Prerequisites #
In this tutorial I’m using:
-
A Hetzner Cloud “CAX11” cloud server with 2 (Arm64) CPU cores, 4GB RAM and 40GB disk space.
-
Debian 12 Bookworm as server OS
-
The domain “jueklu.net” that’s hosted on AWS Route 53
Hetzner Cloud #
Firewall Ports #
- Open the following ports for inbound:
Service | Protocol | Port |
Postfix SMTP | TCP | 25 |
Postfix SMTPS | TCP | 465 |
Postfix Submission | TCP | 587 |
Dovecot IMAP | TCP | 143 |
Dovecot IMAPS | TCP | 993 |
Dovecot POP3 | TCP | 110 |
Dovecot POP3S | TCP | 995 |
Dovecot ManageSieve | TCP | 4190 |
HTTP | TCP | 80 |
HTTPS | TCP | 443 |
Port Details
Port Description
SMTP (Simple Mail Transfer Protocol):
- Port 25 (TCP) for SMTP: Receiving emails from other servers
- Port 587 (TCP) for Submission: For secure email submission from email clients
- Port 465 (TCP) for SMTPS: SMTP over SSL, less common but used by some clients
IMAP (Internet Message Access Protocol):
- Port 143 (TCP) for IMAP: Accessing emails from email clients
- Port 993 (TCP) for IMAPS: IMAP over SSL, for secure email access
POP3 (Post Office Protocol version 3):
- Port 110 (TCP) for POP3: Downloading emails from the server
- Port 995 (TCP) for POP3S: POP3 over SSL, for secure email downloading
- Port 4190 (TCP) for managing sieve scripts via the ManageSieve protocol
HTTP/HTTPS (for accessing the mailcow UI and webmail):
- Port 80 (TCP) for HTTP: Web interface
- Port 443 (TCP) for HTTPS: Web interface secure
data:image/s3,"s3://crabby-images/2ff2b/2ff2be6018d1c81585c2eeb10e0dc6fa6f4157d5" alt=""
- Apply the firewall to the cloud server
Reverse DNS Entry #
# Hetzner Cloud: Projects
https://console.hetzner.cloud/projects
- Go to
Networking
>Edit Reverse DNS
data:image/s3,"s3://crabby-images/a02e5/a02e5bcbfaea3a27776d53189e314c0257c178b5" alt=""
Port 25 and 465 #
Per default port 25 and 465 are blocked for the Hetzner cloud servers. They can be opened via the support menu (after the first bill) or after a few days if you send request per mail.
AWS Route 53: DNS Entries #
Go to Route 53
> Hosted zones
A Record (IPv4) #
Add an A record for the subdomain “mail” (mail.jueklu.net) that points to the IPv4 address of the mail server.
# Name Type Value
mail.jueklu.net A 135.181.97.75
data:image/s3,"s3://crabby-images/b958e/b958e194035115025f5195a790da65faf9116685" alt=""
CNAME Records #
Add CNAME records for “autodiscover” and “autoconfig” that point to your “mail” subdomain “mail.jueklu.net.” These are used by email clients for automatic discovery of mail server settings.
# Name Type Value
autodiscover CNAME mail.jueklu.net.
autoconfig CNAME mail.jueklu.net.
data:image/s3,"s3://crabby-images/fd8d6/fd8d67853454e9800bc8efdcc50952948fee09e7" alt=""
MX Record (Mail Exchange) #
Add an MX record that is used for routing emails.
# Name Type Value
(blank) MX 10 mail.jueklu.net.
data:image/s3,"s3://crabby-images/309f5/309f5d1520b53c2a9b35315e77fd1ee534ce8cdf" alt=""
Debian Prerequisites #
Create Swap File #
# Create swapfile
sudo fallocate -l 4G /swapfile
# Change permissions
sudo chmod 600 /swapfile
# Mark the file as swap space
sudo mkswap /swapfile
# Enable the Swapfile
sudo swapon /swapfile
# Add fstab entry
echo '/swapfile none swap sw 0 0' | sudo tee -a /etc/fstab
Install Docker & Docker Compose #
- Install the Docker ARM version for Debian 12
#!/bin/bash
# REF:https://docs.docker.com/engine/install/debian/#install-using-the-repository
# Add Docker's official GPG key:
sudo apt-get update
sudo apt-get install ca-certificates curl gnupg
sudo install -m 0755 -d /etc/apt/keyrings
curl -fsSL https://download.docker.com/linux/debian/gpg | sudo gpg --dearmor -o /etc/apt/keyrings/docker.gpg
sudo chmod a+r /etc/apt/keyrings/docker.gpg
# Add the repository to Apt sources:
echo \
"deb [arch="$(dpkg --print-architecture)" signed-by=/etc/apt/keyrings/docker.gpg] https://download.docker.com/linux/debian \
"$(. /etc/os-release && echo "$VERSION_CODENAME")" stable" | \
sudo tee /etc/apt/sources.list.d/docker.list > /dev/null
sudo apt-get update
sudo apt-get install docker-ce docker-ce-cli containerd.io docker-buildx-plugin docker-compose-plugin -y
Mailcow Setup #
Install Dependencies #
# Update package index
sudo apt update
# Install dependencies
sudo apt install curl git -y
Docker Compose #
- Clone Mailcow repository
# Change directory
cd /opt
# Clone the Mailcow repository
git clone https://github.com/mailcow/mailcow-dockerized
- Generate configuration
# Change directory
cd /opt/mailcow-dockerized
# Generate configuration file
./generate_config.sh
# Mail server hostname (FQDN) - this is not your mail domain, but your mail servers hostname:
mail.jueklu.net # Enter the FQDN
# Timezone [Europe/Vienna]:
Europe/Vienna # Enter timezone
# Available Branches:
# - master branch (stable updates) | default, recommended [1]
# - nightly branch (unstable updates, testing) | not-production ready [2]
# Choose the Branch with it´s number [1/2]
1 # Select master branch
# Detecting if your IP is listed on Spamhaus Bad ASN List...
# Check completed! Your IP is clean
# Check the configuration, optional change HTTPS & HTTPS ports
cat mailcow.conf
- Create / start container
# Create / start Docker stack
sudo docker compose up -d
# Check
docker ps --format '{{.Names}}'
# Shell output:
mailcowdockerized-ipv6nat-mailcow-1
mailcowdockerized-watchdog-mailcow-1
mailcowdockerized-acme-mailcow-1
mailcowdockerized-rspamd-mailcow-1
mailcowdockerized-ofelia-mailcow-1
mailcowdockerized-nginx-mailcow-1
mailcowdockerized-dovecot-mailcow-1
mailcowdockerized-postfix-mailcow-1
mailcowdockerized-php-fpm-mailcow-1
mailcowdockerized-solr-mailcow-1
mailcowdockerized-redis-mailcow-1
mailcowdockerized-mysql-mailcow-1
mailcowdockerized-clamd-mailcow-1
mailcowdockerized-dockerapi-mailcow-1
mailcowdockerized-netfilter-mailcow-1
mailcowdockerized-olefy-mailcow-1
mailcowdockerized-sogo-mailcow-1
mailcowdockerized-unbound-mailcow-1
mailcowdockerized-memcached-mailcow-1
Mailcow Configuration #
Webinterface #
# Access the Mailcow webinterface
https://mail.jueklu.net
# Default user:
admin
# Default PW:
moohoo
Admin PW & 2FA #
Go to System
> Configuration
> Access
-
Click
Edit
to change the admin password -
Click “Two-factor authentication:”
Please select
to add 2FA
data:image/s3,"s3://crabby-images/dfa13/dfa13b51f7961e88341fe48722e2b13c3cbf21e9" alt=""
Add Mail Domain #
Go to E-Mail
> Configuration
-
Select the
Domains
tab -
Click
+ Add domain
data:image/s3,"s3://crabby-images/98a86/98a86659fafe19bc078d02f037f5098ef0d71173" alt=""
-
Add the domain “jueklu.net”
-
Click
Add domain and restart SOGo
data:image/s3,"s3://crabby-images/103ee/103eee7bb32b3be1a15ddbe67b40898469dcf1eb" alt=""
Remainings DNS Entrys #
- Click on
DNS
to open the DNS status for the new domain
data:image/s3,"s3://crabby-images/cf881/cf8810fb5d2b53225d8679042e8587bbcb69b39a" alt=""
- This provides a overview for the missing DNS entries. Copy the text for the DKIM DNS entry:
data:image/s3,"s3://crabby-images/dfdf4/dfdf4ead5d5af55a4a49b9e9e6469e6d9028ed3e" alt=""
DKIM DNS Entry #
-
Record name:
dkim._domainkey
-
Record type:
TXT
-
Value: Copy from Mailcow
data:image/s3,"s3://crabby-images/a6db2/a6db2bd83d49124894594d99d98642beec6696f6" alt=""
Note: On AWS Route 53 it was necessary to split up the DKIM value into two parts in quotation marks, separated by a space. Like “part1” “part2”
For example:
"v=DKIM1; k=rsa;p=MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQCrL1y615QbUJEdHPfOINx" "6tBw4fFJ5YizSa/en5B5JUxpOUDGiFZWSYti4Ed+I0Y3ERUUE"
Otherwise Route 53 shows the following error:
Error occurred
Bad request.
(InvalidChangeBatch 400: CharacterStringTooLong (Value is too long)
DMARC DNS Entry #
-
Record name:
_dmarc
-
Record type:
TXT
# Value: If an email fails the DKIM and SPF tests, mark it as spam:
v=DMARC1; p=quarantine; adkim=s; aspf=s;
data:image/s3,"s3://crabby-images/d021f/d021f0b4deec6278ef8c4e6b59236592f1fc76d3" alt=""
SPF DNS Entry #
data:image/s3,"s3://crabby-images/6e087/6e087e6dad6a4a0624cb95dce08c3d29d729de7b" alt=""
# Add the following entry
v=spf1 a mx -all
# Link
http://www.open-spf.org/SPF_Record_Syntax/
SRV DNS Entry #
# Route 53 SRV Syntax
priority weight port target
# Example
0 0 443 mail.jueklu.net
data:image/s3,"s3://crabby-images/033c7/033c79c590fcfb7406866930ad21dc505db4cf59" alt=""
TLSA DNS Entry #
TLSA is not supported by AWS Route 53.
Check Mailcow DNS #
Open and close the DNS section of the domain, it should look like this:
data:image/s3,"s3://crabby-images/1856d/1856d24ac619c16e734a81afb1a9c74e3a9a85a0" alt=""
Create a Mailbox #
Go to E-Mail
> Configuration
> Mailboxes
- Click
+ Add Mailbox
data:image/s3,"s3://crabby-images/cf71d/cf71d57a9718146c17265012fbb3018f9a9bb5ab" alt=""
-
Define the “username” (username@domain.com)
-
Define a password
-
Click
Add
to create the mailbox
data:image/s3,"s3://crabby-images/822f5/822f58d524461fd153ecb3624529e08e0cc088df" alt=""
Create Alias #
Optional create one or more aliases for the mailbox.
Go to E-Mail
> Configuration
> Alisas
- Click
+ Add alias
data:image/s3,"s3://crabby-images/db924/db9249b628a53b61e457adef469176620da25437" alt=""
data:image/s3,"s3://crabby-images/368c7/368c732ce88c0b6dacafd57668c7b32d49093a0e" alt=""
SOGO Webmail #
# Access the SoGo webinterface
https://mail.jueklu.net/SOGo/
data:image/s3,"s3://crabby-images/3d9b8/3d9b877c5cf52c2995d39dcde5d60fd8677bb244" alt=""
Test Configuration #
https://mail-tester.com
data:image/s3,"s3://crabby-images/f4aea/f4aea619b6a61f19810987e7829dfdb70d1692e7" alt=""
data:image/s3,"s3://crabby-images/a5148/a514856935d05960fe4e5f53686953e8bbcd0ceb" alt=""
data:image/s3,"s3://crabby-images/f16a9/f16a95fc89a23f511c7c24b43c98342bb0bd693c" alt=""
Links #
# Hetzner Cloud: Pricing
https://www.hetzner.com/cloud/
# Hetzner Cloud: Login
https://accounts.hetzner.com/login
# Hetzner Cloud: Server Overview
https://console.hetzner.cloud/projects
# Mailcow Github
https://github.com/mailcow/mailcow-dockerized
# Mailcow Official Documentation
https://docs.mailcow.email/
# Mailcow Official Documentation: Ports
https://docs.mailcow.email/getstarted/prerequisite-system/#default-ports
# Mailcow Official Documentation: DNS
https://docs.mailcow.email/getstarted/prerequisite-dns/#references