2021-12-08 19:52:46 +00:00
|
|
|
// Copyright 2021 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import { assert } from 'chai';
|
2023-08-10 16:43:33 +00:00
|
|
|
import { v4 as generateUuid } from 'uuid';
|
2021-12-08 19:52:46 +00:00
|
|
|
|
|
|
|
import dataInterface from '../../sql/Client';
|
2023-08-10 16:43:33 +00:00
|
|
|
import { generateAci } from '../../types/ServiceId';
|
2021-12-08 19:52:46 +00:00
|
|
|
|
|
|
|
import type { ReactionType } from '../../types/Reactions';
|
2022-11-16 20:18:02 +00:00
|
|
|
import { DurationInSeconds } from '../../util/durations';
|
2021-12-08 19:52:46 +00:00
|
|
|
import type { MessageAttributesType } from '../../model-types.d';
|
|
|
|
import { ReadStatus } from '../../messages/MessageReadStatus';
|
|
|
|
|
|
|
|
const {
|
|
|
|
_removeAllMessages,
|
|
|
|
_removeAllReactions,
|
|
|
|
_getAllReactions,
|
|
|
|
_getAllMessages,
|
|
|
|
addReaction,
|
|
|
|
saveMessages,
|
|
|
|
getTotalUnreadForConversation,
|
|
|
|
getUnreadByConversationAndMarkRead,
|
|
|
|
getUnreadReactionsAndMarkRead,
|
|
|
|
} = dataInterface;
|
|
|
|
|
|
|
|
describe('sql/markRead', () => {
|
|
|
|
beforeEach(async () => {
|
|
|
|
await _removeAllMessages();
|
|
|
|
await _removeAllReactions();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('properly finds and reads unread messages in current conversation', async () => {
|
2021-12-10 22:51:54 +00:00
|
|
|
assert.lengthOf(await _getAllMessages(), 0);
|
2021-12-08 19:52:46 +00:00
|
|
|
|
|
|
|
const start = Date.now();
|
|
|
|
const readAt = start + 20;
|
2023-08-10 16:43:33 +00:00
|
|
|
const conversationId = generateUuid();
|
|
|
|
const ourAci = generateAci();
|
2021-12-08 19:52:46 +00:00
|
|
|
|
2022-09-30 00:57:11 +00:00
|
|
|
const oldest: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 1',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 1,
|
|
|
|
received_at: start + 1,
|
|
|
|
timestamp: start + 1,
|
|
|
|
readStatus: ReadStatus.Read,
|
|
|
|
};
|
2022-09-30 00:57:11 +00:00
|
|
|
const oldestUnread: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 2',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 2,
|
|
|
|
received_at: start + 2,
|
|
|
|
timestamp: start + 2,
|
|
|
|
readStatus: ReadStatus.Unread,
|
|
|
|
};
|
2022-09-30 00:57:11 +00:00
|
|
|
const unreadInAnotherConvo: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 3',
|
|
|
|
type: 'incoming',
|
2023-08-10 16:43:33 +00:00
|
|
|
conversationId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
sent_at: start + 3,
|
|
|
|
received_at: start + 3,
|
|
|
|
timestamp: start + 3,
|
|
|
|
readStatus: ReadStatus.Unread,
|
|
|
|
};
|
2022-09-30 00:57:11 +00:00
|
|
|
const unread: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 4',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 4,
|
|
|
|
received_at: start + 4,
|
|
|
|
timestamp: start + 4,
|
|
|
|
readStatus: ReadStatus.Unread,
|
|
|
|
};
|
2022-09-30 00:57:11 +00:00
|
|
|
const unreadStory: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 5',
|
|
|
|
type: 'story',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 5,
|
|
|
|
received_at: start + 5,
|
|
|
|
timestamp: start + 5,
|
|
|
|
readStatus: ReadStatus.Unread,
|
2023-08-10 16:43:33 +00:00
|
|
|
storyId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
};
|
2022-09-30 00:57:11 +00:00
|
|
|
const unreadStoryReply: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 6',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 6,
|
|
|
|
received_at: start + 6,
|
|
|
|
timestamp: start + 6,
|
|
|
|
readStatus: ReadStatus.Unread,
|
2023-08-10 16:43:33 +00:00
|
|
|
storyId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
};
|
2022-09-30 00:57:11 +00:00
|
|
|
const newestUnread: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 7',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 7,
|
|
|
|
received_at: start + 7,
|
|
|
|
timestamp: start + 7,
|
|
|
|
readStatus: ReadStatus.Unread,
|
|
|
|
};
|
|
|
|
|
|
|
|
await saveMessages(
|
2022-09-30 00:57:11 +00:00
|
|
|
[
|
|
|
|
oldest,
|
|
|
|
oldestUnread,
|
|
|
|
unreadInAnotherConvo,
|
|
|
|
unread,
|
|
|
|
unreadStory,
|
|
|
|
unreadStoryReply,
|
|
|
|
newestUnread,
|
|
|
|
],
|
2021-12-08 19:52:46 +00:00
|
|
|
{
|
|
|
|
forceSave: true,
|
2023-08-10 16:43:33 +00:00
|
|
|
ourAci,
|
2021-12-08 19:52:46 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2021-12-10 22:51:54 +00:00
|
|
|
assert.lengthOf(await _getAllMessages(), 7);
|
2021-12-08 19:52:46 +00:00
|
|
|
assert.strictEqual(
|
2022-04-27 17:41:24 +00:00
|
|
|
await getTotalUnreadForConversation(conversationId, {
|
|
|
|
storyId: undefined,
|
2022-09-30 00:57:11 +00:00
|
|
|
includeStoryReplies: false,
|
2022-04-27 17:41:24 +00:00
|
|
|
}),
|
2022-09-30 00:57:11 +00:00
|
|
|
3,
|
|
|
|
'no stories/unread count - before'
|
2021-12-08 19:52:46 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
const markedRead = await getUnreadByConversationAndMarkRead({
|
|
|
|
conversationId,
|
2022-09-30 00:57:11 +00:00
|
|
|
newestUnreadAt: unreadStoryReply.received_at,
|
2021-12-08 19:52:46 +00:00
|
|
|
readAt,
|
2022-09-30 00:57:11 +00:00
|
|
|
includeStoryReplies: false,
|
2021-12-08 19:52:46 +00:00
|
|
|
});
|
|
|
|
|
2022-09-30 00:57:11 +00:00
|
|
|
assert.lengthOf(markedRead, 2, 'no stories/two messages marked read');
|
2021-12-08 19:52:46 +00:00
|
|
|
assert.strictEqual(
|
2022-04-27 17:41:24 +00:00
|
|
|
await getTotalUnreadForConversation(conversationId, {
|
|
|
|
storyId: undefined,
|
2022-09-30 00:57:11 +00:00
|
|
|
includeStoryReplies: false,
|
2022-04-27 17:41:24 +00:00
|
|
|
}),
|
2022-09-30 00:57:11 +00:00
|
|
|
1,
|
|
|
|
'no stories/unread count - after'
|
2021-12-08 19:52:46 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
// Sorted in descending order
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead[0].id,
|
2022-09-30 00:57:11 +00:00
|
|
|
unread.id,
|
|
|
|
'no stories/first should be "unread" message'
|
2021-12-08 19:52:46 +00:00
|
|
|
);
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead[1].id,
|
2022-09-30 00:57:11 +00:00
|
|
|
oldestUnread.id,
|
|
|
|
'no stories/second should be oldestUnread'
|
2021-12-08 19:52:46 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
const markedRead2 = await getUnreadByConversationAndMarkRead({
|
|
|
|
conversationId,
|
2022-09-30 00:57:11 +00:00
|
|
|
newestUnreadAt: newestUnread.received_at,
|
2021-12-08 19:52:46 +00:00
|
|
|
readAt,
|
2022-09-30 00:57:11 +00:00
|
|
|
includeStoryReplies: true,
|
2021-12-08 19:52:46 +00:00
|
|
|
});
|
|
|
|
|
2022-09-30 00:57:11 +00:00
|
|
|
assert.lengthOf(markedRead2, 2, 'with stories/two messages marked read');
|
|
|
|
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead2[0].id,
|
|
|
|
newestUnread.id,
|
|
|
|
'with stories/should be newestUnread'
|
|
|
|
);
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead2[1].id,
|
|
|
|
unreadStoryReply.id,
|
|
|
|
'with stories/should be unreadStoryReply'
|
|
|
|
);
|
2021-12-08 19:52:46 +00:00
|
|
|
|
|
|
|
assert.strictEqual(
|
2022-04-27 17:41:24 +00:00
|
|
|
await getTotalUnreadForConversation(conversationId, {
|
|
|
|
storyId: undefined,
|
2022-09-30 00:57:11 +00:00
|
|
|
includeStoryReplies: true,
|
2022-04-27 17:41:24 +00:00
|
|
|
}),
|
2021-12-08 19:52:46 +00:00
|
|
|
0,
|
2022-09-30 00:57:11 +00:00
|
|
|
'with stories/unread count'
|
2021-12-08 19:52:46 +00:00
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('properly finds and reads unread messages in story', async () => {
|
2021-12-10 22:51:54 +00:00
|
|
|
assert.lengthOf(await _getAllMessages(), 0);
|
2021-12-08 19:52:46 +00:00
|
|
|
|
|
|
|
const start = Date.now();
|
|
|
|
const readAt = start + 20;
|
2023-08-10 16:43:33 +00:00
|
|
|
const conversationId = generateUuid();
|
|
|
|
const storyId = generateUuid();
|
|
|
|
const ourAci = generateAci();
|
2021-12-08 19:52:46 +00:00
|
|
|
|
|
|
|
const message1: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 1',
|
|
|
|
type: 'story',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 1,
|
|
|
|
received_at: start + 1,
|
|
|
|
timestamp: start + 1,
|
|
|
|
readStatus: ReadStatus.Read,
|
|
|
|
storyId,
|
|
|
|
};
|
|
|
|
const message2: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 2',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 2,
|
|
|
|
received_at: start + 2,
|
|
|
|
timestamp: start + 2,
|
|
|
|
readStatus: ReadStatus.Unread,
|
|
|
|
storyId,
|
|
|
|
};
|
|
|
|
const message3: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 3',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 3,
|
|
|
|
received_at: start + 3,
|
|
|
|
timestamp: start + 3,
|
|
|
|
readStatus: ReadStatus.Unread,
|
2023-08-10 16:43:33 +00:00
|
|
|
storyId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
};
|
|
|
|
const message4: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 4',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 4,
|
|
|
|
received_at: start + 4,
|
|
|
|
timestamp: start + 4,
|
|
|
|
readStatus: ReadStatus.Unread,
|
|
|
|
storyId,
|
|
|
|
};
|
|
|
|
const message5: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 5',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 5,
|
|
|
|
received_at: start + 5,
|
|
|
|
timestamp: start + 5,
|
|
|
|
readStatus: ReadStatus.Unread,
|
2023-08-10 16:43:33 +00:00
|
|
|
storyId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
};
|
|
|
|
const message6: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 6',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 6,
|
|
|
|
received_at: start + 6,
|
|
|
|
timestamp: start + 6,
|
|
|
|
readStatus: ReadStatus.Unread,
|
2023-08-10 16:43:33 +00:00
|
|
|
storyId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
};
|
|
|
|
const message7: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 7',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 7,
|
|
|
|
received_at: start + 7,
|
|
|
|
timestamp: start + 7,
|
|
|
|
readStatus: ReadStatus.Unread,
|
|
|
|
storyId,
|
|
|
|
};
|
|
|
|
|
|
|
|
await saveMessages(
|
|
|
|
[message1, message2, message3, message4, message5, message6, message7],
|
|
|
|
{
|
|
|
|
forceSave: true,
|
2023-08-10 16:43:33 +00:00
|
|
|
ourAci,
|
2021-12-08 19:52:46 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2021-12-10 22:51:54 +00:00
|
|
|
assert.lengthOf(await _getAllMessages(), 7);
|
2021-12-08 19:52:46 +00:00
|
|
|
|
|
|
|
const markedRead = await getUnreadByConversationAndMarkRead({
|
|
|
|
conversationId,
|
|
|
|
newestUnreadAt: message7.received_at,
|
|
|
|
readAt,
|
|
|
|
storyId,
|
2022-09-30 00:57:11 +00:00
|
|
|
includeStoryReplies: false,
|
2021-12-08 19:52:46 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
assert.lengthOf(markedRead, 3, 'three messages marked read');
|
|
|
|
|
|
|
|
// Sorted in descending order
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead[0].id,
|
|
|
|
message7.id,
|
|
|
|
'first should be message7'
|
|
|
|
);
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead[1].id,
|
|
|
|
message4.id,
|
|
|
|
'first should be message4'
|
|
|
|
);
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead[2].id,
|
|
|
|
message2.id,
|
|
|
|
'second should be message2'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('properly starts disappearing message timer, even if message is already read', async () => {
|
2022-09-22 23:49:06 +00:00
|
|
|
const now = Date.now();
|
2021-12-10 22:51:54 +00:00
|
|
|
assert.lengthOf(await _getAllMessages(), 0);
|
2021-12-08 19:52:46 +00:00
|
|
|
|
|
|
|
const start = Date.now();
|
|
|
|
const readAt = start + 20;
|
2023-08-10 16:43:33 +00:00
|
|
|
const conversationId = generateUuid();
|
2022-11-16 20:18:02 +00:00
|
|
|
const expireTimer = DurationInSeconds.fromSeconds(15);
|
2023-08-10 16:43:33 +00:00
|
|
|
const ourAci = generateAci();
|
2021-12-08 19:52:46 +00:00
|
|
|
|
|
|
|
const message1: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 1',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 1,
|
|
|
|
received_at: start + 1,
|
|
|
|
timestamp: start + 1,
|
|
|
|
expireTimer,
|
|
|
|
expirationStartTimestamp: start + 1,
|
|
|
|
readStatus: ReadStatus.Read,
|
|
|
|
};
|
|
|
|
const message2: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 2',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 2,
|
|
|
|
received_at: start + 2,
|
|
|
|
timestamp: start + 2,
|
|
|
|
expireTimer,
|
|
|
|
readStatus: ReadStatus.Read,
|
|
|
|
};
|
|
|
|
const message3: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 3',
|
|
|
|
type: 'incoming',
|
2023-08-10 16:43:33 +00:00
|
|
|
conversationId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
sent_at: start + 3,
|
|
|
|
received_at: start + 3,
|
|
|
|
timestamp: start + 3,
|
|
|
|
expireTimer,
|
|
|
|
readStatus: ReadStatus.Unread,
|
|
|
|
};
|
|
|
|
const message4: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 4',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 4,
|
|
|
|
received_at: start + 4,
|
|
|
|
timestamp: start + 4,
|
|
|
|
expireTimer,
|
|
|
|
readStatus: ReadStatus.Unread,
|
|
|
|
};
|
|
|
|
const message5: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 5',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 5,
|
|
|
|
received_at: start + 5,
|
|
|
|
timestamp: start + 5,
|
|
|
|
readStatus: ReadStatus.Unread,
|
|
|
|
};
|
|
|
|
|
|
|
|
await saveMessages([message1, message2, message3, message4, message5], {
|
|
|
|
forceSave: true,
|
2023-08-10 16:43:33 +00:00
|
|
|
ourAci,
|
2021-12-08 19:52:46 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
assert.strictEqual(
|
2022-04-27 17:41:24 +00:00
|
|
|
await getTotalUnreadForConversation(conversationId, {
|
|
|
|
storyId: undefined,
|
2022-09-30 00:57:11 +00:00
|
|
|
includeStoryReplies: true,
|
2022-04-27 17:41:24 +00:00
|
|
|
}),
|
2021-12-08 19:52:46 +00:00
|
|
|
2,
|
|
|
|
'unread count'
|
|
|
|
);
|
2021-12-10 22:51:54 +00:00
|
|
|
assert.lengthOf(await _getAllMessages(), 5);
|
2021-12-08 19:52:46 +00:00
|
|
|
|
|
|
|
const markedRead = await getUnreadByConversationAndMarkRead({
|
|
|
|
conversationId,
|
|
|
|
newestUnreadAt: message4.received_at,
|
|
|
|
readAt,
|
2022-09-30 00:57:11 +00:00
|
|
|
includeStoryReplies: false,
|
2022-09-22 23:49:06 +00:00
|
|
|
now,
|
2021-12-08 19:52:46 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
assert.lengthOf(markedRead, 1, 'one message marked read');
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead[0].id,
|
|
|
|
message4.id,
|
|
|
|
'first should be message4'
|
|
|
|
);
|
|
|
|
assert.strictEqual(
|
2022-04-27 17:41:24 +00:00
|
|
|
await getTotalUnreadForConversation(conversationId, {
|
|
|
|
storyId: undefined,
|
2022-09-30 00:57:11 +00:00
|
|
|
includeStoryReplies: true,
|
2022-04-27 17:41:24 +00:00
|
|
|
}),
|
2021-12-08 19:52:46 +00:00
|
|
|
1,
|
|
|
|
'unread count'
|
|
|
|
);
|
|
|
|
|
2021-12-10 22:51:54 +00:00
|
|
|
const allMessages = await _getAllMessages();
|
|
|
|
const sorted = allMessages.sort(
|
|
|
|
(left, right) => left.timestamp - right.timestamp
|
|
|
|
);
|
2021-12-08 19:52:46 +00:00
|
|
|
|
2021-12-10 22:51:54 +00:00
|
|
|
assert.strictEqual(sorted[1].id, message2.id, 'checking message 2');
|
2021-12-08 19:52:46 +00:00
|
|
|
assert.isAtMost(
|
2021-12-10 22:51:54 +00:00
|
|
|
sorted[1].expirationStartTimestamp ?? Infinity,
|
2022-09-22 23:49:06 +00:00
|
|
|
now,
|
2021-12-08 19:52:46 +00:00
|
|
|
'checking message 2 expirationStartTimestamp'
|
|
|
|
);
|
|
|
|
|
2021-12-10 22:51:54 +00:00
|
|
|
assert.strictEqual(sorted[3].id, message4.id, 'checking message 4');
|
2021-12-08 19:52:46 +00:00
|
|
|
assert.isAtMost(
|
2021-12-10 22:51:54 +00:00
|
|
|
sorted[3].expirationStartTimestamp ?? Infinity,
|
2022-09-22 23:49:06 +00:00
|
|
|
now,
|
2021-12-08 19:52:46 +00:00
|
|
|
'checking message 4 expirationStartTimestamp'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('properly finds and reads unread reactions in current conversation', async () => {
|
|
|
|
assert.lengthOf(await _getAllReactions(), 0);
|
|
|
|
|
|
|
|
const start = Date.now();
|
2023-08-10 16:43:33 +00:00
|
|
|
const conversationId = generateUuid();
|
|
|
|
const storyId = generateUuid();
|
|
|
|
const ourAci = generateAci();
|
2021-12-08 19:52:46 +00:00
|
|
|
|
2021-12-21 20:01:09 +00:00
|
|
|
const pad: Array<MessageAttributesType> = Array.from({ length: 4 }, _ => {
|
|
|
|
return {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-21 20:01:09 +00:00
|
|
|
body: 'pad message',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start - 1,
|
|
|
|
received_at: start - 1,
|
|
|
|
timestamp: start - 1,
|
|
|
|
};
|
|
|
|
});
|
2021-12-08 19:52:46 +00:00
|
|
|
const message1: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 1',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 1,
|
|
|
|
received_at: start + 1,
|
|
|
|
timestamp: start + 1,
|
|
|
|
};
|
|
|
|
const message2: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 2',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 2,
|
|
|
|
received_at: start + 2,
|
|
|
|
timestamp: start + 2,
|
|
|
|
storyId,
|
|
|
|
};
|
|
|
|
const message3: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 3',
|
|
|
|
type: 'incoming',
|
2023-08-10 16:43:33 +00:00
|
|
|
conversationId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
sent_at: start + 3,
|
|
|
|
received_at: start + 3,
|
|
|
|
timestamp: start + 3,
|
|
|
|
};
|
|
|
|
const message4: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 4',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 4,
|
|
|
|
received_at: start + 4,
|
|
|
|
timestamp: start + 4,
|
|
|
|
};
|
|
|
|
const message5: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 5',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 5,
|
|
|
|
received_at: start + 5,
|
|
|
|
timestamp: start + 5,
|
|
|
|
};
|
|
|
|
|
2021-12-21 20:01:09 +00:00
|
|
|
await saveMessages(
|
|
|
|
[...pad, message1, message2, message3, message4, message5],
|
|
|
|
{
|
|
|
|
forceSave: true,
|
2023-08-10 16:43:33 +00:00
|
|
|
ourAci,
|
2021-12-21 20:01:09 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
assert.lengthOf(await _getAllMessages(), pad.length + 5);
|
2021-12-08 19:52:46 +00:00
|
|
|
|
|
|
|
const reaction1: ReactionType = {
|
|
|
|
conversationId,
|
|
|
|
emoji: '🎉',
|
2023-08-10 16:43:33 +00:00
|
|
|
fromId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
messageId: message1.id,
|
|
|
|
messageReceivedAt: message1.received_at,
|
2023-08-16 20:54:39 +00:00
|
|
|
targetAuthorAci: generateAci(),
|
2021-12-08 19:52:46 +00:00
|
|
|
targetTimestamp: start,
|
|
|
|
};
|
|
|
|
const reaction2: ReactionType = {
|
|
|
|
conversationId,
|
|
|
|
emoji: '🚀',
|
2023-08-10 16:43:33 +00:00
|
|
|
fromId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
messageId: message2.id,
|
|
|
|
messageReceivedAt: message2.received_at,
|
2023-08-16 20:54:39 +00:00
|
|
|
targetAuthorAci: generateAci(),
|
2021-12-08 19:52:46 +00:00
|
|
|
targetTimestamp: start,
|
|
|
|
};
|
|
|
|
const reaction3: ReactionType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
conversationId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
emoji: '☀️',
|
2023-08-10 16:43:33 +00:00
|
|
|
fromId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
messageId: message3.id,
|
|
|
|
messageReceivedAt: message3.received_at,
|
2023-08-16 20:54:39 +00:00
|
|
|
targetAuthorAci: generateAci(),
|
2021-12-08 19:52:46 +00:00
|
|
|
targetTimestamp: start,
|
|
|
|
};
|
|
|
|
const reaction4: ReactionType = {
|
|
|
|
conversationId,
|
|
|
|
emoji: '❤️🔥',
|
2023-08-10 16:43:33 +00:00
|
|
|
fromId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
messageId: message4.id,
|
|
|
|
messageReceivedAt: message4.received_at,
|
2023-08-16 20:54:39 +00:00
|
|
|
targetAuthorAci: generateAci(),
|
2021-12-08 19:52:46 +00:00
|
|
|
targetTimestamp: start,
|
|
|
|
};
|
|
|
|
const reaction5: ReactionType = {
|
|
|
|
conversationId,
|
|
|
|
emoji: '🆒',
|
2023-08-10 16:43:33 +00:00
|
|
|
fromId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
messageId: message5.id,
|
|
|
|
messageReceivedAt: message5.received_at,
|
2023-08-16 20:54:39 +00:00
|
|
|
targetAuthorAci: generateAci(),
|
2021-12-08 19:52:46 +00:00
|
|
|
targetTimestamp: start,
|
|
|
|
};
|
|
|
|
|
|
|
|
await addReaction(reaction1);
|
|
|
|
await addReaction(reaction2);
|
|
|
|
await addReaction(reaction3);
|
|
|
|
await addReaction(reaction4);
|
|
|
|
await addReaction(reaction5);
|
|
|
|
|
|
|
|
assert.lengthOf(await _getAllReactions(), 5);
|
|
|
|
const markedRead = await getUnreadReactionsAndMarkRead({
|
|
|
|
conversationId,
|
|
|
|
newestUnreadAt: reaction4.messageReceivedAt,
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.lengthOf(markedRead, 2, 'two reactions marked read');
|
|
|
|
|
|
|
|
// Sorted in descending order
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead[0].messageId,
|
|
|
|
reaction4.messageId,
|
|
|
|
'first should be reaction4'
|
|
|
|
);
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead[1].messageId,
|
|
|
|
reaction1.messageId,
|
|
|
|
'second should be reaction1'
|
|
|
|
);
|
|
|
|
|
|
|
|
const markedRead2 = await getUnreadReactionsAndMarkRead({
|
|
|
|
conversationId,
|
|
|
|
newestUnreadAt: reaction5.messageReceivedAt,
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.lengthOf(markedRead2, 1);
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead2[0].messageId,
|
|
|
|
reaction5.messageId,
|
|
|
|
'should be reaction5'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('properly finds and reads unread reactions in story', async () => {
|
|
|
|
assert.lengthOf(await _getAllReactions(), 0);
|
|
|
|
|
|
|
|
const start = Date.now();
|
2023-08-10 16:43:33 +00:00
|
|
|
const conversationId = generateUuid();
|
|
|
|
const storyId = generateUuid();
|
|
|
|
const ourAci = generateAci();
|
2021-12-08 19:52:46 +00:00
|
|
|
|
|
|
|
const message1: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 1',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 1,
|
|
|
|
received_at: start + 1,
|
|
|
|
timestamp: start + 1,
|
|
|
|
storyId,
|
|
|
|
};
|
|
|
|
const message2: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 2',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 2,
|
|
|
|
received_at: start + 2,
|
|
|
|
timestamp: start + 2,
|
2023-08-10 16:43:33 +00:00
|
|
|
storyId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
};
|
|
|
|
const message3: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 3',
|
|
|
|
type: 'incoming',
|
2023-08-10 16:43:33 +00:00
|
|
|
conversationId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
sent_at: start + 3,
|
|
|
|
received_at: start + 3,
|
|
|
|
timestamp: start + 3,
|
|
|
|
};
|
|
|
|
const message4: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 4',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 4,
|
|
|
|
received_at: start + 4,
|
|
|
|
timestamp: start + 4,
|
|
|
|
storyId,
|
|
|
|
};
|
|
|
|
const message5: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
body: 'message 5',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 5,
|
|
|
|
received_at: start + 5,
|
|
|
|
timestamp: start + 5,
|
|
|
|
storyId,
|
|
|
|
};
|
|
|
|
|
|
|
|
await saveMessages([message1, message2, message3, message4, message5], {
|
|
|
|
forceSave: true,
|
2023-08-10 16:43:33 +00:00
|
|
|
ourAci,
|
2021-12-08 19:52:46 +00:00
|
|
|
});
|
2021-12-10 22:51:54 +00:00
|
|
|
assert.lengthOf(await _getAllMessages(), 5);
|
2021-12-08 19:52:46 +00:00
|
|
|
|
|
|
|
const reaction1: ReactionType = {
|
|
|
|
conversationId,
|
|
|
|
emoji: '🎉',
|
2023-08-10 16:43:33 +00:00
|
|
|
fromId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
messageId: message1.id,
|
|
|
|
messageReceivedAt: message1.received_at,
|
2023-08-16 20:54:39 +00:00
|
|
|
targetAuthorAci: generateAci(),
|
2021-12-08 19:52:46 +00:00
|
|
|
targetTimestamp: start,
|
|
|
|
};
|
|
|
|
const reaction2: ReactionType = {
|
|
|
|
conversationId,
|
|
|
|
emoji: '🚀',
|
2023-08-10 16:43:33 +00:00
|
|
|
fromId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
messageId: message2.id,
|
|
|
|
messageReceivedAt: message2.received_at,
|
2023-08-16 20:54:39 +00:00
|
|
|
targetAuthorAci: generateAci(),
|
2021-12-08 19:52:46 +00:00
|
|
|
targetTimestamp: start,
|
|
|
|
};
|
|
|
|
const reaction3: ReactionType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
conversationId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
emoji: '☀️',
|
2023-08-10 16:43:33 +00:00
|
|
|
fromId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
messageId: message3.id,
|
|
|
|
messageReceivedAt: message3.received_at,
|
2023-08-16 20:54:39 +00:00
|
|
|
targetAuthorAci: generateAci(),
|
2021-12-08 19:52:46 +00:00
|
|
|
targetTimestamp: start,
|
|
|
|
};
|
|
|
|
const reaction4: ReactionType = {
|
|
|
|
conversationId,
|
|
|
|
emoji: '❤️🔥',
|
2023-08-10 16:43:33 +00:00
|
|
|
fromId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
messageId: message4.id,
|
|
|
|
messageReceivedAt: message4.received_at,
|
2023-08-16 20:54:39 +00:00
|
|
|
targetAuthorAci: generateAci(),
|
2021-12-08 19:52:46 +00:00
|
|
|
targetTimestamp: start,
|
|
|
|
};
|
|
|
|
const reaction5: ReactionType = {
|
|
|
|
conversationId,
|
|
|
|
emoji: '🆒',
|
2023-08-10 16:43:33 +00:00
|
|
|
fromId: generateUuid(),
|
2021-12-08 19:52:46 +00:00
|
|
|
messageId: message5.id,
|
|
|
|
messageReceivedAt: message5.received_at,
|
2023-08-16 20:54:39 +00:00
|
|
|
targetAuthorAci: generateAci(),
|
2021-12-08 19:52:46 +00:00
|
|
|
targetTimestamp: start,
|
|
|
|
};
|
|
|
|
|
|
|
|
await addReaction(reaction1);
|
|
|
|
await addReaction(reaction2);
|
|
|
|
await addReaction(reaction3);
|
|
|
|
await addReaction(reaction4);
|
|
|
|
await addReaction(reaction5);
|
|
|
|
|
|
|
|
assert.lengthOf(await _getAllReactions(), 5);
|
|
|
|
const markedRead = await getUnreadReactionsAndMarkRead({
|
|
|
|
conversationId,
|
|
|
|
newestUnreadAt: reaction4.messageReceivedAt,
|
|
|
|
storyId,
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.lengthOf(markedRead, 2, 'two reactions marked read');
|
|
|
|
|
|
|
|
// Sorted in descending order
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead[0].messageId,
|
|
|
|
reaction4.messageId,
|
|
|
|
'first should be reaction4'
|
|
|
|
);
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead[1].messageId,
|
|
|
|
reaction1.messageId,
|
|
|
|
'second should be reaction1'
|
|
|
|
);
|
|
|
|
|
|
|
|
const markedRead2 = await getUnreadReactionsAndMarkRead({
|
|
|
|
conversationId,
|
|
|
|
newestUnreadAt: reaction5.messageReceivedAt,
|
|
|
|
storyId,
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.lengthOf(markedRead2, 1);
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead2[0].messageId,
|
|
|
|
reaction5.messageId,
|
|
|
|
'should be reaction5'
|
|
|
|
);
|
|
|
|
});
|
2022-04-20 23:33:38 +00:00
|
|
|
|
|
|
|
it('does not include group story replies', async () => {
|
|
|
|
assert.lengthOf(await _getAllMessages(), 0);
|
|
|
|
|
|
|
|
const start = Date.now();
|
|
|
|
const readAt = start + 20;
|
2023-08-10 16:43:33 +00:00
|
|
|
const conversationId = generateUuid();
|
|
|
|
const storyId = generateUuid();
|
|
|
|
const ourAci = generateAci();
|
2022-04-20 23:33:38 +00:00
|
|
|
|
|
|
|
const message1: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2022-04-20 23:33:38 +00:00
|
|
|
body: 'message 1',
|
|
|
|
type: 'story',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 1,
|
|
|
|
received_at: start + 1,
|
|
|
|
timestamp: start + 1,
|
|
|
|
readStatus: ReadStatus.Read,
|
|
|
|
};
|
|
|
|
const message2: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2022-04-20 23:33:38 +00:00
|
|
|
body: 'message 2',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 2,
|
|
|
|
received_at: start + 2,
|
|
|
|
timestamp: start + 2,
|
|
|
|
readStatus: ReadStatus.Unread,
|
|
|
|
storyId,
|
|
|
|
};
|
|
|
|
const message3: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2022-04-20 23:33:38 +00:00
|
|
|
body: 'message 3',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 3,
|
|
|
|
received_at: start + 3,
|
|
|
|
timestamp: start + 3,
|
|
|
|
readStatus: ReadStatus.Unread,
|
|
|
|
};
|
|
|
|
const message4: MessageAttributesType = {
|
2023-08-10 16:43:33 +00:00
|
|
|
id: generateUuid(),
|
2022-04-20 23:33:38 +00:00
|
|
|
body: 'message 4',
|
|
|
|
type: 'incoming',
|
|
|
|
conversationId,
|
|
|
|
sent_at: start + 4,
|
|
|
|
received_at: start + 4,
|
|
|
|
timestamp: start + 4,
|
|
|
|
readStatus: ReadStatus.Unread,
|
|
|
|
storyId,
|
|
|
|
};
|
|
|
|
|
|
|
|
await saveMessages([message1, message2, message3, message4], {
|
|
|
|
forceSave: true,
|
2023-08-10 16:43:33 +00:00
|
|
|
ourAci,
|
2022-04-20 23:33:38 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
assert.lengthOf(await _getAllMessages(), 4);
|
|
|
|
|
|
|
|
const markedRead = await getUnreadByConversationAndMarkRead({
|
|
|
|
conversationId,
|
2022-09-30 00:57:11 +00:00
|
|
|
includeStoryReplies: false,
|
2022-04-20 23:33:38 +00:00
|
|
|
newestUnreadAt: message4.received_at,
|
|
|
|
readAt,
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.lengthOf(markedRead, 1, '1 message marked read');
|
|
|
|
|
|
|
|
// Sorted in descending order
|
|
|
|
assert.strictEqual(
|
|
|
|
markedRead[0].id,
|
|
|
|
message3.id,
|
|
|
|
'first should be message3'
|
|
|
|
);
|
|
|
|
});
|
2021-12-08 19:52:46 +00:00
|
|
|
});
|