wildduck/README.md

672 lines
26 KiB
Markdown
Raw Normal View History

2017-04-02 04:16:53 +08:00
# Agent Wild Duck
2017-03-06 05:45:50 +08:00
2017-03-06 22:29:25 +08:00
![](https://cldup.com/qlZnwOz0na.jpg)
2017-04-09 17:33:10 +08:00
Wild Duck is a distributed IMAP/POP3 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.
2017-03-11 22:02:41 +08:00
2017-03-11 22:40:59 +08:00
> **NB!** Wild Duck is currently in **beta**. You should not use it in production.
2017-03-06 05:45:50 +08:00
## Goals of the Project
2017-03-06 05:45:50 +08:00
2017-04-09 17:33:10 +08:00
1. Build a scalable and distributed IMAP/POP3 server that uses clustered database instead of single machine file system as mail store
2. Allow using internationalized email addresses
2017-03-08 02:08:59 +08:00
3. 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
2017-04-03 21:59:04 +08:00
4. Provide parsed mailbox and message data over HTTP. This should make creating webmail interfaces super easy, no need to parse RFC822 messages to get text content or attachments
2017-03-06 05:45:50 +08:00
2017-04-05 01:42:44 +08:00
## Alternatives
2017-04-05 01:42:44 +08:00
Here's a list of email/IMAP servers that use database for storing email messages
2017-04-05 01:42:44 +08:00
- [DBMail](http://www.dbmail.org/) (IMAP)
- [Archiveopteryx](http://archiveopteryx.org/) (IMAP)
- [ElasticInbox](http://www.elasticinbox.com/) (POP3)
## Supported features
2017-03-06 22:24:57 +08:00
2017-03-08 02:08:59 +08:00
Wild Duck IMAP server supports the following IMAP standards:
2017-03-06 22:31:49 +08:00
2017-03-30 23:25:59 +08:00
- The entire **IMAP4rev1** suite with some minor differences from the spec. See below for [IMAP Protocol Differences](#imap-protocol-differences) for a complete list
2017-04-05 01:39:32 +08:00
- **IDLE** ([RFC2177](https://tools.ietf.org/html/rfc2177)) notfies about new and deleted messages and also about flag updates
- **CONDSTORE** ([RFC4551](https://tools.ietf.org/html/rfc4551)) and **ENABLE** ([RFC5161](https://tools.ietf.org/html/rfc5161)) supports most of the spec, except metadata stuff which is ignored
- **STARTTLS** ([RFC2595](https://tools.ietf.org/html/rfc2595))
- **NAMESPACE** ([RFC2342](https://tools.ietf.org/html/rfc2342)) minimal support, just lists the single user namespace with hierarchy separator
- **UNSELECT** ([RFC3691](https://tools.ietf.org/html/rfc3691))
- **UIDPLUS** ([RFC4315](https://tools.ietf.org/html/rfc4315))
- **SPECIAL-USE** ([RFC6154](https://tools.ietf.org/html/rfc6154))
- **ID** ([RFC2971](https://tools.ietf.org/html/rfc2971))
- **MOVE** ([RFC6851](https://tools.ietf.org/html/rfc6851))
- **AUTHENTICATE PLAIN** ([RFC4959](https://tools.ietf.org/html/rfc4959)) and **SASL-IR**
- **APPENDLIMIT** ([RFC7889](https://tools.ietf.org/html/rfc7889)) maximum global allowed message size is advertised in CAPABILITY listing
- **UTF8=ACCEPT** ([RFC6855](https://tools.ietf.org/html/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](https://tools.ietf.org/html/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](https://tools.ietf.org/html/rfc822) sources of stored messages. Actual disk usage is larger as there are database overhead per every message.
- **COMPRESS=DEFLATE** ([RFC4978](https://tools.ietf.org/html/rfc4978)) Compress traffic between the client and the server
2017-03-08 02:08:59 +08:00
2017-04-05 01:46:42 +08:00
Wild Duck more or less passes the [ImapTest](https://www.imapwiki.org/ImapTest/TestFeatures). Common errors that arise in the test are unknown labels (Wild Duck doesn't send unsolicited `FLAGS` updates even though it does send unsolicited `FETCH FLAGS` updates) and sometimes NO for `STORE` (messages deleted in one session can not be updated in another).
2017-04-01 02:41:53 +08:00
2017-04-08 03:38:52 +08:00
### POP3 Support
2017-04-09 17:33:10 +08:00
In addition to the required POP3 commands ([RFC1939](https://tools.ietf.org/html/rfc1939)) Wild Duck supports the following extensions:
2017-04-08 03:38:52 +08:00
2017-04-09 17:33:10 +08:00
* **UIDL**
* **USER**
* **PASS**
* **SASL PLAIN**
* **PIPELINING**
2017-04-10 04:21:36 +08:00
* **TOP**
2017-04-09 17:33:10 +08:00
2017-04-09 17:57:10 +08:00
#### POP3 command behaviors
2017-04-10 02:29:42 +08:00
All changes to messages like deleting messages or marking messages as seen are stored in storage only in the UPDATE stage (eg. after calling QUIT). Until then the changes are preserved in memory only. This also means that if a message is downloaded but QUIT is not issued then the message does not get marked as *Seen*.
2017-04-09 17:57:10 +08:00
##### LIST
2017-04-09 17:33:10 +08:00
POP3 listing displays the newest 250 messages in INBOX (configurable)
2017-04-09 18:00:19 +08:00
##### UIDL
Wild Duck uses message `_id` value (24 byte hex) as the unique ID. If a message is moved from one mailbox to another then it might *re-appear* in the listing.
2017-04-09 17:57:10 +08:00
##### RETR
2017-04-09 17:33:10 +08:00
2017-04-10 02:29:42 +08:00
If a messages is downloaded by a client this message gets marked as *Seen*
2017-04-09 17:33:10 +08:00
2017-04-09 17:57:10 +08:00
##### DELE
2017-04-09 17:33:10 +08:00
If a messages is deleted by a client this message gets marked as Seen and moved to Trash folder
2017-04-08 03:38:52 +08:00
## FAQ
2017-03-08 02:08:59 +08:00
### Does it work?
2017-03-08 02:08:59 +08:00
2017-04-09 17:57:10 +08:00
Yes, it does. You can run the server and get working IMAP and POP3 servers 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. The IMAP server hosting уайлддак.орг uses a MongoDB replica set of 3 hosts.
2017-03-08 02:08:59 +08:00
### What are the killer features?
2017-03-21 01:34:48 +08:00
1. 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.
2. 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.
2017-03-11 05:32:44 +08:00
3. Works almost on any OS including Windows. At least if you get MongoDB and Redis ([Windows fork](https://github.com/MSOpenTech/redis)) running first.
4. Focus on internationalization, ie. supporting email addresses with non-ascii characters
2017-03-28 01:59:17 +08:00
5. `+`-labels: _андрис+ööö@уайлддак.орг_ is delivered to _андрис@уайлддак.орг_
2017-04-03 21:59:04 +08:00
6. Access messages both using IMAP and HTTP API. The latter serves parsed data, so no need to fetch RFC822 messages and parse out html, plaintext content or attachments. It is super easy to create a webmail interface on top of this.
### Isn't it bad to use a database as a mail store?
2017-03-08 02:08:59 +08:00
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).
2017-03-21 01:34:48 +08:00
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.
2017-03-08 02:08:59 +08:00
2017-03-31 02:02:41 +08:00
You can see an example mail entry [here](https://gist.github.com/andris9/520d530bcc126768ce5e09e774be8c2e). Lines [184-217](https://gist.github.com/andris9/520d530bcc126768ce5e09e774be8c2e#file-entry-js-L184-L217) demonstrate a node that has its body missing as it was big enough to be moved to GridStore and not be included with the main entry.
### Is the server scalable?
2017-03-08 02:08:59 +08:00
2017-04-05 01:39:32 +08:00
Somewhat yes. Even though on some parts Wild Duck is already fast (Wild Duck is successfully tested with mailboxes up to 200K messages), there are still some important improvements that need to be done:
2017-03-06 22:24:57 +08:00
2017-04-04 19:46:51 +08:00
1. Optimize FETCH queries to load only partial data for BODY subparts
2. Parse incoming message into the mime tree as a stream. Currently the entire message is buffered in memory before being parsed.
3. CPU usage seems a bit too high, there is probably a ton of profiling to do
### How does it work?
2017-04-10 02:29:42 +08:00
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. Attachments 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 and, 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.
2017-03-11 05:15:00 +08:00
2017-04-10 02:29:42 +08:00
Wild Duck tries to keep minimal state for sessions (basically just a list of currently known UIDs and latest MODSEQ value) 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.
2017-03-06 22:24:57 +08:00
2017-03-11 22:40:59 +08:00
### Future considerations
1. Add interoperability with current servers, for example by fetching authentication data from MySQL
2. 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.
2017-03-12 00:12:19 +08:00
2017-03-06 05:45:50 +08:00
## Usage
Assuming you have MongoDB and Redis running somewhere.
### Step 1\. Get the code from github
2017-03-06 05:45:50 +08:00
```
$ git clone git://github.com/wildduck-email/wildduck.git
$ cd wildduck
```
2017-03-06 05:45:50 +08:00
### Step 2\. Install dependencies
Install dependencies from npm
2017-03-06 05:45:50 +08:00
```
$ npm install --production
```
2017-03-06 22:24:57 +08:00
### Step 3\. Modify config
You can either modify the default [config file](./config/default.js) or alternatively generate an environment related config file that gets merged with the default values. Read about the config module [here](https://www.npmjs.com/package/config)
### Step 4\. Run the server
To use the default config file, run the following
2017-03-06 05:45:50 +08:00
```
npm start
```
2017-03-06 05:45:50 +08:00
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
2017-04-03 21:59:04 +08:00
See see [below](#http-api) for details about creating new user accounts
2017-03-06 22:39:00 +08:00
2017-04-03 21:59:04 +08:00
## HTTP API
2017-03-06 22:13:40 +08:00
2017-04-03 21:59:04 +08:00
Users, mailboxes and messages can be managed with HTTP requests against Wild Duck API
TODO:
1. Expose counters (seen/unseen messages, message count in mailbox etc.)
2. Search messages
2017-04-04 15:20:25 +08:00
3. Expose journal updates through WebSocket or similar
2017-03-06 22:13:40 +08:00
### POST /user/create
Creates a new user.
2017-03-06 22:13:40 +08:00
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
2017-03-06 22:13:40 +08:00
**Example**
```
curl -XPOST "http://localhost:8080/user/create" -H 'content-type: application/json' -d '{
"username": "testuser",
"password": "secretpass"
2017-03-06 22:13:40 +08:00
}'
```
The response for successful operation should look like this:
```json
{
"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
2017-03-28 01:59:17 +08:00
- **main** (either _true_ or _false_, defaults to _false_) indicates that this is the default address for that user
2017-03-28 01:59:17 +08:00
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:
```json
{
"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
- **recipients** (optional) is the maximum sending recipients per 24h allowed for this user. Assumes ZoneMTA with [zonemta-wildduck](https://github.com/wildduck-email/zonemta-wildduck) plugin
> At least one limit value must be set
**Example**
```
curl -XPOST "http://localhost:8080/user/quota" -H 'content-type: application/json' -d '{
"username": "testuser",
"quota": 1234567,
"recipients": 500
}'
```
The response for successful operation should look like this:
2017-03-06 22:13:40 +08:00
```json
{
"success": true,
"username": "testuser",
"quota": 1234567,
"recipients": 500
2017-03-06 22:13:40 +08:00
}
```
Quota changes apply immediately.
2017-03-30 01:06:09 +08:00
### POST /user/quota/reset
Recalculates used storage for an user. Use this when it seems that quota counters for an user do not match with reality.
Arguments
- **username** is the username of the user to check
**Example**
```
curl -XPOST "http://localhost:8080/user/quota/reset" -H 'content-type: application/json' -d '{
"username": "testuser"
}'
```
The response for successful operation should look like this:
```json
{
"success": true,
"username": "testuser",
"previousStorageUsed": 1000,
"storageUsed": 800
}
```
Be aware though that this method is not atomic and should be done only if quota counters are way off.
### 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:
```json
{
"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:
```json
{
"success": true,
"username": "testuser",
"quota": 1234567,
"storageUsed": 1822,
"recipients": 500,
"recipientsLimited": false,
"recipientsSent": 47,
"recipientsTtl": 3392,
"addresses": [
{
"id": "58d8fccb645b0deb23d6c37d",
"address": "user@example.com",
"main": true,
"created": "2017-03-27T11:51:39.639Z"
}
]
}
```
Where
* **recipients** is the count of maximum recipients for 24 hour period (starts with the first message)
* **recipientsLimited** if *true* then sending is currently disabled as recipient limit has been reached
* **recipientsSent** how many recipients has been used in the current 24 hour period
* **recipientsTtl** seconds until the end of current period
Recipient limits assume that messages are sent using ZoneMTA with [zonemta-wildduck](https://github.com/wildduck-email/zonemta-wildduck) plugin, otherwise the counters are not updated.
### 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:
```json
{
"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
}
]
}
```
2017-04-03 21:59:04 +08:00
### GET /mailbox/:id
List messages in a mailbox.
Parameters
- **id** is the mailbox ID
- **size** is optional number to limit the length of the messages array (defaults to 20)
2017-04-04 22:09:39 +08:00
- **before** is an optional paging number (see _next_ in response)
- **after** is an optional paging number (see _prev_ in response)
2017-04-03 21:59:04 +08:00
Response includes the following fields
2017-04-04 22:09:39 +08:00
- **mailbox** is an object that lists some metadata about the current mailbox
- **id** is the mailbox ID
- **path** is the folder path
- **next** is an URL fragment for retrieving the next page (or false if there are no more pages)
2017-04-05 01:39:32 +08:00
2017-04-04 22:09:39 +08:00
- **prev** is an URL fragment for retrieving the previous page (or false if it is the first page)
2017-04-05 01:39:32 +08:00
2017-04-04 22:09:39 +08:00
- **messages** is an array of messages in the mailbox
- **id** is the message ID
- **date** is the date when this message was received
- **hasAttachments** is a boolean that indicates if this messages has attachments or not
- **intro** includes the first 256 characters from the message
- **subject** is the message title
- **from** is the From: field
- **to** is the To: field
- **cc** is the Cc: field
- **bcc** is the Bcc: field
2017-04-03 21:59:04 +08:00
The response for successful listing should look like this:
```json
{
"success": true,
"mailbox": {
"id": "58dbf87fcff690a8c30470c7",
"path": "INBOX"
},
"next": "/mailbox/58dbf87fcff690a8c30470c7?before=34&size=20",
"prev": false,
"messages": [
{
"id": "58e25243ab71621c3890417e",
"date": "2017-04-03T13:46:44.226Z",
"hasAttachments": true,
"intro": "Welcome to Ryan Finnie's MIME torture test. This message was designed to introduce a couple of the newer features of MIME-aware MUAs, features that have come around since the days of the original MIME torture test. Just to be clear, this message SUPPLEMENT…",
"subject": "ryan finnie's mime torture test v1.0",
"from": "ryan finnie <rfinnie@domain.dom>",
"to": "bob@domain.dom"
}
]
}
```
### GET /message/:id
Retrieves message information
Parameters
- **id** is the MongoDB _id as a string for a message
- **mailbox** is optional Mailbox id. Use this to verify that the message is located at this mailbox
**Example**
```
curl "http://localhost:8080/message/58d8299c5195c38e77c2daa5"
```
Response message includes the following fields
- **id** is the id of the message
- **headers** is an array that lists all headers of the message. A header is an object:
- **key** is the lowercase key of the header
- **value** is the header value in unicode (all encoded values are decoded to utf-8). The value is capped at around 800 characters.
2017-04-03 21:59:04 +08:00
- **date** is the receive date (not header Date: field)
- **mailbox** is the id of the mailbox this messages belongs to
- **flags** is an array of IMAP flags for this message
- **text** is the plaintext version of the message (derived from html if not present in message source)
- **html** is the HTML version of the message (derived from plaintext if not present in message source). It is an array of strings, each array element corresponds to different MIME node and might have its own html header
2017-04-05 01:39:32 +08:00
2017-04-03 21:59:04 +08:00
- **attachments** is an array of attachment objects. Attachments can be shared between messages.
- **id** is the id of the attachment
- **fileName** is the name of the attachment. Autogenerated from Content-Type if not set in source
- **contentType** is the MIME type of the message
- **disposition** defines Content-Disposition and is either 'inline', 'attachment' or _false_
- **transferEncoding** defines Content-Transfer-Encoding
- **related** is a boolean value that states if the attachment should be hidden (_true_) or not. _Related_ attachments are usually embedded images
- **sizeKb** is the approximate size of the attachment in kilobytes
#### Embedded images
HTML content has embedded images linked with the following URL structure:
2017-04-04 22:09:39 +08:00
```
attachment:MESSAGE_ID/ATTACHMENT_ID
```
2017-04-03 21:59:04 +08:00
For example:
2017-04-04 22:09:39 +08:00
```
<img src="attachment:aaaaaa/bbbbbb">
```
2017-04-03 21:59:04 +08:00
To fetch the actual attachment contents for this image, use the following url:
2017-04-04 22:09:39 +08:00
```
http://localhost:8080/message/aaaaaa/attachment/bbbbbb
```
2017-04-03 21:59:04 +08:00
#### Example response
The response for successful operation should look like this:
```json
{
"success": true,
"message": {
"id": "58d8299c5195c38e77c2daa5",
"mailbox": "58dbf87fcff690a8c30470c7",
"headers": [
{
"key": "delivered-to",
"value": "andris@addrgw.com"
}
],
"date": "2017-04-03T10:34:43.007Z",
"flags": ["\\Seen"],
"text": "Hello world!",
"html": ["<p>Hello world!</p>"],
2017-04-03 21:59:04 +08:00
"attachments": [
{
"id": "58e2254289cccb742fd6c015",
"fileName": "image.png",
"contentType": "image/png",
"disposition": "attachment",
"transferEncoding": "base64",
"related": true,
"sizeKb": 1
}
]
}
}
```
### GET /message/:mid/attachment/:aid
Retrieves an attachment of the message
Parameters
- **mid** is the message ID
- **aid** is the attachment ID
**Example**
```
curl "http://localhost:8080/message/58d8299c5195c38e77c2daa5/attachment/58e2254289cccb742fd6c015"
```
### GET /message/:id/raw
Retrieves RFC822 source of the message
Parameters
- **id** is the MongoDB _id as a string for a message
- **mailbox** is optional Mailbox id. Use this to verify that the message is located at this mailbox
**Example**
```
curl "http://localhost:8080/message/58d8299c5195c38e77c2daa5/raw"
```
### DELETE /message/:id
Deletes a message from a mailbox.
2017-04-03 21:59:04 +08:00
Parameters
- **id** is the MongoDB _id as a string for a message
2017-04-03 21:59:04 +08:00
- **mailbox** is an optional Mailbox id. Use this to verify that the message to be deleted is located at this mailbox
**Example**
```
2017-04-03 21:59:04 +08:00
curl -XDELETE "http://localhost:8080/message/58d8299c5195c38e77c2daa5"
```
The response for successful operation should look like this:
```json
{
"success": true,
2017-04-03 21:59:04 +08:00
"message":{
"id": "58d8299c5195c38e77c2daa5"
}
}
```
2017-03-06 22:13:40 +08:00
## IMAP Protocol Differences
This is a list of known differences from the IMAP specification. Listed differences are either intentional or are bugs that became features.
1. `\Recent` flags is not implemented and most probably never will be (RFC3501 2.3.2.)
2. `RENAME` does not touch subfolders which is against the spec (RFC3501 6.3.5\. _If the name has inferior hierarchical names, then the inferior hierarchical names MUST also be renamed._). Wild Duck stores all folders using flat hierarchy, the "/" separator is fake and only used for listing mailboxes
3. Unsolicited `FLAGS` responses (RFC3501 7.2.6.) and `PERMANENTFLAGS` are not sent (except for as part of `SELECT` and `EXAMINE` responses). Wild Duck notifies about flag updates only with unsolicited FETCH updates.
4. Wild Duck responds with `NO` for `STORE` if matching messages were deleted in another session
5. `CHARSET` argument for the `SEARCH` command is ignored (RFC3501 6.4.4.)
6. Metadata arguments for `SEARCH MODSEQ` are ignored (RFC7162 3.1.5.). You can define `<entry-name>` and `<entry-type-req>` values but these are not used for anything
2017-04-04 15:20:25 +08:00
7. `SEARCH TEXT` and `SEARCH BODY` both use MongoDB [$text index](https://docs.mongodb.com/v3.4/reference/operator/query/text/) against decoded plaintext version of the message. RFC3501 assumes that it should be a string match either against full message (`TEXT`) or body section (`BODY`).
2017-04-05 01:39:32 +08:00
8. What happens when FETCH is called for messages that were deleted in another session? _Not sure, need to check_
Any other differences are most probably real bugs and unintentional.
## Future considerations for IMAP extensions
Wild Duck does not plan to be the most feature-rich IMAP client in the world. Most IMAP extensions are useless because there aren't too many clients that are able to benefit from these extensions. There are a few extensions though that would make sense to be added to Wild Duck
2017-04-05 01:39:32 +08:00
1. IMAP4 non-synchronizing literals, LITERAL- ([RFC7888](https://tools.ietf.org/html/rfc7888)). Synchronized literals are needed for APPEND to check mailbox quota, small values could go with the non-synchronizing version.
2. LIST-STATUS ([RFC5819](https://tools.ietf.org/html/rfc5819))
2017-04-04 22:09:39 +08:00
3. _What else?_ (definitely not NOTIFY nor QRESYNC)
## 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.
2017-04-05 01:39:32 +08:00
## Outbound SMTP
Use [ZoneMTA](https://github.com/zone-eu/zone-mta) with the [ZoneMTA-WildDuck](https://github.com/wildduck-email/zonemta-wildduck) plugin. This gives you an outbound SMTP server that uses Wild Duck accounts for authentication.
2017-03-06 05:45:50 +08:00
## License
Wild Duck Mail Agent is licensed under the [European Union Public License 1.1](http://ec.europa.eu/idabc/eupl.html).