2017-04-01 17:22:51 +08:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
let crypto = require('crypto');
|
|
|
|
let EventEmitter = require('events').EventEmitter;
|
|
|
|
|
|
|
|
// Expects that the folder listing is a Map
|
|
|
|
|
|
|
|
class MemoryNotifier extends EventEmitter {
|
|
|
|
constructor(options) {
|
|
|
|
super();
|
|
|
|
this.folders = options.folders || new Map();
|
|
|
|
|
|
|
|
let logfunc = (...args) => {
|
|
|
|
let level = args.shift() || 'DEBUG';
|
|
|
|
let message = args.shift() || '';
|
|
|
|
|
|
|
|
console.log([level].concat(message || '').join(' '), ...args); // eslint-disable-line no-console
|
|
|
|
};
|
|
|
|
|
|
|
|
this.logger = options.logger || {
|
|
|
|
info: logfunc.bind(null, 'INFO'),
|
|
|
|
debug: logfunc.bind(null, 'DEBUG'),
|
|
|
|
error: logfunc.bind(null, 'ERROR')
|
|
|
|
};
|
|
|
|
|
|
|
|
this._listeners = new EventEmitter();
|
|
|
|
this._listeners.setMaxListeners(0);
|
|
|
|
|
|
|
|
EventEmitter.call(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generates hashed event names for mailbox:username pairs
|
|
|
|
*
|
|
|
|
* @param {String} mailbox
|
|
|
|
* @param {String} username
|
|
|
|
* @returns {String} md5 hex
|
|
|
|
*/
|
|
|
|
_eventName(mailbox, username) {
|
2017-10-03 16:33:11 +08:00
|
|
|
return crypto
|
|
|
|
.createHash('md5')
|
|
|
|
.update(username + ':' + mailbox)
|
|
|
|
.digest('hex');
|
2017-04-01 17:22:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Registers an event handler for mailbox:username events
|
|
|
|
*
|
|
|
|
* @param {String} username
|
|
|
|
* @param {String} mailbox
|
|
|
|
* @param {Function} handler Function to run once there are new entries in the journal
|
|
|
|
*/
|
|
|
|
addListener(session, mailbox, handler) {
|
2017-07-20 18:10:43 +08:00
|
|
|
let eventName = this._eventName(session.user.id, mailbox);
|
2017-04-01 17:22:51 +08:00
|
|
|
this._listeners.addListener(eventName, handler);
|
|
|
|
|
2017-07-20 18:10:43 +08:00
|
|
|
this.logger.debug('[%s] New journal listener for %s ("%s:%s")', session.id, eventName, session.user.id, mailbox);
|
2017-04-01 17:22:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Unregisters an event handler for mailbox:username events
|
|
|
|
*
|
|
|
|
* @param {String} username
|
|
|
|
* @param {String} mailbox
|
|
|
|
* @param {Function} handler Function to run once there are new entries in the journal
|
|
|
|
*/
|
|
|
|
removeListener(session, mailbox, handler) {
|
2017-07-20 18:10:43 +08:00
|
|
|
let eventName = this._eventName(session.user.id, mailbox);
|
2017-04-01 17:22:51 +08:00
|
|
|
this._listeners.removeListener(eventName, handler);
|
|
|
|
|
2017-07-20 18:10:43 +08:00
|
|
|
this.logger.debug('[%s] Removed journal listener from %s ("%s:%s")', session.id, eventName, session.user.id, mailbox);
|
2017-04-01 17:22:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Stores multiple journal entries to db
|
|
|
|
*
|
|
|
|
* @param {String} username
|
|
|
|
* @param {String} mailbox
|
|
|
|
* @param {Array|Object} entries An array of entries to be journaled
|
|
|
|
* @param {Function} callback Runs once the entry is either stored or an error occurred
|
|
|
|
*/
|
|
|
|
addEntries(username, mailbox, entries, callback) {
|
|
|
|
let folder = this.folders.get(mailbox);
|
|
|
|
|
|
|
|
if (!folder) {
|
|
|
|
return callback(null, new Error('Selected mailbox does not exist'));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (entries && !Array.isArray(entries)) {
|
|
|
|
entries = [entries];
|
|
|
|
} else if (!entries || !entries.length) {
|
|
|
|
return callback(null, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// store entires in the folder object
|
|
|
|
if (!folder.journal) {
|
|
|
|
folder.journal = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
entries.forEach(entry => {
|
|
|
|
entry.modseq = ++folder.modifyIndex;
|
|
|
|
folder.journal.push(entry);
|
|
|
|
});
|
|
|
|
|
|
|
|
setImmediate(callback);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sends a notification that there are new updates in the selected mailbox
|
|
|
|
*
|
|
|
|
* @param {String} username
|
|
|
|
* @param {String} mailbox
|
|
|
|
*/
|
|
|
|
fire(username, mailbox, payload) {
|
|
|
|
let eventName = this._eventName(username, mailbox);
|
|
|
|
setImmediate(() => {
|
|
|
|
this._listeners.emit(eventName, payload);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns all entries from the journal that have higher than provided modification index
|
|
|
|
*
|
|
|
|
* @param {String} username
|
|
|
|
* @param {String} mailbox
|
|
|
|
* @param {Number} modifyIndex Last known modification id
|
|
|
|
* @param {Function} callback Returns update entries as an array
|
|
|
|
*/
|
|
|
|
getUpdates(session, mailbox, modifyIndex, callback) {
|
|
|
|
modifyIndex = Number(modifyIndex) || 0;
|
|
|
|
|
|
|
|
if (!this.folders.has(mailbox)) {
|
|
|
|
return callback(null, 'NONEXISTENT');
|
|
|
|
}
|
|
|
|
|
|
|
|
let folder = this.folders.get(mailbox);
|
|
|
|
let minIndex = folder.journal.length;
|
|
|
|
|
|
|
|
for (let i = folder.journal.length - 1; i >= 0; i--) {
|
|
|
|
if (folder.journal[i].modseq > modifyIndex) {
|
|
|
|
minIndex = i;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return callback(null, folder.journal.slice(minIndex));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = MemoryNotifier;
|