Files
cloudron-box/src/mailer.js

323 lines
11 KiB
JavaScript
Raw Normal View History

'use strict';
exports = module.exports = {
passwordReset,
boxUpdateAvailable,
appUpdatesAvailable,
sendInvite,
backupFailed,
2016-01-22 17:37:41 -08:00
certificateRenewalError,
boxUpdateError,
2016-03-19 20:40:03 -07:00
sendTestMail,
_mailQueue: [] // accumulate mails in test mode
};
2017-11-14 20:34:25 -08:00
var assert = require('assert'),
2019-10-24 13:34:14 -07:00
BoxError = require('./boxerror.js'),
debug = require('debug')('box:mailer'),
ejs = require('ejs'),
2019-11-05 19:54:53 -08:00
mail = require('./mail.js'),
nodemailer = require('nodemailer'),
path = require('path'),
safe = require('safetydance'),
2016-10-13 11:14:17 +02:00
settings = require('./settings.js'),
showdown = require('showdown'),
2020-11-19 23:38:59 +01:00
translation = require('./translation.js'),
smtpTransport = require('nodemailer-smtp-transport');
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) {
2020-02-05 14:30:56 -08:00
if (error) debug('Error getting cloudron name: ', error);
settings.getSupportConfig(function (error, supportConfig) {
if (error) debug('Error getting support config: ', error);
callback(null, {
cloudronName: cloudronName || '',
notificationFrom: `"${cloudronName}" <no-reply@${settings.adminDomain()}>`,
supportEmail: supportConfig.email
});
});
});
}
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();
}
2019-11-05 19:54:53 -08:00
mail.getMailAuth(function (error, data) {
if (error) return callback(error);
var transport = nodemailer.createTransport(smtpTransport({
2019-11-05 19:54:53 -08:00
host: data.ip,
port: data.port,
2018-12-28 13:32:37 -08:00
auth: {
user: mailOptions.authUser || `no-reply@${settings.adminDomain()}`,
2019-11-05 19:54:53 -08:00
pass: data.relayToken
}
}));
transport.sendMail(mailOptions, function (error) {
2019-10-24 13:34:14 -07:00
if (error) return callback(new BoxError(BoxError.EXTERNAL_ERROR, error));
debug(`Email "${mailOptions.subject}" sent to ${mailOptions.to}`);
callback(null);
});
});
}
2020-11-19 23:38:59 +01:00
function render(templateFile, params, translationAssets) {
assert.strictEqual(typeof templateFile, 'string');
assert.strictEqual(typeof params, 'object');
var content = null;
2020-11-19 23:38:59 +01:00
var raw = safe.fs.readFileSync(path.join(MAIL_TEMPLATES_DIR, templateFile), 'utf8');
if (raw === null) {
debug(`Error loading ${templateFile}`);
return '';
}
if (typeof translationAssets === 'object') raw = translation.translate(raw, translationAssets.translations || {}, translationAssets.fallback || {});
try {
2020-11-19 23:38:59 +01:00
content = ejs.render(raw, params);
} catch (e) {
2017-08-04 11:15:55 -07:00
debug(`Error rendering ${templateFile}`, e);
}
return content;
}
function sendInvite(user, invitor, inviteLink) {
assert.strictEqual(typeof user, 'object');
assert.strictEqual(typeof invitor, 'object');
assert.strictEqual(typeof inviteLink, 'string');
debug('Sending invite mail');
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
2020-11-19 23:38:59 +01:00
translation.getTranslations(function (error, translationAssets) {
if (error) return debug('Error getting translations:', error);
var templateData = {
2020-11-20 16:10:04 +01:00
user: user.displayName || user.username || user.email,
2020-11-19 23:38:59 +01:00
webadminUrl: settings.adminOrigin(),
inviteLink: inviteLink,
2020-11-20 16:10:04 +01:00
invitor: invitor ? invitor.email : null,
2020-11-19 23:38:59 +01:00
cloudronName: mailConfig.cloudronName,
cloudronAvatarUrl: settings.adminOrigin() + '/api/v1/cloudron/avatar'
};
var mailOptions = {
from: mailConfig.notificationFrom,
to: user.fallbackEmail,
2020-11-21 00:13:07 +01:00
subject: ejs.render(translation.translate('{{ welcomeEmail.subject }}', translationAssets.translations || {}, translationAssets.fallback || {}), { cloudron: mailConfig.cloudronName }),
2020-11-19 23:38:59 +01:00
text: render('welcome_user-text.ejs', templateData, translationAssets),
html: render('welcome_user-html.ejs', templateData, translationAssets)
};
sendMail(mailOptions);
});
2016-10-13 11:14:17 +02:00
});
}
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);
2020-11-20 16:18:24 +01:00
translation.getTranslations(function (error, translationAssets) {
if (error) return debug('Error getting translations:', error);
2016-10-13 16:44:09 +02:00
2020-11-20 16:18:24 +01:00
var templateData = {
user: user.displayName || user.username || user.email,
resetLink: `${settings.adminOrigin()}/login.html?resetToken=${user.resetToken}`,
cloudronName: mailConfig.cloudronName,
cloudronAvatarUrl: settings.adminOrigin() + '/api/v1/cloudron/avatar'
};
2016-10-13 16:44:09 +02:00
2020-11-20 16:18:24 +01:00
var mailOptions = {
from: mailConfig.notificationFrom,
to: user.fallbackEmail,
2020-11-21 00:13:07 +01:00
subject: ejs.render(translation.translate('{{ passwordResetEmail.subject }}', translationAssets.translations || {}, translationAssets.fallback || {}), { cloudron: mailConfig.cloudronName }),
2020-11-20 16:18:24 +01:00
text: render('password_reset-text.ejs', templateData, translationAssets),
html: render('password_reset-html.ejs', templateData, translationAssets)
};
2016-10-13 16:44:09 +02:00
2020-11-20 16:18:24 +01:00
sendMail(mailOptions);
});
2016-10-13 16:44:09 +02:00
});
}
function boxUpdateAvailable(mailTo, updateInfo, callback) {
assert.strictEqual(typeof mailTo, 'string');
assert.strictEqual(typeof updateInfo, 'object');
assert.strictEqual(typeof callback, 'function');
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
var converter = new showdown.Converter();
var templateData = {
webadminUrl: settings.adminOrigin(),
newBoxVersion: updateInfo.version,
changelog: updateInfo.changelog,
changelogHTML: updateInfo.changelog.map(function (e) { return converter.makeHtml(e); }),
cloudronName: mailConfig.cloudronName,
cloudronAvatarUrl: settings.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,
to: mailTo,
2020-12-29 17:51:41 -08:00
subject: `[${mailConfig.cloudronName}] Cloudron update available`,
text: render('box_update_available.ejs', templateDataText),
html: render('box_update_available.ejs', templateDataHTML)
};
sendMail(mailOptions, callback);
});
}
function appUpdatesAvailable(mailTo, apps, callback) {
2019-03-10 16:05:27 -07:00
assert.strictEqual(typeof mailTo, 'string');
assert.strictEqual(typeof apps, 'object');
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: settings.adminOrigin(),
apps: apps,
cloudronName: mailConfig.cloudronName,
cloudronAvatarUrl: settings.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,
2020-12-29 17:51:41 -08:00
subject: `[${mailConfig.cloudronName}] App update available`,
text: render('app_updates_available.ejs', templateDataText),
html: render('app_updates_available.ejs', templateDataHTML)
};
sendMail(mailOptions, callback);
});
}
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,
2020-12-29 17:51:41 -08:00
subject: `[${mailConfig.cloudronName}] Failed to backup`,
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,
2020-12-29 17:51:41 -08:00
subject: `[${mailConfig.cloudronName}] Certificate renewal error`,
text: render('certificate_renewal_error.ejs', { domain: domain, message: message, format: 'text' })
};
sendMail(mailOptions);
});
2016-03-19 20:40:03 -07:00
}
function boxUpdateError(mailTo, message) {
assert.strictEqual(typeof mailTo, 'string');
assert.strictEqual(typeof message, 'string');
getMailConfig(function (error, mailConfig) {
if (error) return debug('Error getting mail details:', error);
var mailOptions = {
from: mailConfig.notificationFrom,
to: mailTo,
2020-12-29 17:51:41 -08:00
subject: `[${mailConfig.cloudronName}] Cloudron update error`,
text: render('box_update_error.ejs', { message: message, 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,
2020-12-29 17:51:41 -08:00
subject: `[${mailConfig.cloudronName}] Test Email`,
text: render('test.ejs', { cloudronName: mailConfig.cloudronName, format: 'text'})
};
2019-04-15 16:47:43 -07:00
sendMail(mailOptions, callback);
});
}