Raymii.org
Quis custodiet ipsos custodes?Home | About | All pages | Cluster Status | RSS Feed
Filtering IMAP mail with imapfilter
Published: 17-01-2015 | Author: Remy van Elst | Text only version of this article
❗ This post is over nine years old. It may no longer be up to date. Opinions may have changed.
Table of Contents
I have several email accounts at different providers. Most of them don't offer filtering capabilites like Sieve, or only their own non exportable rule system (Google Apps). My mail client of choice, Thunderbird, has filtering capabilities but my phone has not and I don't want to leave my machine running Thunderbird all the time since it gets quite slow with huge mailboxes. Imapfilter is a mail filtering utility written in Lua which connects to one or more IMAP accounts and filters on the server using IMAP queries. It is a lightweight command line utility, the configuration can be versioned and is simple text and it is very fast.
Recently I removed all Google Ads from this site due to their invasive tracking, as well as Google Analytics. Please, if you found this content useful, consider a small donation using any of the options below:
I'm developing an open source monitoring app called Leaf Node Monitoring, for windows, linux & android. Go check it out!
Consider sponsoring me on Github. It means the world to me if you show your appreciation and you'll help pay the server costs.
You can also sponsor me by getting a Digital Ocean VPS. With this referral link you'll get $200 credit for 60 days. Spend $25 after your credit expires and I'll get $25!
Imapfilter is configured via a config file. This article will discuss this config file with filtering and other examples. Start with a blank one:
mkdir -p ~/.imapfilter
vim ~/.imapfilter/config.lua
Options
imapfilter has a few global options which are configured via the
options.$OPTION = $VALUE
format. These are the ones I have, the manpage has
more. Comments in the config file are prefix by two dashes (--
).
-- One of the work mailservers is slow.
-- The time in seconds for the program to wait for a mail server's response (default 60)
options.timeout = 120
-- According to the IMAP specification, when trying to write a message to a non-existent mailbox, the server must send a hint to the client, whether it should create the mailbox and try again or not. However some IMAP servers don't follow the specification and don't send the correct response code to the client. By enabling this option the client tries to create the mailbox, despite of the server's response.
options.create = true
-- By enabling this option new mailboxes that were automatically created, get also subscribed; they are set active in order for IMAP clients to recognize them
options.subscribe = true
-- Normally, messages are marked for deletion and are actually deleted when the mailbox is closed. When this option is enabled, messages are expunged immediately after being marked deleted.
options.expunge = true
Accounts
I've defined two example accounts, one for work and one for personal stuff:
account1 = IMAP {
server = "imap.gmail.com",
username = "joe@gmail.com",
password = "P@ssw0rd",
ssl = "tls1"
}
account2 = IMAP {
server = "imap.mywork.org",
username = "joe",
password = "W0rdP@ss",
ssl = "ssl3"
}
You can define as much accounts as needed. You can even get your accounts from offlineimap:
function offlineimap (key)
local status
local value
status, value = pipe_from('grep -A2 mail.gandi.net ~/.offlineimaprc | grep ' .. key .. '|cut -d= -f2')
value = string.gsub(value, ' ', '')
value = string.gsub(value, '\n', '')
return value
end
T = IMAP {
server = offlineimap('remotehost'),
username = offlineimap('remoteuser'),
password = offlineimap('remotepass'),
ssl = 'ssl3',
}
Mailboxes / Folders
imapfilter has the concept of mailboxes
. While technically correct, we general
users just call them (top level) folders. INBOX
is a mailbox, other folders
are as well. After an IMAP account has been initialized, mailboxes residing in
that account can be accessed simply as elements of the account table:
myaccount.mymailbox
If mailbox names don't only include letters, digits and underscores, or begin with a digit, an alternative form must be used:
myaccount['mymailbox']
A mailbox inside a folder (subfolder) can be only accessed by using the alternative form:
myaccount['myfolder/mymailbox']
In this article I use this alternative form for ease of use and consistensy.
Filtering
The filters defined are processed in order from top to bottom. I mostly filter my inbox by moving messages to another folder. If a message matched a filter it is moved, if it would then lower on match another filter that would not apply to that mail because it is already moved.
See the manpage for all configuration options.
If you simply want to filter a message based on the sender, receipient or subject you can use the following. It moves all messages with the Duplicity mailing list address to the mailinglists folder:
messages = account1["INBOX"]:contain_to("duplicity-talk@nongnu.org")
messages:move_messages(account1["Mailinglists/Duplicity-Talk"])
If you want to filter based on a few more parameters, you can use the following
operators, *
for AND, +
for OR and -
for NOT. To filter nagios messages
with a certain subject line:
messages = account1["INBOX"]:contain_from("nagios@monitoring.org")
* account1["INBOX"]:contain_subject("important_hostname")
messages:move_messages(account1["Important/Nagios"])
To move messages from Nagios from less important hosts, but not with the "CRITICAL" subject and mark them as read:
messages = account1["INBOX"]:contain_from("nagios@monitoring.org")
- account1["INBOX"]:contain_subject("CRITICAL:")
messages:mark_seen()
messages:move_messages(account1["Monitoring/Nagios"])
As you can see the mark_seen()
operator marks messages as read.
With these operators you can construct advanced filters.
Copying mail
To copy mail from one account to another account's folder and mark those copied messages as read, for archival purposes for example, use the following filter:
messages = account2['INBOX']:is_unseen()
messages:copy_messages(account1["Backup_of_Account2"])
messages = account1['Backup_of_Account2']:is_unseen()
messages:mark_seen()
Place this at the top of the account2
filtering rules.
pipe_to
Taken from the extended configuration example here is an example piece of code which sends mail to an external program and based on the output deletes the messages the program marked as spam.
-- The auxiliary function pipe_to() is supplied for conveniency. For
-- example if there was a utility named "bayesian-spam-filter", which
-- returned 1 when it considered the message "spam" and 0 otherwise:
all = account1["INBOX"]:is_unseen()
results = Set {}
for _, mesg in ipairs(all) do
mbox, uid = table.unpack(mesg)
text = mbox[uid]:fetch_message()
if (pipe_to('bayesian-spam-filter', text) == 1) then
table.insert(results, mesg)
end
end
results:delete_messages()
Conclusion
The above examples will get you started with message filtering right away. The manpage and the example config and the extended example config will get you even further.
Tags: blog , dovecot , filter , gmail , imap , imapfilter , lua , mail , sieve , smtp