Email Filtering with Sieve
User-Defined Sieve Filters
Advice may be outdated
This section was contributed by the community some time ago and some configuration examples may be outdated.
Sieve allows to specify filtering rules for incoming emails that allow for example sorting mails into different folders depending on the title of an email.
Global vs User order
There are global and user specific filters which are filtering the incoming emails in the following order:
Global-before -> User specific -> Global-after
Global filters are applied to EVERY incoming mail for EVERY email address.
- To specify a global Sieve filter provide a
docker-data/dms/config/before.dovecot.sieve
or adocker-data/dms/config/after.dovecot.sieve
file with your filter rules. - If any filter in this filtering chain discards an incoming mail, the delivery process will stop as well and the mail will not reach any following filters (e.g. global-before stops an incoming spam mail: The mail will get discarded and a user-specific filter won't get applied.)
To specify a user-defined Sieve filter place a .dovecot.sieve
file into a virtual user's mail folder (e.g. /var/mail/example.com/user1/home/.dovecot.sieve
). If this file exists dovecot will apply the filtering rules.
It's even possible to install a user provided Sieve filter at startup during users setup: simply include a Sieve file in the docker-data/dms/config/
path for each user login that needs a filter. The file name provided should be in the form <user_login>.dovecot.sieve
, so for example for user1@example.com
you should provide a Sieve file named docker-data/dms/config/user1@example.com.dovecot.sieve
.
An example of a sieve filter that moves mails to a folder INBOX/spam
depending on the sender address:
Example
require ["fileinto", "reject"];
if address :contains ["From"] "spam@spam.com" {
fileinto "INBOX.spam";
} else {
keep;
}
Warning
That folders have to exist beforehand if sieve should move them.
Another example of a sieve filter that forward mails to a different address:
Example
require ["copy"];
redirect :copy "user2@not-example.com";
Just forward all incoming emails and do not save them locally:
Example
redirect "user2@not-example.com";
You can also use external programs to filter or pipe (process) messages by adding executable scripts in docker-data/dms/config/sieve-pipe
or docker-data/dms/config/sieve-filter
.
This can be used in lieu of a local alias file, for instance to forward an email to a webservice.
- These programs can then be referenced by filename, by all users.
- Note that the process running the scripts run as a privileged user.
- For further information see Dovecot's docs.
require ["vnd.dovecot.pipe"];
pipe "external-program";
For more examples or a detailed description of the Sieve language have a look at the official site. Other resources are available on the internet where you can find several examples.
Automatic Sorting Based on Sub-addresses
When mail is delivered to your account, it is possible to organize storing mail into folders by the subaddress (tag) used.
Example: user+<tag>@example.com
to INBOX/<Tag>
This example sorts mail into inbox folders by their tag:
require ["envelope", "fileinto", "mailbox", "subaddress", "variables"];
# Check if the mail recipient address has a tag (:detail)
if envelope :detail :matches "to" "*" {
# Create a variable `tag`, with the the captured `to` value normalized (SoCIAL => Social)
set :lower :upperfirst "tag" "${1}";
# Store the mail into a folder with the tag name, nested under your inbox folder:
if mailboxexists "INBOX.${tag}" {
fileinto "INBOX.${tag}";
} else {
fileinto :create "INBOX.${tag}";
}
}
When receiving mail for user+social@example.com
it would be delivered into the INBOX/Social
folder.
Only redirect mail for specific tags
If you want to only handle specific tags, you could replace the envelope condition and tag assignment from the prior example with:
# Instead of `:matches`, use the default comparator `:is` (exact match)
if envelope :detail "to" "social" {
set "tag" "Social";
# Alternatively you can also provide a list of values to match:
if envelope :detail "to" ["azure", "aws"] {
set "tag" "Cloud";
# Similar to `:matches`, except `:regex` provides enhanced pattern matching.
# NOTE: This example needs you to `require` the "regex" extension
if envelope :detail :regex "to" "^cloud-(azure|aws)$" {
# Normalize the captured azure/aws tag as the resolved value is no longer fixed:
set :lower :upperfirst "vendor" "${1}";
# If a `.` exists in the tag, it will create nested folders:
set "tag" "Cloud.${vendor}";
NOTE: There is no need to lowercase the tag in the conditional as the to
value is a case-insensitive check.
Technical Details
- Dovecot supports this feature via the Sieve subaddress extension (RFC 5233).
- Only a single tag per subaddress is supported. Any additional tag delimiters are part of the tag value itself.
- The Dovecot setting
recipient_delimiter
(default:+
) configures the tag delimiter. This is where thelocal-part
of the recipient address will split at, providing the:detail
(tag) value for Sieve.
INBOX
is the default namespace configured by Dovecot.
- If you omit the
INBOX.
prefix from the sieve script above, the mailbox (folder) for that tag is created at the top-level alongside your Trash and Junk folders. - The
.
betweenINBOX
and${tag}
is important as a separator to distinguish mailbox names. This can vary by mailbox format or configuration. DMS usesMaildir
by default, which uses.
as the separator. lmtp_save_to_detail_mailbox = yes
can be set in/etc/dovecot/conf.d/20-lmtp.conf
:- This implements the feature globally, except for the tag normalization and
INBOX.
prefix parts of the example script. - However, if the sieve script is also present, the script has precedence and will handle this task instead when the condition is successful, otherwise falling back to the global feature.
- This implements the feature globally, except for the tag normalization and
Manage Sieve
The Manage Sieve extension allows users to modify their Sieve script by themselves. The authentication mechanisms are the same as for the main dovecot service. ManageSieve runs on port 4190
and needs to be enabled using the ENABLE_MANAGESIEVE=1
environment variable.
Example
ports:
- "4190:4190"
environment:
- ENABLE_MANAGESIEVE=1
All user defined sieve scripts that are managed by ManageSieve are stored in the user's home folder in /var/mail/example.com/user1/home/sieve
. Just one Sieve script might be active for a user and is sym-linked to /var/mail/example.com/user1/home/.dovecot.sieve
automatically.
Note
ManageSieve makes sure to not overwrite an existing .dovecot.sieve
file. If a user activates a new sieve script the old one is backed up and moved to the sieve
folder.
The extension is known to work with the following ManageSieve clients:
- Sieve Editor a portable standalone application based on the former Thunderbird plugin.
- Kmail the mail client of KDE's Kontact Suite.