|
|
@ -6,28 +6,25 @@ var async = require('async'); |
|
|
|
var log = require('npmlog'); |
|
|
|
log.debug = log.verbose; |
|
|
|
var fs = require('fs'); |
|
|
|
var nodemailer = require('nodemailer'); |
|
|
|
|
|
|
|
var Model = require('./model'); |
|
|
|
|
|
|
|
var EMAIL_TYPES = { |
|
|
|
'NewTxProposal': { |
|
|
|
filename: 'new_tx_proposal', |
|
|
|
notifyCreator: false, |
|
|
|
notifyDoer: false, |
|
|
|
}, |
|
|
|
'NewOutgoingTx': { |
|
|
|
filename: 'new_outgoing_tx', |
|
|
|
notifyCreator: true, |
|
|
|
notifyDoer: true, |
|
|
|
}, |
|
|
|
'NewIncomingTx': { |
|
|
|
filename: 'new_incoming_tx', |
|
|
|
notifyCreator: true, |
|
|
|
notifyDoer: true, |
|
|
|
}, |
|
|
|
'TxProposalFinallyRejected': { |
|
|
|
filename: 'txp_finally_rejected', |
|
|
|
notifyCreator: true, |
|
|
|
notifyDoer: false, |
|
|
|
}, |
|
|
|
}; |
|
|
@ -36,10 +33,17 @@ var EMAIL_TYPES = { |
|
|
|
function EmailService(opts) { |
|
|
|
this.storage = opts.storage; |
|
|
|
this.lock = opts.lock; |
|
|
|
this.mailer = opts.mailer || nodemailer.createTransport(opts.email); |
|
|
|
$.checkState(this.mailer); |
|
|
|
}; |
|
|
|
|
|
|
|
// TODO: cache for X minutes
|
|
|
|
EmailService.prototype._readTemplate = function(filename, cb) { |
|
|
|
fs.readFile(__dirname + '/templates/' + filename + '.plain', 'utf8', function(err, template) { |
|
|
|
if (err) { |
|
|
|
log.error('Could not read template file ' + filename, err); |
|
|
|
return cb(err); |
|
|
|
} |
|
|
|
var lines = template.split('\n'); |
|
|
|
return cb(null, { |
|
|
|
subject: _.template(lines[0]), |
|
|
@ -50,20 +54,20 @@ EmailService.prototype._readTemplate = function(filename, cb) { |
|
|
|
|
|
|
|
EmailService.prototype._applyTemplate = function(template, data, cb) { |
|
|
|
var result = _.mapValues(template, function(t) { |
|
|
|
// TODO: If this throws exception, log and abort email generation
|
|
|
|
try { |
|
|
|
return t(data); |
|
|
|
} catch (e) { |
|
|
|
log.error('Could not apply data to template', e); |
|
|
|
return cb(e); |
|
|
|
} |
|
|
|
}); |
|
|
|
return cb(null, result); |
|
|
|
}; |
|
|
|
|
|
|
|
EmailService.prototype._generateFromNotification = function(notification, cb) { |
|
|
|
var self = this; |
|
|
|
|
|
|
|
var emailType = EMAIL_TYPES[notification.type]; |
|
|
|
if (!emailType) return cb(); |
|
|
|
|
|
|
|
self.storage.fetchPreferences(notification.walletId, null, function(err, preferences) { |
|
|
|
if (_.isEmpty(preferences)) return cb(); |
|
|
|
EmailService.prototype._getEmailAddresses = function(walletId, cb) { |
|
|
|
self.storage.fetchPreferences(walletId, null, function(err, preferences) { |
|
|
|
if (err) return cb(err); |
|
|
|
if (_.isEmpty(preferences)) return cb(null, {}); |
|
|
|
|
|
|
|
var addressesByCopayer = _.reduce(preferences, function(memo, p) { |
|
|
|
if (p.email) { |
|
|
@ -72,40 +76,70 @@ EmailService.prototype._generateFromNotification = function(notification, cb) { |
|
|
|
return memo; |
|
|
|
}, {}); |
|
|
|
|
|
|
|
if (_.isEmpty(addressesByCopayer)) return cb(); |
|
|
|
return cb(null, addressesByCopayer); |
|
|
|
}); |
|
|
|
}; |
|
|
|
|
|
|
|
self._readTemplate(emailType.filename, function(err, template) { |
|
|
|
if (err) return cb(err); |
|
|
|
EmailService.prototype._send = function(email, cb) { |
|
|
|
var self = this; |
|
|
|
|
|
|
|
self._applyTemplate(template, notification.data, function(err, content) { |
|
|
|
if (err) return cb(err); |
|
|
|
var mailOptions = { |
|
|
|
from: email.from, |
|
|
|
to: email.to, |
|
|
|
subject: email.subject, |
|
|
|
text: email.body, |
|
|
|
}; |
|
|
|
self.mailer.sendMail(mailOptions, function(err, result) { |
|
|
|
if (err) { |
|
|
|
log.error('An error occurred when trying to send email to ' + email.to, err); |
|
|
|
return cb(err); |
|
|
|
} |
|
|
|
log.debug('Message sent: ', result || ''); |
|
|
|
return cb(err, result); |
|
|
|
}); |
|
|
|
}; |
|
|
|
|
|
|
|
_.each(addressesByCopayer, function(emailAddress, copayerId) { |
|
|
|
EmailService.prototype._generateFromNotification = function(notification, cb) { |
|
|
|
var self = this; |
|
|
|
|
|
|
|
var emailType = EMAIL_TYPES[notification.type]; |
|
|
|
if (!emailType) return cb(); |
|
|
|
|
|
|
|
var emailByCopayer; |
|
|
|
|
|
|
|
async.waterfall([ |
|
|
|
|
|
|
|
function(next) { |
|
|
|
self._getEmailAddresses(notification.walletId, next); |
|
|
|
}, |
|
|
|
function(emailAddresses, next) { |
|
|
|
if (_.isEmpty(emailAddresses)) return cb(); |
|
|
|
emailByCopayer = emailAddresses; |
|
|
|
self._readTemplate(emailType.filename, next); |
|
|
|
}, |
|
|
|
function(template, next) { |
|
|
|
self._applyTemplate(template, notification.data, next); |
|
|
|
}, |
|
|
|
function(content, next) { |
|
|
|
_.each(emailByCopayer, function(address, copayerId) { |
|
|
|
var email = Model.Email.create({ |
|
|
|
walletId: notification.walletId, |
|
|
|
copayerId: copayerId, |
|
|
|
to: emailAddress, |
|
|
|
to: address, |
|
|
|
subject: content.subject, |
|
|
|
body: content.body, |
|
|
|
}); |
|
|
|
self.storage.storeEmail(email, function(err) { |
|
|
|
return cb(err); |
|
|
|
}); |
|
|
|
}); |
|
|
|
}); |
|
|
|
return next(err, email); |
|
|
|
}); |
|
|
|
}); |
|
|
|
}, |
|
|
|
function(email, next) { |
|
|
|
self._send(email, next); |
|
|
|
}, |
|
|
|
], cb); |
|
|
|
|
|
|
|
return cb(); |
|
|
|
}; |
|
|
|
|
|
|
|
EmailService.prototype._send = function(cb) { |
|
|
|
var self = this; |
|
|
|
|
|
|
|
this.lock.runLocked('emails', cb, function() { |
|
|
|
//self._fetchUnsentEmails();
|
|
|
|
|
|
|
|
}); |
|
|
|
}; |
|
|
|
|
|
|
|
module.exports = EmailService; |
|
|
|