Files
cloudron-box/src/mailer.js

467 lines
16 KiB
JavaScript
Raw Normal View History

'use strict';
exports = module.exports = {
userAdded: userAdded,
userRemoved: userRemoved,
adminChanged: adminChanged,
passwordReset: passwordReset,
appUpdatesAvailable: appUpdatesAvailable,
2017-07-21 16:13:44 +02:00
sendDigest: sendDigest,
sendInvite: sendInvite,
2019-02-11 12:32:02 -08:00
appUp: appUp,
2015-08-04 14:31:40 +02:00
appDied: appDied,
2019-05-07 12:04:28 +02:00
appUpdated: appUpdated,
oomEvent: oomEvent,
2015-08-04 14:31:40 +02:00
2016-10-14 14:46:34 -07:00
backupFailed: backupFailed,
2016-01-22 17:37:41 -08:00
certificateRenewalError: certificateRenewalError,
2016-03-19 20:40:03 -07:00
sendTestMail: sendTestMail,
_mailQueue: [] // accumulate mails in test mode
};
2017-11-14 20:34:25 -08:00
var assert = require('assert'),
config = require('./config.js'),
debug = require('debug')('box:mailer'),
docker = require('./docker.js').connection,
ejs = require('ejs'),
nodemailer = require('nodemailer'),
path = require('path'),
safe = require('safetydance'),
2016-10-13 11:14:17 +02:00
settings = require('./settings.js'),
showdown = require('showdown'),
smtpTransport = require('nodemailer-smtp-transport'),
2019-01-10 10:51:31 -08:00
util = require('util');
var NOOP_CALLBACK = function (error) { if (error) debug(error); };
var MAIL_TEMPLATES_DIR = path.join(__dirname, 'mail_templates');
// This will collect the most common details required for notification emails
function getMailConfig(callback) {
assert.strictEqual(typeof callback, 'function');
2019-03-10 16:05:27 -07:00
settings.getCloudronName(function (error, cloudronName) {
// this is not fatal
if (error) {
debug(error);
cloudronName = 'Cloudron';
}
2019-03-10 16:05:27 -07:00
callback(null, {
cloudronName: cloudronName,
notificationFrom: `"${cloudronName}" <no-reply@${config.adminDomain()}>`
});
});
}
function sendMail(mailOptions, callback) {
assert.strictEqual(typeof mailOptions, 'object');
callback = callback || NOOP_CALLBACK;
if (process.env.BOX_ENV === 'test') {
exports._mailQueue.push(mailOptions);
return callback();
}
2016-05-16 08:18:33 -07:00
docker.getContainer('mail').inspect(function (error, data) {
if (error) return callback(error);
2016-06-20 18:26:22 -05:00
var mailServerIp = safe.query(data, 'NetworkSettings.Networks.cloudron.IPAddress');
if (!mailServerIp) return callback('Error querying mail server IP');
2018-12-28 13:32:37 -08:00
// extract the relay token for auth
const env = safe.query(data, 'Config.Env', null);
if (!env) return callback(new Error('Error getting mail env'));
const tmp = env.find(function (e) { return e.indexOf('CLOUDRON_RELAY_TOKEN') === 0; });
if (!tmp) return callback(new Error('Error getting CLOUDRON_RELAY_TOKEN env var'));
const relayToken = tmp.slice('CLOUDRON_RELAY_TOKEN'.length + 1); // +1 for the = sign
if (!relayToken) return callback(new Error('Error parsing CLOUDRON_RELAY_TOKEN'));
var transport = nodemailer.createTransport(smtpTransport({
host: mailServerIp,
2018-12-28 13:32:37 -08:00
port: config.get('smtpPort'),
auth: {
user: mailOptions.authUser || `no-reply@${config.adminDomain()}`,
2018-12-28 13:32:37 -08:00
pass: relayToken
}
}));
transport.sendMail(mailOptions, function (error) {
if (error) return callback(error);
debug(`Email "${mailOptions.subject}" sent to ${mailOptions.to}`);
callback(null);
});
});
}
function render(templateFile, params) {
assert.strictEqual(typeof templateFile, 'string');
assert.strictEqual(typeof params, 'object');
var content = null;
try {
content = ejs.render(safe.fs.readFileSync(path.join(MAIL_TEMPLATES_DIR, templateFile), 'utf8'), params);
} catch (e) {
2017-08-04 11:15:55 -07:00
debug(`Error rendering ${templateFile}`, e);
}
return content;
}
2019-01-10 12:00:04 +01:00
function mailUserEvent(mailTo, user, event) {
assert.strictEqual(typeof mailTo, 'string');
assert.strictEqual(typeof user, 'object');
assert.strictEqual(typeof event, 'string');
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
var mailOptions = {
from: mailConfig.notificationFrom,
to: mailTo,
subject: util.format('[%s] %s %s', mailConfig.cloudronName, user.username || user.fallbackEmail || user.email, event),
text: render('user_event.ejs', { user: user, event: event, format: 'text' }),
};
sendMail(mailOptions);
2019-01-10 12:00:04 +01:00
});
}
function sendInvite(user, invitor) {
assert.strictEqual(typeof user, 'object');
assert(typeof invitor === 'object');
debug('Sending invite mail');
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
2016-10-13 11:14:17 +02:00
var templateData = {
user: user,
webadminUrl: config.adminOrigin(),
setupLink: `${config.adminOrigin()}/api/v1/session/account/setup.html?reset_token=${user.resetToken}&email=${encodeURIComponent(user.email)}`,
2016-10-13 11:14:17 +02:00
invitor: invitor,
cloudronName: mailConfig.cloudronName,
2016-10-13 11:14:17 +02:00
cloudronAvatarUrl: config.adminOrigin() + '/api/v1/cloudron/avatar'
};
var templateDataText = JSON.parse(JSON.stringify(templateData));
templateDataText.format = 'text';
var templateDataHTML = JSON.parse(JSON.stringify(templateData));
templateDataHTML.format = 'html';
2016-10-13 11:14:17 +02:00
var mailOptions = {
from: mailConfig.notificationFrom,
to: user.fallbackEmail,
subject: util.format('Welcome to %s', mailConfig.cloudronName),
text: render('welcome_user.ejs', templateDataText),
html: render('welcome_user.ejs', templateDataHTML)
2016-10-13 11:14:17 +02:00
};
sendMail(mailOptions);
2016-10-13 11:14:17 +02:00
});
}
2019-01-10 12:00:04 +01:00
function userAdded(mailTo, user) {
assert.strictEqual(typeof mailTo, 'string');
assert.strictEqual(typeof user, 'object');
2019-01-10 12:00:04 +01:00
debug(`userAdded: Sending mail for added users ${user.fallbackEmail} to ${mailTo}`);
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
var templateData = {
user: user,
cloudronName: mailConfig.cloudronName,
cloudronAvatarUrl: config.adminOrigin() + '/api/v1/cloudron/avatar'
};
2016-10-13 12:37:25 +02:00
var templateDataText = JSON.parse(JSON.stringify(templateData));
templateDataText.format = 'text';
2017-01-17 15:34:50 +01:00
var templateDataHTML = JSON.parse(JSON.stringify(templateData));
templateDataHTML.format = 'html';
2017-01-17 15:34:50 +01:00
var mailOptions = {
from: mailConfig.notificationFrom,
2019-01-10 12:00:04 +01:00
to: mailTo,
subject: util.format('[%s] User %s added', mailConfig.cloudronName, user.fallbackEmail),
text: render('user_added.ejs', templateDataText),
html: render('user_added.ejs', templateDataHTML)
};
2016-10-13 12:37:25 +02:00
sendMail(mailOptions);
});
}
2019-01-10 12:00:04 +01:00
function userRemoved(mailTo, user) {
assert.strictEqual(typeof mailTo, 'string');
assert.strictEqual(typeof user, 'object');
2019-01-10 12:00:04 +01:00
debug('Sending mail for userRemoved.', user.id, user.username, user.email);
2019-01-10 12:00:04 +01:00
mailUserEvent(mailTo, user, 'was removed');
}
2019-01-10 12:00:04 +01:00
function adminChanged(mailTo, user, isAdmin) {
assert.strictEqual(typeof mailTo, 'string');
assert.strictEqual(typeof user, 'object');
2019-01-10 12:00:04 +01:00
assert.strictEqual(typeof isAdmin, 'boolean');
debug('Sending mail for adminChanged');
2019-01-10 12:00:04 +01:00
mailUserEvent(mailTo, user, isAdmin ? 'is now an admin' : 'is no more an admin');
}
function passwordReset(user) {
assert.strictEqual(typeof user, 'object');
debug('Sending mail for password reset for user %s.', user.email, user.id);
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
2016-10-13 16:44:09 +02:00
var templateData = {
user: user,
resetLink: `${config.adminOrigin()}/api/v1/session/password/reset.html?reset_token=${user.resetToken}&email=${encodeURIComponent(user.email)}`,
cloudronName: mailConfig.cloudronName,
2016-10-13 16:44:09 +02:00
cloudronAvatarUrl: config.adminOrigin() + '/api/v1/cloudron/avatar'
};
2016-10-13 16:44:09 +02:00
var templateDataText = JSON.parse(JSON.stringify(templateData));
templateDataText.format = 'text';
var templateDataHTML = JSON.parse(JSON.stringify(templateData));
templateDataHTML.format = 'html';
var mailOptions = {
from: mailConfig.notificationFrom,
to: user.fallbackEmail,
subject: util.format('[%s] Password Reset', mailConfig.cloudronName),
2016-10-13 16:44:09 +02:00
text: render('password_reset.ejs', templateDataText),
html: render('password_reset.ejs', templateDataHTML)
};
sendMail(mailOptions);
2016-10-13 16:44:09 +02:00
});
}
2019-02-11 12:32:02 -08:00
function appUp(mailTo, app) {
assert.strictEqual(typeof mailTo, 'string');
assert.strictEqual(typeof app, 'object');
debug('Sending mail for app %s @ %s up', app.id, app.fqdn);
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
var mailOptions = {
from: mailConfig.notificationFrom,
to: mailTo,
subject: util.format('[%s] App %s is back online', mailConfig.cloudronName, app.fqdn),
text: render('app_up.ejs', { title: app.manifest.title, appFqdn: app.fqdn, format: 'text' })
};
sendMail(mailOptions);
2019-02-11 12:32:02 -08:00
});
}
2019-01-10 12:00:04 +01:00
function appDied(mailTo, app) {
assert.strictEqual(typeof mailTo, 'string');
assert.strictEqual(typeof app, 'object');
2017-01-17 16:02:42 +01:00
debug('Sending mail for app %s @ %s died', app.id, app.fqdn);
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
var mailOptions = {
from: mailConfig.notificationFrom,
2019-01-10 12:00:04 +01:00
to: mailTo,
subject: util.format('[%s] App %s is down', mailConfig.cloudronName, app.fqdn),
text: render('app_down.ejs', { title: app.manifest.title, appFqdn: app.fqdn, format: 'text' })
};
sendMail(mailOptions);
});
}
function appUpdated(mailTo, app, callback) {
2019-05-07 12:04:28 +02:00
assert.strictEqual(typeof mailTo, 'string');
assert.strictEqual(typeof app, 'object');
callback = callback || NOOP_CALLBACK;
2019-05-07 12:04:28 +02:00
debug('Sending mail for app %s @ %s updated', app.id, app.fqdn);
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
var mailOptions = {
from: mailConfig.notificationFrom,
to: mailTo,
subject: util.format('[%s] App %s was updated', mailConfig.cloudronName, app.fqdn),
text: render('app_updated.ejs', { title: app.manifest.title, appFqdn: app.fqdn, version: app.manifest.version, format: 'text' })
};
sendMail(mailOptions, callback);
2019-05-07 12:04:28 +02:00
});
}
function appUpdatesAvailable(mailTo, apps, hasSubscription, callback) {
2019-03-10 16:05:27 -07:00
assert.strictEqual(typeof mailTo, 'string');
assert.strictEqual(typeof apps, 'object');
2017-11-02 18:31:51 -07:00
assert.strictEqual(typeof hasSubscription, 'boolean');
2019-03-10 16:05:27 -07:00
assert.strictEqual(typeof callback, 'function');
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
var converter = new showdown.Converter();
apps.forEach(function (app) {
app.changelogHTML = converter.makeHtml(app.updateInfo.manifest.changelog);
});
var templateData = {
webadminUrl: config.adminOrigin(),
hasSubscription: hasSubscription,
apps: apps,
cloudronName: mailConfig.cloudronName,
cloudronAvatarUrl: config.adminOrigin() + '/api/v1/cloudron/avatar'
};
var templateDataText = JSON.parse(JSON.stringify(templateData));
templateDataText.format = 'text';
var templateDataHTML = JSON.parse(JSON.stringify(templateData));
templateDataHTML.format = 'html';
var mailOptions = {
from: mailConfig.notificationFrom,
2019-03-10 16:05:27 -07:00
to: mailTo,
subject: `New app updates available for ${mailConfig.cloudronName}`,
text: render('app_updates_available.ejs', templateDataText),
html: render('app_updates_available.ejs', templateDataHTML)
};
sendMail(mailOptions, callback);
});
}
2019-03-10 16:05:27 -07:00
function sendDigest(mailTo, info, callback) {
assert.strictEqual(typeof mailTo, 'string');
2017-07-21 16:13:44 +02:00
assert.strictEqual(typeof info, 'object');
2019-03-10 16:05:27 -07:00
assert.strictEqual(typeof callback, 'function');
2017-07-21 16:13:44 +02:00
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
2017-07-21 16:13:44 +02:00
var templateData = {
webadminUrl: config.adminOrigin(),
cloudronName: mailConfig.cloudronName,
cloudronAvatarUrl: config.adminOrigin() + '/api/v1/cloudron/avatar',
info: info
};
var templateDataText = JSON.parse(JSON.stringify(templateData));
templateDataText.format = 'text';
var templateDataHTML = JSON.parse(JSON.stringify(templateData));
templateDataHTML.format = 'html';
var mailOptions = {
from: mailConfig.notificationFrom,
2019-03-10 16:05:27 -07:00
to: mailTo,
subject: util.format('[%s] Weekly activity digest', mailConfig.cloudronName),
text: render('digest.ejs', templateDataText),
html: render('digest.ejs', templateDataHTML)
};
sendMail(mailOptions, callback);
2017-07-21 16:13:44 +02:00
});
}
function backupFailed(mailTo, errorMessage, logUrl) {
assert.strictEqual(typeof mailTo, 'string');
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
2016-10-14 14:46:34 -07:00
var mailOptions = {
from: mailConfig.notificationFrom,
to: mailTo,
subject: util.format('[%s] Failed to backup', mailConfig.cloudronName),
text: render('backup_failed.ejs', { cloudronName: mailConfig.cloudronName, message: errorMessage, logUrl: logUrl, format: 'text' })
};
sendMail(mailOptions);
2016-10-14 14:46:34 -07:00
});
}
function certificateRenewalError(mailTo, domain, message) {
assert.strictEqual(typeof mailTo, 'string');
2016-03-19 20:40:03 -07:00
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof message, 'string');
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
2016-03-19 20:40:03 -07:00
var mailOptions = {
from: mailConfig.notificationFrom,
to: mailTo,
subject: util.format('[%s] Certificate renewal error', domain),
text: render('certificate_renewal_error.ejs', { domain: domain, message: message, format: 'text' })
};
sendMail(mailOptions);
});
2016-03-19 20:40:03 -07:00
}
2019-03-06 11:54:37 -08:00
function oomEvent(mailTo, program, event) {
2019-01-10 12:00:04 +01:00
assert.strictEqual(typeof mailTo, 'string');
assert.strictEqual(typeof program, 'string');
2019-03-06 11:54:37 -08:00
assert.strictEqual(typeof event, 'object');
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
var mailOptions = {
from: mailConfig.notificationFrom,
2019-01-10 12:00:04 +01:00
to: mailTo,
2019-03-06 11:54:37 -08:00
subject: util.format('[%s] %s was restarted (OOM)', mailConfig.cloudronName, program),
2019-03-06 15:59:35 -08:00
text: render('oom_event.ejs', { cloudronName: mailConfig.cloudronName, program: program, event: JSON.stringify(event), format: 'text' })
};
sendMail(mailOptions);
});
}
2019-04-15 16:47:43 -07:00
function sendTestMail(domain, email, callback) {
assert.strictEqual(typeof domain, 'string');
assert.strictEqual(typeof email, 'string');
2019-04-15 16:47:43 -07:00
assert.strictEqual(typeof callback, 'function');
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
var mailOptions = {
authUser: `no-reply@${domain}`,
2018-02-03 18:27:55 -08:00
from: `"${mailConfig.cloudronName}" <no-reply@${domain}>`,
to: email,
subject: util.format('Test Email from %s', mailConfig.cloudronName),
text: render('test.ejs', { cloudronName: mailConfig.cloudronName, format: 'text'})
};
2019-04-15 16:47:43 -07:00
sendMail(mailOptions, callback);
});
}