2020-12-20 21:04:29 +00:00
|
|
|
"""authentik events models"""
|
2019-12-05 15:14:08 +00:00
|
|
|
from inspect import getmodule, stack
|
2021-01-12 20:48:16 +00:00
|
|
|
from smtplib import SMTPException
|
2020-12-20 21:04:29 +00:00
|
|
|
from typing import Optional, Union
|
|
|
|
from uuid import uuid4
|
2019-12-05 15:14:08 +00:00
|
|
|
|
2018-11-23 16:05:41 +00:00
|
|
|
from django.conf import settings
|
|
|
|
from django.db import models
|
2019-12-05 15:14:08 +00:00
|
|
|
from django.http import HttpRequest
|
2018-12-10 12:47:51 +00:00
|
|
|
from django.utils.translation import gettext as _
|
2021-02-12 08:47:37 +00:00
|
|
|
from geoip2.errors import GeoIP2Error
|
2021-01-12 20:48:16 +00:00
|
|
|
from requests import RequestException, post
|
2021-01-01 14:39:43 +00:00
|
|
|
from structlog.stdlib import get_logger
|
2018-11-23 16:05:41 +00:00
|
|
|
|
2021-01-11 17:43:59 +00:00
|
|
|
from authentik import __version__
|
2020-12-05 21:08:42 +00:00
|
|
|
from authentik.core.middleware import (
|
2020-09-19 20:49:40 +00:00
|
|
|
SESSION_IMPERSONATE_ORIGINAL_USER,
|
|
|
|
SESSION_IMPERSONATE_USER,
|
|
|
|
)
|
2021-01-11 17:43:59 +00:00
|
|
|
from authentik.core.models import Group, User
|
2021-02-12 08:47:37 +00:00
|
|
|
from authentik.events.geo import GEOIP_READER
|
2020-12-20 21:04:29 +00:00
|
|
|
from authentik.events.utils import cleanse_dict, get_user, sanitize_dict
|
2021-01-12 20:48:16 +00:00
|
|
|
from authentik.lib.sentry import SentryIgnoredException
|
2020-12-05 21:08:42 +00:00
|
|
|
from authentik.lib.utils.http import get_client_ip
|
2021-01-11 17:43:59 +00:00
|
|
|
from authentik.policies.models import PolicyBindingModel
|
|
|
|
from authentik.stages.email.utils import TemplateEmailMessage
|
2018-11-23 16:05:41 +00:00
|
|
|
|
2020-12-20 21:04:29 +00:00
|
|
|
LOGGER = get_logger("authentik.events")
|
2019-12-31 12:33:07 +00:00
|
|
|
|
|
|
|
|
2021-01-12 20:48:16 +00:00
|
|
|
class NotificationTransportError(SentryIgnoredException):
|
|
|
|
"""Error raised when a notification fails to be delivered"""
|
|
|
|
|
|
|
|
|
2020-09-21 18:16:14 +00:00
|
|
|
class EventAction(models.TextChoices):
|
2020-12-20 21:04:29 +00:00
|
|
|
"""All possible actions to save into the events log"""
|
2019-12-05 15:14:08 +00:00
|
|
|
|
2019-12-31 11:51:16 +00:00
|
|
|
LOGIN = "login"
|
|
|
|
LOGIN_FAILED = "login_failed"
|
|
|
|
LOGOUT = "logout"
|
2020-09-21 18:16:14 +00:00
|
|
|
|
2020-10-05 21:43:56 +00:00
|
|
|
USER_WRITE = "user_write"
|
2019-12-31 11:51:16 +00:00
|
|
|
SUSPICIOUS_REQUEST = "suspicious_request"
|
2020-09-21 18:16:14 +00:00
|
|
|
PASSWORD_SET = "password_set" # noqa # nosec
|
|
|
|
|
2021-02-09 17:18:36 +00:00
|
|
|
SECRET_VIEW = "secret_view" # noqa # nosec
|
2020-10-17 20:26:18 +00:00
|
|
|
|
2019-12-31 11:51:16 +00:00
|
|
|
INVITE_USED = "invitation_used"
|
2020-09-21 18:16:14 +00:00
|
|
|
|
2020-10-05 21:43:56 +00:00
|
|
|
AUTHORIZE_APPLICATION = "authorize_application"
|
2020-09-21 18:30:30 +00:00
|
|
|
SOURCE_LINKED = "source_linked"
|
|
|
|
|
2020-09-18 21:39:37 +00:00
|
|
|
IMPERSONATION_STARTED = "impersonation_started"
|
|
|
|
IMPERSONATION_ENDED = "impersonation_ended"
|
2019-12-05 15:14:08 +00:00
|
|
|
|
2020-12-20 21:04:29 +00:00
|
|
|
POLICY_EXECUTION = "policy_execution"
|
|
|
|
POLICY_EXCEPTION = "policy_exception"
|
|
|
|
PROPERTY_MAPPING_EXCEPTION = "property_mapping_exception"
|
|
|
|
|
2021-01-18 08:34:48 +00:00
|
|
|
SYSTEM_TASK_EXECUTION = "system_task_execution"
|
|
|
|
SYSTEM_TASK_EXCEPTION = "system_task_exception"
|
|
|
|
|
2020-12-27 12:11:38 +00:00
|
|
|
CONFIGURATION_ERROR = "configuration_error"
|
|
|
|
|
2020-09-21 18:16:14 +00:00
|
|
|
MODEL_CREATED = "model_created"
|
|
|
|
MODEL_UPDATED = "model_updated"
|
|
|
|
MODEL_DELETED = "model_deleted"
|
|
|
|
|
2020-12-20 21:04:29 +00:00
|
|
|
UPDATE_AVAILABLE = "update_available"
|
|
|
|
|
2020-09-21 18:16:14 +00:00
|
|
|
CUSTOM_PREFIX = "custom_"
|
2019-12-05 15:14:08 +00:00
|
|
|
|
|
|
|
|
2020-05-20 07:17:06 +00:00
|
|
|
class Event(models.Model):
|
2020-12-20 21:04:29 +00:00
|
|
|
"""An individual Audit/Metrics/Notification/Error Event"""
|
2018-11-23 16:05:41 +00:00
|
|
|
|
2020-05-20 07:17:06 +00:00
|
|
|
event_uuid = models.UUIDField(primary_key=True, editable=False, default=uuid4)
|
2020-09-21 11:20:50 +00:00
|
|
|
user = models.JSONField(default=dict)
|
2020-09-21 18:16:14 +00:00
|
|
|
action = models.TextField(choices=EventAction.choices)
|
2018-11-23 16:05:41 +00:00
|
|
|
app = models.TextField()
|
2020-08-15 19:04:22 +00:00
|
|
|
context = models.JSONField(default=dict, blank=True)
|
2019-12-05 15:14:08 +00:00
|
|
|
client_ip = models.GenericIPAddressField(null=True)
|
2018-12-13 17:01:45 +00:00
|
|
|
created = models.DateTimeField(auto_now_add=True)
|
2018-12-10 12:47:51 +00:00
|
|
|
|
|
|
|
@staticmethod
|
2019-12-05 15:14:08 +00:00
|
|
|
def _get_app_from_request(request: HttpRequest) -> str:
|
|
|
|
if not isinstance(request, HttpRequest):
|
|
|
|
return ""
|
|
|
|
return request.resolver_match.app_name
|
|
|
|
|
|
|
|
@staticmethod
|
2019-12-31 11:51:16 +00:00
|
|
|
def new(
|
2020-09-21 18:16:14 +00:00
|
|
|
action: Union[str, EventAction],
|
2019-12-31 11:51:16 +00:00
|
|
|
app: Optional[str] = None,
|
|
|
|
_inspect_offset: int = 1,
|
|
|
|
**kwargs,
|
|
|
|
) -> "Event":
|
2019-12-05 15:14:08 +00:00
|
|
|
"""Create new Event instance from arguments. Instance is NOT saved."""
|
|
|
|
if not isinstance(action, EventAction):
|
2020-09-21 18:16:14 +00:00
|
|
|
action = EventAction.CUSTOM_PREFIX + action
|
2019-12-05 15:14:08 +00:00
|
|
|
if not app:
|
|
|
|
app = getmodule(stack()[_inspect_offset][0]).__name__
|
2020-06-29 17:13:07 +00:00
|
|
|
cleaned_kwargs = cleanse_dict(sanitize_dict(kwargs))
|
2020-09-21 18:16:14 +00:00
|
|
|
event = Event(action=action, app=app, context=cleaned_kwargs)
|
2019-12-05 15:14:08 +00:00
|
|
|
return event
|
|
|
|
|
2020-12-20 21:04:29 +00:00
|
|
|
def set_user(self, user: User) -> "Event":
|
|
|
|
"""Set `.user` based on user, ensuring the correct attributes are copied.
|
|
|
|
This should only be used when self.from_http is *not* used."""
|
|
|
|
self.user = get_user(user)
|
|
|
|
return self
|
|
|
|
|
2019-12-31 11:51:16 +00:00
|
|
|
def from_http(
|
|
|
|
self, request: HttpRequest, user: Optional[settings.AUTH_USER_MODEL] = None
|
|
|
|
) -> "Event":
|
2019-12-05 15:14:08 +00:00
|
|
|
"""Add data from a Django-HttpRequest, allowing the creation of
|
|
|
|
Events independently from requests.
|
|
|
|
`user` arguments optionally overrides user from requests."""
|
2019-12-31 11:51:16 +00:00
|
|
|
if hasattr(request, "user"):
|
2021-01-11 17:43:59 +00:00
|
|
|
original_user = None
|
|
|
|
if hasattr(request, "session"):
|
|
|
|
original_user = request.session.get(
|
|
|
|
SESSION_IMPERSONATE_ORIGINAL_USER, None
|
|
|
|
)
|
|
|
|
self.user = get_user(request.user, original_user)
|
2019-12-05 15:14:08 +00:00
|
|
|
if user:
|
2020-09-21 11:20:50 +00:00
|
|
|
self.user = get_user(user)
|
2020-09-18 21:39:37 +00:00
|
|
|
# Check if we're currently impersonating, and add that user
|
|
|
|
if hasattr(request, "session"):
|
|
|
|
if SESSION_IMPERSONATE_ORIGINAL_USER in request.session:
|
2020-09-21 11:20:50 +00:00
|
|
|
self.user = get_user(request.session[SESSION_IMPERSONATE_ORIGINAL_USER])
|
|
|
|
self.user["on_behalf_of"] = get_user(
|
2020-09-19 20:49:40 +00:00
|
|
|
request.session[SESSION_IMPERSONATE_USER]
|
2020-09-18 21:39:37 +00:00
|
|
|
)
|
2019-12-05 15:14:08 +00:00
|
|
|
# User 255.255.255.255 as fallback if IP cannot be determined
|
2019-12-31 11:51:16 +00:00
|
|
|
self.client_ip = get_client_ip(request) or "255.255.255.255"
|
2021-02-12 08:47:37 +00:00
|
|
|
# Apply GeoIP Data, when enabled
|
|
|
|
self.with_geoip()
|
2019-12-05 15:14:08 +00:00
|
|
|
# If there's no app set, we get it from the requests too
|
|
|
|
if not self.app:
|
|
|
|
self.app = Event._get_app_from_request(request)
|
|
|
|
self.save()
|
|
|
|
return self
|
2018-11-23 16:05:41 +00:00
|
|
|
|
2021-02-12 08:47:37 +00:00
|
|
|
def with_geoip(self):
|
|
|
|
"""Apply GeoIP Data, when enabled"""
|
2021-02-12 09:43:00 +00:00
|
|
|
if not GEOIP_READER:
|
|
|
|
return
|
2021-02-12 08:47:37 +00:00
|
|
|
try:
|
|
|
|
response = GEOIP_READER.city(self.client_ip)
|
|
|
|
self.context["geo"] = {
|
|
|
|
"continent": response.continent.code,
|
|
|
|
"country": response.country.iso_code,
|
|
|
|
"lat": response.location.latitude,
|
|
|
|
"long": response.location.longitude,
|
|
|
|
}
|
|
|
|
if response.city.name:
|
|
|
|
self.context["geo"]["city"] = response.city.name
|
2021-03-18 14:16:43 +00:00
|
|
|
except GeoIP2Error as exc:
|
|
|
|
LOGGER.warning("Failed to add geoIP Data to event", exc=exc)
|
2021-02-12 08:47:37 +00:00
|
|
|
|
2018-11-23 16:05:41 +00:00
|
|
|
def save(self, *args, **kwargs):
|
2021-02-12 08:47:37 +00:00
|
|
|
if self._state.adding:
|
|
|
|
LOGGER.debug(
|
|
|
|
"Created Event",
|
|
|
|
action=self.action,
|
|
|
|
context=self.context,
|
|
|
|
client_ip=self.client_ip,
|
|
|
|
user=self.user,
|
|
|
|
)
|
2019-12-05 15:14:08 +00:00
|
|
|
return super().save(*args, **kwargs)
|
2018-12-10 12:47:51 +00:00
|
|
|
|
2021-01-11 17:43:59 +00:00
|
|
|
@property
|
|
|
|
def summary(self) -> str:
|
|
|
|
"""Return a summary of this event."""
|
|
|
|
if "message" in self.context:
|
|
|
|
return self.context["message"]
|
|
|
|
return f"{self.action}: {self.context}"
|
|
|
|
|
|
|
|
def __str__(self) -> str:
|
|
|
|
return f"<Event action={self.action} user={self.user} context={self.context}>"
|
|
|
|
|
2018-12-10 12:47:51 +00:00
|
|
|
class Meta:
|
|
|
|
|
2020-12-20 21:04:29 +00:00
|
|
|
verbose_name = _("Event")
|
|
|
|
verbose_name_plural = _("Events")
|
2021-01-11 17:43:59 +00:00
|
|
|
|
|
|
|
|
|
|
|
class TransportMode(models.TextChoices):
|
|
|
|
"""Modes that a notification transport can send a notification"""
|
|
|
|
|
|
|
|
WEBHOOK = "webhook", _("Generic Webhook")
|
|
|
|
WEBHOOK_SLACK = "webhook_slack", _("Slack Webhook (Slack/Discord)")
|
|
|
|
EMAIL = "email", _("Email")
|
|
|
|
|
|
|
|
|
|
|
|
class NotificationTransport(models.Model):
|
2021-01-15 15:23:27 +00:00
|
|
|
"""Action which is executed when a Rule matches"""
|
2021-01-11 17:43:59 +00:00
|
|
|
|
|
|
|
uuid = models.UUIDField(primary_key=True, editable=False, default=uuid4)
|
|
|
|
|
|
|
|
name = models.TextField(unique=True)
|
|
|
|
mode = models.TextField(choices=TransportMode.choices)
|
|
|
|
|
|
|
|
webhook_url = models.TextField(blank=True)
|
2021-02-02 18:34:55 +00:00
|
|
|
send_once = models.BooleanField(
|
|
|
|
default=False,
|
|
|
|
help_text=_(
|
|
|
|
"Only send notification once, for example when sending a webhook into a chat channel."
|
|
|
|
),
|
|
|
|
)
|
2021-01-11 17:43:59 +00:00
|
|
|
|
|
|
|
def send(self, notification: "Notification") -> list[str]:
|
|
|
|
"""Send notification to user, called from async task"""
|
|
|
|
if self.mode == TransportMode.WEBHOOK:
|
|
|
|
return self.send_webhook(notification)
|
|
|
|
if self.mode == TransportMode.WEBHOOK_SLACK:
|
|
|
|
return self.send_webhook_slack(notification)
|
|
|
|
if self.mode == TransportMode.EMAIL:
|
|
|
|
return self.send_email(notification)
|
|
|
|
raise ValueError(f"Invalid mode {self.mode} set")
|
|
|
|
|
|
|
|
def send_webhook(self, notification: "Notification") -> list[str]:
|
|
|
|
"""Send notification to generic webhook"""
|
2021-01-12 20:48:16 +00:00
|
|
|
try:
|
|
|
|
response = post(
|
|
|
|
self.webhook_url,
|
|
|
|
json={
|
|
|
|
"body": notification.body,
|
|
|
|
"severity": notification.severity,
|
2021-01-12 22:28:17 +00:00
|
|
|
"user_email": notification.user.email,
|
|
|
|
"user_username": notification.user.username,
|
2021-01-12 20:48:16 +00:00
|
|
|
},
|
|
|
|
)
|
|
|
|
response.raise_for_status()
|
|
|
|
except RequestException as exc:
|
2021-01-14 16:40:19 +00:00
|
|
|
raise NotificationTransportError(exc.response.text) from exc
|
2021-01-11 17:43:59 +00:00
|
|
|
return [
|
|
|
|
response.status_code,
|
|
|
|
response.text,
|
|
|
|
]
|
|
|
|
|
|
|
|
def send_webhook_slack(self, notification: "Notification") -> list[str]:
|
|
|
|
"""Send notification to slack or slack-compatible endpoints"""
|
2021-01-14 16:40:19 +00:00
|
|
|
fields = [
|
|
|
|
{
|
|
|
|
"title": _("Severity"),
|
|
|
|
"value": notification.severity,
|
|
|
|
"short": True,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"title": _("Dispatched for user"),
|
|
|
|
"value": str(notification.user),
|
|
|
|
"short": True,
|
|
|
|
},
|
|
|
|
]
|
|
|
|
if notification.event:
|
|
|
|
for key, value in notification.event.context.items():
|
|
|
|
if not isinstance(value, str):
|
|
|
|
continue
|
|
|
|
# https://birdie0.github.io/discord-webhooks-guide/other/field_limits.html
|
|
|
|
if len(fields) >= 25:
|
|
|
|
continue
|
|
|
|
fields.append({"title": key[:256], "value": value[:1024]})
|
2021-01-11 17:43:59 +00:00
|
|
|
body = {
|
|
|
|
"username": "authentik",
|
|
|
|
"icon_url": "https://goauthentik.io/img/icon.png",
|
|
|
|
"attachments": [
|
|
|
|
{
|
|
|
|
"author_name": "authentik",
|
|
|
|
"author_link": "https://goauthentik.io",
|
|
|
|
"author_icon": "https://goauthentik.io/img/icon.png",
|
|
|
|
"title": notification.body,
|
|
|
|
"color": "#fd4b2d",
|
2021-01-14 16:40:19 +00:00
|
|
|
"fields": fields,
|
2021-01-11 17:43:59 +00:00
|
|
|
"footer": f"authentik v{__version__}",
|
|
|
|
}
|
|
|
|
],
|
|
|
|
}
|
|
|
|
if notification.event:
|
|
|
|
body["attachments"][0]["title"] = notification.event.action
|
2021-01-12 20:48:16 +00:00
|
|
|
try:
|
|
|
|
response = post(self.webhook_url, json=body)
|
|
|
|
response.raise_for_status()
|
|
|
|
except RequestException as exc:
|
2021-01-14 16:40:19 +00:00
|
|
|
raise NotificationTransportError(exc.response.text) from exc
|
2021-01-11 17:43:59 +00:00
|
|
|
return [
|
|
|
|
response.status_code,
|
|
|
|
response.text,
|
|
|
|
]
|
|
|
|
|
|
|
|
def send_email(self, notification: "Notification") -> list[str]:
|
|
|
|
"""Send notification via global email configuration"""
|
2021-02-04 20:44:06 +00:00
|
|
|
subject = "authentik Notification: "
|
|
|
|
key_value = {}
|
|
|
|
if notification.event:
|
|
|
|
subject += notification.event.action
|
|
|
|
for key, value in notification.event.context.items():
|
|
|
|
if not isinstance(value, str):
|
|
|
|
continue
|
|
|
|
key_value[key] = value
|
|
|
|
else:
|
|
|
|
subject += notification.body[:75]
|
2021-01-11 17:43:59 +00:00
|
|
|
mail = TemplateEmailMessage(
|
2021-02-04 20:44:06 +00:00
|
|
|
subject=subject,
|
2021-01-27 12:22:43 +00:00
|
|
|
template_name="email/generic.html",
|
2021-01-11 17:43:59 +00:00
|
|
|
to=[notification.user.email],
|
|
|
|
template_context={
|
2021-02-04 20:44:06 +00:00
|
|
|
"title": subject,
|
2021-01-11 17:43:59 +00:00
|
|
|
"body": notification.body,
|
2021-02-04 20:44:06 +00:00
|
|
|
"key_value": key_value,
|
2021-01-11 17:43:59 +00:00
|
|
|
},
|
|
|
|
)
|
|
|
|
# Email is sent directly here, as the call to send() should have been from a task.
|
2021-01-12 20:48:16 +00:00
|
|
|
try:
|
2021-01-18 08:34:48 +00:00
|
|
|
from authentik.stages.email.tasks import send_mail
|
|
|
|
|
2021-01-12 20:48:16 +00:00
|
|
|
# pyright: reportGeneralTypeIssues=false
|
|
|
|
return send_mail(mail.__dict__) # pylint: disable=no-value-for-parameter
|
2021-01-17 22:31:58 +00:00
|
|
|
except (SMTPException, ConnectionError, OSError) as exc:
|
2021-01-12 20:48:16 +00:00
|
|
|
raise NotificationTransportError from exc
|
2021-01-11 17:43:59 +00:00
|
|
|
|
2021-01-12 21:03:33 +00:00
|
|
|
def __str__(self) -> str:
|
|
|
|
return f"Notification Transport {self.name}"
|
|
|
|
|
2021-01-11 17:43:59 +00:00
|
|
|
class Meta:
|
|
|
|
|
|
|
|
verbose_name = _("Notification Transport")
|
|
|
|
verbose_name_plural = _("Notification Transports")
|
|
|
|
|
|
|
|
|
|
|
|
class NotificationSeverity(models.TextChoices):
|
|
|
|
"""Severity images that a notification can have"""
|
|
|
|
|
|
|
|
NOTICE = "notice", _("Notice")
|
|
|
|
WARNING = "warning", _("Warning")
|
|
|
|
ALERT = "alert", _("Alert")
|
|
|
|
|
|
|
|
|
|
|
|
class Notification(models.Model):
|
|
|
|
"""Event Notification"""
|
|
|
|
|
|
|
|
uuid = models.UUIDField(primary_key=True, editable=False, default=uuid4)
|
|
|
|
severity = models.TextField(choices=NotificationSeverity.choices)
|
|
|
|
body = models.TextField()
|
|
|
|
created = models.DateTimeField(auto_now_add=True)
|
|
|
|
event = models.ForeignKey(Event, on_delete=models.SET_NULL, null=True, blank=True)
|
|
|
|
seen = models.BooleanField(default=False)
|
|
|
|
user = models.ForeignKey(User, on_delete=models.CASCADE)
|
|
|
|
|
|
|
|
def __str__(self) -> str:
|
|
|
|
body_trunc = (self.body[:75] + "..") if len(self.body) > 75 else self.body
|
|
|
|
return f"Notification for user {self.user}: {body_trunc}"
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
|
|
|
verbose_name = _("Notification")
|
|
|
|
verbose_name_plural = _("Notifications")
|
|
|
|
|
|
|
|
|
2021-01-15 15:23:27 +00:00
|
|
|
class NotificationRule(PolicyBindingModel):
|
2021-01-11 17:43:59 +00:00
|
|
|
"""Decide when to create a Notification based on policies attached to this object."""
|
|
|
|
|
|
|
|
name = models.TextField(unique=True)
|
|
|
|
transports = models.ManyToManyField(
|
|
|
|
NotificationTransport,
|
|
|
|
help_text=_(
|
|
|
|
(
|
|
|
|
"Select which transports should be used to notify the user. If none are "
|
|
|
|
"selected, the notification will only be shown in the authentik UI."
|
|
|
|
)
|
|
|
|
),
|
|
|
|
)
|
|
|
|
severity = models.TextField(
|
|
|
|
choices=NotificationSeverity.choices,
|
|
|
|
default=NotificationSeverity.NOTICE,
|
|
|
|
help_text=_(
|
|
|
|
"Controls which severity level the created notifications will have."
|
|
|
|
),
|
|
|
|
)
|
|
|
|
group = models.ForeignKey(
|
|
|
|
Group,
|
|
|
|
help_text=_(
|
|
|
|
(
|
|
|
|
"Define which group of users this notification should be sent and shown to. "
|
|
|
|
"If left empty, Notification won't ben sent."
|
|
|
|
)
|
|
|
|
),
|
|
|
|
null=True,
|
|
|
|
blank=True,
|
|
|
|
on_delete=models.SET_NULL,
|
|
|
|
)
|
|
|
|
|
2021-01-12 21:03:33 +00:00
|
|
|
def __str__(self) -> str:
|
2021-01-15 15:23:27 +00:00
|
|
|
return f"Notification Rule {self.name}"
|
2021-01-12 21:03:33 +00:00
|
|
|
|
2021-01-11 17:43:59 +00:00
|
|
|
class Meta:
|
|
|
|
|
2021-01-15 15:23:27 +00:00
|
|
|
verbose_name = _("Notification Rule")
|
|
|
|
verbose_name_plural = _("Notification Rules")
|