config | ||
examples | ||
imap-core | ||
lib | ||
.eslintrc | ||
.gitignore | ||
api.js | ||
Gruntfile.js | ||
imap.js | ||
indexes.json | ||
LICENSE | ||
logger.js | ||
package.json | ||
README.md | ||
server.js | ||
smtp.js | ||
worker.js |
Wild Duck Mail Agent
Wild Duck is a distributed IMAP server built with Node.js, MongoDB and Redis. Node.js runs the application, MongoDB is used as the mail store and Redis is used for ephemeral actions like publish/subscribe, locking and caching.
NB! Wild Duck is currently in beta. You should not use it in production.
Goals of the Project
- Build a scalable and distributed IMAP server that uses clustered database instead of single machine file system as mail store
- Push notifications. Your application (eg. a webmail client) should be able to request changes (new and deleted messages, flag changes) to be pushed to client instead of using IMAP to fetch stuff from the server
- Provide Gmail-like features like pushing sent messages automatically to Sent Mail folder or notifying about messages moved to Junk folder so these could be marked as spam
Similar alterntives
Here's a list of Email/IMAP servers that use database for storing email messages
Supported features
Wild Duck IMAP server supports the following IMAP standards:
- The entire IMAP4rev1 suite with some minor differences from the spec. Intentionally missing is the
\Recent
flag as it does not provide any real value, only makes things more complicated. RENAME works a bit differently than spec describes. - IDLE – notfies about new and deleted messages and also about flag updates
- CONDSTORE and ENABLE – supports most of the spec, except metadata stuff which is ignored
- STARTTLS
- NAMESPACE – minimal support, just lists the single user namespace with hierarchy separator
- UNSELECT
- UIDPLUS
- SPECIAL-USE
- ID
- AUTHENTICATE PLAIN and SASL-IR
- APPENDLIMIT (RFC7889) – maximum global allowed message size is advertised in CAPABILITY listing
- UTF8=ACCEPT (RFC6855) – this also means that Wild Duck natively supports unicode email usernames. For example <андрис@уайлддак.орг> is a valid email address that is hosted by a test instance of Wild Duck
- QUOTA (RFC2087) – Quota size is global for an account, using a single quota root. Be aware that quota size does not mean actual byte storage in disk, it is calculated as the sum of the rfc822 sources of stored messages. Actual disk usage is larger as there's a lot of database overhead per every message.
FAQ
Does it work?
Yes, it does. You can run the server and get a working IMAP server for mail store, SMTP server for pushing messages to the mail store and HTTP API server to create new users. All handled by Node.js, MongoDB and Redis, no additional dependencies needed.
What are the killer features?
- Start as many instances as you want. You can start multiple Wild Duck instances in different machines and as long as they share the same MongoDB and Redis settings, users can connect to any instances. This is very different from the traditional IMAP servers where a single user always needs to connect (or be proxied) to the same IMAP server. Wild Duck keeps all required state information in MongoDB, so it does not matter which IMAP instance you use.
- Super easy to tweak. The entire codebase is pure JavaScript, so there's nothing to compile or anything platform specific. If you need to tweak something then change the code, restart the app and you're ready to go. If it works on one machine then most probably it works in every other machine as well.
- Works almost on any OS including Windows. At least if you get MongoDB and Redis (Windows fork) running first.
- Focus on internationalization, ie. supporting email addresses with non-ascii characters
+
-labels: андрис+ööö@уайлддак.орг is delivered to андрис@уайлддак.орг
Isn't it bad to use a database as a mail store?
Yes, historically it has been considered a bad practice to store emails in a database. And for a good reason. The data model of relational databases like MySQL does not work well with tree like structures (email mime tree) or large blobs (email source).
Notice the word "relational"? In fact document stores like MongoDB work very well with emails. Document store is great for storing tree-like structures and while GridFS is not as good as "real" object storage, it is good enough for storing the raw parts of the message. Additionally there's nothing too GridFS specific, so (at least in theory) it could be replaced with any object store.
Is the server scalable?
Not yet exactly. Even though on some parts Wild Duck is already fast, there are still some important improvements that need to be done:
- Optimize SEARCH queries to use MongoDB queries. Currently only simple stuff (flag, internaldate, not flag, modseq) is included in query and more complex comparisons are handled by the application but this means that too much data must be loaded from database (unless it is a very simple query like "SEARCH UNSEEN" that is already optimized)
- Optimize FETCH queries to load only partial data for BODY subparts
- Parse incoming message into the mime tree as a stream. Currently the entire message is buffered in memory before being parsed.
How does it work?
Whenever a message is received Wild Duck parses it into a tree-like structure based on the MIME tree and stores this tree to MongoDB. Larger attachments (anything above 50kB) are removed from the tree and stored separately in GridStore. If a message needs to be loaded then Wild Duck fetches the tree structure first, if needed loads attachments from GridStore and then compiles it back into the original RFC822 message. The result should be identical to the original messages unless the original message used unix newlines, these might be partially replaced with windows newlines.
Wild Duck tries to keep minimal state for sessions to be able to distribute sessions between different hosts. Whenever a mailbox is opened the entire message list is loaded as an array of UID values. The first UID in the array element points to the message #1 in IMAP, second one points to message #2 etc.
Actual update data (information about new and deleted messages, flag updates and such) is stored to a journal log and an update beacon is propagated through Redis pub/sub whenever something happens. If a session detects that there have been some changes in the current mailbox and it is possible to notify the user about it (eg. a NOOP call was made), journaled log is loaded from the database and applied to the UID array one action at a time. Once all journaled updates have applied then the result should match the latest state. If it is not possible to notify the user (eg a FETCH call was made), then journal log is not loaded and the user continues to see the old state.
Future considerations
- Add interoperability with current servers, for example by fetching authentication data from MySQL
- Maybe allow some kind of message manipulation through plugins? This would allow to turn Wild Duck for example into an encrypted mail server – mail data would be encrypted using users public key before storing it to DB and decrypted with users private key whenever the user logs in and FETCHes or SEARCHes messages. Private key would be protected by users password. For the user the encryption layer would be invisible while guaranteeing that if the user is currently not logged in then there would be no way to read the messages as the private key is locked.
- Add proper quota handling. Every time a user gets a new message added to storage, the quota counter should increase. If only a single quota root would be used per account then implementing rfc2087 should be fairly easy. Currently Wild Duck keeps a count on storage used by an user but this is not exposed to the client.
Usage
Assuming you have MongoDB and Redis running somewhere.
Step 1. Get the code from github
$ git clone git://github.com/wildduck-email/wildduck.git
$ cd wildduck
Step 2. Install dependencies
Install dependencies from npm
$ npm install --production
Step 3. Modify config
You can either modify the default config file or alternatively generate an environment related config file that gets merged with the default values. Read about the config module here
Step 4. Run the server
To use the default config file, run the following
npm start
Or if you want to use environment related config file, eg from production.js
, run the following
NODE_ENV=production npm start
Step 5. Create an user account
See see below for details about creating new user accounts
Manage user
Users can be managed with HTTP requests against Wild Duck API
POST /user/create
Creates a new user.
Arguments
- username is the username of the user. This is not an email address but authentication username, use only letters and numbers
- password is the password for the user
- quota (optional) is the maximum storage in bytes allowed for this user. If not set then the default value is used
Example
curl -XPOST "http://localhost:8080/user/create" -H 'content-type: application/json' -d '{
"username": "testuser",
"password": "secretpass"
}'
The response for successful operation should look like this:
{
"success": true,
"username": "testuser"
}
After you have created an user you can use these credentials to log in to the IMAP server. To be able to receive mail for that user you need to register an email address.
POST /user/address/create
Creates a new email address alias for an existing user. You can use internationalized email addresses like андрис@уайлддак.орг.
Arguments
- username is the username
- address is the email address to use as an alias for this user
- main (either true or false, defaults to false) indicates that this is the default address for that user
First added address becomes main by default
Example
curl -XPOST "http://localhost:8080/user/address/create" -H 'content-type: application/json' -d '{
"username": "testuser",
"address": "user@example.com"
}'
The response for successful operation should look like this:
{
"success": true,
"username": "testuser",
"address": "user@example.com"
}
After you have registered a new address then SMTP maildrop server starts accepting mail for it and store the messages to the users mailbox.
POST /user/quota
Updates maximum allowed quota for an user
Arguments
- username is the username of the user to modify
- quota (optional) is the maximum storage in bytes allowed for this user. If not set or zero then the default value is used
Example
curl -XPOST "http://localhost:8080/user/quota" -H 'content-type: application/json' -d '{
"username": "testuser",
"quota": 1234567
}'
The response for successful operation should look like this:
{
"success": true,
"username": "testuser",
"previousQuota": 0,
"quota": 1234567
}
Quota changes apply immediately.
POST /user/password
Updates password for an user
Arguments
- username is the username of the user to modify
- password is the new password for the user
Example
curl -XPOST "http://localhost:8080/user/password" -H 'content-type: application/json' -d '{
"username": "testuser",
"password": "newpass"
}'
The response for successful operation should look like this:
{
"success": true,
"username": "testuser"
}
Password change applies immediately.
GET /user
Returns user information including quota usage and registered addresses
Arguments
- username is the username of the user to modify
Example
curl "http://localhost:8080/user?username=testuser"
The response for successful operation should look like this:
{
"success": true,
"username": "testuser",
"quota": 1234567,
"storageUsed": 1822,
"addresses": [
{
"id": "58d8fccb645b0deb23d6c37d",
"address": "user@example.com",
"main": true,
"created": "2017-03-27T11:51:39.639Z"
}
]
}
GET /user/mailboxes
Returns all mailbox names for the user
Arguments
- username is the username of the user to modify
Example
curl "http://localhost:8080/user/mailboxes?username=testuser"
The response for successful operation should look like this:
{
"success": true,
"username": "testuser",
"mailboxes": [
{
"id": "58d8f2ae240366dfd5d8049c",
"path": "INBOX",
"special": "Inbox",
"messages": 100
},
{
"id": "58d8f2ae240366dfd5d8049d",
"path": "Sent Mail",
"special": "Sent",
"messages": 45
},
{
"id": "58d8f2ae240366dfd5d8049f",
"path": "Junk",
"special": "Junk",
"messages": 10
},
{
"id": "58d8f2ae240366dfd5d8049e",
"path": "Trash",
"special": "Trash",
"messages": 11
}
]
}
DELETE /message
Deletes a message from a mailbox.
Arguments
- id is the MongoDB _id as a string for a message
Example
curl "http://localhost:8080/message?id=58d8299c5195c38e77c2daa5"
The response for successful operation should look like this:
{
"success": true,
"id": "58d8299c5195c38e77c2daa5"
}
Testing
Create an email account and use your IMAP client to connect to it. To send mail to this account, run the example script:
node examples/push-mail.js username@example.com
This should "deliver" a new message to the INBOX of username@example.com by using the built-in SMTP maildrop interface. If your email client is connected then you should promptly see the new message.
License
Wild Duck Mail Agent is licensed under the European Union Public License 1.1.