website

website.git
git clone git://git.lenczewski.org/website.git
Log | Files | Refs

commit 023094ad7b0f49b77b678667333014310d672501
parent 89630a8ad1882d70aeecb8b51ff92e038226db85
Author: MikoĊ‚aj Lenczewski <mblenczewski@gmail.com>
Date:   Mon, 17 Mar 2025 22:05:50 +0000

Initial selfhosted-mail article

Diffstat:
Asrc/blog/selfhosted-mail.html | 1070+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Asrc/blog/selfhosted-mail.html.title.meta | 1+
Msrc/css/style.css | 16++++++++++++++++
Msrc/css/theme.css | 5+++++
Asrc/res/blog/android-15-cert-settings.png | 0
Asrc/res/blog/mail-layout.png | 0
Asrc/res/blog/mail-layout.svg | 5+++++
Asrc/res/blog/thunderbird-final-setup.png | 0
Asrc/res/blog/thunderbird-imap-setup.png | 0
Asrc/res/blog/thunderbird-smtp-setup.png | 0
Asrc/res/blog/thunderbird-user-setup.png | 0
11 files changed, 1097 insertions(+), 0 deletions(-)

diff --git a/src/blog/selfhosted-mail.html b/src/blog/selfhosted-mail.html @@ -0,0 +1,1070 @@ +<h2>Summary</h2> +<p> + I think that E-Mail is very much taken for granted nowadays, with how easy it + is to get access to, and with how much people use it every day. Now, most people + will interact with email via an online portal created by the email provider, + or via a third-party app that talks with the servers operated by said provider. + I, unfortunately, am not as sane as these people, and so want to bring as much of + the infrastructure I use on a daily basis under my own control as possible. This + post attempts to document my journey of setting up a relatively basic mail server + using Postfix and Dovecot. +</p> + +<hr /> + +<div id="intro"> + <h3>Is this for Me?</h3> + <p> + I would love to say "YES", and claim that setting up a mail server is something + that anyone and everyone can and should be doing. I choose to believe that + having control over your data and the code that operates on said data is + valuable. That does not mean that it is the cheapest, simplest, and most + maintenance-free option available. After all, GMail is free, any web-based + mail is as simple as clicking on a URL, and all serious email providers by + necessity have a hands off experience that a child can use (a more accessible + UI means a wider user-base). + </p> + <p> + I chose to set my own mail server up for two reasons: I want to learn how + mail works, and I want to own my infrastructure. I am able to do this because + I have spare compute (in the form of a Raspberry Pi 4B), and because I am + willing to put in the time required to fix and maintain this system if + it breaks. It is not "mission critical" to me. Therefore if this sounds like + you, and you would also like to set up your own mail server, I would encourage + you to do so! + </p> +</div> + +<hr /> + +<div id="links"> + <h3>Links and References</h3> + <p> + Much of the work I did is built on that of those who have done this thing before. + Below, I have tried to compile a list of the links that I used during my setup: + </p> + <ul> + <li> + <span><a href="https://wiki.archlinux.org/title/Mail_server">Arch Wiki Mail Server</a></span> + overview post. Gives an example of some of the software alternatives you can + use when setting up your own mail server. + </li> + <li> + <span><a href="https://wiki.gentoo.org/wiki/Complete_Virtual_Mail_Server">Gentoo Wiki Virtual Mail Server Article</a></span>, + giving an overview of how to set up a mail server with a web-based management + console, and a database backend. + </li> + <li> + <span><a href="https://www.c0ffee.net/blog/mail-server-guide/">c0ffee's mail server guide</a></span>, + an excellent overview of how to set up postfix and dovecot on FreeBSD. Also + implements some additional services (LDAP for unified user authentication + with the mail server, and some text search / mail filtering services). + </li> + <li> + <span><a href="https://www.postfix.org/TLS_README.html">Postfix TLS documentation</a></span>, + necessary when trying to secure our smtp and submissions services. + </li> + <li> + <span><a href="https://doc.dovecot.org/2.3/configuration_manual/howto/simple_virtual_install/">Minimal Dovecot install</a></span>, + without SSL and using a static configuration for mail users. Not directly + useful (due to the lack of encryption), but the static configuration is + something I tend to prefer. More generally, the <span><a href="https://doc.dovecot.org/2.3/">Dovecot docs</a></span> are + a good resource when trying to fix a broken install (if verbose). + </li> + <li> + <span><a href="https://dmarc.org/overview/">DMARC overview</a></span>, + explains what fields exist in a DMARC record and what they mean. + </li> + <li> + <span><a href="https://arminreiter.com/2022/01/create-your-own-certificate-authority-ca-using-openssl/">Armin Reiter's CA certificate tutorial</a></span>. A simple setup, simply explained. + </li> + <li> + <span><a href="https://jamielinux.com/docs/openssl-certificate-authority/introduction.html">Jaime Nguyen's CA tutorial</a></span>, + for a more complex (and realistic) setup. Also excellently explained! + </li> + </ul> +</div> + +<hr /> + +<div id="theory"> + <h3>What are we Actually Trying to do?</h3> + <p> + So we want to set up our own email, how do we go about this? E-Mail is broken + down into largely three independent parts: + </p> + <ol> + <li>software that sends and receives email - i.e. postfix</li> + <li>software that manages and "delivers email" to an electronic mailbox - i.e. dovecot</li> + <li>software that lets you interact with your mailbox - i.e. your email client</li> + </ol> + + <aside> + <small> + Note: in more formal terminology, the software that sends and receives email is + called a <span><a href="https://en.wikipedia.org/wiki/Message_transfer_agent">mail transfer agent</a></span>, + the software that manages the mailbox (in other words, "delivers" mail to the + mailbox) is called a <span><a href="https://en.wikipedia.org/wiki/Mail_delivery_agent">mail delivery agent</a></span>, + and the software that lets you interact with your own mailbox is called a + <span><a href="https://wiki.archlinux.org/title/Email_client">mail user agent</a></span>. + </small> + </aside> + + <p> + So, in order to have our own mail server, we will need to install a MTA and + configure it such that it can talk to other MTAs and both send mail to and + receive mail from them, we will need to install a MDA to deliver any mail + received by our MTA to our mailbox, and finally we will need to setup our + email client to talk to both the MTA (to allow us to send emails we compose) + and the MDA (to fetch and display the contents of our mailbox). + </p> + <p> + As mentioned above, for our MTA we will install postfix, and for our MDA we + will install dovecot. We choose these because they seem fairly common, and + have decent enough configuration options. For the email client, I personally + use mutt on my desktop and laptops, and thunderbird on my Android phone, and + will show the minimal configuration necessary for those. + </p> + + <h3>Our Configuration</h3> + <p> + Behold, a pictoral representation of our desired setup. I will explain later, + but a picture is worth a thousand words anyway: + </p> + + <div class="container"> + <img src="/res/blog/mail-layout.svg" alt="Our Proposed E-Mail Server Setup" /> + </div> + + <p> + So, explanation time. Postfix is our MTA, and so holds both a SMTP client for + sending our mail to external mail servers, and a SMTP server to receive + incoming mail from external servers and our mail clients. Traditionally, + SMTP servers ran over port 25, port 465, or port 587. Port 25 was used for + unencrypted traffic, port 465 for traffic implicitly encrypted with SSL/TLS, + and port 587 was for initially unencrypted traffic that was upgraded using + the STARTTLS command. We will definitely not use STARTTLS due to it being + an upgrade protocol (hence, starting in cleartext) and having known STARTTLS + stripping attacks (see <a href="https://www.usenix.org/conference/usenixsecurity21/presentation/poddebniak">this presentation</a>, + for example). Additionally, since <a href="https://www.rfc-editor.org/rfc/rfc8314.html#section-1.1">RFC8314</a>, + the IETF has designated TCP port 465 as the "submissions" port for implicit + TLS. But this leaves us with the question of using port 25. + </p> + <p> + In a perfect world, we would encrypt all the things, and it would be cheap, + and no-one could every spy on our traffic, and it would all be sunshine and + rainbows. Unfortunately, we do not live in such a world. As such, we have to + accept that despite client-to-server SMTP traffic over port 465 being encrypted, + server-to-server SMTP traffic over port 25 can at best use STARTTLS. This is + unavoidable, as we very much want other email servers to talk to us. After all, + what use is a mail service that only ever sends email? So, we have to keep + both open. Hence, our setup has SMTP over port 25 secured with STARTTLS, + and submissions over port 465, secured with TLS1.3. + </p> + <aside> + <small> + Note: some online sources will tell you that port 465 is somehow "less secure" + than port 587. How any port can be considered less secure than another one + is anyones guess. Seeing as STARTTLS has known issues, and any TLS version + less than TLS1.3 should not be used with security in mind, I will ignore + such recommendations. + </small> + </aside> + + <p> + Of note is the addition of OpenDKIM as a mail filter. This will allow us to + sign outgoing mail with our DKIM keys, allowing external servers to verify + that our mail comes from the correct domain and reducing the chance it is + marked as spam. This is internal to our mail server, and communication + will be done over a unix socket. + </p> + + <p> + Next, our MDA, dovecot. Dovecot will accept incoming mail received by postfix + over the LMTP protocol (literally "Local Mail Transfer Protocol"), but this + is internal to the mail server. However, it will still have to talk to external + mail clients to send them the contents of our mailbox, which is typically done + using the IMAP(S) protocol. Traditionally, port 143 and port 993 have both + been used with port 143 sometimes being unencrypted and sometimes being + encrypted with TLS, while port 993 had tended to be encrypted with SSL. + Technically, TLS <em>is</em> SSL (<a href="https://tls12.xargs.org/#client-hello/annotated">a newer version</a>, + but still internally treated as SSLv3), and so we will use IMAPS over port 993, + again secured with TLS1.3, to avoid any confusion. + </p> + + <p> + Finally, mail clients. I will not talk too much about this, because this is + as much personal preference as anything else. Note that not every mail client + supports every mail server configuration. In particular, using client certificates + seems unsupported or unimplemented in pretty much every mainstream mail client + I have looked at. Granted, Outlook, Gmail, and Apple's Mail are not a massive + sample size, but I believe that they are representative of what most users + will run. As stated in the introduction, I will only cover my use case, which + is mutt for desktop use, and thunderbird for android for mobile use. Your + mileage may vary on this setup. + </p> +</div> + +<hr /> + +<div id="preparation"> + <h3>Preparation</h3> + <p> + First thing to do is to create a user for our mail system. I personally + dislike running a full-blown database for such small setups, especially on + lower-end devices. We will keep our mail server using static configuration + files as much as possible becase 1) this is easier, 2) it is faster and + less resource hungry than spinning up a database engine, and 3) it gives you + all the flexibility you would need for a personal mail server anyway. We must + also set up the DNS records for our domain, to point to our mail server, and + to set up both DMARC and SPF records. + </p> + + <h4>Creating our vmail User</h4> + <p> + As root, perform the following commands: + </p> + +<pre><code class="language-text">$ groupadd -g 5000 vmail +$ useradd -u 5000 -g 5000 -d /var/vmail -M vmail +$ mkdir /var/vmail +$ chown vmail:vmail /var/vmail +$ chmod 2770 /var/vmail +</code></pre> + + <p> + We need to create our vmail user and group. We don't necessarily want the + <code>/var/vmail</code> directory to include any dotfiles (e.g. <code>.bashrc</code>, + <code>.bash_profile</code>) so we create the home directory ourselves. + Because we make it ourselves, we then need to ensure it is owned by the + vmail user, and set its permissions. Instead of the traditional 0755 permission + octal, we choose 2770: 2 meaning that the directory has the setgid access flag, + the first 7 meaning the owner has read, write, and execute permissions, the + next 7 meaning that all users with the same group have the same permissions, + and the final 0 meaning that any other users have no access. + </p> + + <aside> + <small> + Note: the setgid permission is interesting, in that it allows any user with + access to a given directory to perform operations as though they were in + the same group as the owner of the directory. This is similar to the setuid + permission, where the operations are performed as though the user was the + owner of the directory. + </small> + </aside> + + <h4>Creating our Local CA, and our User Certificates</h4> + <p> + To generate per-user certificates, we need to create a root "CA" certificate. + I use the below script to do this: + </p> + <pre><code class="language-bash">#!/bin/sh + +# TODO: replace this with your domain +DOMAIN="example.com" + +# password for out .p12 PCKS certificate file +PCKSPASS="password" + +# TODO: replace with all services you want certificates for +SERVICES="postfix dovecot" + +# TODO: replace will all users you want personal certificates for +USERS="john" + +set -ex + +# certificate authority +# NOTE: this is not how you should do a proper CA. really, the certificate we +# generate here should be help in an airgapped system, and a secondary +# certificate should be generated and signed by it. that way, this secondary +# certificate can be used to sign service / user certificates, and can be +# easily revoked if necessary (while our original, airgapped certificate +# is safe and cannot easily be leaked). +if [ ! "${REFRESH_CA:-z}" = "z" ]; then + openssl ecparam -genkey -name prime256v1 -out ca.key + openssl req -x509 -new -key ca.key -sha512 -days 9999 -out ca.crt -subj "/CN=$DOMAIN" +fi + +# service certificates +mkdir -p services +for serv in $SERVICES; do + mkdir -p services/$serv + openssl ecparam -genkey -name prime256v1 -noout -out services/$serv/ecc.key + openssl req -new -out services/$serv/ecc.csr -key services/$serv/ecc.key -subj "/CN=$DOMAIN/x500UniqueIdentifier=$serv" + openssl x509 -req -in services/$serv/ecc.csr -CA ca.crt -CAkey ca.key -CAcreateserial -out services/$serv/ecc.crt -sha512 -days 365 + cat services/$serv/ecc.crt services/$serv/ecc.key > services/$serv/ecc.$serv.pem + + if [ ! "${SERVICE_RSA_CERTS:-z}" = "z" ]; then + openssl req -new -nodes -out services/$serv/rsa.csr -keyout services/$serv/rsa.key -newkey rsa:4096 -subj "/CN=$DOMAIN/x500UniqueIdentifier=$serv" + openssl x509 -req -in services/$serv/rsa.csr -CA ca.crt -CAkey ca.key -CAcreateserial -out services/$serv/rsa.crt -sha512 -days 365 + cat services/$serv/rsa.crt services/$serv/rsa.key > services/$serv/rsa.$serv.pem + fi +done + +# user certificates +mkdir -p users +for user in $USERS; do + mkdir -p users/$user + openssl ecparam -genkey -name prime256v1 -noout -out users/$user/ecc.key + openssl req -new -out users/$user/ecc.csr -key users/$user/ecc.key -subj "/CN=$DOMAIN/emailAddress=$user@$DOMAIN/x500UniqueIdentifier=$user" + openssl x509 -req -in users/$user/ecc.csr -CA ca.crt -CAkey ca.key -CAcreateserial -out users/$user/ecc.crt -sha512 -days 365 + cat users/$user/ecc.crt users/$user/ecc.key > users/$user/ecc.$user.pem + + # we generate a pcks12 certificate to allow easy importing under android + openssl pkcs12 -export -out users/$user/ecc.$user.pfx.p12 -in users/$user/ecc.crt -inkey users/$user/ecc.key -passout "pass:$PCKSPASS" +done</code></pre> + + <h4>Creating our DNS Records</h4> + <p> + Our mail server needs to be accessible over the web, and to do so we will + edit our DNS zone on our DNS server (or on our DNS provider's + online gateway). We will provide example records for BIND, but please adapt + these to your chosen DNS server (or, the relevant fields and buttons in your + DNS provider's web UI). + </p> + + <p> + At minimum, we need an MX record that will define a mail server for our + domain, and an A/AAAA record to define the IP address of the mail server. Note + that in the below examples, <code>x.y.z.w</code> refers to an IPv4 address, + and <code>w.z.y.x</code> refers to the reversed form of the same address. + </p> + + <pre><code class="language-text">example.com. IN MX 10 mail.example.com. +mail.example.com. IN A x.y.z.w + +# Optionally, for IPv6 connectivity instead +# mail.example.com. IN AAAA a:b:c:d:e:f:g:h + +# Optionally, can define a CNAME record to point mail.example.com to example.com instead of the A record +# if our mail server is hosted on the same server as example.com +# mail.example.com. IN CNAME example.com.</code></pre> + + <p> + We must define a PTR record that allows a mail server to verify that the + IP address an email is coming from is the one defined to send email (via its + MX record). Not defining this PTR record means that you will be marked as + spam, or simply rejected, by most email providers. + </p> + + <pre><code class="language-text">w.z.y.x.in-addr.arpa. IN PTR example.com + +# If you chose to not alias your mail domain, use this instead +# w.z.y.x.in-addr.arpa. IN PTR mail.example.com</code></pre> + + <p> + The next record we must define is an SPF record for our root domain, to + tell other mail servers how to accept mail for our domain. We want to only + allow mail to be received from mail servers specified in our MX records, + and any other mail should not be accepted (as it was not sent by us, and is + spoofed). + </p> + + <pre><code class="language-text">example.com. IN TXT "v=spf1 mx ~all"</code></pre> + + <p> + After SPF, we must set up our DKIM record, which contains the public key that + will be used by external mail recipients to verify mail sent from our email + server. This is an anti-spam measure enforced by pretty much every mainstream + mail provider, and so is pretty much necessary to set up. Note, that the actual + value for this record will be given to us later when we set up OpenDKIM. + </p> + + <pre><code class="language-text">mail._domainkey IN TXT ( "v=DKIM1; k=rsa; p=my/publickey+signature" )</code></pre> + + <p> + Finally, we must set up DMARC, which is used to tell other mail servers what + to do with email that fails the SPF or DKIM checks (and allows reports of + such failures to be sent to a given address). This is again necessary for + mail coming from our mail server to be marked as authenticated by mail providers. + </p> + + <pre><code class="language-text">_dmarc.example.com. IN TXT "v=DMARC1;p=quarantine;ruf=mailto:forensic@example.com;rua=mailto:aggregate@example.com"</code></pre> +</div> + +<hr /> + +<div id="opendkim"> + <h3>OpenDKIM</h3> + <p> + DKIM (or, DomainKeys Identified Mail), is an anti-spam measure enforced by + most mainstream email providers. It will sign all outgoing mail with a private + key, whose corresponding public key is available via a DNS record. + </p> + <p> + After installing the OpenDKIM package, you will need to modify the config + file, possibly modify the postfix and opendkim users to allow for local + unix socket access, and generate keys for your domain. The following + <span><a href="https://wiki.gentoo.org/wiki/OpenDKIM">webpage</a></span> is + what I followed to get a local unix socket, and this + <span><a href="https://wiki.gentoo.org/wiki/Postfix/DKIM">webpage</a></span> + is what I followed to actually create the keys and configure postfix. Your + mileage may vary depending on your OS! + </p> + <p> + My <code>/etc/opendkim/opendkim.conf</code> file is as follows: + </p> + <pre><code class="language-text"># This is a simple config file for signing and verifying + +#LogWhy yes +Syslog yes +SyslogSuccess yes + +Canonicalization relaxed/relaxed + +Domain example.com +Selector mail +KeyFile /var/lib/opendkim/mail.private + +# To use a local socket instead, specify a path here. The "standard" +# location is under /run/opendkim, and it's best to agree +# on that directory so that various init systems can configure its +# permissions and ownership automatically. +Socket local:/run/opendkim/opendkim.sock + +#Socket inet:8891@localhost + +ReportAddress postmaster@example.com +SendReports yes + +## Hosts to sign email for - 127.0.0.1 is default +## See the OPERATION section of opendkim(8) for more information +# +# InternalHosts 192.168.0.0/16, 10.0.0.0/8, 172.16.0.0/12 + +## For secondary mailservers - indicates not to sign or verify messages +## from these hosts +# +# PeerList X.X.X.X + +# PidFile /run/opendkim.pid + +# The UMask is really only used for the PID file (root:root) and the +# local UNIX socket, if you're using one. It should be 0117 for the +# socket. +UMask 0117 +UserID opendkim + +# For use with unbound +#TrustAnchorFile /etc/dnssec/root-anchors.txt</code></pre> +</div> + +<hr /> + +<div id="postfix"> + <h3>Postfix</h3> + <p> + Setting up postfix is relatively straightforward. Install the postfix package + for your relevant distro. In my case, I enabled lmdb support to provide whatever + "database" services postfix needed without resorting to a more heavyweight + database engine. + </p> + <p> + After installing postfix, we need to provide it two configuration files, a + <code>/etc/postfix/main.cf</code> file defines settings for postfix itself, + and a <code>/etc/postfix/master.cf</code> file that defines what services + postfix provides (i.e. smtp(s), submission(s)). + </p> + <p> + For informing postfix of the local email users (and their aliases!) we must + first generate an "aliases" file. I chose to place this at + <code>/etc/mail/aliases</code>. One potential aliases file can look as + follows: + </p> + <pre><code class="language-text"># Basic system aliases -- these MUST be present. +MAILER-DAEMON: postmaster +postmaster: root + +# General redirections for pseudo accounts. +adm: root +bin: root +daemon: root +exim: root +lp: root +mail: root +named: root +nobody: root +postfix: root + +# Well-known aliases -- these should be filled in! +root: john+root +operator: john+operator + +# dmarc report address +report: john+report + +# Standard RFC2142 aliases +abuse: postmaster +ftp: root +hostmaster: root +news: usenet +noc: root +security: root +usenet: root +uucp: root +webmaster: root +www: webmaster + +# trap decode to catch security attacks +decode: /dev/null + +# our actual users and their aliases +john: john</code></pre> + <p> + After creating this file for the first time, make sure to run the following + command: + </p> + <pre><code class="language-text">$ newalias /etc/mail/aliases</code></pre> + <p> + Whenever you update the aliases file, you need to rerun the following, + slightly different command: + </p> + <pre><code class="language-text">$ postalias /etc/mail/aliases</code></pre> + <p> + My <code>/etc/postfix/main.cf</code> file is as follows: + </p> + <pre><code class="language-text">mail_owner = postfix + +# TODO: change these to your real domain +myhostname = mail.example.com +mydomain = example.com + +myorigin = $mydomain + +#inet_interfaces = $myhostname, localhost +inet_interfaces = 192.168.2.101, localhost +inet_protocols = ipv4 + +mydestination = $myhostname, $mydomain, localhost.$mydomain, localhost + +# our local user aliases we generated earlier +alias_maps = lmdb:/etc/mail/aliases +local_recipient_maps = $alias_maps + +address_verify_map = lmdb:$data_directory/verify_cache + +# reject mail +unknown_local_recipient_reject_code = 550 + +mynetworks_style = host +# OR +#mynetworks = 192.168.2.0/24, 127.0.0.0/8 + +recipient_delimiter = + + +virtual_uid_maps = static:980 +virtual_gid_maps = static:980 +virtual_mailbox_base = /var/vmail + +# disable "new mail" notification for local users +biff = no + +# reduce information given to snoopers, "NO UCE" can help with reducing spam +smtpd_banner = $myhostname ESMTP NO UCE + +# require email addresses of form '<user>@<domain>.<tld>' +allow_percent_hack = no +swap_bangpath = no + +# dont allow spammers to use vrfy to scrape valid mail users +disable_vrfy_command = yes + +# dont give information if mailbox does not exist +show_user_unknown_table_name = no + +# maximum mta message size: 32 MiB +# NOTE: we also set the maximum mailbox size to ensure we can receive the +# mta message, but otherwise we don't care about it because we will use +# dovecot for writing the mailbox +message_size_limit = 33554432 +mailbox_size_limit = 33554432 + +# opendkim +smtpd_milters = unix:/run/opendkim/opendkim.sock +non_smtpd_milters = $smtpd_milters + +# lmtp, we will set up dovecot in a second +mailbox_transport = lmtp:unix:private/dovecot-lmtp + +# optional tls for sending mail, with optional DNSSEC verification +smtp_tls_security_level = dane +smtp_dns_support_level = dnssec +# OR +# optional tls for sending mail +#smtp_tls_security_level = may + +# tls certificates to use when sending out mail +# TODO: replace these with your ssl certificates +smtp_tls_CAfile = /etc/ssl/certs/ca-certificates.crt +smtp_tls_eccert_file = /root/certs/example.com/fullchain.pem +smtp_tls_eckey_file = /root/certs/example.com/privkey.pem + +# tls settings for receiving mail, realistically this could be dropped to TLSv1.2 +smtpd_tls_mandatory_protocols = TLSv1.3 +smtpd_tls_mandatory_ciphers = high +tls_ssl_options = no_ticket, no_compression + +# when receiving mail, require sender to use encryption +smtpd_tls_security_level = encrypt + +# tls certificates for receiving mail +# TODO: replace these with your ssl certificates +smtpd_tls_CAfile = /etc/ssl/certs/ca-certificates.crt +smtpd_tls_eccert_file = /root/certs/example.com/fullchain.pem +smtpd_tls_eckey_file = /root/certs/example.com/privkey.pem + +smtp_tls_loglevel = 1 +smtpd_tls_loglevel = 1 + +#smtpd_tls_received_header = yes +smtpd_tls_auth_only = yes + +# require that mail servers identify themselves to try to reduce spam +smtpd_helo_required = yes + +# sasl, again we will set up dovecot in a second +smtpd_sasl_type = dovecot +smtpd_sasl_path = private/auth + +smtpd_sasl_local_domain=$mydomain + +broken_sasl_auth_clients = no + +smtpd_sasl_auth_enable = yes +smtpd_sasl_authenticated_header = yes + +# don't allow plaintext auth methods on unencrypted connections +smtpd_sasl_security_options = noanonymous, noplaintext + +# ... but plaintext auth is fine when using TLS +smtpd_sasl_tls_security_options = noanonymous + +# mail restrictions, to limit spam and unauthorised connections + +smtpd_client_restrictions = permit_mynetworks, + permit_sasl_authenticated, + reject_unknown_reverse_client_hostname, + reject_unauth_pipelining + +smtpd_helo_restrictions = permit_mynetworks, + permit_sasl_authenticated, + reject_invalid_helo_hostname, + reject_non_fqdn_helo_hostname, + reject_unauth_pipelining + +smtpd_sender_restrictions = permit_mynetworks, + permit_sasl_authenticated, + reject_non_fqdn_sender, + reject_unknown_sender_domain, + reject_unauth_pipelining + +smtpd_recipient_restrictions = permit_mynetworks, + permit_sasl_authenticated, + reject_non_fqdn_recipient, + reject_unknown_recipient_domain, + reject_unauth_pipelining, + reject_unverified_recipient + #reject_rbl_client zen.spamhaus.org, + #reject_rbl_client bl.smapcop.net + +smtpd_relay_restrictions = permit_mynetworks, + permit_sasl_authenticated, + reject_unauth_destination + +smtpd_data_restrictions = permit_mynetworks, + permit_sasl_authenticated, + reject_multi_recipient_bounce, + reject_unauth_pipelining</code></pre> + <p> + My <code>/etc/postfix/master.cf</code> file is as follows: + </p> + <pre><code class="language-text"># +# Postfix master process configuration file. For details on the format +# of the file, see the master(5) manual page (command: "man 5 master" or +# on-line: http://www.postfix.org/master.5.html). +# +# Do not forget to execute "postfix reload" after editing this file. +# +# ========================================================================== +# service type private unpriv chroot wakeup maxproc command + args +# (yes) (yes) (no) (never) (100) +# ========================================================================== +smtp inet n - n - - smtpd + -o smtpd_sasl_auth_enable=no + -o smtpd_tls_ask_ccert=yes + +#submission inet n - n - - smtpd +# -o syslog_name=postfix/submission +# -o smtpd_tls_req_ccert=yes + +submissions inet n - n - - smtpd + -o syslog_name=postfix/submissions + -o smtpd_tls_wrappermode=yes + -o smtpd_tls_req_ccert=yes + +pickup unix n - n 60 1 pickup +cleanup unix n - n - 0 cleanup +qmgr unix n - n 300 1 qmgr +#qmgr unix n - n 300 1 oqmgr +tlsmgr unix - - n 1000? 1 tlsmgr +rewrite unix - - n - - trivial-rewrite +bounce unix - - n - 0 bounce +defer unix - - n - 0 bounce +trace unix - - n - 0 bounce +verify unix - - n - 1 verify +flush unix n - n 1000? 0 flush +proxymap unix - - n - - proxymap +proxywrite unix - - n - 1 proxymap +smtp unix - - n - - smtp +relay unix - - n - - smtp + -o syslog_name=${multi_instance_name?{$multi_instance_name}:{postfix}}/$service_name +# -o smtp_helo_timeout=5 -o smtp_connect_timeout=5 +showq unix n - n - - showq +error unix - - n - - error +retry unix - - n - - error +discard unix - - n - - discard +local unix - n n - - local +virtual unix - n n - - virtual +lmtp unix - - n - - lmtp +anvil unix - - n - 1 anvil +scache unix - - n - 1 scache +postlog unix-dgram n - n - 1 postlogd</code></pre> +</div> + +<hr /> + +<div id="dovecot"> + <h3>Dovecot</h3> + <p> + Dovecot is similarly straightforward to install and configure. After installing + the dovecot package, we need to provide it with a single config file. However, + since we want to have a simple password file as well, we must create that + first. + </p> + <p> + I chose to place my mail users' password file at <code>/etc/mail/users</code>. + The contents of this file are similar to <code>/etc/shadow</code>, and + user passwords can be added to it using the following command: + </p> + <pre><code class="language-text">$ echo "john:$(doveadm pw -p yourpassword)::::::" >> /etc/mail/users</code></pre> + <p> + Finally, we can finally create our final dovecot config. + </p> + <p> + My <code>/etc/dovecot/dovecot.conf</code> is as follows: + </p> + <pre><code class="language-text">protocols = imap lmtp + +# debugging help +#auth_verbose = yes +#auth_debug = yes +#mail_debug = yes +#verbose_ssl = yes + +log_path = /var/log/dovecot.log + +dict { + #quota = mysql:/etc/dovecot/dovecot-dict-sql.conf.ext +} + +# auth +# ============================================================================ + +disable_plaintext_auth = yes + +auth_username_chars = abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ01234567890.-_@ + +auth_username_format = %Ln +auth_anonymous_username = anonymous + +auth_mechanisms = plain login + +# to support smtp clients who cannot present client certificates, but require +# them for imap +protocol !smtp { + auth_ssl_require_client_cert = yes +} + +# we will get the username from whatever client certificate was presented +auth_ssl_username_from_cert = yes + +# NOTE: using auth cache means we MUST reload dovecot after changing passwords +# NOTE: using auth cache requires different passdb driver +#auth_cache_size = 10M +#auth_cache_ttl = 300s +#auth_cache_negative_ttl = 60s +#auth_cache_verify_password_with_worker = yes + +# lets use a plaintext password file. really, we could omit this and only require +# the client certificates, but this is a trivial additional layer of security that +# is easy enough to swap out and might buy you some time if the certificate gets +# leaked +passdb { + driver = passwd-file + args = scheme=CRYPT username_format=%Ln /etc/mail/users + # args = cache_key=%u:%w +} + +# we must use our vmail user to store all mail for users +userdb { + driver = static + args = uid=vmail gid=vmail home=/var/vmail/%Ln +# default_fields = quota_rule=*:storage=1G +} + +# ssl +# ============================================================================ + +ssl = required + +# TODO: swap these out with your own tls certificates +ssl_cert = &lt;/root/certs/example.com/fullchain.pem +ssl_key = &lt;/root/certs/example.com/privkey.pem + +# NOTE: this is the root ca for our user certificates +ssl_ca = &lt;/root/ca/ca.crt + +# TODO: do we actually care about verifying that our user certificates expired? +ssl_require_crl = no + +ssl_client_require_valid_cert = yes +ssl_verify_client_cert = yes + +# which certificate field to get the username from +#ssl_cert_username_field = commonName +#ssl_cert_username_field = emailAddress +ssl_cert_username_field = x500UniqueIdentifier + +ssl_min_protocol = TLSv1.3 +ssl_cipher_list = ECDHE-ECDSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-ECDSA-CHACHA20-POLY1305:ECDHE-RSA-CHACHA20-POLY1305:ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES256-SHA384:ECDHE-RSA-AES256-SHA384:ECDHE-ECDSA-AES128-SHA256:ECDHE-RSA-AES128-SHA256 +ssl_curve_list = X25519:prime256v1:secp384r1 +ssl_prefer_server_ciphers = yes +ssl_options = no_ticket, no_compression + +# mail +# ============================================================================ + +mail_location = maildir:~/.maildir + +namespace inbox { + inbox = yes + separator = / +} + +mailbox_idle_check_interval = 30 secs + +# master +# ============================================================================ + +service imap-login { + # NOTE: we do not expose our unsecured imap listener to our local (or wider) network + inet_listener imap { + address = 127.0.0.1, ::1 + port = 143 + } + + inet_listener imaps { + port = 993 + ssl = yes + } + + #service_count = 1 + #process_min_avail = 0 + #vsz_limit = $default_vsz_limit +} + +service imap { + #vsz_limit = $default_vsz_limit + #process_limit = 1024 +} + +# here is the lmtp listener where postfix will deliver mail to +service lmtp { + unix_listener /var/spool/postfix/private/dovecot-lmtp { + mode = 0666 + user = postfix + group = postfix + } +} + +# here is where postfix will attempt to login via sasl +service auth { + # Postfix smtp-auth + unix_listener /var/spool/postfix/private/auth { + mode = 0666 + user = postfix + group = postfix + } +} + +service auth-worker { + #user = root +} + +# mailboxes +# ============================================================================ + +namespace inbox { + # These mailboxes are widely used and could perhaps be created automatically: + mailbox Archive { + special_use = \Archive + auto = subscribe + } + mailbox Drafts { + special_use = \Drafts + auto = subscribe + } + mailbox Junk { + special_use = \Junk + auto = subscribe + } + mailbox Trash { + special_use = \Trash + auto = subscribe + } + + # For \Sent mailboxes there are two widely used names. We'll mark both of + # them as \Sent. User typically deletes one of them if duplicates are created. + mailbox Sent { + special_use = \Sent + auto = subscribe + } + mailbox "Sent Messages" { + special_use = \Sent + } + + # If you have a virtual "All messages" mailbox: + #mailbox virtual/All { + # special_use = \All + # comment = All my messages + #} + + # If you have a virtual "Flagged" mailbox: + #mailbox virtual/Flagged { + # special_use = \Flagged + # comment = All my flagged messages + #} + + # If you have a virtual "Important" mailbox: + #mailbox virtual/Important { + # special_use = \Important + # comment = All my important messages + #} +}</code></pre> +</div> + +<hr /> + +<div id="mutt"> + <h3>Configuring Mutt</h3> + <p> + I won't dwell too much on this, but the core setup that is necessary is + the snippet below. Whether you include this directly in your + <code>.muttrc</code>, or add this snippet in a per-user config, is up to + you! + </p> + <pre><code class="language-text"># TODO: replace this with wherever you store the user certificate we generated +# earlier! +set ssl_client_cert=/home/john/.config/mutt/ecc.john.pem + +# IMAP +# ============================================================================ + +set folder = imaps://mail.example.com/ + +set spoolfile = +INBOX +set postponed = +Drafts +set record = +Sent + +# here you should add all mailboxes you added to your dovecot configuration +mailboxes = +INBOX +Drafts +Sent + +# NOTE: this is not necessary, because we get our username from our client +# certificate. however, include it if you dont want to use such certificates +set imap_user = john + +# allow Mutt to open a new IMAP connection automatically +unset imap_passive + +# keep the IMAP connection alive by polling intermittently (in seconds) +set imap_keepalive = 300 + +# how often to check for new mail (in seconds) +set mail_check = 120 + +# SMTP +# ============================================================================ + +set smtp_authenticators = "login" + +set smtp_url = smtps://$imap_user@mail.example.com/ + +set realname = "John Doe" +set from = "john@example.com"</code></pre> +</div> + +<hr /> + +<div id="thunderbird-android"> + <h3>Configuring Thunderbird on Android</h3> + <p> + All that is required for setting up thunderbird is to first install the + previously generated CA certificate (or intermediate CA certificate, if using + a more proper approach) on your android device. Then, your user certificate + must also be installed, and has to be in the PCKS12 format for this: + </p> + <p> + To install both CA and user certificates, go to your android settings, and + search for "certificates". This page should show up. First install your CA + certificate (must be in .pem format), and then install your user certificate + (must be in .pfx.p12 format): + </p> + <div class="container"> + <img src="/res/blog/android-15-cert-settings.png" alt="Android 15 Certificate Settings Page" /> + </div> + <p> + After installing your certificates, continue with setting up thunderbird as + normal. Add a new account. Note that Thunderbird will not be able to lookup + your server's configuration the first time you press "Next", and will complain. + Just press "Next" again in this case to enter manual configuration: + </p> + <div class="container"> + <img src="/res/blog/thunderbird-user-setup.png" alt="Thunderbird User Setup" /> + </div> + <p> + Configure your "incoming", or imap, settings. This will allow you to read + mail sent to you. Make sure to your a "normal password", not any of the other + options, and select the user certificate we installed earlier: + </p> + <div class="container"> + <img src="/res/blog/thunderbird-imap-setup.png" alt="Thunderbird IMAP Settings" /> + </div> + <p> + Configure your "outgoing", or smtp, settings. This will allow you to send + mail. Make sure to again use "normal password", and provide the user certificate: + </p> + <div class="container"> + <img src="/res/blog/thunderbird-smtp-setup.png" alt="Thunderbird SMTP Settings" /> + </div> + <p> + Finally, enter your email address and choose what real name to give (and + optionally, what signature to have in all outgoing emails): + </p> + <div class="container"> + <img src="/res/blog/thunderbird-final-setup.png" alt="Thunderbird Mail Settings" /> + </div> + <p> + After this, your thunderbird client should be set up and working! + </p> +</div> + +<hr /> + +<div id="conclusion"> + <h3>Conclusion</h3> + <p> + That was how to set up a mail server. Hopefully this was useful to you! + </p> +</div> diff --git a/src/blog/selfhosted-mail.html.title.meta b/src/blog/selfhosted-mail.html.title.meta @@ -0,0 +1 @@ +Self-Hosted Mail with Postfix and Dovecot diff --git a/src/css/style.css b/src/css/style.css @@ -25,6 +25,15 @@ hr { padding: 1em; } +.container img { + max-height: 40em; + max-width: 100%; + + display: block; + margin-left: auto; + margin-right: auto; +} + .navbar { padding-left: 1em; padding-right: 1em; @@ -76,3 +85,10 @@ hr { opacity: 1; transition: opacity .2s; } + +.code-title { + font-family: sans-serif; + font-size: 80%; + font-weight: bold; + margin-bottom: .8em; +} diff --git a/src/css/theme.css b/src/css/theme.css @@ -69,3 +69,8 @@ h2 { h3, h4, h5, h6 { color: var(--theme-h3); } + + +.code-title { + border-bottom: 1px solid var(--theme-text); +} diff --git a/src/res/blog/android-15-cert-settings.png b/src/res/blog/android-15-cert-settings.png Binary files differ. diff --git a/src/res/blog/mail-layout.png b/src/res/blog/mail-layout.png Binary files differ. diff --git a/src/res/blog/mail-layout.svg b/src/res/blog/mail-layout.svg @@ -0,0 +1,4 @@ +<?xml version="1.0" encoding="UTF-8"?> +<!-- Do not edit this file with editors other than draw.io --> +<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> +<svg xmlns="http://www.w3.org/2000/svg" style="background: #ffffff; background-color: light-dark(#ffffff, var(--ge-dark-color, #121212)); color-scheme: light dark;" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1" width="523px" height="361px" viewBox="-0.5 -0.5 523 361" content="&lt;mxfile host=&quot;app.diagrams.net&quot; agent=&quot;Mozilla/5.0 (X11; Linux x86_64; rv:135.0) Gecko/20100101 Firefox/135.0&quot; version=&quot;26.0.16&quot; scale=&quot;1&quot; border=&quot;0&quot;&gt;&#xA; &lt;diagram name=&quot;Page-1&quot; id=&quot;n5D40TtwpLpbVFWp0QJ_&quot;&gt;&#xA; &lt;mxGraphModel dx=&quot;1232&quot; dy=&quot;1130&quot; grid=&quot;1&quot; gridSize=&quot;10&quot; guides=&quot;1&quot; tooltips=&quot;1&quot; connect=&quot;1&quot; arrows=&quot;1&quot; fold=&quot;1&quot; page=&quot;1&quot; pageScale=&quot;1&quot; pageWidth=&quot;850&quot; pageHeight=&quot;1100&quot; math=&quot;0&quot; shadow=&quot;0&quot;&gt;&#xA; &lt;root&gt;&#xA; &lt;mxCell id=&quot;0&quot; /&gt;&#xA; &lt;mxCell id=&quot;1&quot; parent=&quot;0&quot; /&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-1&quot; value=&quot;Postfix&quot; style=&quot;rounded=0;whiteSpace=wrap;html=1;&quot; parent=&quot;1&quot; vertex=&quot;1&quot;&gt;&#xA; &lt;mxGeometry x=&quot;240&quot; y=&quot;320&quot; width=&quot;120&quot; height=&quot;40&quot; as=&quot;geometry&quot; /&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-2&quot; value=&quot;Dovecot&quot; style=&quot;rounded=0;whiteSpace=wrap;html=1;&quot; parent=&quot;1&quot; vertex=&quot;1&quot;&gt;&#xA; &lt;mxGeometry x=&quot;240&quot; y=&quot;440&quot; width=&quot;120&quot; height=&quot;40&quot; as=&quot;geometry&quot; /&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-3&quot; value=&quot;&quot; style=&quot;endArrow=none;dashed=1;html=1;dashPattern=1 3;strokeWidth=2;rounded=0;&quot; parent=&quot;1&quot; edge=&quot;1&quot;&gt;&#xA; &lt;mxGeometry width=&quot;50&quot; height=&quot;50&quot; relative=&quot;1&quot; as=&quot;geometry&quot;&gt;&#xA; &lt;mxPoint x=&quot;40&quot; y=&quot;240&quot; as=&quot;sourcePoint&quot; /&gt;&#xA; &lt;mxPoint x=&quot;560&quot; y=&quot;240&quot; as=&quot;targetPoint&quot; /&gt;&#xA; &lt;/mxGeometry&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-4&quot; value=&quot;&quot; style=&quot;endArrow=classic;html=1;rounded=0;entryX=0.5;entryY=0;entryDx=0;entryDy=0;exitX=0.5;exitY=1;exitDx=0;exitDy=0;exitPerimeter=0;startArrow=classic;startFill=1;&quot; parent=&quot;1&quot; target=&quot;bQRnDDqhK_hAcH7NuNX_-1&quot; edge=&quot;1&quot; source=&quot;bQRnDDqhK_hAcH7NuNX_-20&quot;&gt;&#xA; &lt;mxGeometry width=&quot;50&quot; height=&quot;50&quot; relative=&quot;1&quot; as=&quot;geometry&quot;&gt;&#xA; &lt;mxPoint x=&quot;260&quot; y=&quot;200&quot; as=&quot;sourcePoint&quot; /&gt;&#xA; &lt;mxPoint x=&quot;310&quot; y=&quot;150&quot; as=&quot;targetPoint&quot; /&gt;&#xA; &lt;/mxGeometry&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-8&quot; value=&quot;&amp;lt;div&amp;gt;smtp&amp;lt;/div&amp;gt;&amp;lt;div&amp;gt;tcp:25&amp;lt;/div&amp;gt;&quot; style=&quot;edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];&quot; parent=&quot;bQRnDDqhK_hAcH7NuNX_-4&quot; vertex=&quot;1&quot; connectable=&quot;0&quot;&gt;&#xA; &lt;mxGeometry x=&quot;0.6333&quot; relative=&quot;1&quot; as=&quot;geometry&quot;&gt;&#xA; &lt;mxPoint y=&quot;-18&quot; as=&quot;offset&quot; /&gt;&#xA; &lt;/mxGeometry&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-6&quot; value=&quot;&quot; style=&quot;endArrow=classic;html=1;rounded=0;exitX=0.5;exitY=1;exitDx=0;exitDy=0;entryX=0.5;entryY=0;entryDx=0;entryDy=0;&quot; parent=&quot;1&quot; source=&quot;bQRnDDqhK_hAcH7NuNX_-1&quot; target=&quot;bQRnDDqhK_hAcH7NuNX_-2&quot; edge=&quot;1&quot;&gt;&#xA; &lt;mxGeometry width=&quot;50&quot; height=&quot;50&quot; relative=&quot;1&quot; as=&quot;geometry&quot;&gt;&#xA; &lt;mxPoint x=&quot;400&quot; y=&quot;610&quot; as=&quot;sourcePoint&quot; /&gt;&#xA; &lt;mxPoint x=&quot;450&quot; y=&quot;560&quot; as=&quot;targetPoint&quot; /&gt;&#xA; &lt;/mxGeometry&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-7&quot; value=&quot;&amp;lt;div&amp;gt;lmtp&amp;lt;/div&amp;gt;&amp;lt;div&amp;gt;unix:/var/spool/postfix/postfix-private/dovecot-lmtp&amp;lt;/div&amp;gt;&quot; style=&quot;edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];&quot; parent=&quot;bQRnDDqhK_hAcH7NuNX_-6&quot; vertex=&quot;1&quot; connectable=&quot;0&quot;&gt;&#xA; &lt;mxGeometry x=&quot;-0.1&quot; y=&quot;3&quot; relative=&quot;1&quot; as=&quot;geometry&quot;&gt;&#xA; &lt;mxPoint x=&quot;-3&quot; y=&quot;4&quot; as=&quot;offset&quot; /&gt;&#xA; &lt;/mxGeometry&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-9&quot; value=&quot;OpenDKIM&quot; style=&quot;rounded=1;whiteSpace=wrap;html=1;&quot; parent=&quot;1&quot; vertex=&quot;1&quot;&gt;&#xA; &lt;mxGeometry x=&quot;120&quot; y=&quot;440&quot; width=&quot;80&quot; height=&quot;40&quot; as=&quot;geometry&quot; /&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-11&quot; value=&quot;&quot; style=&quot;endArrow=classic;html=1;rounded=0;exitX=0;exitY=0.5;exitDx=0;exitDy=0;entryX=0.5;entryY=0;entryDx=0;entryDy=0;startArrow=classic;startFill=1;&quot; parent=&quot;1&quot; source=&quot;bQRnDDqhK_hAcH7NuNX_-1&quot; target=&quot;bQRnDDqhK_hAcH7NuNX_-9&quot; edge=&quot;1&quot;&gt;&#xA; &lt;mxGeometry width=&quot;50&quot; height=&quot;50&quot; relative=&quot;1&quot; as=&quot;geometry&quot;&gt;&#xA; &lt;mxPoint x=&quot;400&quot; y=&quot;610&quot; as=&quot;sourcePoint&quot; /&gt;&#xA; &lt;mxPoint x=&quot;450&quot; y=&quot;560&quot; as=&quot;targetPoint&quot; /&gt;&#xA; &lt;Array as=&quot;points&quot;&gt;&#xA; &lt;mxPoint x=&quot;160&quot; y=&quot;340&quot; /&gt;&#xA; &lt;/Array&gt;&#xA; &lt;/mxGeometry&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-12&quot; value=&quot;unix:/var/run/opendkim/opendkim.sock&quot; style=&quot;edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];&quot; parent=&quot;bQRnDDqhK_hAcH7NuNX_-11&quot; vertex=&quot;1&quot; connectable=&quot;0&quot;&gt;&#xA; &lt;mxGeometry x=&quot;0.5143&quot; y=&quot;1&quot; relative=&quot;1&quot; as=&quot;geometry&quot;&gt;&#xA; &lt;mxPoint x=&quot;-1&quot; y=&quot;-16&quot; as=&quot;offset&quot; /&gt;&#xA; &lt;/mxGeometry&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-15&quot; value=&quot;&amp;lt;div&amp;gt;User&amp;#39;s&amp;lt;/div&amp;gt;&amp;lt;div&amp;gt;Mail Client&amp;lt;/div&amp;gt;&quot; style=&quot;rounded=0;whiteSpace=wrap;html=1;&quot; parent=&quot;1&quot; vertex=&quot;1&quot;&gt;&#xA; &lt;mxGeometry x=&quot;400&quot; y=&quot;140&quot; width=&quot;120&quot; height=&quot;40&quot; as=&quot;geometry&quot; /&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-20&quot; value=&quot;&amp;lt;div&amp;gt;External&amp;lt;/div&amp;gt;&amp;lt;div&amp;gt;Mail Servers&amp;lt;/div&amp;gt;&quot; style=&quot;ellipse;shape=cloud;whiteSpace=wrap;html=1;&quot; parent=&quot;1&quot; vertex=&quot;1&quot;&gt;&#xA; &lt;mxGeometry x=&quot;240&quot; y=&quot;120&quot; width=&quot;120&quot; height=&quot;80&quot; as=&quot;geometry&quot; /&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-21&quot; value=&quot;&quot; style=&quot;endArrow=classic;html=1;rounded=0;exitX=0.25;exitY=1;exitDx=0;exitDy=0;entryX=1;entryY=0.5;entryDx=0;entryDy=0;&quot; parent=&quot;1&quot; source=&quot;bQRnDDqhK_hAcH7NuNX_-15&quot; target=&quot;bQRnDDqhK_hAcH7NuNX_-1&quot; edge=&quot;1&quot;&gt;&#xA; &lt;mxGeometry width=&quot;50&quot; height=&quot;50&quot; relative=&quot;1&quot; as=&quot;geometry&quot;&gt;&#xA; &lt;mxPoint x=&quot;400&quot; y=&quot;610&quot; as=&quot;sourcePoint&quot; /&gt;&#xA; &lt;mxPoint x=&quot;450&quot; y=&quot;560&quot; as=&quot;targetPoint&quot; /&gt;&#xA; &lt;Array as=&quot;points&quot;&gt;&#xA; &lt;mxPoint x=&quot;430&quot; y=&quot;340&quot; /&gt;&#xA; &lt;/Array&gt;&#xA; &lt;/mxGeometry&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-24&quot; value=&quot;&amp;lt;div&amp;gt;submissions&amp;lt;/div&amp;gt;&amp;lt;div&amp;gt;tcp:465&amp;lt;/div&amp;gt;&quot; style=&quot;edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];&quot; parent=&quot;bQRnDDqhK_hAcH7NuNX_-21&quot; vertex=&quot;1&quot; connectable=&quot;0&quot;&gt;&#xA; &lt;mxGeometry x=&quot;-0.2231&quot; y=&quot;-1&quot; relative=&quot;1&quot; as=&quot;geometry&quot;&gt;&#xA; &lt;mxPoint y=&quot;11&quot; as=&quot;offset&quot; /&gt;&#xA; &lt;/mxGeometry&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-22&quot; value=&quot;&quot; style=&quot;endArrow=classic;html=1;rounded=0;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=0.75;entryY=1;entryDx=0;entryDy=0;&quot; parent=&quot;1&quot; source=&quot;bQRnDDqhK_hAcH7NuNX_-2&quot; target=&quot;bQRnDDqhK_hAcH7NuNX_-15&quot; edge=&quot;1&quot;&gt;&#xA; &lt;mxGeometry width=&quot;50&quot; height=&quot;50&quot; relative=&quot;1&quot; as=&quot;geometry&quot;&gt;&#xA; &lt;mxPoint x=&quot;400&quot; y=&quot;610&quot; as=&quot;sourcePoint&quot; /&gt;&#xA; &lt;mxPoint x=&quot;450&quot; y=&quot;560&quot; as=&quot;targetPoint&quot; /&gt;&#xA; &lt;Array as=&quot;points&quot;&gt;&#xA; &lt;mxPoint x=&quot;490&quot; y=&quot;460&quot; /&gt;&#xA; &lt;/Array&gt;&#xA; &lt;/mxGeometry&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;mxCell id=&quot;bQRnDDqhK_hAcH7NuNX_-23&quot; value=&quot;&amp;lt;div&amp;gt;imaps&amp;lt;/div&amp;gt;&amp;lt;div&amp;gt;tcp:993&amp;lt;/div&amp;gt;&quot; style=&quot;edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];&quot; parent=&quot;bQRnDDqhK_hAcH7NuNX_-22&quot; vertex=&quot;1&quot; connectable=&quot;0&quot;&gt;&#xA; &lt;mxGeometry x=&quot;0.4895&quot; y=&quot;1&quot; relative=&quot;1&quot; as=&quot;geometry&quot;&gt;&#xA; &lt;mxPoint y=&quot;-5&quot; as=&quot;offset&quot; /&gt;&#xA; &lt;/mxGeometry&gt;&#xA; &lt;/mxCell&gt;&#xA; &lt;/root&gt;&#xA; &lt;/mxGraphModel&gt;&#xA; &lt;/diagram&gt;&#xA;&lt;/mxfile&gt;&#xA;"><defs/><rect fill="#ffffff" style="fill: light-dark(#ffffff, var(--ge-dark-color, #121212));" width="100%" height="100%" x="0" y="0"/><g><g data-cell-id="0"><g data-cell-id="1"><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-1"><g><rect x="201" y="200" width="120" height="40" fill="#ffffff" style="fill: light-dark(#ffffff, var(--ge-dark-color, #121212)); stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke="#000000" pointer-events="all"/></g><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 118px; height: 1px; padding-top: 220px; margin-left: 202px;"><div style="box-sizing: border-box; font-size: 0; text-align: center; color: #000000; "><div style="display: inline-block; font-size: 12px; font-family: &quot;Helvetica&quot;; color: light-dark(#000000, #ffffff); line-height: 1.2; pointer-events: all; white-space: normal; word-wrap: normal; ">Postfix</div></div></div></foreignObject><image x="202" y="213.5" width="118" height="17" xlink:href=""/></switch></g></g></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-2"><g><rect x="201" y="320" width="120" height="40" fill="#ffffff" style="fill: light-dark(#ffffff, var(--ge-dark-color, #121212)); stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke="#000000" pointer-events="all"/></g><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 118px; height: 1px; padding-top: 340px; margin-left: 202px;"><div style="box-sizing: border-box; font-size: 0; text-align: center; color: #000000; "><div style="display: inline-block; font-size: 12px; font-family: &quot;Helvetica&quot;; color: light-dark(#000000, #ffffff); line-height: 1.2; pointer-events: all; white-space: normal; word-wrap: normal; ">Dovecot</div></div></div></foreignObject><image x="202" y="333.5" width="118" height="17" xlink:href=""/></switch></g></g></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-3"><g><path d="M 1 120 L 521 120" fill="none" stroke="#000000" style="stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke-width="2" stroke-miterlimit="10" stroke-dasharray="2 6" pointer-events="stroke"/></g></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-4"><g><path d="M 261 86.37 L 261 193.63" fill="none" stroke="#000000" style="stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 261 81.12 L 264.5 88.12 L 261 86.37 L 257.5 88.12 Z" fill="#000000" style="fill: light-dark(rgb(0, 0, 0), rgb(255, 255, 255)); stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke="#000000" stroke-miterlimit="10" pointer-events="all"/><path d="M 261 198.88 L 257.5 191.88 L 261 193.63 L 264.5 191.88 Z" fill="#000000" style="fill: light-dark(rgb(0, 0, 0), rgb(255, 255, 255)); stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke="#000000" stroke-miterlimit="10" pointer-events="all"/></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-8"><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 1px; height: 1px; padding-top: 161px; margin-left: 262px;"><div style="box-sizing: border-box; font-size: 0; text-align: center; color: #000000; background-color: #ffffff; "><div style="display: inline-block; font-size: 11px; font-family: &quot;Helvetica&quot;; color: light-dark(#000000, #ffffff); line-height: 1.2; pointer-events: all; background-color: light-dark(#ffffff, var(--ge-dark-color, #121212)); white-space: nowrap; "><div>smtp</div><div>tcp:25</div></div></div></div></foreignObject><image x="248.5" y="148.5" width="27" height="28.75" xlink:href=""/></switch></g></g></g></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-6"><g><path d="M 261 240 L 261 313.63" fill="none" stroke="#000000" style="stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 261 318.88 L 257.5 311.88 L 261 313.63 L 264.5 311.88 Z" fill="#000000" style="fill: light-dark(rgb(0, 0, 0), rgb(255, 255, 255)); stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke="#000000" stroke-miterlimit="10" pointer-events="all"/></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-7"><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 1px; height: 1px; padding-top: 281px; margin-left: 262px;"><div style="box-sizing: border-box; font-size: 0; text-align: center; color: #000000; background-color: #ffffff; "><div style="display: inline-block; font-size: 11px; font-family: &quot;Helvetica&quot;; color: light-dark(#000000, #ffffff); line-height: 1.2; pointer-events: all; background-color: light-dark(#ffffff, var(--ge-dark-color, #121212)); white-space: nowrap; "><div>lmtp</div><div>unix:/var/spool/postfix/postfix-private/dovecot-lmtp</div></div></div></div></foreignObject><image x="147" y="268.5" width="230" height="28.75" xlink:href=""/></switch></g></g></g></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-9"><g><rect x="81" y="320" width="80" height="40" rx="6" ry="6" fill="#ffffff" style="fill: light-dark(#ffffff, var(--ge-dark-color, #121212)); stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke="#000000" pointer-events="all"/></g><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 78px; height: 1px; padding-top: 340px; margin-left: 82px;"><div style="box-sizing: border-box; font-size: 0; text-align: center; color: #000000; "><div style="display: inline-block; font-size: 12px; font-family: &quot;Helvetica&quot;; color: light-dark(#000000, #ffffff); line-height: 1.2; pointer-events: all; white-space: normal; word-wrap: normal; ">OpenDKIM</div></div></div></foreignObject><image x="82" y="333.5" width="78" height="17" xlink:href=""/></switch></g></g></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-11"><g><path d="M 194.63 220 L 121 220 L 121 313.63" fill="none" stroke="#000000" style="stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 199.88 220 L 192.88 223.5 L 194.63 220 L 192.88 216.5 Z" fill="#000000" style="fill: light-dark(rgb(0, 0, 0), rgb(255, 255, 255)); stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke="#000000" stroke-miterlimit="10" pointer-events="all"/><path d="M 121 318.88 L 117.5 311.88 L 121 313.63 L 124.5 311.88 Z" fill="#000000" style="fill: light-dark(rgb(0, 0, 0), rgb(255, 255, 255)); stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke="#000000" stroke-miterlimit="10" pointer-events="all"/></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-12"><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 1px; height: 1px; padding-top: 261px; margin-left: 122px;"><div style="box-sizing: border-box; font-size: 0; text-align: center; color: #000000; background-color: #ffffff; "><div style="display: inline-block; font-size: 11px; font-family: &quot;Helvetica&quot;; color: light-dark(#000000, #ffffff); line-height: 1.2; pointer-events: all; background-color: light-dark(#ffffff, var(--ge-dark-color, #121212)); white-space: nowrap; ">unix:/var/run/opendkim/opendkim.sock</div></div></div></foreignObject><image x="35" y="255" width="174" height="15.75" xlink:href=""/></switch></g></g></g></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-15"><g><rect x="361" y="20" width="120" height="40" fill="#ffffff" style="fill: light-dark(#ffffff, var(--ge-dark-color, #121212)); stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke="#000000" pointer-events="all"/></g><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 118px; height: 1px; padding-top: 40px; margin-left: 362px;"><div style="box-sizing: border-box; font-size: 0; text-align: center; color: #000000; "><div style="display: inline-block; font-size: 12px; font-family: &quot;Helvetica&quot;; color: light-dark(#000000, #ffffff); line-height: 1.2; pointer-events: all; white-space: normal; word-wrap: normal; "><div>User's</div><div>Mail Client</div></div></div></div></foreignObject><image x="362" y="26" width="118" height="32" xlink:href=""/></switch></g></g></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-20"><g><path d="M 231 20 C 207 20 201 40 220.2 44 C 201 52.8 222.6 72 238.2 64 C 249 80 285 80 297 64 C 321 64 321 48 306 40 C 321 24 297 8 276 16 C 261 4 237 4 231 20 Z" fill="#ffffff" style="fill: light-dark(#ffffff, var(--ge-dark-color, #121212)); stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke="#000000" stroke-miterlimit="10" pointer-events="all"/></g><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 118px; height: 1px; padding-top: 40px; margin-left: 202px;"><div style="box-sizing: border-box; font-size: 0; text-align: center; color: #000000; "><div style="display: inline-block; font-size: 12px; font-family: &quot;Helvetica&quot;; color: light-dark(#000000, #ffffff); line-height: 1.2; pointer-events: all; white-space: normal; word-wrap: normal; "><div>External</div><div>Mail Servers</div></div></div></div></foreignObject><image x="202" y="26" width="118" height="32" xlink:href=""/></switch></g></g></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-21"><g><path d="M 391 60 L 391 220 L 327.37 220" fill="none" stroke="#000000" style="stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 322.12 220 L 329.12 216.5 L 327.37 220 L 329.12 223.5 Z" fill="#000000" style="fill: light-dark(rgb(0, 0, 0), rgb(255, 255, 255)); stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke="#000000" stroke-miterlimit="10" pointer-events="all"/></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-24"><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 1px; height: 1px; padding-top: 161px; margin-left: 391px;"><div style="box-sizing: border-box; font-size: 0; text-align: center; color: #000000; background-color: #ffffff; "><div style="display: inline-block; font-size: 11px; font-family: &quot;Helvetica&quot;; color: light-dark(#000000, #ffffff); line-height: 1.2; pointer-events: all; background-color: light-dark(#ffffff, var(--ge-dark-color, #121212)); white-space: nowrap; "><div>submissions</div><div>tcp:465</div></div></div></div></foreignObject><image x="364" y="148.5" width="54" height="28.75" xlink:href=""/></switch></g></g></g></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-22"><g><path d="M 321 340 L 451 340 L 451 66.37" fill="none" stroke="#000000" style="stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 451 61.12 L 454.5 68.12 L 451 66.37 L 447.5 68.12 Z" fill="#000000" style="fill: light-dark(rgb(0, 0, 0), rgb(255, 255, 255)); stroke: light-dark(rgb(0, 0, 0), rgb(255, 255, 255));" stroke="#000000" stroke-miterlimit="10" pointer-events="all"/></g><g data-cell-id="bQRnDDqhK_hAcH7NuNX_-23"><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 1px; height: 1px; padding-top: 161px; margin-left: 451px;"><div style="box-sizing: border-box; font-size: 0; text-align: center; color: #000000; background-color: #ffffff; "><div style="display: inline-block; font-size: 11px; font-family: &quot;Helvetica&quot;; color: light-dark(#000000, #ffffff); line-height: 1.2; pointer-events: all; background-color: light-dark(#ffffff, var(--ge-dark-color, #121212)); white-space: nowrap; "><div>imaps</div><div>tcp:993</div></div></div></div></foreignObject><image x="434.5" y="148.5" width="33" height="28.75" xlink:href=""/></switch></g></g></g></g></g></g></g></svg> +\ No newline at end of file diff --git a/src/res/blog/thunderbird-final-setup.png b/src/res/blog/thunderbird-final-setup.png Binary files differ. diff --git a/src/res/blog/thunderbird-imap-setup.png b/src/res/blog/thunderbird-imap-setup.png Binary files differ. diff --git a/src/res/blog/thunderbird-smtp-setup.png b/src/res/blog/thunderbird-smtp-setup.png Binary files differ. diff --git a/src/res/blog/thunderbird-user-setup.png b/src/res/blog/thunderbird-user-setup.png Binary files differ.