mirror of
https://github.com/pretix/pretix.git
synced 2026-05-04 15:04:03 +00:00
* Add data shredders for PII * First working shredder * Add more shredders * Add new shredders and download confirmation * tmp * PayPal, Stripe, banktransfer * Add icon to logs * Untested payment log shredders * Add waiting list shredder * First tests * Add tests for shredders * Improve templats, link to shredder * Test payment info shredders * More tests * Documentation * Fix enabled flag in payment provider overview * Fix minor issues
382 lines
14 KiB
Python
382 lines
14 KiB
Python
import warnings
|
|
from typing import Any, Callable, List, Tuple
|
|
|
|
import django.dispatch
|
|
from django.apps import apps
|
|
from django.conf import settings
|
|
from django.dispatch.dispatcher import NO_RECEIVERS
|
|
|
|
from .models import Event
|
|
|
|
app_cache = {}
|
|
|
|
|
|
def _populate_app_cache():
|
|
global app_cache
|
|
apps.check_apps_ready()
|
|
for ac in apps.app_configs.values():
|
|
app_cache[ac.name] = ac
|
|
|
|
|
|
class EventPluginSignal(django.dispatch.Signal):
|
|
"""
|
|
This is an extension to Django's built-in signals which differs in a way that it sends
|
|
out it's events only to receivers which belong to plugins that are enabled for the given
|
|
Event.
|
|
"""
|
|
|
|
def _is_active(self, sender, receiver):
|
|
if sender is None:
|
|
# Send to all events!
|
|
return True
|
|
|
|
# Find the Django application this belongs to
|
|
searchpath = receiver.__module__
|
|
core_module = any([searchpath.startswith(cm) for cm in settings.CORE_MODULES])
|
|
app = None
|
|
if not core_module:
|
|
while True:
|
|
app = app_cache.get(searchpath)
|
|
if "." not in searchpath or app:
|
|
break
|
|
searchpath, _ = searchpath.rsplit(".", 1)
|
|
|
|
# Only fire receivers from active plugins and core modules
|
|
if core_module or (sender and app and app.name in sender.get_plugins()):
|
|
if not hasattr(app, 'compatibility_errors') or not app.compatibility_errors:
|
|
return True
|
|
return False
|
|
|
|
def send(self, sender: Event, **named) -> List[Tuple[Callable, Any]]:
|
|
"""
|
|
Send signal from sender to all connected receivers that belong to
|
|
plugins enabled for the given Event.
|
|
|
|
sender is required to be an instance of ``pretix.base.models.Event``.
|
|
"""
|
|
if sender and not isinstance(sender, Event):
|
|
raise ValueError("Sender needs to be an event.")
|
|
|
|
responses = []
|
|
if not self.receivers or self.sender_receivers_cache.get(sender) is NO_RECEIVERS:
|
|
return responses
|
|
|
|
if not app_cache:
|
|
_populate_app_cache()
|
|
|
|
for receiver in self._live_receivers(sender):
|
|
if self._is_active(sender, receiver):
|
|
response = receiver(signal=self, sender=sender, **named)
|
|
responses.append((receiver, response))
|
|
return sorted(responses, key=lambda r: (receiver.__module__, receiver.__name__))
|
|
|
|
def send_chained(self, sender: Event, chain_kwarg_name, **named) -> List[Tuple[Callable, Any]]:
|
|
"""
|
|
Send signal from sender to all connected receivers. The return value of the first receiver
|
|
will be used as the keyword argument specified by ``chain_kwarg_name`` in the input to the
|
|
second receiver and so on. The return value of the last receiver is returned by this method.
|
|
|
|
sender is required to be an instance of ``pretix.base.models.Event``.
|
|
"""
|
|
if sender and not isinstance(sender, Event):
|
|
raise ValueError("Sender needs to be an event.")
|
|
|
|
response = named.get(chain_kwarg_name)
|
|
if not self.receivers or self.sender_receivers_cache.get(sender) is NO_RECEIVERS:
|
|
return response
|
|
|
|
if not app_cache:
|
|
_populate_app_cache()
|
|
|
|
for receiver in self._live_receivers(sender):
|
|
if self._is_active(sender, receiver):
|
|
named[chain_kwarg_name] = response
|
|
response = receiver(signal=self, sender=sender, **named)
|
|
return response
|
|
|
|
|
|
class DeprecatedSignal(django.dispatch.Signal):
|
|
|
|
def connect(self, receiver, sender=None, weak=True, dispatch_uid=None):
|
|
warnings.warn('This signal is deprecated and will soon be removed', stacklevel=3)
|
|
super().connect(receiver, sender=None, weak=True, dispatch_uid=None)
|
|
|
|
|
|
event_live_issues = EventPluginSignal(
|
|
providing_args=[]
|
|
)
|
|
"""
|
|
This signal is sent out to determine whether an event can be taken live. If you want to
|
|
prevent the event from going live, return a string that will be displayed to the user
|
|
as the error message. If you don't, your receiver should return ``None``.
|
|
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event.
|
|
"""
|
|
|
|
|
|
register_payment_providers = EventPluginSignal(
|
|
providing_args=[]
|
|
)
|
|
"""
|
|
This signal is sent out to get all known payment providers. Receivers should return a
|
|
subclass of pretix.base.payment.BasePaymentProvider or a list of these
|
|
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event.
|
|
"""
|
|
|
|
register_invoice_renderers = EventPluginSignal(
|
|
providing_args=[]
|
|
)
|
|
"""
|
|
This signal is sent out to get all known invoice renderers. Receivers should return a
|
|
subclass of pretix.base.invoice.BaseInvoiceRenderer or a list of these
|
|
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event.
|
|
"""
|
|
|
|
register_data_shredders = EventPluginSignal(
|
|
providing_args=[]
|
|
)
|
|
"""
|
|
This signal is sent out to get all known data shredders. Receivers should return a
|
|
subclass of pretix.base.shredder.BaseDataShredder or a list of these
|
|
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event.
|
|
"""
|
|
|
|
register_ticket_outputs = EventPluginSignal(
|
|
providing_args=[]
|
|
)
|
|
"""
|
|
This signal is sent out to get all known ticket outputs. Receivers should return a
|
|
subclass of pretix.base.ticketoutput.BaseTicketOutput
|
|
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event.
|
|
"""
|
|
|
|
register_notification_types = EventPluginSignal(
|
|
providing_args=[]
|
|
)
|
|
"""
|
|
This signal is sent out to get all known notification types. Receivers should return an
|
|
instance of a subclass of pretix.base.notifications.NotificationType or a list of such
|
|
instances.
|
|
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event,
|
|
however for this signal, the ``sender`` **may also be None** to allow creating the general
|
|
notification settings!
|
|
"""
|
|
|
|
register_data_exporters = EventPluginSignal(
|
|
providing_args=[]
|
|
)
|
|
"""
|
|
This signal is sent out to get all known data exporters. Receivers should return a
|
|
subclass of pretix.base.exporter.BaseExporter
|
|
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event.
|
|
"""
|
|
|
|
validate_cart = EventPluginSignal(
|
|
providing_args=["positions"]
|
|
)
|
|
"""
|
|
This signal is sent out before the user starts checkout. It includes an iterable
|
|
with the current CartPosition objects.
|
|
The response of receivers will be ignored, but you can raise a CartError with an
|
|
appropriate exception message.
|
|
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event.
|
|
"""
|
|
|
|
order_placed = EventPluginSignal(
|
|
providing_args=["order"]
|
|
)
|
|
"""
|
|
This signal is sent out every time an order is placed. The order object is given
|
|
as the first argument. This signal is *not* sent out if an order is created through
|
|
splitting an existing order, so you can not expect to see all orders by listening
|
|
to this signal.
|
|
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event.
|
|
"""
|
|
|
|
order_paid = EventPluginSignal(
|
|
providing_args=["order"]
|
|
)
|
|
"""
|
|
This signal is sent out every time an order is paid. The order object is given
|
|
as the first argument. This signal is *not* sent out if an order is marked as paid
|
|
because an already-paid order has been split.
|
|
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event.
|
|
"""
|
|
|
|
logentry_display = EventPluginSignal(
|
|
providing_args=["logentry"]
|
|
)
|
|
"""
|
|
To display an instance of the ``LogEntry`` model to a human user,
|
|
``pretix.base.signals.logentry_display`` will be sent out with a ``logentry`` argument.
|
|
|
|
The first received response that is not ``None`` will be used to display the log entry
|
|
to the user. The receivers are expected to return plain text.
|
|
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event.
|
|
"""
|
|
|
|
logentry_object_link = EventPluginSignal(
|
|
providing_args=["logentry"]
|
|
)
|
|
"""
|
|
To display the relationship of an instance of the ``LogEntry`` model to another model
|
|
to a human user, ``pretix.base.signals.logentry_object_link`` will be sent out with a
|
|
``logentry`` argument.
|
|
|
|
The first received response that is not ``None`` will be used to display the related object
|
|
to the user. The receivers are expected to return a HTML link. The internal implementation
|
|
builds the links like this::
|
|
|
|
a_text = _('Tax rule {val}')
|
|
a_map = {
|
|
'href': reverse('control:event.settings.tax.edit', kwargs={
|
|
'event': sender.slug,
|
|
'organizer': sender.organizer.slug,
|
|
'rule': logentry.content_object.id
|
|
}),
|
|
'val': escape(logentry.content_object.name),
|
|
}
|
|
a_map['val'] = '<a href="{href}">{val}</a>'.format_map(a_map)
|
|
return a_text.format_map(a_map)
|
|
|
|
Make sure that any user content in the HTML code you return is properly escaped!
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event.
|
|
"""
|
|
|
|
requiredaction_display = EventPluginSignal(
|
|
providing_args=["action", "request"]
|
|
)
|
|
"""
|
|
To display an instance of the ``RequiredAction`` model to a human user,
|
|
``pretix.base.signals.requiredaction_display`` will be sent out with a ``action`` argument.
|
|
You will also get the current ``request`` in a different argument.
|
|
|
|
The first received response that is not ``None`` will be used to display the log entry
|
|
to the user. The receivers are expected to return HTML code.
|
|
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event.
|
|
"""
|
|
|
|
event_copy_data = EventPluginSignal(
|
|
providing_args=["other", "tax_map", "category_map", "item_map", "question_map", "variation_map"]
|
|
)
|
|
"""
|
|
This signal is sent out when a new event is created as a clone of an existing event, i.e.
|
|
the settings from the older event are copied to the newer one. You can listen to this
|
|
signal to copy data or configuration stored within your plugin's models as well.
|
|
|
|
You don't need to copy data inside the general settings storage which is cloned automatically,
|
|
but you might need to modify that data.
|
|
|
|
The ``sender`` keyword argument will contain the event of the **new** event. The ``other``
|
|
keyword argument will contain the event to **copy from**. The keyword arguments
|
|
``tax_map``, ``category_map``, ``item_map``, ``question_map``, and ``variation_map`` contain
|
|
mappings from object IDs in the original event to objects in the new event of the respective
|
|
types.
|
|
"""
|
|
|
|
item_copy_data = EventPluginSignal(
|
|
providing_args=["source", "target"]
|
|
)
|
|
"""
|
|
This signal is sent out when a new product is created as a clone of an existing product, i.e.
|
|
the settings from the older product are copied to the newer one. You can listen to this
|
|
signal to copy data or configuration stored within your plugin's models as well.
|
|
|
|
The ``sender`` keyword argument will contain the event. The ``target`` will contain the item to
|
|
copy to, the ``source`` keyword argument will contain the product to **copy from**.
|
|
"""
|
|
|
|
periodic_task = django.dispatch.Signal()
|
|
"""
|
|
This is a regular django signal (no pretix event signal) that we send out every
|
|
time the periodic task cronjob runs. This interval is not sharply defined, it can
|
|
be everything between a minute and a day. The actions you perform should be
|
|
idempotent, i.e. it should not make a difference if this is sent out more often
|
|
than expected.
|
|
"""
|
|
|
|
register_global_settings = django.dispatch.Signal()
|
|
"""
|
|
All plugins that are installed may send fields for the global settings form, as
|
|
an OrderedDict of (setting name, form field).
|
|
"""
|
|
|
|
order_fee_calculation = EventPluginSignal(
|
|
providing_args=['positions', 'invoice_address', 'meta_info', 'total']
|
|
)
|
|
"""
|
|
This signals allows you to add fees to an order while it is being created. You are expected to
|
|
return a list of ``OrderFee`` objects that are not yet saved to the database
|
|
(because there is no order yet).
|
|
|
|
As with all plugin signals, the ``sender`` keyword argument will contain the event. A ``positions``
|
|
argument will contain the cart positions and ``invoice_address`` the invoice address (useful for
|
|
tax calculation). The argument ``meta_info`` contains the order's meta dictionary. The ``total``
|
|
keyword argument will contain the total cart sum without any fees. You should not rely on this
|
|
``total`` value for fee calculations as other fees might interfere.
|
|
"""
|
|
|
|
order_fee_type_name = EventPluginSignal(
|
|
providing_args=['request', 'fee']
|
|
)
|
|
"""
|
|
This signals allows you to return a human-readable description for a fee type based on the ``fee_type``
|
|
and ``internal_type`` attributes of the ``OrderFee`` model that you get as keyword arguments. You are
|
|
expected to return a string or None, if you don't know about this fee.
|
|
|
|
As with all plugin signals, the ``sender`` keyword argument will contain the event.
|
|
"""
|
|
|
|
allow_ticket_download = EventPluginSignal(
|
|
providing_args=['order']
|
|
)
|
|
"""
|
|
This signal is sent out to check if tickets for an order can be downloaded. If any receiver returns false,
|
|
a download will not be offered.
|
|
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event.
|
|
"""
|
|
|
|
email_filter = EventPluginSignal(
|
|
providing_args=['message', 'order']
|
|
)
|
|
"""
|
|
This signal allows you to implement a middleware-style filter on all outgoing emails. You are expected to
|
|
return a (possibly modified) copy of the message object passed to you.
|
|
|
|
As with all event-plugin signals, the ``sender`` keyword argument will contain the event.
|
|
The ``message`` argument will contain an ``EmailMultiAlternatives`` object.
|
|
If the email is associated with a specific order, the ``order`` argument will be passed as well, otherwise
|
|
it will be ``None``.
|
|
"""
|
|
|
|
|
|
layout_text_variables = EventPluginSignal()
|
|
"""
|
|
This signal is sent out to collect variables that can be used to display text in ticket-related PDF layouts.
|
|
Receivers are expected to return a dictionary with globally unique identifiers as keys and more
|
|
dictionaries as values that contain keys like in the following example::
|
|
|
|
return {
|
|
"product": {
|
|
"label": _("Product name"),
|
|
"editor_sample": _("Sample product"),
|
|
"evaluate": lambda orderposition, order, event: str(orderposition.item)
|
|
}
|
|
}
|
|
|
|
The evaluate member will be called with the order position, order and event as arguments. The event might
|
|
also be a subevent, if applicable.
|
|
"""
|