Files
NodeBB/src/emailer.js

198 lines
5.2 KiB
JavaScript
Raw Normal View History

2017-02-18 01:56:23 -07:00
'use strict';
2014-03-13 00:49:32 -04:00
var async = require('async');
2016-02-17 20:22:00 +02:00
var winston = require('winston');
var nconf = require('nconf');
var templates = require('templates.js');
var nodemailer = require('nodemailer');
var sendmailTransport = require('nodemailer-sendmail-transport');
var smtpTransport = require('nodemailer-smtp-transport');
2016-02-17 20:22:00 +02:00
var htmlToText = require('html-to-text');
var url = require('url');
var User = require('./user');
var Plugins = require('./plugins');
var meta = require('./meta');
var translator = require('./translator');
var pubsub = require('./pubsub');
2016-02-17 20:22:00 +02:00
var transports = {
sendmail: nodemailer.createTransport(sendmailTransport()),
2017-02-17 19:31:21 -07:00
gmail: undefined,
2016-02-17 20:22:00 +02:00
};
var app;
var fallbackTransport;
2014-11-29 12:12:02 -05:00
2017-05-26 23:55:20 -04:00
var Emailer = module.exports;
Emailer._defaultPayload = {};
2014-11-29 12:12:02 -05:00
2017-05-26 23:55:20 -04:00
Emailer.registerApp = function (expressApp) {
app = expressApp;
2014-11-29 12:12:02 -05:00
2017-07-28 16:43:32 -04:00
var logo = null;
if (meta.configs.hasOwnProperty('brand:emailLogo')) {
logo = (!meta.config['brand:emailLogo'].startsWith('http') ? nconf.get('url') : '') + meta.config['brand:emailLogo'];
}
2017-07-25 09:45:40 -04:00
Emailer._defaultPayload = {
url: nconf.get('url'),
site_title: meta.config.title || 'NodeBB',
logo: {
2017-07-25 09:45:40 -04:00
src: logo,
height: meta.config['brand:emailLogo:height'],
width: meta.config['brand:emailLogo:width'],
},
};
2017-05-26 23:55:20 -04:00
// Enable Gmail transport if enabled in ACP
if (parseInt(meta.config['email:GmailTransport:enabled'], 10) === 1) {
transports.gmail = nodemailer.createTransport(smtpTransport({
host: 'smtp.gmail.com',
port: 465,
secure: true,
auth: {
user: meta.config['email:GmailTransport:user'],
pass: meta.config['email:GmailTransport:pass'],
2017-02-17 19:31:21 -07:00
},
2017-05-26 23:55:20 -04:00
}));
fallbackTransport = transports.gmail;
} else {
fallbackTransport = transports.sendmail;
}
2015-06-28 21:54:21 -04:00
// Update default payload if new logo is uploaded
pubsub.on('config:update', function (config) {
if (config) {
Emailer._defaultPayload.logo.src = config['brand:emailLogo'];
Emailer._defaultPayload.logo.height = config['brand:emailLogo:height'];
Emailer._defaultPayload.logo.width = config['brand:emailLogo:width'];
}
});
2017-05-26 23:55:20 -04:00
return Emailer;
};
2015-06-28 21:54:21 -04:00
2017-05-26 23:55:20 -04:00
Emailer.send = function (template, uid, params, callback) {
callback = callback || function () {};
if (!app) {
winston.warn('[emailer] App not ready!');
return callback();
}
// Combined passed-in payload with default values
params = Object.assign({}, Emailer._defaultPayload, params);
2017-05-26 23:55:20 -04:00
async.waterfall([
function (next) {
async.parallel({
email: async.apply(User.getUserField, uid, 'email'),
settings: async.apply(User.getSettings, uid),
}, next);
},
function (results, next) {
if (!results.email) {
winston.warn('uid : ' + uid + ' has no email, not sending.');
return next();
}
2017-05-26 23:55:20 -04:00
params.uid = uid;
Emailer.sendToEmail(template, results.email, results.settings.userLang, params, next);
},
], callback);
};
2015-11-06 14:01:34 -05:00
2017-05-26 23:55:20 -04:00
Emailer.sendToEmail = function (template, email, language, params, callback) {
callback = callback || function () {};
var lang = language || meta.config.defaultLang || 'en-GB';
async.waterfall([
function (next) {
async.parallel({
html: function (next) {
renderAndTranslate('emails/' + template, params, lang, next);
},
subject: function (next) {
translator.translate(params.subject, lang, function (translated) {
next(null, translated);
});
},
}, next);
},
function (results, next) {
var data = {
_raw: params,
to: email,
from: meta.config['email:from'] || 'no-reply@' + getHostname(),
from_name: meta.config['email:from_name'] || 'NodeBB',
subject: results.subject,
html: results.html,
plaintext: htmlToText.fromString(results.html, {
ignoreImage: true,
}),
template: template,
uid: params.uid,
pid: params.pid,
fromUid: params.fromUid,
};
Plugins.fireHook('filter:email.modify', data, next);
},
function (data, next) {
if (Plugins.hasListeners('filter:email.send')) {
Plugins.fireHook('filter:email.send', data, next);
} else {
Emailer.sendViaFallback(data, next);
}
},
], function (err) {
if (err && err.code === 'ENOENT') {
callback(new Error('[[error:sendmail-not-found]]'));
2015-11-06 14:01:34 -05:00
} else {
2017-05-26 23:55:20 -04:00
callback(err);
2015-11-06 14:01:34 -05:00
}
2017-05-26 23:55:20 -04:00
});
};
2015-11-06 14:01:34 -05:00
2017-05-26 23:55:20 -04:00
Emailer.sendViaFallback = function (data, callback) {
// Some minor alterations to the data to conform to nodemailer standard
data.text = data.plaintext;
delete data.plaintext;
2015-06-28 21:54:21 -04:00
2017-05-26 23:55:20 -04:00
// NodeMailer uses a combined "from"
data.from = data.from_name + '<' + data.from + '>';
delete data.from_name;
winston.verbose('[emailer] Sending email to uid ' + data.uid + ' (' + data.to + ')');
fallbackTransport.sendMail(data, function (err) {
if (err) {
winston.error(err);
}
callback();
});
};
2015-11-08 12:28:48 -05:00
2017-05-26 23:55:20 -04:00
function render(tpl, params, next) {
if (meta.config['email:custom:' + tpl.replace('emails/', '')]) {
var text = templates.parse(meta.config['email:custom:' + tpl.replace('emails/', '')], params);
next(null, text);
} else {
app.render(tpl, params, next);
2016-02-17 20:22:00 +02:00
}
2017-05-26 23:55:20 -04:00
}
function renderAndTranslate(tpl, params, lang, callback) {
render(tpl, params, function (err, html) {
translator.translate(html, lang, function (translated) {
callback(err, translated);
});
});
}
function getHostname() {
var configUrl = nconf.get('url');
var parsed = url.parse(configUrl);
2017-05-26 23:55:20 -04:00
return parsed.hostname;
}