import uuid from django.contrib.postgres.fields import HStoreField from django.db import models from glitchtip.base_models import CreatedModel from glitchtip.model_utils import FromStringIntegerChoices from projects.tasks import update_event_project_hourly_statistic from user_reports.models import UserReport class AbstractEvent(CreatedModel): event_id = models.UUIDField(primary_key=True, default=uuid.uuid4, editable=False) data = models.JSONField(help_text="General event data that is searchable") timestamp = models.DateTimeField( blank=True, null=True, help_text="Date created as claimed by client it came from", ) class Meta: abstract = True def __str__(self): return self.event_id_hex @property def event_id_hex(self): """The public key without dashes""" if self.event_id: if isinstance(self.event_id, str): return self.event_id return self.event_id.hex class LogLevel(FromStringIntegerChoices): NOTSET = 0, "sample" DEBUG = 1, "debug" INFO = 2, "info" WARNING = 3, "warning" ERROR = 4, "error" FATAL = 5, "fatal" @classmethod def from_string(cls, string: str): result = super().from_string(string) if result: return result if string == "critical": return cls.FATAL if string == "log": return cls.INFO return cls.ERROR class Event(AbstractEvent): """ An individual event. An issue is a set of like-events. Most is stored in "data" but some fields benefit from being real relational data types such as dates and foreign keys """ issue = models.ForeignKey( "issues.Issue", on_delete=models.CASCADE, help_text="Sentry calls this a group", ) level = models.PositiveSmallIntegerField( choices=LogLevel.choices, default=LogLevel.ERROR ) errors = models.JSONField( null=True, blank=True, help_text="Event processing errors from event intake, including validation errors", ) tags = HStoreField(default=dict) release = models.ForeignKey( "releases.Release", blank=True, null=True, on_delete=models.SET_NULL ) class Meta: ordering = ["-created"] def save(self, *args, **kwargs): is_new = self._state.adding super().save(*args, **kwargs) if is_new: update_event_project_hourly_statistic( args=[self.issue.project_id, self.created], countdown=60 ) def event_json(self): """ OSS Sentry Compatible raw event JSON Effectively this combines data and relational data """ event = self.data event["event_id"] = self.event_id_hex event["project"] = self.issue.project_id event["level"] = self.get_level_display() event["tags"] = self.tags.items() if self.errors: event["errors"] = self.errors if self.timestamp: event["datetime"] = self.timestamp.isoformat().replace("+00:00", "Z") if self.release: event["release"] = self.release.version return event @property def context(self): return self.data.get("extra") @property def contexts(self): return self.data.get("contexts") @property def culprit(self): return self.data.get("culprit") @property def message(self): return self.data.get("message") @property def user(self): return self.data.get("user") @property def user_report(self): return UserReport.objects.filter(event_id=self.pk).first() def _build_context(self, context: list, base_line_no: int, is_pre: bool): context_length = len(context) result = [] for index, pre_context_line in enumerate(context): if is_pre: line_no = base_line_no - context_length + index else: line_no = base_line_no + 1 + index result.append( [ line_no, pre_context_line, ] ) return result @property def metadata(self): return self.data.get("metadata") @property def packages(self): return self.data.get("modules") @property def platform(self): return self.data.get("platform") @property def sdk(self): return self.data.get("sdk") @property def title(self): return self.data.get("title") @property def type(self): return self.data.get("type") def next(self, *args, **kwargs): try: return self.get_next_by_created(**kwargs) except Event.DoesNotExist: return None def previous(self, *args, **kwargs): """Get previous object by created date, pass filter kwargs""" try: return self.get_previous_by_created(**kwargs) except Event.DoesNotExist: return None