Skip to main content

Markdown template based HTML and text emails for Django.

Project description

Django eMark↓

emark logo: envelope with markdown stamp

Markdown template based HTML and text emails for Django.

  • simple email templates with markdown
  • support for HTML and text emails
  • i18n support
  • built-in UTM tracking
  • built-in sent, open and click tracking
  • automatic CSS inliner via premailer

PyPi Version Test Coverage GitHub License

Setup

python3 -m pip install emark
# settings.py
INSTALLED_APPS = [
    "emark",
    # ...
]
python3 manage.py migrate

Usage

<!-- myapp/my_message.md -->
# Hello World

Hi {{ user.short_name }}!
# myapp/emails.py
from emark.message import MarkdownEmail


class MyMessage(MarkdownEmail):
    subject = "Hello World"
    template_name = "myapp/my_message.md"
# myapp/views.py
from . import emails


def my_view(request):
    message = emails.MyMessage.to_user(request.user)
    message.send()

Templates

You can use Django's template engine, just like you usually would. You can use translations, template tags, filters, blocks, etc.

You may also have a base template, that you inherit form in your individual emails to provide a consistent salutation and farewell.

<!-- base.md -->
{% load static i18n %}
{% block salutation %}Hi {{ user.short_name }}!{% endblock %}

{% block content %}{% endblock %}

{% block farewell %}
{% blocktrans trimmed %}
Best regards,
{{ site_admin }}
{% endblocktrans %}
{% endblock %}

{% block footer %}
Legal footer.
{% endblock %}
<!-- myapp/email.md -->
{% extends "base.md" %}

{% block content %}
This is the content of the email.
{% endblock %}

Context

The context is passed to the template as a dictionary. Furthermore, you may override the get_context_data method to add additional context variables.

# myapp/emails.py
from emark.message import MarkdownEmail


class MyMessage(MarkdownEmail):
    subject = "Hello World"
    template_name = "myapp/email.md"

    def get_context_data(self):
        context = super().get_context_data()
        context["my_variable"] = "Hello World"
        return context

Tracking

Sent, Open & Click Tracking

Django eMark comes with built-in tracking for sent, open and click events. The tracking is done via a tracking pixel and a redirect view.

As an added bonus, this feature also comes with an open-in-browser link that allows the user to view the email in their browser if their email client does not support HTML emails.

This feature is disabled by default. To enable it, you need to use a separate email backend. This backend will send the email via SMTP and also add the tracking pixel and redirect view. However, it will send a separate email for each recipient, which may not be desirable in all cases.

# settings.py
EMAIL_BACKEND = "emark.backends.TrackingSMTPEmailBackend"

Furthermore, you need to add the tracking view to your urls.py:

# urls.py
from django.urls import include, path

urlpatterns = [
    # … other urls
    path("emark/", include("emark.urls")),
]

You will need to provide a domain name for the tracking pixel and redirect view. This can be done via the DOMAIN setting:

# settings.py
EMARK = {"DOMAIN": "example.com"}

If the site framework is installed and no settings are provided, the domain will be automatically set to the current site's domain.

The tracking data is stored in the database. You need to run migrations to create the necessary tables:

python3 manage.py migrate

You can analyze the tracking data via the tables emark_sent, emark_open and emark_click.

UTM Tracking

Every MarkdownEmail subclass comes with automatic UTM tracking. UTM parameters are added to all links in the email. Existing UTM params on link that have been explicitly set, are not overridden. The default parameters are:

  • utm_source: website
  • utm_medium: email
  • utm_campaign: {{ EMAIL_CLASS_NAME }}

The global UTM parameters can be overridden via the EMARK_UTM_PARAMS setting, which is a dictionary of parameters:

# settings.py
EMARK = {
    "UTM_PARAMS": {
        "utm_source": "website",  # default
        "utm_medium": "email",  # default
    }
}

You may also change the UTM parameters by overriding the get_utm_params or passing a utm_params dictionary to class constructor.

# myapp/emails.py
from emark.message import MarkdownEmail


class MyMessage(MarkdownEmail):
    subject = "Hello World"
    template_name = "myapp/email.md"

    # override the parameters for this email class
    def get_utm_params(self):
        return {
            "utm_source": "myapp",
            "utm_medium": "email",
            "utm_campaign": "my-campaign",
        }


# or alternatively during instantiation
MyMessage(utm_params={"utm_campaign": "my-other-campaign"}).send()

Development

Pretty HTML emails are great, unless they spam your console during development. To prevent this, you can use the ConsoleEmailBackend:

# settings.py
EMAIL_BACKEND = "emark.backends.ConsoleEmailBackend"

The ConsoleEmailBackend will only print the plain text version of the email.

Email Dashboard

Django eMark comes with a simple email dashboard to preview your templates.

To enable the dashboard, add the app to your INSTALLED_APPS setting

# settings.py
INSTALLED_APPS = [
    # ...
    "emark",
    "emark.contrib.dashboard",  # needs to be added before Django's admin app
    # ...
    "django.contrib.admin",  # required for the dashboard
    # ...
]

and add the following to your urls.py:

# urls.py
from django.urls import include, path


urlpatterns = [
    # … other urls
    path(
        "emark/",
        include(
            [
                path("", include("emark.urls")),
                path("dashboard/", include("emark.contrib.dashboard.urls")),
            ]
        ),
    ),
]

Next you need to register the email classes you want to preview in the dashboard:

# myapp/emails.py
from emark.message import MarkdownEmail
from emark.contrib import dashboard


@dashboard.register
class MyMessage(MarkdownEmail):
    subject = "Hello World"
    template_name = "myapp/email.md"

Credits

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

emark-2.2.0.tar.gz (21.0 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

emark-2.2.0-py3-none-any.whl (24.7 kB view details)

Uploaded Python 3

File details

Details for the file emark-2.2.0.tar.gz.

File metadata

  • Download URL: emark-2.2.0.tar.gz
  • Upload date:
  • Size: 21.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for emark-2.2.0.tar.gz
Algorithm Hash digest
SHA256 3f1bfed44c3922c3180eae9e0293b702a434f251898b3bb0c261d9c47c1b0df4
MD5 b3c54811e1818dfea580dad06b255b01
BLAKE2b-256 0d37822a45516a144d2d0bafc934ce690f431c243c27b82e281c61d82406ccf1

See more details on using hashes here.

Provenance

The following attestation bundles were made for emark-2.2.0.tar.gz:

Publisher: release.yml on voiio/emark

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file emark-2.2.0-py3-none-any.whl.

File metadata

  • Download URL: emark-2.2.0-py3-none-any.whl
  • Upload date:
  • Size: 24.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for emark-2.2.0-py3-none-any.whl
Algorithm Hash digest
SHA256 3fd1c74dfd46979152005301237dc9e4d9c191a44113e87630deacd2f96b8848
MD5 d0933eed7eaed969aee939cc175bca64
BLAKE2b-256 2242b24b29af11da766c3e9fbfa36b6eebaa35e59a3821c175752f0d1fdf4924

See more details on using hashes here.

Provenance

The following attestation bundles were made for emark-2.2.0-py3-none-any.whl:

Publisher: release.yml on voiio/emark

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

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