From dc364aa8b917b2eccaa90fe282b7e222b9385566 Mon Sep 17 00:00:00 2001 From: Andy Balaam Date: Thu, 18 May 2023 17:08:54 +0100 Subject: [PATCH 1/6] Attempt a potential workaround for stuck notifs --- src/models/event-timeline-set.ts | 93 ++++++++++++++++++++++++++++++++ src/models/event-timeline.ts | 37 +++++++++++++ src/models/thread.ts | 30 ++++++++++- 3 files changed, 159 insertions(+), 1 deletion(-) diff --git a/src/models/event-timeline-set.ts b/src/models/event-timeline-set.ts index 5cb04997e8b..9df897b182c 100644 --- a/src/models/event-timeline-set.ts +++ b/src/models/event-timeline-set.ts @@ -756,6 +756,99 @@ export class EventTimelineSet extends TypedEventEmitter event.getTs()) { + // We found an event later than ours, so insert before that. + break; + } + } + // If we got to the end of the loop, insertIndex points at the end of + // the list. + + const eventId = event.getId()!; + timeline.insertEvent(event, insertIndex, roomState); + this._eventIdToTimeline.set(eventId, timeline); + + this.relations.aggregateParentEvent(event); + this.relations.aggregateChildEvent(event, this); + + const data: IRoomTimelineData = { + timeline: timeline, + liveEvent: timeline == this.liveTimeline, + }; + this.emit(RoomEvent.Timeline, event, this.room, false, false, data); + } + /** * Replaces event with ID oldEventId with one with newEventId, if oldEventId is * recognised. Otherwise, add to the live timeline. Used to handle remote echos. diff --git a/src/models/event-timeline.ts b/src/models/event-timeline.ts index d1ba3210365..24f63a99965 100644 --- a/src/models/event-timeline.ts +++ b/src/models/event-timeline.ts @@ -427,6 +427,43 @@ export class EventTimeline { } } + /** + * Insert a new event into the timeline, and update the state. + * + * TEMPORARY: until we have recursive relations, we need this function + * to exist to allow us to insert events in timeline order, which is our + * best guess for Sync Order. + * This is a copy of addEvent above, modified to allow inserting an event at + * a specific index. + */ + public insertEvent(event: MatrixEvent, insertIndex: number, roomState: RoomState): void { + const timelineSet = this.getTimelineSet(); + + if (timelineSet.room) { + EventTimeline.setEventMetadata(event, roomState, false); + + // modify state but only on unfiltered timelineSets + if (event.isState() && timelineSet.room.getUnfilteredTimelineSet() === timelineSet) { + roomState.setStateEvents([event], {}); + // it is possible that the act of setting the state event means we + // can set more metadata (specifically sender/target props), so try + // it again if the prop wasn't previously set. It may also mean that + // the sender/target is updated (if the event set was a room member event) + // so we want to use the *updated* member (new avatar/name) instead. + // + // However, we do NOT want to do this on member events if we're going + // back in time, else we'll set the .sender value for BEFORE the given + // member event, whereas we want to set the .sender value for the ACTUAL + // member event itself. + if (!event.sender || event.getType() === EventType.RoomMember) { + EventTimeline.setEventMetadata(event, roomState!, false); + } + } + } + + this.events.splice(insertIndex, 0, event); // insert element + } + /** * Remove an event from the timeline * diff --git a/src/models/thread.ts b/src/models/thread.ts index c31499b2b34..fb7d84d98c1 100644 --- a/src/models/thread.ts +++ b/src/models/thread.ts @@ -236,6 +236,31 @@ export class Thread extends ReadReceipt { } } + /** + * TEMPORARY. Only call this when MSC3981 is not available, and we have some + * late-arriving events to insert, because we recursively found them as part + * of populating a thread. When we have MSC3981 we won't need it, because + * they will all be supplied by the homeserver in one request, and they will + * already be in the right order in that response. + * This is a copy of addEventToTimeline above, modified to call + * insertEventIntoTimeline so this event is inserted into our best guess of + * the right place based on timestamp. (We should be using Sync Order but we + * don't have it.) + */ + public insertEventIntoTimeline(event: MatrixEvent): void { + const eventId = event.getId(); + if (!eventId) { + return; + } + if (this.findEventById(eventId)) { + return; + } + this.timelineSet.insertEventIntoTimeline(event, this.liveTimeline, this.roomState); + + // As far as we know, timeline should always be the same as events + this.timeline = this.events; + } + public addEvents(events: MatrixEvent[], toStartOfTimeline: boolean): void { events.forEach((ev) => this.addEvent(ev, toStartOfTimeline, false)); this.updateThreadMetadata(); @@ -281,7 +306,8 @@ export class Thread extends ReadReceipt { */ this.replayEvents?.push(event); } else { - this.addEventToTimeline(event, toStartOfTimeline); + // TODO: check with Germain is this right? + this.insertEventIntoTimeline(event); } // Apply annotations and replace relations to the relations of the timeline only this.timelineSet.relations?.aggregateParentEvent(event); @@ -473,6 +499,8 @@ export class Thread extends ReadReceipt { .then((relations) => { if (relations.events.length) { event.makeReplaced(relations.events[0]); + // TODO: check with Germain: is this right? + this.insertEventIntoTimeline(event); } }) .catch((e) => { From a1bece1c5d13c8b4226297c7808de1dc1136e1de Mon Sep 17 00:00:00 2001 From: Andy Balaam Date: Fri, 19 May 2023 09:22:29 +0100 Subject: [PATCH 2/6] Remove TODOs --- src/models/thread.ts | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/models/thread.ts b/src/models/thread.ts index fb7d84d98c1..9ab21da5e81 100644 --- a/src/models/thread.ts +++ b/src/models/thread.ts @@ -306,7 +306,6 @@ export class Thread extends ReadReceipt { */ this.replayEvents?.push(event); } else { - // TODO: check with Germain is this right? this.insertEventIntoTimeline(event); } // Apply annotations and replace relations to the relations of the timeline only @@ -499,7 +498,6 @@ export class Thread extends ReadReceipt { .then((relations) => { if (relations.events.length) { event.makeReplaced(relations.events[0]); - // TODO: check with Germain: is this right? this.insertEventIntoTimeline(event); } }) From 5e93e11c7f5b650e7c6284e44f5a85ba58efae4f Mon Sep 17 00:00:00 2001 From: Andy Balaam Date: Fri, 19 May 2023 10:55:53 +0100 Subject: [PATCH 3/6] Fix backwards logic about server support for MSC3981 in fetchEditsWhereNeeded --- src/models/thread.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/models/thread.ts b/src/models/thread.ts index 9ab21da5e81..2db66801c61 100644 --- a/src/models/thread.ts +++ b/src/models/thread.ts @@ -485,7 +485,7 @@ export class Thread extends ReadReceipt { // XXX: Workaround for https://github.com/matrix-org/matrix-spec-proposals/pull/2676/files#r827240084 private async fetchEditsWhereNeeded(...events: MatrixEvent[]): Promise { const recursionSupport = this.client.canSupport.get(Feature.RelationsRecursion) ?? ServerSupport.Unsupported; - if (recursionSupport !== ServerSupport.Unsupported) { + if (recursionSupport === ServerSupport.Unsupported) { return Promise.all( events .filter((e) => e.isEncrypted()) From 389b7c50839fdac16e344802e508052aeefe11bf Mon Sep 17 00:00:00 2001 From: Andy Balaam Date: Fri, 19 May 2023 10:56:22 +0100 Subject: [PATCH 4/6] Check for lack of MSC3981 server support before calling insertEventIntoTimeline --- src/models/thread.ts | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/src/models/thread.ts b/src/models/thread.ts index 2db66801c61..27f54e22b85 100644 --- a/src/models/thread.ts +++ b/src/models/thread.ts @@ -306,7 +306,14 @@ export class Thread extends ReadReceipt { */ this.replayEvents?.push(event); } else { - this.insertEventIntoTimeline(event); + const recursionSupport = + this.client.canSupport.get(Feature.RelationsRecursion) ?? ServerSupport.Unsupported; + + if (recursionSupport === ServerSupport.Unsupported) { + this.insertEventIntoTimeline(event); + } else { + this.addEventToTimeline(event, toStartOfTimeline); + } } // Apply annotations and replace relations to the relations of the timeline only this.timelineSet.relations?.aggregateParentEvent(event); From 1aee0a84a98b61da5901f71775b5791062fb5f20 Mon Sep 17 00:00:00 2001 From: Andy Balaam Date: Fri, 19 May 2023 12:11:59 +0100 Subject: [PATCH 5/6] If no parent event is found, insert purely based on timestamp --- src/models/event-timeline-set.ts | 15 ++++----------- 1 file changed, 4 insertions(+), 11 deletions(-) diff --git a/src/models/event-timeline-set.ts b/src/models/event-timeline-set.ts index 9df897b182c..4fab9e5da0d 100644 --- a/src/models/event-timeline-set.ts +++ b/src/models/event-timeline-set.ts @@ -811,19 +811,12 @@ export class EventTimelineSet extends TypedEventEmitter Date: Fri, 19 May 2023 12:23:30 +0100 Subject: [PATCH 6/6] Mark temporary methods as internal --- src/models/event-timeline-set.ts | 2 ++ src/models/event-timeline.ts | 2 ++ src/models/thread.ts | 2 ++ 3 files changed, 6 insertions(+) diff --git a/src/models/event-timeline-set.ts b/src/models/event-timeline-set.ts index 4fab9e5da0d..dfe9694c2ed 100644 --- a/src/models/event-timeline-set.ts +++ b/src/models/event-timeline-set.ts @@ -769,6 +769,8 @@ export class EventTimelineSet extends TypedEventEmitter { * insertEventIntoTimeline so this event is inserted into our best guess of * the right place based on timestamp. (We should be using Sync Order but we * don't have it.) + * + * @internal */ public insertEventIntoTimeline(event: MatrixEvent): void { const eventId = event.getId();