@unwanted/matrix-sdk-mini 34.12.0-7 → 34.12.0-9

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (64) hide show
  1. package/git-revision.txt +1 -1
  2. package/lib/autodiscovery.js +3 -3
  3. package/lib/autodiscovery.js.map +1 -1
  4. package/lib/client.d.ts.map +1 -1
  5. package/lib/client.js +21 -21
  6. package/lib/client.js.map +1 -1
  7. package/lib/content-repo.js +2 -2
  8. package/lib/content-repo.js.map +1 -1
  9. package/lib/embedded.d.ts.map +1 -1
  10. package/lib/embedded.js +35 -2
  11. package/lib/embedded.js.map +1 -1
  12. package/lib/http-api/prefix.d.ts +6 -7
  13. package/lib/http-api/prefix.d.ts.map +1 -1
  14. package/lib/http-api/prefix.js +6 -7
  15. package/lib/http-api/prefix.js.map +1 -1
  16. package/lib/models/event-timeline-set.d.ts +6 -10
  17. package/lib/models/event-timeline-set.d.ts.map +1 -1
  18. package/lib/models/event-timeline-set.js +28 -36
  19. package/lib/models/event-timeline-set.js.map +1 -1
  20. package/lib/models/event-timeline.d.ts +7 -2
  21. package/lib/models/event-timeline.d.ts.map +1 -1
  22. package/lib/models/event-timeline.js +8 -9
  23. package/lib/models/event-timeline.js.map +1 -1
  24. package/lib/models/event.d.ts.map +1 -1
  25. package/lib/models/event.js +1 -1
  26. package/lib/models/event.js.map +1 -1
  27. package/lib/models/room-state.d.ts +2 -13
  28. package/lib/models/room-state.d.ts.map +1 -1
  29. package/lib/models/room-state.js +2 -30
  30. package/lib/models/room-state.js.map +1 -1
  31. package/lib/models/room.d.ts +2 -2
  32. package/lib/models/room.d.ts.map +1 -1
  33. package/lib/models/room.js +27 -14
  34. package/lib/models/room.js.map +1 -1
  35. package/lib/models/thread.d.ts.map +1 -1
  36. package/lib/models/thread.js +5 -3
  37. package/lib/models/thread.js.map +1 -1
  38. package/lib/sliding-sync-sdk.d.ts +1 -1
  39. package/lib/sliding-sync-sdk.d.ts.map +1 -1
  40. package/lib/sliding-sync-sdk.js +15 -13
  41. package/lib/sliding-sync-sdk.js.map +1 -1
  42. package/lib/sync-accumulator.d.ts +6 -4
  43. package/lib/sync-accumulator.d.ts.map +1 -1
  44. package/lib/sync-accumulator.js +23 -12
  45. package/lib/sync-accumulator.js.map +1 -1
  46. package/lib/sync.d.ts +10 -1
  47. package/lib/sync.d.ts.map +1 -1
  48. package/lib/sync.js +95 -44
  49. package/lib/sync.js.map +1 -1
  50. package/package.json +7 -7
  51. package/src/autodiscovery.ts +3 -3
  52. package/src/client.ts +20 -21
  53. package/src/content-repo.ts +2 -2
  54. package/src/embedded.ts +35 -2
  55. package/src/http-api/prefix.ts +6 -8
  56. package/src/models/event-timeline-set.ts +17 -38
  57. package/src/models/event-timeline.ts +10 -5
  58. package/src/models/event.ts +3 -1
  59. package/src/models/room-state.ts +2 -29
  60. package/src/models/room.ts +18 -6
  61. package/src/models/thread.ts +4 -2
  62. package/src/sliding-sync-sdk.ts +8 -11
  63. package/src/sync-accumulator.ts +33 -16
  64. package/src/sync.ts +113 -47
@@ -1692,10 +1692,11 @@ export class Room extends ReadReceipt<RoomEmittedEvents, RoomEventHandlerMap> {
1692
1692
  public addEventsToTimeline(
1693
1693
  events: MatrixEvent[],
1694
1694
  toStartOfTimeline: boolean,
1695
+ addToState: boolean,
1695
1696
  timeline: EventTimeline,
1696
1697
  paginationToken?: string,
1697
1698
  ): void {
1698
- timeline.getTimelineSet().addEventsToTimeline(events, toStartOfTimeline, timeline, paginationToken);
1699
+ timeline.getTimelineSet().addEventsToTimeline(events, toStartOfTimeline, addToState, timeline, paginationToken);
1699
1700
  }
1700
1701
 
1701
1702
  /**
@@ -1860,7 +1861,7 @@ export class Room extends ReadReceipt<RoomEmittedEvents, RoomEventHandlerMap> {
1860
1861
  // see https://github.com/vector-im/vector-web/issues/2109
1861
1862
 
1862
1863
  unfilteredLiveTimeline.getEvents().forEach(function (event) {
1863
- timelineSet.addLiveEvent(event);
1864
+ timelineSet.addLiveEvent(event, { addToState: false }); // Filtered timeline sets should not track state
1864
1865
  });
1865
1866
 
1866
1867
  // find the earliest unfiltered timeline
@@ -1947,6 +1948,7 @@ export class Room extends ReadReceipt<RoomEmittedEvents, RoomEventHandlerMap> {
1947
1948
  if (filterType !== ThreadFilterType.My || currentUserParticipated) {
1948
1949
  timelineSet.getLiveTimeline().addEvent(thread.rootEvent!, {
1949
1950
  toStartOfTimeline: false,
1951
+ addToState: false,
1950
1952
  });
1951
1953
  }
1952
1954
  });
@@ -2020,6 +2022,7 @@ export class Room extends ReadReceipt<RoomEmittedEvents, RoomEventHandlerMap> {
2020
2022
  const opts = {
2021
2023
  duplicateStrategy: DuplicateStrategy.Ignore,
2022
2024
  fromCache: false,
2025
+ addToState: false,
2023
2026
  roomState,
2024
2027
  };
2025
2028
  this.threadsTimelineSets[0]?.addLiveEvent(rootEvent, opts);
@@ -2128,6 +2131,7 @@ export class Room extends ReadReceipt<RoomEmittedEvents, RoomEventHandlerMap> {
2128
2131
  duplicateStrategy: DuplicateStrategy.Replace,
2129
2132
  fromCache: false,
2130
2133
  roomState,
2134
+ addToState: false,
2131
2135
  });
2132
2136
  }
2133
2137
  }
@@ -2319,9 +2323,13 @@ export class Room extends ReadReceipt<RoomEmittedEvents, RoomEventHandlerMap> {
2319
2323
  duplicateStrategy: DuplicateStrategy.Replace,
2320
2324
  fromCache: false,
2321
2325
  roomState: this.currentState,
2326
+ addToState: false,
2322
2327
  });
2323
2328
  } else {
2324
- timelineSet.addEventToTimeline(thread.rootEvent, timelineSet.getLiveTimeline(), { toStartOfTimeline });
2329
+ timelineSet.addEventToTimeline(thread.rootEvent, timelineSet.getLiveTimeline(), {
2330
+ toStartOfTimeline,
2331
+ addToState: false,
2332
+ });
2325
2333
  }
2326
2334
  }
2327
2335
  };
@@ -2478,7 +2486,7 @@ export class Room extends ReadReceipt<RoomEmittedEvents, RoomEventHandlerMap> {
2478
2486
  * Fires {@link RoomEvent.Timeline}
2479
2487
  */
2480
2488
  private addLiveEvent(event: MatrixEvent, addLiveEventOptions: IAddLiveEventOptions): void {
2481
- const { duplicateStrategy, timelineWasEmpty, fromCache } = addLiveEventOptions;
2489
+ const { duplicateStrategy, timelineWasEmpty, fromCache, addToState } = addLiveEventOptions;
2482
2490
 
2483
2491
  // add to our timeline sets
2484
2492
  for (const timelineSet of this.timelineSets) {
@@ -2486,6 +2494,7 @@ export class Room extends ReadReceipt<RoomEmittedEvents, RoomEventHandlerMap> {
2486
2494
  duplicateStrategy,
2487
2495
  fromCache,
2488
2496
  timelineWasEmpty,
2497
+ addToState,
2489
2498
  });
2490
2499
  }
2491
2500
 
@@ -2569,11 +2578,13 @@ export class Room extends ReadReceipt<RoomEmittedEvents, RoomEventHandlerMap> {
2569
2578
  if (timelineSet.getFilter()!.filterRoomTimeline([event]).length) {
2570
2579
  timelineSet.addEventToTimeline(event, timelineSet.getLiveTimeline(), {
2571
2580
  toStartOfTimeline: false,
2581
+ addToState: false, // We don't support localEcho of state events yet
2572
2582
  });
2573
2583
  }
2574
2584
  } else {
2575
2585
  timelineSet.addEventToTimeline(event, timelineSet.getLiveTimeline(), {
2576
2586
  toStartOfTimeline: false,
2587
+ addToState: false, // We don't support localEcho of state events yet
2577
2588
  });
2578
2589
  }
2579
2590
  }
@@ -2824,8 +2835,8 @@ export class Room extends ReadReceipt<RoomEmittedEvents, RoomEventHandlerMap> {
2824
2835
  * @param addLiveEventOptions - addLiveEvent options
2825
2836
  * @throws If `duplicateStrategy` is not falsey, 'replace' or 'ignore'.
2826
2837
  */
2827
- public async addLiveEvents(events: MatrixEvent[], addLiveEventOptions?: IAddLiveEventOptions): Promise<void> {
2828
- const { duplicateStrategy, fromCache, timelineWasEmpty = false } = addLiveEventOptions ?? {};
2838
+ public async addLiveEvents(events: MatrixEvent[], addLiveEventOptions: IAddLiveEventOptions): Promise<void> {
2839
+ const { duplicateStrategy, fromCache, timelineWasEmpty = false, addToState } = addLiveEventOptions;
2829
2840
  if (duplicateStrategy && ["replace", "ignore"].indexOf(duplicateStrategy) === -1) {
2830
2841
  throw new Error("duplicateStrategy MUST be either 'replace' or 'ignore'");
2831
2842
  }
@@ -2840,6 +2851,7 @@ export class Room extends ReadReceipt<RoomEmittedEvents, RoomEventHandlerMap> {
2840
2851
  duplicateStrategy,
2841
2852
  fromCache,
2842
2853
  timelineWasEmpty,
2854
+ addToState,
2843
2855
  };
2844
2856
 
2845
2857
  // List of extra events to check for being parents of any relations encountered
@@ -208,6 +208,7 @@ export class Thread extends ReadReceipt<ThreadEmittedEvents, ThreadEventHandlerM
208
208
 
209
209
  public static setServerSideSupport(status: FeatureSupport): void {
210
210
  Thread.hasServerSideSupport = status;
211
+ // XXX: This global latching behaviour is really unexpected and means that you can't undo when moving to a server without support
211
212
  if (status !== FeatureSupport.Stable) {
212
213
  FILTER_RELATED_BY_SENDERS.setPreferUnstable(true);
213
214
  FILTER_RELATED_BY_REL_TYPES.setPreferUnstable(true);
@@ -317,6 +318,7 @@ export class Thread extends ReadReceipt<ThreadEmittedEvents, ThreadEventHandlerM
317
318
  toStartOfTimeline,
318
319
  fromCache: false,
319
320
  roomState: this.roomState,
321
+ addToState: false,
320
322
  });
321
323
  }
322
324
  }
@@ -343,7 +345,7 @@ export class Thread extends ReadReceipt<ThreadEmittedEvents, ThreadEventHandlerM
343
345
  if (this.findEventById(eventId)) {
344
346
  return;
345
347
  }
346
- this.timelineSet.insertEventIntoTimeline(event, this.liveTimeline, this.roomState);
348
+ this.timelineSet.insertEventIntoTimeline(event, this.liveTimeline, this.roomState, false);
347
349
  }
348
350
 
349
351
  public addEvents(events: MatrixEvent[], toStartOfTimeline: boolean): void {
@@ -617,7 +619,7 @@ export class Thread extends ReadReceipt<ThreadEmittedEvents, ThreadEventHandlerM
617
619
  // if the thread has regular events, this will just load the last reply.
618
620
  // if the thread is newly created, this will load the root event.
619
621
  if (this.replyCount === 0 && this.rootEvent) {
620
- this.timelineSet.addEventsToTimeline([this.rootEvent], true, this.liveTimeline, null);
622
+ this.timelineSet.addEventsToTimeline([this.rootEvent], true, false, this.liveTimeline, null);
621
623
  this.liveTimeline.setPaginationToken(null, Direction.Backward);
622
624
  } else {
623
625
  this.initalEventFetchProm = this.client.paginateEventTimeline(this.liveTimeline, {
@@ -550,7 +550,7 @@ export class SlidingSyncSdk {
550
550
  timelineEvents = newEvents;
551
551
  if (oldEvents.length > 0) {
552
552
  // old events are scrollback, insert them now
553
- room.addEventsToTimeline(oldEvents, true, room.getLiveTimeline(), roomData.prev_batch);
553
+ room.addEventsToTimeline(oldEvents, true, false, room.getLiveTimeline(), roomData.prev_batch);
554
554
  }
555
555
  }
556
556
 
@@ -684,7 +684,7 @@ export class SlidingSyncSdk {
684
684
  /**
685
685
  * Injects events into a room's model.
686
686
  * @param stateEventList - A list of state events. This is the state
687
- * at the *START* of the timeline list if it is supplied.
687
+ * at the *END* of the timeline list if it is supplied.
688
688
  * @param timelineEventList - A list of timeline events. Lower index
689
689
  * is earlier in time. Higher index is later.
690
690
  * @param numLive - the number of events in timelineEventList which just happened,
@@ -693,13 +693,9 @@ export class SlidingSyncSdk {
693
693
  public async injectRoomEvents(
694
694
  room: Room,
695
695
  stateEventList: MatrixEvent[],
696
- timelineEventList?: MatrixEvent[],
697
- numLive?: number,
696
+ timelineEventList: MatrixEvent[] = [],
697
+ numLive: number = 0,
698
698
  ): Promise<void> {
699
- timelineEventList = timelineEventList || [];
700
- stateEventList = stateEventList || [];
701
- numLive = numLive || 0;
702
-
703
699
  // If there are no events in the timeline yet, initialise it with
704
700
  // the given state events
705
701
  const liveTimeline = room.getLiveTimeline();
@@ -750,16 +746,17 @@ export class SlidingSyncSdk {
750
746
  timelineEventList = timelineEventList.slice(0, -1 * liveTimelineEvents.length);
751
747
  }
752
748
 
753
- // execute the timeline events. This will continue to diverge the current state
754
- // if the timeline has any state events in it.
749
+ // Execute the timeline events.
755
750
  // This also needs to be done before running push rules on the events as they need
756
751
  // to be decorated with sender etc.
757
752
  await room.addLiveEvents(timelineEventList, {
758
753
  fromCache: true,
754
+ addToState: false,
759
755
  });
760
756
  if (liveTimelineEvents.length > 0) {
761
757
  await room.addLiveEvents(liveTimelineEvents, {
762
758
  fromCache: false,
759
+ addToState: false,
763
760
  });
764
761
  }
765
762
 
@@ -896,7 +893,7 @@ export class SlidingSyncSdk {
896
893
  return a.getTs() - b.getTs();
897
894
  });
898
895
  this.notifEvents.forEach((event) => {
899
- this.client.getNotifTimelineSet()?.addLiveEvent(event);
896
+ this.client.getNotifTimelineSet()?.addLiveEvent(event, { addToState: false });
900
897
  });
901
898
  this.notifEvents = [];
902
899
  }
@@ -77,7 +77,9 @@ export interface ITimeline {
77
77
 
78
78
  export interface IJoinedRoom {
79
79
  "summary": IRoomSummary;
80
- "state": IState;
80
+ // One of `state` or `state_after` is required.
81
+ "state"?: IState;
82
+ "org.matrix.msc4222.state_after"?: IState; // https://github.com/matrix-org/matrix-spec-proposals/pull/4222
81
83
  "timeline": ITimeline;
82
84
  "ephemeral": IEphemeral;
83
85
  "account_data": IAccountData;
@@ -106,9 +108,11 @@ export interface IInvitedRoom {
106
108
  }
107
109
 
108
110
  export interface ILeftRoom {
109
- state: IState;
110
- timeline: ITimeline;
111
- account_data: IAccountData;
111
+ // One of `state` or `state_after` is required.
112
+ "state"?: IState;
113
+ "org.matrix.msc4222.state_after"?: IState;
114
+ "timeline": ITimeline;
115
+ "account_data": IAccountData;
112
116
  }
113
117
 
114
118
  export interface IKnockedRoom {
@@ -481,13 +485,18 @@ export class SyncAccumulator {
481
485
  // Work out the current state. The deltas need to be applied in the order:
482
486
  // - existing state which didn't come down /sync.
483
487
  // - State events under the 'state' key.
484
- // - State events in the 'timeline'.
488
+ // - State events under the 'state_after' key OR state events in the 'timeline' if 'state_after' is not present.
485
489
  data.state?.events?.forEach((e) => {
486
490
  setState(currentData._currentState, e);
487
491
  });
488
- data.timeline?.events?.forEach((e, index) => {
489
- // this nops if 'e' isn't a state event
492
+ data["org.matrix.msc4222.state_after"]?.events?.forEach((e) => {
490
493
  setState(currentData._currentState, e);
494
+ });
495
+ data.timeline?.events?.forEach((e, index) => {
496
+ if (!data["org.matrix.msc4222.state_after"]) {
497
+ // this nops if 'e' isn't a state event
498
+ setState(currentData._currentState, e);
499
+ }
491
500
  // append the event to the timeline. The back-pagination token
492
501
  // corresponds to the first event in the timeline
493
502
  let transformedEvent: TaggedEvent;
@@ -563,17 +572,22 @@ export class SyncAccumulator {
563
572
  });
564
573
  Object.keys(this.joinRooms).forEach((roomId) => {
565
574
  const roomData = this.joinRooms[roomId];
566
- const roomJson: IJoinedRoom = {
567
- ephemeral: { events: [] },
568
- account_data: { events: [] },
569
- state: { events: [] },
570
- timeline: {
575
+ const roomJson: IJoinedRoom & {
576
+ // We track both `state` and `state_after` for downgrade compatibility
577
+ "state": IState;
578
+ "org.matrix.msc4222.state_after": IState;
579
+ } = {
580
+ "ephemeral": { events: [] },
581
+ "account_data": { events: [] },
582
+ "state": { events: [] },
583
+ "org.matrix.msc4222.state_after": { events: [] },
584
+ "timeline": {
571
585
  events: [],
572
586
  prev_batch: null,
573
587
  },
574
- unread_notifications: roomData._unreadNotifications,
575
- unread_thread_notifications: roomData._unreadThreadNotifications,
576
- summary: roomData._summary as IRoomSummary,
588
+ "unread_notifications": roomData._unreadNotifications,
589
+ "unread_thread_notifications": roomData._unreadThreadNotifications,
590
+ "summary": roomData._summary as IRoomSummary,
577
591
  };
578
592
  // Add account data
579
593
  Object.keys(roomData._accountData).forEach((evType) => {
@@ -650,8 +664,11 @@ export class SyncAccumulator {
650
664
  Object.keys(roomData._currentState).forEach((evType) => {
651
665
  Object.keys(roomData._currentState[evType]).forEach((stateKey) => {
652
666
  let ev = roomData._currentState[evType][stateKey];
667
+ // Push to both fields to provide downgrade compatibility in the sync accumulator db
668
+ // the code will prefer `state_after` if it is present
669
+ roomJson["org.matrix.msc4222.state_after"].events.push(ev);
670
+ // Roll the state back to the value at the start of the timeline if it was changed
653
671
  if (rollBackState[evType] && rollBackState[evType][stateKey]) {
654
- // use the reverse clobbered event instead.
655
672
  ev = rollBackState[evType][stateKey];
656
673
  }
657
674
  roomJson.state.events.push(ev);
package/src/sync.ts CHANGED
@@ -48,7 +48,7 @@ import {
48
48
  IToDeviceEvent,
49
49
  } from "./sync-accumulator.ts";
50
50
  import { MatrixEvent } from "./models/event.ts";
51
- import { MatrixError, Method, VersionsPrefix } from "./http-api/index.ts";
51
+ import { MatrixError, Method } from "./http-api/index.ts";
52
52
  import { ISavedSync } from "./store/index.ts";
53
53
  import { EventType } from "./@types/event.ts";
54
54
  import { IPushRules } from "./@types/PushRules.ts";
@@ -161,14 +161,15 @@ export enum SetPresence {
161
161
  }
162
162
 
163
163
  interface ISyncParams {
164
- filter?: string;
165
- timeout: number;
166
- since?: string;
164
+ "filter"?: string;
165
+ "timeout": number;
166
+ "since"?: string;
167
167
  // eslint-disable-next-line camelcase
168
- full_state?: boolean;
168
+ "full_state"?: boolean;
169
169
  // eslint-disable-next-line camelcase
170
- set_presence?: SetPresence;
171
- _cacheBuster?: string | number; // not part of the API itself
170
+ "set_presence"?: SetPresence;
171
+ "_cacheBuster"?: string | number; // not part of the API itself
172
+ "org.matrix.msc4222.use_state_after"?: boolean; // https://github.com/matrix-org/matrix-spec-proposals/pull/4222
172
173
  }
173
174
 
174
175
  type WrappedRoom<T> = T & {
@@ -330,8 +331,9 @@ export class SyncApi {
330
331
  );
331
332
 
332
333
  const qps: ISyncParams = {
333
- timeout: 0, // don't want to block since this is a single isolated req
334
- filter: filterId,
334
+ "timeout": 0, // don't want to block since this is a single isolated req
335
+ "filter": filterId,
336
+ "org.matrix.msc4222.use_state_after": true,
335
337
  };
336
338
 
337
339
  const data = await client.http.authedRequest<ISyncResponse>(Method.Get, "/sync", qps as any, undefined, {
@@ -361,21 +363,17 @@ export class SyncApi {
361
363
  prev_batch: null,
362
364
  events: [],
363
365
  };
364
- const events = this.mapSyncEventsFormat(leaveObj.timeline, room);
365
-
366
- const stateEvents = this.mapSyncEventsFormat(leaveObj.state, room);
367
-
368
366
  // set the back-pagination token. Do this *before* adding any
369
367
  // events so that clients can start back-paginating.
370
368
  room.getLiveTimeline().setPaginationToken(leaveObj.timeline.prev_batch, EventTimeline.BACKWARDS);
371
369
 
372
- await this.injectRoomEvents(room, stateEvents, events);
370
+ const { timelineEvents } = await this.mapAndInjectRoomEvents(leaveObj);
373
371
 
374
372
  room.recalculate();
375
373
  client.store.storeRoom(room);
376
374
  client.emit(ClientEvent.Room, room);
377
375
 
378
- this.processEventsForNotifs(room, events);
376
+ this.processEventsForNotifs(room, timelineEvents);
379
377
  return room;
380
378
  }),
381
379
  );
@@ -450,6 +448,7 @@ export class SyncApi {
450
448
  this._peekRoom.addEventsToTimeline(
451
449
  messages.reverse(),
452
450
  true,
451
+ true,
453
452
  this._peekRoom.getLiveTimeline(),
454
453
  response.messages.start,
455
454
  );
@@ -537,7 +536,7 @@ export class SyncApi {
537
536
  })
538
537
  .map(this.client.getEventMapper());
539
538
 
540
- await peekRoom.addLiveEvents(events);
539
+ await peekRoom.addLiveEvents(events, { addToState: true });
541
540
  this.peekPoll(peekRoom, res.end);
542
541
  },
543
542
  (err) => {
@@ -938,7 +937,11 @@ export class SyncApi {
938
937
  filter = this.getGuestFilter();
939
938
  }
940
939
 
941
- const qps: ISyncParams = { filter, timeout };
940
+ const qps: ISyncParams = {
941
+ filter,
942
+ timeout,
943
+ "org.matrix.msc4222.use_state_after": true,
944
+ };
942
945
 
943
946
  if (this.opts.disablePresence) {
944
947
  qps.set_presence = SetPresence.Offline;
@@ -1200,7 +1203,7 @@ export class SyncApi {
1200
1203
  const room = inviteObj.room;
1201
1204
  const stateEvents = this.mapSyncEventsFormat(inviteObj.invite_state, room);
1202
1205
 
1203
- await this.injectRoomEvents(room, stateEvents);
1206
+ await this.injectRoomEvents(room, stateEvents, undefined);
1204
1207
 
1205
1208
  if (inviteObj.isBrandNewRoom) {
1206
1209
  room.recalculate();
@@ -1219,15 +1222,24 @@ export class SyncApi {
1219
1222
  await promiseMapSeries(joinRooms, async (joinObj) => {
1220
1223
  const room = joinObj.room;
1221
1224
  const stateEvents = this.mapSyncEventsFormat(joinObj.state, room);
1225
+ const stateAfterEvents = this.mapSyncEventsFormat(joinObj["org.matrix.msc4222.state_after"], room);
1222
1226
  // Prevent events from being decrypted ahead of time
1223
1227
  // this helps large account to speed up faster
1224
1228
  // room::decryptCriticalEvent is in charge of decrypting all the events
1225
1229
  // required for a client to function properly
1226
- const events = this.mapSyncEventsFormat(joinObj.timeline, room, false);
1230
+ const timelineEvents = this.mapSyncEventsFormat(joinObj.timeline, room, false);
1227
1231
  const ephemeralEvents = this.mapSyncEventsFormat(joinObj.ephemeral);
1228
1232
  const accountDataEvents = this.mapSyncEventsFormat(joinObj.account_data);
1229
1233
 
1230
- const encrypted = this.isRoomEncrypted(room, stateEvents, events);
1234
+ // If state_after is present, this is the events that form the state at the end of the timeline block and
1235
+ // regular timeline events do *not* count towards state. If it's not present, then the state is formed by
1236
+ // the state events plus the timeline events. Note mapSyncEventsFormat returns an empty array if the field
1237
+ // is absent so we explicitly check the field on the original object.
1238
+ const eventsFormingFinalState = joinObj["org.matrix.msc4222.state_after"]
1239
+ ? stateAfterEvents
1240
+ : stateEvents.concat(timelineEvents);
1241
+
1242
+ const encrypted = this.isRoomEncrypted(room, eventsFormingFinalState);
1231
1243
  // We store the server-provided value first so it's correct when any of the events fire.
1232
1244
  if (joinObj.unread_notifications) {
1233
1245
  /**
@@ -1315,8 +1327,8 @@ export class SyncApi {
1315
1327
  // which we'll try to paginate but not get any new events (which
1316
1328
  // will stop us linking the empty timeline into the chain).
1317
1329
  //
1318
- for (let i = events.length - 1; i >= 0; i--) {
1319
- const eventId = events[i].getId()!;
1330
+ for (let i = timelineEvents.length - 1; i >= 0; i--) {
1331
+ const eventId = timelineEvents[i].getId()!;
1320
1332
  if (room.getTimelineForEvent(eventId)) {
1321
1333
  debuglog(`Already have event ${eventId} in limited sync - not resetting`);
1322
1334
  limited = false;
@@ -1324,7 +1336,7 @@ export class SyncApi {
1324
1336
  // we might still be missing some of the events before i;
1325
1337
  // we don't want to be adding them to the end of the
1326
1338
  // timeline because that would put them out of order.
1327
- events.splice(0, i);
1339
+ timelineEvents.splice(0, i);
1328
1340
 
1329
1341
  // XXX: there's a problem here if the skipped part of the
1330
1342
  // timeline modifies the state set in stateEvents, because
@@ -1353,7 +1365,17 @@ export class SyncApi {
1353
1365
  }
1354
1366
 
1355
1367
  try {
1356
- await this.injectRoomEvents(room, stateEvents, events, syncEventData.fromCache);
1368
+ if ("org.matrix.msc4222.state_after" in joinObj) {
1369
+ await this.injectRoomEvents(
1370
+ room,
1371
+ undefined,
1372
+ stateAfterEvents,
1373
+ timelineEvents,
1374
+ syncEventData.fromCache,
1375
+ );
1376
+ } else {
1377
+ await this.injectRoomEvents(room, stateEvents, undefined, timelineEvents, syncEventData.fromCache);
1378
+ }
1357
1379
  } catch (e) {
1358
1380
  logger.error(`Failed to process events on room ${room.roomId}:`, e);
1359
1381
  }
@@ -1377,11 +1399,11 @@ export class SyncApi {
1377
1399
  client.emit(ClientEvent.Room, room);
1378
1400
  }
1379
1401
 
1380
- this.processEventsForNotifs(room, events);
1402
+ this.processEventsForNotifs(room, timelineEvents);
1381
1403
 
1382
1404
  const emitEvent = (e: MatrixEvent): boolean => client.emit(ClientEvent.Event, e);
1383
1405
  stateEvents.forEach(emitEvent);
1384
- events.forEach(emitEvent);
1406
+ timelineEvents.forEach(emitEvent);
1385
1407
  ephemeralEvents.forEach(emitEvent);
1386
1408
  accountDataEvents.forEach(emitEvent);
1387
1409
  });
@@ -1389,11 +1411,9 @@ export class SyncApi {
1389
1411
  // Handle leaves (e.g. kicked rooms)
1390
1412
  await promiseMapSeries(leaveRooms, async (leaveObj) => {
1391
1413
  const room = leaveObj.room;
1392
- const stateEvents = this.mapSyncEventsFormat(leaveObj.state, room);
1393
- const events = this.mapSyncEventsFormat(leaveObj.timeline, room);
1414
+ const { timelineEvents, stateEvents, stateAfterEvents } = await this.mapAndInjectRoomEvents(leaveObj);
1394
1415
  const accountDataEvents = this.mapSyncEventsFormat(leaveObj.account_data);
1395
1416
 
1396
- await this.injectRoomEvents(room, stateEvents, events);
1397
1417
  room.addAccountData(accountDataEvents);
1398
1418
 
1399
1419
  room.recalculate();
@@ -1402,12 +1422,15 @@ export class SyncApi {
1402
1422
  client.emit(ClientEvent.Room, room);
1403
1423
  }
1404
1424
 
1405
- this.processEventsForNotifs(room, events);
1425
+ this.processEventsForNotifs(room, timelineEvents);
1406
1426
 
1407
- stateEvents.forEach(function (e) {
1427
+ stateEvents?.forEach(function (e) {
1408
1428
  client.emit(ClientEvent.Event, e);
1409
1429
  });
1410
- events.forEach(function (e) {
1430
+ stateAfterEvents?.forEach(function (e) {
1431
+ client.emit(ClientEvent.Event, e);
1432
+ });
1433
+ timelineEvents.forEach(function (e) {
1411
1434
  client.emit(ClientEvent.Event, e);
1412
1435
  });
1413
1436
  accountDataEvents.forEach(function (e) {
@@ -1420,7 +1443,7 @@ export class SyncApi {
1420
1443
  const room = knockObj.room;
1421
1444
  const stateEvents = this.mapSyncEventsFormat(knockObj.knock_state, room);
1422
1445
 
1423
- await this.injectRoomEvents(room, stateEvents);
1446
+ await this.injectRoomEvents(room, stateEvents, undefined);
1424
1447
 
1425
1448
  if (knockObj.isBrandNewRoom) {
1426
1449
  room.recalculate();
@@ -1445,7 +1468,7 @@ export class SyncApi {
1445
1468
  return a.getTs() - b.getTs();
1446
1469
  });
1447
1470
  this.notifEvents.forEach(function (event) {
1448
- client.getNotifTimelineSet()?.addLiveEvent(event);
1471
+ client.getNotifTimelineSet()?.addLiveEvent(event, { addToState: true });
1449
1472
  });
1450
1473
  }
1451
1474
  }
@@ -1508,7 +1531,7 @@ export class SyncApi {
1508
1531
  this.client.http
1509
1532
  .request(
1510
1533
  Method.Get,
1511
- VersionsPrefix,
1534
+ "/_matrix/client/versions",
1512
1535
  undefined, // queryParams
1513
1536
  undefined, // data
1514
1537
  {
@@ -1572,7 +1595,7 @@ export class SyncApi {
1572
1595
  }
1573
1596
 
1574
1597
  private mapSyncEventsFormat(
1575
- obj: IInviteState | ITimeline | IEphemeral,
1598
+ obj: IInviteState | ITimeline | IEphemeral | undefined,
1576
1599
  room?: Room,
1577
1600
  decrypt = true,
1578
1601
  ): MatrixEvent[] {
@@ -1640,28 +1663,69 @@ export class SyncApi {
1640
1663
 
1641
1664
  // When processing the sync response we cannot rely on Room.hasEncryptionStateEvent we actually
1642
1665
  // inject the events into the room object, so we have to inspect the events themselves.
1643
- private isRoomEncrypted(room: Room, stateEventList: MatrixEvent[], timelineEventList?: MatrixEvent[]): boolean {
1644
- return (
1645
- room.hasEncryptionStateEvent() ||
1646
- !!this.findEncryptionEvent(stateEventList) ||
1647
- !!this.findEncryptionEvent(timelineEventList)
1666
+ private isRoomEncrypted(room: Room, eventsFormingFinalState: MatrixEvent[]): boolean {
1667
+ return room.hasEncryptionStateEvent() || !!this.findEncryptionEvent(eventsFormingFinalState);
1668
+ }
1669
+
1670
+ private async mapAndInjectRoomEvents(wrappedRoom: WrappedRoom<ILeftRoom>): Promise<{
1671
+ timelineEvents: MatrixEvent[];
1672
+ stateEvents?: MatrixEvent[];
1673
+ stateAfterEvents?: MatrixEvent[];
1674
+ }> {
1675
+ const stateEvents = this.mapSyncEventsFormat(wrappedRoom.state, wrappedRoom.room);
1676
+ const stateAfterEvents = this.mapSyncEventsFormat(
1677
+ wrappedRoom["org.matrix.msc4222.state_after"],
1678
+ wrappedRoom.room,
1648
1679
  );
1680
+ const timelineEvents = this.mapSyncEventsFormat(wrappedRoom.timeline, wrappedRoom.room);
1681
+
1682
+ if ("org.matrix.msc4222.state_after" in wrappedRoom) {
1683
+ await this.injectRoomEvents(wrappedRoom.room, undefined, stateAfterEvents, timelineEvents);
1684
+ } else {
1685
+ await this.injectRoomEvents(wrappedRoom.room, stateEvents, undefined, timelineEvents);
1686
+ }
1687
+
1688
+ return { timelineEvents, stateEvents, stateAfterEvents };
1649
1689
  }
1650
1690
 
1651
1691
  /**
1652
1692
  * Injects events into a room's model.
1653
1693
  * @param stateEventList - A list of state events. This is the state
1654
1694
  * at the *START* of the timeline list if it is supplied.
1695
+ * @param stateAfterEventList - A list of state events. This is the state
1696
+ * at the *END* of the timeline list if it is supplied.
1655
1697
  * @param timelineEventList - A list of timeline events, including threaded. Lower index
1656
1698
  * is earlier in time. Higher index is later.
1657
1699
  * @param fromCache - whether the sync response came from cache
1700
+ *
1701
+ * No more than one of stateEventList and stateAfterEventList must be supplied. If
1702
+ * stateEventList is supplied, the events in timelineEventList are added to the state
1703
+ * after stateEventList. If stateAfterEventList is supplied, the events in timelineEventList
1704
+ * are not added to the state.
1658
1705
  */
1659
1706
  public async injectRoomEvents(
1660
1707
  room: Room,
1661
1708
  stateEventList: MatrixEvent[],
1709
+ stateAfterEventList: undefined,
1710
+ timelineEventList?: MatrixEvent[],
1711
+ fromCache?: boolean,
1712
+ ): Promise<void>;
1713
+ public async injectRoomEvents(
1714
+ room: Room,
1715
+ stateEventList: undefined,
1716
+ stateAfterEventList: MatrixEvent[],
1717
+ timelineEventList?: MatrixEvent[],
1718
+ fromCache?: boolean,
1719
+ ): Promise<void>;
1720
+ public async injectRoomEvents(
1721
+ room: Room,
1722
+ stateEventList: MatrixEvent[] | undefined,
1723
+ stateAfterEventList: MatrixEvent[] | undefined,
1662
1724
  timelineEventList?: MatrixEvent[],
1663
1725
  fromCache = false,
1664
1726
  ): Promise<void> {
1727
+ const eitherStateEventList = stateAfterEventList ?? stateEventList!;
1728
+
1665
1729
  // If there are no events in the timeline yet, initialise it with
1666
1730
  // the given state events
1667
1731
  const liveTimeline = room.getLiveTimeline();
@@ -1675,10 +1739,11 @@ export class SyncApi {
1675
1739
  // push actions cache elsewhere so we can freeze MatrixEvents, or otherwise
1676
1740
  // find some solution where MatrixEvents are immutable but allow for a cache
1677
1741
  // field.
1678
- for (const ev of stateEventList) {
1742
+
1743
+ for (const ev of eitherStateEventList) {
1679
1744
  this.client.getPushActionsForEvent(ev);
1680
1745
  }
1681
- liveTimeline.initialiseState(stateEventList, {
1746
+ liveTimeline.initialiseState(eitherStateEventList, {
1682
1747
  timelineWasEmpty,
1683
1748
  });
1684
1749
  }
@@ -1710,17 +1775,18 @@ export class SyncApi {
1710
1775
  // XXX: As above, don't do this...
1711
1776
  //room.addLiveEvents(stateEventList || []);
1712
1777
  // Do this instead...
1713
- room.oldState.setStateEvents(stateEventList || []);
1714
- room.currentState.setStateEvents(stateEventList || []);
1778
+ room.oldState.setStateEvents(eitherStateEventList);
1779
+ room.currentState.setStateEvents(eitherStateEventList);
1715
1780
  }
1716
1781
 
1717
- // Execute the timeline events. This will continue to diverge the current state
1718
- // if the timeline has any state events in it.
1782
+ // Execute the timeline events. If addToState is true the timeline has any state
1783
+ // events in it, this will continue to diverge the current state.
1719
1784
  // This also needs to be done before running push rules on the events as they need
1720
1785
  // to be decorated with sender etc.
1721
1786
  await room.addLiveEvents(timelineEventList || [], {
1722
1787
  fromCache,
1723
1788
  timelineWasEmpty,
1789
+ addToState: stateAfterEventList === undefined,
1724
1790
  });
1725
1791
  this.client.processBeaconEvents(room, timelineEventList);
1726
1792
  }