Source code for salmon.mail

"""
The salmon.mail module contains nothing more than wrappers around the big work
done in salmon.encoding.  These are the actual APIs that you'll interact with
when doing email, and they mostly replicate the salmon.encoding.MailBase
functionality.

The main design criteria is that MailRequest is mostly for reading email
that you've received, so it doesn't have functions for attaching files and such.
MailResponse is used when you are going to write an email, so it has the
APIs for doing attachments and such.
"""
from email.utils import parseaddr
import mimetypes
import os
import warnings

from salmon import bounce, encoding

# You can change this to 'Delivered-To' on servers that support it like Postfix
ROUTABLE_TO_HEADER = 'to'


def _decode_header_randomness(addr):
    """
    This fixes the given address so that it is *always* a set() of
    just email addresses suitable for routing.
    """
    if not addr:
        return set()
    elif isinstance(addr, (list, tuple)):
        addr_set = set()
        for a in addr:
            for returned_addr in _decode_header_randomness(a):
                addr_set.add(returned_addr)

        return addr_set
    elif isinstance(addr, str):
        return set([parseaddr(addr.lower())[1]])
    elif isinstance(addr, bytes):
        addr = addr.decode()
        return set([parseaddr(addr.lower())[1]])
    else:
        raise encoding.EncodingError("Address must be a string or a list not: %r", type(addr))


[docs]class MailRequest: """ This is what is given to your message handlers. The information you get out of this is *ALWAYS* in Python str and should be usable by any API. Modifying this object will cause other handlers that deal with it to get your modifications, but in general you don't want to do more than maybe tag a few headers. """ def __init__(self, Peer, From, To, Data): """ Peer is the remote peer making the connection (sometimes the queue name). From and To are what you think they are. Data is the raw full email as received by the server. NOTE: It does not handle multiple From headers, if that's even possible. It will parse the From into a list and take the first one. """ self.Peer = Peer self.Data = Data try: self.From = _decode_header_randomness(From).pop() except KeyError: self.From = None try: self.To = _decode_header_randomness(To).pop() except KeyError: self.To = None self.base = encoding.from_string(self.Data) if 'from' not in self.base: self.base['from'] = self.From if 'to' not in self.base: # do NOT use ROUTABLE_TO here self.base['to'] = self.To self.From = self.From or self.base['from'] self.To = self.To or self.base[ROUTABLE_TO_HEADER] self.bounce = None def __repr__(self): return "From: {}".format([self.Peer, self.From, self.To])
[docs] def all_parts(self): """Returns all multipart mime parts. This could be an empty list.""" return self.base.parts
[docs] def body(self): """ Always returns a body if there is one. If the message is multipart then it returns the first part's body, if it's not then it just returns the body. If returns None then this message has nothing for a body. """ if self.base.parts: return self.base.parts[0].body else: return self.base.body
def __contains__(self, key): return self.base.__contains__(key) def __getitem__(self, name): return self.base.__getitem__(name) def __setitem__(self, name, val): self.base.__setitem__(name, val) def __delitem__(self, name): del self.base[name] def __str__(self): """ Converts this to a string usable for storage into a queue or transmission. """ return encoding.to_string(self.base)
[docs] def items(self): return self.base.items()
[docs] def keys(self): return self.base.keys()
[docs] def to_message(self): """ Converts this to a Python email message you can use to interact with the python mail APIs. """ return encoding.to_message(self.base)
[docs] def walk(self): """Recursively walks all attached parts and their children.""" for x in self.base.walk(): yield x
[docs] def is_bounce(self, threshold=0.3): """ Determines whether the message is a bounce message based on salmon.bounce.BounceAnalzyer given threshold. 0.3 is a good conservative base. """ if not self.bounce: self.bounce = bounce.detect(self) return self.bounce.score > threshold
@property def original(self): warnings.warn("MailRequest.original is deprecated, use MailRequest.Data instead", category=DeprecationWarning, stacklevel=2) return self.Data
[docs]class MailResponse: """ You are given MailResponse objects from the salmon.view methods, and whenever you want to generate an email to send to someone. It has the same basic functionality as MailRequest, but it is designed to be written to, rather than read from (although you can do both). You can easily set a Body or Html during creation or after by passing it as __init__ parameters, or by setting those attributes. You can initially set the From, To, and Subject, but they are headers so use the dict notation to change them: ``msg['From'] = 'joe@test.com'``. The message is not fully crafted until right when you convert it with MailResponse.to_message. This lets you change it and work with it, then send it out when it's ready. """ def __init__(self, To=None, From=None, Subject=None, Body=None, Html=None): self.Body = Body self.Html = Html self.base = encoding.MailBase([('To', To), ('From', From), ('Subject', Subject)]) self.multipart = self.Body and self.Html self.attachments = [] def __contains__(self, key): return self.base.__contains__(key) def __getitem__(self, key): return self.base.__getitem__(key) def __setitem__(self, key, val): return self.base.__setitem__(key, val) def __delitem__(self, name): del self.base[name]
[docs] def attach(self, filename=None, content_type=None, data=None, disposition=None): """ Simplifies attaching files from disk or data as files. To attach simple text simple give data and a content_type. To attach a file, give the data/content_type/filename/disposition combination. For convenience, if you don't give data and only a filename, then it will read that file's contents when you call to_message() later. If you give data and filename then it will assume you've filled data with what the file's contents are and filename is just the name to use. """ if data is None: if filename is None: raise TypeError("You must give a filename or some data to attach.") elif not os.path.exists(filename): raise TypeError("File doesn't exist, and no data given.") self.multipart = True if filename and not content_type: content_type, encoding = mimetypes.guess_type(filename) if not content_type: raise ValueError("No content type given, and couldn't guess from the filename: %r" % filename) self.attachments.append({ 'filename': filename, 'content_type': content_type, 'data': data, 'disposition': disposition, })
[docs] def attach_part(self, part): """ Attaches a raw MailBase part from a MailRequest (or anywhere) so that you can copy it over. """ self.multipart = True self.attachments.append({'filename': None, 'content_type': None, 'data': None, 'disposition': None, 'part': part, })
[docs] def attach_all_parts(self, mail_request): """ Used for copying the attachment parts of a mail.MailRequest object for mailing lists that need to maintain attachments. """ for part in mail_request.all_parts(): self.attach_part(part)
[docs] def clear(self): """ Clears out the attachments so you can redo them. Use this to keep the headers for a series of different messages with different attachments. """ del self.attachments[:] del self.base.parts[:] self.multipart = False
[docs] def update(self, message): """ Used to easily set a bunch of headers from another dict like object. """ for k in message.keys(): self.base[k] = message[k]
def __str__(self): """ Converts to a string. """ return self.to_message().as_string() def _encode_attachment(self, filename=None, content_type=None, data=None, disposition=None, part=None): """ Used internally to take the attachments mentioned in self.attachments and do the actual encoding in a lazy way when you call to_message. """ if part: self.base.parts.append(part) elif filename: if not data: data = open(filename).read() self.base.attach_file(filename, data, content_type, disposition or 'attachment') else: self.base.attach_text(data, content_type) ctype = self.base.content_encoding['Content-Type'][0] if ctype and not ctype.startswith('multipart'): self.base.content_encoding['Content-Type'] = ('multipart/mixed', {})
[docs] def to_message(self): """ Figures out all the required steps to finally craft the message you need and return it. The resulting message is also available as a self.base attribute. What is returned is a Python email API message you can use with those APIs. The self.base attribute is the raw salmon.encoding.MailBase. """ del self.base.parts[:] if self.Body and self.Html: self.multipart = True self.base.content_encoding['Content-Type'] = ('multipart/alternative', {}) if self.multipart: self.base.body = None if self.Body: self.base.attach_text(self.Body, 'text/plain') if self.Html: self.base.attach_text(self.Html, 'text/html') for args in self.attachments: self._encode_attachment(**args) elif self.Body: self.base.body = self.Body self.base.content_encoding['Content-Type'] = ('text/plain', {}) elif self.Html: self.base.body = self.Html self.base.content_encoding['Content-Type'] = ('text/html', {}) return encoding.to_message(self.base)
[docs] def all_parts(self): """ Returns all the encoded parts. Only useful for debugging or inspecting after calling to_message(). """ return self.base.parts
[docs] def items(self): return self.base.items()
[docs] def keys(self): return self.base.keys()