Skip to main content

A robust smtp email send tool, with asyncronous send capability

Project description

emailee

PyPI Version Build Status Code Coverage Code Quality Code Factor versions license

A robust SMTP mailer library, with async mail send features built-in using Threads and Multiprocessing. Zero PyPI (production) dependencies.

Installation

Install with pip install emailee.

Some quick examples

Send a simple text email

import emailee

my_email = emailee.Emailee()
my_email.sender("john.smith@fakeemail.com")
my_email.subject("Test email")
my_email.msgContent("some poorly formatted raw text")
my_email.sendTo(["jill.smith@fakeemail.com"])
my_email.server("smtp.fakeemail.com")

if my_email.ready():
    my_email.send()

Send multiple asyncronous emails

import emailee

emails_list = [{...}]
server_dict = {...}

emails = emailee.AsyncThreads(emails_list, server_dict, outputFile='output.txt')

Documentation

There are only three classes exposed to the user of this library, Emailee, AsyncThreads and AsyncMP. The latter two utilise the Emailee class to send emails via the Threading or Multiprocessing async APIs respectively.

Emailee class

If you have used the smtplib and email libraries in the past, you may of found how difficult it is getting the email logic set up properly to handle attachments and different sending scenarios. All customisations are built into Emailee, which takes the guess work out of how to configure.

The following methods are exposed to the user:

Emailee.sender(sender: str, replyTo: str = "") -> None

  • sender - email address of who is sending the email
  • replyTo (optional) - alternative email address the receiver can reply to

Emailee.subject(subject: str = "") -> None

  • subject (optional) - email subject line, maximum of 255 characters

Emailee.msgContent(msgText: str = "", msgHTML: str = "") -> None

  • msgText (optional) - email message in raw text format
  • msgHTML (optional) - email message in email HTML format

It's a good idea to use both of these to replicate your email, so users who can't read HTML emails can read your fallback raw text email. Look online for HTML email generators

Emailee.sendTo(to: List[str] = [], cc: List[str] = [], bcc: List[str] = [], ignoreErrors: bool = False) -> None

  • to (optional) - list of to addresses to send emails to
  • cc (optional) - list of cc addresses to send emails to
  • bcc (optional) - list of bcc addresses to send emails to
  • ignoreErrors (optional) - all addresses are validated by regex to be in a valid email format. By default, if an email address is found to be invalid the program will raise an exception. If changed to True, the program will ignore and remove invalid emails and continue with sending the email.

Emailee.attachmentFiles(attachmentFiles: List[str] = []) -> None

  • attachmentFiles (optional) - list of attachments for email, can be listed by relative or full path

Emailee.server(smtpServer: str, port: int = 0, SSLTLS: str = "", authUsername: str = "", authPassword: str = "", timeout: int = 30) -> None

  • smtpServer - server name or IP address of SMTP server
  • port (optional) - port number for SMTP server, if connection not authenticated and port left blank, port will default to 25. If connection set to SSL or TLS and port left blank, port will default to 465 or 587 respectively.
  • SSLTLS (optional) - encrypted connection setting, value either "SSL" or "TLS"
  • authUsername (optional) - username for logging into the SMTP server, if authentication required. If left blank, but authPassword filled in, this will default to the sender email.
  • authPassword (optional) - password for logging into the SMTP server
  • timeout (optional) - server connection and email send timeout limit. Extend this value if on a slow network, or sending emails with large attachments

Emailee.ready() -> bool

Returns True if the minimal fields to send an email have been set.

Email.send(outputFile: str) -> bool

  • outputFile (optional) - relative or full path location of a file to write report output metadata to, required for async classes when sending mail, file must be empty or not currently exist

Returns True if successful, does not guarantee email was delivered, just sent.

Email.repr() -> str

Print class function, will print a dictionary of email metadata, useful for testing and debugging.


Data for async classes

Both async classes are very similar in their function and both require emails and server config in the exact same format:

mail_list = [
    {
        'sender': sender # see Emailee.sender()
        'replyTo': replyTo # see Emailee.sender()
        'subject': email_subject # see Emailee.subject()
        'msgText': raw_text_message # see Emailee.msgContent()
        'msgHTML': HTML_formatted_message # see Emailee.msgContent()
        'to': list_of_to_emails # see Emailee.sendTo()
        'cc': list_of_cc_emails # see Emailee.sendTo()
        'bcc': list_of_bcc_emails # see Emailee.sendTo()
        'ignoreErrors': True # see Emailee.sendTo()
        'attachmentFiles': list_of_attachment_file_paths # see Emailee.attachmentFiles()
    },
    {...},
]
server_dict = {
    'smtpServer': SMTP_server # see Emailee.server()
    'port': port_number # see Emailee.server()
    'SSLTLS': 'TLS' # see Emailee.server()
    'authUsername': authenticated_username # see Emailee.server()
    'authPassword': authenticated_password # see Emailee.server()
}

AsyncThreads class

AsyncThreads uses the threading API to enable asyncronous sending of email. Threading only utilises the same CPU core that the Python program is currently running on, so it can only maximise a single core usage, but the benefits are you can throttle how many threads are allowed to run concurrently and it has a lower overhead compared to AsyncMP.

AsyncThreads(mailList: List[Dict], serverDict: Dict, outputFile: str, maxThreads: int = 10, waitTime: float = 0)

  • mailList - see mail_list above for format
  • serverDict - see server_dict above for format
  • outputFile - relative or full path location of a file to write report output metadata to, file must be empty or not currently exist. Mandatory for async classes to handle errors.
  • maxThreads - maximum number of concurrent threads to send emails on
  • waitTime - wait time in seconds to pause between each thread being spawned, this helps if your SMTP server throttles your connection if you send too many emails in a short time span

AsyncMP class

AsyncMP uses the multiprocessing API to enable asyncronous sending of email. Multiprocessing utilises all CPU cores of the system the Python program is running on, but does not allow throttling, so you run the risk of maxing out your CPU and RAM, slowing down your system.

AsyncMP(mailList: List[Dict], serverDict: Dict, outputFile: str, waitTime: float = 0)

  • mailList - see mail_list above for format
  • serverDict - see server_dict above for format
  • outputFile - relative or full path location of a file to write report output metadata to, file must be empty or not currently exist. Mandatory for async classes to handle errors.
  • waitTime - wait time in seconds to pause between each process being spawned, this helps if your SMTP server throttles your connection if you send too many emails in a short time span

Reporting on async output

Upon completion of either async class, you can call the emailReport() method to return a metadata list of all emails sent. This won't work if an exception is thrown during the sending process, hence the outputFile requirement. The output file will be written to after each successful email send so you can analyse against emailReport() or see who received emails before a thrown exception or manual cancellation of the sending process.

Development and Testing

Emailee is built with poetry, tested with pytest, tox and coverage, type checked with mypy and formatted with black.

Dev Installation

Install with Poetry

Clone the repo, cd into it and run poetry install

Install with pip

Clone the repo, cd into it, build and run a new virtual environment, then open the pyproject.toml file and pip install all packages listed under [tool.poetry.dev-dependencies]

Testing

To successfully run local tests you will need to rename tests/example.test.env.toml to tests/test.env.toml and modify the config inside the file to contain valid email and SMTP server connection data.

Testing and coverage can then be run with pytest --cov-report term-missing --cov=emailee tests

Tox can be used to run pytest against all support Python environments. Open tox.ini and check if you have all versions of Python installed listed under envlist.

Pre-commit hooks

Run pre-commit install to install the pre-commit hooks in the .pre-commit-config.yaml file. Then run pre-commit run --all-files to auto-check every file for issues.

Feel free to send me any changes or feedback.


Thanks for reading :)

:envelope: :envelope: :envelope: :envelope: :envelope:

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

emailee-1.0.0.tar.gz (15.7 kB view hashes)

Uploaded Source

Built Distribution

emailee-1.0.0-py3-none-any.whl (13.0 kB view hashes)

Uploaded Python 3

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page