2013-09-24 14:46:30 +04:00
|
|
|
var cp = require('child_process'),
|
2014-02-05 12:40:30 +04:00
|
|
|
_ = require('lodash'),
|
2013-09-24 14:46:30 +04:00
|
|
|
when = require('when'),
|
|
|
|
nodefn = require('when/node/function'),
|
2013-12-06 12:51:35 +04:00
|
|
|
nodemailer = require('nodemailer'),
|
|
|
|
api = require('./api'),
|
|
|
|
config = require('./config');
|
2013-08-21 00:19:47 +04: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 12:51:35 +04:00
|
|
|
GhostMailer.prototype.init = function () {
|
|
|
|
var self = this;
|
2014-02-07 14:17:22 +04:00
|
|
|
if (config().mail && config().mail.transport) {
|
2013-12-20 16:57:21 +04:00
|
|
|
this.createTransport();
|
2013-08-21 00:19:47 +04:00
|
|
|
return when.resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Attempt to detect and fallback to `sendmail`
|
|
|
|
return this.detectSendmail().then(function (binpath) {
|
|
|
|
self.transport = nodemailer.createTransport('sendmail', {
|
|
|
|
path: binpath
|
|
|
|
});
|
|
|
|
self.usingSendmail();
|
|
|
|
}, function () {
|
|
|
|
self.emailDisabled();
|
|
|
|
}).ensure(function () {
|
|
|
|
return when.resolve();
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
GhostMailer.prototype.isWindows = function () {
|
|
|
|
return process.platform === 'win32';
|
|
|
|
};
|
|
|
|
|
|
|
|
GhostMailer.prototype.detectSendmail = function () {
|
|
|
|
if (this.isWindows()) {
|
|
|
|
return when.reject();
|
|
|
|
}
|
|
|
|
return when.promise(function (resolve, reject) {
|
|
|
|
cp.exec('which sendmail', function (err, stdout) {
|
|
|
|
if (err && !/bin\/sendmail/.test(stdout)) {
|
|
|
|
return reject();
|
|
|
|
}
|
2013-09-26 17:07:52 +04:00
|
|
|
resolve(stdout.toString().replace(/(\n|\r|\r\n)$/, ''));
|
2013-08-21 00:19:47 +04:00
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2013-12-20 16:57:21 +04:00
|
|
|
GhostMailer.prototype.createTransport = function () {
|
2014-02-07 14:17:22 +04:00
|
|
|
this.transport = nodemailer.createTransport(config().mail.transport, _.clone(config().mail.options) || {});
|
2013-08-21 00:19:47 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
GhostMailer.prototype.usingSendmail = function () {
|
2013-12-06 12:51:35 +04:00
|
|
|
api.notifications.add({
|
2013-08-21 00:19:47 +04:00
|
|
|
type: 'info',
|
|
|
|
message: [
|
|
|
|
"Ghost is attempting to use your server's <b>sendmail</b> to send e-mail.",
|
|
|
|
"It is recommended that you explicitly configure an e-mail service,",
|
2013-09-16 21:34:20 +04:00
|
|
|
"See <a href=\"http://docs.ghost.org/mail\">http://docs.ghost.org/mail</a> for instructions"
|
2014-04-29 00:58:18 +04:00
|
|
|
].join(' ')
|
2013-08-21 00:19:47 +04:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
GhostMailer.prototype.emailDisabled = function () {
|
2013-12-06 12:51:35 +04:00
|
|
|
api.notifications.add({
|
2013-08-21 00:19:47 +04:00
|
|
|
type: 'warn',
|
|
|
|
message: [
|
|
|
|
"Ghost is currently unable to send e-mail.",
|
2013-09-16 21:34:20 +04:00
|
|
|
"See <a href=\"http://docs.ghost.org/mail\">http://docs.ghost.org/mail</a> for instructions"
|
2014-04-29 00:58:18 +04:00
|
|
|
].join(' ')
|
2013-08-21 00:19:47 +04:00
|
|
|
});
|
|
|
|
this.transport = null;
|
|
|
|
};
|
|
|
|
|
2014-02-24 20:28:07 +04:00
|
|
|
GhostMailer.prototype.fromAddress = function () {
|
|
|
|
var from = config().mail && config().mail.fromaddress,
|
|
|
|
domain;
|
|
|
|
|
|
|
|
if (!from) {
|
|
|
|
// Extract the domain name from url set in config.js
|
|
|
|
domain = config().url.match(new RegExp("^https?://([^/:?#]+)(?:[/:?#]|$)", "i"));
|
|
|
|
domain = domain && domain[1];
|
|
|
|
|
2014-03-06 15:03:00 +04:00
|
|
|
// Default to ghost@[blog.url]
|
|
|
|
from = 'ghost@' + domain;
|
2014-02-24 20:28:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return from;
|
|
|
|
};
|
|
|
|
|
2013-08-21 00:19:47 +04:00
|
|
|
// Sends an e-mail message enforcing `to` (blog owner) and `from` fields
|
|
|
|
GhostMailer.prototype.send = function (message) {
|
2013-12-20 16:57:21 +04:00
|
|
|
var self = this;
|
|
|
|
|
2013-08-21 00:19:47 +04:00
|
|
|
if (!this.transport) {
|
2013-09-04 17:57:41 +04:00
|
|
|
return when.reject(new Error('Email Error: No e-mail transport configured.'));
|
2013-08-21 00:19:47 +04:00
|
|
|
}
|
|
|
|
if (!(message && message.subject && message.html)) {
|
2013-09-04 17:57:41 +04:00
|
|
|
return when.reject(new Error('Email Error: Incomplete message data.'));
|
2013-08-21 00:19:47 +04:00
|
|
|
}
|
2013-12-20 16:57:21 +04:00
|
|
|
|
2014-05-07 04:49:25 +04:00
|
|
|
return api.settings.read.call({ internal: true }, 'email').then(function (response) {
|
2014-04-28 03:28:50 +04:00
|
|
|
|
|
|
|
var email = response.settings[0],
|
|
|
|
to = message.to || email.value;
|
2013-08-21 00:19:47 +04:00
|
|
|
|
2013-12-06 12:51:35 +04:00
|
|
|
message = _.extend(message, {
|
2014-02-24 20:28:07 +04:00
|
|
|
from: self.fromAddress(),
|
2013-12-06 12:51:35 +04:00
|
|
|
to: to,
|
|
|
|
generateTextFromHTML: true
|
|
|
|
});
|
|
|
|
}).then(function () {
|
2013-12-20 16:57:21 +04:00
|
|
|
var sendMail = nodefn.lift(self.transport.sendMail.bind(self.transport));
|
2013-12-06 12:51:35 +04:00
|
|
|
return sendMail(message);
|
|
|
|
}).otherwise(function (error) {
|
2013-09-04 17:57:41 +04:00
|
|
|
// Proxy the error message so we can add 'Email Error:' to the beginning to make it clearer.
|
|
|
|
error = _.isString(error) ? 'Email Error:' + error : (_.isObject(error) ? 'Email Error: ' + error.message : 'Email Error: Unknown Email Error');
|
|
|
|
return when.reject(new Error(error));
|
2013-08-21 00:19:47 +04:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2013-11-28 06:45:01 +04:00
|
|
|
module.exports = new GhostMailer();
|