Asynchronous email sending library for Django with enhanced templating options.
Full documentation can be found here
sendmail provides a set of powerful features, such as:
- Handling millions of emails efficiently.
- Allows you to send emails asynchronously.
- Multi backend support.
- Support for inlined images.
- 2-phase email templating to allow non-technical users contributing to emails creation.
- Scheduling support.
- Works with task queues like RQ and Celery.
- Uses multiprocessing to send emails in parallel.
- Support of different storages.
Installing nh3 is strongly encouraged for security reasons. Only with installed nh3 emails will be rendered in admin interface.
pip install django-sendmail
To enable JSON editing functionality for Newsletter in the admin interface, install with django-jsoneditor
optional dependency.
pip install django-sendmail[jsoneditor]
Add sendmail and ckeditor to your installed app in settings.py:
INSTALLED_APPS = [
# other apps,
'ckeditor',
'ckeditor_uploader',
'sendmail',
'jsoneditor' # if installed
]
To your settings.py also add email server configurations:
EMAIL_HOST = '<your-smtp-host.com>'
EMAIL_PORT = '<SMTP-PORT>'
EMAIL_USE_TLS = True
DEFAULT_FROM_EMAIL = '[email protected]'
To your list of template engines (TEMPLATES
) settings add a special template backend:
TEMPLATES = [
{
'BACKEND': 'sendmail.template.backends.sendmail.SendMailTemplates',
'APP_DIRS': True,
'DIRS': [BASE_DIR / 'templates', ...],
'OPTIONS': {
'context_processors': [
'django.contrib.auth.context_processors.auth',
'django.template.context_processors.debug',
'django.template.context_processors.i18n',
'django.template.context_processors.media',
'django.template.context_processors.static',
'django.template.context_processors.tz',
'django.template.context_processors.request',
]
}
},
...
]
Add CKEDITOR_UPLOAD_PATH
. This path will be used to store ckeditor uploaded images inside MEDIA_ROOT
:
CKEDITOR_UPLOAD_PATH = 'ckeditor_uploads'
To your main urls.py
add ckeditor_uploader
urls:
urlpatterns = [
...,
path('ckeditor/', include('ckeditor_uploader.urls')),
]
Add STATIC_URL
and STATIC_ROOT
STATIC_URL = '/static/'
STATIC_ROOT = BASE_DIR / 'staticfiles'
Add MEDIA_URL
and MEDIA_ROOT
MEDIA_URL = '/media/'
MEDIA_ROOT = BASE_DIR / 'workdir' / 'media'
Run migrate
:
python manage.py migrate
Run collectstatic
:
python manage.py collectstatic
Set sendmail.EmailBackend
as your EMAIL_BACKEND
in Django's settings.py
:
EMAIL_BACKEND = 'sendmail.EmailBackend'
Send a simple email is really easy:
from sendmail import mail
mail.send(
'[email protected]', # List of email addresses also accepted
'[email protected]',
subject='My email',
message='Hi there!',
html_message='Hi <strong>there</strong>!',
)
If you want to use templates:
- In your templates folder create an .html file with your email markup. Inside it you can leave placeholders or use context vars. For example something like this:
{% load sendmail %}
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Example email template</title>
</head>
<body>
Hello, {{ username }}
{% placeholder 'main' %}
</body>
</html>
- Register your template in
settings.py
:
SENDMAIL = {
'EMAIL_TEMPLATES': [
('your-file/path', _('Your-Name')),
]
}
You can use relative path from your templates
folder or absolute file path.
-
Specify list of translation languages in
settings.LANGUAGES
or disable internalization by settingsettings.USE_I18N
toFalse
. -
Open your admin interface and create a new EmailMergeModel instance:
-
Enter name. This will be used as identifier for your template and click “Save and continue editing”.
-
Select Base File which you have created.
-
You will be requested to enter values to the placeholders you entered in the template. (main in the example). You can specify variables to be filled with the context. Syntax is
#var#
. Example: This is a simple mail created by #generator# -
If you have more than 1 language configured you will be able to fill values for all languages in
LANGUAGES
:- Create element in Email Template Contents inline with desired language
- Click
Save and Continue Editing
- Fill in placeholders
-
Save your instance.
-
You can now send test email with created object using
Send test email button
. In test email you can make use ofdemo context
that can be refreshed usingReparse Context
button.
-
-
To send an email with the created template:
from sendmail import mail
mail.send(
'[email protected]', # List of email addresses or list of EmailAddress also accepted
'[email protected]',
emailmerge='your-template-here', # Could be an EmailMergeModel instance or name
context={'generator': 'sendmail',
'username': 'michaelpoi', }, # Context is used to fill both {{ var }} in html and #var# in ckeditor.
language='en' # If not specified settings.LANGUAGE_CODE is used
)
The above command will put your email on the queue so you can use the command in your webapp without slowing down the request/response cycle too much. To actually send them out, run
python manage.py sendmail all
You can schedule this management command to run regularly via cron:
* * * * * (/usr/bin/python manage.py sendmail all >> send_mail.log 2>&1)
Full documentation can be found here