Creating a plugin

It is possible to extend pretix with custom Python code using the official plugin API. Every plugin has to be implemented as an independent Django ‘app’ living in its own python package installed like any other python module. There are also some official plugins inside the pretix/plugins/ directory of your pretix installation.

The communication between pretix and the plugins happens mostly using Django’s signal dispatcher feature. The core modules of pretix, pretix.base, pretix.control and pretix.presale expose a number of signals which are documented on the next pages.

To create a new plugin, create a new python package which must be a valid Django app and must contain plugin metadata, as described below. There is some boilerplate that you will need for every plugin to get started. To save your time, we created a cookiecutter template that you can use like this:

$ pip install cookiecutter
$ cookiecutter https://github.com/pretix/pretix-plugin-cookiecutter

This will ask you some questions and then create a project folder for your plugin.

The following pages go into detail about the several types of plugins currently supported. While these instructions don’t assume that you know a lot about pretix, they do assume that you have prior knowledge about Django (e.g. its view layer, how its ORM works, etc.).

Plugin metadata

The plugin metadata lives inside a PretixPluginMeta class inside your app’s configuration class. The metadata class must define the following attributes:

Attribute

Type

Description

name

string

The human-readable name of your plugin

author

string

Your name

version

string

A human-readable version code of your plugin

description

string

A more verbose description of what your plugin does. May contain HTML.

category

string

Category of a plugin. Either one of "FEATURE", "PAYMENT", "INTEGRATION", "CUSTOMIZATION", "FORMAT", or "API", or any other string.

picture

string (optional)

Path to a picture resolvable through the static file system.

featured

boolean (optional)

False by default, can promote a plugin if it’s something many users will want, use carefully.

visible

boolean (optional)

True by default, can hide a plugin so it cannot be normally activated.

restricted

boolean (optional)

False by default, restricts a plugin such that it can only be enabled for an event by system administrators / superusers.

experimental

boolean (optional)

False by default, marks a plugin as an experimental feature in the plugins list.

compatibility

string

Specifier for compatible pretix versions.

A working example would be:

 1try:
 2    from pretix.base.plugins import PluginConfig
 3except ImportError:
 4    raise RuntimeError("Please use pretix 2.7 or above to run this plugin!")
 5from django.utils.translation import gettext_lazy as _
 6
 7
 8class PaypalApp(PluginConfig):
 9    name = 'pretix_paypal'
10    verbose_name = _("PayPal")
11
12    class PretixPluginMeta:
13        name = _("PayPal")
14        author = _("the pretix team")
15        version = '1.0.0'
16        category = 'PAYMENT'
17        picture = 'pretix_paypal/paypal_logo.svg'
18        visible = True
19        featured = False
20        restricted = False
21        description = _("This plugin allows you to receive payments via PayPal")
22        compatibility = "pretix>=2.7.0"
23
24
25default_app_config = 'pretix_paypal.PaypalApp'

The AppConfig class may implement a property compatibility_errors, that checks whether the pretix installation meets all requirements of the plugin. If so, it should contain None or an empty list, otherwise a list of strings containing human-readable error messages. We recommend using the django.utils.functional.cached_property decorator, as it might get called a lot. You can also implement compatibility_warnings, those will be displayed but not block the plugin execution.

The AppConfig class may implement a method is_available(event) that checks if a plugin is available for a specific event. If not, it will not be shown in the plugin list of that event. You should not define is_available and restricted on the same plugin.

Plugin registration

Somehow, pretix needs to know that your plugin exists at all. For this purpose, we make use of the entry point feature of setuptools. To register a plugin that lives in a separate python package, your setup.py should contain something like this:

1setup(
2    args...,
3    entry_points="""
4[pretix.plugin]
5pretix_paypal=pretix_paypal:PretixPluginMeta
6"""
7)

This will automatically make pretix discover this plugin as soon as it is installed e.g. through pip. During development, you can just run python setup.py develop inside your plugin source directory to make it discoverable.

Signals

The various components of pretix define a number of signals which your plugin can listen for. We will go into the details of the different signals in the following pages. We suggest that you put your signal receivers into a signals submodule of your plugin. You should extend your AppConfig (see above) by the following method to make your receivers available:

1class PaypalApp(AppConfig):
2    
3
4    def ready(self):
5        from . import signals  # NOQA

You can optionally specify code that is executed when your plugin is activated for an event in the installed method:

1class PaypalApp(AppConfig):
2    
3
4    def installed(self, event):
5        pass  # Your code here

Note that installed will not be called if the plugin is indirectly activated for an event because the event is created with settings copied from another event.

Registries

Many signals in pretix are used so that plugins can “register” a class, e.g. a payment provider or a ticket renderer.

However, for some of them (types of Log Entries) we use a different method to keep track of them: In a Registry, classes are collected at application startup, along with a unique key (in case of LogEntryType, the action_type) as well as which plugin registered them.

To register a class, you can use one of several decorators provided by the Registry object:

class pretix.base.logentrytypes.LogEntryTypeRegistry
new(*args, **kwargs)

Instantiate the decorated class with the given *args and **kwargs, and register the instance in this registry. May be used multiple times.

1@animal_sound_registry.new("meow")
2@animal_sound_registry.new("woof")
3class AnimalSound:
4  def __init__(self, sound):
5    # ...
new_from_dict(data)

Register multiple instance of a LogEntryType class with different action_type and plain text strings, as given by the items of the specified data dictionary.

This method is designed to be used as a decorator as follows:

1@log_entry_types.new_from_dict({
2    'pretix.event.item.added': _('The product has been created.'),
3    'pretix.event.item.changed': _('The product has been changed.'),
4    # ...
5})
6class CoreItemLogEntryType(ItemLogEntryType):
7    # ...
Parameters:

data – action types and descriptions {"some_action_type": "Plain text description", ...}

register(*objs)

Register one or more entries in this registry.

Usable as a regular method or as decorator on a class or function. If used on a class, the class type object itself is registered, not an instance of the class. To register an instance, use the new method.

@some_registry.register
def my_new_entry(foo):
  # ...

All files in which classes are registered need to be imported in the AppConfig.ready as explained in Signals above.

Views

Your plugin may define custom views. If you put an urls submodule into your plugin module, pretix will automatically import it and include it into the root URL configuration with the namespace plugins:<label>:, where <label> is your Django app label.

Warning

If you define custom URLs and views, you are currently on your own with checking that the calling user is logged in, has appropriate permissions, etc. We plan on providing native support for this in a later version.