Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/Flolagale/mailin
Artisanal inbound emails for every web app
https://github.com/Flolagale/mailin
Last synced: about 1 month ago
JSON representation
Artisanal inbound emails for every web app
- Host: GitHub
- URL: https://github.com/Flolagale/mailin
- Owner: Flolagale
- License: mit
- Created: 2014-03-20T16:49:11.000Z (over 10 years ago)
- Default Branch: master
- Last Pushed: 2021-08-27T14:15:31.000Z (over 3 years ago)
- Last Synced: 2024-03-14T11:47:35.665Z (9 months ago)
- Language: Python
- Homepage: mailin.io
- Size: 605 KB
- Stars: 1,944
- Watchers: 52
- Forks: 152
- Open Issues: 58
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- awesome-github-repos - Flolagale/mailin - Artisanal inbound emails for every web app (Python)
- awesome-starred - Flolagale/mailin - Artisanal inbound emails for every web app (others)
README
# Mailin [![Build Status](https://travis-ci.org/Flolagale/mailin.svg?branch=master)](https://travis-ci.org/Flolagale/mailin)
__Artisanal inbound emails for every web app__
Mailin is an smtp server that listens for emails, parses them and posts them as json to the url of your choice.
It checks the incoming emails [dkim](http://en.wikipedia.org/wiki/DomainKeys_Identified_Mail), [spf](http://en.wikipedia.org/wiki/Sender_Policy_Framework), spam score (using [spamassassin](http://spamassassin.apache.org/)) and tells you in which language the email is written.Mailin can be used as a standalone application directly from the command line, or embedded inside a node application.
Mailin relies heavily on the excellent work of [@andris9](https://github.com/andris9) for the smtp and mail parsing services.
Why? Because we needed it for our startup [jokund.com](http://jokund.com).
### Show me a demo!
Sure! A demo is live at [mailin.io](http://mailin.io). Please note that it is running on the smallest Digital Ocean instance, so be fair if it is overloaded.### Initial setup
#### Dependencies
Mailin can run without any dependencies other than node itself, but having them allow you to use some additional features.
So first make sure the node is available, and the ```node``` command as well. On Debian/Ubuntu boxes:
```
sudo aptitude install nodejs ; sudo ln -s $(which nodejs) /usr/bin/node
```To handle dkim and spf checking, Mailin depends on Python 2.7. On Linux machines, it is very not likely that you don't have a decent version of Python available.
To handle the spam score computation, Mailin depends on spamassassin and its server interface spamc. Both should be available as packages on your machine. For instance on Debian/Ubuntu boxes:
```
sudo aptitude install spamassassin spamc
```
Spamassassin is not enabled by default, enable it in ```/etc/default/spamassassin```.#### Node versions
The latest version of Mailin (^3.0.0) runs on node ~0.12.0 or iojs ^2.0.0. If you are running an older version of node such as ~0.10.0, you can install Mailin ^2.0.0:
```
npm install [email protected]
```#### The crux: setting up your DNS correctly
In order to receive emails, your smtp server address should be made available somewhere. Two records should be added to your DNS records. Let us pretend that we want to receive emails at ```*@subdomain.domain.com```:
* First an MX record: ```subdomain.domain.com MX 10 mxsubdomain.domain.com```. This means that the mail server for addresses like ```*@subdomain.domain.com``` will be ```mxsubdomain.domain.com```.
* Then an A record: ```mxsubdomain.domain.com A the.ip.address.of.your.mailin.server```. This tells at which ip address the mail server can be found.You can fire up Mailin (see next section) and use an [smtp server tester](http://mxtoolbox.com/diagnostic.aspx) to verify that everything is correct.
### Using Mailin
#### From the command lineInstall mailin globally.
```
sudo npm install -g mailin
```Run it, specifying your webhook url (addtionnal help can be found using ```mailin --help```). By default, Mailin will listen on port 25, the standard smtp port. you can change this port for testing purpose using the ```--port``` option. However, do not change this port if you want to receive emails from the real world.
Ports number under 1000 are reserved to root user. So two options here. Either run Mailin as root:
```
sudo mailin --webhook http://mydomain.com/incoming_emails
```
Or, prefered choice, use something like ```authbind``` to run Mailin with a standard user while still using port 25.
Here comes a [tutorial on how to setup authbind](http://respectthecode.tumblr.com/post/16461876216/using-authbind-to-run-node-js-on-port-80-with-dreamhost). In this case, do something like:
```
authbind --deep mailin --webhook http://mydomain.com/incoming_emails
```
and make sure that your user can write to the log file.At this point, Mailin will listen for incoming emails, parse them and post an urlencoded form ```multipart/form-data``` to your webhook url.
##### Webhook format
The webhook payload is a multipart form with a ```mailinMsg``` fields always present and some optional additional fields containing the content of the attachments. How to handle this? We have got you covered, there is a working example using node and express in [mailin/samples/server.js](https://github.com/Flolagale/mailin/blob/master/samples/server.js). Anyway, once parsed, you should end up with something like:
```
{
mailinMsg:
{
html: 'Hello world!',
text: 'Hello world!',
headers: {
from: 'John Doe ',
to: 'Jane Doe ',
'content-type': 'multipart/mixed; boundary="----mailcomposer-?=_1-1395066415427"',
'mime-version': '1.0'
},
priority: 'normal',
from: [{
address: '[email protected]',
name: 'John Doe'
}],
to: [{
address: '[email protected]',
name: 'Jane Doe'
}],
attachments: [{
contentType: 'text/plain',
fileName: 'dummyFile.txt',
contentDisposition: 'attachment',
transferEncoding: 'base64',
generatedFileName: 'dummyFile.txt',
contentId: '6e4a9c577e603de61e554abab84f6297@mailparser',
checksum: 'e9fa6319356c536b962650eda9399a44',
length: '28'
}],
connection:
from: 'John Doe ',
to: ['Jane Doe '],
remoteAddress: '91.142.31.23',
authentication: { username: false, authenticated: false, state: 'NORMAL' },
id: '0e9b7099'
},
dkim: 'failed',
spf: 'pass',
spamScore: 3.3,
language: 'english',
cc: [{
address: '[email protected]',
name: 'James'
}],
envelopeFrom: [ { address: '[email protected]', name: 'John Doe' } ],
envelopeTo: [ { address: '[email protected]', name: 'Jane Doe' } ]
},
'dummyFile.txt': 'a-base64-encoded-string=='
}
```##### Gotchas
* ```error: listen EACCES```: your user do not have sufficients privileges to run on the given port. Ports under 1000 are restricted to root user. Try with [sudo](http://xkcd.com/149/).
* ```error: listen EADDRINUSE```: the current port is already used by something. Most likely, you are trying to use port 25 and your machine's [mail transport agent](http://en.wikipedia.org/wiki/Message_transfer_agent) is already running. Stop it with something like ```sudo service exim4 stop``` or ```sudo service postfix stop``` before using Mailin.
* ```error: Unable to compute spam score ECONNREFUSED```: it is likely that spamassassin is not enabled on your machine, check the ```/etc/default/spamassassin``` file.
* ```node: command not found```: most likely, your system does not have node installed or it is installed with a different name. For instance on Debian/Ubuntu, the node interpreter is called nodejs. The quick fix is making a symlink: ```ln -s $(which nodejs) /usr/bin/node``` to make the node command available.
* ```Uncaught SenderError: Mail from command failed - 450 4.1.8 : Sender address rejected: Domain not found```: The smtpOption `disableDNSValidation` is set to `false` and an email was sent from an invalid domain.#### Embedded inside a node application
Install Mailin locally.
```
sudo npm install --save mailin
```Start the Mailin server and listen to events.
```javascript
var mailin = require('mailin');/* Start the Mailin server. The available options are:
* options = {
* port: 25,
* webhook: 'http://mydomain.com/mailin/incoming,
* disableWebhook: false,
* logFile: '/some/local/path',
* logLevel: 'warn', // One of silly, info, debug, warn, error
* smtpOptions: { // Set of options directly passed to simplesmtp.createServer(smtpOptions)
* SMTPBanner: 'Hi from a custom Mailin instance',
* // By default, the DNS validation of the sender and recipient domains is disabled so.
* // You can enable it as follows:
* disableDNSValidation: false
* }
* };
* Here disable the webhook posting so that you can do what you want with the
* parsed message. */
mailin.start({
port: 25,
disableWebhook: true // Disable the webhook posting.
});/* Access simplesmtp server instance. */
mailin.on('authorizeUser', function(connection, username, password, done) {
if (username == "johnsmith" && password == "mysecret") {
done(null, true);
} else {
done(new Error("Unauthorized!"), false);
}
});/* Event emitted when a connection with the Mailin smtp server is initiated. */
mailin.on('startMessage', function (connection) {
/* connection = {
from: '[email protected]',
to: '[email protected]',
id: 't84h5ugf',
authentication: { username: null, authenticated: false, status: 'NORMAL' }
}
}; */
console.log(connection);
});/* Event emitted after a message was received and parsed. */
mailin.on('message', function (connection, data, content) {
console.log(data);
/* Do something useful with the parsed message here.
* Use parsed message `data` directly or use raw message `content`. */
});
```##### Events
* **startData** *(connection)* - DATA stream is opened by the client.
* **data** *(connection, chunk)* - E-mail data chunk is passed from the client.
* **dataReady** *(connection, callback)* - Client has finished passing e-mail data. `callback` returns the queue id to the client.
* **authorizeUser** *(connection, username, password, callback)* - Emitted if `requireAuthentication` option is set to true. `callback` has two parameters *(err, success)* where `success` is a Boolean and should be true, if user is authenticated successfully.
* **validateSender** *(connection, email, callback)* - Emitted if `validateSender` listener is set up.
* **senderValidationFailed** *(connection, email, callback)* - Emitted if a sender DNS validation failed.
* **validateRecipient** *(connection, email, callback)* - Emitted if `validateRecipients` listener is set up.
* **recipientValidationFailed** *(connection, email, callback)* - Emitted if a recipient DNS validation failed.
* **close** *(connection)* - Emitted when the connection to a client is closed.
* **startMessage** *(connection)* - Connection with the Mailin smtp server is initiated.
* **message** *(connection, data, content)* - Message was received and parsed.### Todo
If webhook fails, schedule some retries.Notice: Postman image copyright [Charlie Allen](http://charlieallensblog.blogspot.fr)