|
|
|
'use strict';
|
|
|
|
|
|
|
|
var async = require('async');
|
|
|
|
|
|
|
|
var db = require('../database');
|
|
|
|
var user = require('../user');
|
|
|
|
var utils = require('../utils');
|
|
|
|
var plugins = require('../plugins');
|
|
|
|
|
|
|
|
const intFields = ['timestamp', 'edited', 'fromuid', 'roomId', 'deleted'];
|
|
|
|
|
|
|
|
module.exports = function (Messaging) {
|
|
|
|
Messaging.newMessageCutoff = 1000 * 60 * 3;
|
|
|
|
|
|
|
|
Messaging.getMessagesFields = function (mids, fields, callback) {
|
|
|
|
if (!Array.isArray(mids) || !mids.length) {
|
|
|
|
return callback(null, []);
|
|
|
|
}
|
|
|
|
|
|
|
|
async.waterfall([
|
|
|
|
function (next) {
|
|
|
|
const keys = mids.map(mid => 'message:' + mid);
|
|
|
|
if (fields.length) {
|
|
|
|
db.getObjectsFields(keys, fields, next);
|
|
|
|
} else {
|
|
|
|
db.getObjects(keys, next);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
function (messages, next) {
|
|
|
|
messages.forEach(message => modifyMessage(message, fields));
|
|
|
|
next(null, messages);
|
|
|
|
},
|
|
|
|
], callback);
|
|
|
|
};
|
|
|
|
|
|
|
|
Messaging.getMessageField = function (mid, field, callback) {
|
|
|
|
Messaging.getMessageFields(mid, [field], function (err, fields) {
|
|
|
|
callback(err, fields ? fields[field] : null);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
Messaging.getMessageFields = function (mid, fields, callback) {
|
|
|
|
Messaging.getMessagesFields([mid], fields, function (err, messages) {
|
|
|
|
callback(err, messages ? messages[0] : null);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
Messaging.setMessageField = function (mid, field, content, callback) {
|
|
|
|
db.setObjectField('message:' + mid, field, content, callback);
|
|
|
|
};
|
|
|
|
|
|
|
|
Messaging.setMessageFields = function (mid, data, callback) {
|
|
|
|
db.setObject('message:' + mid, data, callback);
|
|
|
|
};
|
|
|
|
|
|
|
|
Messaging.getMessagesData = function (mids, uid, roomId, isNew, callback) {
|
|
|
|
var messages;
|
|
|
|
|
|
|
|
async.waterfall([
|
|
|
|
function (next) {
|
|
|
|
Messaging.getMessagesFields(mids, [], next);
|
|
|
|
},
|
|
|
|
async.apply(user.blocks.filter, uid, 'fromuid'),
|
|
|
|
function (_messages, next) {
|
|
|
|
messages = _messages.map(function (msg, idx) {
|
|
|
|
if (msg) {
|
|
|
|
msg.messageId = parseInt(mids[idx], 10);
|
|
|
|
msg.ip = undefined;
|
|
|
|
}
|
|
|
|
return msg;
|
|
|
|
}).filter(Boolean);
|
|
|
|
|
|
|
|
const uids = messages.map(msg => msg && msg.fromuid);
|
|
|
|
|
|
|
|
user.getUsersFields(uids, ['uid', 'username', 'userslug', 'picture', 'status', 'banned'], next);
|
|
|
|
},
|
|
|
|
function (users, next) {
|
|
|
|
messages.forEach(function (message, index) {
|
|
|
|
message.fromUser = users[index];
|
|
|
|
message.fromUser.banned = !!message.fromUser.banned;
|
|
|
|
message.fromUser.deleted = message.fromuid !== message.fromUser.uid && message.fromUser.uid === 0;
|
|
|
|
|
|
|
|
var self = message.fromuid === parseInt(uid, 10);
|
|
|
|
message.self = self ? 1 : 0;
|
|
|
|
|
|
|
|
message.newSet = false;
|
|
|
|
message.roomId = String(message.roomId || roomId);
|
|
|
|
message.deleted = !!message.deleted;
|
|
|
|
});
|
|
|
|
|
|
|
|
async.map(messages, function (message, next) {
|
|
|
|
Messaging.parse(message.content, message.fromuid, uid, roomId, isNew, function (err, result) {
|
|
|
|
if (err) {
|
|
|
|
return next(err);
|
|
|
|
}
|
|
|
|
message.content = result;
|
|
|
|
message.cleanedContent = utils.stripHTMLTags(utils.decodeHTMLEntities(result));
|
|
|
|
next(null, message);
|
|
|
|
});
|
|
|
|
}, next);
|
|
|
|
},
|
|
|
|
function (messages, next) {
|
|
|
|
if (messages.length > 1) {
|
|
|
|
// Add a spacer in between messages with time gaps between them
|
|
|
|
messages = messages.map(function (message, index) {
|
|
|
|
// Compare timestamps with the previous message, and check if a spacer needs to be added
|
|
|
|
if (index > 0 && message.timestamp > messages[index - 1] + Messaging.newMessageCutoff) {
|
|
|
|
// If it's been 5 minutes, this is a new set of messages
|
|
|
|
message.newSet = true;
|
|
|
|
} else if (index > 0 && message.fromuid !== messages[index - 1].fromuid) {
|
|
|
|
// If the previous message was from the other person, this is also a new set
|
|
|
|
message.newSet = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return message;
|
|
|
|
});
|
|
|
|
|
|
|
|
next(undefined, messages);
|
|
|
|
} else if (messages.length === 1) {
|
|
|
|
// For single messages, we don't know the context, so look up the previous message and compare
|
|
|
|
var key = 'uid:' + uid + ':chat:room:' + roomId + ':mids';
|
|
|
|
async.waterfall([
|
|
|
|
async.apply(db.sortedSetRank, key, messages[0].messageId),
|
|
|
|
function (index, next) {
|
|
|
|
// Continue only if this isn't the first message in sorted set
|
|
|
|
if (index > 0) {
|
|
|
|
db.getSortedSetRange(key, index - 1, index - 1, next);
|
|
|
|
} else {
|
|
|
|
messages[0].newSet = true;
|
|
|
|
return next(undefined, messages);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
function (mid, next) {
|
|
|
|
Messaging.getMessageFields(mid, ['fromuid', 'timestamp'], next);
|
|
|
|
},
|
|
|
|
function (fields, next) {
|
|
|
|
if ((messages[0].timestamp > fields.timestamp + Messaging.newMessageCutoff) ||
|
|
|
|
(messages[0].fromuid !== fields.fromuid)) {
|
|
|
|
// If it's been 5 minutes, this is a new set of messages
|
|
|
|
messages[0].newSet = true;
|
|
|
|
}
|
|
|
|
next(null, messages);
|
|
|
|
},
|
|
|
|
], next);
|
|
|
|
} else {
|
|
|
|
next(null, []);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
function (messages, next) {
|
|
|
|
plugins.fireHook('filter:messaging.getMessages', {
|
|
|
|
messages: messages,
|
|
|
|
uid: uid,
|
|
|
|
roomId: roomId,
|
|
|
|
isNew: isNew,
|
|
|
|
mids: mids,
|
|
|
|
}, function (err, data) {
|
|
|
|
next(err, data && data.messages);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
], callback);
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
function modifyMessage(message, fields) {
|
|
|
|
if (message) {
|
|
|
|
db.parseIntFields(message, intFields, fields);
|
|
|
|
if (message.hasOwnProperty('timestamp')) {
|
|
|
|
message.timestampISO = utils.toISOString(message.timestamp);
|
|
|
|
}
|
|
|
|
if (message.hasOwnProperty('edited')) {
|
|
|
|
message.editedISO = utils.toISOString(message.edited);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|