<?xml version="1.0"?>
<feed xmlns="http://www.w3.org/2005/Atom" xml:lang="en">
	<id>https://mywikibiz.com/index.php?action=history&amp;feed=atom&amp;title=Module%3ARfx</id>
	<title>Module:Rfx - Revision history</title>
	<link rel="self" type="application/atom+xml" href="https://mywikibiz.com/index.php?action=history&amp;feed=atom&amp;title=Module%3ARfx"/>
	<link rel="alternate" type="text/html" href="https://mywikibiz.com/index.php?title=Module:Rfx&amp;action=history"/>
	<updated>2026-07-05T07:43:43Z</updated>
	<subtitle>Revision history for this page on the wiki</subtitle>
	<generator>MediaWiki 1.35.3</generator>
	<entry>
		<id>https://mywikibiz.com/index.php?title=Module:Rfx&amp;diff=479141&amp;oldid=prev</id>
		<title>Zoran: Pywikibot 6.4.0</title>
		<link rel="alternate" type="text/html" href="https://mywikibiz.com/index.php?title=Module:Rfx&amp;diff=479141&amp;oldid=prev"/>
		<updated>2021-07-16T05:39:31Z</updated>

		<summary type="html">&lt;p&gt;Pywikibot 6.4.0&lt;/p&gt;
&lt;p&gt;&lt;b&gt;New page&lt;/b&gt;&lt;/p&gt;&lt;div&gt;----------------------------------------------------------------------&lt;br /&gt;
--                          Module:Rfx                              --&lt;br /&gt;
-- This is a library for retrieving information about requests      --&lt;br /&gt;
-- for adminship and requests for bureaucratship on the English     --&lt;br /&gt;
-- Wikipedia. Please see the module documentation for instructions. --&lt;br /&gt;
----------------------------------------------------------------------&lt;br /&gt;
&lt;br /&gt;
local libraryUtil = require('libraryUtil')&lt;br /&gt;
local lang = mw.getContentLanguage()&lt;br /&gt;
local textSplit = mw.text.split&lt;br /&gt;
local umatch = mw.ustring.match&lt;br /&gt;
local newTitle = mw.title.new&lt;br /&gt;
&lt;br /&gt;
local rfx = {}&lt;br /&gt;
&lt;br /&gt;
--------------------------------------&lt;br /&gt;
--         Helper functions         --&lt;br /&gt;
--------------------------------------&lt;br /&gt;
&lt;br /&gt;
local function getTitleObject(title)&lt;br /&gt;
	local success, titleObject = pcall(newTitle, title)&lt;br /&gt;
	if success and titleObject then&lt;br /&gt;
		return titleObject&lt;br /&gt;
	else&lt;br /&gt;
		return nil&lt;br /&gt;
	end&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
local function parseVoteBoundaries(section)&lt;br /&gt;
	-- Returns an array containing the raw wikitext of RfX votes in a given section.&lt;br /&gt;
	section = section:match('^.-\n#(.*)$') -- Strip non-votes from the start.&lt;br /&gt;
	if not section then&lt;br /&gt;
		return {}&lt;br /&gt;
	end&lt;br /&gt;
	section = section:match('^(.-)\n[^#]') or section -- Discard subsequent numbered lists.&lt;br /&gt;
	local comments = textSplit(section, '\n#')&lt;br /&gt;
	local votes = {}&lt;br /&gt;
	for i, comment in ipairs(comments) do&lt;br /&gt;
		if comment:find('^[^#*;:].*%S') then&lt;br /&gt;
			votes[#votes + 1] = comment&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	return votes&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
local function parseVote(vote)&lt;br /&gt;
	-- parses a username from an RfX vote.&lt;br /&gt;
	local userStart, userEnd, userMatch = vote:find('%[%[[%s_]*[uU][sS][eE][rR][%s_]*:[%s_]*(.-)[%s_]*%]%].-$')&lt;br /&gt;
	local talkStart, talkEnd, talkMatch = vote:find('%[%[[%s_]*[uU][sS][eE][rR][%s_]+[tT][aA][lL][kK][%s_]*:[%s_]*(.-)[%s_]*%]%].-$')&lt;br /&gt;
	local username&lt;br /&gt;
	if userStart and talkStart then&lt;br /&gt;
		if userStart &amp;gt; talkStart then&lt;br /&gt;
			username = userMatch&lt;br /&gt;
		else&lt;br /&gt;
			username = talkMatch&lt;br /&gt;
		end&lt;br /&gt;
	elseif userStart then&lt;br /&gt;
		username = userMatch&lt;br /&gt;
	elseif talkStart then&lt;br /&gt;
		username = talkMatch&lt;br /&gt;
	else&lt;br /&gt;
		return string.format( &amp;quot;'''Error parsing signature''': ''%s''&amp;quot;, vote )&lt;br /&gt;
	end&lt;br /&gt;
	username = username:match('^[^|/#]*')&lt;br /&gt;
	return username&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
local function parseVoters(votes)&lt;br /&gt;
	local voters = {}&lt;br /&gt;
	for i, vote in ipairs(votes) do&lt;br /&gt;
		voters[#voters + 1] = parseVote(vote)&lt;br /&gt;
	end&lt;br /&gt;
	return voters&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
local function dupesExist(...)&lt;br /&gt;
	local exists = {}&lt;br /&gt;
	local tables = {...}&lt;br /&gt;
	for i, usernames in ipairs(tables) do&lt;br /&gt;
		for j, username in ipairs(usernames) do&lt;br /&gt;
			username = lang:ucfirst(username)&lt;br /&gt;
			if exists[username] then&lt;br /&gt;
				return true&lt;br /&gt;
			else&lt;br /&gt;
				exists[username] = true&lt;br /&gt;
			end&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	return false&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
------------------------------------------&lt;br /&gt;
--   Define the constructor function    --&lt;br /&gt;
------------------------------------------&lt;br /&gt;
&lt;br /&gt;
function rfx.new(title)&lt;br /&gt;
	local obj = {}&lt;br /&gt;
	local data = {}&lt;br /&gt;
	local checkSelf = libraryUtil.makeCheckSelfFunction( 'Module:Rfx', 'rfx', obj, 'rfx object' )&lt;br /&gt;
	&lt;br /&gt;
	-- Get the title object and check to see whether we are a subpage of WP:RFA or WP:RFB.&lt;br /&gt;
	title = getTitleObject(title)&lt;br /&gt;
	if not title then&lt;br /&gt;
		return nil&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	function data:getTitleObject()&lt;br /&gt;
		checkSelf(self, 'getTitleObject')&lt;br /&gt;
		return title&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	if title.namespace == 4 then&lt;br /&gt;
		local rootText = title.rootText&lt;br /&gt;
		if rootText == 'Requests for adminship' then&lt;br /&gt;
			data.type = 'rfa'&lt;br /&gt;
		elseif rootText == 'Requests for bureaucratship' then&lt;br /&gt;
			data.type = 'rfb'&lt;br /&gt;
		else&lt;br /&gt;
			return nil&lt;br /&gt;
		end&lt;br /&gt;
	else&lt;br /&gt;
		return nil&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	-- Get the page content and divide it into sections.&lt;br /&gt;
	local pageText = title:getContent()&lt;br /&gt;
	if not pageText then&lt;br /&gt;
		return nil&lt;br /&gt;
	end&lt;br /&gt;
	local introText, supportText, opposeText, neutralText = umatch(&lt;br /&gt;
		pageText,&lt;br /&gt;
		'^(.-)\n====[^=\n][^\n]-====.-'&lt;br /&gt;
		.. '\n=====%s*[sS]upport%s*=====(.-)'&lt;br /&gt;
		.. '\n=====%s*[oO]ppose%s*=====(.-)'&lt;br /&gt;
		.. '\n=====%s*[nN]eutral%s*=====(.-)$'&lt;br /&gt;
	)&lt;br /&gt;
	if not introText then&lt;br /&gt;
		introText, supportText, opposeText, neutralText = umatch(&lt;br /&gt;
			pageText,&lt;br /&gt;
			&amp;quot;^(.-\n'''[^\n]-%(%d+/%d+/%d+%)[^\n]-''')\n.-&amp;quot;&lt;br /&gt;
			.. &amp;quot;\n'''Support'''(.-)\n'''Oppose'''(.-)\n'''Neutral'''(.-)&amp;quot;&lt;br /&gt;
		)&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	-- Get vote counts.&lt;br /&gt;
	local supportVotes, opposeVotes, neutralVotes&lt;br /&gt;
	if supportText and opposeText and neutralText then&lt;br /&gt;
		supportVotes = parseVoteBoundaries(supportText)&lt;br /&gt;
		opposeVotes = parseVoteBoundaries(opposeText)&lt;br /&gt;
		neutralVotes = parseVoteBoundaries(neutralText)&lt;br /&gt;
	end&lt;br /&gt;
	local supports, opposes, neutrals&lt;br /&gt;
	if supportVotes and opposeVotes and neutralVotes then&lt;br /&gt;
		supports = #supportVotes&lt;br /&gt;
		data.supports = supports&lt;br /&gt;
		opposes = #opposeVotes&lt;br /&gt;
		data.opposes = opposes&lt;br /&gt;
		neutrals = #neutralVotes&lt;br /&gt;
		data.neutrals = neutrals&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	-- Voter methods and dupe check.&lt;br /&gt;
&lt;br /&gt;
	function data:getSupportUsers()&lt;br /&gt;
		checkSelf(self, 'getSupportUsers')&lt;br /&gt;
		if supportVotes then&lt;br /&gt;
			return parseVoters(supportVotes)&lt;br /&gt;
		else&lt;br /&gt;
			return nil&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	function data:getOpposeUsers()&lt;br /&gt;
		checkSelf(self, 'getOpposeUsers')&lt;br /&gt;
		if opposeVotes then&lt;br /&gt;
			return parseVoters(opposeVotes)&lt;br /&gt;
		else&lt;br /&gt;
			return nil&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	function data:getNeutralUsers()&lt;br /&gt;
		checkSelf(self, 'getNeutralUsers')&lt;br /&gt;
		if neutralVotes then&lt;br /&gt;
			return parseVoters(neutralVotes)&lt;br /&gt;
		else&lt;br /&gt;
			return nil&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	function data:dupesExist()&lt;br /&gt;
		checkSelf(self, 'dupesExist')&lt;br /&gt;
		local supportUsers = self:getSupportUsers()&lt;br /&gt;
		local opposeUsers = self:getOpposeUsers()&lt;br /&gt;
		local neutralUsers = self:getNeutralUsers()&lt;br /&gt;
		if not (supportUsers and opposeUsers and neutralUsers) then&lt;br /&gt;
			return nil&lt;br /&gt;
		end&lt;br /&gt;
		return dupesExist(supportUsers, opposeUsers, neutralUsers)&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if supports and opposes then&lt;br /&gt;
		local total = supports + opposes&lt;br /&gt;
		if total &amp;lt;= 0 then&lt;br /&gt;
			data.percent = 0&lt;br /&gt;
		else&lt;br /&gt;
			data.percent = math.floor((supports / total * 100) + 0.5)&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	if introText then&lt;br /&gt;
		data.endTime = umatch(introText, '(%d%d:%d%d, %d+ %w+ %d+) %(UTC%)')&lt;br /&gt;
		data.user = umatch(introText, '===%s*%[%[[_%s]*[wW]ikipedia[_%s]*:[_%s]*[rR]equests[_ ]for[_ ]%w+/.-|[_%s]*(.-)[_%s]*%]%][_%s]*===')&lt;br /&gt;
		if not data.user then&lt;br /&gt;
			data.user = umatch(introText, '===%s*([^\n]-)%s*===')&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- Methods for seconds left and time left.&lt;br /&gt;
	&lt;br /&gt;
	function data:getSecondsLeft()&lt;br /&gt;
		checkSelf(self, 'getSecondsLeft')&lt;br /&gt;
		local endTime = self.endTime&lt;br /&gt;
		if not endTime then&lt;br /&gt;
			return nil&lt;br /&gt;
		end&lt;br /&gt;
		local now = tonumber(lang:formatDate(&amp;quot;U&amp;quot;))&lt;br /&gt;
		local success, endTimeU = pcall(lang.formatDate, lang, 'U', endTime)&lt;br /&gt;
		if not success then&lt;br /&gt;
			return nil&lt;br /&gt;
		end&lt;br /&gt;
		endTimeU = tonumber(endTimeU)&lt;br /&gt;
		if not endTimeU then&lt;br /&gt;
			return nil&lt;br /&gt;
		end&lt;br /&gt;
		local secondsLeft = endTimeU - now&lt;br /&gt;
		if secondsLeft &amp;lt;= 0 then&lt;br /&gt;
			return 0&lt;br /&gt;
		else&lt;br /&gt;
			return secondsLeft&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	function data:getTimeLeft()&lt;br /&gt;
		checkSelf(self, 'getTimeLeft')&lt;br /&gt;
		local secondsLeft = self:getSecondsLeft()&lt;br /&gt;
		if not secondsLeft then&lt;br /&gt;
			return nil&lt;br /&gt;
		end&lt;br /&gt;
		return mw.ustring.gsub(lang:formatDuration(secondsLeft, {'days', 'hours'}), ' and', ',')&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	function data:getReport()&lt;br /&gt;
		-- Gets the URI object for Vote History tool&lt;br /&gt;
		checkSelf(self, 'getReport')&lt;br /&gt;
		return mw.uri.new('//apersonbot.toolforge.org/vote-history?page=' .. mw.uri.encode(title.prefixedText))&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	function data:getStatus()&lt;br /&gt;
		-- Gets the current status of the RfX. Returns either &amp;quot;successful&amp;quot;, &amp;quot;unsuccessful&amp;quot;,&lt;br /&gt;
		-- &amp;quot;open&amp;quot;, or &amp;quot;pending closure&amp;quot;. Returns nil if the status could not be found.&lt;br /&gt;
		checkSelf( self, 'getStatus' )&lt;br /&gt;
		local rfxType = data.type&lt;br /&gt;
		if rfxType == 'rfa' then&lt;br /&gt;
			if umatch(&lt;br /&gt;
				pageText,&lt;br /&gt;
				'%[%[[%s_]*[cC][aA][tT][eE][gG][oO][rR][yY][%s_]*:[%s_]*[sS]uccessful requests for adminship(.-)[%s_]*%]%]'&lt;br /&gt;
			) then&lt;br /&gt;
				return 'successful'&lt;br /&gt;
			elseif umatch(&lt;br /&gt;
				pageText,&lt;br /&gt;
				'%[%[[%s_]*[cC][aA][tT][eE][gG][oO][rR][yY][%s_]*:[%s_]*[uU]nsuccessful requests for adminship(.-)[%s_]*%]%]'&lt;br /&gt;
			) then&lt;br /&gt;
				return 'unsuccessful'&lt;br /&gt;
			end&lt;br /&gt;
		elseif rfxType == 'rfb' then&lt;br /&gt;
			if umatch(&lt;br /&gt;
				pageText,&lt;br /&gt;
				'%[%[[%s_]*[cC][aA][tT][eE][gG][oO][rR][yY][%s_]*:[%s_]*[sS]uccessful requests for bureaucratship(.-)[%s_]*%]%]'&lt;br /&gt;
			) then&lt;br /&gt;
				return 'successful'&lt;br /&gt;
			elseif umatch(&lt;br /&gt;
				pageText,&lt;br /&gt;
				'%[%[[%s_]*[cC][aA][tT][eE][gG][oO][rR][yY][%s_]*:[%s_]*[uU]nsuccessful requests for bureaucratship(.-)[%s_]*%]%]'&lt;br /&gt;
			) then&lt;br /&gt;
				return 'unsuccessful'&lt;br /&gt;
			end&lt;br /&gt;
		end&lt;br /&gt;
		local secondsLeft = self:getSecondsLeft()&lt;br /&gt;
		if secondsLeft and secondsLeft &amp;gt; 0 then&lt;br /&gt;
			return 'open'&lt;br /&gt;
		elseif secondsLeft and secondsLeft &amp;lt;= 0 then&lt;br /&gt;
			return 'pending closure'&lt;br /&gt;
		else&lt;br /&gt;
			return nil&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- Specify which fields are read-only, and prepare the metatable.&lt;br /&gt;
	local readOnlyFields = {&lt;br /&gt;
		getTitleObject = true,&lt;br /&gt;
		['type'] = true,&lt;br /&gt;
		getSupportUsers = true,&lt;br /&gt;
		getOpposeUsers = true,&lt;br /&gt;
		getNeutralUsers = true,&lt;br /&gt;
		supports = true,&lt;br /&gt;
		opposes = true,&lt;br /&gt;
		neutrals = true,&lt;br /&gt;
		endTime = true,&lt;br /&gt;
		percent = true,&lt;br /&gt;
		user = true,&lt;br /&gt;
		dupesExist = true,&lt;br /&gt;
		getSecondsLeft = true,&lt;br /&gt;
		getTimeLeft = true,&lt;br /&gt;
		getReport = true,&lt;br /&gt;
		getStatus = true&lt;br /&gt;
	}&lt;br /&gt;
	&lt;br /&gt;
	local function pairsfunc( t, k )&lt;br /&gt;
		local v&lt;br /&gt;
		repeat&lt;br /&gt;
			k = next( readOnlyFields, k )&lt;br /&gt;
			if k == nil then&lt;br /&gt;
				return nil&lt;br /&gt;
			end&lt;br /&gt;
			v = t[k]&lt;br /&gt;
		until v ~= nil&lt;br /&gt;
		return k, v&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return setmetatable( obj, {&lt;br /&gt;
		__pairs = function ( t )&lt;br /&gt;
			return pairsfunc, t, nil&lt;br /&gt;
		end,&lt;br /&gt;
		__index = data,&lt;br /&gt;
		__newindex = function( t, key, value )&lt;br /&gt;
			if readOnlyFields[ key ] then&lt;br /&gt;
				error( 'index &amp;quot;' .. key .. '&amp;quot; is read-only', 2 )&lt;br /&gt;
			else&lt;br /&gt;
				rawset( t, key, value )&lt;br /&gt;
			end&lt;br /&gt;
		end,&lt;br /&gt;
		__tostring = function( t )&lt;br /&gt;
			return t:getTitleObject().prefixedText&lt;br /&gt;
		end&lt;br /&gt;
	} )&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
return rfx&lt;/div&gt;</summary>
		<author><name>Zoran</name></author>
	</entry>
</feed>