#!/usr/bin/env python3"""Used to find expensive templates that are subject to be converted to Lua.It counts parser functions and then orders templates by number of theseand uploads the first n titles or alternatively templates having count()>n.Parameters:-start Will start from the given title (it does not have to exist). Parameter may be given as "-start" or "-start:title". Defaults to '!'.-first Returns the first n results in decreasing order of number of hits (or without ordering if used with -nosort) Parameter may be given as "-first" or "-first:n".-atleast Returns templates with at least n hits. Parameter may be given as "-atleast" or "-atleast:n".-nosort Keeps the original order of templates. Default behaviour is to sort them by decreasing order of count(parserfunctions).-save Saves the results. The file is in the form you may upload it to a wikipage. May be given as "-save:<filename>". If it exists, titles will be appended.-upload Specify a page in your wiki where results will be uploaded. Parameter may be given as "-upload" or "-upload:title". Say good-bye to previous content if existed.Precedence of evaluation: results are first sorted in decreasing order oftemplates, unless nosort is switched on. Then first n templates are taken iffirst is specified, and at last atleast is evaluated. If nosort and first areused together, the program will stop at the nth hit without scanning the restof the template namespace. This may be used to run it in more sessions(continue with -start next time).First is strict. That means if results #90-120 have the same number of parserfunctions and you specify -first:100, only the first 100 will be listed (evenif atleast is used as well).Should you specify neither first nor atleast, all templates using parserfunctions will be listed."""## (C) Pywikibot team, 2013-2022## Distributed under the terms of the MIT license.#from__future__importannotationsimportcodecsimportrefromcollectionsimportCounterimportpywikibotfrompywikibotimportpagegeneratorsfrompywikibot.botimportExistingPageBot,SingleSiteBot# TODO:# * Using xml and xmlstart# * Using categories# * Error handling for uploading (anyway, that's the last action, it's only# for the beauty of the program, does not effect anything).
[docs]classParserFunctionCountBot(SingleSiteBot,ExistingPageBot):"""Bot class used for obtaining Parser function Count."""use_redirects=Falseupdate_options={'atleast':None,'first':None,'nosort':False,'save':None,'start':'!','upload':None,}def__init__(self,**kwargs)->None:"""Initializer."""super().__init__(**kwargs)editcomment={# This will be used for uploading the list to your wiki.'en':'Bot: uploading list of templates having too many parser ''functions','hu':'A túl sok parserfüggvényt használó sablonok listájának ''feltöltése',}# Limitations for result:ifself.opt.first:try:self.opt.first=int(self.opt.first)ifself.opt.first<1:self.opt.first=NoneexceptValueError:self.opt.first=Noneifself.opt.atleast:try:self.opt.atleast=int(self.opt.atleast)# 1 has no effect, don't waste resources.ifself.opt.atleast<2:self.opt.atleast=NoneexceptValueError:self.opt.atleast=Nonelang=self.site.langself.summary=editcomment.get(lang,editcomment['en'])@propertydefgenerator(self):"""Generator."""gen=self.site.allpages(start=self.opt.start,namespace=10,filterredir=False)ifself.site.doc_subpage:gen=pagegenerators.RegexFilterPageGenerator(gen,self.site.doc_subpage,quantifier='none')returngen
[docs]defsetup(self)->None:"""Setup magic words, regex and result counter."""pywikibot.info('Hold on, this will need some time. ''You will be notified by 50 templates.')magicwords=[]formagic_wordinself.site.siteinfo['magicwords']:magicwords+=magic_word['aliases']self.regex=re.compile(r'#({}):'.format('|'.join(magicwords)),re.I)self.results=Counter()
[docs]deftreat(self,page)->None:"""Process a single template."""title=page.title()if(self.counter['read']+1)%50==0:# Don't let the poor user panic in front of a black screen.pywikibot.info('{}th template is being processed: {}'.format(self.counter['read']+1,title))text=page.textfunctions=self.regex.findall(text)iffunctionsand(self.opt.atleastisNoneorself.opt.atleast<=len(functions)):self.results[title]=len(functions)ifself.opt.nosortandself.opt.first \
andlen(self.results)>=self.opt.first:self.stop()
[docs]defteardown(self)->None:"""Final processing."""resultlist='\n'.join(f'# [[{result[0]}]] ({result[1]})'forresultinself.results.most_common(self.opt.first))pywikibot.info(resultlist)pywikibot.info(f'{len(self.results)} templates were found.')# File operations:ifself.opt.save:# This opens in strict error mode, that means bot will stop# on encoding errors with ValueError.# See https://docs.python.org/3/library/codecs.html#codecs.opentry:withcodecs.open(self.opt.save,encoding='utf-8',mode='a')asf:f.write(resultlist)exceptOSErrorase:pywikibot.error(e)ifself.opt.upload:page=pywikibot.Page(self.site,self.opt.upload)self.userPut(page,page.text,resultlist,ignore_save_related_errors=True,summary=self.summary)
[docs]defmain(*args:str)->None:"""Process command line arguments and invoke ParserFunctionCountBot."""local_args=pywikibot.handle_args(*args)options={}# Parse command line argumentsforarginlocal_args:opt,_,value=arg.partition(':')ifnotopt.startswith('-'):continueopt=opt[1:]ifopt=='start':options[opt]=valueorpywikibot.input('From which title do you want to continue?')elifopt=='save':options[opt]=valueorpywikibot.input('Please enter the filename:')elifopt=='upload':options[opt]=valueorpywikibot.input('Please enter the pagename:')elifopt=='first':options[opt]=valueorpywikibot.input('Please enter the max. number of templates to display:')elifopt=='atleast':options[opt]=valueorpywikibot.input('Please enter the min. number of functions to display:')elifopt=='nosort':options[opt]=Truebot=ParserFunctionCountBot(**options)bot.run()