You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
nodebb/src/notifications.js

271 lines
7.2 KiB
JavaScript

11 years ago
'use strict';
11 years ago
var async = require('async'),
winston = require('winston'),
cron = require('cron').CronJob,
nconf = require('nconf'),
S = require('string'),
_ = require('underscore'),
11 years ago
db = require('./database'),
utils = require('../public/src/utils'),
events = require('./events'),
User = require('./user'),
groups = require('./groups'),
meta = require('./meta'),
plugins = require('./plugins');
11 years ago
(function(Notifications) {
11 years ago
Notifications.init = function() {
if (process.env.NODE_ENV === 'development') {
winston.info('[notifications.init] Registering jobs.');
}
11 years ago
new cron('*/30 * * * *', Notifications.prune, null, true);
11 years ago
};
11 years ago
11 years ago
Notifications.get = function(nid, callback) {
Notifications.getMultiple([nid], function(err, notifications) {
callback(err, Array.isArray(notifications) && notifications.length ? notifications[0] : null);
});
};
Notifications.getMultiple = function(nids, callback) {
var keys = nids.map(function(nid) {
return 'notifications:' + nid;
});
db.getObjects(keys, function(err, notifications) {
11 years ago
if (err) {
11 years ago
return callback(err);
}
async.map(notifications, function(notification, next) {
if (!notification) {
return next(null, null);
}
// Backwards compatibility for old notification schema
// Remove this block when NodeBB v0.6.0 is released.
if (notification.hasOwnProperty('text')) {
notification.bodyShort = notification.text;
notification.bodyLong = '';
notification.text = S(notification.text).escapeHTML().s;
}
11 years ago
if (notification.bodyShort) {
notification.bodyShort = S(notification.bodyShort).escapeHTML().s;
}
if (notification.bodyLong) {
notification.bodyLong = S(notification.bodyLong).escapeHTML().s;
}
if (notification.from && !notification.image) {
User.getUserField(notification.from, 'picture', function(err, picture) {
if (err) {
return next(err);
}
notification.image = picture;
next(null, notification);
});
return;
} else if (notification.image) {
switch(notification.image) {
case 'brand:logo':
notification.image = meta.config['brand:logo'] || nconf.get('relative_path') + '/logo.png';
break;
11 years ago
}
11 years ago
return next(null, notification);
}
}, callback);
});
11 years ago
};
11 years ago
Notifications.create = function(data, callback) {
if (!data.nid) {
return callback(new Error('no-notification-id'));
}
data.importance = data.importance || 5;
db.getObject('notifications:' + data.nid, function(err, oldNotification) {
11 years ago
if (err) {
return callback(err);
}
if (oldNotification) {
if (parseInt(oldNotification.pid, 10) === parseInt(data.pid, 10) && parseInt(oldNotification.importance, 10) > parseInt(data.importance, 10)) {
return callback(null, null);
}
}
var now = Date.now();
data.datetime = now;
async.parallel([
function(next) {
db.sortedSetAdd('notifications', now, data.nid, next);
},
function(next) {
db.setObject('notifications:' + data.nid, data, next);
}
], function(err) {
callback(err, data);
});
11 years ago
});
};
11 years ago
Notifications.push = function(notification, uids, callback) {
11 years ago
callback = callback || function() {};
var websockets = require('./socket.io');
11 years ago
if (!Array.isArray(uids)) {
uids = [uids];
}
var unreadKeys = [];
var readKeys = [];
uids.filter(Boolean).forEach(function(uid) {
unreadKeys.push('uid:' + uid + ':notifications:unread');
readKeys.push('uid:' + uid + ':notifications:read');
11 years ago
});
async.parallel([
function(next) {
db.sortedSetsAdd(unreadKeys, notification.datetime, notification.nid, next);
},
function(next) {
db.sortedSetsRemove(readKeys, notification.nid, next);
}
], function(err) {
11 years ago
if (err) {
return callback(err);
}
var oneWeekAgo = Date.now() - 604800000;
db.sortedSetsRemoveRangeByScore(unreadKeys, 0, oneWeekAgo);
db.sortedSetsRemoveRangeByScore(readKeys, 0, oneWeekAgo);
plugins.fireHook('action:notification.pushed', {notification: notification, uids: uids});
11 years ago
for(var i=0; i<uids.length; ++i) {
websockets.in('uid_' + uids[i]).emit('event:new_notification', notification);
11 years ago
}
});
};
11 years ago
Notifications.pushGroup = function(notification, groupName, callback) {
callback = callback || function() {};
groups.get(groupName, {}, function(err, groupObj) {
if (err || !groupObj || !Array.isArray(groupObj.members) || !groupObj.members.length) {
return callback(err);
}
Notifications.push(notification, groupObj.members, callback);
});
};
11 years ago
Notifications.markRead = function(nid, uid, callback) {
11 years ago
callback = callback || function() {};
if (!parseInt(uid, 10) || !parseInt(nid, 10)) {
11 years ago
return callback();
11 years ago
}
Notifications.markReadMultiple([nid], uid, callback);
};
Notifications.markReadMultiple = function(nids, uid, callback) {
callback = callback || function() {};
if (!Array.isArray(nids) || !nids.length) {
return callback();
}
var notificationKeys = nids.map(function(nid) {
return 'notifications:' + nid;
});
11 years ago
db.getObjectsFields(notificationKeys, ['datetime'], function(err, notificationData) {
if (err) {
11 years ago
return callback(err);
}
var datetimes = notificationData.map(function(notification) {
return notification && notification.datetime;
});
11 years ago
async.parallel([
function(next) {
db.sortedSetRemove('uid:' + uid + ':notifications:unread', nids, next);
},
function(next) {
db.sortedSetAdd('uid:' + uid + ':notifications:read', datetimes, nids, next);
}
11 years ago
], callback);
});
};
11 years ago
11 years ago
Notifications.markAllRead = function(uid, callback) {
db.getSortedSetRange('uid:' + uid + ':notifications:unread', 0, 99, function(err, nids) {
11 years ago
if (err) {
return callback(err);
}
11 years ago
if (!Array.isArray(nids) || !nids.length) {
return callback();
}
11 years ago
Notifications.markReadMultiple(nids, uid, callback);
11 years ago
});
};
11 years ago
Notifications.prune = function() {
var start = process.hrtime();
11 years ago
if (process.env.NODE_ENV === 'development') {
winston.info('[notifications.prune] Removing expired notifications from the database.');
}
var week = 604800000,
11 years ago
numPruned = 0;
var cutoffTime = Date.now() - week;
11 years ago
db.getSortedSetRangeByScore('notifications', 0, 500, 0, cutoffTime, function(err, nids) {
11 years ago
if (err) {
return winston.error(err.message);
}
if (!Array.isArray(nids) || !nids.length) {
return events.log('No notifications to prune');
}
11 years ago
var keys = nids.map(function(nid) {
return 'notifications:' + nid;
});
numPruned = nids.length;
events.log('Notification pruning. Expired Nids = ' + numPruned);
async.parallel([
function(next) {
db.sortedSetRemove('notifications', nids, next);
},
function(next) {
db.deleteAll(keys, next);
}
], function(err) {
if (err) {
return winston.error('Encountered error pruning notifications: ' + err.message);
}
if (process.env.NODE_ENV === 'development') {
winston.info('[notifications.prune] Notification pruning completed. ' + numPruned + ' expired notification' + (numPruned !== 1 ? 's' : '') + ' removed.');
}
var diff = process.hrtime(start);
events.log('Pruning '+ numPruned + ' notifications took : ' + (diff[0] * 1e3 + diff[1] / 1e6) + ' ms');
});
11 years ago
});
};
11 years ago
11 years ago
}(exports));