Source code for pretalx.common.models.log

# SPDX-FileCopyrightText: 2017-present Tobias Kunze
# SPDX-License-Identifier: AGPL-3.0-only WITH LicenseRef-Pretalx-AGPL-3.0-Terms

import logging

from django.contrib.contenttypes.fields import GenericForeignKey
from django.contrib.contenttypes.models import ContentType
from django.db import models
from django.utils.functional import cached_property
from django_scopes import ScopedManager

from pretalx.common.signals import activitylog_display, activitylog_object_link


[docs] class ActivityLog(models.Model): """This model logs actions within an event. It is **not** designed to provide a complete or reliable audit trail. :param is_orga_action: True, if the logged action was performed by a privileged user. """ event = models.ForeignKey( to="event.Event", on_delete=models.PROTECT, related_name="log_entries", null=True, blank=True, ) person = models.ForeignKey( to="person.User", on_delete=models.PROTECT, related_name="log_entries", null=True, blank=True, ) content_type = models.ForeignKey(to=ContentType, on_delete=models.CASCADE) object_id = models.PositiveIntegerField(db_index=True) content_object = GenericForeignKey("content_type", "object_id") timestamp = models.DateTimeField(auto_now_add=True, db_index=True) action_type = models.CharField(max_length=200) data = models.JSONField(null=True, blank=True, default=dict) is_orga_action = models.BooleanField(default=False) objects = ScopedManager(event="event") class Meta: ordering = ("-timestamp",) def __str__(self): """Custom __str__ to help with debugging.""" event = getattr(self.event, "slug", "None") person = getattr(self.person, "name", "None") return f"ActivityLog(event={event}, person={person}, content_object={self.content_object}, action_type={self.action_type})" @cached_property def json_data(self): # Kept for backwards compatibility, as well as to avoid None-checks return self.data or {} @cached_property def display(self) -> str: for _receiver, response in activitylog_display.send( self.event, activitylog=self ): if response: return response logger = logging.getLogger(__name__) logger.warning('Unknown log action "%s".', self.action_type) return self.action_type @cached_property def display_object(self) -> str: """Returns a link (formatted HTML) to the object in question.""" try: if not self.content_object: return "" except ( AttributeError ): # pragma: no cover — stale ContentType whose model class was removed return "" for _receiver, response in activitylog_object_link.send( sender=self.event, activitylog=self ): if response: return response return "" @cached_property def changes(self): from pretalx.common.log import ( # noqa: PLC0415 -- thin method resolve_log_changes, ) return resolve_log_changes(self)