mirror of
https://github.com/Limnoria/Limnoria-doc.git
synced 2025-04-04 14:29:46 +00:00
- Remove the discussion of README and copyright header. These are already very self documenting - Add links to the `help` syntax reference in use/getting_started - Update the headings to include the contents of each file in addition to its name for easier navigation - Further remove redundant language
458 lines
19 KiB
ReStructuredText
458 lines
19 KiB
ReStructuredText
.. _plugin-tutorial:
|
|
|
|
**********************************
|
|
Writing Your First Limnoria Plugin
|
|
**********************************
|
|
|
|
Introduction
|
|
============
|
|
|
|
This page is a top-down guide on how to write new plugins for Limnoria.
|
|
|
|
Before you start, you should install a copy of Limnoria on the machine you
|
|
intend to develop plugins on, as it includes some additional scripts like
|
|
:command:`supybot-plugin-create` to generate the plugin skeleton.
|
|
|
|
We'll go through this tutorial by writing a new plugin, named Random with a
|
|
few commands.
|
|
|
|
Generating the Plugin template
|
|
==============================
|
|
|
|
The recommended way to start writing a plugin is to use the
|
|
:command:`supybot-plugin-create` wizard. You can run this from within your bot's
|
|
plugins directory, or make a separate directory for all your own plugins and run
|
|
it there. (You can add additional plugin directories to your bot config using
|
|
``config directories.plugins``). The latter approach is probably easier if you
|
|
intend to publish your code afterwards, as it keeps your code separate from any
|
|
other plugins you've installed.
|
|
|
|
Here's an example session::
|
|
|
|
$ supybot-plugin-create
|
|
What should the name of the plugin be? Random
|
|
|
|
Sometimes you'll want a callback to be threaded. If its methods
|
|
(command or regexp-based, either one) will take a significant amount
|
|
of time to run, you'll want to thread them so they don't block the
|
|
entire bot.
|
|
|
|
Does your plugin need to be threaded? [y/n] n
|
|
|
|
What is your name, so I can fill in the copyright and license
|
|
appropriately? John Doe
|
|
|
|
Do you wish to use Supybot's license for your plugin? [y/n] y
|
|
|
|
Please provide a short description of the plugin: This plugin contains
|
|
commands relating to random numbers, including random sampling from a list
|
|
and a simple dice roller.
|
|
|
|
.. _init-py:
|
|
Plugin metadata (__init__.py)
|
|
=============================
|
|
The first file we'll look at is :file:`__init__.py`. This is the "glue" that
|
|
makes a directory a Python package, and sets up how the plugin is actually
|
|
loaded.
|
|
|
|
For Limnoria plugins specifically, there are some metadata fields you may wish
|
|
to fill in:
|
|
|
|
* ``__version__``: the plugin version (optional)
|
|
* ``__author__`` describes the author of the plugin, in the form of a
|
|
:class:`supybot.Author` instance::
|
|
|
|
__author__ = supybot.Author(name='Daniel DiPaolo', nick='Strike',
|
|
email='somewhere@someplace.xxx')
|
|
|
|
* ``__contributors__`` is a dictionary mapping :class:`supybot.Author`
|
|
instances to lists of things they contributed. See e.g. `in the Plugin plugin
|
|
<https://github.com/progval/Limnoria/blob/master/plugins/Plugin/__init__.py#L42-L49>`_.
|
|
This is most useful if a plugin has had multiple authors / contributors in
|
|
its history.
|
|
|
|
* ``__url__`` references the download URL for the plugin. Since this is just an
|
|
example, we'll leave this blank.
|
|
|
|
The rest of :file:`__init__.py` can be left alone unless you are using
|
|
third-party modules in your plugin. If you are, you may wish to add
|
|
additional import statements and ``reload`` calls to those modules, so that
|
|
they get reloaded with the rest of the plugin::
|
|
|
|
from . import config
|
|
from . import plugin
|
|
from importlib import reload
|
|
reload(plugin) # In case we're being reloaded.
|
|
# Add more reloads here if you add third-party modules and want them
|
|
# to be reloaded when this plugin is reloaded. Don't forget to
|
|
# import them as well!
|
|
|
|
.. note::
|
|
If your plugin has any third-party dependencies, you may want to list them in
|
|
the README and add a `requirements.txt <https://pip.pypa.io/en/stable/reference/requirements-file-format/>`_
|
|
file for easy installation (via ``pip install -r requirements.txt``).
|
|
|
|
.. _config-py:
|
|
Configuration (config.py)
|
|
=========================
|
|
:file:`config.py` defines the configuration variables for a plugin.
|
|
For this tutorial, the Random plugin is simple enough that it doesn't
|
|
have any config variables, so the template can be left as is. Writing plugin
|
|
configuration is explained in depth in the :ref:`Plugin Config Tutorial <configuration-tutorial>`.
|
|
|
|
To briefly outline this file's structure: the ``configure`` function is used by
|
|
the :command:`supybot-wizard` wizard and allows users to configure the plugin
|
|
further if it's present when the bot is first installed. (In practice though,
|
|
this is seldomly used by third-party plugins as they're generally installed
|
|
*after* configuring the bot.)
|
|
|
|
The following line registers an entry for the plugin in Limnoria's config
|
|
registry, followed by an example of how to register a config variable::
|
|
|
|
Random = conf.registerPlugin('Random')
|
|
# This is where your configuration variables (if any) should go. For example:
|
|
# conf.registerGlobalValue(Random, 'someConfigVariableName',
|
|
# registry.Boolean(False, _("""Help for someConfigVariableName.""")))
|
|
|
|
.. _plugin-py:
|
|
Plugin commands (plugin.py)
|
|
===========================
|
|
``plugin.py`` includes the core functionality of the plugin. For most plugins this will
|
|
include command handlers, as well as anything else relevant to its
|
|
purpose (database queries,
|
|
:ref:`HTTP server endpoints <http_plugins>`,
|
|
:ref:`IRC command triggers <do-method-handlers>`, etc.)
|
|
|
|
As with any Python module, you'll need to import any dependencies you want,
|
|
in addition to the standard ``supybot`` imports included in the plugin
|
|
template::
|
|
|
|
import random
|
|
|
|
The bulk of the plugin definition then resides in a subclass of
|
|
:class:`callbacks.Plugin`. By convention, the class name is equal to the name of
|
|
the plugin, though this is not strictly required (the actual linkage is done by
|
|
the ``Class = Random`` statement at the end of the file). Note that the docstring
|
|
of a plugin class also can be shown using the
|
|
``plugin help <plugin name>`` command. ::
|
|
|
|
class Random(callbacks.Plugin):
|
|
"""This plugin contains commands relating to random numbers, including random sampling from a list and a simple dice roller."""
|
|
|
|
def __init__(self, irc):
|
|
# Make sure to call the superclass' constructor when you define a custom one
|
|
super().__init__(irc)
|
|
self.rng = random.Random() # create our rng
|
|
self.rng.seed() # automatically seeds with current time
|
|
|
|
For this sample plugin, we define a custom constructor (``__init__``) that
|
|
instantiates a random number generator instance and pre-seeds it. This isn't
|
|
technically necessary when using Python's ``random`` module, but it helps outline
|
|
how to write a similar constructor. Notice in particular how you must pass in
|
|
an ``irc`` argument in addition to ``self``.
|
|
|
|
.. warning::
|
|
Because Limnoria is a multi-network bot, you should generally ignore
|
|
the ``irc`` instance passed to the plugin constructor.
|
|
On a manual ``load`` call to a live bot, this will be set to the network
|
|
the command was run on, but on bot startup, ``irc`` will be (arbitrarily)
|
|
set to the first network that the bot decides to connect to.
|
|
|
|
Basic command handler
|
|
---------------------
|
|
|
|
Our first command definition can immediately follow:
|
|
|
|
..
|
|
note: turn off automatic dedent so that the functions appear at the right
|
|
indent level relative to the class definition
|
|
|
|
.. code-block::
|
|
:dedent: 0
|
|
|
|
@wrap
|
|
def random(self, irc, msg, args):
|
|
"""takes no arguments
|
|
|
|
Returns the next random number from the random number generator.
|
|
"""
|
|
irc.reply(str(self.rng.random()))
|
|
|
|
.. note::
|
|
All functions used as commands must have an all lowercase name.
|
|
|
|
A command function taking in no arguments from IRC requires 4 arguments at base;
|
|
they are as follows:
|
|
|
|
- ``self``: refers to the class instance. It is common to keep local state
|
|
for the plugin as instance variables within the plugin class.
|
|
- ``irc``: a :class:`supybot.callbacks.ReplyIrcProxy` instance; refers to the
|
|
IRC network instance the command was called on
|
|
- ``msg``: a :class:`supybot.ircmsgs.IrcMsg` instance; refers to the
|
|
IRC message that triggered this command.
|
|
- ``args``: a raw list of remaining unconverted arguments; new plugins that
|
|
use :ref:`@wrap <using-wrap>` for automatic argument type conversion should
|
|
never need to interact with ``args`` directly.
|
|
|
|
The function docstring is shown in the ``help`` command, and is expected to be
|
|
in a specific format:
|
|
|
|
- The first line shows the :ref:`argument list <help-syntax>` when someone calls the
|
|
``help`` command on this command (i.e., ``help random``).
|
|
- The second line is blank.
|
|
- Everything afterwards is the actual description of the command. This should be
|
|
brief enough to ideally fit in one IRC message, which is around a 450 character limit.
|
|
Indentation and line breaks are normalized by the ``help`` command, so it is fine
|
|
to wrap longer text onto multiple lines.
|
|
|
|
The :py:meth:`irc.reply <supybot.callbacks.ReplyIrcProxy.reply>` call
|
|
is a bit of magic: it issues a reply the same place as the message that
|
|
triggered the command. i.e. this may be in a channel or in a private
|
|
conversation with the bot.
|
|
|
|
Lastly, command functions are registered with the bot using the :ref:`@wrap <using-wrap>`
|
|
decorator. For commands that require no parameters, calling ``@wrap`` with no
|
|
arguments is sufficient.
|
|
|
|
Command handler with parameters
|
|
-------------------------------
|
|
|
|
Now let's create a command with an argument.
|
|
The ``seed`` command lets the user pick a specific RNG seed:
|
|
|
|
.. code-block::
|
|
:dedent: 0
|
|
|
|
@wrap(['float'])
|
|
def seed(self, irc, msg, args, seed):
|
|
"""<seed>
|
|
|
|
Sets the internal RNG's seed value to <seed>. <seed> must be a
|
|
floating point number.
|
|
"""
|
|
self.rng.seed(seed)
|
|
irc.replySuccess()
|
|
|
|
Command arguments are declared by:
|
|
|
|
1. Adding :ref:`type converters <wrap-converter-list>`, one for each parameter,
|
|
to the list passed into ``@wrap``
|
|
2. Adding one function parameter per argument to the command function
|
|
definition. (i.e. ``def seed(self, irc, msg, args, seed)`` instead of
|
|
``def seed(self, irc, msg, args)``)
|
|
|
|
We also modify the docstring to document this function. Note the syntax
|
|
on the first line: :ref:`by convention <help-syntax>`, required arguments go in ``<>`` and optional
|
|
arguments should be surrounded by ``[]``.
|
|
|
|
The function body includes a new method
|
|
:py:meth:`irc.replySuccess <supybot.callbacks.RichReplyMethods.replySuccess>`.
|
|
This is a generic "I succeeded" command which responds with whatever the
|
|
bot owner has configured in ``config supybot.replies.success``.
|
|
Also, by using ``@wrap``, we don't need to do any type checking inside the
|
|
function itself - this is handled separately, and invalid argument values will
|
|
cause the command to error before it reaches the wrapped function.
|
|
|
|
With this alone you'd be able to make a range of useful plugin commands, but
|
|
we'll go include some more examples to illustrate common patterns.
|
|
|
|
Command handler with list-type arguments
|
|
----------------------------------------
|
|
The next command is named ``sample``: it takes a random
|
|
sample of arbitrary size from a list provided by the user:
|
|
|
|
.. code-block::
|
|
:dedent: 0
|
|
|
|
@wrap(['int', many('anything')])
|
|
def sample(self, irc, msg, args, n, items):
|
|
"""<number of items> <item1> [<item2> ...]
|
|
|
|
Returns a sample of the <number of items> taken from the remaining
|
|
arguments. <number of items> must be less than the number
|
|
of arguments given.
|
|
"""
|
|
if n > len(items):
|
|
# Calling irc.error with Raise=True is an alternative early return
|
|
irc.error('<number of items> must be less than the number '
|
|
'of arguments.', Raise=True)
|
|
sample = self.rng.sample(items, n)
|
|
sample.sort()
|
|
irc.reply(utils.str.commaAndify(sample))
|
|
|
|
The important thing to note is that list type arguments are rolled into one
|
|
parameter in the command function by the ``many`` filter. Similar "multiplicity"
|
|
handlers are documented :ref:`here <wrap-multiplicity-handlers>`.
|
|
|
|
We also update the docstring to use the ``[]`` convention when surrounding
|
|
optional arguments.
|
|
|
|
For this function's body,
|
|
:py:meth:`irc.error <supybot.callbacks.ReplyIrcProxy.error>`
|
|
is like
|
|
:py:meth:`irc.replySuccess <supybot.callbacks.ReplyIrcProxy.replySuccess>`
|
|
but for error messages. We prefer using this instead of ``irc.reply`` for error
|
|
signaling because its behaviour can be configured specially. For example, you
|
|
can force all errors to go in private by setting the ``reply.error.inPrivate``
|
|
option, and this can help reduce noise on a busy channel.
|
|
Also, ``irc.error()`` with no text will return a generic error message
|
|
configured in ``supybot.replies.error``, but this is not a valid call to
|
|
:py:meth:`irc.reply <supybot.callbacks.ReplyIrcProxy.reply>`.
|
|
|
|
``utils.str.commaAndify`` is a helper that takes a list of strings
|
|
and turns it into "item1, item2, item3, item4, and item5" for an arbitrary
|
|
length. Limnoria has accumulated many such helpers in its lifetime, many of
|
|
which are described in the :ref:`Using Utils <using-utils>` page.
|
|
|
|
Command handler with optional arguments
|
|
---------------------------------------
|
|
Now for the last command that we will add to our plugin.py. This ``diceroll``
|
|
command will allow the bot users to roll an arbitrary n-sided die, with n
|
|
defaulting to 6:
|
|
|
|
.. code-block::
|
|
:dedent: 0
|
|
|
|
@wrap([additional(('int', 'number of sides'), 6)])
|
|
def diceroll(self, irc, msg, args, n):
|
|
"""[<number of sides>]
|
|
|
|
Rolls a die with <number of sides> sides. The default number of sides
|
|
is 6.
|
|
"""
|
|
s = 'rolls a %s' % self.rng.randrange(1, n)
|
|
irc.reply(s, action=True)
|
|
|
|
The only new thing described here is that ``irc.reply(..., action=True)`` makes
|
|
the bot perform a `/me`. There are some other flags described in the
|
|
:py:meth:`irc.reply <supybot.callbacks.ReplyIrcProxy.reply>`
|
|
documentation too: common ones include ``private=True``, which
|
|
forces a private message, and ``notice=True``, which forces the reply to use
|
|
NOTICE instead of PRIVMSG.
|
|
|
|
.. _test-py:
|
|
Unit tests (test.py)
|
|
====================
|
|
The easy way to test any plugin would be to start up a bot, load the plugin, and
|
|
run all the commands a few times to verify that they work. But this takes time,
|
|
and as a project grows larger, starts to be a tedious and error-prone process...
|
|
|
|
This is where automated testing comes in. Limnoria has a test harness built upon
|
|
the `Python unittest library <https://docs.python.org/3/library/unittest.html>`_
|
|
that abstracts away all the dependencies of live testing (i.e. the IRC
|
|
client and server) and allows you to cover your entire plugin's functionality
|
|
within a few seconds.
|
|
|
|
How it works
|
|
------------
|
|
|
|
Plugin test cases inherit from
|
|
:class:`supybot.test.PluginTestCase` or
|
|
:class:`supybot.test.ChannelPluginTestCase` and include
|
|
:ref:`several methods <plugin-test-methods>` to interact with a simulated
|
|
instance of the bot, in addition to the
|
|
`standard assertion functions <https://docs.python.org/3/library/unittest.html#assert-methods>`_
|
|
provided by the unittest library.
|
|
|
|
Running the tests for a Limnoria plugin is done using the
|
|
:command:`supybot-test` command: i.e. ``supybot-test /path/to/your/Plugin``
|
|
|
|
The structure of these test classes, as well
|
|
as interactions with features like Limnoria's config system are described in
|
|
detail in the :ref:`Advanced Plugin Testing guide <plugin-testing-guide>`.
|
|
|
|
Functional testing examples
|
|
---------------------------
|
|
|
|
For a command where we don't care about the exact output, the usual approach is
|
|
to check that invocations raise or don't raise an error. For a command that
|
|
generates a purely random output, this applies too since we can't predict what
|
|
the result will be::
|
|
|
|
class RandomTestCase(PluginTestCase):
|
|
# This tuple determines which plugins to load in the test case
|
|
plugins = ('Random',)
|
|
|
|
def testRandom(self):
|
|
self.assertNotError('random')
|
|
|
|
# This throws, because the command doesn't expect any arguments
|
|
self.assertError('random abcdef')
|
|
|
|
However, this is less true if you pre-seed the RNG, as then you're guaranteed
|
|
a repeatable result. The following snippet introduces
|
|
``assertResponse(commandPlusArgs, expectedOutput)``, where ``commandPlusArgs``
|
|
is the full bot command including arguments, all as one string:
|
|
|
|
.. code-block::
|
|
:dedent: 0
|
|
|
|
def testSeed(self):
|
|
self.assertNotError('seed 20')
|
|
self.assertResponse('random', '0.9056396761745207')
|
|
self.assertResponse('random', '0.6862541570267026')
|
|
self.assertNotError('seed 20')
|
|
self.assertResponse('random', '0.9056396761745207')
|
|
self.assertNotError('seed 1234')
|
|
self.assertResponse('random', '0.9664535356921388')
|
|
|
|
Alternatively, you can use ``getMsg(command)`` to fetch the output of a bot
|
|
command as a string and reuse it:
|
|
|
|
.. code-block::
|
|
:dedent: 0
|
|
|
|
def testSeed(self):
|
|
self.assertNotError('seed 20')
|
|
num1 = self.getMsg('random')
|
|
num2 = self.getMsg('random')
|
|
|
|
self.assertNotError('seed 20')
|
|
num1_again = self.getMsg('random')
|
|
|
|
self.assertEqual(num1, num1_again)
|
|
self.assertNotEqual(num1, num2)
|
|
|
|
Another common practice is to use regular expressions to match the output of
|
|
a command:
|
|
|
|
.. note::
|
|
The :func:`assertRegexp` defined in Limnoria is `not` the same as
|
|
:func:`assertRegex` from the standard unittest library. The latter
|
|
compares a regexp against a bare string, not the output of a bot command.
|
|
(For historical reasons, we have this confusing name.)
|
|
|
|
.. code-block::
|
|
:dedent: 0
|
|
|
|
def testSample(self):
|
|
self.assertError('sample 20 foo') # can't sample 20 from only 1 element
|
|
self.assertResponse('sample 1 foo', 'foo')
|
|
self.assertRegexp('sample 2 foo bar', '... and ...')
|
|
self.assertRegexp('sample 3 foo bar baz', '..., ..., and ...')
|
|
# assertNotRegexp(commandWithArgs, regexp) also works as expected
|
|
|
|
def testDiceRoll(self):
|
|
self.assertActionRegexp('diceroll', 'rolls a \d')
|
|
|
|
Conclusion
|
|
==========
|
|
You are now well prepared to write Limnoria plugins. A few words of wisdom:
|
|
|
|
* Read other people's plugins, especially the included plugins and ones by
|
|
the core developers. We can't possibly document all the things that Limnoria
|
|
can do, though we try our best.
|
|
|
|
* Hack new functionality into existing plugins first if writing a new
|
|
plugin is too daunting.
|
|
|
|
* Come ask us questions in #limnoria on Libera. Going back to the
|
|
first point above, the developers themselves can help you even more than
|
|
the docs can (though we prefer you read the docs first).
|
|
|
|
* :ref:`Share your plugins with the world <distributing-plugins>`
|
|
and make Limnoria all that more attractive for other users so they will want
|
|
to write their plugins for Limnoria as well.
|
|
|
|
* And of course, have fun!
|