Files
NodeBB/src/user/digest.js

153 lines
4.2 KiB
JavaScript
Raw Normal View History

2017-02-18 01:56:23 -07:00
'use strict';
2014-11-17 13:37:07 -05:00
2016-11-23 17:15:31 +03:00
var async = require('async');
2016-04-01 15:12:01 +03:00
var winston = require('winston');
var nconf = require('nconf');
2017-06-23 12:41:40 -04:00
var batch = require('../batch');
2016-04-01 15:12:01 +03:00
var meta = require('../meta');
var user = require('../user');
var topics = require('../topics');
var plugins = require('../plugins');
var emailer = require('../emailer');
var utils = require('../utils');
2014-11-17 13:37:07 -05:00
2017-05-16 17:14:50 -04:00
var Digest = module.exports;
2016-11-23 15:52:35 +03:00
Digest.execute = function (payload, callback) {
2017-05-16 17:14:50 -04:00
callback = callback || function () {};
2014-11-17 13:37:07 -05:00
2017-05-16 17:14:50 -04:00
var digestsDisabled = parseInt(meta.config.disableEmailSubscriptions, 10) === 1;
if (digestsDisabled) {
winston.info('[user/jobs] Did not send digests (' + payload.interval + ') because subscription system is disabled.');
2017-05-16 17:14:50 -04:00
return callback();
}
var subscribers;
async.waterfall([
function (next) {
async.parallel({
topics: async.apply(topics.getLatestTopics, 0, 0, 9, payload.interval),
subscribers: function (next) {
if (payload.subscribers) {
setImmediate(next, undefined, payload.subscribers);
} else {
Digest.getSubscribers(payload.interval, next);
}
},
2017-05-16 17:14:50 -04:00
}, next);
},
function (data, next) {
subscribers = data.subscribers;
if (!data.subscribers.length) {
return callback();
}
// Fix relative paths in topic data
data.topics.topics = data.topics.topics.map(function (topicObj) {
var user = topicObj.hasOwnProperty('teaser') && topicObj.teaser !== undefined ? topicObj.teaser.user : topicObj.user;
if (user && user.picture && utils.isRelativeUrl(user.picture)) {
user.picture = nconf.get('base_url') + user.picture;
2015-02-01 20:21:18 -05:00
}
2017-05-16 17:14:50 -04:00
return topicObj;
});
2016-11-23 15:52:35 +03:00
data.interval = payload.interval;
2017-05-16 17:14:50 -04:00
Digest.send(data, next);
},
], function (err) {
if (err) {
winston.error('[user/jobs] Could not send digests (' + payload.interval + '): ' + err.message);
2017-05-16 17:14:50 -04:00
} else {
winston.info('[user/jobs] Digest (' + payload.interval + ') scheduling completed. ' + subscribers.length + ' email(s) sent.');
2017-05-16 17:14:50 -04:00
}
2016-11-23 15:52:35 +03:00
2017-05-16 17:14:50 -04:00
callback(err);
});
};
2016-11-23 15:52:35 +03:00
2017-05-16 17:14:50 -04:00
Digest.getSubscribers = function (interval, callback) {
async.waterfall([
function (next) {
2017-06-23 12:41:40 -04:00
var subs = [];
batch.processSortedSet('users:joindate', function (uids, next) {
user.getMultipleUserSettings(uids, function (err, settings) {
2017-06-23 13:08:15 -04:00
if (err) {
return next(err);
}
2017-06-23 12:41:40 -04:00
settings.forEach(function (hash) {
if (hash.dailyDigestFreq === interval) {
subs.push(hash.uid);
}
2017-06-23 11:03:54 -04:00
});
2017-06-23 12:41:40 -04:00
next();
2017-06-23 11:03:54 -04:00
});
2017-06-23 12:41:40 -04:00
}, { interval: 1000 }, function () {
next(null, subs);
2017-06-23 11:03:54 -04:00
});
},
2017-05-16 17:14:50 -04:00
function (subscribers, next) {
plugins.fireHook('filter:digest.subscribers', {
interval: interval,
subscribers: subscribers,
}, next);
},
function (results, next) {
next(null, results.subscribers);
},
], callback);
};
2014-11-17 13:37:07 -05:00
2017-05-16 17:14:50 -04:00
Digest.send = function (data, callback) {
if (!data || !data.subscribers || !data.subscribers.length) {
return callback();
}
var now = new Date();
2014-11-17 13:37:07 -05:00
2017-05-16 17:14:50 -04:00
async.waterfall([
function (next) {
user.getUsersFields(data.subscribers, ['uid', 'username', 'userslug', 'lastonline'], next);
},
function (users, next) {
async.eachLimit(users, 100, function (userObj, next) {
async.waterfall([
function (next) {
user.notifications.getDailyUnread(userObj.uid, next);
},
function (notifications, next) {
notifications = notifications.filter(Boolean);
// If there are no notifications and no new topics, don't bother sending a digest
if (!notifications.length && !data.topics.topics.length) {
return next();
}
2014-11-17 13:37:07 -05:00
2017-05-16 17:14:50 -04:00
notifications.forEach(function (notification) {
if (notification.image && !notification.image.startsWith('http')) {
notification.image = nconf.get('url') + notification.image;
2016-11-23 15:52:35 +03:00
}
2017-05-16 17:14:50 -04:00
});
2016-11-23 15:52:35 +03:00
2017-05-16 17:14:50 -04:00
emailer.send('digest', userObj.uid, {
subject: '[' + meta.config.title + '] [[email:digest.subject, ' + (now.getFullYear() + '/' + (now.getMonth() + 1) + '/' + now.getDate()) + ']]',
username: userObj.username,
userslug: userObj.userslug,
url: nconf.get('url'),
site_title: meta.config.title || meta.config.browserTitle || 'NodeBB',
notifications: notifications,
recent: data.topics.topics,
interval: data.interval,
});
next();
},
], next);
}, next);
},
], function (err) {
callback(err);
});
};