mirror of
https://github.com/NodeBB/NodeBB.git
synced 2025-10-27 09:06:15 +01:00
216 lines
6.7 KiB
JavaScript
216 lines
6.7 KiB
JavaScript
'use strict';
|
|
|
|
const _ = require('lodash');
|
|
const validator = require('validator');
|
|
|
|
const db = require('../database');
|
|
const user = require('../user');
|
|
const posts = require('../posts');
|
|
const utils = require('../utils');
|
|
const plugins = require('../plugins');
|
|
|
|
const intFields = ['mid', 'timestamp', 'edited', 'fromuid', 'roomId', 'deleted', 'system'];
|
|
|
|
module.exports = function (Messaging) {
|
|
Messaging.newMessageCutoff = 1000 * 60 * 3;
|
|
|
|
Messaging.getMessagesFields = async (mids, fields) => {
|
|
if (!Array.isArray(mids) || !mids.length) {
|
|
return [];
|
|
}
|
|
|
|
const keys = mids.map(mid => `message:${mid}`);
|
|
const messages = await db.getObjects(keys, fields);
|
|
|
|
return await Promise.all(messages.map(
|
|
async (message, idx) => modifyMessage(message, fields, parseInt(mids[idx], 10))
|
|
));
|
|
};
|
|
|
|
Messaging.getMessageField = async (mid, field) => {
|
|
const fields = await Messaging.getMessageFields(mid, [field]);
|
|
return fields ? fields[field] : null;
|
|
};
|
|
|
|
Messaging.getMessageFields = async (mid, fields) => {
|
|
const messages = await Messaging.getMessagesFields([mid], fields);
|
|
return messages ? messages[0] : null;
|
|
};
|
|
|
|
Messaging.setMessageField = async (mid, field, content) => {
|
|
await db.setObjectField(`message:${mid}`, field, content);
|
|
};
|
|
|
|
Messaging.setMessageFields = async (mid, data) => {
|
|
await db.setObject(`message:${mid}`, data);
|
|
};
|
|
|
|
Messaging.getMessagesData = async (mids, uid, roomId, isNew) => {
|
|
let messages = await Messaging.getMessagesFields(mids, []);
|
|
messages = messages
|
|
.map((msg, idx) => {
|
|
if (msg) {
|
|
msg.messageId = parseInt(mids[idx], 10);
|
|
msg.ip = undefined;
|
|
msg.isOwner = msg.fromuid === parseInt(uid, 10);
|
|
}
|
|
return msg;
|
|
})
|
|
.filter(Boolean);
|
|
messages = await user.blocks.filter(uid, 'fromuid', messages);
|
|
const users = await user.getUsersFields(
|
|
messages.map(msg => msg && msg.fromuid),
|
|
['uid', 'username', 'userslug', 'picture', 'status', 'banned']
|
|
);
|
|
|
|
messages.forEach((message, index) => {
|
|
message.fromUser = users[index];
|
|
message.fromUser.banned = !!message.fromUser.banned;
|
|
message.fromUser.deleted = message.fromuid !== message.fromUser.uid && message.fromUser.uid === 0;
|
|
|
|
const self = message.fromuid === parseInt(uid, 10);
|
|
message.self = self ? 1 : 0;
|
|
|
|
message.newSet = false;
|
|
message.roomId = String(message.roomId || roomId);
|
|
});
|
|
|
|
await parseMessages(messages, uid, roomId, isNew);
|
|
|
|
if (messages.length > 1) {
|
|
// Add a spacer in between messages with time gaps between them
|
|
messages = messages.map((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].timestamp + 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;
|
|
} else if (index > 0 && messages[index - 1].system) {
|
|
message.newSet = true;
|
|
} else if (index === 0 || message.toMid) {
|
|
message.newSet = true;
|
|
}
|
|
|
|
return message;
|
|
});
|
|
} else if (messages.length === 1) {
|
|
// For single messages, we don't know the context, so look up the previous message and compare
|
|
const key = `chat:room:${roomId}:mids`;
|
|
const index = await db.sortedSetRank(key, messages[0].messageId);
|
|
if (index > 0) {
|
|
const mid = await db.getSortedSetRange(key, index - 1, index - 1);
|
|
const fields = await Messaging.getMessageFields(mid, ['fromuid', 'timestamp']);
|
|
if ((messages[0].timestamp > fields.timestamp + Messaging.newMessageCutoff) ||
|
|
(messages[0].fromuid !== fields.fromuid) ||
|
|
messages[0].system || messages[0].toMid) {
|
|
// If it's been 5 minutes, this is a new set of messages
|
|
messages[0].newSet = true;
|
|
}
|
|
} else {
|
|
messages[0].newSet = true;
|
|
}
|
|
}
|
|
|
|
await addParentMessages(messages, uid, roomId);
|
|
|
|
const data = await plugins.hooks.fire('filter:messaging.getMessages', {
|
|
messages: messages,
|
|
uid: uid,
|
|
roomId: roomId,
|
|
isNew: isNew,
|
|
mids: mids,
|
|
});
|
|
|
|
return data && data.messages;
|
|
};
|
|
|
|
async function addParentMessages(messages, uid, roomId) {
|
|
let parentMids = messages.map(msg => (msg && msg.hasOwnProperty('toMid') ? parseInt(msg.toMid, 10) : null)).filter(Boolean);
|
|
|
|
if (!parentMids.length) {
|
|
return;
|
|
}
|
|
parentMids = _.uniq(parentMids);
|
|
const canView = await Messaging.canViewMessage(parentMids, roomId, uid);
|
|
parentMids = parentMids.filter((mid, idx) => canView[idx]);
|
|
|
|
const parentMessages = await Messaging.getMessagesFields(parentMids, [
|
|
'fromuid', 'content', 'timestamp', 'deleted',
|
|
]);
|
|
const parentUids = _.uniq(parentMessages.map(msg => msg && msg.fromuid));
|
|
const usersMap = _.zipObject(
|
|
parentUids,
|
|
await user.getUsersFields(parentUids, ['uid', 'username', 'userslug', 'picture'])
|
|
);
|
|
|
|
await Promise.all(parentMessages.map(async (parentMsg) => {
|
|
if (parentMsg.deleted && parentMsg.fromuid !== parseInt(uid, 10)) {
|
|
parentMsg.content = `<p>[[modules:chat.message-deleted]]</p>`;
|
|
return;
|
|
}
|
|
const foundMsg = messages.find(msg => parseInt(msg.mid, 10) === parseInt(parentMsg.mid, 10));
|
|
if (foundMsg) {
|
|
parentMsg.content = foundMsg.content;
|
|
return;
|
|
}
|
|
parentMsg.content = await parseMessage(parentMsg, uid, roomId, false);
|
|
}));
|
|
|
|
const parents = {};
|
|
parentMessages.forEach((msg, i) => {
|
|
if (usersMap[msg.fromuid]) {
|
|
msg.user = usersMap[msg.fromuid];
|
|
parents[parentMids[i]] = msg;
|
|
}
|
|
});
|
|
|
|
messages.forEach((msg) => {
|
|
if (parents[msg.toMid]) {
|
|
msg.parent = parents[msg.toMid];
|
|
msg.parent.mid = msg.toMid;
|
|
}
|
|
});
|
|
}
|
|
|
|
async function parseMessages(messages, uid, roomId, isNew) {
|
|
await Promise.all(messages.map(async (msg) => {
|
|
if (msg.deleted && !msg.isOwner) {
|
|
msg.content = `<p>[[modules:chat.message-deleted]]</p>`;
|
|
return;
|
|
}
|
|
msg.content = await parseMessage(msg, uid, roomId, isNew);
|
|
}));
|
|
}
|
|
async function parseMessage(message, uid, roomId, isNew) {
|
|
if (message.system) {
|
|
return validator.escape(String(message.content));
|
|
} else if (!utils.isNumber(message.mid)) {
|
|
return posts.sanitize(message.content);
|
|
}
|
|
|
|
return await Messaging.parse(message.content, message.fromuid, uid, roomId, isNew);
|
|
}
|
|
};
|
|
|
|
async function modifyMessage(message, fields, mid) {
|
|
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);
|
|
}
|
|
}
|
|
|
|
const payload = await plugins.hooks.fire('filter:messaging.getFields', {
|
|
mid: mid,
|
|
message: message,
|
|
fields: fields,
|
|
});
|
|
|
|
return payload.message;
|
|
}
|