2017-03-07 00:27:04 +08:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
const config = require('config');
|
|
|
|
const log = require('npmlog');
|
|
|
|
const SMTPServer = require('smtp-server').SMTPServer;
|
|
|
|
const crypto = require('crypto');
|
2017-03-21 06:07:23 +08:00
|
|
|
const tools = require('./lib/tools');
|
|
|
|
const MessageHandler = require('./lib/message-handler');
|
2017-04-12 03:50:20 +08:00
|
|
|
const MessageSplitter = require('./lib/message-splitter');
|
2017-03-22 16:30:10 +08:00
|
|
|
const os = require('os');
|
2017-03-27 15:36:45 +08:00
|
|
|
const db = require('./lib/db');
|
|
|
|
|
2017-04-12 16:32:57 +08:00
|
|
|
const maxStorage = config.maxStorage * 1024 * 1024;
|
2017-03-27 15:36:45 +08:00
|
|
|
const maxMessageSize = config.smtp.maxMB * 1024 * 1024;
|
2017-03-07 00:27:04 +08:00
|
|
|
|
2017-03-21 06:07:23 +08:00
|
|
|
let messageHandler;
|
2017-03-07 00:27:04 +08:00
|
|
|
|
|
|
|
const server = new SMTPServer({
|
|
|
|
|
|
|
|
// log to console
|
|
|
|
logger: {
|
|
|
|
info(...args) {
|
|
|
|
args.shift();
|
|
|
|
log.info('SMTP', ...args);
|
|
|
|
},
|
|
|
|
debug(...args) {
|
|
|
|
args.shift();
|
|
|
|
log.silly('SMTP', ...args);
|
|
|
|
},
|
|
|
|
error(...args) {
|
|
|
|
args.shift();
|
|
|
|
log.error('SMTP', ...args);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
name: false,
|
|
|
|
|
|
|
|
// not required but nice-to-have
|
|
|
|
banner: 'Welcome to Wild Duck Mail Agent',
|
|
|
|
|
|
|
|
// disable STARTTLS to allow authentication in clear text mode
|
|
|
|
disabledCommands: ['AUTH', 'STARTTLS'],
|
|
|
|
|
|
|
|
// Accept messages up to 10 MB
|
2017-03-27 15:36:45 +08:00
|
|
|
size: maxMessageSize,
|
2017-03-07 00:27:04 +08:00
|
|
|
|
2017-03-30 01:06:09 +08:00
|
|
|
onMailFrom(address, session, callback) {
|
|
|
|
|
|
|
|
// reset session entries
|
|
|
|
session.users = new Map();
|
|
|
|
|
|
|
|
// accept sender address
|
|
|
|
return callback();
|
|
|
|
},
|
|
|
|
|
2017-03-07 00:27:04 +08:00
|
|
|
// Validate RCPT TO envelope address. Example allows all addresses that do not start with 'deny'
|
|
|
|
// If this method is not set, all addresses are allowed
|
2017-03-22 16:30:10 +08:00
|
|
|
onRcptTo(rcpt, session, callback) {
|
|
|
|
let originalRecipient = tools.normalizeAddress(rcpt.address);
|
|
|
|
let recipient = originalRecipient.replace(/\+[^@]*@/, '@');
|
2017-03-21 06:07:23 +08:00
|
|
|
|
2017-03-30 01:06:09 +08:00
|
|
|
if (session.users.has(recipient)) {
|
2017-03-21 06:07:23 +08:00
|
|
|
return callback();
|
|
|
|
}
|
2017-03-07 00:27:04 +08:00
|
|
|
|
2017-03-27 15:36:45 +08:00
|
|
|
db.database.collection('addresses').findOne({
|
2017-03-22 16:30:10 +08:00
|
|
|
address: recipient
|
|
|
|
}, (err, address) => {
|
2017-03-07 00:27:04 +08:00
|
|
|
if (err) {
|
|
|
|
log.error('SMTP', err);
|
|
|
|
return callback(new Error('Database error'));
|
|
|
|
}
|
2017-03-22 16:30:10 +08:00
|
|
|
if (!address) {
|
2017-03-07 00:27:04 +08:00
|
|
|
return callback(new Error('Unknown recipient'));
|
|
|
|
}
|
|
|
|
|
2017-03-27 15:36:45 +08:00
|
|
|
db.database.collection('users').findOne({
|
|
|
|
_id: address.user
|
|
|
|
}, (err, user) => {
|
|
|
|
if (err) {
|
|
|
|
log.error('SMTP', err);
|
|
|
|
return callback(new Error('Database error'));
|
|
|
|
}
|
2017-03-28 01:53:13 +08:00
|
|
|
if (!user) {
|
2017-03-27 15:36:45 +08:00
|
|
|
return callback(new Error('Unknown recipient'));
|
|
|
|
}
|
2017-03-07 00:27:04 +08:00
|
|
|
|
2017-03-27 15:36:45 +08:00
|
|
|
if (!session.users) {
|
|
|
|
session.users = new Map();
|
|
|
|
}
|
2017-03-07 00:27:04 +08:00
|
|
|
|
2017-03-27 15:36:45 +08:00
|
|
|
let storageAvailable = (Number(user.quota || 0) || maxStorage) - Number(user.storageUsed || 0);
|
|
|
|
|
2017-04-12 03:50:20 +08:00
|
|
|
if (storageAvailable <= 0) {
|
2017-03-27 15:36:45 +08:00
|
|
|
err = new Error('Insufficient channel storage: ' + originalRecipient);
|
|
|
|
err.responseCode = 452;
|
|
|
|
return callback(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
session.users.set(recipient, {
|
|
|
|
recipient: originalRecipient,
|
2017-04-12 03:50:20 +08:00
|
|
|
user: address.user
|
2017-03-27 15:36:45 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
callback();
|
|
|
|
});
|
2017-03-07 00:27:04 +08:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
// Handle message stream
|
|
|
|
onData(stream, session, callback) {
|
|
|
|
let chunks = [];
|
|
|
|
let chunklen = 0;
|
|
|
|
let hash = crypto.createHash('md5');
|
2017-03-27 15:36:45 +08:00
|
|
|
|
2017-04-12 03:50:20 +08:00
|
|
|
let splitter = new MessageSplitter();
|
|
|
|
|
|
|
|
splitter.on('readable', () => {
|
2017-03-07 00:27:04 +08:00
|
|
|
let chunk;
|
2017-04-12 03:50:20 +08:00
|
|
|
while ((chunk = splitter.read()) !== null) {
|
2017-03-27 15:36:45 +08:00
|
|
|
if (chunklen < maxMessageSize) {
|
|
|
|
chunks.push(chunk);
|
|
|
|
chunklen += chunk.length;
|
|
|
|
}
|
2017-03-07 00:27:04 +08:00
|
|
|
hash.update(chunk);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
stream.once('error', err => {
|
|
|
|
log.error('SMTP', err);
|
|
|
|
callback(new Error('Error reading from stream'));
|
|
|
|
});
|
|
|
|
|
2017-04-12 03:50:20 +08:00
|
|
|
splitter.once('end', () => {
|
2017-03-07 00:27:04 +08:00
|
|
|
let err;
|
2017-03-27 15:36:45 +08:00
|
|
|
|
|
|
|
// too large message
|
2017-03-07 00:27:04 +08:00
|
|
|
if (stream.sizeExceeded) {
|
|
|
|
err = new Error('Error: message exceeds fixed maximum message size ' + config.smtp.maxMB + ' MB');
|
|
|
|
err.responseCode = 552;
|
|
|
|
return callback(err);
|
|
|
|
}
|
|
|
|
|
2017-03-27 15:36:45 +08:00
|
|
|
// no recipients defined
|
2017-03-07 00:27:04 +08:00
|
|
|
if (!session.users || !session.users.size) {
|
|
|
|
return callback(new Error('Nowhere to save the mail to'));
|
|
|
|
}
|
|
|
|
|
2017-04-12 03:50:20 +08:00
|
|
|
chunks.unshift(splitter.rawHeaders);
|
|
|
|
chunklen += splitter.rawHeaders.length;
|
2017-03-27 15:36:45 +08:00
|
|
|
|
2017-03-22 16:30:10 +08:00
|
|
|
let queueId = hash.digest('hex').toUpperCase();
|
2017-03-07 00:27:04 +08:00
|
|
|
let users = Array.from(session.users);
|
|
|
|
let stored = 0;
|
|
|
|
let storeNext = () => {
|
|
|
|
if (stored >= users.length) {
|
2017-03-22 16:30:10 +08:00
|
|
|
return callback(null, 'Message queued as ' + queueId);
|
2017-03-07 00:27:04 +08:00
|
|
|
}
|
|
|
|
|
2017-03-22 16:30:10 +08:00
|
|
|
let recipient = users[stored][0];
|
|
|
|
let rcptData = users[stored][1] || {};
|
2017-03-21 06:07:23 +08:00
|
|
|
stored++;
|
2017-03-07 00:27:04 +08:00
|
|
|
|
2017-03-22 16:30:10 +08:00
|
|
|
// create Delivered-To and Received headers
|
|
|
|
let header = Buffer.from(
|
|
|
|
'Delivered-To: ' + recipient + '\r\n' +
|
|
|
|
'Received: ' + generateReceivedHeader(session, queueId, os.hostname(), recipient) + '\r\n'
|
|
|
|
);
|
|
|
|
|
2017-03-07 00:27:04 +08:00
|
|
|
chunks.unshift(header);
|
|
|
|
chunklen += header.length;
|
|
|
|
|
2017-04-12 03:50:20 +08:00
|
|
|
let mailboxQueryKey = 'path';
|
|
|
|
let mailboxQueryValue = 'INBOX';
|
|
|
|
|
|
|
|
if (Array.isArray(splitter.headers)) {
|
|
|
|
for (let i = splitter.headers.length - 1; i >= 0; i--) {
|
|
|
|
let header = splitter.headers[i];
|
|
|
|
|
|
|
|
// check if the header is used for detecting spam
|
|
|
|
if (config.spamHeader && config.spamHeader.toLowerCase() === header.key) {
|
|
|
|
let value = header.line.substr(header.line.indexOf(':') + 1).trim();
|
|
|
|
if (/^yes\b/i.test(value)) {
|
|
|
|
mailboxQueryKey = 'specialUse';
|
|
|
|
mailboxQueryValue = '\\Junk';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-21 06:07:23 +08:00
|
|
|
messageHandler.add({
|
2017-03-22 16:30:10 +08:00
|
|
|
user: rcptData.user,
|
2017-04-12 03:50:20 +08:00
|
|
|
[mailboxQueryKey]: mailboxQueryValue,
|
2017-03-21 06:07:23 +08:00
|
|
|
meta: {
|
|
|
|
source: 'SMTP',
|
|
|
|
from: tools.normalizeAddress(session.envelope.mailFrom && session.envelope.mailFrom.address || ''),
|
2017-03-22 16:30:10 +08:00
|
|
|
to: rcptData.recipient,
|
2017-03-21 06:07:23 +08:00
|
|
|
origin: session.remoteAddress,
|
|
|
|
originhost: session.clientHostname,
|
|
|
|
transhost: session.hostNameAppearsAs,
|
|
|
|
transtype: session.transmissionType,
|
|
|
|
time: Date.now()
|
|
|
|
},
|
|
|
|
date: false,
|
|
|
|
flags: false,
|
2017-04-10 22:12:47 +08:00
|
|
|
raw: Buffer.concat(chunks, chunklen),
|
|
|
|
|
|
|
|
// if similar message exists, then skip
|
|
|
|
skipExisting: true
|
|
|
|
}, (err, inserted) => {
|
2017-03-07 00:27:04 +08:00
|
|
|
// remove Delivered-To
|
|
|
|
chunks.shift();
|
|
|
|
chunklen -= header.length;
|
|
|
|
|
|
|
|
if (err) {
|
2017-03-22 16:30:10 +08:00
|
|
|
log.error('SMTP', err);
|
2017-04-10 22:12:47 +08:00
|
|
|
} else if (!inserted) {
|
|
|
|
log.debug('SMTP', 'Message was not inserted');
|
2017-03-07 00:27:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
storeNext();
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
storeNext();
|
|
|
|
});
|
2017-04-12 03:50:20 +08:00
|
|
|
|
|
|
|
stream.pipe(splitter);
|
2017-03-07 00:27:04 +08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2017-03-21 06:07:23 +08:00
|
|
|
module.exports = done => {
|
2017-03-07 00:27:04 +08:00
|
|
|
if (!config.smtp.enabled) {
|
|
|
|
return setImmediate(() => done(null, false));
|
|
|
|
}
|
|
|
|
|
2017-03-27 15:36:45 +08:00
|
|
|
messageHandler = new MessageHandler(db.database);
|
2017-03-07 00:27:04 +08:00
|
|
|
|
2017-03-27 15:36:45 +08:00
|
|
|
let started = false;
|
2017-03-07 00:27:04 +08:00
|
|
|
|
2017-03-27 15:36:45 +08:00
|
|
|
server.on('error', err => {
|
|
|
|
if (!started) {
|
2017-03-07 00:27:04 +08:00
|
|
|
started = true;
|
2017-03-27 15:36:45 +08:00
|
|
|
return done(err);
|
|
|
|
}
|
|
|
|
log.error('SMTP', err);
|
|
|
|
});
|
|
|
|
|
|
|
|
server.listen(config.smtp.port, config.smtp.host, () => {
|
|
|
|
if (started) {
|
|
|
|
return server.close();
|
|
|
|
}
|
|
|
|
started = true;
|
|
|
|
done(null, server);
|
2017-03-07 00:27:04 +08:00
|
|
|
});
|
|
|
|
};
|
2017-03-22 16:30:10 +08:00
|
|
|
|
|
|
|
function generateReceivedHeader(session, queueId, hostname, recipient) {
|
|
|
|
let origin = session.remoteAddress ? '[' + session.remoteAddress + ']' : '';
|
|
|
|
let originhost = session.clientHostname && session.clientHostname.charAt(0) !== '[' ? session.clientHostname : false;
|
|
|
|
origin = [].concat(origin || []).concat(originhost || []);
|
|
|
|
|
|
|
|
if (origin.length > 1) {
|
|
|
|
origin = '(' + origin.join(' ') + ')';
|
|
|
|
} else {
|
|
|
|
origin = origin.join(' ').trim() || 'localhost';
|
|
|
|
}
|
|
|
|
|
|
|
|
let value = '' +
|
|
|
|
// from ehlokeyword
|
|
|
|
'from' + (session.hostNameAppearsAs ? ' ' + session.hostNameAppearsAs : '') +
|
|
|
|
// [1.2.3.4]
|
|
|
|
' ' + origin +
|
|
|
|
(originhost ? '\r\n' : '') +
|
|
|
|
|
|
|
|
// by smtphost
|
|
|
|
' by ' + hostname +
|
|
|
|
|
|
|
|
// with ESMTP
|
|
|
|
' with ' + session.transmissionType +
|
|
|
|
// id 12345678
|
|
|
|
' id ' + queueId +
|
|
|
|
'\r\n' +
|
|
|
|
|
|
|
|
// for <receiver@example.com>
|
|
|
|
' for <' + recipient + '>' +
|
|
|
|
// (version=TLSv1/SSLv3 cipher=ECDHE-RSA-AES128-GCM-SHA256)
|
|
|
|
(session.tlsOptions ? '\r\n (version=' + session.tlsOptions.version + ' cipher=' + session.tlsOptions.name + ')' : '') +
|
|
|
|
|
|
|
|
';' +
|
|
|
|
'\r\n' +
|
|
|
|
|
|
|
|
// Wed, 03 Aug 2016 11:32:07 +0000
|
|
|
|
' ' + new Date().toUTCString().replace(/GMT/, '+0000');
|
|
|
|
return value;
|
|
|
|
}
|