2014-08-31 10:36:32 -05:00
|
|
|
var _ = require('lodash'),
|
2014-08-17 01:17:23 -05:00
|
|
|
Promise = require('bluebird'),
|
2013-12-06 03:51:35 -05:00
|
|
|
nodemailer = require('nodemailer'),
|
2014-09-20 16:11:30 -05:00
|
|
|
validator = require('validator'),
|
2013-12-06 03:51:35 -05:00
|
|
|
config = require('./config');
|
2013-08-20 15:19:47 -05:00
|
|
|
|
|
|
|
function GhostMailer(opts) {
|
|
|
|
opts = opts || {};
|
|
|
|
this.transport = opts.transport || null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ## E-mail transport setup
|
|
|
|
// *This promise should always resolve to avoid halting Ghost::init*.
|
2013-12-06 03:51:35 -05:00
|
|
|
GhostMailer.prototype.init = function () {
|
|
|
|
var self = this;
|
2014-06-08 07:39:28 -05:00
|
|
|
self.state = {};
|
2014-07-17 09:33:21 -05:00
|
|
|
if (config.mail && config.mail.transport) {
|
2013-12-20 07:57:21 -05:00
|
|
|
this.createTransport();
|
2014-08-17 01:17:23 -05:00
|
|
|
return Promise.resolve();
|
2013-08-20 15:19:47 -05:00
|
|
|
}
|
|
|
|
|
2014-08-31 10:36:32 -05:00
|
|
|
self.transport = nodemailer.createTransport('direct');
|
|
|
|
self.state.usingDirect = true;
|
2014-08-17 01:17:23 -05:00
|
|
|
|
2014-08-31 10:36:32 -05:00
|
|
|
return Promise.resolve();
|
2013-08-20 15:19:47 -05:00
|
|
|
};
|
|
|
|
|
2013-12-20 07:57:21 -05:00
|
|
|
GhostMailer.prototype.createTransport = function () {
|
2014-07-17 09:33:21 -05:00
|
|
|
this.transport = nodemailer.createTransport(config.mail.transport, _.clone(config.mail.options) || {});
|
2013-08-20 15:19:47 -05:00
|
|
|
};
|
|
|
|
|
2014-09-20 16:11:30 -05:00
|
|
|
GhostMailer.prototype.from = function () {
|
|
|
|
var from = config.mail && (config.mail.from || config.mail.fromaddress);
|
2014-02-24 11:28:07 -05:00
|
|
|
|
2014-09-20 16:11:30 -05:00
|
|
|
// If we don't have a from address at all
|
2014-02-24 11:28:07 -05:00
|
|
|
if (!from) {
|
2014-03-06 06:03:00 -05:00
|
|
|
// Default to ghost@[blog.url]
|
2014-09-20 16:11:30 -05:00
|
|
|
from = 'ghost@' + this.getDomain();
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we do have a from address, and it's just an email
|
|
|
|
if (validator.isEmail(from)) {
|
|
|
|
if (!config.theme.title) {
|
|
|
|
config.theme.title = 'Ghost at ' + this.getDomain();
|
|
|
|
}
|
2015-01-09 16:03:03 -05:00
|
|
|
from = '"' + config.theme.title + '" <' + from + '>';
|
2014-02-24 11:28:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return from;
|
|
|
|
};
|
|
|
|
|
2014-09-20 16:11:30 -05:00
|
|
|
// Moved it to its own module
|
|
|
|
GhostMailer.prototype.getDomain = function () {
|
|
|
|
var domain = config.url.match(new RegExp('^https?://([^/:?#]+)(?:[/:?#]|$)', 'i'));
|
|
|
|
return domain && domain[1];
|
|
|
|
};
|
|
|
|
|
2013-08-20 15:19:47 -05:00
|
|
|
// Sends an e-mail message enforcing `to` (blog owner) and `from` fields
|
2015-03-14 16:50:00 -05:00
|
|
|
// This assumes that api.settings.read('email') was already done on the API level
|
2014-06-08 07:39:28 -05:00
|
|
|
GhostMailer.prototype.send = function (message) {
|
2014-05-08 18:12:18 -05:00
|
|
|
var self = this,
|
2014-06-08 07:39:28 -05:00
|
|
|
to,
|
|
|
|
sendMail;
|
|
|
|
|
|
|
|
message = message || {};
|
|
|
|
to = message.to || false;
|
2013-12-20 07:57:21 -05:00
|
|
|
|
2013-08-20 15:19:47 -05:00
|
|
|
if (!this.transport) {
|
2014-08-17 01:17:23 -05:00
|
|
|
return Promise.reject(new Error('Email Error: No e-mail transport configured.'));
|
2013-08-20 15:19:47 -05:00
|
|
|
}
|
2014-06-08 07:39:28 -05:00
|
|
|
if (!(message && message.subject && message.html && message.to)) {
|
2014-08-17 01:17:23 -05:00
|
|
|
return Promise.reject(new Error('Email Error: Incomplete message data.'));
|
2013-08-20 15:19:47 -05:00
|
|
|
}
|
2014-08-17 01:17:23 -05:00
|
|
|
sendMail = Promise.promisify(self.transport.sendMail.bind(self.transport));
|
2013-12-20 07:57:21 -05:00
|
|
|
|
2014-06-08 07:39:28 -05:00
|
|
|
message = _.extend(message, {
|
2014-09-20 16:11:30 -05:00
|
|
|
from: self.from(),
|
2014-06-08 07:39:28 -05:00
|
|
|
to: to,
|
2014-10-28 08:03:32 -05:00
|
|
|
generateTextFromHTML: true,
|
2014-11-16 13:14:18 -05:00
|
|
|
encoding: 'base64'
|
2013-08-20 15:19:47 -05:00
|
|
|
});
|
2014-08-31 10:36:32 -05:00
|
|
|
|
|
|
|
return new Promise(function (resolve, reject) {
|
|
|
|
sendMail(message, function (error, response) {
|
|
|
|
if (error) {
|
|
|
|
return reject(new Error(error));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (self.transport.transportType !== 'DIRECT') {
|
|
|
|
return resolve(response);
|
|
|
|
}
|
|
|
|
|
2014-09-09 23:06:24 -05:00
|
|
|
response.statusHandler.once('failed', function (data) {
|
2014-08-31 10:36:32 -05:00
|
|
|
var reason = 'Email Error: Failed sending email';
|
2015-05-10 09:56:15 -05:00
|
|
|
|
|
|
|
if (data.error && data.error.errno === 'ENOTFOUND') {
|
2014-08-31 10:36:32 -05:00
|
|
|
reason += ': there is no mail server at this address: ' + data.domain;
|
|
|
|
}
|
|
|
|
reason += '.';
|
|
|
|
return reject(new Error(reason));
|
|
|
|
});
|
|
|
|
|
2014-09-09 23:06:24 -05:00
|
|
|
response.statusHandler.once('requeue', function (data) {
|
2015-05-10 09:56:15 -05:00
|
|
|
var errorMessage = 'Email Error: message was not sent, requeued. Probably will not be sent. :(';
|
|
|
|
|
|
|
|
if (data.error && data.error.message) {
|
|
|
|
errorMessage += '\nMore info: ' + data.error.message;
|
|
|
|
}
|
|
|
|
|
|
|
|
return reject(new Error(errorMessage));
|
2014-08-31 10:36:32 -05:00
|
|
|
});
|
|
|
|
|
2014-09-09 23:06:24 -05:00
|
|
|
response.statusHandler.once('sent', function () {
|
|
|
|
return resolve('Message was accepted by the mail server. Make sure to check inbox and spam folders. :)');
|
2014-08-31 10:36:32 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2013-08-20 15:19:47 -05:00
|
|
|
};
|
|
|
|
|
2013-11-27 21:45:01 -05:00
|
|
|
module.exports = new GhostMailer();
|