/wikichangewatcher

Python library for easy monitoring of all Wikipedia page edits

Primary LanguagePythonApache License 2.0Apache-2.0

https://raw.githubusercontent.com/eriknyquist/wikichangewatcher/5f8e204db0af39d0a0ed00e5884a38544e11321a/images/wikiwatcher_github_banner.png

tests_badge cov_badge version_badge license_badge

Wikipedia provides an SSE Stream of all edits made to any page across Wikipedia, which allows you to watch all edits made to all wikipedia pages in real time.

WikiChangeWatcher is an SSE client that watches the SSE stream of wikipedia page edits, with some filtering features that allow you to watch for page edit events with specific attributes (e.g. "anonymous" edits with IP addresses in specific ranges, or edits made to a specific page, or edits made by a wikipedia user whose username matches a specific regular expression).

This package is inspired by Tom Scott's WikiParliament project.

Install using pip.

pip install wikichangewatcher

Some example scripts illustrating how to use WikiChangeWatcher are presented in the following sections.

The following example code watches for edits made by any IPv4 or IPv6 address.

# Example script showing how to use WikiChangeWatcher to watch for "anonymous" edits to any
# wikipedia page from any IPv4 or IPv6 address

import time
from wikichangewatcher import WikiChangeWatcher, IpV4Filter, IpV6Filter

# Callback function to run whenever an event matching our IPv4 address pattern is seen
def match_handler(json_data):
    """
    json_data is a JSON-encoded event from the WikiMedia "recent changes" event stream,
    as described here: https://www.mediawiki.org/wiki/Manual:RCFeed
    """
    print("{user} edited {title_url}".format(**json_data))

# Watch for anonymous edits from any IPv4  or IPv6 address
wc = WikiChangeWatcher((IpV4Filter() | IpV6Filter()).on_match(match_handler))
wc.run()

# Watch for page edits forever until KeyboardInterrupt
try:
    while wc.is_running():
        time.sleep(0.1)
except KeyboardInterrupt:
    wc.stop()

The following example code watches for edits made by 3 specific IPv4 address ranges.

# Example script showing how to use WikiChangeWatcher to watch for "anonymous" edits to any
# wikipedia page from specific IP address ranges

import time
from wikichangewatcher import WikiChangeWatcher, IpV4Filter, IpV6Filter

# Callback function to run whenever an event matching our IPv4 address pattern is seen
def match_handler(json_data):
    """
    json_data is a JSON-encoded event from the WikiMedia "recent changes" event stream,
    as described here: https://www.mediawiki.org/wiki/Manual:RCFeed
    """
    print("{user} edited {title_url}".format(**json_data))

# Watch for anonymous edits from some specific IP address ranges
wc = WikiChangeWatcher(IpV4Filter("192.60.38.225-230").on_match(match_handler),
                       IpV6Filter("2601:205:4882:810:5D1D:BC41:61BB:0-ffff").on_match(match_handler))

# Wildcard '*' character can be used in place of a IPv4 or IP46 address field, to ignore that field entirely.
# IPV6 filter with some fields ignored: IpV6Filter("*:*:*:810:5D1D:BC41:*:0-ffff")
# IPV6 filter with some fields ignored: IpV4Filter("192.*.*.225-230")

wc.run()

# Watch for page edits forever until KeyboardInterrupt
try:
    while True:
        time.sleep(0.1)
except KeyboardInterrupt:
    wc.stop()

The following example code watches for edits made by signed-in users with usernames that contain one or more strings matching a regular expression.

# Example script showing how to use WikiChangeWatcher to watch for NON-"anonymous" edits to any
# wikipedia page, by usernames that contain a string matching a provided regular expression

import time
from wikichangewatcher import WikiChangeWatcher, UsernameRegexSearchFilter

# Callback function to run whenever an edit by a user with a username containing our regex is seen
def match_handler(json_data):
    """
    json_data is a JSON-encoded event from the WikiMedia "recent changes" event stream,
    as described here: https://www.mediawiki.org/wiki/Manual:RCFeed
    """
    print("{user} edited {title_url}".format(**json_data))

# Watch for edits made by users with "bot" in their username
wc = WikiChangeWatcher(UsernameRegexSearchFilter(r"[Bb]ot|BOT").on_match(match_handler))

wc.run()

# Watch for page edits forever until KeyboardInterrupt
try:
    while True:
        time.sleep(0.1)
except KeyboardInterrupt:
    wc.stop()

The following example code watches for any page edit events where the specified JSON field matches contains one or more matches of a regular expression (available JSON fields and their descriptions can be found here).

# Example script showing how to use WikiChangeWatcher to filter page edit events
# by a regular expression match in an arbitrary named field from the JSON event
# provided by the SSE stream of wikipedia page edits

import time
from wikichangewatcher import WikiChangeWatcher, FieldRegexSearchFilter

# Callback function to run whenever an edit is made to a page that has a regex match in the page URL
def match_handler(json_data):
    """
    json_data is a JSON-encoded event from the WikiMedia "recent changes" event stream,
    as described here: https://www.mediawiki.org/wiki/Manual:RCFeed
    """
    print("{user} edited {title_url}".format(**json_data))

# Watch for edits made to any page that has the word "publish" in the page URL
# ("title_url" field in the JSON object)
wc = WikiChangeWatcher(FieldRegexSearchFilter("title_url", r"[Pp]ublish").on_match(match_handler))

wc.run()

# Watch for page edits forever until KeyboardInterrupt
try:
    while True:
        time.sleep(0.1)
except KeyboardInterrupt:
    wc.stop()

The following example watches for anonymous page edits to a specific page URL.

# Example script showing how to use WikiChangeWatcher to watch for "anonymous" edits to
# a specific wikipedia page

import time
from wikichangewatcher import WikiChangeWatcher, FilterCollection, IpV4Filter, PageUrlFilter

# Callback function to run whenever an event matching our filters is seen
def match_handler(json_data):
    """
    json_data is a JSON-encoded event from the WikiMedia "recent changes" event stream,
    as described here: https://www.mediawiki.org/wiki/Manual:RCFeed
    """
    print("{user} edited {title_url}".format(**json_data))

# Default match type is is MatchType.ALL
filters = FilterCollection(
    # Filter for any edits to a specific wikipedia page URL
    PageUrlFilter("https://es.wikipedia.org/wiki/Reclus_(La_Rioja)"),

    # Filter for any IP address (any anonymous edit)
    IpV4Filter("*.*.*.*"),
).on_match(match_handler)


wc = WikiChangeWatcher(filters)

wc.run()

# Watch for page edits forever until KeyboardInterrupt
try:
    while True:
        time.sleep(0.1)
except KeyboardInterrupt:
    wc.stop()

The following example watches for page edits to several specific page URLs made by user with the word "bot" in their username.

# Example script showing how to use WikiChangeWatcher to watch for edit to specific
# wikipedia page URLs by users with the word "bot" in their name

import time
from wikichangewatcher import WikiChangeWatcher, FilterCollection, UsernameRegexSearchFilter, PageUrlFilter, MatchType

# Callback function to run whenever an event matching our filters is seen
def match_handler(json_data):
    """
    json_data is a JSON-encoded event from the WikiMedia "recent changes" event stream,
    as described here: https://www.mediawiki.org/wiki/Manual:RCFeed
    """
    print("{user} edited {title_url}".format(**json_data))

# Make a filter collection that matches any one of several wikipedia pages
page_urls = FilterCollection(
    # Filters for any edits to multiple specific wikipedia page URLs
    PageUrlFilter("https://en.wikipedia.org/wiki/Python_(programming_language)"),
    PageUrlFilter("https://en.wikipedia.org/wiki/CPython"),
    PageUrlFilter("https://en.wikipedia.org/wiki/Server-sent_events"),
).set_match_type(MatchType.ANY)

# Make a filter collection that matches one of the page URLs, *and* a specific username regex
main_filter = FilterCollection(
    page_urls,
    UsernameRegexSearchFilter(r"[Bb][Oo][Tt]")
).set_match_type(MatchType.ALL).on_match(match_handler)

wc = WikiChangeWatcher(main_filter)

wc.run()

# Watch for page edits forever until KeyboardInterrupt
try:
    while True:
        time.sleep(0.1)
except KeyboardInterrupt:
    wc.stop()

Instead of creating FilterCollection classes directly, you can instead use bitwise AND & and bitwise OR | to combine filter objects.

For example, this code uses the bitwise OR operator to create a filter that matches any IPv4 address, or any IPv6 address:

from wikichangewatcher import IpV4Filter, IpV6Filter

# Callback function to run whenever an event matching our filters is seen
def match_handler(json_data):
    print("{user} edited {title_url}".format(**json_data))

filter_collection = (IpV4Filter() | IpV6Filter()).on_match(match_handler)

And this code creates an equivalent filter, but uses the FilterCollection class directly instead:

from wikichangewatcher import IpV4Filter, IpV6Filter, FilterCollection, MatchType

# Callback function to run whenever an event matching our filters is seen
def match_handler(json_data):
    print("{user} edited {title_url}".format(**json_data))

filter_collection = FilterCollection(
    IpV4Filter(), IpV6Filter()
).set_match_type(MatchType.ANY).on_match(match_handler)

Finally, here is a slightly more complex example, which uses both bitwise AND / OR operators together to create a filter that matches any IPv4 or IPv6 address, and a specific page URL:

from wikichangewatcher import IpV4Filter, IpV6Filter, PageUrlFilter

PAGE_URL = "https://en.wikipedia.org/wiki/Hayaguchi_Station"

# Callback function to run whenever an event matching our filters is seen
def match_handler(json_data):
    print("{user} edited {title_url}".format(**json_data))

filter_collection = ((IpV4Filter() | IpV6Filter()) & PageUrlFilter(PAGE_URL)).on_match(match_handler)

The following example watches for anonymous page edits to any wikipedia page, from IP address ranges that were found to be publicly listed as owned by various US government department and agencies (mostly California, some federal).

If you want to look up some IP addresses owned by your local governments, or companies, it's pretty easy, I just went to https://ip-netblocks.whoisxmlapi.com/ and searched for "california department of" as the company name.

# Example script showing how to use WikiChangeWatcher to watch for "anonymous" edits to any
# wikipedia page from IP address ranges that are publicly listed as being owned by various US government departments

import time
from wikichangewatcher import WikiChangeWatcher, FilterCollection, IpV4Filter, IpV6Filter, MatchType

# Callback function to run whenever an event matching one of our IPv4 address ranges is seen
def match_handler(json_data):
    """
    json_data is a JSON-encoded event from the WikiMedia "recent changes" event stream,
    as described here: https://www.mediawiki.org/wiki/Manual:RCFeed
    """
    print("{user} edited {title_url}".format(**json_data))


filter_collection = FilterCollection(
    IpV4Filter("136.200.0-255.0-255"),                                    # IP4 range assigned to CA dept. of water resources
    IpV4Filter("151.143.0-255.0-255"),                                    # IP4 range assigned to CA dept. of technology
    IpV4Filter("160.88.0-255.0-255"),                                     # IP4 range assigned to CA dept. of insurance
    IpV4Filter("192.56.110.0-255"),                                       # IP4 range #1 assigned to CA dept. of corrections
    IpV4Filter("153.48.0-255.0-255"),                                     # IP4 range #2 assigned to CA dept. of corrections
    IpV4Filter("149.136.0-255.0-255"),                                    # IP4 range assigned to CA dept. of transportation
    IpV6Filter("2602:814:5000-5fff:0-ffff:0-ffff:0-ffff:0-ffff:0-ffff"),  # IP6 range assigned CA dept. of transportation
    IpV4Filter("192.251.92.0-255"),                                       # IP4 range assigned to CA dept. of general services
    IpV4Filter("159.145.0-255.0-255"),                                    # IP4 range assigned to CA dept. of consumer affairs
    IpV4Filter("167.10.0-255.0-255"),                                     # IP4 range assigned to CA dept. of justice
    IpV4Filter("192.58.200-203.0-255"),                                   # IP4 range assigned to Bureau of Justice Statistics in WA
    IpV6Filter("2607:f330:0-ffff:0-ffff:0-ffff:0-ffff:0-ffff:0-ffff")     # IP6 range assigned to the US dept. of justice in WA
).set_match_type(MatchType.ALL).on_match(match_handler)

wc = WikiChangeWatcher(filter_collection)
wc.run()

# Watch for page edits forever until KeyboardInterrupt
try:
    while True:
        time.sleep(0.1)
except KeyboardInterrupt:
    wc.stop()

The following example watches for any edit to any wikipedia page, and updates a running average of the rate of page edits per minute, which is printed to stdout once every 5 seconds.

# Example script showing how to use WikiChangeWatcher to watch for "anonymous" edits to any
# wikipedia page from specific IP address ranges

import time
import statistics
import queue

from wikichangewatcher import WikiChangeWatcher


# Max. number of samples in the averaging window
MAX_WINDOW_LEN = 6

# Interval between new samples for the averaging window, in seconds
INTERVAL_SECS = 5


class EditRateCounter():
    """
    Tracks total number of page edits per minute across all of wikipedia,
    using a simple averaging window
    """
    def __init__(self, interval_secs=INTERVAL_SECS):
        self._edit_count = 0
        self._start_time = None
        self._interval_secs = interval_secs
        self._queue = queue.Queue()
        self._window = []

    # Callback function to run whenever an edit event is seen
    def edit_handler(self, json_data):
        """
        json_data is a JSON-encoded event from the WikiMedia "recent changes" event stream,
        as described here: https://www.mediawiki.org/wiki/Manual:RCFeed
        """
        self._edit_count += 1

    # Add an edit rate sample to the averaging window, and return the new average
    def _add_to_window(self, edits_per_min):
        self._window.append(edits_per_min)
        if len(self._window) > MAX_WINDOW_LEN:
            self._window.pop(0)

        return statistics.mean(self._window)

    def run(self):
        if self._start_time is None:
            self._start_time = time.time()

        if (time.time() - self._start_time) >= self._interval_secs:
            # interval is up, calculate new rate and put it on the queue
            edits_per_min = float(self._edit_count) * (60.0 / self._interval_secs)
            self._queue.put((self._add_to_window(edits_per_min), self._edit_count))
            self._edit_count = 0
            self._start_time = time.time()

    def get_rate(self):
        ret = None

        try:
            ret = self._queue.get(block=False)
        except queue.Empty:
            pass

        return ret

# Create rate counter class to monitor page edit rate over time
ratecounter = EditRateCounter()

# Create a watcher with no filters-- we want to see every single edit
wc = WikiChangeWatcher().on_edit(ratecounter.edit_handler)

wc.run()

# Watch for page edits forever until KeyboardInterrupt
try:
    while True:
        ratecounter.run()
        new_rate = ratecounter.get_rate()
        if new_rate:
            rate, since_last = new_rate
            print(f"{rate:.2f} avg. page edits per min. ({since_last} in the last {INTERVAL_SECS} secs)")
except KeyboardInterrupt:
    wc.stop()

A CLI program called wikiwatch is provided, which uses the wikichangewatcher package to provide some monitoring capabilities at the command line:

usage: wikiwatch [-h] [-a ADDRESS] [-u USERNAME_REGEX] [-f FIELD_NAME VALUE_RGX]
                 [-s FORMAT_STRING] [--version]

Real-time monitoring of global Wikipedia page edits, with flexible filtering
features.

options:
  -h, --help            show this help message and exit
  -a ADDRESS, --address ADDRESS
                        Adds an IPv4 or Ipv6 address range to look for. Any
                        anonymous edits made by IP addresses in this range
                        will be displayed. Each dot-separated field (for IPv4
                        addresses) or colon-separated field (for IPv6 addresses)
                        may be optionally replaced with with an asterisk (which
                        acts as a wildcard, matching any value), or a range of
                        values. For example, the address range "*.22.33.0-55"
                        would match all IPv4 addresses in the range 0.22.33.0
                        through 255.22.33.50. This option can be used multiple
                        times to add multiple IP address filters.
  -u USERNAME_REGEX, --username-regex USERNAME_REGEX
                        Adds a username regex to look for. Any edits made by
                        logged-in users with a username that matches this
                        regular expression will be displayed. This option can be
                        used multiple times to add multiple username filters.
  -f FIELD_NAME VALUE_RGX, --field FIELD_NAME VALUE_RGX
                        Adds a regex to look for in a specific named field in
                        the JSON event provided by the wikimedia recent changes
                        stream (described here
                        https://www.mediawiki.org/wiki/Manual:RCFeed). Any edit
                        events which have a value matching the VALUE_RGX regular
                        expression stored in the FIELD_NAME field will be
                        displayed. This option can be used multiple times to add
                        multiple named field filters.
  -s FORMAT_STRING, --format-string FORMAT_STRING
                        Define a custom format string to control how filtered
                        results are displayed. Format tokens may be used to
                        display data from any named field in the JSON event
                        described at
                        https://www.mediawiki.org/wiki/Manual:RCFeed. Format
                        tokens must be in the form "{field_name}", where
                        "field_name" is the name of any field from the JSON
                        event. This option can only be used once (Default:
                        "{user} edited {title_url}").
  --version             Show version and exit.

NOTE: if run without arguments, then all anonymous edits (any IPv4 or IPv6
address) will be shown.

EXAMPLES:

Show only edits made by one of two specific IP addresses:

    wikiwatch -a 89.44.33.22 -a 2001:0db8:85a3:0000:0000:8a2e:0370:7334

Show only edits made by IPv4 addresses in the range 88.44.0-33.0-22:

    wikiwatch -a 88.44.0-33.0-22

Show only edits made by IPv4 addresses in the range 232.22.0-255.0-255:

    wikiwatch -a 232.22.*.*

Show only edits made by usernames that contain the word "Bot" or "bot":

    wikiwatch -f user "[Bb]ot"

Contributions are welcome, please open a pull request at https://github.com/eriknyquist/wikichangewatcher/pulls. You will need to install packages required for development by doing pip install -r dev_requirements.txt.

Please ensure that all existing tests pass, new test(s) are added if required, and the code coverage check passes.

  • Run tests with python setup.py test.
  • Run tests and and generate code coverage report with python code_coverage.py (this script will report an error if coverage is below 90%)

If you have any questions about / need help with contributions or tests, please contact Erik at eknyquist@gmail.com.