stages/email: Implement MonitoredTask, but only for failed emails

This commit is contained in:
Jens Langhammer 2020-10-16 14:31:01 +02:00
parent 4ac87d8739
commit 8fedd9ec07
2 changed files with 48 additions and 24 deletions

View file

@ -69,10 +69,14 @@ class TaskInfo:
class MonitoredTask(Task): class MonitoredTask(Task):
"""Task which can save its state to the cache""" """Task which can save its state to the cache"""
# For tasks that should only be listed if they failed, set this to False
save_on_success: bool
_result: TaskResult _result: TaskResult
def __init__(self, *args, **kwargs) -> None: def __init__(self, *args, **kwargs) -> None:
super().__init__(*args, **kwargs) super().__init__(*args, **kwargs)
self.save_on_success = True
self._result = TaskResult(status=TaskResultStatus.ERROR, messages=[]) self._result = TaskResult(status=TaskResultStatus.ERROR, messages=[])
def set_status(self, result: TaskResult): def set_status(self, result: TaskResult):
@ -83,16 +87,17 @@ class MonitoredTask(Task):
def after_return( def after_return(
self, status, retval, task_id, args: List[Any], kwargs: Dict[str, Any], einfo self, status, retval, task_id, args: List[Any], kwargs: Dict[str, Any], einfo
): ):
TaskInfo( if self.save_on_success:
task_name=self.__name__, TaskInfo(
task_description=self.__doc__, task_name=self.__name__,
finish_timestamp=datetime.now(), task_description=self.__doc__,
result=self._result, finish_timestamp=datetime.now(),
task_call_module=self.__module__, result=self._result,
task_call_func=self.__name__, task_call_module=self.__module__,
task_call_args=args, task_call_func=self.__name__,
task_call_kwargs=kwargs, task_call_args=args,
).save() task_call_kwargs=kwargs,
).save()
return super().after_return(status, retval, task_id, args, kwargs, einfo=einfo) return super().after_return(status, retval, task_id, args, kwargs, einfo=einfo)
# pylint: disable=too-many-arguments # pylint: disable=too-many-arguments

View file

@ -1,11 +1,14 @@
"""email stage tasks""" """email stage tasks"""
from email.utils import make_msgid
from smtplib import SMTPException from smtplib import SMTPException
from typing import Any, Dict, List from typing import Any, Dict, List
from celery import group from celery import group
from django.core.mail import EmailMultiAlternatives from django.core.mail import EmailMultiAlternatives
from django.core.mail.utils import DNS_NAME
from structlog import get_logger from structlog import get_logger
from passbook.lib.tasks import MonitoredTask, TaskResult, TaskResultStatus
from passbook.root.celery import CELERY_APP from passbook.root.celery import CELERY_APP
from passbook.stages.email.models import EmailStage from passbook.stages.email.models import EmailStage
@ -16,7 +19,7 @@ def send_mails(stage: EmailStage, *messages: List[EmailMultiAlternatives]):
"""Wrapper to convert EmailMessage to dict and send it from worker""" """Wrapper to convert EmailMessage to dict and send it from worker"""
tasks = [] tasks = []
for message in messages: for message in messages:
tasks.append(_send_mail_task.s(stage.pk, message.__dict__)) tasks.append(send_mail.s(stage.pk, message.__dict__))
lazy_group = group(*tasks) lazy_group = group(*tasks)
promise = lazy_group() promise = lazy_group()
return promise return promise
@ -29,19 +32,35 @@ def send_mails(stage: EmailStage, *messages: List[EmailMultiAlternatives]):
ConnectionError, ConnectionError,
), ),
retry_backoff=True, retry_backoff=True,
base=MonitoredTask,
) )
# pylint: disable=unused-argument def send_mail(self: MonitoredTask, email_stage_pk: int, message: Dict[Any, Any]):
def _send_mail_task(self, email_stage_pk: int, message: Dict[Any, Any]):
"""Send Email according to EmailStage parameters from background worker. """Send Email according to EmailStage parameters from background worker.
Automatically retries if message couldn't be sent.""" Automatically retries if message couldn't be sent."""
stage: EmailStage = EmailStage.objects.get(pk=email_stage_pk) self.save_on_success = False
backend = stage.backend try:
backend.open() stage: EmailStage = EmailStage.objects.get(pk=email_stage_pk)
# Since django's EmailMessage objects are not JSON serialisable, backend = stage.backend
# we need to rebuild them from a dict backend.open()
message_object = EmailMultiAlternatives() # Since django's EmailMessage objects are not JSON serialisable,
for key, value in message.items(): # we need to rebuild them from a dict
setattr(message_object, key, value) message_object = EmailMultiAlternatives()
message_object.from_email = stage.from_address for key, value in message.items():
LOGGER.debug("Sending mail", to=message_object.to) setattr(message_object, key, value)
stage.backend.send_messages([message_object]) message_object.from_email = stage.from_address
# Because we use the Message-ID as UID for the task, manually assign it
message_id = make_msgid(DNS_NAME)
message_object.extra_headers["Message-ID"] = message_id
LOGGER.debug("Sending mail", to=message_object.to)
stage.backend.send_messages([message_object])
self.set_status(
TaskResult(
TaskResultStatus.SUCCESSFUL,
messages=["Successfully sent Mail."],
uid=message_id,
)
)
except (SMTPException, ConnectionError) as exc:
self.set_status(TaskResult(TaskResultStatus.ERROR, [str(exc)], exc))
raise exc