From 384a3c64e573ddfdb63b7ac3cc2ef061a5932c00 Mon Sep 17 00:00:00 2001 From: Andy Balaam Date: Thu, 25 May 2023 03:49:07 +0100 Subject: [PATCH 1/4] Add a test for creating local echo receipts in threads --- spec/unit/models/thread.spec.ts | 54 ++++++++++++++++++++++++++++++--- 1 file changed, 49 insertions(+), 5 deletions(-) diff --git a/spec/unit/models/thread.spec.ts b/spec/unit/models/thread.spec.ts index bd17f00c8aa..432c15bebd2 100644 --- a/spec/unit/models/thread.spec.ts +++ b/spec/unit/models/thread.spec.ts @@ -17,9 +17,9 @@ limitations under the License. import { mocked } from "jest-mock"; import { MatrixClient, PendingEventOrdering } from "../../../src/client"; -import { Room } from "../../../src/models/room"; +import { Room, RoomEvent } from "../../../src/models/room"; import { Thread, THREAD_RELATION_TYPE, ThreadEvent, FeatureSupport } from "../../../src/models/thread"; -import { mkThread } from "../../test-utils/thread"; +import { makeThreadEvent, mkThread } from "../../test-utils/thread"; import { TestClient } from "../../TestClient"; import { emitPromise, mkMessage, mkReaction, mock } from "../../test-utils/test-utils"; import { Direction, EventStatus, MatrixEvent } from "../../../src"; @@ -429,7 +429,7 @@ describe("Thread", () => { }); describe("insertEventIntoTimeline", () => { - it("Inserts a reply in timestamp order", () => { + it("Inserts a reaction in timestamp order", () => { // Assumption: no server side support because if we have it, events // can only be added to the timeline after the thread has been // initialised, and we are not properly initialising it here. @@ -449,11 +449,11 @@ describe("Thread", () => { ts: 100, // Events will be at ts 100, 101, 102, 103, 104 and 105 }); - // When we insert a reply to the second thread message + // When we insert a reaction to the second thread message const replyEvent = mkReaction(events[2], client, userId, room.roomId, 104); thread.insertEventIntoTimeline(replyEvent); - // Then the reply is inserted based on its timestamp + // Then the reaction is inserted based on its timestamp expect(thread.events.map((ev) => ev.getId())).toEqual([ events[0].getId(), events[1].getId(), @@ -465,6 +465,50 @@ describe("Thread", () => { ]); }); + it("Creates a local echo receipt for new events", async () => { + // Assumption: no server side support because if we have it, events + // can only be added to the timeline after the thread has been + // initialised, and we are not properly initialising it here. + expect(Thread.hasServerSideSupport).toBe(FeatureSupport.None); + + const client = createClientWithEventMapper(); + const userId = "user1"; + const room = new Room("room1", client, userId); + + // Given a thread + const { thread } = mkThread({ + room, + client, + authorId: userId, + participantUserIds: [], + ts: 1, + }); + + const awaitTimelineEvent = new Promise((res) => thread.on(RoomEvent.Timeline, () => res())); + + // When we add a message + const message = makeThreadEvent({ + event: true, + rootEventId: thread.id, + replyToEventId: thread.id, + user: userId, + room: room.roomId, + ts: 100, + }); + await thread.addEvent(message, false, true); + await awaitTimelineEvent; + + // Then a receipt was added to the thread + const receipt = thread.getReadReceiptForUserId(userId); + expect(receipt).toBeTruthy(); + expect(receipt?.eventId).toEqual(message.getId()); + expect(receipt?.data.ts).toEqual(100); + expect(receipt?.data.thread_id).toEqual(thread.id); + + // (And the receipt was synthetic) + expect(thread.getReadReceiptForUserId(userId, true)).toBeNull(); + }); + function createClientWithEventMapper(): MatrixClient { const client = mock(MatrixClient, "MatrixClient"); client.reEmitter = mock(ReEmitter, "ReEmitter"); From 37bb4c26341ed44989bae5df4252ff05cab1fca7 Mon Sep 17 00:00:00 2001 From: Andy Balaam Date: Thu, 25 May 2023 03:53:16 +0100 Subject: [PATCH 2/4] Only add local receipt if it's after existing receipt --- spec/unit/models/thread.spec.ts | 40 +++++++++++++++++++++++++++++++++ src/models/thread.ts | 24 +++++++++++++++++++- 2 files changed, 63 insertions(+), 1 deletion(-) diff --git a/spec/unit/models/thread.spec.ts b/spec/unit/models/thread.spec.ts index 432c15bebd2..1b1f39c0c61 100644 --- a/spec/unit/models/thread.spec.ts +++ b/spec/unit/models/thread.spec.ts @@ -509,6 +509,46 @@ describe("Thread", () => { expect(thread.getReadReceiptForUserId(userId, true)).toBeNull(); }); + it("Doesn't create a local echo receipt for events before an existing receipt", async () => { + // Assumption: no server side support because if we have it, events + // can only be added to the timeline after the thread has been + // initialised, and we are not properly initialising it here. + expect(Thread.hasServerSideSupport).toBe(FeatureSupport.None); + + const client = createClientWithEventMapper(); + const userId = "user1"; + const room = new Room("room1", client, userId); + + // Given a thread + const { thread } = mkThread({ + room, + client, + authorId: userId, + participantUserIds: [], + ts: 200, // Timestamp is after the event we add in a second + }); + // Sanity: the current receipt is for the thread root + expect(thread.getReadReceiptForUserId(userId)?.eventId).toEqual(thread.rootEvent?.getId()); + + const awaitTimelineEvent = new Promise((res) => thread.on(RoomEvent.Timeline, () => res())); + + // When we add a message that is before the latest receipt + const message = makeThreadEvent({ + event: true, + rootEventId: thread.id, + replyToEventId: thread.id, + user: userId, + room: room.roomId, + ts: 100, + }); + await thread.addEvent(message, false, true); + await awaitTimelineEvent; + + // Then no receipt was added to the thread (the receipt is still for + // the thread root). + expect(thread.getReadReceiptForUserId(userId)?.eventId).toEqual(thread.rootEvent?.getId()); + }); + function createClientWithEventMapper(): MatrixClient { const client = mock(MatrixClient, "MatrixClient"); client.reEmitter = mock(ReEmitter, "ReEmitter"); diff --git a/src/models/thread.ts b/src/models/thread.ts index f2420842dfc..d593ed3b105 100644 --- a/src/models/thread.ts +++ b/src/models/thread.ts @@ -203,11 +203,33 @@ export class Thread extends ReadReceipt { ): void => { // Add a synthesized receipt when paginating forward in the timeline if (!toStartOfTimeline) { - room!.addLocalEchoReceipt(event.getSender()!, event, ReceiptType.Read); + const sender = event.getSender(); + if (sender && room && this.shouldSendLocalEchoReceipt(sender, event)) { + room.addLocalEchoReceipt(sender, event, ReceiptType.Read); + } } this.onEcho(event, toStartOfTimeline ?? false); }; + private shouldSendLocalEchoReceipt(sender: string, event: MatrixEvent): boolean { + const recursionSupport = this.client.canSupport.get(Feature.RelationsRecursion) ?? ServerSupport.Unsupported; + + if (recursionSupport === ServerSupport.Unsupported) { + // Normally we add a local receipt, but if we don't have + // recursion support, then events may arrive out of order, so we + // only create a receipt if it's after our existing receipt. + const oldReceiptEventId = this.getReadReceiptForUserId(sender)?.eventId; + if (oldReceiptEventId) { + const receiptEvent = this.findEventById(oldReceiptEventId); + if (receiptEvent && receiptEvent.getTs() > event.getTs()) { + return false; + } + } + } + + return true; + } + private onLocalEcho = (event: MatrixEvent): void => { this.onEcho(event, false); }; From 4168003c5eb7323052543245cbd60a67b28499e9 Mon Sep 17 00:00:00 2001 From: Andy Balaam Date: Thu, 25 May 2023 04:06:30 +0100 Subject: [PATCH 3/4] Refactor local receipt tests to be shorter --- spec/unit/models/thread.spec.ts | 97 ++++++++++++++++----------------- 1 file changed, 46 insertions(+), 51 deletions(-) diff --git a/spec/unit/models/thread.spec.ts b/spec/unit/models/thread.spec.ts index 1b1f39c0c61..8b86cd66e15 100644 --- a/spec/unit/models/thread.spec.ts +++ b/spec/unit/models/thread.spec.ts @@ -465,58 +465,55 @@ describe("Thread", () => { ]); }); - it("Creates a local echo receipt for new events", async () => { - // Assumption: no server side support because if we have it, events - // can only be added to the timeline after the thread has been - // initialised, and we are not properly initialising it here. - expect(Thread.hasServerSideSupport).toBe(FeatureSupport.None); - - const client = createClientWithEventMapper(); - const userId = "user1"; - const room = new Room("room1", client, userId); - - // Given a thread - const { thread } = mkThread({ - room, - client, - authorId: userId, - participantUserIds: [], - ts: 1, + describe("Without relations recursion support", () => { + it("Creates a local echo receipt for new events", async () => { + // Assumption: no server side support because if we have it, events + // can only be added to the timeline after the thread has been + // initialised, and we are not properly initialising it here. + expect(Thread.hasServerSideSupport).toBe(FeatureSupport.None); + + // Given a client without relations recursion support + const client = createClientWithEventMapper(); + + // And a thread with an added event (with later timestamp) + const userId = "user1"; + const { thread, message } = await createThreadAndEvent(client, 1, 100, userId); + + // Then a receipt was added to the thread + const receipt = thread.getReadReceiptForUserId(userId); + expect(receipt).toBeTruthy(); + expect(receipt?.eventId).toEqual(message.getId()); + expect(receipt?.data.ts).toEqual(100); + expect(receipt?.data.thread_id).toEqual(thread.id); + + // (And the receipt was synthetic) + expect(thread.getReadReceiptForUserId(userId, true)).toBeNull(); }); - const awaitTimelineEvent = new Promise((res) => thread.on(RoomEvent.Timeline, () => res())); + it("Doesn't create a local echo receipt for events before an existing receipt", async () => { + // Assumption: no server side support because if we have it, events + // can only be added to the timeline after the thread has been + // initialised, and we are not properly initialising it here. + expect(Thread.hasServerSideSupport).toBe(FeatureSupport.None); - // When we add a message - const message = makeThreadEvent({ - event: true, - rootEventId: thread.id, - replyToEventId: thread.id, - user: userId, - room: room.roomId, - ts: 100, - }); - await thread.addEvent(message, false, true); - await awaitTimelineEvent; + // Given a client without relations recursion support + const client = createClientWithEventMapper(); - // Then a receipt was added to the thread - const receipt = thread.getReadReceiptForUserId(userId); - expect(receipt).toBeTruthy(); - expect(receipt?.eventId).toEqual(message.getId()); - expect(receipt?.data.ts).toEqual(100); - expect(receipt?.data.thread_id).toEqual(thread.id); + // And a thread with an added event with a lower timestamp than its other events + const userId = "user1"; + const { thread } = await createThreadAndEvent(client, 200, 100, userId); - // (And the receipt was synthetic) - expect(thread.getReadReceiptForUserId(userId, true)).toBeNull(); + // Then no receipt was added to the thread (the receipt is still for the thread root). + expect(thread.getReadReceiptForUserId(userId)?.eventId).toEqual(thread.rootEvent?.getId()); + }); }); - it("Doesn't create a local echo receipt for events before an existing receipt", async () => { - // Assumption: no server side support because if we have it, events - // can only be added to the timeline after the thread has been - // initialised, and we are not properly initialising it here. - expect(Thread.hasServerSideSupport).toBe(FeatureSupport.None); - - const client = createClientWithEventMapper(); - const userId = "user1"; + async function createThreadAndEvent( + client: MatrixClient, + rootTs: number, + eventTs: number, + userId: string, + ): Promise<{ thread: Thread; message: MatrixEvent }> { const room = new Room("room1", client, userId); // Given a thread @@ -525,7 +522,7 @@ describe("Thread", () => { client, authorId: userId, participantUserIds: [], - ts: 200, // Timestamp is after the event we add in a second + ts: rootTs, }); // Sanity: the current receipt is for the thread root expect(thread.getReadReceiptForUserId(userId)?.eventId).toEqual(thread.rootEvent?.getId()); @@ -539,15 +536,13 @@ describe("Thread", () => { replyToEventId: thread.id, user: userId, room: room.roomId, - ts: 100, + ts: eventTs, }); await thread.addEvent(message, false, true); await awaitTimelineEvent; - // Then no receipt was added to the thread (the receipt is still for - // the thread root). - expect(thread.getReadReceiptForUserId(userId)?.eventId).toEqual(thread.rootEvent?.getId()); - }); + return { thread, message }; + } function createClientWithEventMapper(): MatrixClient { const client = mock(MatrixClient, "MatrixClient"); From 68270273e2b128b95090f1bdcfe62ebf3ccb5a91 Mon Sep 17 00:00:00 2001 From: Andy Balaam Date: Thu, 25 May 2023 04:12:01 +0100 Subject: [PATCH 4/4] Tests for local receipts where we DO have recursive relations support --- spec/unit/models/thread.spec.ts | 54 +++++++++++++++++++++++++++++++-- 1 file changed, 51 insertions(+), 3 deletions(-) diff --git a/spec/unit/models/thread.spec.ts b/spec/unit/models/thread.spec.ts index 8b86cd66e15..bd3715bc68a 100644 --- a/spec/unit/models/thread.spec.ts +++ b/spec/unit/models/thread.spec.ts @@ -503,11 +503,59 @@ describe("Thread", () => { const userId = "user1"; const { thread } = await createThreadAndEvent(client, 200, 100, userId); - // Then no receipt was added to the thread (the receipt is still for the thread root). + // Then no receipt was added to the thread (the receipt is still + // for the thread root). This happens because since we have no + // recursive relations support, we know that sometimes events + // appear out of order, so we have to check their timestamps as + // a guess of the correct order. expect(thread.getReadReceiptForUserId(userId)?.eventId).toEqual(thread.rootEvent?.getId()); }); }); + describe("With relations recursion support", () => { + it("Creates a local echo receipt for new events", async () => { + // Assumption: no server side support because if we have it, events + // can only be added to the timeline after the thread has been + // initialised, and we are not properly initialising it here. + expect(Thread.hasServerSideSupport).toBe(FeatureSupport.None); + + // Given a client WITH relations recursion support + const client = createClientWithEventMapper( + new Map([[Feature.RelationsRecursion, ServerSupport.Stable]]), + ); + + // And a thread with an added event (with later timestamp) + const userId = "user1"; + const { thread, message } = await createThreadAndEvent(client, 1, 100, userId); + + // Then a receipt was added to the thread + const receipt = thread.getReadReceiptForUserId(userId); + expect(receipt?.eventId).toEqual(message.getId()); + }); + + it("Creates a local echo receipt even for events BEFORE an existing receipt", async () => { + // Assumption: no server side support because if we have it, events + // can only be added to the timeline after the thread has been + // initialised, and we are not properly initialising it here. + expect(Thread.hasServerSideSupport).toBe(FeatureSupport.None); + + // Given a client WITH relations recursion support + const client = createClientWithEventMapper( + new Map([[Feature.RelationsRecursion, ServerSupport.Stable]]), + ); + + // And a thread with an added event with a lower timestamp than its other events + const userId = "user1"; + const { thread, message } = await createThreadAndEvent(client, 200, 100, userId); + + // Then a receipt was added to the thread, because relations + // recursion is available, so we trust the server to have + // provided us with events in the right order. + const receipt = thread.getReadReceiptForUserId(userId); + expect(receipt?.eventId).toEqual(message.getId()); + }); + }); + async function createThreadAndEvent( client: MatrixClient, rootTs: number, @@ -544,10 +592,10 @@ describe("Thread", () => { return { thread, message }; } - function createClientWithEventMapper(): MatrixClient { + function createClientWithEventMapper(canSupport: Map = new Map()): MatrixClient { const client = mock(MatrixClient, "MatrixClient"); client.reEmitter = mock(ReEmitter, "ReEmitter"); - client.canSupport = new Map(); + client.canSupport = canSupport; jest.spyOn(client, "getEventMapper").mockReturnValue(eventMapperFor(client, {})); mocked(client.supportsThreads).mockReturnValue(true); return client;