"""
Send E-Mail through Postmark
Adapted from `repoze.sendmail<https://github.com/repoze/repoze.sendmail>`_.
Usage::
qp = PostmarkQueueProcessor(token, maildir, maildir, ..., limit=x)
qp.send_messages()
"""
from __future__ import annotations
import json
import pycurl
from io import BytesIO
from .core import log, MailQueueProcessor
[docs]
class PostmarkMailQueueProcessor(MailQueueProcessor):
def __init__(
self,
postmark_token: str,
*paths: str,
limit: int | None = None
):
super().__init__(*paths, limit=limit)
# Keep a pycurl object around, to use HTTP keep-alive - though pycurl
# is much worse in terms of it's API, the performance is *much* better
# than requests and it supports modern features like HTTP/2 or HTTP/3
[docs]
self.url = 'https://api.postmarkapp.com/email/batch'
[docs]
self.curl = pycurl.Curl()
self.curl.setopt(pycurl.TCP_KEEPALIVE, 1)
self.curl.setopt(pycurl.URL, self.url)
self.curl.setopt(pycurl.HTTPHEADER, [
'Accept:application/json',
'Content-Type:application/json',
f'X-Postmark-Server-Token:{postmark_token}'
])
self.curl.setopt(pycurl.POST, 1)
[docs]
def send(self, filename: str, payload: str) -> bool:
""" Sends the mail and returns success as bool """
code, body = self.send_request(payload)
if 400 <= code < 600:
raise RuntimeError(f'{code} calling {self.url}: {body}')
result = json.loads(body)
# If we don't get a list we definitely hit an error
if not isinstance(result, list):
log.error(f'Invalid API response in mail batch {filename}')
return False
# If any list entry contains errors we return failure
success = True
for index, status in enumerate(result, start=1):
error_code = status.get('ErrorCode', 0)
message = status.get('Message', f'ErrorCode: {error_code}')
if error_code == 406:
# inactive recipient, error can be ignored but still log it
log.warning(
f'Inactive recipient in mail batch {filename} at '
f'index {index}: {message}'
)
elif error_code != 0:
log.error(
f'Error in mail batch {filename} at index '
f'{index}: {message}'
)
success = False
return success
[docs]
def send_request(self, payload: str) -> tuple[int, str]:
""" Performes the API request using the given payload. """
body = BytesIO()
self.curl.setopt(pycurl.WRITEDATA, body)
self.curl.setopt(pycurl.POSTFIELDS, payload)
self.curl.perform()
code = self.curl.getinfo(pycurl.RESPONSE_CODE)
body.seek(0)
body_str = body.read().decode('utf-8')
return code, body_str