✉️ Composable all-in-one mail server. https://maddy.email
Find a file
fox.cpp dbfdfd0728
dist: Add LogsDirectory to systemd unit
See https://github.com/foxcpp/maddy/wiki/fail2ban-configuration

Without that it will be impossible for maddy to write to /var/log/maddy
due to sandboxing (ProtectSystem=strict).
2019-09-19 01:59:50 +03:00
.github Update issue templates 2019-09-07 19:22:56 +03:00
address Restructure code tree 2019-09-08 16:06:38 +03:00
atomicbool dispatcher: Add basic logic for check scoring 2019-08-28 04:12:02 +03:00
auth Rework how work directories are handled 2019-09-15 04:03:24 +03:00
buffer buffer: Move to a separate package 2019-08-25 20:23:13 +03:00
check dkim: Implement minimal DKIM verfication 2019-09-18 22:20:04 +03:00
cmd cmd/maddy: Prefer Go 1.12 embedded build info over version constant 2019-09-19 01:48:54 +03:00
config Rework how work directories are handled 2019-09-15 04:03:24 +03:00
dispatcher dispatcher: Remove dd.cancelCtx 2019-09-18 22:59:31 +03:00
dist dist: Add LogsDirectory to systemd unit 2019-09-19 01:59:50 +03:00
dns Perform rDNS lookup asynchronously and generate Received early 2019-09-18 22:59:31 +03:00
dsn dsn: Implement DSN-generation library 2019-08-29 22:08:01 +03:00
endpoint smtp: Set logger name to 'submission' if submission alias is used 2019-09-18 23:18:39 +03:00
future future: Implement a minimal 'future' implementation 2019-09-18 22:59:31 +03:00
log log: Don't add timestamp to messages when logging to stderr 2019-09-15 04:01:36 +03:00
modify module, dispatcher: Implement message rewriting framework 2019-09-14 21:58:54 +03:00
module Perform rDNS lookup asynchronously and generate Received early 2019-09-18 22:59:31 +03:00
mtasts all: Rename module and update imports 2019-08-25 20:32:53 +03:00
storage/sql Perform rDNS lookup asynchronously and generate Received early 2019-09-18 22:59:31 +03:00
target remote: Convert textproto.Error into smtp.SMTPError 2019-09-19 00:39:57 +03:00
testutils dispatcher: Split AddRcpt and Start, fix *State object leaks 2019-09-14 21:58:55 +03:00
.build.yml ci: Remove -race flag 2019-08-26 23:19:44 +03:00
.gitignore Fix-up 35c3b1c 2019-09-08 17:15:18 +03:00
config.go cmd/maddy: Allow to set logging target from command line 2019-09-15 04:03:24 +03:00
dummy.go all: Rename module and update imports 2019-08-25 20:32:53 +03:00
go.mod Add 'go 1.12' to go.mod 2019-09-12 00:47:19 +03:00
go.sum Bump go-imap-sql version 2019-09-06 21:41:55 +03:00
LICENSE docs: Update copyright, repo links and IRC channel in README 2019-08-25 20:29:35 +03:00
maddy.1.scd Update README.md 2019-09-15 04:03:24 +03:00
maddy.conf dkim: Implement minimal DKIM verfication 2019-09-18 22:20:04 +03:00
maddy.conf.5.scd dkim: Implement minimal DKIM verfication 2019-09-18 22:20:04 +03:00
maddy.go cmd/maddy: Remove debug directive hack for command line argument 2019-09-18 22:59:31 +03:00
README.md docs: Update Features section in README 2019-09-19 01:52:20 +03:00

maddy

builds.sr.ht status

Simple, fast, secure all-in-one mail server.

⚠️ Disclaimer: maddy is in early development, many planned features are missing and bugs are waiting to eat your messages

Join ##maddy on irc.freenode.net, if you have any questions or just want to talk about maddy.

Table of contents

Features

  • Single binary for easy deployment
  • Minimal configuration changes required to get almost complete email stack running
  • MTA-STS support
  • DNS sanity checks for incoming deliveries
  • Built-in DKIM verification support

Planned features:

Installation

Pre-built binaries for releases are available here.

Building from source

Dependencies

  • Go toolchain (1.11.4 or newer)

    If your distribution ships an outdated Go version, you can use following commands to get a newer version:

    go get golang.org/dl/go1.13
    go1.13 download
    

    Obviously, you need to have $GOPATH/bin ($HOME/go/bin by default) in $PATH. Then use go1.13 instead of go in commands below.

  • C compiler (optional, set CGO_ENABLED env. variable to 0 to disable)

    Required for SQLite3-based storage (default configuration) and PAM authentication. SQLite3 support can be disabled using nosqlite3 build tag:

    GO111MODULE=on go get github.com/foxcpp/maddy/cmd/maddy@master -tags 'nosqlite3' 
    
  • libpam (optional, not needed by default)

    Used for PAM auth helper binary or direct PAM use, later is disabled by default and can be enabled using 'libpam' build tag (e.g. go get ... -tags 'libpam nosqlite3')

Note: Main executable built with CGO_ENABLED=0 does not depend on any system library and can be moved freely between systems. Executable built without libpam but with CGO_ENABLED=1 (default) depends only on libc. Executable built with libpam depends on system libpam, obviously.

Building

First, make sure Go Modules support is enabled:

export GO111MODULE=on

Command to build latest commit from master branch:

go get github.com/foxcpp/maddy/cmd/maddy@master

Command to build release X.Y.Z:

go get github.com/foxcpp/maddy/cmd/maddy@vX.Y.Z

There is no need to clone this repo, go get command will take care of it.

maddy executable will be placed in $GOPATH/bin directory (defaults to $HOME/go/bin).

Quick start

You need to make sure configuration is placed in /etc/maddy/maddy.conf (copy maddy.conf from this repo) and also /var/lib/maddy and /run/maddy exist and writable.

Then, simply run the executable:

maddy 

You can specify custom locations for all directories and config file, so here is no need to mess with directories in your system:

maddy -config /maddy.conf -state /state -runtime /tmp 

maddy will create specified directories for you.

systemd unit

Alternatively, you can use the systemd unit file from dist/ directory in this repo. It will automatically set-up user account and directories. Additionally, it will apply strict sandbox to maddy to ensure additional security.

You need a relatively new systemd version (235+) both of these things to work properly. Otherwise, you still have to manually create a user account and the state + runtime directories with read-write permissions for the maddy user.

Configuration

Start by copying contents of the maddy.conf to /etc/maddy/maddy.conf (default configuration location).

You need to change the following directives to your values:

  • tls cert_file key_file Change to paths to TLS certificate and key.
  • hostname Server identifier. Put your domain here if you have only one server.
  • autogenerated_msg_domain Put the "main" domain you are handling messages for here.
  • auth_domains, destination in smtp and submission blocks Change to the list of domains you are handling messages for.

With that configuration you will get the following:

  • SQLite-based storage for messages
  • Authentication using SQLite-based virtual users DB (use imapsql-ctl to create user accounts)
  • SMTP endpoint for incoming messages on 25 port.
  • SMTP Submission endpoint for messages from your users, on both 587 (STARTTLS) and 465 (TLS) ports.
  • IMAP endpoint for access to user mailboxes, on both 143 (STARTTLS) and 993 (TLS) ports.
  • Two basic DNS checks for incoming messages

Configuration syntax, high-level structure, and all implemented options are documented in maddy.conf(5) man page.

SQL-based database

Currently, the only supported storage and authentication DB implementation is SQL-based go-imap-sql library.

Use the following commands to install the imapsql-ctl utility:

export GO111MODULE=on
go get github.com/foxcpp/go-imap-sql/cmd/imapsql-ctl@dev

It can be used to create/delete virtual users as well as mailboxes and messages in them.

Here is the command to use to create a new virtual user NAME:

imapsql-ctl --driver DRIVER --dsn DSN users create NAME

Replace DRIVER and DSN with your values from maddy config. For default configuration it is --driver sqlite3 --dsn /var/lib/maddy/all.db.

PostgreSQL instead of SQLite

If you want to use PostgreSQL instead of SQLite 3 (e.g. you want better access concurrency), here is what you need to do:

  1. Install, configure and start PostgreSQL server

There is plenty of tutorials on the web. Note that you need PostgreSQL 9.6 or newer.

  1. Create the database and user role for maddy
psql -U postgres -c 'CREATE USER maddy'
psql -U postgres -c 'CREATE DATABASE maddy'
  1. Update maddy.conf to use PostgreSQL and that database
sql {
    driver postgres
    dsn user=maddy dbname=maddy sslmode=disable
}

Add host= option is server is running on the different machine (configure TLS and remove sslmode=disable then!).

See https://godoc.org/github.com/lib/pq for driver options documentation.

  1. Consider using fsstore

By default, maddy will store messages in table rows. This requires lesser amount of internal bookkeeping and works fine in most cases.

However, you may want to make maddy store message in a filesystem directory instead. To do so add fsstore directive to the sql config block.

You may optionally specify the directory to use instead of default (/var/lib/maddy/sql-sql-fsstore):

sql {
    ...
    fsstore /var/lib/maddy/s3-messages
}

Documentation

Configuration reference is maintained as a single man page in the scdoc format. You can view page source here (it is readable).

Tutorials and misc articles will be added to the project wiki.

System authentication helper binaries

By default maddy is running as a unprivileged user, meaning it can't read system account passwords. There are two options:

  • Run maddy as a privileged user instead (not recommended)

This way you can use maddy without messing with helper binaries, but that will also give maddy a little bit too many permissions.

  • Let maddy start privileged (setuid) helper binary to perform authentication

Compile cmd/maddy-pam-helper and/or cmd/maddy-shadow-helper.

Put them into /var/lib/maddy and make them setuid (there are also other more restrictive options, check README in the binary directories).

Add use_helper to pam or shadow configuration block in your config. Modify systemd unit to use less strict sandbox and disable DynamicUser.

Contributing

See .github/CONTRIBUTING.md.

License

MIT. Just do whatever you want.