Resolved conflicts in __init__.py and managers.pytags/v0.1^2
@@ -1,4 +1,6 @@ | |||||
*.pyc | *.pyc | ||||
*.egg | |||||
*.egg-info | *.egg-info | ||||
.DS_Store | .DS_Store | ||||
build/ | |||||
build | |||||
docs/_build |
@@ -1,39 +0,0 @@ | |||||
[EarwigBot](http://en.wikipedia.org/wiki/User:EarwigBot) is a | |||||
[Python](http://python.org/) robot that edits | |||||
[Wikipedia](http://en.wikipedia.org/) and interacts with people over | |||||
[IRC](http://en.wikipedia.org/wiki/Internet_Relay_Chat). | |||||
# History | |||||
Development began, based on the | |||||
[Pywikipedia framework](http://pywikipediabot.sourceforge.net/), in early 2009. | |||||
Approval for its fist task, a | |||||
[copyright violation detector](http://en.wikipedia.org/wiki/Wikipedia:Bots/Requests_for_approval/EarwigBot_1), | |||||
was carried out in May, and the bot has been running consistently ever since | |||||
(with the exception of Jan/Feb 2011). It currently handles | |||||
[several ongoing tasks](http://en.wikipedia.org/wiki/User:EarwigBot#Tasks), | |||||
ranging from statistics generation to category cleanup, and on-demand tasks | |||||
such as WikiProject template tagging. Since it started running, the bot has | |||||
made over 45,000 edits. | |||||
A project to rewrite it from scratch began in early April 2011, thus moving | |||||
away from the Pywikipedia framework and allowing for less overall code, better | |||||
integration between bot parts, and easier maintenance. | |||||
# Installation | |||||
## Dependencies | |||||
EarwigBot uses the MySQL library | |||||
[oursql](http://packages.python.org/oursql/) (>= 0.9.2) for communicating with | |||||
MediaWiki databases, and some tasks use their own tables for storage. | |||||
Additionally, the afc_history task uses | |||||
[matplotlib](http://matplotlib.sourceforge.net/) and | |||||
[numpy](http://numpy.scipy.org/) for graphing AfC statistics. Neither of these | |||||
modules are required for the main bot itself. | |||||
`earwigbot.wiki.copyright` requires access to a search engine for detecting | |||||
copyright violations. Currently, | |||||
[Yahoo! BOSS](http://developer.yahoo.com/search/boss/) is the only engine | |||||
supported, and this requires | |||||
[oauth2](https://github.com/simplegeo/python-oauth2). |
@@ -0,0 +1,510 @@ | |||||
EarwigBot | |||||
========= | |||||
EarwigBot_ is a Python_ robot that edits Wikipedia_ and interacts with people | |||||
over IRC_. This file provides a basic overview of how to install and setup the | |||||
bot; more detailed information is located in the ``docs/`` directory (available | |||||
online at PyPI_). | |||||
History | |||||
------- | |||||
Development began, based on the `Pywikipedia framework`_, in early 2009. | |||||
Approval for its fist task, a `copyright violation detector`_, was carried out | |||||
in May, and the bot has been running consistently ever since (with the | |||||
exception of Jan/Feb 2011). It currently handles `several ongoing tasks`_ | |||||
ranging from statistics generation to category cleanup, and on-demand tasks | |||||
such as WikiProject template tagging. Since it started running, the bot has | |||||
made over 50,000 edits. | |||||
A project to rewrite it from scratch began in early April 2011, thus moving | |||||
away from the Pywikipedia framework and allowing for less overall code, better | |||||
integration between bot parts, and easier maintenance. | |||||
Installation | |||||
------------ | |||||
This package contains the core ``earwigbot``, abstracted enough that it should | |||||
be usable and customizable by anyone running a bot on a MediaWiki site. Since | |||||
it is component-based, the IRC components can be disabled if desired. IRC | |||||
commands and bot tasks specific to `my instance of EarwigBot`_ that I don't | |||||
feel the average user will need are available from the repository | |||||
`earwigbot-plugins`_. | |||||
It's recommended to run the bot's unit tests before installing. Run ``python | |||||
setup.py test`` from the project's root directory. Note that some | |||||
tests require an internet connection, and others may take a while to run. | |||||
Coverage is currently rather incomplete. | |||||
Latest release (v0.1) | |||||
~~~~~~~~~~~~~~~~~~~~~ | |||||
EarwigBot is available from the `Python Package Index`_, so you can install the | |||||
latest release with ``pip install earwigbot`` (`get pip`_). | |||||
You can also install it from source [1]_ directly:: | |||||
curl -Lo earwigbot.tgz https://github.com/earwig/earwigbot/tarball/v0.1 | |||||
tar -xf earwigbot.tgz | |||||
cd earwig-earwigbot-* | |||||
python setup.py install | |||||
cd .. | |||||
rm -r earwigbot.tgz earwig-earwigbot-* | |||||
Development version | |||||
~~~~~~~~~~~~~~~~~~~ | |||||
You can install the development version of the bot from ``git`` by using | |||||
setuptools/distribute's ``develop`` command [1]_, probably on the ``develop`` | |||||
branch which contains (usually) working code. ``master`` contains the latest | |||||
release. EarwigBot uses `git flow`_, so you're free to | |||||
browse by tags or by new features (``feature/*`` branches):: | |||||
git clone git://github.com/earwig/earwigbot.git earwigbot | |||||
cd earwigbot | |||||
python setup.py develop | |||||
Setup | |||||
----- | |||||
The bot stores its data in a "working directory", including its config file and | |||||
databases. This is also the location where you will place custom IRC commands | |||||
and bot tasks, which will be explained later. It doesn't matter where this | |||||
directory is, as long as the bot can write to it. | |||||
Start the bot with ``earwigbot path/to/working/dir``, or just ``earwigbot`` if | |||||
the working directory is the current directory. It will notice that no | |||||
``config.yml`` file exists and take you through the setup process. | |||||
There is currently no way to edit the ``config.yml`` file from within the bot | |||||
after it has been created, but YAML is a very straightforward format, so you | |||||
should be able to make any necessary changes yourself. Check out the | |||||
`explanation of YAML`_ on Wikipedia for help. | |||||
After setup, the bot will start. This means it will connect to the IRC servers | |||||
it has been configured for, schedule bot tasks to run at specific times, and | |||||
then wait for instructions (as commands on IRC). For a list of commands, say | |||||
"``!help``" (commands are messages prefixed with an exclamation mark). | |||||
You can stop the bot at any time with Control+C, same as you stop a normal | |||||
Python program, and it will try to exit safely. You can also use the | |||||
"``!quit``" command on IRC. | |||||
Customizing | |||||
----------- | |||||
The bot's working directory contains a ``commands`` subdirectory and a | |||||
``tasks`` subdirectory. Custom IRC commands can be placed in the former, | |||||
whereas custom wiki bot tasks go into the latter. Developing custom modules is | |||||
explained below, and in more detail through the bot's documentation on PyPI_. | |||||
Note that custom commands will override built-in commands and tasks with the | |||||
same name. | |||||
``Bot`` and ``BotConfig`` | |||||
~~~~~~~~~~~~~~~~~~~~~~~~~ | |||||
`earwigbot.bot.Bot`_ is EarwigBot's main class. You don't have to instantiate | |||||
this yourself, but it's good to be familiar with its attributes and methods, | |||||
because it is the main way to communicate with other parts of the bot. A | |||||
``Bot`` object is accessible as an attribute of commands and tasks (i.e., | |||||
``self.bot``). | |||||
The most useful attributes are: | |||||
- ``bot.config``: an instance of ``BotConfig``, for accessing the bot's | |||||
configuration data (see below). | |||||
- ``bot.commands``: the bot's ``CommandManager``, which is used internally to | |||||
run IRC commands (through ``bot.commands.call()``, which you shouldn't have | |||||
to use); you can safely reload all commands with ``bot.commands.load()``. | |||||
- ``bot.tasks``: the bot's ``TaskManager``, which can be used to start tasks | |||||
with ``bot.tasks.start(task_name, **kwargs)``. ``bot.tasks.load()`` can be | |||||
used to safely reload all tasks. | |||||
- ``bot.frontend`` / ``bot.watcher``: instances of ``earwigbot.irc.Frontend`` | |||||
and ``earwigbot.irc.Watcher``, respectively, which represent the bot's | |||||
connections to these two servers; you can, for example, send a message to the | |||||
frontend with ``bot.frontend.say(chan, msg)`` (more on communicating with IRC | |||||
below). | |||||
- ``bot.wiki``: interface with `the Wiki Toolset`_ (see below). | |||||
- Finally, ``bot.restart()`` (restarts IRC components and reloads config, | |||||
commands, and tasks) and ``bot.stop()`` can be used almost anywhere. Both | |||||
take an optional "reason" that will be logged and used as the quit message | |||||
when disconnecting from IRC. | |||||
`earwigbot.config.BotConfig`_ stores configuration information for the bot. Its | |||||
docstring explains what each attribute is used for, but essentially each "node" | |||||
(one of ``config.components``, ``wiki``, ``tasks``, ``irc``, and ``metadata``) | |||||
maps to a section of the bot's ``config.yml`` file. For example, if | |||||
``config.yml`` includes something like:: | |||||
irc: | |||||
frontend: | |||||
nick: MyAwesomeBot | |||||
channels: | |||||
- "##earwigbot" | |||||
- "#channel" | |||||
- "#other-channel" | |||||
...then ``config.irc["frontend"]["nick"]`` will be ``"MyAwesomeBot"`` and | |||||
``config.irc["frontend"]["channels"]`` will be ``["##earwigbot", "#channel", | |||||
"#other-channel"]``. | |||||
Custom IRC commands | |||||
~~~~~~~~~~~~~~~~~~~ | |||||
Custom commands are subclasses of `earwigbot.commands.BaseCommand`_ that | |||||
override ``BaseCommand``'s ``process()`` (and optionally ``check()``) methods. | |||||
``BaseCommand``'s docstrings should explain what each attribute and method is | |||||
for and what they should be overridden with, but these are the basics: | |||||
- Class attribute ``name`` is the name of the command. This must be specified. | |||||
- Class attribute ``hooks`` is a list of the "IRC events" that this command | |||||
might respond to. It defaults to ``["msg"]``, but options include | |||||
``"msg_private"`` (for private messages only), ``"msg_public"`` (for channel | |||||
messages only), and ``"join"`` (for when a user joins a channel). See the | |||||
afc_status_ plugin for a command that responds to other hook types. | |||||
- Method ``check()`` is passed a ``Data`` [2]_ object, and should return | |||||
``True`` if you want to respond to this message, or ``False`` otherwise. The | |||||
default behavior is to return ``True`` only if ``data.is_command`` is | |||||
``True`` and ``data.command == self.name``, which is suitable for most cases. | |||||
A common, straightforward reason for overriding is if a command has aliases | |||||
(see chanops_ for an example). Note that by returning ``True``, you prevent | |||||
any other commands from responding to this message. | |||||
- Method ``process()`` is passed the same ``Data`` object as ``check()``, but | |||||
only if ``check()`` returned ``True``. This is where the bulk of your command | |||||
goes. To respond to IRC messages, there are a number of methods of | |||||
``BaseCommand`` at your disposal. See the the test_ command for a simple | |||||
example, or look in BaseCommand's ``__init__`` method for the full list. | |||||
The most common ones are ``self.say(chan_or_user, msg)``, | |||||
``self.reply(data, msg)`` (convenience function; sends a reply to the | |||||
issuer of the command in the channel it was received), | |||||
``self.action(chan_or_user, msg)``, ``self.notice(chan_or_user, msg)``, | |||||
``self.join(chan)``, and ``self.part(chan)``. | |||||
It's important to name the command class ``Command`` within the file, or else | |||||
the bot might not recognize it as a command. The name of the file doesn't | |||||
really matter and need not match the command's name, but this is recommended | |||||
for readability. | |||||
The bot has a wide selection of built-in commands and plugins to act as sample | |||||
code and/or to give ideas. Start with test_, and then check out chanops_ and | |||||
afc_status_ for some more complicated scripts. | |||||
Custom bot tasks | |||||
~~~~~~~~~~~~~~~~ | |||||
Custom tasks are subclasses of `earwigbot.tasks.BaseTask`_ that override | |||||
``BaseTask``'s ``run()`` (and optionally ``setup()``) methods. | |||||
``BaseTask``'s docstrings should explain what each attribute and method is for | |||||
and what they should be overridden with, but these are the basics: | |||||
- Class attribute ``name`` is the name of the task. This must be specified. | |||||
- Class attribute ``number`` can be used to store an optional "task number", | |||||
possibly for use in edit summaries (to be generated with ``make_summary()``). | |||||
For example, EarwigBot's ``config.wiki["summary"]`` is | |||||
``"([[WP:BOT|Bot]]; [[User:EarwigBot#Task $1|Task $1]]): $2"``, which the | |||||
task class's ``make_summary(comment)`` method will take and replace ``$1`` | |||||
with the task number and ``$2`` with the details of the edit. | |||||
Additionally, ``shutoff_enabled()`` (which checks whether the bot has been | |||||
told to stop on-wiki by checking the content of a particular page) can check | |||||
a different page for each task using similar variables. EarwigBot's | |||||
``config.wiki["shutoff"]["page"]`` is ``"User:$1/Shutoff/Task $2"``; ``$1`` | |||||
is substituted with the bot's username, and ``$2`` is substituted with the | |||||
task number, so, e.g., task #14 checks the page | |||||
``[[User:EarwigBot/Shutoff/Task 14]].`` If the page's content does *not* | |||||
match ``config.wiki["shutoff"]["disabled"]`` (``"run"`` by default), then | |||||
shutoff is considered to be *enabled* and ``shutoff_enabled()`` will return | |||||
``True``, indicating the task should not run. If you don't intend to use | |||||
either of these methods, feel free to leave this attribute blank. | |||||
- Method ``setup()`` is called *once* with no arguments immediately after the | |||||
task is first loaded. Does nothing by default; treat it like an | |||||
``__init__()`` if you want (``__init__()`` does things by default and a | |||||
dedicated setup method is often easier than overriding ``__init__()`` and | |||||
using ``super``). | |||||
- Method ``run()`` is called with any number of keyword arguments every time | |||||
the task is executed (by ``bot.tasks.start(task_name, **kwargs)``, usually). | |||||
This is where the bulk of the task's code goes. For interfacing with | |||||
MediaWiki sites, read up on `the Wiki Toolset`_ below. | |||||
Tasks have access to ``config.tasks[task_name]`` for config information, which | |||||
is a node in ``config.yml`` like every other attribute of ``bot.config``. This | |||||
can be used to store, for example, edit summaries, or templates to append to | |||||
user talk pages, so that these can be easily changed without modifying the task | |||||
itself. | |||||
It's important to name the task class ``Task`` within the file, or else the bot | |||||
might not recognize it as a task. The name of the file doesn't really matter | |||||
and need not match the task's name, but this is recommended for readability. | |||||
See the built-in wikiproject_tagger_ task for a relatively straightforward | |||||
task, or the afc_statistics_ plugin for a more complicated one. | |||||
The Wiki Toolset | |||||
---------------- | |||||
EarwigBot's answer to the `Pywikipedia framework`_ is the Wiki Toolset | |||||
(``earwigbot.wiki``), which you will mainly access through ``bot.wiki``. | |||||
``bot.wiki`` provides three methods for the management of Sites - | |||||
``get_site()``, ``add_site()``, and ``remove_site()``. Sites are objects that | |||||
simply represent a MediaWiki site. A single instance of EarwigBot (i.e. a | |||||
single *working directory*) is expected to relate to a single site or group of | |||||
sites using the same login info (like all WMF wikis with CentralAuth). | |||||
Load your default site (the one that you picked during setup) with | |||||
``site = bot.wiki.get_site()``. | |||||
Dealing with other sites | |||||
~~~~~~~~~~~~~~~~~~~~~~~~ | |||||
*Skip this section if you're only working with one site.* | |||||
If a site is *already known to the bot* (meaning that it is stored in the | |||||
``sites.db`` file, which includes just your default wiki at first), you can | |||||
load a site with ``site = bot.wiki.get_site(name)``, where ``name`` might be | |||||
``"enwiki"`` or ``"frwiktionary"`` (you can also do | |||||
``site = bot.wiki.get_site(project="wikipedia", lang="en")``). Recall that not | |||||
giving any arguments to ``get_site()`` will return the default site. | |||||
``add_site()`` is used to add new sites to the sites database. It may be called | |||||
with similar arguments as ``get_site()``, but the difference is important. | |||||
``get_site()`` only needs enough information to identify the site in its | |||||
database, which is usually just its name; the database stores all other | |||||
necessary connection info. With ``add_site()``, you need to provide enough | |||||
connection info so the toolset can successfully access the site's API/SQL | |||||
databases and store that information for later. That might not be much; for | |||||
WMF wikis, you can usually use code like this:: | |||||
project, lang = "wikipedia", "es" | |||||
try: | |||||
site = bot.wiki.get_site(project=project, lang=lang) | |||||
except earwigbot.SiteNotFoundError: | |||||
# Load site info from http://es.wikipedia.org/w/api.php: | |||||
site = bot.wiki.add_site(project=project, lang=lang) | |||||
This works because EarwigBot assumes that the URL for the site is | |||||
``"//{lang}.{project}.org"`` and the API is at ``/w/api.php``; this might | |||||
change if you're dealing with non-WMF wikis, where the code might look | |||||
something more like:: | |||||
project, lang = "mywiki", "it" | |||||
try: | |||||
site = bot.wiki.get_site(project=project, lang=lang) | |||||
except earwigbot.SiteNotFoundError: | |||||
# Load site info from http://mysite.net/mywiki/it/s/api.php: | |||||
base_url = "http://mysite.net/" + project + "/" + lang | |||||
db_name = lang + project + "_p" | |||||
sql = {host: "sql.mysite.net", db: db_name} | |||||
site = bot.wiki.add_site(base_url=base_url, script_path="/s", sql=sql) | |||||
``remove_site()`` does the opposite of ``add_site()``: give it a site's name | |||||
or a project/lang pair like ``get_site()`` takes, and it'll remove that site | |||||
from the sites database. | |||||
Sites | |||||
~~~~~ | |||||
``Site`` objects provide the following attributes: | |||||
- ``name``: the site's name (or "wikiid"), like ``"enwiki"`` | |||||
- ``project``: the site's project name, like ``"wikipedia"`` | |||||
- ``lang``: the site's language code, like ``"en"`` | |||||
- ``domain``: the site's web domain, like ``"en.wikipedia.org"`` | |||||
and the following methods: | |||||
- ``api_query(**kwargs)``: does an API query with the given keyword arguments | |||||
as params | |||||
- ``sql_query(query, params=(), ...)``: does an SQL query and yields its | |||||
results (as a generator) | |||||
- ``get_replag()``: returns the estimated database replication lag (if we have | |||||
the site's SQL connection info) | |||||
- ``namespace_id_to_name(id, all=False)``: given a namespace ID, returns the | |||||
primary associated namespace name (or a list of all names when ``all`` is | |||||
``True``) | |||||
- ``namespace_name_to_id(name)``: given a namespace name, returns the | |||||
associated namespace ID | |||||
- ``get_page(title, follow_redirects=False)``: returns a ``Page`` object for | |||||
the given title (or a ``Category`` object if the page's namespace is | |||||
"``Category:``") | |||||
- ``get_category(catname, follow_redirects=False)``: returns a ``Category`` | |||||
object for the given title (sans namespace) | |||||
- ``get_user(username)``: returns a ``User`` object for the given username | |||||
Pages (and Categories) | |||||
~~~~~~~~~~~~~~~~~~~~~~ | |||||
Create ``Page`` objects with ``site.get_page(title)``, | |||||
``page.toggle_talk()``, ``user.get_userpage()``, or ``user.get_talkpage()``. | |||||
They provide the following attributes: | |||||
- ``title``: the page's title, or pagename | |||||
- ``exists``: whether the page exists | |||||
- ``pageid``: an integer ID representing the page | |||||
- ``url``: the page's URL | |||||
- ``namespace``: the page's namespace as an integer | |||||
- ``protection``: the page's current protection status | |||||
- ``is_talkpage``: ``True`` if the page is a talkpage, else ``False`` | |||||
- ``is_redirect``: ``True`` if the page is a redirect, else ``False`` | |||||
and the following methods: | |||||
- ``reload()``: forcibly reload the page's attributes (emphasis on *reload* - | |||||
this is only necessary if there is reason to believe they have changed) | |||||
- ``toggle_talk(...)``: returns a content page's talk page, or vice versa | |||||
- ``get()``: returns page content | |||||
- ``get_redirect_target()``: if the page is a redirect, returns its destination | |||||
- ``get_creator()``: returns a ``User`` object representing the first user to | |||||
edit the page | |||||
- ``edit(text, summary, minor=False, bot=True, force=False)``: replaces the | |||||
page's content with ``text`` or creates a new page | |||||
- ``add_section(text, title, minor=False, bot=True, force=False)``: adds a new | |||||
section named ``title`` at the bottom of the page | |||||
- ``copyvio_check(...)``: checks the page for copyright violations | |||||
- ``copyvio_compare(url, ...)``: checks the page like ``copyvio_check()``, but | |||||
against a specific URL | |||||
Additionally, ``Category`` objects (created with ``site.get_category(name)`` or | |||||
``site.get_page(title)`` where ``title`` is in the ``Category:`` namespace) | |||||
provide the following additional method: | |||||
- ``get_members(use_sql=False, limit=None)``: returns a list of page titles in | |||||
the category (limit is ``50`` by default if using the API) | |||||
Users | |||||
~~~~~ | |||||
Create ``User`` objects with ``site.get_user(name)`` or | |||||
``page.get_creator()``. They provide the following attributes: | |||||
- ``name``: the user's username | |||||
- ``exists``: ``True`` if the user exists, or ``False`` if they do not | |||||
- ``userid``: an integer ID representing the user | |||||
- ``blockinfo``: information about any current blocks on the user (``False`` if | |||||
no block, or a dict of ``{"by": blocking_user, "reason": block_reason, | |||||
"expiry": block_expire_time}``) | |||||
- ``groups``: a list of the user's groups | |||||
- ``rights``: a list of the user's rights | |||||
- ``editcount``: the number of edits made by the user | |||||
- ``registration``: the time the user registered as a ``time.struct_time`` | |||||
- ``emailable``: ``True`` if you can email the user, ``False`` if you cannot | |||||
- ``gender``: the user's gender (``"male"``, ``"female"``, or ``"unknown"``) | |||||
and the following methods: | |||||
- ``reload()``: forcibly reload the user's attributes (emphasis on *reload* - | |||||
this is only necessary if there is reason to believe they have changed) | |||||
- ``get_userpage()``: returns a ``Page`` object representing the user's | |||||
userpage | |||||
- ``get_talkpage()``: returns a ``Page`` object representing the user's | |||||
talkpage | |||||
Additional features | |||||
~~~~~~~~~~~~~~~~~~~ | |||||
Not all aspects of the toolset are covered here. Explore `its code and | |||||
docstrings`_ to learn how to use it in a more hands-on fashion. For reference, | |||||
``bot.wiki`` is an instance of ``earwigbot.wiki.SitesDB`` tied to the | |||||
``sites.db`` file in the bot's working directory. | |||||
Tips | |||||
---- | |||||
- Logging_ is a fantastic way to monitor the bot's progress as it runs. It has | |||||
a slew of built-in loggers, and enabling log retention (so logs are saved to | |||||
``logs/`` in the working directory) is highly recommended. In the normal | |||||
setup, there are three log files, each of which "rotate" at a specific time | |||||
(``filename.log`` becomes ``filename.log.2012-04-10``, for example). The | |||||
``debug.log`` file rotates every hour, and maintains six hours of logs of | |||||
every level (``DEBUG`` and up). ``bot.log`` rotates every day at midnight, | |||||
and maintains seven days of non-debug logs (``INFO`` and up). Finally, | |||||
``error.log`` rotates every Sunday night, and maintains four weeks of logs | |||||
indicating unexpected events (``WARNING`` and up). | |||||
To use logging in your commands or tasks (recommended), ``BaseCommand`` and | |||||
``BaseTask`` provide ``logger`` attributes configured for the specific | |||||
command or task. If you're working with other classes, ``bot.logger`` is the | |||||
root logger (``logging.getLogger("earwigbot")`` by default), so you can use | |||||
``getChild`` to make your logger. For example, task loggers are essentially | |||||
``bot.logger.getChild("tasks").getChild(task.name)``. | |||||
- A very useful IRC command is "``!reload``", which reloads all commands and | |||||
tasks without restarting the bot. [3]_ Combined with using the `!git plugin`_ | |||||
for pulling repositories from IRC, this can provide a seamless command/task | |||||
development workflow if the bot runs on an external server and you set up | |||||
its working directory as a git repo. | |||||
- You can run a task by itself instead of the entire bot with ``earwigbot | |||||
path/to/working/dir --task task_name``. | |||||
- Questions, comments, or suggestions about the documentation? `Let me know`_ | |||||
so I can improve it for other people. | |||||
Footnotes | |||||
--------- | |||||
.. [1] ``python setup.py install``/``develop`` may require root, or use the | |||||
``--user`` switch to install for the current user only. | |||||
.. [2] ``Data`` objects are instances of ``earwigbot.irc.Data`` that contain | |||||
information about a single message sent on IRC. Their useful attributes | |||||
are ``chan`` (channel the message was sent from, equal to ``nick`` if | |||||
it's a private message), ``nick`` (nickname of the sender), ``ident`` | |||||
(ident_ of the sender), ``host`` (hostname of the sender), ``msg`` (text | |||||
of the sent message), ``is_command`` (boolean telling whether or not | |||||
this message is a bot command, i.e., whether it is prefixed by ``!``), | |||||
``command`` (if the message is a command, this is the name of the | |||||
command used), and ``args`` (if the message is a command, this is a list | |||||
of the command arguments - for example, if issuing "``!part ##earwig | |||||
Goodbye guys``", ``args`` will equal ``["##earwig", "Goodbye", | |||||
"guys"]``). Note that not all ``Data`` objects will have all of these | |||||
attributes: ``Data`` objects generated by private messages will, but | |||||
ones generated by joins will only have ``chan``, ``nick``, ``ident``, | |||||
and ``host``. | |||||
.. [3] In reality, all this does is call ``bot.commands.load()`` and | |||||
``bot.tasks.load()``! | |||||
.. _EarwigBot: http://en.wikipedia.org/wiki/User:EarwigBot | |||||
.. _Python: http://python.org/ | |||||
.. _Wikipedia: http://en.wikipedia.org/ | |||||
.. _IRC: http://en.wikipedia.org/wiki/Internet_Relay_Chat | |||||
.. _PyPI: http://packages.python.org/earwigbot | |||||
.. _Pywikipedia framework: http://pywikipediabot.sourceforge.net/ | |||||
.. _copyright violation detector: http://en.wikipedia.org/wiki/Wikipedia:Bots/Requests_for_approval/EarwigBot_1 | |||||
.. _several ongoing tasks: http://en.wikipedia.org/wiki/User:EarwigBot#Tasks | |||||
.. _my instance of EarwigBot: http://en.wikipedia.org/wiki/User:EarwigBot | |||||
.. _earwigbot-plugins: https://github.com/earwig/earwigbot-plugins | |||||
.. _Python Package Index: http://pypi.python.org | |||||
.. _get pip: http://pypi.python.org/pypi/pip | |||||
.. _git flow: http://nvie.com/posts/a-successful-git-branching-model/ | |||||
.. _explanation of YAML: http://en.wikipedia.org/wiki/YAML | |||||
.. _earwigbot.bot.Bot: https://github.com/earwig/earwigbot/blob/develop/earwigbot/bot.py | |||||
.. _earwigbot.config.BotConfig: https://github.com/earwig/earwigbot/blob/develop/earwigbot/config.py | |||||
.. _earwigbot.commands.BaseCommand: https://github.com/earwig/earwigbot/blob/develop/earwigbot/commands/__init__.py | |||||
.. _afc_status: https://github.com/earwig/earwigbot-plugins/blob/develop/commands/afc_status.py | |||||
.. _chanops: https://github.com/earwig/earwigbot/blob/develop/earwigbot/commands/chanops.py | |||||
.. _test: https://github.com/earwig/earwigbot/blob/develop/earwigbot/commands/test.py | |||||
.. _earwigbot.tasks.BaseTask: https://github.com/earwig/earwigbot/blob/develop/earwigbot/tasks/__init__.py | |||||
.. _wikiproject_tagger: https://github.com/earwig/earwigbot/blob/develop/earwigbot/tasks/wikiproject_tagger.py | |||||
.. _afc_statistics: https://github.com/earwig/earwigbot-plugins/blob/develop/tasks/afc_statistics.py | |||||
.. _its code and docstrings: https://github.com/earwig/earwigbot/tree/develop/earwigbot/wiki | |||||
.. _logging: http://docs.python.org/library/logging.html | |||||
.. _Let me know: ben.kurtovic@verizon.net | |||||
.. _!git plugin: https://github.com/earwig/earwigbot-plugins/blob/develop/commands/git.py | |||||
.. _ident: http://en.wikipedia.org/wiki/Ident |
@@ -0,0 +1,153 @@ | |||||
# Makefile for Sphinx documentation | |||||
# | |||||
# You can set these variables from the command line. | |||||
SPHINXOPTS = | |||||
SPHINXBUILD = sphinx-build | |||||
PAPER = | |||||
BUILDDIR = _build | |||||
# Internal variables. | |||||
PAPEROPT_a4 = -D latex_paper_size=a4 | |||||
PAPEROPT_letter = -D latex_paper_size=letter | |||||
ALLSPHINXOPTS = -d $(BUILDDIR)/doctrees $(PAPEROPT_$(PAPER)) $(SPHINXOPTS) . | |||||
# the i18n builder cannot share the environment and doctrees with the others | |||||
I18NSPHINXOPTS = $(PAPEROPT_$(PAPER)) $(SPHINXOPTS) . | |||||
.PHONY: help clean html dirhtml singlehtml pickle json htmlhelp qthelp devhelp epub latex latexpdf text man changes linkcheck doctest gettext | |||||
help: | |||||
@echo "Please use \`make <target>' where <target> is one of" | |||||
@echo " html to make standalone HTML files" | |||||
@echo " dirhtml to make HTML files named index.html in directories" | |||||
@echo " singlehtml to make a single large HTML file" | |||||
@echo " pickle to make pickle files" | |||||
@echo " json to make JSON files" | |||||
@echo " htmlhelp to make HTML files and a HTML help project" | |||||
@echo " qthelp to make HTML files and a qthelp project" | |||||
@echo " devhelp to make HTML files and a Devhelp project" | |||||
@echo " epub to make an epub" | |||||
@echo " latex to make LaTeX files, you can set PAPER=a4 or PAPER=letter" | |||||
@echo " latexpdf to make LaTeX files and run them through pdflatex" | |||||
@echo " text to make text files" | |||||
@echo " man to make manual pages" | |||||
@echo " texinfo to make Texinfo files" | |||||
@echo " info to make Texinfo files and run them through makeinfo" | |||||
@echo " gettext to make PO message catalogs" | |||||
@echo " changes to make an overview of all changed/added/deprecated items" | |||||
@echo " linkcheck to check all external links for integrity" | |||||
@echo " doctest to run all doctests embedded in the documentation (if enabled)" | |||||
clean: | |||||
-rm -rf $(BUILDDIR)/* | |||||
html: | |||||
$(SPHINXBUILD) -b html $(ALLSPHINXOPTS) $(BUILDDIR)/html | |||||
@echo | |||||
@echo "Build finished. The HTML pages are in $(BUILDDIR)/html." | |||||
dirhtml: | |||||
$(SPHINXBUILD) -b dirhtml $(ALLSPHINXOPTS) $(BUILDDIR)/dirhtml | |||||
@echo | |||||
@echo "Build finished. The HTML pages are in $(BUILDDIR)/dirhtml." | |||||
singlehtml: | |||||
$(SPHINXBUILD) -b singlehtml $(ALLSPHINXOPTS) $(BUILDDIR)/singlehtml | |||||
@echo | |||||
@echo "Build finished. The HTML page is in $(BUILDDIR)/singlehtml." | |||||
pickle: | |||||
$(SPHINXBUILD) -b pickle $(ALLSPHINXOPTS) $(BUILDDIR)/pickle | |||||
@echo | |||||
@echo "Build finished; now you can process the pickle files." | |||||
json: | |||||
$(SPHINXBUILD) -b json $(ALLSPHINXOPTS) $(BUILDDIR)/json | |||||
@echo | |||||
@echo "Build finished; now you can process the JSON files." | |||||
htmlhelp: | |||||
$(SPHINXBUILD) -b htmlhelp $(ALLSPHINXOPTS) $(BUILDDIR)/htmlhelp | |||||
@echo | |||||
@echo "Build finished; now you can run HTML Help Workshop with the" \ | |||||
".hhp project file in $(BUILDDIR)/htmlhelp." | |||||
qthelp: | |||||
$(SPHINXBUILD) -b qthelp $(ALLSPHINXOPTS) $(BUILDDIR)/qthelp | |||||
@echo | |||||
@echo "Build finished; now you can run "qcollectiongenerator" with the" \ | |||||
".qhcp project file in $(BUILDDIR)/qthelp, like this:" | |||||
@echo "# qcollectiongenerator $(BUILDDIR)/qthelp/EarwigBot.qhcp" | |||||
@echo "To view the help file:" | |||||
@echo "# assistant -collectionFile $(BUILDDIR)/qthelp/EarwigBot.qhc" | |||||
devhelp: | |||||
$(SPHINXBUILD) -b devhelp $(ALLSPHINXOPTS) $(BUILDDIR)/devhelp | |||||
@echo | |||||
@echo "Build finished." | |||||
@echo "To view the help file:" | |||||
@echo "# mkdir -p $$HOME/.local/share/devhelp/EarwigBot" | |||||
@echo "# ln -s $(BUILDDIR)/devhelp $$HOME/.local/share/devhelp/EarwigBot" | |||||
@echo "# devhelp" | |||||
epub: | |||||
$(SPHINXBUILD) -b epub $(ALLSPHINXOPTS) $(BUILDDIR)/epub | |||||
@echo | |||||
@echo "Build finished. The epub file is in $(BUILDDIR)/epub." | |||||
latex: | |||||
$(SPHINXBUILD) -b latex $(ALLSPHINXOPTS) $(BUILDDIR)/latex | |||||
@echo | |||||
@echo "Build finished; the LaTeX files are in $(BUILDDIR)/latex." | |||||
@echo "Run \`make' in that directory to run these through (pdf)latex" \ | |||||
"(use \`make latexpdf' here to do that automatically)." | |||||
latexpdf: | |||||
$(SPHINXBUILD) -b latex $(ALLSPHINXOPTS) $(BUILDDIR)/latex | |||||
@echo "Running LaTeX files through pdflatex..." | |||||
$(MAKE) -C $(BUILDDIR)/latex all-pdf | |||||
@echo "pdflatex finished; the PDF files are in $(BUILDDIR)/latex." | |||||
text: | |||||
$(SPHINXBUILD) -b text $(ALLSPHINXOPTS) $(BUILDDIR)/text | |||||
@echo | |||||
@echo "Build finished. The text files are in $(BUILDDIR)/text." | |||||
man: | |||||
$(SPHINXBUILD) -b man $(ALLSPHINXOPTS) $(BUILDDIR)/man | |||||
@echo | |||||
@echo "Build finished. The manual pages are in $(BUILDDIR)/man." | |||||
texinfo: | |||||
$(SPHINXBUILD) -b texinfo $(ALLSPHINXOPTS) $(BUILDDIR)/texinfo | |||||
@echo | |||||
@echo "Build finished. The Texinfo files are in $(BUILDDIR)/texinfo." | |||||
@echo "Run \`make' in that directory to run these through makeinfo" \ | |||||
"(use \`make info' here to do that automatically)." | |||||
info: | |||||
$(SPHINXBUILD) -b texinfo $(ALLSPHINXOPTS) $(BUILDDIR)/texinfo | |||||
@echo "Running Texinfo files through makeinfo..." | |||||
make -C $(BUILDDIR)/texinfo info | |||||
@echo "makeinfo finished; the Info files are in $(BUILDDIR)/texinfo." | |||||
gettext: | |||||
$(SPHINXBUILD) -b gettext $(I18NSPHINXOPTS) $(BUILDDIR)/locale | |||||
@echo | |||||
@echo "Build finished. The message catalogs are in $(BUILDDIR)/locale." | |||||
changes: | |||||
$(SPHINXBUILD) -b changes $(ALLSPHINXOPTS) $(BUILDDIR)/changes | |||||
@echo | |||||
@echo "The overview file is in $(BUILDDIR)/changes." | |||||
linkcheck: | |||||
$(SPHINXBUILD) -b linkcheck $(ALLSPHINXOPTS) $(BUILDDIR)/linkcheck | |||||
@echo | |||||
@echo "Link check complete; look for any errors in the above output " \ | |||||
"or in $(BUILDDIR)/linkcheck/output.txt." | |||||
doctest: | |||||
$(SPHINXBUILD) -b doctest $(ALLSPHINXOPTS) $(BUILDDIR)/doctest | |||||
@echo "Testing of doctests in the sources finished, look at the " \ | |||||
"results in $(BUILDDIR)/doctest/output.txt." |
@@ -0,0 +1,163 @@ | |||||
commands Package | |||||
================ | |||||
:mod:`commands` Package | |||||
----------------------- | |||||
.. automodule:: earwigbot.commands | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`_old` Module | |||||
------------------ | |||||
.. automodule:: earwigbot.commands._old | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`afc_report` Module | |||||
------------------------ | |||||
.. automodule:: earwigbot.commands.afc_report | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`afc_status` Module | |||||
------------------------ | |||||
.. automodule:: earwigbot.commands.afc_status | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`calc` Module | |||||
------------------ | |||||
.. automodule:: earwigbot.commands.calc | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`chanops` Module | |||||
--------------------- | |||||
.. automodule:: earwigbot.commands.chanops | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`crypt` Module | |||||
------------------- | |||||
.. automodule:: earwigbot.commands.crypt | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`ctcp` Module | |||||
------------------ | |||||
.. automodule:: earwigbot.commands.ctcp | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`editcount` Module | |||||
----------------------- | |||||
.. automodule:: earwigbot.commands.editcount | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`git` Module | |||||
----------------- | |||||
.. automodule:: earwigbot.commands.git | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`help` Module | |||||
------------------ | |||||
.. automodule:: earwigbot.commands.help | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`link` Module | |||||
------------------ | |||||
.. automodule:: earwigbot.commands.link | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`praise` Module | |||||
-------------------- | |||||
.. automodule:: earwigbot.commands.praise | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`quit` Module | |||||
------------------ | |||||
.. automodule:: earwigbot.commands.quit | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`registration` Module | |||||
-------------------------- | |||||
.. automodule:: earwigbot.commands.registration | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`remind` Module | |||||
-------------------- | |||||
.. automodule:: earwigbot.commands.remind | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`replag` Module | |||||
-------------------- | |||||
.. automodule:: earwigbot.commands.replag | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`rights` Module | |||||
-------------------- | |||||
.. automodule:: earwigbot.commands.rights | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`test` Module | |||||
------------------ | |||||
.. automodule:: earwigbot.commands.test | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`threads` Module | |||||
--------------------- | |||||
.. automodule:: earwigbot.commands.threads | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
@@ -0,0 +1,51 @@ | |||||
irc Package | |||||
=========== | |||||
:mod:`irc` Package | |||||
------------------ | |||||
.. automodule:: earwigbot.irc | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`connection` Module | |||||
------------------------ | |||||
.. automodule:: earwigbot.irc.connection | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`data` Module | |||||
------------------ | |||||
.. automodule:: earwigbot.irc.data | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`frontend` Module | |||||
---------------------- | |||||
.. automodule:: earwigbot.irc.frontend | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`rc` Module | |||||
---------------- | |||||
.. automodule:: earwigbot.irc.rc | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`watcher` Module | |||||
--------------------- | |||||
.. automodule:: earwigbot.irc.watcher | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
@@ -0,0 +1,61 @@ | |||||
earwigbot Package | |||||
================= | |||||
:mod:`earwigbot` Package | |||||
------------------------ | |||||
.. automodule:: earwigbot.__init__ | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`blowfish` Module | |||||
---------------------- | |||||
.. automodule:: earwigbot.blowfish | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`bot` Module | |||||
----------------- | |||||
.. automodule:: earwigbot.bot | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`config` Module | |||||
-------------------- | |||||
.. automodule:: earwigbot.config | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`managers` Module | |||||
---------------------- | |||||
.. automodule:: earwigbot.managers | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`util` Module | |||||
------------------ | |||||
.. automodule:: earwigbot.util | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
Subpackages | |||||
----------- | |||||
.. toctree:: | |||||
earwigbot.commands | |||||
earwigbot.irc | |||||
earwigbot.tasks | |||||
earwigbot.wiki | |||||
@@ -0,0 +1,91 @@ | |||||
tasks Package | |||||
============= | |||||
:mod:`tasks` Package | |||||
-------------------- | |||||
.. automodule:: earwigbot.tasks | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`afc_catdelink` Module | |||||
--------------------------- | |||||
.. automodule:: earwigbot.tasks.afc_catdelink | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`afc_copyvios` Module | |||||
-------------------------- | |||||
.. automodule:: earwigbot.tasks.afc_copyvios | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`afc_dailycats` Module | |||||
--------------------------- | |||||
.. automodule:: earwigbot.tasks.afc_dailycats | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`afc_history` Module | |||||
------------------------- | |||||
.. automodule:: earwigbot.tasks.afc_history | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`afc_statistics` Module | |||||
---------------------------- | |||||
.. automodule:: earwigbot.tasks.afc_statistics | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`afc_undated` Module | |||||
------------------------- | |||||
.. automodule:: earwigbot.tasks.afc_undated | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`blptag` Module | |||||
-------------------- | |||||
.. automodule:: earwigbot.tasks.blptag | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`feed_dailycats` Module | |||||
---------------------------- | |||||
.. automodule:: earwigbot.tasks.feed_dailycats | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`wikiproject_tagger` Module | |||||
-------------------------------- | |||||
.. automodule:: earwigbot.tasks.wikiproject_tagger | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`wrongmime` Module | |||||
----------------------- | |||||
.. automodule:: earwigbot.tasks.wrongmime | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
@@ -0,0 +1,75 @@ | |||||
wiki Package | |||||
============ | |||||
:mod:`wiki` Package | |||||
------------------- | |||||
.. automodule:: earwigbot.wiki | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`category` Module | |||||
---------------------- | |||||
.. automodule:: earwigbot.wiki.category | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`constants` Module | |||||
----------------------- | |||||
.. automodule:: earwigbot.wiki.constants | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`copyright` Module | |||||
----------------------- | |||||
.. automodule:: earwigbot.wiki.copyright | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`exceptions` Module | |||||
------------------------ | |||||
.. automodule:: earwigbot.wiki.exceptions | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`page` Module | |||||
------------------ | |||||
.. automodule:: earwigbot.wiki.page | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`site` Module | |||||
------------------ | |||||
.. automodule:: earwigbot.wiki.site | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`sitesdb` Module | |||||
--------------------- | |||||
.. automodule:: earwigbot.wiki.sitesdb | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
:mod:`user` Module | |||||
------------------ | |||||
.. automodule:: earwigbot.wiki.user | |||||
:members: | |||||
:undoc-members: | |||||
:show-inheritance: | |||||
@@ -0,0 +1,7 @@ | |||||
earwigbot | |||||
========= | |||||
.. toctree:: | |||||
:maxdepth: 4 | |||||
earwigbot |
@@ -0,0 +1,242 @@ | |||||
# -*- coding: utf-8 -*- | |||||
# | |||||
# EarwigBot documentation build configuration file, created by | |||||
# sphinx-quickstart on Sun Apr 29 01:42:25 2012. | |||||
# | |||||
# This file is execfile()d with the current directory set to its containing dir. | |||||
# | |||||
# Note that not all possible configuration values are present in this | |||||
# autogenerated file. | |||||
# | |||||
# All configuration values have a default; values that are commented out | |||||
# serve to show the default. | |||||
import sys, os | |||||
# If extensions (or modules to document with autodoc) are in another directory, | |||||
# add these directories to sys.path here. If the directory is relative to the | |||||
# documentation root, use os.path.abspath to make it absolute, like shown here. | |||||
sys.path.insert(0, os.path.abspath('..')) | |||||
# -- General configuration ----------------------------------------------------- | |||||
# If your documentation needs a minimal Sphinx version, state it here. | |||||
#needs_sphinx = '1.0' | |||||
# Add any Sphinx extension module names here, as strings. They can be extensions | |||||
# coming with Sphinx (named 'sphinx.ext.*') or your custom ones. | |||||
extensions = ['sphinx.ext.autodoc', 'sphinx.ext.coverage', 'sphinx.ext.viewcode'] | |||||
# Add any paths that contain templates here, relative to this directory. | |||||
templates_path = ['_templates'] | |||||
# The suffix of source filenames. | |||||
source_suffix = '.rst' | |||||
# The encoding of source files. | |||||
#source_encoding = 'utf-8-sig' | |||||
# The master toctree document. | |||||
master_doc = 'index' | |||||
# General information about the project. | |||||
project = u'EarwigBot' | |||||
copyright = u'2009, 2010, 2011, 2012 by Ben Kurtovic' | |||||
# The version info for the project you're documenting, acts as replacement for | |||||
# |version| and |release|, also used in various other places throughout the | |||||
# built documents. | |||||
# | |||||
# The short X.Y version. | |||||
version = '0.1' | |||||
# The full version, including alpha/beta/rc tags. | |||||
release = '0.1.dev' | |||||
# The language for content autogenerated by Sphinx. Refer to documentation | |||||
# for a list of supported languages. | |||||
#language = None | |||||
# There are two options for replacing |today|: either, you set today to some | |||||
# non-false value, then it is used: | |||||
#today = '' | |||||
# Else, today_fmt is used as the format for a strftime call. | |||||
#today_fmt = '%B %d, %Y' | |||||
# List of patterns, relative to source directory, that match files and | |||||
# directories to ignore when looking for source files. | |||||
exclude_patterns = ['_build'] | |||||
# The reST default role (used for this markup: `text`) to use for all documents. | |||||
#default_role = None | |||||
# If true, '()' will be appended to :func: etc. cross-reference text. | |||||
#add_function_parentheses = True | |||||
# If true, the current module name will be prepended to all description | |||||
# unit titles (such as .. function::). | |||||
#add_module_names = True | |||||
# If true, sectionauthor and moduleauthor directives will be shown in the | |||||
# output. They are ignored by default. | |||||
#show_authors = False | |||||
# The name of the Pygments (syntax highlighting) style to use. | |||||
pygments_style = 'sphinx' | |||||
# A list of ignored prefixes for module index sorting. | |||||
#modindex_common_prefix = [] | |||||
# -- Options for HTML output --------------------------------------------------- | |||||
# The theme to use for HTML and HTML Help pages. See the documentation for | |||||
# a list of builtin themes. | |||||
html_theme = 'nature' | |||||
# Theme options are theme-specific and customize the look and feel of a theme | |||||
# further. For a list of options available for each theme, see the | |||||
# documentation. | |||||
#html_theme_options = {} | |||||
# Add any paths that contain custom themes here, relative to this directory. | |||||
#html_theme_path = [] | |||||
# The name for this set of Sphinx documents. If None, it defaults to | |||||
# "<project> v<release> documentation". | |||||
#html_title = None | |||||
# A shorter title for the navigation bar. Default is the same as html_title. | |||||
#html_short_title = None | |||||
# The name of an image file (relative to this directory) to place at the top | |||||
# of the sidebar. | |||||
#html_logo = None | |||||
# The name of an image file (within the static path) to use as favicon of the | |||||
# docs. This file should be a Windows icon file (.ico) being 16x16 or 32x32 | |||||
# pixels large. | |||||
#html_favicon = None | |||||
# Add any paths that contain custom static files (such as style sheets) here, | |||||
# relative to this directory. They are copied after the builtin static files, | |||||
# so a file named "default.css" will overwrite the builtin "default.css". | |||||
html_static_path = ['_static'] | |||||
# If not '', a 'Last updated on:' timestamp is inserted at every page bottom, | |||||
# using the given strftime format. | |||||
#html_last_updated_fmt = '%b %d, %Y' | |||||
# If true, SmartyPants will be used to convert quotes and dashes to | |||||
# typographically correct entities. | |||||
#html_use_smartypants = True | |||||
# Custom sidebar templates, maps document names to template names. | |||||
#html_sidebars = {} | |||||
# Additional templates that should be rendered to pages, maps page names to | |||||
# template names. | |||||
#html_additional_pages = {} | |||||
# If false, no module index is generated. | |||||
#html_domain_indices = True | |||||
# If false, no index is generated. | |||||
#html_use_index = True | |||||
# If true, the index is split into individual pages for each letter. | |||||
#html_split_index = False | |||||
# If true, links to the reST sources are added to the pages. | |||||
#html_show_sourcelink = True | |||||
# If true, "Created using Sphinx" is shown in the HTML footer. Default is True. | |||||
#html_show_sphinx = True | |||||
# If true, "(C) Copyright ..." is shown in the HTML footer. Default is True. | |||||
#html_show_copyright = True | |||||
# If true, an OpenSearch description file will be output, and all pages will | |||||
# contain a <link> tag referring to it. The value of this option must be the | |||||
# base URL from which the finished HTML is served. | |||||
#html_use_opensearch = '' | |||||
# This is the file name suffix for HTML files (e.g. ".xhtml"). | |||||
#html_file_suffix = None | |||||
# Output file base name for HTML help builder. | |||||
htmlhelp_basename = 'EarwigBotdoc' | |||||
# -- Options for LaTeX output -------------------------------------------------- | |||||
latex_elements = { | |||||
# The paper size ('letterpaper' or 'a4paper'). | |||||
#'papersize': 'letterpaper', | |||||
# The font size ('10pt', '11pt' or '12pt'). | |||||
#'pointsize': '10pt', | |||||
# Additional stuff for the LaTeX preamble. | |||||
#'preamble': '', | |||||
} | |||||
# Grouping the document tree into LaTeX files. List of tuples | |||||
# (source start file, target name, title, author, documentclass [howto/manual]). | |||||
latex_documents = [ | |||||
('index', 'EarwigBot.tex', u'EarwigBot Documentation', | |||||
u'Ben Kurtovic', 'manual'), | |||||
] | |||||
# The name of an image file (relative to this directory) to place at the top of | |||||
# the title page. | |||||
#latex_logo = None | |||||
# For "manual" documents, if this is true, then toplevel headings are parts, | |||||
# not chapters. | |||||
#latex_use_parts = False | |||||
# If true, show page references after internal links. | |||||
#latex_show_pagerefs = False | |||||
# If true, show URL addresses after external links. | |||||
#latex_show_urls = False | |||||
# Documents to append as an appendix to all manuals. | |||||
#latex_appendices = [] | |||||
# If false, no module index is generated. | |||||
#latex_domain_indices = True | |||||
# -- Options for manual page output -------------------------------------------- | |||||
# One entry per manual page. List of tuples | |||||
# (source start file, name, description, authors, manual section). | |||||
man_pages = [ | |||||
('index', 'earwigbot', u'EarwigBot Documentation', | |||||
[u'Ben Kurtovic'], 1) | |||||
] | |||||
# If true, show URL addresses after external links. | |||||
#man_show_urls = False | |||||
# -- Options for Texinfo output ------------------------------------------------ | |||||
# Grouping the document tree into Texinfo files. List of tuples | |||||
# (source start file, target name, title, author, | |||||
# dir menu entry, description, category) | |||||
texinfo_documents = [ | |||||
('index', 'EarwigBot', u'EarwigBot Documentation', | |||||
u'Ben Kurtovic', 'EarwigBot', 'One line description of project.', | |||||
'Miscellaneous'), | |||||
] | |||||
# Documents to append as an appendix to all manuals. | |||||
#texinfo_appendices = [] | |||||
# If false, no module index is generated. | |||||
#texinfo_domain_indices = True | |||||
# How to display URL addresses: 'footnote', 'no', or 'inline'. | |||||
#texinfo_show_urls = 'footnote' |
@@ -0,0 +1,238 @@ | |||||
Customizing | |||||
=========== | |||||
The bot's working directory contains a :file:`commands` subdirectory and a | |||||
:file:`tasks` subdirectory. Custom IRC commands can be placed in the former, | |||||
whereas custom wiki bot tasks go into the latter. Developing custom modules is | |||||
explained in detail in this documentation. | |||||
Note that custom commands will override built-in commands and tasks with the | |||||
same name. | |||||
:py:class:`~earwigbot.bot.Bot` and :py:class:`~earwigbot.bot.BotConfig` | |||||
----------------------------------------------------------------------- | |||||
:py:class:`earwigbot.bot.Bot` is EarwigBot's main class. You don't have to | |||||
instantiate this yourself, but it's good to be familiar with its attributes and | |||||
methods, because it is the main way to communicate with other parts of the bot. | |||||
A :py:class:`~earwigbot.bot.Bot` object is accessible as an attribute of | |||||
commands and tasks (i.e., :py:attr:`self.bot`). | |||||
The most useful attributes are: | |||||
- :py:attr:`~earwigbot.bot.Bot.config`: an instance of | |||||
:py:class:`~earwigbot.config.BotConfig`, for accessing the bot's | |||||
configuration data (see below). | |||||
- :py:attr:`~earwigbot.bot.Bot.commands`: the bot's | |||||
:py:class:`~earwigbot.managers.CommandManager`, which is used internally to | |||||
run IRC commands (through | |||||
:py:meth:`commands.call() <earwigbot.managers.CommandManager.call>`, which | |||||
you shouldn't have to use); you can safely reload all commands with | |||||
:py:meth:`commands.load() <earwigbot.managers._ResourceManager.load>`. | |||||
- :py:attr:`~earwigbot.bot.Bot.tasks`: the bot's | |||||
:py:class:`~earwigbot.managers.TaskManager`, which can be used to start tasks | |||||
with :py:meth:`tasks.start(task_name, **kwargs) | |||||
<earwigbot.managers.TaskManager.start>`. :py:meth:`tasks.load() | |||||
<earwigbot.managers._ResourceManager.load>` can be used to safely reload all | |||||
tasks. | |||||
- :py:attr:`~earwigbot.bot.Bot.frontend` / | |||||
:py:attr:`~earwigbot.bot.Bot.watcher`: instances of | |||||
:py:class:`earwigbot.irc.Frontend <earwigbot.irc.frontend.Frontend>` and | |||||
:py:class:`earwigbot.irc.Watcher <earwigbot.irc.watcher.Watcher>`, | |||||
respectively, which represent the bot's connections to these two servers; you | |||||
can, for example, send a message to the frontend with | |||||
:py:meth:`frontend.say(chan, msg) | |||||
<earwigbot.irc.connection.IRCConnection.say>` (more on communicating with IRC | |||||
below). | |||||
- :py:attr:`~earwigbot.bot.Bot.wiki`: interface with the | |||||
:doc:`Wiki Toolset <toolset>`. | |||||
- Finally, :py:meth:`~earwigbot.bot.Bot.restart` (restarts IRC components and | |||||
reloads config, commands, and tasks) and :py:meth:`~earwigbot.bot.Bot.stop` | |||||
can be used almost anywhere. Both take an optional "reason" that will be | |||||
logged and used as the quit message when disconnecting from IRC. | |||||
:py:class:`earwigbot.config.BotConfig` stores configuration information for the | |||||
bot. Its docstring explains what each attribute is used for, but essentially | |||||
each "node" (one of :py:attr:`config.components`, :py:attr:`wiki`, | |||||
:py:attr:`tasks`, :py:attr:`tasks`, or :py:attr:`metadata`) maps to a section | |||||
of the bot's :file:`config.yml` file. For example, if :file:`config.yml` | |||||
includes something like:: | |||||
irc: | |||||
frontend: | |||||
nick: MyAwesomeBot | |||||
channels: | |||||
- "##earwigbot" | |||||
- "#channel" | |||||
- "#other-channel" | |||||
...then :py:attr:`config.irc["frontend"]["nick"]` will be ``"MyAwesomeBot"`` | |||||
and :py:attr:`config.irc["frontend"]["channels"]` will be | |||||
``["##earwigbot", "#channel", "#other-channel"]``. | |||||
Custom IRC commands | |||||
------------------- | |||||
Custom commands are subclasses of :py:class:`earwigbot.commands.BaseCommand` | |||||
that override :py:class:`~earwigbot.commands.BaseCommand`'s | |||||
:py:meth:`~earwigbot.commands.BaseCommand.process` (and optionally | |||||
:py:meth:`~earwigbot.commands.BaseCommand.check`) methods. | |||||
:py:class:`~earwigbot.commands.BaseCommand`'s docstrings should explain what | |||||
each attribute and method is for and what they should be overridden with, but | |||||
these are the basics: | |||||
- Class attribute :py:attr:`~earwigbot.commands.BaseCommand.name` is the name | |||||
of the command. This must be specified. | |||||
- Class attribute :py:attr:`~earwigbot.commands.BaseCommand.hooks` is a list of | |||||
the "IRC events" that this command might respond to. It defaults to | |||||
``["msg"]``, but options include ``"msg_private"`` (for private messages | |||||
only), ``"msg_public"`` (for channel messages only), and ``"join"`` (for when | |||||
a user joins a channel). See the afc_status_ plugin for a command that | |||||
responds to other hook types. | |||||
- Method :py:meth:`~earwigbot.commands.BaseCommand.check` is passed a | |||||
:py:class:`~earwigbot.irc.data.Data` [1]_ object, and should return ``True`` | |||||
if you want to respond to this message, or ``False`` otherwise. The default | |||||
behavior is to return ``True`` only if | |||||
:py:attr:`data.is_command` is ``True`` and :py:attr:`data.command` == | |||||
:py:attr:`~earwigbot.commands.BaseCommand.name`, which is suitable for most | |||||
cases. A common, straightforward reason for overriding is if a command has | |||||
aliases (see chanops_ for an example). Note that by returning ``True``, you | |||||
prevent any other commands from responding to this message. | |||||
- Method :py:meth:`~earwigbot.commands.BaseCommand.process` is passed the same | |||||
:py:class:`~earwigbot.irc.data.Data` object as | |||||
:py:meth:`~earwigbot.commands.BaseCommand.check`, but only if | |||||
:py:meth:`~earwigbot.commands.BaseCommand.check` returned ``True``. This is | |||||
where the bulk of your command goes. To respond to IRC messages, there are a | |||||
number of methods of :py:class:`~earwigbot.commands.BaseCommand` at your | |||||
disposal. See the the test_ command for a simple example, or look in | |||||
:py:class:`~earwigbot.commands.BaseCommand`'s | |||||
:py:meth:`~earwigbot.commands.BaseCommand.__init__` method for the full list. | |||||
The most common ones are :py:meth:`say(chan_or_user, msg) | |||||
<earwigbot.irc.connection.IRCConnection.say>`, :py:meth:`reply(data, msg) | |||||
<earwigbot.irc.connection.IRCConnection.reply>` (convenience function; sends | |||||
a reply to the issuer of the command in the channel it was received), | |||||
:py:meth:`action(chan_or_user, msg) | |||||
<earwigbot.irc.connection.IRCConnection.action>`, | |||||
:py:meth:`notice(chan_or_user, msg) | |||||
<earwigbot.irc.connection.IRCConnection.notice>`, :py:meth:`join(chan) | |||||
<earwigbot.irc.connection.IRCConnection.join>`, and | |||||
:py:meth:`part(chan) <earwigbot.irc.connection.IRCConnection.part>`. | |||||
It's important to name the command class :py:class:`Command` within the file, | |||||
or else the bot might not recognize it as a command. The name of the file | |||||
doesn't really matter and need not match the command's name, but this is | |||||
recommended for readability. | |||||
The bot has a wide selection of built-in commands and plugins to act as sample | |||||
code and/or to give ideas. Start with test_, and then check out chanops_ and | |||||
afc_status_ for some more complicated scripts. | |||||
Custom bot tasks | |||||
---------------- | |||||
Custom tasks are subclasses of :py:class:`earwigbot.tasks.BaseTask` that | |||||
override :py:class:`~earwigbot.tasks.BaseTask`'s | |||||
:py:meth:`~earwigbot.tasks.BaseTask.run` (and optionally | |||||
:py:meth:`~earwigbot.tasks.BaseTask.setup`) methods. | |||||
:py:class:`~earwigbot.tasks.BaseTask`'s docstrings should explain what each | |||||
attribute and method is for and what they should be overridden with, but these | |||||
are the basics: | |||||
- Class attribute :py:attr:`~earwigbot.tasks.BaseTask.name` is the name of the | |||||
task. This must be specified. | |||||
- Class attribute :py:attr:`~earwigbot.tasks.BaseTask.number` can be used to | |||||
store an optional "task number", possibly for use in edit summaries (to be | |||||
generated with :py:meth:`~earwigbot.tasks.BaseTask.make_summary`). For | |||||
example, EarwigBot's :py:attr:`config.wiki["summary"]` is | |||||
``"([[WP:BOT|Bot]]; [[User:EarwigBot#Task $1|Task $1]]): $2"``, which the | |||||
task class's :py:meth:`make_summary(comment) | |||||
<earwigbot.tasks.BaseTask.make_summary>` method will take and replace | |||||
``$1`` with the task number and ``$2`` with the details of the edit. | |||||
Additionally, :py:meth:`~earwigbot.tasks.BaseTask.shutoff_enabled` (which | |||||
checks whether the bot has been told to stop on-wiki by checking the content | |||||
of a particular page) can check a different page for each task using similar | |||||
variables. EarwigBot's :py:attr:`config.wiki["shutoff"]["page"]` is | |||||
``"User:$1/Shutoff/Task $2"``; ``$1`` is substituted with the bot's username, | |||||
and ``$2`` is substituted with the task number, so, e.g., task #14 checks the | |||||
page ``[[User:EarwigBot/Shutoff/Task 14]].`` If the page's content does *not* | |||||
match :py:attr:`config.wiki["shutoff"]["disabled"]` (``"run"`` by default), | |||||
then shutoff is considered to be *enabled* and | |||||
:py:meth:`~earwigbot.tasks.BaseTask.shutoff_enabled` will return ``True``, | |||||
indicating the task should not run. If you don't intend to use either of | |||||
these methods, feel free to leave this attribute blank. | |||||
- Method :py:meth:`~earwigbot.tasks.BaseTask.setup` is called *once* with no | |||||
arguments immediately after the task is first loaded. Does nothing by | |||||
default; treat it like an :py:meth:`__init__` if you want | |||||
(:py:meth:`~earwigbot.tasks.BaseTask.__init__` does things by default and a | |||||
dedicated setup method is often easier than overriding | |||||
:py:meth:`~earwigbot.tasks.BaseTask.__init__` and using :py:obj:`super`). | |||||
- Method :py:meth:`~earwigbot.tasks.BaseTask.run` is called with any number of | |||||
keyword arguments every time the task is executed (by | |||||
:py:meth:`tasks.start(task_name, **kwargs) | |||||
<earwigbot.managers.TaskManager.start>`, usually). This is where the bulk of | |||||
the task's code goes. For interfacing with MediaWiki sites, read up on the | |||||
:doc:`Wiki Toolset <toolset>`. | |||||
Tasks have access to :py:attr:`config.tasks[task_name]` for config information, | |||||
which is a node in :file:`config.yml` like every other attribute of | |||||
:py:attr:`bot.config`. This can be used to store, for example, edit summaries, | |||||
or templates to append to user talk pages, so that these can be easily changed | |||||
without modifying the task itself. | |||||
It's important to name the task class :py:class:`Task` within the file, or else | |||||
the bot might not recognize it as a task. The name of the file doesn't really | |||||
matter and need not match the task's name, but this is recommended for | |||||
readability. | |||||
See the built-in wikiproject_tagger_ task for a relatively straightforward | |||||
task, or the afc_statistics_ plugin for a more complicated one. | |||||
.. rubric:: Footnotes | |||||
.. [1] :py:class:`~earwigbot.irc.data.Data` objects are instances of | |||||
:py:class:`earwigbot.irc.Data <earwigbot.irc.data.Data>` that contain | |||||
information about a single message sent on IRC. Their useful attributes | |||||
are :py:attr:`~earwigbot.irc.data.Data.chan` (channel the message was | |||||
sent from, equal to :py:attr:`~earwigbot.irc.data.Data.nick` if it's a | |||||
private message), :py:attr:`~earwigbot.irc.data.Data.nick` (nickname of | |||||
the sender), :py:attr:`~earwigbot.irc.data.Data.ident` (ident_ of the | |||||
sender), :py:attr:`~earwigbot.irc.data.Data.host` (hostname of the | |||||
sender), :py:attr:`~earwigbot.irc.data.Data.msg` (text of the sent | |||||
message), :py:attr:`~earwigbot.irc.data.Data.is_command` (boolean | |||||
telling whether or not this message is a bot command, e.g., whether it | |||||
is prefixed by ``!``), :py:attr:`~earwigbot.irc.data.Data.command` (if | |||||
the message is a command, this is the name of the command used), and | |||||
:py:attr:`~earwigbot.irc.data.Data.args` (if the message is a command, | |||||
this is a list of the command arguments - for example, if issuing | |||||
"``!part ##earwig Goodbye guys``", | |||||
:py:attr:`~earwigbot.irc.data.Data.args` will equal | |||||
``["##earwig", "Goodbye", "guys"]``). Note that not all | |||||
:py:class:`~earwigbot.irc.data.Data` objects will have all of these | |||||
attributes: :py:class:`~earwigbot.irc.data.Data` objects generated by | |||||
private messages will, but ones generated by joins will only have | |||||
:py:attr:`~earwigbot.irc.data.Data.chan`, | |||||
:py:attr:`~earwigbot.irc.data.Data.nick`, | |||||
:py:attr:`~earwigbot.irc.data.Data.ident`, | |||||
and :py:attr:`~earwigbot.irc.data.Data.host`. | |||||
.. _afc_status: https://github.com/earwig/earwigbot-plugins/blob/develop/commands/afc_status.py | |||||
.. _chanops: https://github.com/earwig/earwigbot/blob/develop/earwigbot/commands/chanops.py | |||||
.. _test: https://github.com/earwig/earwigbot/blob/develop/earwigbot/commands/test.py | |||||
.. _wikiproject_tagger: https://github.com/earwig/earwigbot/blob/develop/earwigbot/tasks/wikiproject_tagger.py | |||||
.. _afc_statistics: https://github.com/earwig/earwigbot-plugins/blob/develop/tasks/afc_statistics.py | |||||
.. _ident: http://en.wikipedia.org/wiki/Ident |
@@ -0,0 +1,48 @@ | |||||
EarwigBot v0.1 Documentation | |||||
============================ | |||||
EarwigBot_ is a Python_ robot that edits Wikipedia_ and interacts with people | |||||
over IRC_. | |||||
History | |||||
------- | |||||
Development began, based on the `Pywikipedia framework`_, in early 2009. | |||||
Approval for its fist task, a `copyright violation detector`_, was carried out | |||||
in May, and the bot has been running consistently ever since (with the | |||||
exception of Jan/Feb 2011). It currently handles `several ongoing tasks`_ | |||||
ranging from statistics generation to category cleanup, and on-demand tasks | |||||
such as WikiProject template tagging. Since it started running, the bot has | |||||
made over 50,000 edits. | |||||
A project to rewrite it from scratch began in early April 2011, thus moving | |||||
away from the Pywikipedia framework and allowing for less overall code, better | |||||
integration between bot parts, and easier maintenance. | |||||
.. _EarwigBot: http://en.wikipedia.org/wiki/User:EarwigBot | |||||
.. _Python: http://python.org/ | |||||
.. _Wikipedia: http://en.wikipedia.org/ | |||||
.. _IRC: http://en.wikipedia.org/wiki/Internet_Relay_Chat | |||||
.. _Pywikipedia framework: http://pywikipediabot.sourceforge.net/ | |||||
.. _copyright violation detector: http://en.wikipedia.org/wiki/Wikipedia:Bots/Requests_for_approval/EarwigBot_1 | |||||
.. _several ongoing tasks: http://en.wikipedia.org/wiki/User:EarwigBot#Tasks | |||||
Contents | |||||
-------- | |||||
.. toctree:: | |||||
:maxdepth: 2 | |||||
installation | |||||
setup | |||||
customizing | |||||
toolset | |||||
tips | |||||
API Reference <api/modules> | |||||
Indices and tables | |||||
------------------ | |||||
* :ref:`genindex` | |||||
* :ref:`modindex` | |||||
* :ref:`search` |
@@ -0,0 +1,55 @@ | |||||
Installation | |||||
============ | |||||
This package contains the core :py:mod:`earwigbot`, abstracted enough that it | |||||
should be usable and customizable by anyone running a bot on a MediaWiki site. | |||||
Since it is component-based, the IRC components can be disabled if desired. IRC | |||||
commands and bot tasks specific to `my instance of EarwigBot`_ that I don't | |||||
feel the average user will need are available from the repository | |||||
`earwigbot-plugins`_. | |||||
It's recommended to run the bot's unit tests before installing. Run | |||||
:command:`python setup.py test` from the project's root directory. Note that | |||||
some tests require an internet connection, and others may take a while to run. | |||||
Coverage is currently rather incomplete. | |||||
Latest release (v0.1) | |||||
--------------------- | |||||
EarwigBot is available from the `Python Package Index`_, so you can install the | |||||
latest release with :command:`pip install earwigbot` (`get pip`_). | |||||
You can also install it from source [1]_ directly:: | |||||
curl -Lo earwigbot.tgz https://github.com/earwig/earwigbot/tarball/v0.1 | |||||
tar -xf earwigbot.tgz | |||||
cd earwig-earwigbot-* | |||||
python setup.py install | |||||
cd .. | |||||
rm -r earwigbot.tgz earwig-earwigbot-* | |||||
Development version | |||||
------------------- | |||||
You can install the development version of the bot from :command:`git` by using | |||||
setuptools/`distribute`_'s :command:`develop` command [1]_, probably on the | |||||
``develop`` branch which contains (usually) working code. ``master`` contains | |||||
the latest release. EarwigBot uses `git flow`_, so you're free to browse by | |||||
tags or by new features (``feature/*`` branches):: | |||||
git clone git://github.com/earwig/earwigbot.git earwigbot | |||||
cd earwigbot | |||||
python setup.py develop | |||||
.. rubric:: Footnotes | |||||
.. [1] :command:`python setup.py install`/:command:`develop` may require root, | |||||
or use the :command:`--user` switch to install for the current user | |||||
only. | |||||
.. _my instance of EarwigBot: http://en.wikipedia.org/wiki/User:EarwigBot | |||||
.. _earwigbot-plugins: https://github.com/earwig/earwigbot-plugins | |||||
.. _Python Package Index: http://pypi.python.org | |||||
.. _get pip: http://pypi.python.org/pypi/pip | |||||
.. _distribute: http://pypi.python.org/pypi/distribute | |||||
.. _git flow: http://nvie.com/posts/a-successful-git-branching-model/ |
@@ -0,0 +1,28 @@ | |||||
Setup | |||||
===== | |||||
The bot stores its data in a "working directory", including its config file and | |||||
databases. This is also the location where you will place custom IRC commands | |||||
and bot tasks, which will be explained later. It doesn't matter where this | |||||
directory is, as long as the bot can write to it. | |||||
Start the bot with :command:`earwigbot path/to/working/dir`, or just | |||||
:command:`earwigbot` if the working directory is the current directory. It will | |||||
notice that no :file:`config.yml` file exists and take you through the setup | |||||
process. | |||||
There is currently no way to edit the :file:`config.yml` file from within the | |||||
bot after it has been created, but YAML is a very straightforward format, so | |||||
you should be able to make any necessary changes yourself. Check out the | |||||
`explanation of YAML`_ on Wikipedia for help. | |||||
After setup, the bot will start. This means it will connect to the IRC servers | |||||
it has been configured for, schedule bot tasks to run at specific times, and | |||||
then wait for instructions (as commands on IRC). For a list of commands, say | |||||
"``!help``" (commands are messages prefixed with an exclamation mark). | |||||
You can stop the bot at any time with :kbd:`Control-c`, same as you stop a | |||||
normal Python program, and it will try to exit safely. You can also use the | |||||
"``!quit``" command on IRC. | |||||
.. _explanation of YAML: http://en.wikipedia.org/wiki/YAML |
@@ -0,0 +1,46 @@ | |||||
Tips | |||||
==== | |||||
- Logging_ is a fantastic way to monitor the bot's progress as it runs. It has | |||||
a slew of built-in loggers, and enabling log retention (so logs are saved to | |||||
:file:`logs/` in the working directory) is highly recommended. In the normal | |||||
setup, there are three log files, each of which "rotate" at a specific time | |||||
(:file:`filename.log` becomes :file:`filename.log.2012-04-10`, for example). | |||||
The :file:`debug.log` file rotates every hour, and maintains six hours of | |||||
logs of every level (``DEBUG`` and up). :file:`bot.log` rotates every day at | |||||
midnight, and maintains seven days of non-debug logs (``INFO`` and up). | |||||
Finally, :file:`error.log` rotates every Sunday night, and maintains four | |||||
weeks of logs indicating unexpected events (``WARNING`` and up). | |||||
To use logging in your commands or tasks (recommended), | |||||
:py:class:~earwigbot.commands.BaseCommand` and | |||||
:py:class:~earwigbot.tasks.BaseTask` provide :py:attr:`logger` attributes | |||||
configured for the specific command or task. If you're working with other | |||||
classes, :py:attr:`bot.logger` is the root logger | |||||
(:py:obj:`logging.getLogger("earwigbot")` by default), so you can use | |||||
:py:func:`~logging.Logger.getChild` to make your logger. For example, task | |||||
loggers are essentially | |||||
:py:attr:`bot.logger.getChild("tasks").getChild(task.name) <bot.logger>`. | |||||
- A very useful IRC command is "``!reload``", which reloads all commands and | |||||
tasks without restarting the bot. [1]_ Combined with using the `!git plugin`_ | |||||
for pulling repositories from IRC, this can provide a seamless command/task | |||||
development workflow if the bot runs on an external server and you set up | |||||
its working directory as a git repo. | |||||
- You can run a task by itself instead of the entire bot with | |||||
:command:`earwigbot path/to/working/dir --task task_name`. | |||||
- Questions, comments, or suggestions about the documentation? `Let me know`_, | |||||
or `create an issue`_ so I can improve it for other people. | |||||
.. rubric:: Footnotes | |||||
.. [1] In reality, all this does is call :py:meth:`bot.commands.load() | |||||
<earwigbot.managers._ResourceManager.load>` and | |||||
:py:meth:`bot.tasks.load() <earwigbot.managers._ResourceManager.load>`! | |||||
.. _logging: http://docs.python.org/library/logging.html | |||||
.. _!git plugin: https://github.com/earwig/earwigbot-plugins/blob/develop/commands/git.py | |||||
.. _Let me know: ben.kurtovic@verizon.net | |||||
.. _create an issue: https://github.com/earwig/earwigbot/issues |
@@ -0,0 +1,220 @@ | |||||
The Wiki Toolset | |||||
================ | |||||
EarwigBot's answer to the `Pywikipedia framework`_ is the Wiki Toolset | |||||
(:py:mod:`earwigbot.wiki`), which you will mainly access through | |||||
:py:attr:`bot.wiki <earwigbot.bot.Bot.wiki>`. | |||||
:py:attr:`bot.wiki <earwigbot.bot.Bot.wiki>` provides three methods for the | |||||
management of Sites - :py:meth:`~earwigbot.wiki.sitesdb.SitesDB.get_site`, | |||||
:py:meth:`~earwigbot.wiki.sitesdb.SitesDB.add_site`, and | |||||
:py:meth:`~earwigbot.wiki.sitesdb.SitesDB.remove_site`. Sites are objects that | |||||
simply represent a MediaWiki site. A single instance of EarwigBot (i.e. a | |||||
single *working directory*) is expected to relate to a single site or group of | |||||
sites using the same login info (like all WMF wikis with `CentralAuth`_). | |||||
Load your default site (the one that you picked during setup) with | |||||
``site = bot.wiki.get_site()``. | |||||
Dealing with other sites | |||||
~~~~~~~~~~~~~~~~~~~~~~~~ | |||||
*Skip this section if you're only working with one site.* | |||||
If a site is *already known to the bot* (meaning that it is stored in the | |||||
:file:`sites.db` file, which includes just your default wiki at first), you can | |||||
load a site with ``site = bot.wiki.get_site(name)``, where ``name`` might be | |||||
``"enwiki"`` or ``"frwiktionary"`` (you can also do | |||||
``site = bot.wiki.get_site(project="wikipedia", lang="en")``). Recall that not | |||||
giving any arguments to ``get_site()`` will return the default site. | |||||
:py:meth:`~earwigbot.wiki.sitesdb.SitesDB.add_site` is used to add new sites to | |||||
the sites database. It may be called with similar arguments as | |||||
:py:meth:`~earwigbot.wiki.sitesdb.SitesDB.get_site`, but the difference is | |||||
important. :py:meth:`~earwigbot.wiki.sitesdb.SitesDB.get_site` only needs | |||||
enough information to identify the site in its database, which is usually just | |||||
its name; the database stores all other necessary connection info. With | |||||
:py:meth:`~earwigbot.wiki.sitesdb.SitesDB.add_site`, you need to provide enough | |||||
connection info so the toolset can successfully access the site's API/SQL | |||||
databases and store that information for later. That might not be much; for WMF | |||||
wikis, you can usually use code like this:: | |||||
project, lang = "wikipedia", "es" | |||||
try: | |||||
site = bot.wiki.get_site(project=project, lang=lang) | |||||
except earwigbot.SiteNotFoundError: | |||||
# Load site info from http://es.wikipedia.org/w/api.php: | |||||
site = bot.wiki.add_site(project=project, lang=lang) | |||||
This works because EarwigBot assumes that the URL for the site is | |||||
``"//{lang}.{project}.org"`` and the API is at ``/w/api.php``; this might | |||||
change if you're dealing with non-WMF wikis, where the code might look | |||||
something more like:: | |||||
project, lang = "mywiki", "it" | |||||
try: | |||||
site = bot.wiki.get_site(project=project, lang=lang) | |||||
except earwigbot.SiteNotFoundError: | |||||
# Load site info from http://mysite.net/mywiki/it/s/api.php: | |||||
base_url = "http://mysite.net/" + project + "/" + lang | |||||
db_name = lang + project + "_p" | |||||
sql = {host: "sql.mysite.net", db: db_name} | |||||
site = bot.wiki.add_site(base_url=base_url, script_path="/s", sql=sql) | |||||
:py:meth:`~earwigbot.wiki.sitesdb.SitesDB.remove_site` does the opposite of | |||||
:py:meth:`~earwigbot.wiki.sitesdb.SitesDB.add_site`: give it a site's name or a | |||||
project/lang pair like :py:meth:`~earwigbot.wiki.sitesdb.SitesDB.get_site` | |||||
takes, and it'll remove that site from the sites database. | |||||
Sites | |||||
~~~~~ | |||||
:py:class:`earwigbot.wiki.Site <earwigbot.wiki.site.Site>` objects provide the | |||||
following attributes: | |||||
- :py:attr:`~earwigbot.wiki.site.Site.name`: the site's name (or "wikiid"), | |||||
like ``"enwiki"`` | |||||
- :py:attr:`~earwigbot.wiki.site.Site.project`: the site's project name, like | |||||
``"wikipedia"`` | |||||
- :py:attr:`~earwigbot.wiki.site.Site.lang`: the site's language code, like | |||||
``"en"`` | |||||
- :py:attr:`~earwigbot.wiki.site.Site.domain`: the site's web domain, like | |||||
``"en.wikipedia.org"`` | |||||
and the following methods: | |||||
- :py:meth:`api_query(**kwargs) <earwigbot.wiki.site.Site.api_query>`: does an | |||||
API query with the given keyword arguments as params | |||||
- :py:meth:`sql_query(query, params=(), ...) | |||||
<earwigbot.wiki.site.Site.sql_query>`: does an SQL query and yields its | |||||
results (as a generator) | |||||
- :py:meth:`~earwigbot.wiki.site.Site.get_replag`: returns the estimated | |||||
database replication lag (if we have the site's SQL connection info) | |||||
- :py:meth:`namespace_id_to_name(id, all=False) | |||||
<earwigbot.wiki.site.Site.namespace_id_to_name>`: given a namespace ID, | |||||
returns the primary associated namespace name (or a list of all names when | |||||
``all`` is ``True``) | |||||
- :py:meth:`namespace_name_to_id(name) | |||||
<earwigbot.wiki.site.Site.namespace_name_to_id>`: given a namespace name, | |||||
returns the associated namespace ID | |||||
- :py:meth:`get_page(title, follow_redirects=False) | |||||
<earwigbot.wiki.site.Site.get_page>`: returns a ``Page`` object for the given | |||||
title (or a :py:class:`~earwigbot.wiki.category.Category` object if the | |||||
page's namespace is "``Category:``") | |||||
- :py:meth:`get_category(catname, follow_redirects=False) | |||||
<earwigbot.wiki.site.Site.get_category>`: returns a ``Category`` object for | |||||
the given title (sans namespace) | |||||
- :py:meth:`get_user(username) <earwigbot.wiki.site.Site.get_user>`: returns a | |||||
:py:class:`~earwigbot.wiki.user.User` object for the given username | |||||
Pages and categories | |||||
~~~~~~~~~~~~~~~~~~~~ | |||||
Create :py:class:`earwigbot.wiki.Page <earwigbot.wiki.page.Page>` objects with | |||||
:py:meth:`site.get_page(title) <earwigbot.wiki.site.Site.get_page>`, | |||||
:py:meth:`page.toggle_talk() <earwigbot.wiki.page.Page.toggle_talk>`, | |||||
:py:meth:`user.get_userpage() <earwigbot.wiki.user.User.get_userpage>`, or | |||||
:py:meth:`user.get_talkpage() <earwigbot.wiki.user.User.get_talkpage>`. They | |||||
provide the following attributes: | |||||
- :py:attr:`~earwigbot.wiki.page.Page.title`: the page's title, or pagename | |||||
- :py:attr:`~earwigbot.wiki.page.Page.exists`: whether the page exists | |||||
- :py:attr:`~earwigbot.wiki.page.Page.pageid`: an integer ID representing the | |||||
page | |||||
- :py:attr:`~earwigbot.wiki.page.Page.url`: the page's URL | |||||
- :py:attr:`~earwigbot.wiki.page.Page.namespace`: the page's namespace as an | |||||
integer | |||||
- :py:attr:`~earwigbot.wiki.page.Page.protection`: the page's current | |||||
protection status | |||||
- :py:attr:`~earwigbot.wiki.page.Page.is_talkpage`: ``True`` if the page is a | |||||
talkpage, else ``False`` | |||||
- :py:attr:`~earwigbot.wiki.page.Page.is_redirect`: ``True`` if the page is a | |||||
redirect, else ``False`` | |||||
and the following methods: | |||||
- :py:meth:`~earwigbot.wiki.page.Page.reload`: forcibly reload the page's | |||||
attributes (emphasis on *reload* - this is only necessary if there is reason | |||||
to believe they have changed) | |||||
- :py:meth:`toggle_talk(...) <earwigbot.wiki.page.Page.toggle_talk>`: returns a | |||||
content page's talk page, or vice versa | |||||
- :py:meth:`~earwigbot.wiki.page.Page.get`: returns page content | |||||
- :py:meth:`~earwigbot.wiki.page.Page.get_redirect_target`: if the page is a | |||||
redirect, returns its destination | |||||
- :py:meth:`~earwigbot.wiki.page.Page.get_creator`: returns a | |||||
:py:class:`~earwigbot.wiki.user.User` object representing the first user to | |||||
edit the page | |||||
- :py:meth:`edit(text, summary, minor=False, bot=True, force=False) | |||||
<earwigbot.wiki.page.Page.edit>`: replaces the page's content with ``text`` | |||||
or creates a new page | |||||
- :py:meth:`add_section(text, title, minor=False, bot=True, force=False) | |||||
<earwigbot.wiki.page.Page.add_section>`: adds a new section named ``title`` | |||||
at the bottom of the page | |||||
- :py:meth:`copyvio_check(...) | |||||
<earwigbot.wiki.copyvios.CopyvioMixin.copyvio_check>`: checks the page for | |||||
copyright violations | |||||
- :py:meth:`copyvio_compare(url, ...) | |||||
<earwigbot.wiki.copyvios.CopyvioMixin.copyvio_compare>`: checks the page like | |||||
:py:meth:`~earwigbot.wiki.copyvios.CopyvioMixin.copyvio_check`, but | |||||
against a specific URL | |||||
Additionally, :py:class:`~earwigbot.wiki.category.Category` objects (created | |||||
with :py:meth:`site.get_category(name) <earwigbot.wiki.site.Site.get_category>` | |||||
or :py:meth:`site.get_page(title) <earwigbot.wiki.site.Site.get_page>` where | |||||
``title`` is in the ``Category:`` namespace) provide the following additional | |||||
method: | |||||
- :py:meth:`get_members(use_sql=False, limit=None) | |||||
<earwigbot.wiki.category.Category.get_members>`: returns a list of page | |||||
titles in the category (limit is ``50`` by default if using the API) | |||||
Users | |||||
~~~~~ | |||||
Create :py:class:`earwigbot.wiki.User <earwigbot.wiki.user.User>` objects with | |||||
:py:meth:`site.get_user(name) <earwigbot.wiki.site.Site.get_user>` or | |||||
:py:meth:`page.get_creator() <earwigbot.wiki.page.Page.get_creator>`. They | |||||
provide the following attributes: | |||||
- :py:attr:`~earwigbot.wiki.user.User.name`: the user's username | |||||
- :py:attr:`~earwigbot.wiki.user.User.exists`: ``True`` if the user exists, or | |||||
``False`` if they do not | |||||
- :py:attr:`~earwigbot.wiki.user.User.userid`: an integer ID representing the | |||||
user | |||||
- :py:attr:`~earwigbot.wiki.user.User.blockinfo`: information about any current | |||||
blocks on the user (``False`` if no block, or a dict of | |||||
``{"by": blocking_user, "reason": block_reason, | |||||
"expiry": block_expire_time}``) | |||||
- :py:attr:`~earwigbot.wiki.user.User.groups`: a list of the user's groups | |||||
- :py:attr:`~earwigbot.wiki.user.User.rights`: a list of the user's rights | |||||
- :py:attr:`~earwigbot.wiki.user.User.editcount`: the number of edits made by | |||||
the user | |||||
- :py:attr:`~earwigbot.wiki.user.User.registration`: the time the user | |||||
registered as a :py:obj:`time.struct_time` | |||||
- :py:attr:`~earwigbot.wiki.user.User.emailable`: ``True`` if you can email the | |||||
user, ``False`` if you cannot | |||||
- :py:attr:`~earwigbot.wiki.user.User.gender`: the user's gender (``"male"``, | |||||
``"female"``, or ``"unknown"``) | |||||
and the following methods: | |||||
- :py:meth:`~earwigbot.wiki.user.User.reload`: forcibly reload the user's | |||||
attributes (emphasis on *reload* - this is only necessary if there is reason | |||||
to believe they have changed) | |||||
- :py:meth:`~earwigbot.wiki.user.User.get_userpage`: returns a | |||||
:py:class:`~earwigbot.wiki.page.Page` object representing the user's userpage | |||||
- :py:meth:`~earwigbot.wiki.user.User.get_talkpage`: returns a | |||||
:py:class:`~earwigbot.wiki.page.Page` object representing the user's talkpage | |||||
Additional features | |||||
~~~~~~~~~~~~~~~~~~~ | |||||
Not all aspects of the toolset are covered here. Explore `its code and | |||||
docstrings`_ to learn how to use it in a more hands-on fashion. For reference, | |||||
:py:attr:`bot.wiki <earwigbot.bot.Bot.wiki>` is an instance of | |||||
:py:class:`earwigbot.wiki.SitesDB <earwigbot.wiki.sitesdb.SitesDB>` tied to the | |||||
:file:`sites.db` file in the bot's working directory. | |||||
.. _Pywikipedia framework: http://pywikipediabot.sourceforge.net/ | |||||
.. _CentralAuth: http://www.mediawiki.org/wiki/Extension:CentralAuth | |||||
.. _its code and docstrings: https://github.com/earwig/earwigbot/tree/develop/earwigbot/wiki |
@@ -22,9 +22,10 @@ | |||||
""" | """ | ||||
EarwigBot is a Python robot that edits Wikipedia and interacts with people over | EarwigBot is a Python robot that edits Wikipedia and interacts with people over | ||||
IRC. - http://earwig.github.com/earwig/earwigbot | |||||
IRC. - https://github.com/earwig/earwigbot | |||||
See README.md for a basic overview, or the docs/ directory for details. | |||||
See README.rst for an overview, or the docs/ directory for details. This | |||||
documentation is also available online at http://packages.python.org/earwigbot. | |||||
""" | """ | ||||
__author__ = "Ben Kurtovic" | __author__ = "Ben Kurtovic" | ||||
@@ -35,17 +36,17 @@ __email__ = "ben.kurtovic@verizon.net" | |||||
__release__ = False | __release__ = False | ||||
if not __release__: | if not __release__: | ||||
def _add_git_commit_id_to_version(version): | |||||
def _add_git_commit_id_to_version_string(version): | |||||
from git import Repo | from git import Repo | ||||
from os.path import split, dirname | from os.path import split, dirname | ||||
path = split(dirname(__file__))[0] | path = split(dirname(__file__))[0] | ||||
commit_id = Repo(path).head.object.hexsha | commit_id = Repo(path).head.object.hexsha | ||||
return version + ".git+" + commit_id[:8] | return version + ".git+" + commit_id[:8] | ||||
try: | try: | ||||
__version__ = _add_git_commit_id_to_version(__version__) | |||||
__version__ = _add_git_commit_id_to_version_string(__version__) | |||||
except Exception: | except Exception: | ||||
pass | pass | ||||
finally: | finally: | ||||
del _add_git_commit_id_to_version | |||||
del _add_git_commit_id_to_version_string | |||||
from earwigbot import bot, commands, config, irc, managers, tasks, util, wiki | from earwigbot import bot, commands, config, irc, managers, tasks, util, wiki |
@@ -39,6 +39,7 @@ class Bot(object): | |||||
EarwigBot has three components that can run independently of each other: an | EarwigBot has three components that can run independently of each other: an | ||||
IRC front-end, an IRC watcher, and a wiki scheduler. | IRC front-end, an IRC watcher, and a wiki scheduler. | ||||
* The IRC front-end runs on a normal IRC server and expects users to | * The IRC front-end runs on a normal IRC server and expects users to | ||||
interact with it/give it commands. | interact with it/give it commands. | ||||
* The IRC watcher runs on a wiki recent-changes server and listens for | * The IRC watcher runs on a wiki recent-changes server and listens for | ||||
@@ -20,20 +20,18 @@ | |||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
# SOFTWARE. | # SOFTWARE. | ||||
""" | |||||
EarwigBot's IRC Commands | |||||
This package provides the IRC "commands" used by the bot's front-end component. | |||||
This module contains the BaseCommand class (import with | |||||
`from earwigbot.commands import BaseCommand`), whereas the package contains | |||||
various built-in commands. Additional commands can be installed as plugins in | |||||
the bot's working directory. | |||||
""" | |||||
__all__ = ["BaseCommand"] | __all__ = ["BaseCommand"] | ||||
class BaseCommand(object): | class BaseCommand(object): | ||||
"""A base class for commands on IRC. | |||||
""" | |||||
EarwigBot's Base IRC Command | |||||
This package provides built-in IRC "commands" used by the bot's front-end | |||||
component. Additional commands can be installed as plugins in the bot's | |||||
working directory. | |||||
This class (import with `from earwigbot.commands import BaseCommand`), | |||||
can be subclassed to create custom IRC commands. | |||||
This docstring is reported to the user when they use !help <command>. | This docstring is reported to the user when they use !help <command>. | ||||
""" | """ | ||||
@@ -29,9 +29,7 @@ class Command(BaseCommand): | |||||
def check(self, data): | def check(self, data): | ||||
cmnds = ["chanops", "voice", "devoice", "op", "deop", "join", "part"] | cmnds = ["chanops", "voice", "devoice", "op", "deop", "join", "part"] | ||||
if data.is_command and data.command in cmnds: | |||||
return True | |||||
return False | |||||
return data.is_command and data.command in cmnds | |||||
def process(self, data): | def process(self, data): | ||||
if data.command == "chanops": | if data.command == "chanops": | ||||
@@ -77,11 +75,11 @@ class Command(BaseCommand): | |||||
reason = None | reason = None | ||||
if data.args: | if data.args: | ||||
if data.args[0].startswith("#"): | if data.args[0].startswith("#"): | ||||
# !part #channel reason for parting | |||||
# "!part #channel reason for parting" | |||||
channel = data.args[0] | channel = data.args[0] | ||||
if data.args[1:]: | if data.args[1:]: | ||||
reason = " ".join(data.args[1:]) | reason = " ".join(data.args[1:]) | ||||
else: # !part reason for parting; assume current channel | |||||
else: # "!part reason for parting"; assume current channel | |||||
reason = " ".join(data.args) | reason = " ".join(data.args) | ||||
msg = "Requested by {0}".format(data.nick) | msg = "Requested by {0}".format(data.nick) | ||||
@@ -40,6 +40,7 @@ class BotConfig(object): | |||||
from scratch at the inital bot run. | from scratch at the inital bot run. | ||||
BotConfig has a few properties and functions, including the following: | BotConfig has a few properties and functions, including the following: | ||||
* config.root_dir - bot's working directory; contains config.yml, logs/ | * config.root_dir - bot's working directory; contains config.yml, logs/ | ||||
* config.path - path to the bot's config file | * config.path - path to the bot's config file | ||||
* config.components - enabled components | * config.components - enabled components | ||||
@@ -50,6 +51,7 @@ class BotConfig(object): | |||||
* config.schedule() - tasks scheduled to run at a given time | * config.schedule() - tasks scheduled to run at a given time | ||||
BotConfig also has some functions used in config loading: | BotConfig also has some functions used in config loading: | ||||
* config.load() - loads and parses our config file, returning True if | * config.load() - loads and parses our config file, returning True if | ||||
passwords are stored encrypted or False otherwise; | passwords are stored encrypted or False otherwise; | ||||
can also be used to easily reload config | can also be used to easily reload config | ||||
@@ -150,7 +152,10 @@ class BotConfig(object): | |||||
#else: | #else: | ||||
# is_encrypted = False | # is_encrypted = False | ||||
raise NotImplementedError() | raise NotImplementedError() | ||||
# yaml.dumps() | |||||
# yaml.dumps() config.yml file (self._config_path) | |||||
# Create root_dir/, root_dir/commands/, root_dir/tasks/ | |||||
# Give a reasonable message after config has been created regarding | |||||
# what to do next... | |||||
@property | @property | ||||
def root_dir(self): | def root_dir(self): | ||||
@@ -57,7 +57,7 @@ class Frontend(IRCConnection): | |||||
data.nick, data.ident, data.host = self.sender_regex.findall(line[0])[0] | data.nick, data.ident, data.host = self.sender_regex.findall(line[0])[0] | ||||
data.chan = line[2] | data.chan = line[2] | ||||
data.parse_args() | data.parse_args() | ||||
self.bot.commands.check("join", data) | |||||
self.bot.commands.call("join", data) | |||||
elif line[1] == "PRIVMSG": | elif line[1] == "PRIVMSG": | ||||
data.nick, data.ident, data.host = self.sender_regex.findall(line[0])[0] | data.nick, data.ident, data.host = self.sender_regex.findall(line[0])[0] | ||||
@@ -69,13 +69,13 @@ class Frontend(IRCConnection): | |||||
# This is a privmsg to us, so set 'chan' as the nick of the | # This is a privmsg to us, so set 'chan' as the nick of the | ||||
# sender, then check for private-only command hooks: | # sender, then check for private-only command hooks: | ||||
data.chan = data.nick | data.chan = data.nick | ||||
self.bot.commands.check("msg_private", data) | |||||
self.bot.commands.call("msg_private", data) | |||||
else: | else: | ||||
# Check for public-only command hooks: | # Check for public-only command hooks: | ||||
self.bot.commands.check("msg_public", data) | |||||
self.bot.commands.call("msg_public", data) | |||||
# Check for command hooks that apply to all messages: | # Check for command hooks that apply to all messages: | ||||
self.bot.commands.check("msg", data) | |||||
self.bot.commands.call("msg", data) | |||||
elif line[0] == "PING": # If we are pinged, pong back | elif line[0] == "PING": # If we are pinged, pong back | ||||
self.pong(line[1]) | self.pong(line[1]) | ||||
@@ -163,8 +163,8 @@ class CommandManager(_ResourceManager): | |||||
e = "Error executing command '{0}':" | e = "Error executing command '{0}':" | ||||
self.logger.exception(e.format(data.command)) | self.logger.exception(e.format(data.command)) | ||||
def check(self, hook, data): | |||||
"""Given an IRC event, check if there's anything we can respond to.""" | |||||
def call(self, hook, data): | |||||
"""Given a hook type and a Data object, respond appropriately.""" | |||||
self.lock.acquire() | self.lock.acquire() | ||||
for command in self._resources.itervalues(): | for command in self._resources.itervalues(): | ||||
if hook in command.hooks and self._wrap_check(command, data): | if hook in command.hooks and self._wrap_check(command, data): | ||||
@@ -20,24 +20,24 @@ | |||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
# SOFTWARE. | # SOFTWARE. | ||||
""" | |||||
EarwigBot's Bot Tasks | |||||
This package provides the wiki bot "tasks" EarwigBot runs. This module contains | |||||
the BaseTask class (import with `from earwigbot.tasks import BaseTask`), | |||||
whereas the package contains various built-in tasks. Additional tasks can be | |||||
installed as plugins in the bot's working directory. | |||||
To run a task, use bot.tasks.start(name, **kwargs). **kwargs get passed to the | |||||
Task's run() function. | |||||
""" | |||||
from earwigbot import wiki | from earwigbot import wiki | ||||
__all__ = ["BaseTask"] | __all__ = ["BaseTask"] | ||||
class BaseTask(object): | class BaseTask(object): | ||||
"""A base class for bot tasks that edit Wikipedia.""" | |||||
""" | |||||
EarwigBot's Base Bot Task | |||||
This package provides built-in wiki bot "tasks" EarwigBot runs. Additional | |||||
tasks can be installed as plugins in the bot's working directory. | |||||
This class (import with `from earwigbot.tasks import BaseTask`) can be | |||||
subclassed to create custom bot tasks. | |||||
To run a task, use :py:meth:`bot.tasks.start(name, **kwargs) | |||||
<earwigbot.managers.TaskManager.start>`. ``**kwargs`` get passed to the | |||||
Task's run() function. | |||||
""" | |||||
name = None | name = None | ||||
number = 0 | number = 0 | ||||
@@ -25,8 +25,8 @@ from earwigbot.tasks import BaseTask | |||||
__all__ = ["Task"] | __all__ = ["Task"] | ||||
class Task(BaseTask): | class Task(BaseTask): | ||||
"""A task to add |blp=yes to {{WPB}} or {{WPBS}} when it is used along with | |||||
{{WP Biography}}.""" | |||||
"""A task to add |blp=yes to ``{{WPB}}`` or ``{{WPBS}}`` when it is used | |||||
along with ``{{WP Biography}}``.""" | |||||
name = "blptag" | name = "blptag" | ||||
def setup(self): | def setup(self): | ||||
@@ -38,19 +38,23 @@ class Page(CopyrightMixin): | |||||
about the page, getting page content, and so on. Category is a subclass of | about the page, getting page content, and so on. Category is a subclass of | ||||
Page with additional methods. | Page with additional methods. | ||||
Attributes: | |||||
title -- the page's title, or pagename | |||||
exists -- whether the page exists | |||||
pageid -- an integer ID representing the page | |||||
url -- the page's URL | |||||
namespace -- the page's namespace as an integer | |||||
protection -- the page's current protection status | |||||
is_talkpage -- True if the page is a talkpage, else False | |||||
is_redirect -- True if the page is a redirect, else False | |||||
Public methods: | Public methods: | ||||
title -- returns the page's title, or pagename | |||||
exists -- returns whether the page exists | |||||
pageid -- returns an integer ID representing the page | |||||
url -- returns the page's URL | |||||
namespace -- returns the page's namespace as an integer | |||||
protection -- returns the page's current protection status | |||||
creator -- returns the page's creator (first user to edit) | |||||
is_talkpage -- returns True if the page is a talkpage, else False | |||||
is_redirect -- returns True if the page is a redirect, else False | |||||
reload -- forcibly reload the page's attributes | |||||
toggle_talk -- returns a content page's talk page, or vice versa | toggle_talk -- returns a content page's talk page, or vice versa | ||||
get -- returns page content | get -- returns page content | ||||
get_redirect_target -- if the page is a redirect, returns its destination | get_redirect_target -- if the page is a redirect, returns its destination | ||||
get_creator -- returns a User object representing the first person | |||||
to edit the page | |||||
edit -- replaces the page's content or creates a new page | edit -- replaces the page's content or creates a new page | ||||
add_section -- adds a new section at the bottom of the page | add_section -- adds a new section at the bottom of the page | ||||
copyvio_check -- checks the page for copyright violations | copyvio_check -- checks the page for copyright violations | ||||
@@ -56,16 +56,18 @@ class Site(object): | |||||
instances, tools.add_site() for adding new ones to config, and | instances, tools.add_site() for adding new ones to config, and | ||||
tools.del_site() for removing old ones from config, should suffice. | tools.del_site() for removing old ones from config, should suffice. | ||||
Attributes: | |||||
name -- the site's name (or "wikiid"), like "enwiki" | |||||
project -- the site's project name, like "wikipedia" | |||||
lang -- the site's language code, like "en" | |||||
domain -- the site's web domain, like "en.wikipedia.org" | |||||
Public methods: | Public methods: | ||||
name -- returns our name (or "wikiid"), like "enwiki" | |||||
project -- returns our project name, like "wikipedia" | |||||
lang -- returns our language code, like "en" | |||||
domain -- returns our web domain, like "en.wikipedia.org" | |||||
api_query -- does an API query with the given kwargs as params | api_query -- does an API query with the given kwargs as params | ||||
sql_query -- does an SQL query and yields its results | sql_query -- does an SQL query and yields its results | ||||
get_replag -- returns the estimated database replication lag | get_replag -- returns the estimated database replication lag | ||||
namespace_id_to_name -- given a namespace ID, returns associated name(s) | namespace_id_to_name -- given a namespace ID, returns associated name(s) | ||||
namespace_name_to_id -- given a namespace name, returns associated id | |||||
namespace_name_to_id -- given a namespace name, returns the associated ID | |||||
get_page -- returns a Page object for the given title | get_page -- returns a Page object for the given title | ||||
get_category -- returns a Category object for the given title | get_category -- returns a Category object for the given title | ||||
get_user -- returns a User object for the given username | get_user -- returns a User object for the given username | ||||
@@ -36,17 +36,20 @@ class User(object): | |||||
information about the user, such as editcount and user rights, methods for | information about the user, such as editcount and user rights, methods for | ||||
returning the user's userpage and talkpage, etc. | returning the user's userpage and talkpage, etc. | ||||
Attributes: | |||||
name -- the user's username | |||||
exists -- True if the user exists, or False if they do not | |||||
userid -- an integer ID representing the user | |||||
blockinfo -- information about any current blocks on the user | |||||
groups -- a list of the user's groups | |||||
rights -- a list of the user's rights | |||||
editcount -- the number of edits made by the user | |||||
registration -- the time the user registered as a time.struct_time | |||||
emailable -- True if you can email the user, False if you cannot | |||||
gender -- the user's gender ("male", "female", or "unknown") | |||||
Public methods: | Public methods: | ||||
name -- returns the user's username | |||||
exists -- returns True if the user exists, False if they do not | |||||
userid -- returns an integer ID representing the user | |||||
blockinfo -- returns information about a current block on the user | |||||
groups -- returns a list of the user's groups | |||||
rights -- returns a list of the user's rights | |||||
editcount -- returns the number of edits made by the user | |||||
registration -- returns the time the user registered as a time.struct_time | |||||
emailable -- returns True if you can email the user, False if you cannot | |||||
gender -- returns the user's gender ("male", "female", or "unknown") | |||||
reload -- forcibly reload the user's attributes | |||||
get_userpage -- returns a Page object representing the user's userpage | get_userpage -- returns a Page object representing the user's userpage | ||||
get_talkpage -- returns a Page object representing the user's talkpage | get_talkpage -- returns a Page object representing the user's talkpage | ||||
""" | """ | ||||