Source code for scripts.image

#!/usr/bin/env python3
"""
This script can be used to change one image to another or remove an image.

Syntax:

    python pwb.py image image_name [new_image_name]

If only one command-line parameter is provided then that image will be removed;
if two are provided, then the first image will be replaced by the second one on
all pages.

Command line options:

-summary:  Provide a custom edit summary. If the summary includes spaces,
           surround it with single quotes, such as:
           -summary:'My edit summary'
-always    Don't prompt to make changes, just do them.
-loose     Do loose replacements. This will replace all occurrences of the name
           of the image (and not just explicit image syntax). This should work
           to catch all instances of the image, including where it is used as a
           template parameter or in image galleries. However, it can also make
           more mistakes. This only works with image replacement, not image
           removal.

Examples
--------

The image "FlagrantCopyvio.jpg" is about to be deleted, so let's first remove
it from everything that displays it:

    python pwb.py image FlagrantCopyvio.jpg

The image "Flag.svg" has been uploaded, making the old "Flag.jpg" obsolete:

    python pwb.py image Flag.jpg Flag.svg

"""
#
# (C) Pywikibot team, 2013-2022
#
# Distributed under the terms of the MIT license.
#
from __future__ import annotations

import re

import pywikibot
from pywikibot import i18n, pagegenerators
from pywikibot.bot import SingleSiteBot
from pywikibot.textlib import case_escape, ignore_case
from scripts.replace import ReplaceRobot as ReplaceBot


[docs] class ImageRobot(ReplaceBot): """This bot will replace or remove all occurrences of an old image.""" def __init__(self, generator, old_image: str, new_image: str = '', **kwargs) -> None: """ Initializer. :param generator: the pages to work on :type generator: iterable :param old_image: the title of the old image (without namespace) :param new_image: the title of the new image (without namespace), or None if you want to remove the image """ self.available_options.update({ 'summary': None, 'loose': False, }) SingleSiteBot.__init__(self, **kwargs) self.old_image = old_image self.new_image = new_image param = { 'old': self.old_image, 'new': self.new_image, 'file': self.old_image, } summary = self.opt.summary or i18n.twtranslate( self.site, 'image-replace' if self.new_image else 'image-remove', param) namespace = self.site.namespaces[6] escaped = case_escape(namespace.case, self.old_image, underscore=True) if not self.opt.loose or not self.new_image: image_regex = re.compile( r'\[\[ *(?:{})\s*:\s*{} *(?P<parameters>\|' r'(?:[^\[\]]|\[\[[^\]]+\]\]|\[[^\]]+\])*|) *\]\]' .format('|'.join(ignore_case(s) for s in namespace), escaped)) else: image_regex = re.compile(r'' + escaped) replacements = [] if not self.opt.loose and self.new_image: replacements.append((image_regex, '[[{}:{}\\g<parameters>]]' .format( self.site.namespaces.FILE.custom_name, self.new_image))) else: replacements.append((image_regex, self.new_image)) super().__init__(generator, replacements, always=self.opt.always, site=self.site, summary=summary)
[docs] def main(*args: str) -> None: """ Process command line arguments and invoke bot. If args is an empty list, sys.argv is used. :param args: command line arguments """ old_image = '' new_image = '' options = {} for argument in pywikibot.handle_args(args): arg, _, value = argument.partition(':') if arg in ('-always', '-loose'): options[arg[1:]] = True elif arg == '-summary': options[arg[1:]] = value or pywikibot.input( 'Choose an edit summary: ') elif old_image: new_image = arg else: old_image = arg if old_image: site = pywikibot.Site() old_imagepage = pywikibot.FilePage(site, old_image) gen = old_imagepage.using_pages() preloading_gen = pagegenerators.PreloadingGenerator(gen) bot = ImageRobot(preloading_gen, old_image, new_image, site=site, **options) bot.run() else: pywikibot.bot.suggest_help(missing_parameters=['old image'])
if __name__ == '__main__': main()