2016-04-07 15:03:22 +02:00
|
|
|
#!/usr/bin/env python
|
2013-11-01 19:31:00 +01:00
|
|
|
|
|
|
|
"""
|
|
|
|
Deliver email messages that have been queued by various things
|
|
|
|
(at this time invitation reminders and day1/day2 followup emails).
|
|
|
|
|
|
|
|
This management command is run via supervisor. Do not run on multiple machines,
|
|
|
|
as you may encounter multiple sends in a specific race condition.
|
|
|
|
"""
|
|
|
|
|
|
|
|
from __future__ import absolute_import
|
|
|
|
|
|
|
|
from django.conf import settings
|
|
|
|
from django.core.management.base import BaseCommand
|
|
|
|
from django.core.mail import EmailMultiAlternatives, get_connection
|
|
|
|
from django.utils.html import format_html
|
|
|
|
|
|
|
|
from zerver.models import ScheduledJob
|
|
|
|
from zerver.lib.context_managers import lockfile
|
|
|
|
|
|
|
|
import time
|
|
|
|
import logging
|
|
|
|
from datetime import datetime
|
|
|
|
from ujson import loads
|
2016-09-12 08:07:51 +02:00
|
|
|
from typing import Any
|
2013-11-01 19:31:00 +01:00
|
|
|
|
|
|
|
## Setup ##
|
|
|
|
log_format = "%(asctime)s: %(message)s"
|
|
|
|
logging.basicConfig(format=log_format)
|
|
|
|
|
|
|
|
formatter = logging.Formatter(log_format)
|
|
|
|
file_handler = logging.FileHandler(settings.EMAIL_DELIVERER_LOG_PATH)
|
|
|
|
file_handler.setFormatter(formatter)
|
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
logger.setLevel(logging.DEBUG)
|
|
|
|
logger.addHandler(file_handler)
|
|
|
|
|
|
|
|
|
|
|
|
def get_recipient_as_string(dictionary):
|
2016-09-12 08:07:51 +02:00
|
|
|
# type: (Dict[str, str]) -> str
|
2013-11-01 19:31:00 +01:00
|
|
|
if not dictionary["recipient_name"]:
|
|
|
|
return dictionary["recipient_email"]
|
|
|
|
return format_html(u"\"{0}\" <{1}>", dictionary["recipient_name"], dictionary["recipient_email"])
|
|
|
|
|
|
|
|
def get_sender_as_string(dictionary):
|
2016-09-12 08:07:51 +02:00
|
|
|
# type: (Dict[str, str]) -> str
|
2013-11-16 01:37:55 +01:00
|
|
|
if dictionary["sender_email"]:
|
|
|
|
return dictionary["sender_email"] if not dictionary["sender_name"] else format_html(u"\"{0}\" <{1}>",
|
|
|
|
dictionary["sender_name"],
|
|
|
|
dictionary["sender_email"])
|
2013-11-01 19:31:00 +01:00
|
|
|
return settings.DEFAULT_FROM_EMAIL
|
|
|
|
|
|
|
|
def send_email_job(job):
|
2016-09-12 08:07:51 +02:00
|
|
|
# type: (ScheduledJob) -> bool
|
2013-11-01 19:31:00 +01:00
|
|
|
data = loads(job.data)
|
|
|
|
fields = {'subject': data["email_subject"],
|
|
|
|
'body': data["email_text"],
|
|
|
|
'from_email': get_sender_as_string(data),
|
|
|
|
'to': [get_recipient_as_string(data)]}
|
|
|
|
|
|
|
|
msg = EmailMultiAlternatives(**fields)
|
|
|
|
if data["email_html"]:
|
|
|
|
msg.attach_alternative(data["email_html"], "text/html")
|
|
|
|
return msg.send() > 0
|
|
|
|
|
|
|
|
class Command(BaseCommand):
|
2016-07-17 19:38:15 +02:00
|
|
|
help = """Deliver emails queued by various parts of Zulip
|
|
|
|
(either for immediate sending or sending at a specified time).
|
2013-11-01 19:31:00 +01:00
|
|
|
|
2015-08-21 18:34:54 +02:00
|
|
|
Run this command under supervisor. We use Mandrill for zulip.com; this is for SMTP email delivery.
|
2013-11-01 19:31:00 +01:00
|
|
|
|
2016-11-22 01:44:16 +01:00
|
|
|
Usage: ./manage.py deliver_email
|
2013-11-01 19:31:00 +01:00
|
|
|
"""
|
|
|
|
|
2013-11-13 22:59:04 +01:00
|
|
|
def handle(self, *args, **options):
|
2016-09-12 08:07:51 +02:00
|
|
|
# type: (*Any, **Any) -> None
|
2013-11-01 19:31:00 +01:00
|
|
|
# TODO: this only acquires a lock on the system, not on the DB:
|
|
|
|
# be careful not to run this on multiple systems.
|
2016-08-13 04:24:45 +02:00
|
|
|
|
|
|
|
# In the meantime, we have an option to prevent this job from
|
|
|
|
# running on >1 machine
|
|
|
|
if settings.EMAIL_DELIVERER_DISABLED:
|
|
|
|
return
|
|
|
|
|
2013-11-01 19:31:00 +01:00
|
|
|
with lockfile("/tmp/zulip_email_deliver.lockfile"):
|
|
|
|
while True:
|
2016-07-08 18:57:01 +02:00
|
|
|
# make sure to use utcnow, otherwise it gets confused when you set the time with utcnow(),
|
|
|
|
# and select with now()
|
2014-02-10 21:18:13 +01:00
|
|
|
email_jobs_to_deliver = ScheduledJob.objects.filter(type=ScheduledJob.EMAIL,
|
2016-12-03 00:04:17 +01:00
|
|
|
scheduled_timestamp__lte=datetime.utcnow())
|
2013-11-01 19:31:00 +01:00
|
|
|
if email_jobs_to_deliver:
|
|
|
|
for job in email_jobs_to_deliver:
|
|
|
|
if not send_email_job(job):
|
|
|
|
logger.warn("No exception raised, but %r sent as 0 bytes" % (job,))
|
|
|
|
else:
|
|
|
|
job.delete()
|
|
|
|
time.sleep(10)
|
|
|
|
else:
|
|
|
|
# Less load on the db during times of activity, and more responsiveness when the load is low
|
|
|
|
time.sleep(2)
|