2014-07-10 07:18:01 +00:00
#!/bin/bash
#
# User Authentication and Destination Validation
2014-09-21 17:43:21 +00:00
# ----------------------------------------------
2014-07-10 07:18:01 +00:00
#
# This script configures user authentication for Dovecot
# and Postfix (which relies on Dovecot) and destination
2015-06-27 17:23:15 +00:00
# validation by quering an Sqlite3 database of mail users.
2014-07-10 07:18:01 +00:00
source setup/functions.sh # load our functions
source /etc/mailinabox.conf # load global vars
2014-09-21 17:43:21 +00:00
# ### User and Alias Database
2014-07-10 07:18:01 +00:00
# The database of mail users (i.e. authenticated users, who have mailboxes)
# and aliases (forwarders).
db_path = $STORAGE_ROOT /mail/users.sqlite
# Create an empty database if it doesn't yet exist.
if [ ! -f $db_path ] ; then
echo Creating new user database: $db_path ;
2014-08-08 12:31:22 +00:00
echo "CREATE TABLE users (id INTEGER PRIMARY KEY AUTOINCREMENT, email TEXT NOT NULL UNIQUE, password TEXT NOT NULL, extra, privileges TEXT NOT NULL DEFAULT '');" | sqlite3 $db_path ;
2015-07-04 15:31:11 +00:00
echo "CREATE TABLE aliases (id INTEGER PRIMARY KEY AUTOINCREMENT, address TEXT NOT NULL UNIQUE, receivers TEXT NOT NULL, senders TEXT NOT NULL);" | sqlite3 $db_path ;
2014-07-10 07:18:01 +00:00
fi
2014-09-21 17:43:21 +00:00
# ### User Authentication
2014-07-10 07:18:01 +00:00
# Have Dovecot query our database, and not system users, for authentication.
sed -i "s/#*\(\!include auth-system.conf.ext\)/#\1/" /etc/dovecot/conf.d/10-auth.conf
sed -i "s/#\(\!include auth-sql.conf.ext\)/\1/" /etc/dovecot/conf.d/10-auth.conf
# Specify how the database is to be queried for user authentication (passdb)
# and where user mailboxes are stored (userdb).
cat > /etc/dovecot/conf.d/auth-sql.conf.ext << EOF;
passdb {
driver = sql
args = /etc/dovecot/dovecot-sql.conf.ext
}
userdb {
driver = static
args = uid = mail gid = mail home = $STORAGE_ROOT /mail/mailboxes/%d/%n
}
EOF
# Configure the SQL to query for a user's password.
cat > /etc/dovecot/dovecot-sql.conf.ext << EOF;
driver = sqlite
connect = $db_path
default_pass_scheme = SHA512-CRYPT
password_query = SELECT email as user, password FROM users WHERE email = '%u' ;
EOF
chmod 0600 /etc/dovecot/dovecot-sql.conf.ext # per Dovecot instructions
# Have Dovecot provide an authorization service that Postfix can access & use.
cat > /etc/dovecot/conf.d/99-local-auth.conf << EOF;
service auth {
unix_listener /var/spool/postfix/private/auth {
mode = 0666
user = postfix
group = postfix
}
}
EOF
# And have Postfix use that service.
tools/editconf.py /etc/postfix/main.cf \
smtpd_sasl_type = dovecot \
smtpd_sasl_path = private/auth \
smtpd_sasl_auth_enable = yes
2015-06-20 10:27:18 +00:00
# ### Sender Validation
# Use a Sqlite3 database to set login maps. This is used with
2015-06-27 17:23:15 +00:00
# reject_authenticated_sender_login_mismatch to see if the user is
# allowed to send mail as the FROM address specified in the request.
2015-06-20 10:27:18 +00:00
tools/editconf.py /etc/postfix/main.cf \
smtpd_sender_login_maps = sqlite:/etc/postfix/sender-login-maps.cf
2015-06-27 17:23:15 +00:00
# SQL statement that returns a list of addresses/domains the logged in username
# is allowed to send as. This is similar to virtual-alias-maps.cf (see below).
# Matches from the users table take priority over (direct) aliases.
2015-06-20 10:27:18 +00:00
cat > /etc/postfix/sender-login-maps.cf << EOF;
dbpath = $db_path
2015-07-04 15:31:11 +00:00
query = SELECT senders from ( SELECT senders, 0 as priority FROM aliases WHERE address = '%s' UNION SELECT email as senders, 1 as priority FROM users WHERE email = '%s' ) ORDER BY priority LIMIT 1;
2015-06-20 10:27:18 +00:00
EOF
2014-09-21 17:43:21 +00:00
# ### Destination Validation
2014-07-10 07:18:01 +00:00
# Use a Sqlite3 database to check whether a destination email address exists,
# and to perform any email alias rewrites in Postfix.
tools/editconf.py /etc/postfix/main.cf \
virtual_mailbox_domains = sqlite:/etc/postfix/virtual-mailbox-domains.cf \
virtual_mailbox_maps = sqlite:/etc/postfix/virtual-mailbox-maps.cf \
virtual_alias_maps = sqlite:/etc/postfix/virtual-alias-maps.cf \
local_recipient_maps = \$ virtual_mailbox_maps
# SQL statement to check if we handle mail for a domain, either for users or aliases.
cat > /etc/postfix/virtual-mailbox-domains.cf << EOF;
dbpath = $db_path
2015-07-04 15:31:11 +00:00
query = SELECT 1 FROM users WHERE email LIKE '%%@%s' UNION SELECT 1 FROM aliases WHERE address LIKE '%%@%s'
2014-07-10 07:18:01 +00:00
EOF
# SQL statement to check if we handle mail for a user.
cat > /etc/postfix/virtual-mailbox-maps.cf << EOF;
dbpath = $db_path
query = SELECT 1 FROM users WHERE email = '%s'
EOF
# SQL statement to rewrite an email address if an alias is present.
2015-06-21 12:22:11 +00:00
#
# Postfix makes multiple queries for each incoming mail. It first
# queries the whole email address, then just the user part in certain
# locally-directed cases (but we don't use this), then just `@`+the
# domain part. The first query that returns something wins. See
# http://www.postfix.org/virtual.5.html.
#
# virtual-alias-maps has precedence over virtual-mailbox-maps, but
# we don't want catch-alls and domain aliases to catch mail for users
# that have been defined on those domains. To fix this, we not only
# query the aliases table but also the users table when resolving
# aliases, i.e. we turn users into aliases from themselves to
# themselves. That means users will match in postfix's first query
# before postfix gets to the third query for catch-alls/domain alises.
#
2014-09-27 11:39:50 +00:00
# If there is both an alias and a user for the same address either
# might be returned by the UNION, so the whole query is wrapped in
2015-06-21 12:22:11 +00:00
# another select that prioritizes the alias definition to preserve
# postfix's preference for aliases for whole email addresses.
2014-07-10 07:18:01 +00:00
cat > /etc/postfix/virtual-alias-maps.cf << EOF;
dbpath = $db_path
2015-07-04 15:31:11 +00:00
query = SELECT receivers from ( SELECT receivers, 0 as priority FROM aliases WHERE address = '%s' UNION SELECT email as receivers, 1 as priority FROM users WHERE email = '%s' ) ORDER BY priority LIMIT 1;
2014-07-10 07:18:01 +00:00
EOF
# Restart Services
##################
2014-07-16 13:06:45 +00:00
restart_service postfix
restart_service dovecot
2014-07-10 07:18:01 +00:00