FitTrackee/fittrackee/emails/email.py

179 lines
5.9 KiB
Python
Raw Normal View History

import logging
import smtplib
import ssl
from email.mime.multipart import MIMEMultipart
from email.mime.text import MIMEText
2022-07-02 19:25:07 +02:00
from typing import Dict, List, Optional, Type, Union
from urllib.parse import unquote
2022-07-02 19:25:07 +02:00
from babel.support import Translations
2021-01-02 19:28:03 +01:00
from flask import Flask
from jinja2 import Environment, FileSystemLoader, select_autoescape
2022-02-16 13:35:38 +01:00
from urllib3.util import parse_url
2022-02-16 13:35:38 +01:00
from .exceptions import InvalidEmailUrlScheme
email_log = logging.getLogger('fittrackee_api_email')
email_log.setLevel(logging.DEBUG)
class EmailMessage:
2021-01-02 19:28:03 +01:00
def __init__(
self, sender: str, recipient: str, subject: str, html: str, text: str
) -> None:
self.sender = sender
self.recipient = recipient
self.subject = subject
self.html = html
self.text = text
2021-01-02 19:28:03 +01:00
def generate_message(self) -> MIMEMultipart:
message = MIMEMultipart('alternative')
message['Subject'] = self.subject
message['From'] = self.sender
message['To'] = self.recipient
part1 = MIMEText(self.text, 'plain')
part2 = MIMEText(self.html, 'html')
message.attach(part1)
message.attach(part2)
return message
class EmailTemplate:
2022-07-02 19:25:07 +02:00
def __init__(
self,
template_directory: str,
translations_directory: str,
languages: List[str],
) -> None:
self._translations = self._get_translations(
translations_directory, languages
)
self._env = Environment(
autoescape=select_autoescape(['html', 'htm', 'xml']),
loader=FileSystemLoader(template_directory),
2022-07-02 19:25:07 +02:00
extensions=['jinja2.ext.i18n'],
)
@staticmethod
def _get_translations(
translations_directory: str, languages: List[str]
) -> Dict:
translations = {}
for language in languages:
translations[language] = Translations.load(
dirname=translations_directory, locales=[language]
)
return translations
def _load_translation(self, lang: str) -> None:
self._env.install_gettext_translations( # type: ignore
self._translations[lang],
newstyle=True,
)
2021-01-02 19:28:03 +01:00
def get_content(
self, template_name: str, lang: str, part: str, data: Dict
) -> str:
2022-07-02 19:25:07 +02:00
self._load_translation(lang)
template = self._env.get_template(f'{template_name}/{part}')
return template.render(data)
2021-01-02 19:28:03 +01:00
def get_all_contents(self, template: str, lang: str, data: Dict) -> Dict:
output = {}
for part in ['subject.txt', 'body.txt', 'body.html']:
output[part] = self.get_content(template, lang, part, data)
return output
2021-01-02 19:28:03 +01:00
def get_message(
self, template: str, lang: str, sender: str, recipient: str, data: Dict
) -> MIMEMultipart:
output = self.get_all_contents(template, lang, data)
message = EmailMessage(
sender,
recipient,
output['subject.txt'],
output['body.html'],
output['body.txt'],
)
return message.generate_message()
2022-02-16 13:35:38 +01:00
class EmailService:
2021-01-02 19:28:03 +01:00
def __init__(self, app: Optional[Flask] = None) -> None:
self.host = 'localhost'
self.port = 25
self.use_tls = False
self.use_ssl = False
self.username = None
self.password = None
self.sender_email = 'no-reply@example.com'
2021-01-02 19:28:03 +01:00
self.email_template: Optional[EmailTemplate] = None
if app is not None:
self.init_email(app)
2021-01-02 19:28:03 +01:00
def init_email(self, app: Flask) -> None:
2022-02-16 13:35:38 +01:00
parsed_url = self.parse_email_url(app.config['EMAIL_URL'])
self.host = parsed_url['host']
self.port = parsed_url['port']
self.use_tls = parsed_url['use_tls']
self.use_ssl = parsed_url['use_ssl']
self.username = parsed_url['username']
self.password = parsed_url['password']
2021-01-02 19:28:03 +01:00
self.sender_email = app.config['SENDER_EMAIL']
2022-07-02 19:25:07 +02:00
self.email_template = EmailTemplate(
app.config['TEMPLATES_FOLDER'],
app.config['TRANSLATIONS_FOLDER'],
app.config['LANGUAGES'],
)
2022-02-16 13:35:38 +01:00
@staticmethod
def parse_email_url(email_url: str) -> Dict:
parsed_url = parse_url(email_url)
if parsed_url.scheme != 'smtp':
raise InvalidEmailUrlScheme()
2023-10-02 11:31:25 +02:00
username, password = (
2022-08-27 19:32:54 +02:00
parsed_url.auth.split(':')
if parsed_url.auth
else [None, None] # type: ignore
2022-02-16 13:35:38 +01:00
)
return {
'host': parsed_url.host,
'port': 25 if parsed_url.port is None else parsed_url.port,
2023-10-02 11:31:25 +02:00
'use_tls': parsed_url.query == 'tls=True',
'use_ssl': parsed_url.query == 'ssl=True',
'username': username,
'password': (
2023-10-02 11:31:25 +02:00
unquote(password) if isinstance(password, str) else password
),
2022-02-16 13:35:38 +01:00
}
@property
2021-01-02 19:28:03 +01:00
def smtp(self) -> Type[Union[smtplib.SMTP_SSL, smtplib.SMTP]]:
return smtplib.SMTP_SSL if self.use_ssl else smtplib.SMTP
2021-01-02 19:28:03 +01:00
def send(
self, template: str, lang: str, recipient: str, data: Dict
) -> None:
if not self.email_template:
raise Exception('No email template defined.')
message = self.email_template.get_message(
template, lang, self.sender_email, recipient, data
)
connection_params = {}
if self.use_ssl or self.use_tls:
context = ssl.create_default_context()
if self.use_ssl:
connection_params.update({'context': context})
2021-01-02 19:28:03 +01:00
with self.smtp(
self.host, self.port, **connection_params # type: ignore
) as smtp:
if self.use_tls:
2022-04-23 06:11:08 +02:00
smtp.ehlo()
smtp.starttls(context=context)
2022-04-23 06:11:08 +02:00
smtp.ehlo()
if self.username and self.password:
smtp.login(self.username, self.password) # type: ignore
smtp.sendmail(self.sender_email, recipient, message.as_string())
smtp.quit()