Files
cloudron-box/src/mail.js

1098 lines
39 KiB
JavaScript
Raw Normal View History

2017-06-28 17:06:12 -05:00
'use strict';
exports = module.exports = {
getStatus: getStatus,
2018-04-03 14:37:52 -07:00
getDomains: getDomains,
2018-04-03 14:37:52 -07:00
getDomain: getDomain,
addDomain: addDomain,
removeDomain: removeDomain,
updateDomain: updateDomain,
2018-03-08 20:08:01 -08:00
addDnsRecords: addDnsRecords,
validateName: validateName,
setMailFromValidation: setMailFromValidation,
setCatchAllAddress: setCatchAllAddress,
setMailRelay: setMailRelay,
2018-01-20 23:17:39 -08:00
setMailEnabled: setMailEnabled,
startMail: restartMail,
sendTestMail: sendTestMail,
getMailboxes: getMailboxes,
2018-02-11 01:18:29 -08:00
removeMailboxes: removeMailboxes,
getMailbox: getMailbox,
addMailbox: addMailbox,
updateMailbox: updateMailbox,
removeMailbox: removeMailbox,
listAliases: listAliases,
getAliases: getAliases,
setAliases: setAliases,
2018-01-26 10:22:50 +01:00
getLists: getLists,
getList: getList,
addList: addList,
updateList: updateList,
2018-01-26 10:22:50 +01:00
removeList: removeList,
2018-01-25 15:38:29 -08:00
_readDkimPublicKeySync: readDkimPublicKeySync,
2018-01-20 18:19:26 -08:00
MailError: MailError
2017-06-28 17:06:12 -05:00
};
var assert = require('assert'),
async = require('async'),
config = require('./config.js'),
DatabaseError = require('./databaseerror.js'),
debug = require('debug')('box:mail'),
dns = require('./native-dns.js'),
domains = require('./domains.js'),
infra = require('./infra_version.js'),
mailboxdb = require('./mailboxdb.js'),
maildb = require('./maildb.js'),
mailer = require('./mailer.js'),
2017-06-28 17:06:12 -05:00
net = require('net'),
nodemailer = require('nodemailer'),
os = require('os'),
path = require('path'),
paths = require('./paths.js'),
reverseProxy = require('./reverseproxy.js'),
safe = require('safetydance'),
shell = require('./shell.js'),
2017-06-28 17:06:12 -05:00
smtpTransport = require('nodemailer-smtp-transport'),
sysinfo = require('./sysinfo.js'),
2018-04-29 10:58:45 -07:00
users = require('./users.js'),
2017-06-28 17:06:12 -05:00
util = require('util'),
_ = require('underscore');
2018-02-08 14:43:49 -08:00
const DNS_OPTIONS = { server: '127.0.0.1', timeout: 5000 }; // unbound runs on 127.0.0.1
var NOOP_CALLBACK = function (error) { if (error) debug(error); };
2018-01-20 18:19:26 -08:00
function MailError(reason, errorOrMessage) {
2017-06-28 17:06:12 -05:00
assert.strictEqual(typeof reason, 'string');
assert(errorOrMessage instanceof Error || typeof errorOrMessage === 'string' || typeof errorOrMessage === 'undefined');
Error.call(this);
Error.captureStackTrace(this, this.constructor);
this.name = this.constructor.name;
this.reason = reason;
if (typeof errorOrMessage === 'undefined') {
this.message = reason;
} else if (typeof errorOrMessage === 'string') {
this.message = errorOrMessage;
} else {
this.message = 'Internal error';
this.nestedError = errorOrMessage;
}
}
2018-01-20 18:19:26 -08:00
util.inherits(MailError, Error);
MailError.INTERNAL_ERROR = 'Internal Error';
2018-05-29 13:31:40 +02:00
MailError.EXTERNAL_ERROR = 'External Error';
2018-01-20 18:19:26 -08:00
MailError.BAD_FIELD = 'Bad Field';
2018-01-24 15:38:19 +01:00
MailError.ALREADY_EXISTS = 'Already Exists';
2018-01-21 00:16:58 -08:00
MailError.NOT_FOUND = 'Not Found';
2018-02-11 00:04:28 -08:00
MailError.IN_USE = 'In Use';
2018-05-29 13:31:40 +02:00
MailError.BILLING_REQUIRED = 'Billing Required';
2017-06-28 17:06:12 -05:00
2018-04-03 09:36:41 -07:00
function validateName(name) {
assert.strictEqual(typeof name, 'string');
2018-04-03 09:36:41 -07:00
if (name.length < 1) return new MailError(MailError.BAD_FIELD, 'mailbox name must be atleast 1 char');
if (name.length >= 200) return new MailError(MailError.BAD_FIELD, 'mailbox name too long');
// +/- can be tricky in emails. also need to consider valid LDAP characters here (e.g '+' is reserved)
2018-04-03 09:36:41 -07:00
if (/[^a-zA-Z0-9.]/.test(name)) return new MailError(MailError.BAD_FIELD, 'mailbox name can only contain alphanumerals and dot');
// app emails are sent using the .app suffix
2018-04-03 09:36:41 -07:00
if (name.indexOf('.app') !== -1) return new MailError(MailError.BAD_FIELD, 'mailbox name pattern is reserved for apps');
return null;
}
function checkOutboundPort25(callback) {
2017-06-28 17:06:12 -05:00
assert.strictEqual(typeof callback, 'function');
var smtpServer = _.sample([
'smtp.gmail.com',
'smtp.live.com',
'smtp.mail.yahoo.com',
'smtp.o2.ie',
'smtp.comcast.net',
2018-02-08 11:48:55 -08:00
'smtp.1und1.de',
]);
var relay = {
value: 'OK',
status: false
};
var client = new net.Socket();
client.setTimeout(5000);
client.connect(25, smtpServer);
client.on('connect', function () {
relay.status = true;
relay.value = 'OK';
client.destroy(); // do not use end() because it still triggers timeout
callback(null, relay);
});
client.on('timeout', function () {
relay.status = false;
relay.value = 'Connect to ' + smtpServer + ' timed out';
client.destroy();
callback(new Error('Timeout'), relay);
});
client.on('error', function (error) {
relay.status = false;
relay.value = 'Connect to ' + smtpServer + ' failed: ' + error.message;
client.destroy();
callback(error, relay);
});
}
function checkSmtpRelay(relay, callback) {
var result = {
value: 'OK',
status: false
};
2017-06-28 17:06:12 -05:00
var transporter = nodemailer.createTransport(smtpTransport({
2018-07-23 17:05:15 -07:00
connectionTimeout: 5000,
greetingTimeout: 5000,
2017-06-28 17:06:12 -05:00
host: relay.host,
port: relay.port,
auth: {
user: relay.username,
pass: relay.password
}
}));
transporter.verify(function(error) {
2017-06-29 10:11:55 -05:00
result.status = !error;
if (error) {
result.value = error.message;
return callback(error, result);
}
2017-06-28 17:06:12 -05:00
2017-12-04 17:10:02 +05:30
callback(null, result);
});
2017-06-28 17:06:12 -05:00
}
function verifyRelay(relay, callback) {
assert.strictEqual(typeof relay, 'object');
2017-06-28 17:06:12 -05:00
assert.strictEqual(typeof callback, 'function');
2018-03-08 23:23:02 -08:00
// we used to verify cloudron-smtp with checkOutboundPort25 but that is unreliable given that we just
// randomly select some smtp server
if (relay.provider === 'cloudron-smtp') return callback();
2017-06-28 17:06:12 -05:00
2018-03-08 23:23:02 -08:00
checkSmtpRelay(relay, function (error) {
2018-01-20 18:19:26 -08:00
if (error) return callback(new MailError(MailError.BAD_FIELD, error.message));
2017-06-28 17:06:12 -05:00
callback();
});
}
2017-06-28 17:06:12 -05:00
2018-01-24 11:33:09 -08:00
function checkDkim(domain, callback) {
var dkim = {
2018-01-24 11:33:09 -08:00
domain: config.dkimSelector() + '._domainkey.' + domain,
type: 'TXT',
expected: null,
value: null,
status: false
};
2017-06-28 17:06:12 -05:00
2018-01-25 14:51:07 -08:00
var dkimKey = readDkimPublicKeySync(domain);
if (!dkimKey) return callback(new Error('Failed to read dkim public key'), dkim);
2017-06-28 17:06:12 -05:00
dkim.expected = 'v=DKIM1; t=s; p=' + dkimKey;
2017-06-28 17:06:12 -05:00
dns.resolve(dkim.domain, dkim.type, DNS_OPTIONS, function (error, txtRecords) {
if (error) return callback(error, dkim);
2017-06-28 17:06:12 -05:00
if (txtRecords.length !== 0) {
dkim.value = txtRecords[0].join('');
dkim.status = (dkim.value === dkim.expected);
}
2017-06-28 17:06:12 -05:00
callback(null, dkim);
});
}
2017-06-28 17:06:12 -05:00
2018-01-24 11:33:09 -08:00
function checkSpf(domain, callback) {
var spf = {
2018-01-24 11:33:09 -08:00
domain: domain,
type: 'TXT',
value: null,
expected: 'v=spf1 a:' + config.mailFqdn() + ' ~all',
status: false
};
dns.resolve(spf.domain, spf.type, DNS_OPTIONS, function (error, txtRecords) {
if (error) return callback(error, spf);
var i;
for (i = 0; i < txtRecords.length; i++) {
let txtRecord = txtRecords[i].join(''); // https://agari.zendesk.com/hc/en-us/articles/202952749-How-long-can-my-SPF-record-be-
if (txtRecord.indexOf('v=spf1 ') !== 0) continue; // not SPF
spf.value = txtRecord;
spf.status = spf.value.indexOf(' a:' + config.adminFqdn()) !== -1;
break;
}
2017-06-28 17:06:12 -05:00
if (spf.status) {
spf.expected = spf.value;
} else if (i !== txtRecords.length) {
spf.expected = 'v=spf1 a:' + config.adminFqdn() + ' ' + spf.value.slice('v=spf1 '.length);
}
2017-06-28 17:06:12 -05:00
callback(null, spf);
});
}
2017-06-28 17:06:12 -05:00
2018-01-24 11:33:09 -08:00
function checkMx(domain, callback) {
var mx = {
2018-01-24 11:33:09 -08:00
domain: domain,
type: 'MX',
value: null,
expected: '10 ' + config.mailFqdn() + '.',
status: false
};
dns.resolve(mx.domain, mx.type, DNS_OPTIONS, function (error, mxRecords) {
if (error) return callback(error, mx);
if (mxRecords.length !== 0) {
mx.status = mxRecords.length == 1 && mxRecords[0].exchange === config.mailFqdn();
mx.value = mxRecords.map(function (r) { return r.priority + ' ' + r.exchange + '.'; }).join(' ');
}
2017-06-28 17:06:12 -05:00
callback(null, mx);
});
}
2017-06-28 17:06:12 -05:00
2018-01-24 11:33:09 -08:00
function checkDmarc(domain, callback) {
var dmarc = {
2018-01-24 11:33:09 -08:00
domain: '_dmarc.' + domain,
type: 'TXT',
value: null,
expected: 'v=DMARC1; p=reject; pct=100',
status: false
};
dns.resolve(dmarc.domain, dmarc.type, DNS_OPTIONS, function (error, txtRecords) {
if (error) return callback(error, dmarc);
if (txtRecords.length !== 0) {
dmarc.value = txtRecords[0].join('');
dmarc.status = (dmarc.value === dmarc.expected);
}
2017-06-28 17:06:12 -05:00
callback(null, dmarc);
});
}
2017-06-28 17:06:12 -05:00
function checkPtr(callback) {
var ptr = {
domain: null,
type: 'PTR',
value: null,
expected: config.mailFqdn(), // any trailing '.' is added by client software (https://lists.gt.net/spf/devel/7918)
status: false
};
2017-06-28 17:06:12 -05:00
sysinfo.getPublicIp(function (error, ip) {
if (error) return callback(error, ptr);
2017-06-28 17:06:12 -05:00
ptr.domain = ip.split('.').reverse().join('.') + '.in-addr.arpa';
2017-06-28 17:06:12 -05:00
dns.resolve(ptr.domain, 'PTR', DNS_OPTIONS, function (error, ptrRecords) {
if (error) return callback(error, ptr);
2017-06-28 17:06:12 -05:00
if (ptrRecords.length !== 0) {
ptr.value = ptrRecords.join(' ');
ptr.status = ptrRecords.some(function (v) { return v === ptr.expected; });
}
2017-06-28 17:06:12 -05:00
return callback(null, ptr);
2017-06-28 17:06:12 -05:00
});
});
}
2017-06-28 17:06:12 -05:00
2017-09-08 11:50:11 -07:00
// https://raw.githubusercontent.com/jawsome/node-dnsbl/master/list.json
const RBL_LIST = [
2018-03-05 14:26:53 -08:00
{
'name': 'Abuse.ch',
'dns': 'spam.abuse.ch',
'site': 'http://abuse.ch/'
},
2017-09-08 11:50:11 -07:00
{
2017-12-04 17:10:02 +05:30
'name': 'Barracuda',
'dns': 'b.barracudacentral.org',
'site': 'http://www.barracudacentral.org/rbl/removal-request'
2017-09-08 11:50:11 -07:00
},
{
2018-03-05 14:26:53 -08:00
'name': 'Composite Blocking List',
'dns': 'cbl.abuseat.org',
'site': 'http://www.abuseat.org'
},
{
'name': 'Multi SURBL',
'dns': 'multi.surbl.org',
'site': 'http://www.surbl.org'
},
{
'name': 'Passive Spam Block List',
'dns': 'psbl.surriel.com',
'site': 'https://psbl.org'
2017-09-08 11:50:11 -07:00
},
{
2017-12-04 17:10:02 +05:30
'name': 'Sorbs Aggregate Zone',
'dns': 'dnsbl.sorbs.net',
'site': 'http://dnsbl.sorbs.net/'
2017-09-08 11:50:11 -07:00
},
{
2017-12-04 17:10:02 +05:30
'name': 'Sorbs spam.dnsbl Zone',
'dns': 'spam.dnsbl.sorbs.net',
'site': 'http://sorbs.net'
2017-09-08 11:50:11 -07:00
},
{
2018-03-05 14:26:53 -08:00
'name': 'SpamCop',
'dns': 'bl.spamcop.net',
'site': 'http://spamcop.net'
},
{
'name': 'SpamHaus Zen',
'dns': 'zen.spamhaus.org',
'site': 'http://spamhaus.org'
2017-09-08 11:50:11 -07:00
},
{
2017-12-04 17:10:02 +05:30
'name': 'The Unsubscribe Blacklist(UBL)',
'dns': 'ubl.unsubscore.com ',
'site': 'http://www.lashback.com/blacklist/'
2017-09-08 11:50:11 -07:00
},
{
2017-12-04 17:10:02 +05:30
'name': 'UCEPROTECT Network',
'dns': 'dnsbl-1.uceprotect.net',
'site': 'http://www.uceprotect.net/en'
2017-09-08 11:50:11 -07:00
}
];
2018-01-24 11:33:09 -08:00
// this function currently only looks for black lists based on IP. TODO: also look up by domain
function checkRblStatus(domain, callback) {
2017-09-13 22:39:42 -07:00
assert.strictEqual(typeof callback, 'function');
2017-09-08 11:50:11 -07:00
sysinfo.getPublicIp(function (error, ip) {
if (error) return callback(error, ip);
var flippedIp = ip.split('.').reverse().join('.');
// https://tools.ietf.org/html/rfc5782
async.map(RBL_LIST, function (rblServer, iteratorDone) {
dns.resolve(flippedIp + '.' + rblServer.dns, 'A', DNS_OPTIONS, function (error, records) {
2017-09-08 11:50:11 -07:00
if (error || !records) return iteratorDone(null, null); // not listed
2018-01-24 11:33:09 -08:00
debug('checkRblStatus: %s (ip: %s) is in the blacklist of %j', domain, flippedIp, rblServer);
2017-09-08 11:50:11 -07:00
var result = _.extend({ }, rblServer);
dns.resolve(flippedIp + '.' + rblServer.dns, 'TXT', DNS_OPTIONS, function (error, txtRecords) {
result.txtRecords = error || !txtRecords ? 'No txt record' : txtRecords.map(x => x.join(''));
2017-09-08 11:50:11 -07:00
2018-01-24 11:33:09 -08:00
debug('checkRblStatus: %s (error: %s) (txtRecords: %j)', domain, error, txtRecords);
2017-09-08 11:50:11 -07:00
return iteratorDone(null, result);
});
});
}, function (ignoredError, blacklistedServers) {
blacklistedServers = blacklistedServers.filter(function(b) { return b !== null; });
2018-01-24 11:33:09 -08:00
debug('checkRblStatus: %s (ip: %s) servers: %j', domain, ip, blacklistedServers);
2017-09-13 22:39:42 -07:00
return callback(null, { status: blacklistedServers.length === 0, ip: ip, servers: blacklistedServers });
});
});
}
2018-01-21 00:40:30 -08:00
function getStatus(domain, callback) {
assert.strictEqual(typeof domain, 'string');
2017-09-13 22:39:42 -07:00
assert.strictEqual(typeof callback, 'function');
// ensure we always have a valid toplevel properties for the api
var results = {
dns: {},
rbl: {},
relay: {}
};
2017-09-13 22:39:42 -07:00
function recordResult(what, func) {
return function (callback) {
func(function (error, result) {
if (error) debug('Ignored error - ' + what + ':', error);
safe.set(results, what, result);
callback();
});
};
}
2017-09-08 11:50:11 -07:00
2018-04-03 14:37:52 -07:00
getDomain(domain, function (error, result) {
2017-09-13 22:39:42 -07:00
if (error) return callback(error);
2017-09-08 11:50:11 -07:00
2017-09-13 22:39:42 -07:00
var checks = [
2018-01-24 11:33:09 -08:00
recordResult('dns.mx', checkMx.bind(null, domain)),
recordResult('dns.dmarc', checkDmarc.bind(null, domain))
2017-09-13 22:39:42 -07:00
];
2018-01-24 11:33:09 -08:00
if (result.relay.provider === 'cloudron-smtp') {
2017-09-13 22:39:42 -07:00
// these tests currently only make sense when using Cloudron's SMTP server at this point
checks.push(
2018-01-24 11:33:09 -08:00
recordResult('dns.spf', checkSpf.bind(null, domain)),
recordResult('dns.dkim', checkDkim.bind(null, domain)),
2017-09-13 22:39:42 -07:00
recordResult('dns.ptr', checkPtr),
recordResult('relay', checkOutboundPort25),
2018-01-24 11:33:09 -08:00
recordResult('rbl', checkRblStatus.bind(null, domain))
2017-09-13 22:39:42 -07:00
);
} else {
2018-01-24 11:33:09 -08:00
checks.push(recordResult('relay', checkSmtpRelay.bind(null, result.relay)));
2017-09-13 22:39:42 -07:00
}
async.parallel(checks, function () {
callback(null, results);
2017-09-08 11:50:11 -07:00
});
});
}
function createMailConfig(callback) {
assert.strictEqual(typeof callback, 'function');
debug('createMailConfig: generating mail config');
2018-04-03 14:37:52 -07:00
getDomains(function (error, mailDomains) {
2018-01-24 11:33:09 -08:00
if (error) return callback(error);
2018-04-29 10:58:45 -07:00
users.getOwner(function (error, owner) {
2018-02-05 15:02:34 -08:00
const mailFqdn = config.mailFqdn();
const defaultDomain = config.adminDomain();
const alertsFrom = `no-reply@${defaultDomain}`;
2018-01-24 11:33:09 -08:00
const alertsTo = config.provider() === 'caas' ? [ 'support@cloudron.io' ] : [ ];
alertsTo.concat(error ? [] : owner.email).join(','); // owner may not exist yet
2018-02-05 15:02:34 -08:00
const mailOutDomains = mailDomains.map(function (d) { return d.domain; }).join(',');
const mailInDomains = mailDomains.filter(function (d) { return d.enabled; }).map(function (d) { return d.domain; }).join(',');
2018-02-05 15:02:34 -08:00
if (!safe.fs.writeFileSync(path.join(paths.ADDON_CONFIG_DIR, 'mail/mail.ini'),
2018-02-10 21:29:00 -08:00
`mail_in_domains=${mailInDomains}\nmail_out_domains=${mailOutDomains}\nmail_default_domain=${defaultDomain}\nmail_server_name=${mailFqdn}\nalerts_from=${alertsFrom}\nalerts_to=${alertsTo}\n\n`, 'utf8')) {
return callback(new Error('Could not create mail var file:' + safe.error.message));
}
// enable_outbound makes plugin forward email for relayed mail. non-relayed mail always hits LMTP plugin first
if (!safe.fs.writeFileSync(path.join(paths.ADDON_CONFIG_DIR, 'mail/smtp_forward.ini'), 'enable_outbound=false\ndomain_selector=mail_from\n', 'utf8')) {
2018-02-05 15:02:34 -08:00
return callback(new Error('Could not create smtp forward file:' + safe.error.message));
}
2018-02-05 15:02:34 -08:00
// create sections for per-domain configuration
mailDomains.forEach(function (domain) {
const catchAll = domain.catchAll.map(function (c) { return `${c}@${domain.domain}`; }).join(',');
const mailFromValidation = domain.mailFromValidation;
2018-02-05 15:02:34 -08:00
if (!safe.fs.appendFileSync(path.join(paths.ADDON_CONFIG_DIR, 'mail/mail.ini'),
`[${domain.domain}]\ncatch_all=${catchAll}\nmail_from_validation=${mailFromValidation}\n\n`, 'utf8')) {
return callback(new Error('Could not create mail var file:' + safe.error.message));
}
const relay = domain.relay;
2018-02-05 15:02:34 -08:00
const enableRelay = relay.provider !== 'cloudron-smtp',
2018-02-05 15:02:34 -08:00
host = relay.host || '',
port = relay.port || 25,
username = relay.username || '',
password = relay.password || '';
if (!enableRelay) return;
2018-02-05 15:02:34 -08:00
if (!safe.fs.appendFileSync(paths.ADDON_CONFIG_DIR + '/mail/smtp_forward.ini',
`[${domain.domain}]\nenable_outbound=true\nhost=${host}\nport=${port}\nenable_tls=true\nauth_type=plain\nauth_user=${username}\nauth_pass=${password}\n\n`, 'utf8')) {
2018-02-05 15:02:34 -08:00
return callback(new Error('Could not create mail var file:' + safe.error.message));
}
});
2018-02-05 15:02:34 -08:00
callback(null, mailInDomains.length !== 0 /* allowInbound */);
});
});
}
function restartMail(callback) {
// mail (note: 2525 is hardcoded in mail container and app use this port)
// MAIL_SERVER_NAME is the hostname of the mailserver i.e server uses these certs
// MAIL_DOMAIN is the domain for which this server is relaying mails
// mail container uses /app/data for backed up data and /run for restart-able data
2018-01-21 00:16:58 -08:00
if (process.env.BOX_ENV === 'test' && !process.env.TEST_CREATE_INFRA) return callback();
2018-01-21 00:06:08 -08:00
const tag = infra.images.mail.tag;
const memoryLimit = Math.max((1 + Math.round(os.totalmem()/(1024*1024*1024)/4)) * 128, 256);
// admin and mail share the same certificate
2018-02-08 15:07:49 +01:00
reverseProxy.getCertificate({ fqdn: config.adminFqdn(), domain: config.adminDomain() }, function (error, bundle) {
if (error) return callback(error);
// the setup script copies dhparams.pem to /addons/mail
const mailCertFilePath = path.join(paths.ADDON_CONFIG_DIR, 'mail/tls_cert.pem');
2018-01-31 22:31:14 -08:00
const mailKeyFilePath = path.join(paths.ADDON_CONFIG_DIR, 'mail/tls_key.pem');
if (!safe.child_process.execSync(`cp ${bundle.certFilePath} ${mailCertFilePath}`)) return callback(new Error('Could not create cert file:' + safe.error.message));
if (!safe.child_process.execSync(`cp ${bundle.keyFilePath} ${mailKeyFilePath}`)) return callback(new Error('Could not create key file:' + safe.error.message));
2018-01-24 11:33:09 -08:00
shell.execSync('startMail', 'docker rm -f mail || true');
2018-01-24 11:33:09 -08:00
createMailConfig(function (error, allowInbound) {
if (error) return callback(error);
2018-01-24 11:33:09 -08:00
var ports = allowInbound ? '-p 587:2525 -p 993:9993 -p 4190:4190 -p 25:2525' : '';
2018-01-24 11:33:09 -08:00
const cmd = `docker run --restart=always -d --name="mail" \
--net cloudron \
--net-alias mail \
2018-06-06 13:41:46 +02:00
--log-driver syslog \
--log-opt syslog-address=udp://127.0.0.1:2514 \
--log-opt syslog-format=rfc5424 \
--log-opt tag=mail \
2018-01-24 11:33:09 -08:00
-m ${memoryLimit}m \
--memory-swap ${memoryLimit * 2}m \
--dns 172.18.0.1 \
--dns-search=. \
-v "${paths.MAIL_DATA_DIR}:/app/data" \
-v "${paths.PLATFORM_DATA_DIR}/addons/mail:/etc/mail" \
${ports} \
2018-03-21 23:15:30 -07:00
-p 127.0.0.1:2020:2020 \
2018-01-24 11:33:09 -08:00
--read-only -v /run -v /tmp ${tag}`;
2018-01-24 11:33:09 -08:00
shell.execSync('startMail', cmd);
2018-01-24 11:33:09 -08:00
callback();
});
});
}
2018-04-03 14:37:52 -07:00
function getDomain(domain, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof callback, 'function');
maildb.get(domain, function (error, result) {
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND));
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
return callback(null, result);
});
}
2018-04-03 14:37:52 -07:00
function getDomains(callback) {
2018-01-24 11:33:09 -08:00
assert.strictEqual(typeof callback, 'function');
maildb.getAll(function (error, results) {
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
return callback(null, results);
});
}
// https://agari.zendesk.com/hc/en-us/articles/202952749-How-long-can-my-SPF-record-be-
function txtRecordsWithSpf(domain, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof callback, 'function');
2018-02-08 12:05:29 -08:00
domains.getDnsRecords('', domain, 'TXT', function (error, txtRecords) {
if (error) return callback(error);
debug('txtRecordsWithSpf: current txt records - %j', txtRecords);
var i, matches, validSpf;
for (i = 0; i < txtRecords.length; i++) {
matches = txtRecords[i].match(/^("?v=spf1) /); // DO backend may return without quotes
if (matches === null) continue;
// this won't work if the entry is arbitrarily "split" across quoted strings
validSpf = txtRecords[i].indexOf('a:' + config.mailFqdn()) !== -1;
break; // there can only be one SPF record
}
if (validSpf) return callback(null, null);
if (!matches) { // no spf record was found, create one
txtRecords.push('"v=spf1 a:' + config.mailFqdn() + ' ~all"');
debug('txtRecordsWithSpf: adding txt record');
} else { // just add ourself
txtRecords[i] = matches[1] + ' a:' + config.mailFqdn() + txtRecords[i].slice(matches[1].length);
debug('txtRecordsWithSpf: inserting txt record');
}
return callback(null, txtRecords);
});
}
function ensureDkimKeySync(domain) {
assert.strictEqual(typeof domain, 'string');
const dkimPath = path.join(paths.MAIL_DATA_DIR, `dkim/${domain}`);
const dkimPrivateKeyFile = path.join(dkimPath, 'private');
const dkimPublicKeyFile = path.join(dkimPath, 'public');
const dkimSelectorFile = path.join(dkimPath, 'selector');
2018-03-08 15:29:18 -08:00
if (safe.fs.existsSync(dkimPublicKeyFile) &&
safe.fs.existsSync(dkimPublicKeyFile) &&
safe.fs.existsSync(dkimPublicKeyFile)) {
debug(`Reusing existing DKIM keys for ${domain}`);
return null;
}
debug(`Generating new DKIM keys for ${domain}`);
if (!safe.fs.mkdirSync(dkimPath) && safe.error.code !== 'EEXIST') {
debug('Error creating dkim.', safe.error);
return new MailError(MailError.INTERNAL_ERROR, safe.error);
}
if (!safe.child_process.execSync('openssl genrsa -out ' + dkimPrivateKeyFile + ' 1024')) return new MailError(MailError.INTERNAL_ERROR, safe.error);
if (!safe.child_process.execSync('openssl rsa -in ' + dkimPrivateKeyFile + ' -out ' + dkimPublicKeyFile + ' -pubout -outform PEM')) return new MailError(MailError.INTERNAL_ERROR, safe.error);
if (!safe.fs.writeFileSync(dkimSelectorFile, config.dkimSelector(), 'utf8')) return new MailError(MailError.INTERNAL_ERROR, safe.error);
return null;
}
2018-01-25 14:51:07 -08:00
function readDkimPublicKeySync(domain) {
assert.strictEqual(typeof domain, 'string');
var dkimPath = path.join(paths.MAIL_DATA_DIR, `dkim/${domain}`);
var dkimPublicKeyFile = path.join(dkimPath, 'public');
var publicKey = safe.fs.readFileSync(dkimPublicKeyFile, 'utf8');
if (publicKey === null) {
debug('Error reading dkim public key.', safe.error);
return null;
}
// remove header, footer and new lines
publicKey = publicKey.split('\n').slice(1, -2).join('');
return publicKey;
}
function addDnsRecords(domain, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof callback, 'function');
var error = ensureDkimKeySync(domain);
if (error) return callback(error);
2018-03-08 16:02:13 -08:00
if (process.env.BOX_ENV === 'test') return callback();
2018-01-25 14:51:07 -08:00
var dkimKey = readDkimPublicKeySync(domain);
if (!dkimKey) return callback(new MailError(MailError.INTERNAL_ERROR, new Error('Failed to read dkim public key')));
// t=s limits the domainkey to this domain and not it's subdomains
var dkimRecord = { subdomain: config.dkimSelector() + '._domainkey', domain: domain, type: 'TXT', values: [ '"v=DKIM1; t=s; p=' + dkimKey + '"' ] };
var records = [ ];
records.push(dkimRecord);
debug('addDnsRecords: %j', records);
2018-03-08 17:53:44 -08:00
txtRecordsWithSpf(domain, function (error, txtRecords) {
if (error) return callback(error);
2018-03-08 17:53:44 -08:00
if (txtRecords) records.push({ subdomain: '', domain: domain, type: 'TXT', values: txtRecords });
2018-03-08 17:53:44 -08:00
debug('addDnsRecords: will update %j', records);
2018-03-08 17:53:44 -08:00
async.mapSeries(records, function (record, iteratorCallback) {
domains.upsertDnsRecords(record.subdomain, record.domain, record.type, record.values, iteratorCallback);
}, function (error, changeIds) {
if (error) debug('addDnsRecords: failed to update : %s. will retry', error);
else debug('addDnsRecords: records %j added with changeIds %j', records, changeIds);
2018-03-08 17:53:44 -08:00
callback(error);
});
});
}
2018-04-03 14:37:52 -07:00
function addDomain(domain, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof callback, 'function');
maildb.add(domain, function (error) {
if (error && error.reason === DatabaseError.ALREADY_EXISTS) return callback(new MailError(MailError.ALREADY_EXISTS, 'Domain already exists'));
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND, 'No such domain'));
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
async.series([
addDnsRecords.bind(null, domain), // do this first to ensure DKIM keys
restartMail
], NOOP_CALLBACK); // do these asynchronously
callback();
});
}
// this is just a way to resync the mail "dns" records via the UI
2018-04-03 14:37:52 -07:00
function updateDomain(domain, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof callback, 'function');
2018-04-03 14:37:52 -07:00
getDomain(domain, function (error) {
if (error) return callback(error);
addDnsRecords(domain, NOOP_CALLBACK);
callback();
});
}
2018-04-03 14:37:52 -07:00
function removeDomain(domain, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof callback, 'function');
maildb.del(domain, function (error) {
if (error && error.reason === DatabaseError.IN_USE) return callback(new MailError(MailError.IN_USE));
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND, error.message));
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
restartMail(NOOP_CALLBACK);
callback();
});
}
2018-01-20 23:17:39 -08:00
function setMailFromValidation(domain, enabled, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof enabled, 'boolean');
assert.strictEqual(typeof callback, 'function');
2018-01-21 00:06:08 -08:00
maildb.update(domain, { mailFromValidation: enabled }, function (error) {
2018-01-20 23:17:39 -08:00
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND));
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
restartMail(NOOP_CALLBACK); // have to restart mail container since haraka cannot watch symlinked config files (mail.ini)
callback(null);
});
}
function setCatchAllAddress(domain, addresses, callback) {
2018-01-20 23:17:39 -08:00
assert.strictEqual(typeof domain, 'string');
assert(Array.isArray(addresses));
assert.strictEqual(typeof callback, 'function');
maildb.update(domain, { catchAll: addresses }, function (error) {
2018-01-20 23:17:39 -08:00
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND));
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
restartMail(NOOP_CALLBACK); // have to restart mail container since haraka cannot watch symlinked config files (mail.ini)
callback(null);
});
}
2018-01-20 23:17:39 -08:00
function setMailRelay(domain, relay, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof relay, 'object');
assert.strictEqual(typeof callback, 'function');
verifyRelay(relay, function (error) {
if (error) return callback(error);
2018-01-20 23:17:39 -08:00
maildb.update(domain, { relay: relay }, function (error) {
2018-06-13 07:51:22 -07:00
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND));
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
restartMail(NOOP_CALLBACK);
callback(null);
});
});
}
2018-01-20 23:17:39 -08:00
function setMailEnabled(domain, enabled, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof enabled, 'boolean');
assert.strictEqual(typeof callback, 'function');
2018-06-09 18:26:00 -07:00
maildb.update(domain, { enabled: enabled }, function (error) {
2018-06-13 07:51:22 -07:00
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND));
2018-06-09 18:26:00 -07:00
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
2018-01-24 11:33:09 -08:00
2018-06-09 18:26:00 -07:00
restartMail(NOOP_CALLBACK);
2018-01-24 11:33:09 -08:00
2018-06-09 18:26:00 -07:00
if (!enabled || process.env.BOX_ENV === 'test') return callback(null);
2018-01-24 11:33:09 -08:00
2018-06-09 18:26:00 -07:00
// Add MX and DMARC record. Note that DMARC policy depends on DKIM signing and thus works
// only if we use our internal mail server.
var records = [
{ subdomain: '_dmarc', type: 'TXT', values: [ '"v=DMARC1; p=reject; pct=100"' ] },
{ subdomain: '', type: 'MX', values: [ '10 ' + config.mailFqdn() + '.' ] }
];
2018-05-29 13:31:40 +02:00
2018-06-09 18:26:00 -07:00
async.mapSeries(records, function (record, iteratorCallback) {
domains.upsertDnsRecords(record.subdomain, domain, record.type, record.values, iteratorCallback);
}, NOOP_CALLBACK);
2018-05-29 13:31:40 +02:00
2018-06-09 18:26:00 -07:00
callback(null);
});
}
function sendTestMail(domain, to, callback) {
assert.strictEqual(typeof domain, 'string');
2018-02-03 18:27:55 -08:00
assert.strictEqual(typeof to, 'string');
assert.strictEqual(typeof callback, 'function');
2018-04-03 14:37:52 -07:00
getDomain(domain, function (error, result) {
if (error) return callback(error);
mailer.sendTestMail(result.domain, to);
callback();
});
}
function getMailboxes(domain, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof callback, 'function');
mailboxdb.listMailboxes(domain, function (error, result) {
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
callback(null, result);
});
}
2018-02-11 01:18:29 -08:00
function removeMailboxes(domain, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof callback, 'function');
mailboxdb.delByDomain(domain, function (error) {
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
callback();
});
}
function getMailbox(name, domain, callback) {
assert.strictEqual(typeof name, 'string');
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof callback, 'function');
mailboxdb.getMailbox(name, domain, function (error, result) {
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND, 'no such mailbox'));
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
callback(null, result);
});
}
function addMailbox(name, domain, userId, callback) {
assert.strictEqual(typeof name, 'string');
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof userId, 'string');
assert.strictEqual(typeof callback, 'function');
2018-04-05 16:07:51 -07:00
name = name.toLowerCase();
2018-04-05 16:07:51 -07:00
var error = validateName(name);
if (error) return callback(error);
2018-04-03 14:27:09 -07:00
2018-04-07 18:33:30 -07:00
mailboxdb.addMailbox(name, domain, userId, mailboxdb.OWNER_TYPE_USER, function (error) {
if (error && error.reason === DatabaseError.ALREADY_EXISTS) return callback(new MailError(MailError.ALREADY_EXISTS, `mailbox ${name} already exists`));
2018-04-05 16:07:51 -07:00
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
callback(null);
});
}
function updateMailbox(name, domain, userId, callback) {
assert.strictEqual(typeof name, 'string');
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof userId, 'string');
assert.strictEqual(typeof callback, 'function');
2018-04-05 16:07:51 -07:00
name = name.toLowerCase();
var error = validateName(name);
if (error) return callback(error);
2018-04-07 18:33:30 -07:00
mailboxdb.updateMailbox(name, domain, userId, mailboxdb.OWNER_TYPE_USER, function (error) {
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND, 'no such mailbox'));
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
callback(null);
});
}
function removeMailbox(name, domain, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof name, 'string');
assert.strictEqual(typeof callback, 'function');
mailboxdb.del(name, domain, function (error) {
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND, 'no such mailbox'));
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
callback(null);
});
}
function listAliases(domain, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof callback, 'function');
mailboxdb.listAliases(domain, function (error, result) {
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND, error.message));
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
callback(null, result);
});
}
function getAliases(name, domain, callback) {
assert.strictEqual(typeof name, 'string');
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof callback, 'function');
getMailbox(name, domain, function (error) {
if (error) return callback(error);
mailboxdb.getAliasesForName(name, domain, function (error, aliases) {
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND, 'no such mailbox'));
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
callback(null, aliases);
});
});
}
function setAliases(name, domain, aliases, callback) {
assert.strictEqual(typeof name, 'string');
assert.strictEqual(typeof domain, 'string');
assert(Array.isArray(aliases));
assert.strictEqual(typeof callback, 'function');
for (var i = 0; i < aliases.length; i++) {
aliases[i] = aliases[i].toLowerCase();
2018-04-03 09:36:41 -07:00
var error = validateName(aliases[i]);
if (error) return callback(error);
}
mailboxdb.setAliasesForName(name, domain, aliases, function (error) {
if (error && error.reason === DatabaseError.ALREADY_EXISTS && error.message.indexOf('mailboxes_name_domain_unique_index') !== -1) {
2018-06-09 18:26:00 -07:00
var aliasMatch = error.message.match(new RegExp(`^ER_DUP_ENTRY: Duplicate entry '(.*)-${domain}' for key 'mailboxes_name_domain_unique_index'$`));
if (!aliasMatch) return callback(new MailError(MailError.ALREADY_EXISTS, error.message));
return callback(new MailError(MailError.ALREADY_EXISTS, `Mailbox, mailinglist or alias for ${aliasMatch[1]} already exists`));
}
if (error && error.reason === DatabaseError.ALREADY_EXISTS) return callback(new MailError(MailError.ALREADY_EXISTS, error.message));
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND, 'no such mailbox'));
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
callback(null);
});
}
2018-01-26 10:22:50 +01:00
function getLists(domain, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof callback, 'function');
mailboxdb.listGroups(domain, function (error, result) {
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
callback(null, result);
});
}
function getList(domain, listName, callback) {
2018-01-26 10:22:50 +01:00
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof listName, 'string');
2018-01-26 10:22:50 +01:00
assert.strictEqual(typeof callback, 'function');
mailboxdb.getGroup(listName, domain, function (error, result) {
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND, 'no such list'));
2018-01-26 10:22:50 +01:00
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
callback(null, result);
2018-01-26 10:22:50 +01:00
});
}
2018-04-05 16:07:51 -07:00
function addList(name, domain, members, callback) {
2018-01-26 10:22:50 +01:00
assert.strictEqual(typeof domain, 'string');
2018-04-05 16:07:51 -07:00
assert.strictEqual(typeof name, 'string');
assert(Array.isArray(members));
2018-01-26 10:22:50 +01:00
assert.strictEqual(typeof callback, 'function');
2018-04-05 16:07:51 -07:00
name = name.toLowerCase();
var error = validateName(name);
if (error) return callback(error);
for (var i = 0; i < members.length; i++) {
members[i] = members[i].toLowerCase();
error = validateName(members[i]);
if (error) return callback(error);
}
mailboxdb.addGroup(name, domain, members, function (error) {
if (error && error.reason === DatabaseError.ALREADY_EXISTS) return callback(new MailError(MailError.ALREADY_EXISTS, 'list already exits'));
2018-01-26 10:22:50 +01:00
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
callback();
2018-01-26 10:22:50 +01:00
});
}
function updateList(name, domain, members, callback) {
assert.strictEqual(typeof name, 'string');
assert.strictEqual(typeof domain, 'string');
assert(Array.isArray(members));
assert.strictEqual(typeof callback, 'function');
2018-04-05 16:07:51 -07:00
name = name.toLowerCase();
var error = validateName(name);
if (error) return callback(error);
for (var i = 0; i < members.length; i++) {
members[i] = members[i].toLowerCase();
error = validateName(members[i]);
if (error) return callback(error);
}
mailboxdb.updateList(name, domain, members, function (error) {
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND, 'no such mailbox'));
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
callback(null);
});
}
function removeList(domain, listName, callback) {
2018-01-26 10:22:50 +01:00
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof listName, 'string');
2018-01-26 10:22:50 +01:00
assert.strictEqual(typeof callback, 'function');
mailboxdb.del(listName, domain, function (error) {
if (error && error.reason === DatabaseError.NOT_FOUND) return callback(new MailError(MailError.NOT_FOUND, 'no such list'));
2018-01-26 10:22:50 +01:00
if (error) return callback(new MailError(MailError.INTERNAL_ERROR, error));
callback();
2018-01-26 10:22:50 +01:00
});
}