2021-03-31 19:48:10 +00:00
|
|
|
// Copyright 2020-2021 Signal Messenger, LLC
|
2020-10-30 20:34:04 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2020-01-17 22:23:19 +00:00
|
|
|
/* global
|
|
|
|
Backbone,
|
|
|
|
Whisper,
|
2020-03-20 19:07:44 +00:00
|
|
|
MessageController,
|
|
|
|
ConversationController
|
2020-01-17 22:23:19 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/* eslint-disable more/no-then */
|
|
|
|
|
|
|
|
// eslint-disable-next-line func-names
|
2020-11-18 15:15:42 +00:00
|
|
|
(function () {
|
2020-01-17 22:23:19 +00:00
|
|
|
window.Whisper = window.Whisper || {};
|
|
|
|
Whisper.Reactions = new (Backbone.Collection.extend({
|
|
|
|
forMessage(message) {
|
|
|
|
if (message.isOutgoing()) {
|
2020-03-20 19:07:44 +00:00
|
|
|
const outgoingReactions = this.filter({
|
2020-01-17 22:23:19 +00:00
|
|
|
targetTimestamp: message.get('sent_at'),
|
|
|
|
});
|
|
|
|
|
2020-03-20 19:07:44 +00:00
|
|
|
if (outgoingReactions.length > 0) {
|
2020-01-17 22:23:19 +00:00
|
|
|
window.log.info('Found early reaction for outgoing message');
|
2020-03-20 19:07:44 +00:00
|
|
|
this.remove(outgoingReactions);
|
|
|
|
return outgoingReactions;
|
2020-01-17 22:23:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-01 15:06:41 +00:00
|
|
|
const senderId = message.getContactId();
|
|
|
|
const sentAt = message.get('sent_at');
|
2020-03-20 19:07:44 +00:00
|
|
|
const reactionsBySource = this.filter(re => {
|
2020-11-01 15:06:41 +00:00
|
|
|
const targetSenderId = ConversationController.ensureContactIds({
|
2020-07-10 18:28:49 +00:00
|
|
|
uuid: re.get('targetAuthorUuid'),
|
|
|
|
});
|
2020-11-01 15:06:41 +00:00
|
|
|
const targetTimestamp = re.get('targetTimestamp');
|
|
|
|
return targetSenderId === senderId && targetTimestamp === sentAt;
|
2020-01-17 22:23:19 +00:00
|
|
|
});
|
|
|
|
|
2020-03-20 19:07:44 +00:00
|
|
|
if (reactionsBySource.length > 0) {
|
2020-01-17 22:23:19 +00:00
|
|
|
window.log.info('Found early reaction for message');
|
2020-03-20 19:07:44 +00:00
|
|
|
this.remove(reactionsBySource);
|
|
|
|
return reactionsBySource;
|
2020-01-17 22:23:19 +00:00
|
|
|
}
|
|
|
|
|
2020-03-20 19:07:44 +00:00
|
|
|
return [];
|
2020-01-17 22:23:19 +00:00
|
|
|
},
|
|
|
|
async onReaction(reaction) {
|
|
|
|
try {
|
2020-11-02 22:49:07 +00:00
|
|
|
// The conversation the target message was in; we have to find it in the database
|
|
|
|
// to to figure that out.
|
2020-07-10 18:24:58 +00:00
|
|
|
const targetConversation = await ConversationController.getConversationForTargetMessage(
|
2020-07-10 18:28:49 +00:00
|
|
|
ConversationController.ensureContactIds({
|
|
|
|
uuid: reaction.get('targetAuthorUuid'),
|
|
|
|
}),
|
2020-07-10 18:24:58 +00:00
|
|
|
reaction.get('targetTimestamp')
|
2020-01-17 22:23:19 +00:00
|
|
|
);
|
2020-07-10 18:24:58 +00:00
|
|
|
if (!targetConversation) {
|
2020-07-27 18:15:32 +00:00
|
|
|
window.log.info(
|
2020-11-02 22:49:07 +00:00
|
|
|
'No target conversation for reaction',
|
2020-07-27 18:15:32 +00:00
|
|
|
reaction.get('targetAuthorUuid'),
|
|
|
|
reaction.get('targetTimestamp')
|
|
|
|
);
|
2021-05-13 19:10:20 +00:00
|
|
|
return undefined;
|
2020-07-10 18:24:58 +00:00
|
|
|
}
|
2020-01-17 22:23:19 +00:00
|
|
|
|
2020-07-10 18:24:58 +00:00
|
|
|
// awaiting is safe since `onReaction` is never called from inside the queue
|
2021-06-14 21:55:14 +00:00
|
|
|
return await targetConversation.queueJob(
|
|
|
|
'Reactions.onReaction',
|
|
|
|
async () => {
|
2020-07-10 18:24:58 +00:00
|
|
|
window.log.info(
|
2021-06-14 21:55:14 +00:00
|
|
|
'Handling reaction for',
|
2020-07-10 18:24:58 +00:00
|
|
|
reaction.get('targetTimestamp')
|
|
|
|
);
|
|
|
|
|
2021-06-14 21:55:14 +00:00
|
|
|
const messages = await window.Signal.Data.getMessagesBySentAt(
|
|
|
|
reaction.get('targetTimestamp'),
|
|
|
|
{
|
|
|
|
MessageCollection: Whisper.MessageCollection,
|
|
|
|
}
|
|
|
|
);
|
|
|
|
// Message is fetched inside the conversation queue so we have the
|
|
|
|
// most recent data
|
|
|
|
const targetMessage = messages.find(m => {
|
|
|
|
const contact = m.getContact();
|
|
|
|
|
|
|
|
if (!contact) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const mcid = contact.get('id');
|
|
|
|
const recid = ConversationController.ensureContactIds({
|
|
|
|
uuid: reaction.get('targetAuthorUuid'),
|
2020-07-10 18:24:58 +00:00
|
|
|
});
|
2021-06-14 21:55:14 +00:00
|
|
|
return mcid === recid;
|
|
|
|
});
|
2020-07-10 18:24:58 +00:00
|
|
|
|
2021-06-14 21:55:14 +00:00
|
|
|
if (!targetMessage) {
|
|
|
|
window.log.info(
|
|
|
|
'No message for reaction',
|
|
|
|
reaction.get('targetAuthorUuid'),
|
|
|
|
reaction.get('targetTimestamp')
|
|
|
|
);
|
|
|
|
|
|
|
|
// Since we haven't received the message for which we are removing a
|
|
|
|
// reaction, we can just remove those pending reactions
|
|
|
|
if (reaction.get('remove')) {
|
|
|
|
this.remove(reaction);
|
|
|
|
const oldReaction = this.where({
|
|
|
|
targetAuthorUuid: reaction.get('targetAuthorUuid'),
|
|
|
|
targetTimestamp: reaction.get('targetTimestamp'),
|
|
|
|
emoji: reaction.get('emoji'),
|
|
|
|
});
|
|
|
|
oldReaction.forEach(r => this.remove(r));
|
|
|
|
}
|
|
|
|
|
|
|
|
return undefined;
|
|
|
|
}
|
2020-01-17 22:23:19 +00:00
|
|
|
|
2021-06-14 21:55:14 +00:00
|
|
|
const message = MessageController.register(
|
|
|
|
targetMessage.id,
|
|
|
|
targetMessage
|
|
|
|
);
|
2020-01-17 22:23:19 +00:00
|
|
|
|
2021-06-14 21:55:14 +00:00
|
|
|
const oldReaction = await message.handleReaction(reaction);
|
2020-01-17 22:23:19 +00:00
|
|
|
|
2021-06-14 21:55:14 +00:00
|
|
|
this.remove(reaction);
|
2021-05-13 19:10:20 +00:00
|
|
|
|
2021-06-14 21:55:14 +00:00
|
|
|
return oldReaction;
|
|
|
|
}
|
|
|
|
);
|
2020-01-17 22:23:19 +00:00
|
|
|
} catch (error) {
|
|
|
|
window.log.error(
|
|
|
|
'Reactions.onReaction error:',
|
|
|
|
error && error.stack ? error.stack : error
|
|
|
|
);
|
2021-05-13 19:10:20 +00:00
|
|
|
return undefined;
|
2020-01-17 22:23:19 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
}))();
|
|
|
|
})();
|