Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Move common functionality to destination base class #936

Open
wants to merge 20 commits into
base: master
Choose a base branch
from
Open
6 changes: 6 additions & 0 deletions changelog.d/+cleanup-destination-plugins.changed.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
Moved a lot of common infrastructure from our NotificationMedium subclasses to
the parent. This should make it easier to create more media of high quality.
Also, it should make it easier to use the plugins for validating the
settings-file elsewhere than just the API.

This might break 3rd party notification plugins.
144 changes: 101 additions & 43 deletions docs/integrations/notifications/writing-notification-plugins.rst
Original file line number Diff line number Diff line change
Expand Up @@ -28,29 +28,49 @@ implement the following:
Class constants
---------------

You need to set the constants `MEDIA_SLUG`, `MEDIA_NAME` and
`MEDIA_JSON_SCHEMA`.

The media name is the name of the service you want to send notifications by.
This is used only for display purposes so you might want to keep it short and
sweet. So for example `Email`, `SMS` or `MS Teams`.

The media slug is the slugified version of that, so the name simplified to only
contain lowercase letters, numbers, underscores and hyphens. Always have it
start with a letter, a-z. For example `email`, `sms` or `msteams`.

The media `json schema <https://json-schema.org/>`_ is a representation of how
a destination that will be used by this notification plugin should look like.
Such a destination should include all necessary information that is needed to
send notifications with your notification plugin. In case of SMS that is a
phone number or for MS Teams a webhook.
You must set the constants ``MEDIA_SLUG`, `MEDIA_NAME`` and
``MEDIA_JSON_SCHEMA``. If your plugin only takes or needs a single
configuration flag you should also set ``MEDIA_SETTINGS_KEY``.

MEDIA_NAME
The media name is the name of the service you want to send notifications by.
This is used only for display purposes so you might want to keep it short
and sweet. So for example ``"Email"``, ``"SMS"`` or ``"MS Teams"``.

MEDIA_SLUG
The media slug is the slugified version of that, so the name simplified to
only contain lowercase letters, numbers, underscores and hyphens. Always
have it start with a letter, a-z. For example ``"email"``, ``"sms"`` or
``"msteams"``.

MEDIA_JSON_SCHEMA
The media `json schema <https://json-schema.org/>`_ is a representation of
how a destination that will be used by this notification plugin should look
like, so that it is possible to autogenerate a form with javascript. It will
be accessible via the API. Such a destination should include all necessary
information that is needed to send notifications with your notification
plugin. In case of SMS that is a phone number or for MS Teams a webhook.

MEDIA_SETTINGS_KEY
The media settings key is the name of the most important key in the settings
JSON field. It is used to cut down on the amount of code you need to write
if there is only one piece of config you need to send the notification.
Among other things, it is used to check for duplicate entries, so in a way
it acts as the primary key for your plugin. For that reason, it must be
required in the json schema. For example for an email plugin this would be
"email_address".

Form
The ``forms.Form`` used to validate the settings-field.

Class methods for sending notifications
---------------------------------------

.. autoclass:: argus.notificationprofile.media.base.NotificationMedium
:members: send

This MUST be overridden.

The ``send`` method is the method that does the actual sending of the
notification. It gets the Argus event and a list of destinations as input and
returns a boolean indicating if the sending was successful.
Expand All @@ -62,35 +82,73 @@ The rest is very dependent on the notification medium and, if used, the Python
library. The given event can be used to extract relevant information that
should be included in the message that will be sent to each destination.

Class methods for destinations
------------------------------
Helper class methods
--------------------

.. autoclass:: argus.notificationprofile.media.base.NotificationMedium
:members: get_label, has_duplicate, raise_if_not_deletable, update, validate
:noindex:

Your implementation of ``get_label`` should show a reasonable representation
for a destination of that type that makes it easy to identify. For SMS that
would simply be the phone number.

The method ``has_duplicate`` will receive a QuerySet of destinations and a dict
of settings for a possible destination and should return True if a destination
with such settings exists in the given QuerySet.

``raise_if_not_deletable`` should check if a given destination can be deleted.
This is used in case some destinations are synced from an outside source and
should not be able to be deleted by a user. If that is the case a
``NotDeletableError`` should be raised. If not simply return None.

The method ``update`` only has to be implemented if the regular update method
of Django isn't sufficient. This can be the case if additional settings need to
be updated.

Finally the function ``validate`` makes sure that a destination with the given
settings can be updated or created. The function ``has_duplicate`` can be used
here to ensure that not two destinations with the same settings will be
created. Additionally the settings themselves should also be validated. For
example for SMS the given phone number will be checked. Django forms can be
helpful for validation. A ``ValidationError`` should be raised if the given
settings are invalid and the validated and cleaned data should be returned if
not.
With a little luck you might not need to override any of these.

clean
This method will do any additional cleaning beyond what is defined by the
defined ``Form``. Expects a valid form instance and optional
DestinationConfig instance, and returns the updated valid form instance. If
you have fields that shouldn't be set by a user, or values that need extra
conversion, you can do that in this method. Use the passed in instance if
you need to fall back to defaults. This method should not be used to
validate anything and thus should never raise a validation Exception.

get_label
Your implementation of ``get_label`` should show a reasonable representation
for a destination of that type that makes it easy to identify. For SMS that
would simply be the phone number. By default it shows the label stored in
the destination. If no label has been set, it uses MEDIA_SETTINGS_KEY to
look up the most important piece of information in the settings and uses
that directly. The included plugins need not override ``get_label`` for this
reason. If the label would be very long, for instance if the needed setting
is a very long url (40+ characters), you ought to write your own
``get_label``.

get_relevant_destination_settings
Used by ``send``. You should only need to override this if the key in
MEDIA_SETTINGS_KEY is insuffcient to look up the actual configuration of the
destinations of the type set by MEDIA_SLUG.

has_duplicate
The method ``has_duplicate`` will receive a QuerySet of destinations and
a dict of settings for a possible destination and should return True if
a destination with such settings exists in the given QuerySet. By default it
will use MEDIA_SETTINGS_KEY to lookup the most important piece of
information in the settings.

raise_if_not_deletable
``raise_if_not_deletable`` should check if a given destination can be
deleted. This is necessary in case the destination is in use by a profile,
or some destinations are synced from an outside source or otherwise
auto-generated, and should not be able to be deleted by a user. If that is
the case a ``NotDeletableError`` should be raised. If not simply return
None.

update
The method ``update`` only has to be implemented if the regular update
method is insufficient. This can be the case if there is more than one
key-value pair in settings that need to be updated.

validate
The function ``validate`` makes sure that a destination with the given
settings can be updated or created. It uses the ``validate_settings`` method
to validate the settings-field, a form (CommonDestinationConfigForm) to
validate the media and label-fields, and an optional DestinationConfig
instance for the sake of the ``clean``-method. The validated form is
returned if ok, otherwise a ``ValidationError`` should be raised. It is
unlikely that you will ever need to override this method.

validate_settings
This method validates the actual contents of the settings-field using the
``Form`` that is defined and an optional DestinationConfig instance for the
sake of the ``clean``-method. The function ``has_duplicate`` can be used
here to ensure that no two destinations with the same settings will be
created. A ``ValidationError`` should be raised if the given settings are
invalid, and the validated and cleaned data should be returned if not.
143 changes: 122 additions & 21 deletions src/argus/notificationprofile/media/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,62 +3,151 @@
from abc import ABC, abstractmethod
from typing import TYPE_CHECKING

from django import forms
from django.core.exceptions import ValidationError

from ..models import DestinationConfig

if TYPE_CHECKING:
from collections.abc import Iterable

from types import NoneType
from typing import Union
from typing import Union, Generator

from django.contrib.auth import get_user_model
from django.db.models.query import QuerySet

from argus.incident.models import Event
from ..models import DestinationConfig
from ..serializers import RequestDestinationConfigSerializer

User = get_user_model()


__all__ = ["NotificationMedium"]


class CommonDestinationConfigForm(forms.ModelForm):
class Meta:
model = DestinationConfig
fields = ["media", "label"]


class NotificationMedium(ABC):
"""
Must be defined by subclasses:

Class attributes:

- MEDIA_SLUG: short string id for the medium, lowercase
- MEDIA_NAME: human friendly id for the medium
- MEDIA_SETTINGS_KEY: the field in settings that is specific for this medium
hmpf marked this conversation as resolved.
Show resolved Hide resolved
- MEDIA_JSON_SCHEMA: A json-schema to describe the settings field to
javascript, used by the API
- Form: a django.forms.Form that validates the contents of the
settings-field. MEDIA_SETTINGS_KEY must be the field-name of a required
field in the form.

Class methods:

- send(event, destinations): How to send the given event to the given
destinations of type MEDIA_SLUG.
"""

error_messages = {
"readonly_medium": "Media cannot be updated, only settings.",
"readonly_user": "User cannot be changed, only settings.",
"settings_type": "Settings has to be a dictionary.",
"empty_settings": '"settings" cannot be empty',
}

class NotDeletableError(Exception):
"""
Custom exception class that is raised when a destination cannot be
deleted
"""

@classmethod
@abstractmethod
def validate(cls, instance: RequestDestinationConfigSerializer, dict: dict, user: User) -> dict:
def validate(
cls, data: dict, user: User, instance: DestinationConfig = None, exception_class=ValidationError
) -> CommonDestinationConfigForm:
if instance:
if data.get("media", "") != instance.media.slug:
johannaengland marked this conversation as resolved.
Show resolved Hide resolved
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This messes things up for me since the data I pass in (self.cleaned_data) includes an actual Media object, and not just the slug

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You're supposed to pass in the raw stuff. I'll finish moving the error handling then I'll take a look.

raise exception_class(cls.error_messages["readonly_media"])
form = CommonDestinationConfigForm(data, instance=instance)
if instance.user != user:
raise exception_class(cls.error_messages["readonly_user"])
else:
form = CommonDestinationConfigForm(data)

if not form.is_valid():
raise exception_class(form.errors)
Comment on lines +81 to +82
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

How can I associate each error with the field that gave the error when its raised like this? I would kinda like to get the form object even though it contains errors so I can show it with errors included

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If these checks are moved to the DestinationConfigForm.clean (I think) they will be converted into errors in form.errors. I'll do that.

I'll show you some debugging techniques for forms.


settings_form = cls.validate_settings(data, user, instance=instance)
form.cleaned_data["settings"] = settings_form.cleaned_data
Comment on lines +84 to +85
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

validate_settings returns a dict (form.cleaned_data), but it looks like this assumes it returns the form itself?
Got error here when trying to use this in HTMX version of destinations page:

File "/home/simon/repos/Argus/src/argus/notificationprofile/media/base.py", line 85, in validate
    form.cleaned_data["settings"] = settings_form.cleaned_data
AttributeError: 'dict' object has no attribute 'cleaned_data'

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I fix!

form.cleaned_data["user"] = user
return form

@classmethod
def validate_settings(
cls, data: dict, user: User, exception_class=ValidationError, instance: DestinationConfig = None
) -> dict:
"""
Validates the settings of destination and returns a dict with
validated and cleaned data
"""
pass
settings = data["settings"]
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

should this raise keyerror if settings does not exist? Or would settings = data.get("settings") be better?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If the key is not set or empty, validate_settings has been used wrong. The serializer skips validate(), and feeds the entire JSON to validate_settings directly. It doesn't have to do that though!

I don't have anything that uses validate() yet so now we are in Design Decision Needed territory. What do you think would be best?

We could remove the settings-lookup entirely, and change the serializer to feed just the contents of "settings" to validate_settings. Then on the html-side we could have multiple Django forms in a single HTML <form> and feed the request.POST querydict directly to validate_settings.

if not isinstance(settings, dict):
raise exception_class(cls.error_messages["settings_type"])
if not settings:
raise exception_class(cls.error_messages["empty_settings"])

form = cls.Form(settings)
if not form.is_valid():
raise exception_class(form.errors)

form = cls.clean(form, instance)

if cls.has_duplicate(user.destinations, settings):
raise exception_class(cls.MEDIA_SETTINGS_KEY, f"{cls.MEDIA_NAME} already exists")

return form.cleaned_data

@classmethod
def clean(cls, form: forms.Form, instance: DestinationConfig = None) -> forms.Form:
"""Can change the cleaned data of a valid form"""
return form

@classmethod
@abstractmethod
def has_duplicate(cls, queryset: QuerySet, settings: dict) -> bool:
"""
Returns True if a destination with the given settings already exists
in the given queryset
"""
pass
key = f"settings__{cls.MEDIA_SETTINGS_KEY}"
value = settings[cls.MEDIA_SETTINGS_KEY]
return queryset.filter(media_id=cls.MEDIA_SLUG, **{key: value}).exists()

@staticmethod
@abstractmethod
def get_label(destination: DestinationConfig) -> str:
@classmethod
def get_label(cls, destination: DestinationConfig) -> str:
"""
Returns a descriptive label for this destination.
Returns a descriptive label for this destination if none is stored
"""
pass
return destination.label if destination.label else destination.settings.get(cls.MEDIA_SETTINGS_KEY)

# No querysets beyond this point!

@classmethod
def _get_relevant_destinations(cls, destinations: Iterable[DestinationConfig]) -> Generator[DestinationConfig]:
return (destination for destination in destinations if destination.media_id == cls.MEDIA_SLUG)

@classmethod
def get_relevant_addresses(cls, destinations: Iterable[DestinationConfig]) -> set[DestinationConfig]:
def get_relevant_destination_settings(cls, destinations: Iterable[DestinationConfig]) -> set[str]:
"""Returns a set of addresses the message should be sent to"""
pass
destinations = [
destination.settings[cls.MEDIA_SETTINGS_KEY] for destination in cls._get_relevant_destinations(destinations)
]
return set(destinations)

get_relevant_addresses = get_relevant_destination_settings

@classmethod
@abstractmethod
Expand All @@ -79,11 +168,23 @@ def raise_if_not_deletable(cls, destination: DestinationConfig) -> NoneType:
f"Cannot delete this destination since it is in use in the notification profile(s): {profiles}."
)

@staticmethod
def update(destination: DestinationConfig, validated_data: dict) -> Union[DestinationConfig, NoneType]:
@classmethod
def _update_destination(cls, destination: DestinationConfig, validated_data: dict) -> DestinationConfig:
# adapted from rest_framework.serializers.ModelSerializer.update
# DestinationConfig MUST NOT have any m2m-relations so this is safe

for attr, value in validated_data.items():
setattr(destination, attr, value)

return destination

@classmethod
def update(cls, destination: DestinationConfig, validated_data: dict) -> Union[DestinationConfig, NoneType]:
"""
Updates a destination in case the normal update function is not
sufficient and returns the updated destination in that case,
returns None otherwise
Updates a destination

Override in case the normal update function is not sufficient
"""
return None
instance = cls._update_destination(destination, validated_data)
instance.save()
return instance
Loading
Loading