User:Veledan/FP Promoter/Source
Appearance
Main module
[edit]"""Script to help a human promote new Featured Pictures. by Stephen Lawson (en:User:Veledan) released under the GPL. version 1.3beta 13-May-06 """ import re, time import wikipedia from v_MyPage import MyPage as Page try: set except NameError: from future import set mySite = wikipedia.Site('en') wikipedia.setAction('FP promotion script') #should always be overridden # This is a purely interactive script with limited edits. Set the delay low. wikipedia.put_throttle.setDelay(5) #relevant pages on en wikipedia fpPages = { 'fp' : 'Wikipedia:Featured pictures', 'fpt' : 'Wikipedia:Featured pictures thumbs', 'fpv' : 'Wikipedia:Featured pictures visible', 'fpc' : 'Wikipedia:Featured picture candidates', 'nfp' : 'Template:Announcements/New featured pages', 'go' : 'Wikipedia:Goings-on', 'arch': 'Wikipedia:Featured pictures candidates/'} #archives #local exceptions class ModificationFailed(wikipedia.Error): """Modification failed""" class Error(Exception): """Promobot error""" class DropOutError(Error): """Fatal error. Abandon script""" #classes to represent the main FP pages. #Catch wikipedia exceptions from these in the calling code class FpPage(Page): """Wikipedia:Featured pictures""" def __init__(self): Page.__init__(self, mySite, fpPages['fp']) def addNewFp(self, newFp): """Add a new FP to WP:FP. Takes a NewFP object""" oldtext=self.get() headerRegex = newFp.fpSection.replace(' ','[ ]') sectionsearch = re.compile(r""" ={3,5}[ ]?""" + headerRegex + """[ ]?={3,5} #match the header .*? #plus the section contents (?= #and look ahead but don't match ==|<!--[ ]end[ ]of[ ]list[ ]--> #the next header or end of list )""", re.DOTALL|re.VERBOSE) #Section headings are frequently duplicated. Work out which one to use sections = [] for s in sectionsearch.finditer(oldtext): sections.append(s) if len(sections)==0: raise ModificationFailed('Unable to match section ' + newFp.fpSection) if len(sections)==2 and not newFp.createdByWikipedian: section = sections[1] else: section = sections[0] #make the new entry creator = newFp.creator if newFp.createdByWikipedian: creator="[[" + creator + "|]]" newEntry = "\n* '''[[:" + newFp.title() + "|" + newFp.pipedName + \ "]]''' at [[" + newFp.mainArticle + "]], by " + creator + "\n" #put it together sectiontext = section.group().rstrip() sectiontext += newEntry newtext = oldtext[:section.start()] + sectiontext + oldtext[section.end():] #update the FP counter numberofFps=len(self.imagelinks()) + 1 countersearch=r"are currently '''\d\d\d\d?''' featured pictures" countertext=r"are currently '''" + str(numberofFps) + r"''' featured pictures" counter=re.search(countersearch, newtext) if counter: newtext=re.sub(countersearch,countertext,newtext,1) editsum = 'Adding ' + newFp.title() self.put(newtext,editsum,minorEdit=False) if not counter: raise ModificationFailed('Image added to WP:FP ok but failed to match FP counter.') def headerlinks(self, createdByWikipedian): """Return a list of headers from WP:FP. Avoid ambiguity from the duplicate section names by specifying up front whether 'created by' or 'found by' wikipedian is appropriate. """ oldtext=self.get() s = r'== Images created by Wikipedians ==(?P<createdby>.*?)== Images ' \ 'found by Wikipedians ==(?P<foundby>.*?)<!-- end of list -->' sections=re.search(s, oldtext, re.DOTALL) if not sections: raise DropOutError('ERROR: Unable to read headings from WP:FP') if createdByWikipedian: return self._listheaders(sections.group('createdby')) else: return self._listheaders(sections.group('foundby')) def _listheaders(self, text): """Return list of ===, ====, ===== level headers. Show level by -(s) in front""" headersearch=r'===.+===|====.+====|=====.+=====' headers=[] for header in re.finditer(headersearch,text): fullheader=header.group() if '=====' in fullheader: headers.append('--' + fullheader.strip(' =')) elif '====' in fullheader: headers.append('-' + fullheader.strip(' =')) elif '===' in fullheader: headers.append(fullheader.strip(' =')) else: raise Error('debug: header regex not working') return headers class FptPage(Page): """Wikipedia:Featured pictures thumbs""" def __init__(self): fptRedirPage = Page(mySite, fpPages['fpt']) Page.__init__(self, mySite, fptRedirPage.getRedirectTarget()) fptRedirPage = None def addNewFp(self, newFp): """Add a new FP to WP:FPT. Takes a NewFP object""" oldtext=self.get(force=True) gallerystart = re.search(r'<gallery>\s*(?=Image:)',oldtext) if not gallerystart: raise ModificationFailed('Unable to locate start of gallery') newentry = newFp.title() + '|' + newFp.pipedName + '\n' newtext=oldtext[:gallerystart.end()] + newentry + oldtext[gallerystart.end():] editsum = "Adding " + newFp.title() self.put(newtext,editsum,minorEdit=False) class FpvPage(Page): """Wikipedia:Featured pictures visible""" def __init__(self): Page.__init__(self, mySite, fpPages['fpv']) def addNewFp(self, newFp): """Add a new FP to WP:FPV. Takes a NewFP object""" oldtext=self.get() headerRegex = newFp.fpSection.replace(' ','[ ]') sectionsearch = re.compile(r""" ={3,5}[ ]?""" + headerRegex + """[ ]?={3,5} #match the header .*? #plus the section contents (?= #and look ahead but don't match ==|<!--[ ]end[ ]of[ ]list[ ]--> #the next header or end of list )""", re.DOTALL|re.VERBOSE) #Section headings are frequently duplicated. Work out which one to use sections = [] for s in sectionsearch.finditer(oldtext): sections.append(s) if len(sections)==0: raise ModificationFailed('Unable to match section ' + newFp.fpSection) if len(sections)==2 and not newFp.createdByWikipedian: section = sections[1] else: section = sections[0] #make the new entry creator = newFp.creator if newFp.createdByWikipedian: creator="[[" + creator + "|]]" if newFp.fpvSize: newEntry = "\n:[[" + newFp.title() + "|thumb|none|" + newFp.fpvSize \ + "|" + newFp.pipedName + r"<br />at [[" + newFp.mainArticle \ + "]] by " + creator + "]]\n" else: newEntry = "\n:[[" + newFp.title() + "|frame|none|" \ + newFp.pipedName + r"<br />at [[" + newFp.mainArticle \ + "]] by " + creator + "]]\n" #finally put it together and save sectiontext = section.group().rstrip() sectiontext += newEntry newtext = oldtext[:section.start()] + sectiontext + oldtext[section.end():] editsum = 'Adding ' + newFp.title() self.put(newtext,editsum,minorEdit=False) def headerlinks(self, createdByWikipedian): """Return a list of headers from WP:FP. Avoid ambiguity from the duplicate section names by specifying up front whether 'created by' or 'found by' wikipedian is appropriate. """ oldtext=self.get() s = r'== Images created by Wikipedians ==(?P<createdby>.*?)== Images ' \ 'found by Wikipedians ==(?P<foundby>.*?)<!-- end of list -->' sections=re.search(s, oldtext, re.DOTALL) if not sections: raise DropOutError('ERROR: Unable to read headings from WP:FP') if createdByWikipedian: return self._listheaders(sections.group('createdby')) else: return self._listheaders(sections.group('foundby')) def _listheaders(self, text): """Return list of ===, ====, ===== level headers. Show level by -(s) in front""" headersearch=r'===[^=\n]+===|====[^=\n]+====|=====[^=\n]+=====' headers=[] for header in re.finditer(headersearch,text): fullheader=header.group() if '=====' in fullheader: headers.append('--' + fullheader.strip(' =')) elif '====' in fullheader: headers.append('-' + fullheader.strip(' =')) elif '===' in fullheader: headers.append(fullheader.strip(' =')) else: raise Error('debug: header regex not working') return headers class FpcPage(Page): """Wikipedia:Featured picture candidates""" def __init__(self): Page.__init__(self, mySite, fpPages['fpc']) def removeNomination(self, nom): """Remove one nomination from WP:FPC. Takes nomination Page as arg.""" oldtext = self.get() editsum = 'Closing %s' % nom.title() #build regex to match nomination subst page searchexp = nom.title() searchexp=searchexp.replace(' ','[ _]') searchexp=r'{{ ?' + searchexp + r' ?}}\n|\[\[ ?' + searchexp + r' ?\]\]\n' if not re.search(searchexp, oldtext): raise Error('%s isn\'t listed at WP:FPC' % nom.title()) newtext=re.sub(searchexp, '', oldtext) self.put(newtext,editsum,minorEdit=False) class AnnouncePage(Page): """Template:Announcements/New featured pages""" def __init__(self): Page.__init__(self, mySite, fpPages['nfp']) def addNewFp(self, newFp): """Add a new FP to WP:FPV. Takes a NewFP object""" #TODO: add in auto-limiting of number of new FPs listed at any one time #find the current list oldtext = self.get(force=True) fpSectionPattern = """'''\[\[Wikipedia:Featured pictures\|Pictures\]\] recently awarded "featured" status'''.*(?='''\[\[Wikipedia:Featured portals\|Portals\]\] recently)""" fpSection = re.search(fpSectionPattern, oldtext, re.DOTALL) if not fpSection: raise ModificationFailed('Failed to match FP section') #make new entry newentry = '* [[:' + newFp.title() + '|' + newFp.pipedName + ']] ' newentry += '(' + unicode(time.strftime('%B %d',time.gmtime())) + ')' #insert and save sectionText = fpSection.group().rstrip() sectionText += '\n' + newentry + '\n\n' newtext = oldtext[:fpSection.start()] + sectionText + oldtext[fpSection.end():] editsum = 'Adding ' + newFp.title() self.put(newtext, editsum, minorEdit=False) class GoPage(Page): """Wikipedia:Goings-on""" def __init__(self): Page.__init__(self, mySite, fpPages['go']) def addNewFp(self, newFp): """Add a new FP to WP:FPV. Takes a NewFP object""" #TODO: add in auto-limiting of number of new FPs listed at any one time #find the current list oldtext = self.get(force=True) fpSectionPattern = """'''\[\[Wikipedia:Featured pictures\|Pictures\]\] that gained "featured" status'''.*(?='''\[\[Wikipedia:Featured portals\|Portals\]\] that)""" fpSection = re.search(fpSectionPattern, oldtext, re.DOTALL) if not fpSection: raise ModificationFailed('Failed to match FP section') #make new entry newentry = '* [[:' + newFp.title() + '|' + newFp.pipedName + ']] ' newentry += '(' + time.strftime('%B %d',time.gmtime()) + ')' #insert and save sectionText = fpSection.group().rstrip() sectionText += '\n' + newentry + '\n\n' newtext = oldtext[:fpSection.start()] + sectionText + oldtext[fpSection.end():] editsum = 'Adding ' + newFp.title() self.put(newtext, editsum, minorEdit=False) class ArchivePage(Page): """Wikipedia:FPC Archives""" def __init__(self): currentMonth = time.strftime('%B-%Y',time.gmtime()) currentArchive = fpPages['arch'] + currentMonth Page.__init__(self, mySite, currentArchive) def addNomination(self, nom): """Insert one nomination from WP:FPC. Takes nomination Page as arg. Create new archive if necessary.""" try: oldtext = self.get(force=True) except NoPage: oldtext='{{FPCArchiveBar}}\n\n' editsum = 'Adding %s' % nom.title() newtext = oldtext + '\n{{%s}}\n' % nom.title() #save page self.put(newtext,editsum,minorEdit=False) class NominationPage(Page): """WP:FPC/subpage for nomination.""" def verdict(self, isPromoted, image=None, promotionMessage=''): """Add verdict to the nomination page""" #make verdict string & edit summary if isPromoted: verdict = '{{FPCresult|Promoted|%s}} %s ~~~~' % (image.title(), promotionMessage) editsum = 'Promoted' else: verdict = '{{FPCresult|Not promoted| }} %s ~~~~' % promotionMessage editsum = 'Not promoted' #insert verdict in place of reserve note oldtext=self.get(force=True) reservemarker=r'FP Promotion bot — closure.*<!-- end note -->' marker = re.search(reservemarker, oldtext) if not marker: raise DropOutError('ERROR: Reservation marker has gone missing. Please complete manually') newtext=oldtext[:marker.start()] + verdict + oldtext[marker.end():] #save page self.put(newtext, editsum, minorEdit=False) def firstName(self): """Return the first wikilinked username on page""" username=re.search(r'\[\[(User:[^\]\|]+)',self.get()) if username: return username.group(1) def reserve(self): """Add a line to the nomination page effectively 'locking' it""" reservetext = r'FP Promotion bot — closure in progress. Please do ' \ 'not amend this page while this note is showing' es = 'Adding reserve marker prior to closure' #First check not already reserved by someone else using the bot oldtext = self.get() #we've already checked this exists reservemarker=reservetext + r'[^\[]*\[\[(?P<sig>[^\|\]]+)' alreadyreserved = re.search(reservemarker, oldtext) if alreadyreserved: raise DropOutError('The page has already been reserved by ' + alreadyreserved.group('sig')) #find where to insert our line. Any of three end markers are matched, or if not, end of str$ endmarker=re.search(r'<!-- additional votes go above this line -->|\{\{-\}\}|\{\{breakafterimages\}\}|$', oldtext) #insert reserve marker reservetext = '\n' + reservetext + '. ~~~~ <!-- end note -->\n\n' newtext = oldtext[:endmarker.start()] + reservetext + oldtext[endmarker.start():] #save page self.put(newtext, es) #class for the image due for promotion class NewFP(wikipedia.ImagePage): """An image to be promoted. Call gatherInfo() after construction. """ def __init__(self, site, imageTitle, otherVersions, nominationPage): wikipedia.ImagePage.__init__(self, site, imageTitle) self.otherVersions = otherVersions # list of pages self.nominationPage = nominationPage self.nominator = None self.creator = None self.createdByWikipedian = None #boolean self.fpSection = None #string self.fpvSection = None #string self.isPanorama = None self.isAnimated = None self.fpvSize = None #string, '300px' self.pipedName = None self.mainArticle = None #str title self.promotionMessage = None #string self.talkPageMessage = None #string self.mainVersionToReplace = None #Page def gatherInfo(self): #gather all required info self.pipedName = self._getPipedName() self.mainArticle = self._getArticle() self.mainVersionToReplace = self._getMainVersion() self.nominator=self._getNominator() self.creator, self.createdByWikipedian = self._getCreator() self.isPanorama, self.isAnimated, self.fpvSize = self._getSize() self.fpSection = self._getFpSection() self.fpvSection = self._getFpvSection() self.promotionMessage=self._getPromotionMessage() self.talkPageMessage=self._getTalkPageMessage() def supplantOriginalVersion(self): #Replace original image with self (promoted version) in articles editsum='Updating image after promotion of new version at %s' % self.nominationPage.title() for article in self.mainVersionToReplace.usingPages(): if article.namespace()==0: article.put(article.get().replace(self.mainVersionToReplace.title(),self.title()),editsum) def addFpTemplate(self): oldtext='' if self.exists(): oldtext=self.get() if '{{FPC}}' in oldtext: newtext = oldtext.replace('{{FPC}}','{{FeaturedPicture}}') else: newtext = '{{FeaturedPicture}}\n' + oldtext editsum = '{{FeaturedPicture}}' self.put(newtext,editsum) def _getPromotionMessage(self): return fpInput('Would you like to add any extra message to the Promotion ' \ 'template on the nomination page? Leave this blank if not.' \ '\n\nIf you enter a message it will appear between the verdict ' \ 'and your signature on the nomination page') def _getTalkPageMessage(self): return fpInput('Would you like to add any extra message to the Promotion ' \ 'template on the nominator\'s and/or creator\'s talk page?' \ 'Leave this blank if not.\n\nIf you enter a message it ' \ 'will appear between the promotion template and your ' \ 'signature on the nomination page') def _getArticle(self): while True: choice = fpInput('What is the title of this image\'s main article?') choice = unicode(choice) page = Page(mySite, choice) if page.exists(): return page.title() output('That page doesn\'t exist. Try again') def _getMainVersion(self): replace = letteredquestion('Is the promoted image an edit that should replace ' 'the original version in mainspace articles?') if replace=='[n]o': return None oldversion=numberedquestion('Which was the original version?\n\nThe image you ' 'select will be replaced in mainspace articles ' 'with ' + self.title(), [i.title() for i in self.otherVersions]) return wikipedia.ImagePage(mySite,oldversion) def _getPipedName(self): return fpInput('What wording should be used in piping text links to the ' 'image? (e.g. Image:blue_bird02_edit.jpg might be "Blue bird")') def _getSize(self): choice=letteredquestion('Is this a wide panorama? (This affects the size ' 'it will be shown at in Wikipedia:Featured ' 'pictures visible)') isPanorama = (choice=='[y]es') choice=letteredquestion('Is this an animated gif? (These are not resized)') isAnimated = (choice=='[y]es') if isPanorama: size='600px' elif isAnimated: size='' else: size='300px' return isPanorama, isAnimated, size def _getFpSection(self): question = 'What section should this be listed under at Wikipedia:Featured pictures?' return numberedquestion(question, fp.headerlinks(self.createdByWikipedian)).strip(' -') def _getFpvSection(self): question = 'What section should this be listed under at Wikipedia:Featured pictures visible?' if self.isAnimated or self.isPanorama: question += '\n\nYour previous answers indicate this is a panorama ' \ 'or animation. Please choose the matching section as the ' \ 'image will be sized differently from normal.' return numberedquestion(question,fpv.headerlinks(self.createdByWikipedian)).strip(' -') def _getNominator(self): probableNominator = self.nominationPage.firstName() while True: question='Which wikipedian nominated the image? (enter username, ' \ 'not nickname)' if probableNominator: question+='\n\nYou can leave this blank if it was ' + probableNominator choice = fpInput(question) if choice.strip()=='': choice=probableNominator if ':' not in choice: choice='User:' + choice userpage = Page(mySite, choice) try: userpage.get() except wikipedia.Error: choice=letteredquestion('Page %s doesn\'t exist. Please confirm ' \ 'the username was correct (in which case the bot will ' \ 'create the talk page if necessary when leaving a message) ' \ 'or choose No to re-enter the name:' % userpage.title()) else: choice='[y]es' if choice=='[y]es': break return userpage.title() def _getCreator(self): choice = letteredquestion('Was %s also the creator of the image?' % self.nominator) if choice=='[y]es': return self.nominator, True choice = letteredquestion('Is the creator of the image a wikipedian?') createdByWikipedian = (choice=='[y]es') if createdByWikipedian: question="Please enter the username of the wikipedian who created" \ "the image (with or without the \'User:\')" else: question="Please give a name or other suitable attribution for " \ "the creator of the image (e.g. 'John Smith' or 'NASA')" while True: choice = fpInput(question) if not createdByWikipedian: creator = choice break #no validation in that case if ':' not in choice: choice='User:' + choice userpage = Page(mySite, choice) try: userpage.get() except wikipedia.Error: choice=letteredquestion('Page %s doesn\'t exist. Please confirm ' \ 'the username was correct (in which case the bot will ' \ 'create the talk page if necessary when leaving a message) ' \ 'or choose No to re-enter the name:' % userpage.title()) else: choice='[y]es' if choice=='[y]es': creator = userpage.title() break return creator, createdByWikipedian #main control class class FpPromoter(object): def __init__(self): #initialize page objects if necessary try: fp except NameError: createGlobalPageObjects() #instance variables self.nomPage=None #Page self.newFp=None #Page self.promoted=None #boolean self.report=['== FP Promotion Tool started =='] #List of outcomes def run(self): """Main control procedure.""" output('== FP Promotion Tool started ==') #Ask which nomination to close self.nomPage = self._getNominatedPage() self._report('%s selected' % self.nomPage.title()) #Reserve the page to stop edit conflicts output('Attempting to reserve nomination page...') try: self.nomPage.reserve() except wikipedia.Error, error: raise DropOutError(error) self._report('Reserve marker added to page') #Ask whether the image is promoted or not self.promoted = self._getResult() #promote or not promote if self.promoted: self.promote() else: self.dontpromote() def promote(self): """Perform the steps to close a nomination and promote an image""" #find out which image is to be promoted images=self.nomPage.imagelinks() imagelist = [i.title() for i in images] promotedimage = numberedquestion('Which image to promote?',imagelist) #make list of imagepages not being promoted otherimages = set([i for i in images if i.title() != promotedimage]) #create newFp object self.newFp = NewFP(mySite, promotedimage, otherimages, self.nomPage) #new FP will ask the rest of the questions self.newFp.gatherInfo() #PROMOTION STEPS #1. Add verdict output('Adding verdict to nomination page...') try: self.nomPage.verdict(True, self.newFp, self.newFp.promotionMessage) except wikipedia.Error, error: raise DropOutError('ERROR: Failed to add verdict: ' + str(error)) else: self._report('Promoted. Verdict added to nomination page') #2. Add to archive output('Adding entry to %s archive...' % time.strftime('%B',time.gmtime())) try: arc.addNomination(self.nomPage) except wikipedia.Error, error: self._report('Failed to add entry to archive: ' + str(error)) else: self._report('Added entry to archive OK') output('Removing entry from WP:FPC...') try: fpc.removeNomination(self.nomPage) except wikipedia.Error, error: self._report('Failed to remove entry from WP:FPC: ' + str(error)) except Error, error: self._report(str(error)) else: self._report('Removed from WP:FPC OK') #3. Add to New Featured content template output('Adding to Template:Announcements/New featured pages...') try: nfp.addNewFp(self.newFp) except wikipedia.Error, error: self._report('Failed to add entry to Template:Announcements/New featured pages: ' + str(error)) else: self._report('Added to Template:Announcements/New featured pages OK') #4. Add to Goings-on output('Adding to Wikipedia:Goings-on...') try: goo.addNewFp(self.newFp) except wikipedia.Error, error: self._report('Failed to add entry to Wikipedia.Goings-on: ' + str(error)) else: self._report('Added to Wikipedia:Goings-on OK') #5. Add to Featured Pictures output('Adding to WP:FP...') try: fp.addNewFp(self.newFp) except wikipedia.Error, error: self._report('Failed to add entry to Wikipedia:Featured pictures: ' + str(error)) else: self._report('Added to Wikipedia:Featured pictures OK') #6. Add to WP:FPV output('Adding to WP:FPV...') try: fpv.addNewFp(self.newFp) except wikipedia.Error, error: self._report('Failed to add entry to Wikipedia:Featured pictures visible: ' + str(error)) else: self._report('Added to Wikipedia:Featured pictures visible OK') #7. Add to WP:FPT output('Adding to WP:FPT...') try: fpt.addNewFp(self.newFp) except wikipedia.Error, error: self._report('Failed to add entry to Wikipedia:Featured pictures thumbs: ' + str(error)) else: self._report('Added to Wikipedia:Featured pictures thumbs OK') #8. Update {{FPC}} tags output('Adding {{FeaturedPicture}}...') try: self.newFp.addFpTemplate() except wikipedia.Error, error: self._report('Failed to add {{FeaturedPicture}}: ' + str(error)) else: self._report('Added {{FeaturedPicture}} OK') self._report('Removing {{FPC}} from all other images in Nom page...') for image in self.newFp.otherVersions: newtext='' if not image.exists(): #no need to remove from empty page continue oldtext=image.get() if '{{FPC}}' in oldtext: newtext=oldtext.replace('{{FPC}}','') try: image.put(newtext,'Removing {{FPC}}') except wikipedia.Error: self._report('Failed to remove {{FPC}} from ' + image.title()) continue self._report('{{FPC}} removed from ' + image.title()) #9. Notify nominator output('Notifying nominator...') editsum='New featured pic' try: nominatorsTalkPage=Page(mySite,self.newFp.nominator).switchTalkPage() newmessage='\n==Featured picture promotion==\n{{subst:PromotedFPC|%s}}<br />%s ~~~~' % (self.newFp.title(), self.newFp.talkPageMessage) nominatorsTalkPage.put(nominatorsTalkPage.get()+newmessage,editsum,minorEdit=False) except wikipedia.Error, error: self._report('Failed to leave message for nominator: ' + str(error)) else: self._report('Notified nominator OK') #10. Notify creator if self.newFp.nominator != self.newFp.creator and self.newFp.createdByWikipedian: output('Notifying creator...') editsum='New featured pic' try: creatorsTalkPage=Page(mySite,self.newFp.creator).switchTalkPage() newmessage='\n==Featured picture promotion==\n{{subst:UploadedFP|%s}}<br />%s ~~~~' % (self.newFp.title(), self.newFp.talkPageMessage) creatorsTalkPage.put(creatorsTalkPage.get()+newmessage,editsum,minorEdit=False) except wikipedia.Error, error: self._report('Failed to leave message for nominator: ' + str(error)) else: self._report('Creator nominator OK') self._report('Promotion completed') #11. Replace original image with edited version if necessary if self.newFp.mainVersionToReplace: output('Replacing image in articles...') self.newFp.supplantOriginalVersion() def dontpromote(self): """Perform the steps to close a nom with no promotion""" #add verdict message = fpInput('Would you like to add any extra message to the non-Promotion ' \ 'template on the nomination page? Leave this blank if not.' \ '\n\nIf you enter a message it will appear between the verdict ' \ 'and your signature on the nomination page') output('Adding verdict to nomination page...') try: self.nomPage.verdict(False, None, message) except wikipedia.Error, error: raise DropOutError('ERROR: Failed to add verdict:' + str(error)) else: self._report('Not promoted. Verdict added to nomination page') #remove {{FPC}} from versions self._report('Removing {{FPC}} from all images in Nom page...') images = set(self.nomPage.imagelinks()) for image in images: newtext='' if not image.exists(): #no need to remove from empty page continue oldtext=image.get() if '{{FPC}}' in oldtext: newtext=oldtext.replace('{{FPC}}','') try: image.put(newtext,'Removing {{FPC}}') except wikipedia.Error: self._report('Failed to remove {{FPC}} from ' + image.title()) continue self._report('{{FPC}} removed from ' + image.title()) #Remove from Wp:FPC output('Removing entry from WP:FPC...') try: fpc.removeNomination(self.nomPage) except wikipedia.Error, error: self._report('Failed to remove entry from WP:FPC: ' + str(error)) except Error, error: self._report(str(error)) else: self._report('Removed from WP:FPC OK') #Add to archive output('Adding entry to %s archive' % time.strftime('%B',time.gmtime())) try: arc.addNomination(self.nomPage) except wikipedia.Error, error: self._report('Failed to add entry to archive: ' + str(error)) else: self._report('Added entry to archive OK') self._report('Non-promotion completed') def _report(self, message, echo=True): self.report.append(message) if echo: output(message) def _getResult(self): return (letteredquestion('Has %s resulted in a promotion?' % self.nomPage.title())=='[y]es') def _getNominatedPage(self): while True: choice = fpInput("What page to work on? Please enter the full title " \ "pasted from the subpage, e.g. 'Wikipedia:Featured picture " \ "candidates/Street Trinidad Cuba'") nomPage=NominationPage(mySite, choice) if nomPage.exists(): break else: output('Error: %s doesn\'t exist. Please try again...' % choice) return nomPage #functions for use by all classes def fpInput(message): output('\n' + message) answer=raw_input('--> ') answer = unicode(answer.strip(),'utf-8') output(answer,logonly=True) output('\n\n') return answer def output(message, logonly=False): if not logonly: wikipedia.output(unicode('\n' + message)) #also write to log if there is one try: promobotlogfile except NameError: pass else: promobotlogfile.write(time.strftime('[%d%b%y %H:%M:%S] ',time.gmtime()) + message + '\n') def numberedquestion(message, answers): """Show a numbered list of 'answers' and return the selected item""" if message[:-1] != '\n': message+='\n' for item in answers: message+=('%u. ' % (answers.index(item) + 1)) + item + '\n' while True: choice = fpInput(message) try: choice=int(choice) except ValueError: choice=0 if choice > len(answers) or choice < 1: output('ERROR: You have to choose a number from the menu. Try again.\n') else: return answers[choice-1] def letteredquestion(message, answers=['[y]es','[n]o']): """Show a list of options with [s]ignicant letters and return selected item""" if message[:-1] != '\n': message+='\n' answerlist='' for item in answers: message+=(' - ') + item + '\n' keyletter=re.search(r'\[([^\[\]])\]', item).group(1).lower() if keyletter=='': raise DropOutError('Bad list of possible answers - no [] in %s' % item) answerlist+=keyletter while True: choice=fpInput(message).lower()[:1] if not choice or choice not in answerlist: output('ERROR: You have to choose a letter from the menu. Try again.\n') else: return answers[answerlist.index(choice)] def createGlobalPageObjects(): global fp, fpc, fpv, fpt, nfp, goo, arc fp = FpPage() fpv = FpvPage() fpc = FpcPage() fpt = FptPage() nfp = AnnouncePage() goo = GoPage() arc = ArchivePage() def createLogFile(): global promobotlogfile promobotlogfile = file('promobotlog.txt','a') def closeLogFile(): promobotlogfile.close() if __name__=='__main__': try: createLogFile() wikipedia.Site.forceLogin(mySite) #make global FP page objects fp = FpPage() fpv = FpvPage() fpc = FpcPage() fpt = FptPage() nfp = AnnouncePage() goo = GoPage() arc = ArchivePage() #go try: promobot = FpPromoter() promobot.run() except DropOutError, error: output('\nThe script was terminated early. Reason::') output(str(error) + ' -- Please complete manually') finally: wikipedia.stopme() #if the report exists, output it try: promobot.report except: pass else: output('=================================') output('Progress report::') output('\n'.join(promobot.report)) closeLogFile()
v_MyPage.py
[edit]"""Class wikipedia.Page is overridden because the regular expression in the original fails to match image links that have external links embedded in their captions, and mismatches image links with wikilinks in their captions """ import wikipedia, re class MyPage(wikipedia.Page): #def __init__(self, site, title): # """temp, for testing fp promotor macro""" # if 'Wikipedia:Sandbox/Veledan/' not in title: # title='Wikipedia:Sandbox/Veledan/' + title # wikipedia.Page.__init__(self, site, title) def linkedPages(self): """Gives the normal (not-interwiki, non-category) pages the page links to, as a list of Page objects Amended regex to spot links to images with an external link in their caption, hence overwritten """ result = [] try: thistxt = wikipedia.removeLanguageLinks(self.get()) except NoPage: return [] except IsRedirectPage: raise thistxt = wikipedia.removeCategoryLinks(thistxt, self.site()) Rlink = re.compile( \ r'''\[\[ #opening wikilink brackets (?P<title>[^\]\|]*) #title=everything up to first ] or | ( #then an optional, repeatable group [^\]\[] #containing at least one non-bracket char OR | \[[^\[\]]*\] #an embedded external link with both brackets [...] OR | \[\[[^\[\]]*\]\] #an embedded complete wikilink [[...]] )* #all that zero or more times \]\] #until ]] is reached. This will not be matched by above expression ''', re.VERBOSE) for match in Rlink.finditer(thistxt): title = match.group('title') if not self.site().isInterwikiLink(title): page = MyPage(self.site(), title) result.append(page) return result