Skip to content

Emails

You need an email service to handle login links, abandoned cart notifications, and other automated emails. The following are instructions for Mailgun, AWS SES and Cloudflare (receiving only)

Setup - Sending Emails

  1. Sign up at Mailgun

  2. Go to Sending > Domains > Add New Domain

  3. Copy the DNS records from the Mailgun instructions and add them to your DNS provider.

    Type Name Value
    TXT email._domainkey.yourdomain.com k=rsa; p=lorem-ipsum
    TXT yourdomain.com v=spf1 include:mailgun.org ~all
    CNAME email.yourdomain.com eu.mailgun.org
    Type Name Value Priority
    MX yourdomain.com mxa.eu.mailgun.org 10
    MX yourdomain.com mxb.eu.mailgun.org 10

    Info

    If you use a subdomain, make sure it's reflected in your DNS records, e.g.: mail.yourdomain.com instead of yourdomain.com. Mailgun will show you the correct subdomain to use in the instructions.

  4. Add this DMARC record for better delivery:

    Type Name Value
    TXT _dmarc.yourdomain.com v=DMARC1; p=none
  5. Get your SMTP credentials from Domain Overview > SMTP Credentials

  6. Add the password to your .env.prod file (or .env.local for local development):

    .env.prod
    EMAIL_HOST_PASSWORD=[copied_password]
    
  7. And change the following in your settings.py:

    config/settings.py
    EMAIL_HOST = 'smtp.mailgun.org'
    EMAIL_HOST_USER = '[email protected]'
    EMAIL_HOST_PASSWORD = config('EMAIL_HOST_PASSWORD', default='')
    DEFAULT_FROM_EMAIL = '[email protected]'
    NOREPLY_EMAIL = '[email protected]'
    EMAIL_PORT = '587'
    
  1. Sign up at AWS and go to AWS SES

  2. Click on Get Started and enter an email address where you can receive emails (most likely not (yet) the domain you want to use for sending emails), click on Next

  3. Enter the following and click Next:

    • yourdomain.com in Sending domain
    • mail-from.yourdomain.com in MAIL FROM domain - optional, but recommended for better deliverability (this subdomain must not be used for anything else, neither sending nor receiving emails, this is why we chose mail-from.yourdomain.com instead of e.g. mail.yourdomain.com)
  4. Optional: Enable Virtual Deliverability Manager to get insights into your email deliverability

  5. Review and click Get Started

  6. Now check Open tasks, the most important ones are:

    • Verify email address - click on the verification link in the email sent to the address you entered in step 2
    • Verify sending domain - click on Get DNS Records and enter the records in your DNS provider
    • Request production access - fill in the form and click on Submit request, the mail type is Transactional

    Info

    The MX record of the MAIL FROM records is slightly strange formatted if you are using Cloudflare. The value field 10 feedback-smtp.eu-west-1.amazonses.com is actually feedback-smtp.eu-west-1.amazonses.com with a priority of 10.

  7. Click on SMTP Settings in the sidebar -> Create SMTP credentials -> Create User

  8. Add the password to your .env.prod file (or .env.local for local development):

    .env.prod
    EMAIL_HOST_PASSWORD=[copied_password]
    
  9. And change the following in your settings.py:

    config/settings.py
    EMAIL_HOST = 'email-smtp.[your-region].amazonaws.com'
    EMAIL_HOST_USER = 'AKIA[your-access-key-id]'
    EMAIL_HOST_PASSWORD = config('EMAIL_HOST_PASSWORD', default='')
    DEFAULT_FROM_EMAIL = '[email protected]'
    NOREPLY_EMAIL = '[email protected]'
    EMAIL_PORT = '587'
    

Tip

Verify your DNS records at MxToolbox

Setup - Receiving Emails

  1. Go to Receiving -> Create Route

  2. Set Expression Type to Catch All

  3. Toggle Forward and enter your personal email address

  4. Click Save

  5. Test by sending an email to your new address

AWS SES does not support forwarding emails to a personal email address, you would need to set up a lambda function, etc.

This is why I recommend using Cloudflare's Email Routing to forward emails to your personal email address.

  1. Go to your Domain in Cloudflare

  2. Go to Email -> Email Routing -> Get Started and follow the instructions

  3. After setting up, you can create a catch all rule to forward all emails to your personal email address.
    Routing Rules -> Catch-all address -> Edit:

    • Action = Send to an email
    • Destination = your personal email address
  4. Then toggle the status of the Catch-all rule to Active

Extra: Setup - Sending Emails via Gmail

In the first Setup - Sending Emails we set up sending emails via Mailgun. These emails will be sent in your Django app automation. However, you might want to respond to emails from your users directly - without having to use the API.
You can do this by integrating Gmail with the provider you set up above.

  1. Go to Gmail -> Settings gear ⚙️ -> See all settings
  2. Go to the Accounts and Import tab and click on Add another email address
  3. A pop-up will appear, enter the name and email address from which you want to send emails, uncheck Treat as an alias and click on Next step.

    Gmail - Add another email address

  4. Enter your SMTP credentials from Setup - Sending Emails (EMAIL_HOST, EMAIL_HOST_USER, EMAIL_HOST_PASSWORD) and click on Add account.

    Gmail - Add SMTP credentials

  5. A verification email will be sent to the address you entered, click on the link to verify your address.

  6. You are now ready to send emails via Gmail, you can select the email address you want to send from by clicking on the From mail address in the New message window in Gmail.

Avoiding landing in the spam folder

✅ Configure DNS records:

  • SPF record
  • DKIM record
  • DMARC record

✅ Email content best practices:

  • Include unsubscribe link
  • Use recipient's name
  • Keep formatting simple
  • Link only to trusted sites

❌ What to avoid:

  • Marketing terms like "free" or "earn money"
  • Unusual fonts or layouts
  • Excessive punctuation
  • Links to suspicious sites

Development

While developing you may want to skip sending emails and just log them. Do this by setting EMAIL_BACKEND in your settings.py file to the following:

config/settings.py
EMAIL_BACKEND = "django.core.mail.backends.console.EmailBackend"

The mail sending will be skipped, instead the emails will be displayed in the console.