matrix-js-sdk 41.5.0 → 41.6.0
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.
- package/CHANGELOG.md +7 -0
- package/lib/ReEmitter.js +1 -1
- package/lib/ReEmitter.js.map +1 -1
- package/lib/ToDeviceMessageQueue.js +2 -2
- package/lib/ToDeviceMessageQueue.js.map +1 -1
- package/lib/autodiscovery.js +0 -1
- package/lib/autodiscovery.js.map +1 -1
- package/lib/client.js +112 -165
- package/lib/client.js.map +1 -1
- package/lib/common-crypto/CryptoBackend.js +0 -2
- package/lib/common-crypto/CryptoBackend.js.map +1 -1
- package/lib/content-helpers.js +3 -5
- package/lib/content-helpers.js.map +1 -1
- package/lib/content-repo.js +7 -1
- package/lib/content-repo.js.map +1 -1
- package/lib/crypto/store/indexeddb-crypto-store-backend.js +9 -11
- package/lib/crypto/store/indexeddb-crypto-store-backend.js.map +1 -1
- package/lib/crypto/store/indexeddb-crypto-store.js +0 -2
- package/lib/crypto/store/indexeddb-crypto-store.js.map +1 -1
- package/lib/crypto/store/localStorage-crypto-store.js +11 -9
- package/lib/crypto/store/localStorage-crypto-store.js.map +1 -1
- package/lib/crypto/store/memory-crypto-store.js +11 -9
- package/lib/crypto/store/memory-crypto-store.js.map +1 -1
- package/lib/crypto-api/index.js +58 -58
- package/lib/crypto-api/index.js.map +1 -1
- package/lib/embedded.js +29 -45
- package/lib/embedded.js.map +1 -1
- package/lib/extensible_events_v1/MessageEvent.js +15 -15
- package/lib/extensible_events_v1/MessageEvent.js.map +1 -1
- package/lib/extensible_events_v1/PollEndEvent.js +8 -8
- package/lib/extensible_events_v1/PollEndEvent.js.map +1 -1
- package/lib/extensible_events_v1/PollResponseEvent.js +0 -1
- package/lib/extensible_events_v1/PollResponseEvent.js.map +1 -1
- package/lib/extensible_events_v1/PollStartEvent.js +28 -28
- package/lib/extensible_events_v1/PollStartEvent.js.map +1 -1
- package/lib/feature.js +5 -1
- package/lib/feature.js.map +1 -1
- package/lib/filter-component.js +4 -1
- package/lib/filter-component.js.map +1 -1
- package/lib/filter.js +1 -3
- package/lib/filter.js.map +1 -1
- package/lib/http-api/errors.js +20 -16
- package/lib/http-api/errors.js.map +1 -1
- package/lib/http-api/fetch.js +4 -6
- package/lib/http-api/fetch.js.map +1 -1
- package/lib/http-api/refresh.js +10 -12
- package/lib/http-api/refresh.js.map +1 -1
- package/lib/interactive-auth.js +1 -11
- package/lib/interactive-auth.js.map +1 -1
- package/lib/logger.js +0 -2
- package/lib/logger.js.map +1 -1
- package/lib/matrixrtc/CallMembership.js +43 -57
- package/lib/matrixrtc/CallMembership.js.map +1 -1
- package/lib/matrixrtc/MatrixRTCSession.js +19 -24
- package/lib/matrixrtc/MatrixRTCSession.js.map +1 -1
- package/lib/matrixrtc/MatrixRTCSessionManager.js +2 -3
- package/lib/matrixrtc/MatrixRTCSessionManager.js.map +1 -1
- package/lib/matrixrtc/MembershipManager.js +23 -30
- package/lib/matrixrtc/MembershipManager.js.map +1 -1
- package/lib/matrixrtc/MembershipManagerActionScheduler.js +1 -2
- package/lib/matrixrtc/MembershipManagerActionScheduler.js.map +1 -1
- package/lib/matrixrtc/RTCEncryptionManager.js +8 -11
- package/lib/matrixrtc/RTCEncryptionManager.js.map +1 -1
- package/lib/matrixrtc/ToDeviceKeyTransport.js +4 -4
- package/lib/matrixrtc/ToDeviceKeyTransport.js.map +1 -1
- package/lib/matrixrtc/utils.js +7 -1
- package/lib/matrixrtc/utils.js.map +1 -1
- package/lib/models/MSC3089TreeSpace.js +5 -7
- package/lib/models/MSC3089TreeSpace.js.map +1 -1
- package/lib/models/beacon.js +5 -11
- package/lib/models/beacon.js.map +1 -1
- package/lib/models/device.js +0 -7
- package/lib/models/device.js.map +1 -1
- package/lib/models/event-context.js +1 -2
- package/lib/models/event-context.js.map +1 -1
- package/lib/models/event-timeline-set.js +16 -26
- package/lib/models/event-timeline-set.js.map +1 -1
- package/lib/models/event-timeline.js +8 -15
- package/lib/models/event-timeline.js.map +1 -1
- package/lib/models/event.js +20 -37
- package/lib/models/event.js.map +1 -1
- package/lib/models/invites-ignorer.js +8 -12
- package/lib/models/invites-ignorer.js.map +1 -1
- package/lib/models/poll.js +7 -13
- package/lib/models/poll.js.map +1 -1
- package/lib/models/read-receipt.js +8 -4
- package/lib/models/read-receipt.js.map +1 -1
- package/lib/models/related-relations.js +0 -2
- package/lib/models/related-relations.js.map +1 -1
- package/lib/models/relations-container.js +4 -6
- package/lib/models/relations-container.js.map +1 -1
- package/lib/models/relations.js +7 -10
- package/lib/models/relations.js.map +1 -1
- package/lib/models/room-member.js +2 -24
- package/lib/models/room-member.js.map +1 -1
- package/lib/models/room-receipts.js +25 -22
- package/lib/models/room-receipts.js.map +1 -1
- package/lib/models/room-state.js +2 -2
- package/lib/models/room-state.js.map +1 -1
- package/lib/models/room-sticky-events.js +20 -5
- package/lib/models/room-sticky-events.js.map +1 -1
- package/lib/models/room.js +82 -105
- package/lib/models/room.js.map +1 -1
- package/lib/models/thread.js +12 -57
- package/lib/models/thread.js.map +1 -1
- package/lib/models/user.js +1 -20
- package/lib/models/user.js.map +1 -1
- package/lib/oidc/authorize.js +23 -32
- package/lib/oidc/authorize.js.map +1 -1
- package/lib/oidc/tokenRefresher.js +8 -11
- package/lib/oidc/tokenRefresher.js.map +1 -1
- package/lib/pushprocessor.js +8 -5
- package/lib/pushprocessor.js.map +1 -1
- package/lib/receipt-accumulator.js +12 -3
- package/lib/receipt-accumulator.js.map +1 -1
- package/lib/rendezvous/MSC4108SignInWithQR.d.ts.map +1 -1
- package/lib/rendezvous/MSC4108SignInWithQR.js +10 -23
- package/lib/rendezvous/MSC4108SignInWithQR.js.map +1 -1
- package/lib/rendezvous/channels/MSC4108SecureChannel.js +5 -10
- package/lib/rendezvous/channels/MSC4108SecureChannel.js.map +1 -1
- package/lib/rendezvous/transports/MSC4108RendezvousSession.js +5 -15
- package/lib/rendezvous/transports/MSC4108RendezvousSession.js.map +1 -1
- package/lib/room-hierarchy.js +7 -12
- package/lib/room-hierarchy.js.map +1 -1
- package/lib/rust-crypto/DehydratedDeviceManager.js +2 -3
- package/lib/rust-crypto/DehydratedDeviceManager.js.map +1 -1
- package/lib/rust-crypto/KeyClaimManager.js +1 -2
- package/lib/rust-crypto/KeyClaimManager.js.map +1 -1
- package/lib/rust-crypto/OutgoingRequestProcessor.js +11 -4
- package/lib/rust-crypto/OutgoingRequestProcessor.js.map +1 -1
- package/lib/rust-crypto/OutgoingRequestsManager.js +12 -12
- package/lib/rust-crypto/OutgoingRequestsManager.js.map +1 -1
- package/lib/rust-crypto/PerSessionKeyBackupDownloader.js +3 -5
- package/lib/rust-crypto/PerSessionKeyBackupDownloader.js.map +1 -1
- package/lib/rust-crypto/RoomEncryptor.js +6 -6
- package/lib/rust-crypto/RoomEncryptor.js.map +1 -1
- package/lib/rust-crypto/backup.js +22 -16
- package/lib/rust-crypto/backup.js.map +1 -1
- package/lib/rust-crypto/device-converter.js +13 -4
- package/lib/rust-crypto/device-converter.js.map +1 -1
- package/lib/rust-crypto/index.js +1 -3
- package/lib/rust-crypto/index.js.map +1 -1
- package/lib/rust-crypto/libolm_migration.js +13 -15
- package/lib/rust-crypto/libolm_migration.js.map +1 -1
- package/lib/rust-crypto/rust-crypto.js +52 -59
- package/lib/rust-crypto/rust-crypto.js.map +1 -1
- package/lib/rust-crypto/verification.js +10 -10
- package/lib/rust-crypto/verification.js.map +1 -1
- package/lib/scheduler.js +2 -2
- package/lib/scheduler.js.map +1 -1
- package/lib/secret-storage.js +16 -10
- package/lib/secret-storage.js.map +1 -1
- package/lib/serverCapabilities.js +2 -5
- package/lib/serverCapabilities.js.map +1 -1
- package/lib/sliding-sync-sdk.js +4 -9
- package/lib/sliding-sync-sdk.js.map +1 -1
- package/lib/sliding-sync.js +4 -9
- package/lib/sliding-sync.js.map +1 -1
- package/lib/store/indexeddb-local-backend.js +13 -8
- package/lib/store/indexeddb-local-backend.js.map +1 -1
- package/lib/store/indexeddb-remote-backend.js +6 -7
- package/lib/store/indexeddb-remote-backend.js.map +1 -1
- package/lib/store/indexeddb-store-worker.js +1 -2
- package/lib/store/indexeddb-store-worker.js.map +1 -1
- package/lib/store/indexeddb.js +4 -2
- package/lib/store/indexeddb.js.map +1 -1
- package/lib/store/memory.js +0 -4
- package/lib/store/memory.js.map +1 -1
- package/lib/sync-accumulator.js +2 -4
- package/lib/sync-accumulator.js.map +1 -1
- package/lib/sync.js +36 -43
- package/lib/sync.js.map +1 -1
- package/lib/timeline-window.js +2 -6
- package/lib/timeline-window.js.map +1 -1
- package/lib/utils/decryptAESSecretStorageItem.js +5 -1
- package/lib/utils/decryptAESSecretStorageItem.js.map +1 -1
- package/lib/utils/encryptAESSecretStorageItem.js +5 -1
- package/lib/utils/encryptAESSecretStorageItem.js.map +1 -1
- package/lib/utils.js +35 -20
- package/lib/utils.js.map +1 -1
- package/lib/webrtc/call.js +13 -45
- package/lib/webrtc/call.js.map +1 -1
- package/lib/webrtc/callEventHandler.js +0 -5
- package/lib/webrtc/callEventHandler.js.map +1 -1
- package/lib/webrtc/callFeed.js +0 -15
- package/lib/webrtc/callFeed.js.map +1 -1
- package/lib/webrtc/groupCall.js +82 -89
- package/lib/webrtc/groupCall.js.map +1 -1
- package/lib/webrtc/groupCallEventHandler.js +6 -7
- package/lib/webrtc/groupCallEventHandler.js.map +1 -1
- package/lib/webrtc/mediaHandler.js +8 -15
- package/lib/webrtc/mediaHandler.js.map +1 -1
- package/lib/webrtc/stats/callStatsReportGatherer.js +2 -5
- package/lib/webrtc/stats/callStatsReportGatherer.js.map +1 -1
- package/lib/webrtc/stats/connectionStatsReportBuilder.js +5 -1
- package/lib/webrtc/stats/connectionStatsReportBuilder.js.map +1 -1
- package/lib/webrtc/stats/groupCallStats.js +3 -4
- package/lib/webrtc/stats/groupCallStats.js.map +1 -1
- package/lib/webrtc/stats/media/mediaTrackStats.js +3 -3
- package/lib/webrtc/stats/media/mediaTrackStats.js.map +1 -1
- package/lib/webrtc/stats/media/mediaTrackStatsHandler.js +1 -1
- package/lib/webrtc/stats/media/mediaTrackStatsHandler.js.map +1 -1
- package/package.json +3 -3
- package/src/rendezvous/MSC4108SignInWithQR.ts +2 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"indexeddb.js","names":["MemoryStore","LocalIndexedDBStoreBackend","RemoteIndexedDBStoreBackend","MatrixEvent","logger","TypedEventEmitter","WRITE_DELAY_MS","IndexedDBStore","exists","indexedDB","dbName","constructor","opts","_defineProperty","emitter","emit","degradable","backend","getSavedSync","isNewlyCreated","getNextBatchToken","deleteAllData","clearDatabase","then","log","err","error","concat","syncTs","Date","now","userTuples","u","getUsers","userModifiedMap","userId","getLastModifiedTime","events","presence","push","event","syncToDatabase","syncData","setSyncData","roomId","getOutOfBandMembers","membershipEvents","setOutOfBandMembers","clearOutOfBandMembers","getClientOptions","options","storeClientOptions","Error","workerFactory","on","handler","startup","startedUp","Promise","resolve","connect","onClose","getUserPresenceEvents","userPresenceEvents","forEach","_ref","rawEvent","createUser","setPresenceEvent","storeUser","destroy","wantsSave","save","force","arguments","length","undefined","reallySave","func","fallback","_this","fallbackFn","_asyncToGenerator","_len","args","Array","_key","call","e","warn","getPendingEvents","_superprop_getGetPendingEvents","_this2","localStorage","serialized","getItem","pendingEventsKey","JSON","parse","setPendingEvents","_superprop_getSetPendingEvents","_this3","setItem","stringify","removeItem","saveToDeviceBatches","batches","getOldestToDeviceBatch","removeToDeviceBatch","id","getUserProfile","_this4","storeUserProfiles","userProfiles","_this5","removeUserProfiles","userIds","_this6"],"sources":["../../src/store/indexeddb.ts"],"sourcesContent":["/*\nCopyright 2017 - 2021 Vector Creations Ltd\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\n/* eslint-disable @babel/no-invalid-this */\n\nimport { MemoryStore, type IOpts as IBaseOpts } from \"./memory.ts\";\nimport { LocalIndexedDBStoreBackend } from \"./indexeddb-local-backend.ts\";\nimport { RemoteIndexedDBStoreBackend } from \"./indexeddb-remote-backend.ts\";\nimport { type IEvent, MatrixEvent } from \"../models/event.ts\";\nimport { logger } from \"../logger.ts\";\nimport { type ISavedSync } from \"./index.ts\";\nimport { type IIndexedDBBackend } from \"./indexeddb-backend.ts\";\nimport { type ISyncResponse } from \"../sync-accumulator.ts\";\nimport { type EventEmitterEvents, TypedEventEmitter } from \"../models/typed-event-emitter.ts\";\nimport { type IStateEventWithRoomId } from \"../@types/search.ts\";\nimport { type IndexedToDeviceBatch, type ToDeviceBatchWithTxnId } from \"../models/ToDeviceMessage.ts\";\nimport { type IStoredClientOpts } from \"../client.ts\";\nimport { type SyncUserProfile } from \"../models/user.ts\";\n\n/**\n * This is an internal module. See {@link IndexedDBStore} for the public class.\n */\n\n// If this value is too small we'll be writing very often which will cause\n// noticeable stop-the-world pauses. If this value is too big we'll be writing\n// so infrequently that the /sync size gets bigger on reload. Writing more\n// often does not affect the length of the pause since the entire /sync\n// response is persisted each time.\nconst WRITE_DELAY_MS = 1000 * 60 * 5; // once every 5 minutes\n\ninterface IOpts extends IBaseOpts {\n /** The Indexed DB interface e.g. `window.indexedDB` */\n indexedDB?: IDBFactory;\n /** Optional database name. The same name must be used to open the same database. */\n dbName?: string;\n /** Optional factory to spin up a Worker to execute the IDB transactions within. */\n workerFactory?: () => Worker;\n}\n\ntype EventHandlerMap = {\n // Fired when an IDB command fails on a degradable path, and the store falls back to MemoryStore\n // This signals the potential for data volatility.\n degraded: (e: Error) => void;\n // Fired when the IndexedDB gets closed unexpectedly, for example, if the underlying storage is removed or\n // if the user clears the database in the browser's history preferences.\n closed: () => void;\n};\n\nexport class IndexedDBStore extends MemoryStore {\n public static exists(indexedDB: IDBFactory, dbName: string): Promise<boolean> {\n return LocalIndexedDBStoreBackend.exists(indexedDB, dbName);\n }\n\n /**\n * The backend instance.\n * Call through to this API if you need to perform specific indexeddb actions like deleting the database.\n */\n public readonly backend: IIndexedDBBackend;\n\n private startedUp = false;\n private syncTs = 0;\n // Records the last-modified-time of each user at the last point we saved\n // the database, such that we can derive the set if users that have been\n // modified since we last saved.\n private userModifiedMap: Record<string, number> = {}; // user_id : timestamp\n private emitter = new TypedEventEmitter<keyof EventHandlerMap, EventHandlerMap>();\n\n /**\n * Construct a new Indexed Database store, which extends MemoryStore.\n *\n * This store functions like a MemoryStore except it periodically persists\n * the contents of the store to an IndexedDB backend.\n *\n * All data is still kept in-memory but can be loaded from disk by calling\n * `startup()`. This can make startup times quicker as a complete\n * sync from the server is not required. This does not reduce memory usage as all\n * the data is eagerly fetched when `startup()` is called.\n * ```\n * let opts = { indexedDB: window.indexedDB, localStorage: window.localStorage };\n * let store = new IndexedDBStore(opts);\n * let client = sdk.createClient({\n * store: store,\n * });\n * await store.startup(); // load from indexed db, must be called after createClient\n * client.startClient();\n * client.on(\"sync\", function(state, prevState, data) {\n * if (state === \"PREPARED\") {\n * console.log(\"Started up, now with go faster stripes!\");\n * }\n * });\n * ```\n *\n * @param opts - Options object.\n */\n public constructor(opts: IOpts) {\n super(opts);\n\n if (!opts.indexedDB) {\n throw new Error(\"Missing required option: indexedDB\");\n }\n\n if (opts.workerFactory) {\n this.backend = new RemoteIndexedDBStoreBackend(opts.workerFactory, opts.dbName);\n } else {\n this.backend = new LocalIndexedDBStoreBackend(opts.indexedDB, opts.dbName);\n }\n }\n\n /** Re-exports `TypedEventEmitter.on` */\n public on(event: EventEmitterEvents | \"degraded\" | \"closed\", handler: (...args: any[]) => void): void {\n this.emitter.on(event, handler);\n }\n\n /**\n * @returns Resolved when loaded from indexed db.\n */\n public startup(): Promise<void> {\n if (this.startedUp) {\n logger.log(`IndexedDBStore.startup: already started`);\n return Promise.resolve();\n }\n\n logger.log(`IndexedDBStore.startup: connecting to backend`);\n return this.backend\n .connect(this.onClose)\n .then(() => {\n logger.log(`IndexedDBStore.startup: loading presence events`);\n return this.backend.getUserPresenceEvents();\n })\n .then((userPresenceEvents) => {\n logger.log(`IndexedDBStore.startup: processing presence events`);\n userPresenceEvents.forEach(([userId, rawEvent]) => {\n if (!this.createUser) {\n throw new Error(\n \"`IndexedDBStore.startup` must be called after assigning it to the client, not before!\",\n );\n }\n const u = this.createUser(userId);\n if (rawEvent) {\n u.setPresenceEvent(new MatrixEvent(rawEvent));\n }\n this.userModifiedMap[u.userId] = u.getLastModifiedTime();\n this.storeUser(u);\n });\n this.startedUp = true;\n });\n }\n\n /*\n * Close the database and destroy any associated workers\n */\n public destroy(): Promise<void> {\n return this.backend.destroy();\n }\n\n private onClose = (): void => {\n this.emitter.emit(\"closed\");\n };\n\n /**\n * @returns Promise which resolves with a sync response to restore the\n * client state to where it was at the last save, or null if there\n * is no saved sync data.\n */\n public getSavedSync = this.degradable((): Promise<ISavedSync | null> => {\n return this.backend.getSavedSync();\n }, \"getSavedSync\");\n\n /** @returns whether or not the database was newly created in this session. */\n public isNewlyCreated = this.degradable((): Promise<boolean> => {\n return this.backend.isNewlyCreated();\n }, \"isNewlyCreated\");\n\n /**\n * @returns If there is a saved sync, the nextBatch token\n * for this sync, otherwise null.\n */\n public getSavedSyncToken = this.degradable((): Promise<string | null> => {\n return this.backend.getNextBatchToken();\n }, \"getSavedSyncToken\");\n\n /**\n * Delete all data from this store.\n * @returns Promise which resolves if the data was deleted from the database.\n */\n public deleteAllData = this.degradable((): Promise<void> => {\n super.deleteAllData();\n return this.backend.clearDatabase().then(\n () => {\n logger.log(\"Deleted indexeddb data.\");\n },\n (err) => {\n logger.error(`Failed to delete indexeddb data: ${err}`);\n throw err;\n },\n );\n }, null);\n\n /**\n * Whether this store would like to save its data\n * Note that obviously whether the store wants to save or\n * not could change between calling this function and calling\n * save().\n *\n * @returns True if calling save() will actually save\n * (at the time this function is called).\n */\n public wantsSave(): boolean {\n const now = Date.now();\n return now - this.syncTs > WRITE_DELAY_MS;\n }\n\n /**\n * Possibly write data to the database.\n *\n * @param force - True to force a save to happen\n * @returns Promise resolves after the write completes\n * (or immediately if no write is performed)\n */\n public save(force = false): Promise<void> {\n if (force || this.wantsSave()) {\n return this.reallySave();\n }\n return Promise.resolve();\n }\n\n private reallySave = this.degradable((): Promise<void> => {\n this.syncTs = Date.now(); // set now to guard against multi-writes\n\n // work out changed users (this doesn't handle deletions but you\n // can't 'delete' users as they are just presence events).\n const userTuples: [userId: string, presenceEvent: Partial<IEvent>][] = [];\n for (const u of this.getUsers()) {\n if (this.userModifiedMap[u.userId] === u.getLastModifiedTime()) continue;\n if (!u.events.presence) continue;\n\n userTuples.push([u.userId, u.events.presence.event]);\n\n // note that we've saved this version of the user\n this.userModifiedMap[u.userId] = u.getLastModifiedTime();\n }\n\n return this.backend.syncToDatabase(userTuples);\n }, null);\n\n public setSyncData = this.degradable((syncData: ISyncResponse): Promise<void> => {\n return this.backend.setSyncData(syncData);\n }, \"setSyncData\");\n\n /**\n * Returns the out-of-band membership events for this room that\n * were previously loaded.\n * @returns the events, potentially an empty array if OOB loading didn't yield any new members\n * @returns in case the members for this room haven't been stored yet\n */\n public getOutOfBandMembers = this.degradable((roomId: string): Promise<IStateEventWithRoomId[] | null> => {\n return this.backend.getOutOfBandMembers(roomId);\n }, \"getOutOfBandMembers\");\n\n /**\n * Stores the out-of-band membership events for this room. Note that\n * it still makes sense to store an empty array as the OOB status for the room is\n * marked as fetched, and getOutOfBandMembers will return an empty array instead of null\n * @param membershipEvents - the membership events to store\n * @returns when all members have been stored\n */\n public setOutOfBandMembers = this.degradable(\n (roomId: string, membershipEvents: IStateEventWithRoomId[]): Promise<void> => {\n super.setOutOfBandMembers(roomId, membershipEvents);\n return this.backend.setOutOfBandMembers(roomId, membershipEvents);\n },\n \"setOutOfBandMembers\",\n );\n\n public clearOutOfBandMembers = this.degradable((roomId: string) => {\n super.clearOutOfBandMembers(roomId);\n return this.backend.clearOutOfBandMembers(roomId);\n }, \"clearOutOfBandMembers\");\n\n public getClientOptions = this.degradable((): Promise<IStoredClientOpts | undefined> => {\n return this.backend.getClientOptions();\n }, \"getClientOptions\");\n\n public storeClientOptions = this.degradable((options: IStoredClientOpts): Promise<void> => {\n super.storeClientOptions(options);\n return this.backend.storeClientOptions(options);\n }, \"storeClientOptions\");\n\n /**\n * All member functions of `IndexedDBStore` that access the backend use this wrapper to\n * watch for failures after initial store startup, including `QuotaExceededError` as\n * free disk space changes, etc.\n *\n * When IndexedDB fails via any of these paths, we degrade this back to a `MemoryStore`\n * in place so that the current operation and all future ones are in-memory only.\n *\n * @param func - The degradable work to do.\n * @param fallback - The method name for fallback.\n * @returns A wrapped member function.\n */\n private degradable<A extends Array<any>, F extends keyof MemoryStore | null, R = void>(\n func: DegradableFn<A, R>,\n fallback: F,\n ): DegradableFn<A, F extends string ? R : void> {\n const fallbackFn = fallback ? (super[fallback] as (...args: A) => Promise<R>) : null;\n\n return (async (...args) => {\n try {\n return await func.call(this, ...args);\n } catch (e) {\n logger.error(\"IndexedDBStore failure, degrading to MemoryStore\", e);\n this.emitter.emit(\"degraded\", e as Error);\n try {\n // We try to delete IndexedDB after degrading since this store is only a\n // cache (the app will still function correctly without the data).\n // It's possible that deleting repair IndexedDB for the next app load,\n // potentially by making a little more space available.\n logger.log(\"IndexedDBStore trying to delete degraded data\");\n await this.backend.clearDatabase();\n logger.log(\"IndexedDBStore delete after degrading succeeded\");\n } catch (e) {\n logger.warn(\"IndexedDBStore delete after degrading failed\", e);\n }\n // Degrade the store from being an instance of `IndexedDBStore` to instead be\n // an instance of `MemoryStore` so that future API calls use the memory path\n // directly and skip IndexedDB entirely. This should be safe as\n // `IndexedDBStore` already extends from `MemoryStore`, so we are making the\n // store become its parent type in a way. The mutator methods of\n // `IndexedDBStore` also maintain the state that `MemoryStore` uses (many are\n // not overridden at all).\n if (fallbackFn) {\n return fallbackFn.call(this, ...args);\n }\n }\n }) as DegradableFn<A, F extends string ? R : void>;\n }\n\n // XXX: ideally these would be stored in indexeddb as part of the room but,\n // we don't store rooms as such and instead accumulate entire sync responses atm.\n public async getPendingEvents(roomId: string): Promise<Partial<IEvent>[]> {\n if (!this.localStorage) return super.getPendingEvents(roomId);\n\n const serialized = this.localStorage.getItem(pendingEventsKey(roomId));\n if (serialized) {\n try {\n return JSON.parse(serialized);\n } catch (e) {\n logger.error(\"Could not parse persisted pending events\", e);\n }\n }\n return [];\n }\n\n public async setPendingEvents(roomId: string, events: Partial<IEvent>[]): Promise<void> {\n if (!this.localStorage) return super.setPendingEvents(roomId, events);\n\n if (events.length > 0) {\n this.localStorage.setItem(pendingEventsKey(roomId), JSON.stringify(events));\n } else {\n this.localStorage.removeItem(pendingEventsKey(roomId));\n }\n }\n\n public saveToDeviceBatches(batches: ToDeviceBatchWithTxnId[]): Promise<void> {\n return this.backend.saveToDeviceBatches(batches);\n }\n\n public getOldestToDeviceBatch(): Promise<IndexedToDeviceBatch | null> {\n return this.backend.getOldestToDeviceBatch();\n }\n\n public removeToDeviceBatch(id: number): Promise<void> {\n return this.backend.removeToDeviceBatch(id);\n }\n\n public async getUserProfile(userId: string): Promise<SyncUserProfile | undefined> {\n return this.backend.getUserProfile(userId);\n }\n\n public async storeUserProfiles(userProfiles: Map<string, SyncUserProfile>): Promise<void> {\n return this.backend.storeUserProfiles(userProfiles);\n }\n\n public async removeUserProfiles(userIds: string[]): Promise<void> {\n return this.backend.removeUserProfiles(userIds);\n }\n}\n\n/**\n * @param roomId - ID of the current room\n * @returns Storage key to retrieve pending events\n */\nfunction pendingEventsKey(roomId: string): string {\n return `mx_pending_events_${roomId}`;\n}\n\ntype DegradableFn<A extends Array<any>, T> = (...args: A) => Promise<T>;\n"],"mappings":"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","ignoreList":[]}
|
|
1
|
+
{"version":3,"file":"indexeddb.js","names":["MemoryStore","LocalIndexedDBStoreBackend","RemoteIndexedDBStoreBackend","MatrixEvent","logger","TypedEventEmitter","WRITE_DELAY_MS","IndexedDBStore","exists","indexedDB","dbName","constructor","opts","_defineProperty","emitter","emit","degradable","backend","getSavedSync","isNewlyCreated","getNextBatchToken","deleteAllData","clearDatabase","then","log","err","error","concat","syncTs","Date","now","userTuples","u","getUsers","userModifiedMap","userId","getLastModifiedTime","events","presence","push","event","syncToDatabase","syncData","setSyncData","roomId","getOutOfBandMembers","membershipEvents","setOutOfBandMembers","clearOutOfBandMembers","getClientOptions","options","storeClientOptions","Error","workerFactory","on","handler","startup","startedUp","Promise","resolve","connect","onClose","getUserPresenceEvents","userPresenceEvents","forEach","_ref","_ref2","_slicedToArray","rawEvent","createUser","setPresenceEvent","storeUser","destroy","wantsSave","save","force","arguments","length","undefined","reallySave","func","fallback","_this","fallbackFn","_asyncToGenerator","_len","args","Array","_key","call","e","warn","getPendingEvents","_superprop_getGetPendingEvents","_this2","localStorage","serialized","getItem","pendingEventsKey","JSON","parse","setPendingEvents","_superprop_getSetPendingEvents","_this3","setItem","stringify","removeItem","saveToDeviceBatches","batches","getOldestToDeviceBatch","removeToDeviceBatch","id","getUserProfile","_this4","storeUserProfiles","userProfiles","_this5","removeUserProfiles","userIds","_this6"],"sources":["../../src/store/indexeddb.ts"],"sourcesContent":["/*\nCopyright 2017 - 2021 Vector Creations Ltd\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\n/* eslint-disable @babel/no-invalid-this */\n\nimport { MemoryStore, type IOpts as IBaseOpts } from \"./memory.ts\";\nimport { LocalIndexedDBStoreBackend } from \"./indexeddb-local-backend.ts\";\nimport { RemoteIndexedDBStoreBackend } from \"./indexeddb-remote-backend.ts\";\nimport { type IEvent, MatrixEvent } from \"../models/event.ts\";\nimport { logger } from \"../logger.ts\";\nimport { type ISavedSync } from \"./index.ts\";\nimport { type IIndexedDBBackend } from \"./indexeddb-backend.ts\";\nimport { type ISyncResponse } from \"../sync-accumulator.ts\";\nimport { type EventEmitterEvents, TypedEventEmitter } from \"../models/typed-event-emitter.ts\";\nimport { type IStateEventWithRoomId } from \"../@types/search.ts\";\nimport { type IndexedToDeviceBatch, type ToDeviceBatchWithTxnId } from \"../models/ToDeviceMessage.ts\";\nimport { type IStoredClientOpts } from \"../client.ts\";\nimport { type SyncUserProfile } from \"../models/user.ts\";\n\n/**\n * This is an internal module. See {@link IndexedDBStore} for the public class.\n */\n\n// If this value is too small we'll be writing very often which will cause\n// noticeable stop-the-world pauses. If this value is too big we'll be writing\n// so infrequently that the /sync size gets bigger on reload. Writing more\n// often does not affect the length of the pause since the entire /sync\n// response is persisted each time.\nconst WRITE_DELAY_MS = 1000 * 60 * 5; // once every 5 minutes\n\ninterface IOpts extends IBaseOpts {\n /** The Indexed DB interface e.g. `window.indexedDB` */\n indexedDB?: IDBFactory;\n /** Optional database name. The same name must be used to open the same database. */\n dbName?: string;\n /** Optional factory to spin up a Worker to execute the IDB transactions within. */\n workerFactory?: () => Worker;\n}\n\ntype EventHandlerMap = {\n // Fired when an IDB command fails on a degradable path, and the store falls back to MemoryStore\n // This signals the potential for data volatility.\n degraded: (e: Error) => void;\n // Fired when the IndexedDB gets closed unexpectedly, for example, if the underlying storage is removed or\n // if the user clears the database in the browser's history preferences.\n closed: () => void;\n};\n\nexport class IndexedDBStore extends MemoryStore {\n public static exists(indexedDB: IDBFactory, dbName: string): Promise<boolean> {\n return LocalIndexedDBStoreBackend.exists(indexedDB, dbName);\n }\n\n /**\n * The backend instance.\n * Call through to this API if you need to perform specific indexeddb actions like deleting the database.\n */\n public readonly backend: IIndexedDBBackend;\n\n private startedUp = false;\n private syncTs = 0;\n // Records the last-modified-time of each user at the last point we saved\n // the database, such that we can derive the set if users that have been\n // modified since we last saved.\n private userModifiedMap: Record<string, number> = {}; // user_id : timestamp\n private emitter = new TypedEventEmitter<keyof EventHandlerMap, EventHandlerMap>();\n\n /**\n * Construct a new Indexed Database store, which extends MemoryStore.\n *\n * This store functions like a MemoryStore except it periodically persists\n * the contents of the store to an IndexedDB backend.\n *\n * All data is still kept in-memory but can be loaded from disk by calling\n * `startup()`. This can make startup times quicker as a complete\n * sync from the server is not required. This does not reduce memory usage as all\n * the data is eagerly fetched when `startup()` is called.\n * ```\n * let opts = { indexedDB: window.indexedDB, localStorage: window.localStorage };\n * let store = new IndexedDBStore(opts);\n * let client = sdk.createClient({\n * store: store,\n * });\n * await store.startup(); // load from indexed db, must be called after createClient\n * client.startClient();\n * client.on(\"sync\", function(state, prevState, data) {\n * if (state === \"PREPARED\") {\n * console.log(\"Started up, now with go faster stripes!\");\n * }\n * });\n * ```\n *\n * @param opts - Options object.\n */\n public constructor(opts: IOpts) {\n super(opts);\n\n if (!opts.indexedDB) {\n throw new Error(\"Missing required option: indexedDB\");\n }\n\n if (opts.workerFactory) {\n this.backend = new RemoteIndexedDBStoreBackend(opts.workerFactory, opts.dbName);\n } else {\n this.backend = new LocalIndexedDBStoreBackend(opts.indexedDB, opts.dbName);\n }\n }\n\n /** Re-exports `TypedEventEmitter.on` */\n public on(event: EventEmitterEvents | \"degraded\" | \"closed\", handler: (...args: any[]) => void): void {\n this.emitter.on(event, handler);\n }\n\n /**\n * @returns Resolved when loaded from indexed db.\n */\n public startup(): Promise<void> {\n if (this.startedUp) {\n logger.log(`IndexedDBStore.startup: already started`);\n return Promise.resolve();\n }\n\n logger.log(`IndexedDBStore.startup: connecting to backend`);\n return this.backend\n .connect(this.onClose)\n .then(() => {\n logger.log(`IndexedDBStore.startup: loading presence events`);\n return this.backend.getUserPresenceEvents();\n })\n .then((userPresenceEvents) => {\n logger.log(`IndexedDBStore.startup: processing presence events`);\n userPresenceEvents.forEach(([userId, rawEvent]) => {\n if (!this.createUser) {\n throw new Error(\n \"`IndexedDBStore.startup` must be called after assigning it to the client, not before!\",\n );\n }\n const u = this.createUser(userId);\n if (rawEvent) {\n u.setPresenceEvent(new MatrixEvent(rawEvent));\n }\n this.userModifiedMap[u.userId] = u.getLastModifiedTime();\n this.storeUser(u);\n });\n this.startedUp = true;\n });\n }\n\n /*\n * Close the database and destroy any associated workers\n */\n public destroy(): Promise<void> {\n return this.backend.destroy();\n }\n\n private onClose = (): void => {\n this.emitter.emit(\"closed\");\n };\n\n /**\n * @returns Promise which resolves with a sync response to restore the\n * client state to where it was at the last save, or null if there\n * is no saved sync data.\n */\n public getSavedSync = this.degradable((): Promise<ISavedSync | null> => {\n return this.backend.getSavedSync();\n }, \"getSavedSync\");\n\n /** @returns whether or not the database was newly created in this session. */\n public isNewlyCreated = this.degradable((): Promise<boolean> => {\n return this.backend.isNewlyCreated();\n }, \"isNewlyCreated\");\n\n /**\n * @returns If there is a saved sync, the nextBatch token\n * for this sync, otherwise null.\n */\n public getSavedSyncToken = this.degradable((): Promise<string | null> => {\n return this.backend.getNextBatchToken();\n }, \"getSavedSyncToken\");\n\n /**\n * Delete all data from this store.\n * @returns Promise which resolves if the data was deleted from the database.\n */\n public deleteAllData = this.degradable((): Promise<void> => {\n super.deleteAllData();\n return this.backend.clearDatabase().then(\n () => {\n logger.log(\"Deleted indexeddb data.\");\n },\n (err) => {\n logger.error(`Failed to delete indexeddb data: ${err}`);\n throw err;\n },\n );\n }, null);\n\n /**\n * Whether this store would like to save its data\n * Note that obviously whether the store wants to save or\n * not could change between calling this function and calling\n * save().\n *\n * @returns True if calling save() will actually save\n * (at the time this function is called).\n */\n public wantsSave(): boolean {\n const now = Date.now();\n return now - this.syncTs > WRITE_DELAY_MS;\n }\n\n /**\n * Possibly write data to the database.\n *\n * @param force - True to force a save to happen\n * @returns Promise resolves after the write completes\n * (or immediately if no write is performed)\n */\n public save(force = false): Promise<void> {\n if (force || this.wantsSave()) {\n return this.reallySave();\n }\n return Promise.resolve();\n }\n\n private reallySave = this.degradable((): Promise<void> => {\n this.syncTs = Date.now(); // set now to guard against multi-writes\n\n // work out changed users (this doesn't handle deletions but you\n // can't 'delete' users as they are just presence events).\n const userTuples: [userId: string, presenceEvent: Partial<IEvent>][] = [];\n for (const u of this.getUsers()) {\n if (this.userModifiedMap[u.userId] === u.getLastModifiedTime()) continue;\n if (!u.events.presence) continue;\n\n userTuples.push([u.userId, u.events.presence.event]);\n\n // note that we've saved this version of the user\n this.userModifiedMap[u.userId] = u.getLastModifiedTime();\n }\n\n return this.backend.syncToDatabase(userTuples);\n }, null);\n\n public setSyncData = this.degradable((syncData: ISyncResponse): Promise<void> => {\n return this.backend.setSyncData(syncData);\n }, \"setSyncData\");\n\n /**\n * Returns the out-of-band membership events for this room that\n * were previously loaded.\n * @returns the events, potentially an empty array if OOB loading didn't yield any new members\n * @returns in case the members for this room haven't been stored yet\n */\n public getOutOfBandMembers = this.degradable((roomId: string): Promise<IStateEventWithRoomId[] | null> => {\n return this.backend.getOutOfBandMembers(roomId);\n }, \"getOutOfBandMembers\");\n\n /**\n * Stores the out-of-band membership events for this room. Note that\n * it still makes sense to store an empty array as the OOB status for the room is\n * marked as fetched, and getOutOfBandMembers will return an empty array instead of null\n * @param membershipEvents - the membership events to store\n * @returns when all members have been stored\n */\n public setOutOfBandMembers = this.degradable(\n (roomId: string, membershipEvents: IStateEventWithRoomId[]): Promise<void> => {\n super.setOutOfBandMembers(roomId, membershipEvents);\n return this.backend.setOutOfBandMembers(roomId, membershipEvents);\n },\n \"setOutOfBandMembers\",\n );\n\n public clearOutOfBandMembers = this.degradable((roomId: string) => {\n super.clearOutOfBandMembers(roomId);\n return this.backend.clearOutOfBandMembers(roomId);\n }, \"clearOutOfBandMembers\");\n\n public getClientOptions = this.degradable((): Promise<IStoredClientOpts | undefined> => {\n return this.backend.getClientOptions();\n }, \"getClientOptions\");\n\n public storeClientOptions = this.degradable((options: IStoredClientOpts): Promise<void> => {\n super.storeClientOptions(options);\n return this.backend.storeClientOptions(options);\n }, \"storeClientOptions\");\n\n /**\n * All member functions of `IndexedDBStore` that access the backend use this wrapper to\n * watch for failures after initial store startup, including `QuotaExceededError` as\n * free disk space changes, etc.\n *\n * When IndexedDB fails via any of these paths, we degrade this back to a `MemoryStore`\n * in place so that the current operation and all future ones are in-memory only.\n *\n * @param func - The degradable work to do.\n * @param fallback - The method name for fallback.\n * @returns A wrapped member function.\n */\n private degradable<A extends Array<any>, F extends keyof MemoryStore | null, R = void>(\n func: DegradableFn<A, R>,\n fallback: F,\n ): DegradableFn<A, F extends string ? R : void> {\n const fallbackFn = fallback ? (super[fallback] as (...args: A) => Promise<R>) : null;\n\n return (async (...args) => {\n try {\n return await func.call(this, ...args);\n } catch (e) {\n logger.error(\"IndexedDBStore failure, degrading to MemoryStore\", e);\n this.emitter.emit(\"degraded\", e as Error);\n try {\n // We try to delete IndexedDB after degrading since this store is only a\n // cache (the app will still function correctly without the data).\n // It's possible that deleting repair IndexedDB for the next app load,\n // potentially by making a little more space available.\n logger.log(\"IndexedDBStore trying to delete degraded data\");\n await this.backend.clearDatabase();\n logger.log(\"IndexedDBStore delete after degrading succeeded\");\n } catch (e) {\n logger.warn(\"IndexedDBStore delete after degrading failed\", e);\n }\n // Degrade the store from being an instance of `IndexedDBStore` to instead be\n // an instance of `MemoryStore` so that future API calls use the memory path\n // directly and skip IndexedDB entirely. This should be safe as\n // `IndexedDBStore` already extends from `MemoryStore`, so we are making the\n // store become its parent type in a way. The mutator methods of\n // `IndexedDBStore` also maintain the state that `MemoryStore` uses (many are\n // not overridden at all).\n if (fallbackFn) {\n return fallbackFn.call(this, ...args);\n }\n }\n }) as DegradableFn<A, F extends string ? R : void>;\n }\n\n // XXX: ideally these would be stored in indexeddb as part of the room but,\n // we don't store rooms as such and instead accumulate entire sync responses atm.\n public async getPendingEvents(roomId: string): Promise<Partial<IEvent>[]> {\n if (!this.localStorage) return super.getPendingEvents(roomId);\n\n const serialized = this.localStorage.getItem(pendingEventsKey(roomId));\n if (serialized) {\n try {\n return JSON.parse(serialized);\n } catch (e) {\n logger.error(\"Could not parse persisted pending events\", e);\n }\n }\n return [];\n }\n\n public async setPendingEvents(roomId: string, events: Partial<IEvent>[]): Promise<void> {\n if (!this.localStorage) return super.setPendingEvents(roomId, events);\n\n if (events.length > 0) {\n this.localStorage.setItem(pendingEventsKey(roomId), JSON.stringify(events));\n } else {\n this.localStorage.removeItem(pendingEventsKey(roomId));\n }\n }\n\n public saveToDeviceBatches(batches: ToDeviceBatchWithTxnId[]): Promise<void> {\n return this.backend.saveToDeviceBatches(batches);\n }\n\n public getOldestToDeviceBatch(): Promise<IndexedToDeviceBatch | null> {\n return this.backend.getOldestToDeviceBatch();\n }\n\n public removeToDeviceBatch(id: number): Promise<void> {\n return this.backend.removeToDeviceBatch(id);\n }\n\n public async getUserProfile(userId: string): Promise<SyncUserProfile | undefined> {\n return this.backend.getUserProfile(userId);\n }\n\n public async storeUserProfiles(userProfiles: Map<string, SyncUserProfile>): Promise<void> {\n return this.backend.storeUserProfiles(userProfiles);\n }\n\n public async removeUserProfiles(userIds: string[]): Promise<void> {\n return this.backend.removeUserProfiles(userIds);\n }\n}\n\n/**\n * @param roomId - ID of the current room\n * @returns Storage key to retrieve pending events\n */\nfunction pendingEventsKey(roomId: string): string {\n return `mx_pending_events_${roomId}`;\n}\n\ntype DegradableFn<A extends Array<any>, T> = (...args: A) => Promise<T>;\n"],"mappings":"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","ignoreList":[]}
|
package/lib/store/memory.js
CHANGED
|
@@ -46,15 +46,11 @@ export class MemoryStore {
|
|
|
46
46
|
// }
|
|
47
47
|
_defineProperty(this, "filters", new MapWithDefault(() => new Map()));
|
|
48
48
|
_defineProperty(this, "accountData", new Map());
|
|
49
|
-
// type: content
|
|
50
|
-
_defineProperty(this, "localStorage", void 0);
|
|
51
49
|
_defineProperty(this, "oobMembers", new Map());
|
|
52
50
|
// roomId: [member events]
|
|
53
51
|
_defineProperty(this, "pendingEvents", {});
|
|
54
|
-
_defineProperty(this, "clientOptions", void 0);
|
|
55
52
|
_defineProperty(this, "pendingToDeviceBatches", []);
|
|
56
53
|
_defineProperty(this, "nextToDeviceBatchId", 0);
|
|
57
|
-
_defineProperty(this, "createUser", void 0);
|
|
58
54
|
_defineProperty(this, "userProfiles", new Map());
|
|
59
55
|
/**
|
|
60
56
|
* Called when a room member in a room being tracked by this store has been
|
package/lib/store/memory.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"memory.js","names":["RoomStateEvent","MapWithDefault","KnownMembership","isValidFilterId","filterId","isValidStr","MemoryStore","constructor","opts","arguments","length","undefined","_defineProperty","Map","event","state","member","_this$createUser","membership","Invite","user","users","userId","createUser","call","name","setDisplayName","events","setRawDisplayName","getDirectionalContent","displayname","getContent","avatar_url","setAvatarUrl","localStorage","getSyncToken","syncToken","isNewlyCreated","Promise","resolve","setSyncToken","token","storeRoom","room","rooms","roomId","currentState","on","Members","onRoomMember","getMembers","forEach","m","setUserCreator","creator","getRoom","getRooms","Object","values","removeRoom","removeListener","getRoomSummaries","map","summary","storeUser","getUser","getUsers","scrollback","limit","storeEvents","toStart","storeFilter","filter","filters","getOrCreate","set","getFilter","_this$filters$get","get","getFilterIdByName","filterName","key","value","getItem","_unused","setFilterIdByName","setItem","removeItem","_unused2","storeAccountDataEvents","isDeleted","keys","accountData","delete","getType","getAccountData","eventType","setSyncData","syncData","wantsSave","save","force","startup","getSavedSync","getSavedSyncToken","deleteAllData","getOutOfBandMembers","oobMembers","setOutOfBandMembers","membershipEvents","clearOutOfBandMembers","getClientOptions","clientOptions","storeClientOptions","options","assign","getPendingEvents","_this","_asyncToGenerator","_this$pendingEvents$r","pendingEvents","setPendingEvents","_this2","saveToDeviceBatches","batches","batch","pendingToDeviceBatches","push","id","nextToDeviceBatchId","txnId","getOldestToDeviceBatch","_this3","removeToDeviceBatch","getUserProfile","_this4","userProfiles","storeUserProfiles","_this5","profile","removeUserProfiles","userIds","_this6","destroy"],"sources":["../../src/store/memory.ts"],"sourcesContent":["/*\nCopyright 2015 - 2021 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\n/**\n * This is an internal module. See {@link MemoryStore} for the public class.\n */\n\nimport { type EventType } from \"../@types/event.ts\";\nimport { type Room } from \"../models/room.ts\";\nimport { type SyncUserProfile, type User } from \"../models/user.ts\";\nimport { type IEvent, type MatrixEvent } from \"../models/event.ts\";\nimport { type RoomState, RoomStateEvent } from \"../models/room-state.ts\";\nimport { type RoomMember } from \"../models/room-member.ts\";\nimport { type Filter } from \"../filter.ts\";\nimport { type ISavedSync, type IStore, type UserCreator } from \"./index.ts\";\nimport { type RoomSummary } from \"../models/room-summary.ts\";\nimport { type ISyncResponse } from \"../sync-accumulator.ts\";\nimport { type IStateEventWithRoomId } from \"../@types/search.ts\";\nimport { type IndexedToDeviceBatch, type ToDeviceBatchWithTxnId } from \"../models/ToDeviceMessage.ts\";\nimport { type IStoredClientOpts } from \"../client.ts\";\nimport { MapWithDefault } from \"../utils.ts\";\nimport { KnownMembership } from \"../@types/membership.ts\";\n\nfunction isValidFilterId(filterId?: string | number | null): boolean {\n const isValidStr =\n typeof filterId === \"string\" &&\n !!filterId &&\n filterId !== \"undefined\" && // exclude these as we've serialized undefined in localStorage before\n filterId !== \"null\";\n\n return isValidStr || typeof filterId === \"number\";\n}\n\nexport interface IOpts {\n /** The local storage instance to persist some forms of data such as tokens. Rooms will NOT be stored. */\n localStorage?: Storage;\n}\n\nexport class MemoryStore implements IStore {\n private rooms: Record<string, Room> = {}; // roomId: Room\n private users: Record<string, User> = {}; // userId: User\n private syncToken: string | null = null;\n // userId: {\n // filterId: Filter\n // }\n private filters: MapWithDefault<string, Map<string, Filter>> = new MapWithDefault(() => new Map());\n public accountData: Map<string, MatrixEvent> = new Map(); // type: content\n protected readonly localStorage?: Storage;\n private oobMembers: Map<string, IStateEventWithRoomId[]> = new Map(); // roomId: [member events]\n private pendingEvents: { [roomId: string]: Partial<IEvent>[] } = {};\n private clientOptions?: IStoredClientOpts;\n private pendingToDeviceBatches: IndexedToDeviceBatch[] = [];\n private nextToDeviceBatchId = 0;\n protected createUser?: UserCreator;\n public readonly userProfiles = new Map<string, SyncUserProfile>();\n\n /**\n * Construct a new in-memory data store for the Matrix Client.\n * @param opts - Config options\n */\n public constructor(opts: IOpts = {}) {\n this.localStorage = opts.localStorage;\n }\n\n /**\n * Retrieve the token to stream from.\n * @returns The token or null.\n */\n public getSyncToken(): string | null {\n return this.syncToken;\n }\n\n /** @returns whether or not the database was newly created in this session. */\n public isNewlyCreated(): Promise<boolean> {\n return Promise.resolve(true);\n }\n\n /**\n * Set the token to stream from.\n * @param token - The token to stream from.\n */\n public setSyncToken(token: string): void {\n this.syncToken = token;\n }\n\n /**\n * Store the given room.\n * @param room - The room to be stored. All properties must be stored.\n */\n public storeRoom(room: Room): void {\n this.rooms[room.roomId] = room;\n // add listeners for room member changes so we can keep the room member\n // map up-to-date.\n room.currentState.on(RoomStateEvent.Members, this.onRoomMember);\n // add existing members\n room.currentState.getMembers().forEach((m) => {\n this.onRoomMember(null, room.currentState, m);\n });\n }\n\n public setUserCreator(creator: UserCreator): void {\n this.createUser = creator;\n }\n\n /**\n * Called when a room member in a room being tracked by this store has been\n * updated.\n */\n private onRoomMember = (event: MatrixEvent | null, state: RoomState, member: RoomMember): void => {\n if (member.membership === KnownMembership.Invite) {\n // We do NOT add invited members because people love to typo user IDs\n // which would then show up in these lists (!)\n return;\n }\n\n const user = this.users[member.userId] || this.createUser?.(member.userId);\n if (member.name) {\n user.setDisplayName(member.name);\n if (member.events.member) {\n user.setRawDisplayName(member.events.member.getDirectionalContent().displayname);\n }\n }\n if (member.events.member && member.events.member.getContent().avatar_url) {\n user.setAvatarUrl(member.events.member.getContent().avatar_url);\n }\n this.users[user.userId] = user;\n };\n\n /**\n * Retrieve a room by its' room ID.\n * @param roomId - The room ID.\n * @returns The room or null.\n */\n public getRoom(roomId: string): Room | null {\n return this.rooms[roomId] || null;\n }\n\n /**\n * Retrieve all known rooms.\n * @returns A list of rooms, which may be empty.\n */\n public getRooms(): Room[] {\n return Object.values(this.rooms);\n }\n\n /**\n * Permanently delete a room.\n */\n public removeRoom(roomId: string): void {\n if (this.rooms[roomId]) {\n this.rooms[roomId].currentState.removeListener(RoomStateEvent.Members, this.onRoomMember);\n }\n delete this.rooms[roomId];\n }\n\n /**\n * Retrieve a summary of all the rooms.\n * @returns A summary of each room.\n */\n public getRoomSummaries(): RoomSummary[] {\n return Object.values(this.rooms).map(function (room) {\n return room.summary!;\n });\n }\n\n /**\n * Store a User.\n * @param user - The user to store.\n */\n public storeUser(user: User): void {\n this.users[user.userId] = user;\n }\n\n /**\n * Retrieve a User by its' user ID.\n * @param userId - The user ID.\n * @returns The user or null.\n */\n public getUser(userId: string): User | null {\n return this.users[userId] || null;\n }\n\n /**\n * Retrieve all known users.\n * @returns A list of users, which may be empty.\n */\n public getUsers(): User[] {\n return Object.values(this.users);\n }\n\n /**\n * Retrieve scrollback for this room.\n * @param room - The matrix room\n * @param limit - The max number of old events to retrieve.\n * @returns An array of objects which will be at most 'limit'\n * length and at least 0. The objects are the raw event JSON.\n */\n public scrollback(room: Room, limit: number): MatrixEvent[] {\n return [];\n }\n\n /**\n * Store events for a room. The events have already been added to the timeline\n * @param room - The room to store events for.\n * @param events - The events to store.\n * @param token - The token associated with these events.\n * @param toStart - True if these are paginated results.\n */\n public storeEvents(room: Room, events: MatrixEvent[], token: string | null, toStart: boolean): void {\n // no-op because they've already been added to the room instance.\n }\n\n /**\n * Store a filter.\n */\n public storeFilter(filter: Filter): void {\n if (!filter?.userId || !filter?.filterId) return;\n this.filters.getOrCreate(filter.userId).set(filter.filterId, filter);\n }\n\n /**\n * Retrieve a filter.\n * @returns A filter or null.\n */\n public getFilter(userId: string, filterId: string): Filter | null {\n return this.filters.get(userId)?.get(filterId) || null;\n }\n\n /**\n * Retrieve a filter ID with the given name.\n * @param filterName - The filter name.\n * @returns The filter ID or null.\n */\n public getFilterIdByName(filterName: string): string | null {\n if (!this.localStorage) {\n return null;\n }\n const key = \"mxjssdk_memory_filter_\" + filterName;\n // XXX Storage.getItem doesn't throw ...\n // or are we using something different\n // than window.localStorage in some cases\n // that does throw?\n // that would be very naughty\n try {\n const value = this.localStorage.getItem(key);\n if (isValidFilterId(value)) {\n return value;\n }\n } catch {}\n return null;\n }\n\n /**\n * Set a filter name to ID mapping.\n */\n public setFilterIdByName(filterName: string, filterId?: string): void {\n if (!this.localStorage) {\n return;\n }\n const key = \"mxjssdk_memory_filter_\" + filterName;\n try {\n if (isValidFilterId(filterId)) {\n this.localStorage.setItem(key, filterId!);\n } else {\n this.localStorage.removeItem(key);\n }\n } catch {}\n }\n\n /**\n * Store user-scoped account data events.\n * N.B. that account data only allows a single event per type, so multiple\n * events with the same type will replace each other.\n * @param events - The events to store.\n */\n public storeAccountDataEvents(events: MatrixEvent[]): void {\n events.forEach((event) => {\n // MSC3391: an event with content of {} should be interpreted as deleted\n const isDeleted = !Object.keys(event.getContent()).length;\n if (isDeleted) {\n this.accountData.delete(event.getType());\n } else {\n this.accountData.set(event.getType(), event);\n }\n });\n }\n\n /**\n * Get account data event by event type\n * @param eventType - The event type being queried\n * @returns the user account_data event of given type, if any\n */\n public getAccountData(eventType: EventType | string): MatrixEvent | undefined {\n return this.accountData.get(eventType);\n }\n\n /**\n * setSyncData does nothing as there is no backing data store.\n *\n * @param syncData - The sync data\n * @returns An immediately resolved promise.\n */\n public setSyncData(syncData: ISyncResponse): Promise<void> {\n return Promise.resolve();\n }\n\n /**\n * We never want to save becase we have nothing to save to.\n *\n * @returns If the store wants to save\n */\n public wantsSave(): boolean {\n return false;\n }\n\n /**\n * Save does nothing as there is no backing data store.\n * @param force - True to force a save (but the memory\n * store still can't save anything)\n */\n public save(force: boolean): Promise<void> {\n return Promise.resolve();\n }\n\n /**\n * Startup does nothing as this store doesn't require starting up.\n * @returns An immediately resolved promise.\n */\n public startup(): Promise<void> {\n return Promise.resolve();\n }\n\n /**\n * @returns Promise which resolves with a sync response to restore the\n * client state to where it was at the last save, or null if there\n * is no saved sync data.\n */\n public getSavedSync(): Promise<ISavedSync | null> {\n return Promise.resolve(null);\n }\n\n /**\n * @returns If there is a saved sync, the nextBatch token\n * for this sync, otherwise null.\n */\n public getSavedSyncToken(): Promise<string | null> {\n return Promise.resolve(null);\n }\n\n /**\n * Delete all data from this store.\n * @returns An immediately resolved promise.\n */\n public deleteAllData(): Promise<void> {\n this.rooms = {\n // roomId: Room\n };\n this.users = {\n // userId: User\n };\n this.syncToken = null;\n this.filters = new MapWithDefault(() => new Map());\n this.accountData = new Map(); // type : content\n return Promise.resolve();\n }\n\n /**\n * Returns the out-of-band membership events for this room that\n * were previously loaded.\n * @returns the events, potentially an empty array if OOB loading didn't yield any new members\n * @returns in case the members for this room haven't been stored yet\n */\n public getOutOfBandMembers(roomId: string): Promise<IStateEventWithRoomId[] | null> {\n return Promise.resolve(this.oobMembers.get(roomId) || null);\n }\n\n /**\n * Stores the out-of-band membership events for this room. Note that\n * it still makes sense to store an empty array as the OOB status for the room is\n * marked as fetched, and getOutOfBandMembers will return an empty array instead of null\n * @param membershipEvents - the membership events to store\n * @returns when all members have been stored\n */\n public setOutOfBandMembers(roomId: string, membershipEvents: IStateEventWithRoomId[]): Promise<void> {\n this.oobMembers.set(roomId, membershipEvents);\n return Promise.resolve();\n }\n\n public clearOutOfBandMembers(roomId: string): Promise<void> {\n this.oobMembers.delete(roomId);\n return Promise.resolve();\n }\n\n public getClientOptions(): Promise<IStoredClientOpts | undefined> {\n return Promise.resolve(this.clientOptions);\n }\n\n public storeClientOptions(options: IStoredClientOpts): Promise<void> {\n this.clientOptions = Object.assign({}, options);\n return Promise.resolve();\n }\n\n public async getPendingEvents(roomId: string): Promise<Partial<IEvent>[]> {\n return this.pendingEvents[roomId] ?? [];\n }\n\n public async setPendingEvents(roomId: string, events: Partial<IEvent>[]): Promise<void> {\n this.pendingEvents[roomId] = events;\n }\n\n public saveToDeviceBatches(batches: ToDeviceBatchWithTxnId[]): Promise<void> {\n for (const batch of batches) {\n this.pendingToDeviceBatches.push({\n id: this.nextToDeviceBatchId++,\n eventType: batch.eventType,\n txnId: batch.txnId,\n batch: batch.batch,\n });\n }\n return Promise.resolve();\n }\n\n public async getOldestToDeviceBatch(): Promise<IndexedToDeviceBatch | null> {\n if (this.pendingToDeviceBatches.length === 0) return null;\n return this.pendingToDeviceBatches[0];\n }\n\n public removeToDeviceBatch(id: number): Promise<void> {\n this.pendingToDeviceBatches = this.pendingToDeviceBatches.filter((batch) => batch.id !== id);\n return Promise.resolve();\n }\n\n public async getUserProfile(userId: string): Promise<SyncUserProfile | undefined> {\n return this.userProfiles.get(userId);\n }\n\n public async storeUserProfiles(userProfiles: Map<string, SyncUserProfile>): Promise<void> {\n userProfiles.forEach((profile, userId) => this.userProfiles.set(userId, profile));\n }\n\n public async removeUserProfiles(userIds: string[]): Promise<void> {\n userIds.forEach((userId) => this.userProfiles.delete(userId));\n }\n\n public async destroy(): Promise<void> {\n // Nothing to do\n }\n}\n"],"mappings":"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","ignoreList":[]}
|
|
1
|
+
{"version":3,"file":"memory.js","names":["RoomStateEvent","MapWithDefault","KnownMembership","isValidFilterId","filterId","isValidStr","MemoryStore","constructor","opts","arguments","length","undefined","_defineProperty","Map","event","state","member","_this$createUser","membership","Invite","user","users","userId","createUser","call","name","setDisplayName","events","setRawDisplayName","getDirectionalContent","displayname","getContent","avatar_url","setAvatarUrl","localStorage","getSyncToken","syncToken","isNewlyCreated","Promise","resolve","setSyncToken","token","storeRoom","room","rooms","roomId","currentState","on","Members","onRoomMember","getMembers","forEach","m","setUserCreator","creator","getRoom","getRooms","Object","values","removeRoom","removeListener","getRoomSummaries","map","summary","storeUser","getUser","getUsers","scrollback","limit","storeEvents","toStart","storeFilter","filter","filters","getOrCreate","set","getFilter","_this$filters$get","get","getFilterIdByName","filterName","key","value","getItem","_unused","setFilterIdByName","setItem","removeItem","_unused2","storeAccountDataEvents","isDeleted","keys","accountData","delete","getType","getAccountData","eventType","setSyncData","syncData","wantsSave","save","force","startup","getSavedSync","getSavedSyncToken","deleteAllData","getOutOfBandMembers","oobMembers","setOutOfBandMembers","membershipEvents","clearOutOfBandMembers","getClientOptions","clientOptions","storeClientOptions","options","assign","getPendingEvents","_this","_asyncToGenerator","_this$pendingEvents$r","pendingEvents","setPendingEvents","_this2","saveToDeviceBatches","batches","batch","pendingToDeviceBatches","push","id","nextToDeviceBatchId","txnId","getOldestToDeviceBatch","_this3","removeToDeviceBatch","getUserProfile","_this4","userProfiles","storeUserProfiles","_this5","profile","removeUserProfiles","userIds","_this6","destroy"],"sources":["../../src/store/memory.ts"],"sourcesContent":["/*\nCopyright 2015 - 2021 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\n/**\n * This is an internal module. See {@link MemoryStore} for the public class.\n */\n\nimport { type EventType } from \"../@types/event.ts\";\nimport { type Room } from \"../models/room.ts\";\nimport { type SyncUserProfile, type User } from \"../models/user.ts\";\nimport { type IEvent, type MatrixEvent } from \"../models/event.ts\";\nimport { type RoomState, RoomStateEvent } from \"../models/room-state.ts\";\nimport { type RoomMember } from \"../models/room-member.ts\";\nimport { type Filter } from \"../filter.ts\";\nimport { type ISavedSync, type IStore, type UserCreator } from \"./index.ts\";\nimport { type RoomSummary } from \"../models/room-summary.ts\";\nimport { type ISyncResponse } from \"../sync-accumulator.ts\";\nimport { type IStateEventWithRoomId } from \"../@types/search.ts\";\nimport { type IndexedToDeviceBatch, type ToDeviceBatchWithTxnId } from \"../models/ToDeviceMessage.ts\";\nimport { type IStoredClientOpts } from \"../client.ts\";\nimport { MapWithDefault } from \"../utils.ts\";\nimport { KnownMembership } from \"../@types/membership.ts\";\n\nfunction isValidFilterId(filterId?: string | number | null): boolean {\n const isValidStr =\n typeof filterId === \"string\" &&\n !!filterId &&\n filterId !== \"undefined\" && // exclude these as we've serialized undefined in localStorage before\n filterId !== \"null\";\n\n return isValidStr || typeof filterId === \"number\";\n}\n\nexport interface IOpts {\n /** The local storage instance to persist some forms of data such as tokens. Rooms will NOT be stored. */\n localStorage?: Storage;\n}\n\nexport class MemoryStore implements IStore {\n private rooms: Record<string, Room> = {}; // roomId: Room\n private users: Record<string, User> = {}; // userId: User\n private syncToken: string | null = null;\n // userId: {\n // filterId: Filter\n // }\n private filters: MapWithDefault<string, Map<string, Filter>> = new MapWithDefault(() => new Map());\n public accountData: Map<string, MatrixEvent> = new Map(); // type: content\n protected readonly localStorage?: Storage;\n private oobMembers: Map<string, IStateEventWithRoomId[]> = new Map(); // roomId: [member events]\n private pendingEvents: { [roomId: string]: Partial<IEvent>[] } = {};\n private clientOptions?: IStoredClientOpts;\n private pendingToDeviceBatches: IndexedToDeviceBatch[] = [];\n private nextToDeviceBatchId = 0;\n protected createUser?: UserCreator;\n public readonly userProfiles = new Map<string, SyncUserProfile>();\n\n /**\n * Construct a new in-memory data store for the Matrix Client.\n * @param opts - Config options\n */\n public constructor(opts: IOpts = {}) {\n this.localStorage = opts.localStorage;\n }\n\n /**\n * Retrieve the token to stream from.\n * @returns The token or null.\n */\n public getSyncToken(): string | null {\n return this.syncToken;\n }\n\n /** @returns whether or not the database was newly created in this session. */\n public isNewlyCreated(): Promise<boolean> {\n return Promise.resolve(true);\n }\n\n /**\n * Set the token to stream from.\n * @param token - The token to stream from.\n */\n public setSyncToken(token: string): void {\n this.syncToken = token;\n }\n\n /**\n * Store the given room.\n * @param room - The room to be stored. All properties must be stored.\n */\n public storeRoom(room: Room): void {\n this.rooms[room.roomId] = room;\n // add listeners for room member changes so we can keep the room member\n // map up-to-date.\n room.currentState.on(RoomStateEvent.Members, this.onRoomMember);\n // add existing members\n room.currentState.getMembers().forEach((m) => {\n this.onRoomMember(null, room.currentState, m);\n });\n }\n\n public setUserCreator(creator: UserCreator): void {\n this.createUser = creator;\n }\n\n /**\n * Called when a room member in a room being tracked by this store has been\n * updated.\n */\n private onRoomMember = (event: MatrixEvent | null, state: RoomState, member: RoomMember): void => {\n if (member.membership === KnownMembership.Invite) {\n // We do NOT add invited members because people love to typo user IDs\n // which would then show up in these lists (!)\n return;\n }\n\n const user = this.users[member.userId] || this.createUser?.(member.userId);\n if (member.name) {\n user.setDisplayName(member.name);\n if (member.events.member) {\n user.setRawDisplayName(member.events.member.getDirectionalContent().displayname);\n }\n }\n if (member.events.member && member.events.member.getContent().avatar_url) {\n user.setAvatarUrl(member.events.member.getContent().avatar_url);\n }\n this.users[user.userId] = user;\n };\n\n /**\n * Retrieve a room by its' room ID.\n * @param roomId - The room ID.\n * @returns The room or null.\n */\n public getRoom(roomId: string): Room | null {\n return this.rooms[roomId] || null;\n }\n\n /**\n * Retrieve all known rooms.\n * @returns A list of rooms, which may be empty.\n */\n public getRooms(): Room[] {\n return Object.values(this.rooms);\n }\n\n /**\n * Permanently delete a room.\n */\n public removeRoom(roomId: string): void {\n if (this.rooms[roomId]) {\n this.rooms[roomId].currentState.removeListener(RoomStateEvent.Members, this.onRoomMember);\n }\n delete this.rooms[roomId];\n }\n\n /**\n * Retrieve a summary of all the rooms.\n * @returns A summary of each room.\n */\n public getRoomSummaries(): RoomSummary[] {\n return Object.values(this.rooms).map(function (room) {\n return room.summary!;\n });\n }\n\n /**\n * Store a User.\n * @param user - The user to store.\n */\n public storeUser(user: User): void {\n this.users[user.userId] = user;\n }\n\n /**\n * Retrieve a User by its' user ID.\n * @param userId - The user ID.\n * @returns The user or null.\n */\n public getUser(userId: string): User | null {\n return this.users[userId] || null;\n }\n\n /**\n * Retrieve all known users.\n * @returns A list of users, which may be empty.\n */\n public getUsers(): User[] {\n return Object.values(this.users);\n }\n\n /**\n * Retrieve scrollback for this room.\n * @param room - The matrix room\n * @param limit - The max number of old events to retrieve.\n * @returns An array of objects which will be at most 'limit'\n * length and at least 0. The objects are the raw event JSON.\n */\n public scrollback(room: Room, limit: number): MatrixEvent[] {\n return [];\n }\n\n /**\n * Store events for a room. The events have already been added to the timeline\n * @param room - The room to store events for.\n * @param events - The events to store.\n * @param token - The token associated with these events.\n * @param toStart - True if these are paginated results.\n */\n public storeEvents(room: Room, events: MatrixEvent[], token: string | null, toStart: boolean): void {\n // no-op because they've already been added to the room instance.\n }\n\n /**\n * Store a filter.\n */\n public storeFilter(filter: Filter): void {\n if (!filter?.userId || !filter?.filterId) return;\n this.filters.getOrCreate(filter.userId).set(filter.filterId, filter);\n }\n\n /**\n * Retrieve a filter.\n * @returns A filter or null.\n */\n public getFilter(userId: string, filterId: string): Filter | null {\n return this.filters.get(userId)?.get(filterId) || null;\n }\n\n /**\n * Retrieve a filter ID with the given name.\n * @param filterName - The filter name.\n * @returns The filter ID or null.\n */\n public getFilterIdByName(filterName: string): string | null {\n if (!this.localStorage) {\n return null;\n }\n const key = \"mxjssdk_memory_filter_\" + filterName;\n // XXX Storage.getItem doesn't throw ...\n // or are we using something different\n // than window.localStorage in some cases\n // that does throw?\n // that would be very naughty\n try {\n const value = this.localStorage.getItem(key);\n if (isValidFilterId(value)) {\n return value;\n }\n } catch {}\n return null;\n }\n\n /**\n * Set a filter name to ID mapping.\n */\n public setFilterIdByName(filterName: string, filterId?: string): void {\n if (!this.localStorage) {\n return;\n }\n const key = \"mxjssdk_memory_filter_\" + filterName;\n try {\n if (isValidFilterId(filterId)) {\n this.localStorage.setItem(key, filterId!);\n } else {\n this.localStorage.removeItem(key);\n }\n } catch {}\n }\n\n /**\n * Store user-scoped account data events.\n * N.B. that account data only allows a single event per type, so multiple\n * events with the same type will replace each other.\n * @param events - The events to store.\n */\n public storeAccountDataEvents(events: MatrixEvent[]): void {\n events.forEach((event) => {\n // MSC3391: an event with content of {} should be interpreted as deleted\n const isDeleted = !Object.keys(event.getContent()).length;\n if (isDeleted) {\n this.accountData.delete(event.getType());\n } else {\n this.accountData.set(event.getType(), event);\n }\n });\n }\n\n /**\n * Get account data event by event type\n * @param eventType - The event type being queried\n * @returns the user account_data event of given type, if any\n */\n public getAccountData(eventType: EventType | string): MatrixEvent | undefined {\n return this.accountData.get(eventType);\n }\n\n /**\n * setSyncData does nothing as there is no backing data store.\n *\n * @param syncData - The sync data\n * @returns An immediately resolved promise.\n */\n public setSyncData(syncData: ISyncResponse): Promise<void> {\n return Promise.resolve();\n }\n\n /**\n * We never want to save becase we have nothing to save to.\n *\n * @returns If the store wants to save\n */\n public wantsSave(): boolean {\n return false;\n }\n\n /**\n * Save does nothing as there is no backing data store.\n * @param force - True to force a save (but the memory\n * store still can't save anything)\n */\n public save(force: boolean): Promise<void> {\n return Promise.resolve();\n }\n\n /**\n * Startup does nothing as this store doesn't require starting up.\n * @returns An immediately resolved promise.\n */\n public startup(): Promise<void> {\n return Promise.resolve();\n }\n\n /**\n * @returns Promise which resolves with a sync response to restore the\n * client state to where it was at the last save, or null if there\n * is no saved sync data.\n */\n public getSavedSync(): Promise<ISavedSync | null> {\n return Promise.resolve(null);\n }\n\n /**\n * @returns If there is a saved sync, the nextBatch token\n * for this sync, otherwise null.\n */\n public getSavedSyncToken(): Promise<string | null> {\n return Promise.resolve(null);\n }\n\n /**\n * Delete all data from this store.\n * @returns An immediately resolved promise.\n */\n public deleteAllData(): Promise<void> {\n this.rooms = {\n // roomId: Room\n };\n this.users = {\n // userId: User\n };\n this.syncToken = null;\n this.filters = new MapWithDefault(() => new Map());\n this.accountData = new Map(); // type : content\n return Promise.resolve();\n }\n\n /**\n * Returns the out-of-band membership events for this room that\n * were previously loaded.\n * @returns the events, potentially an empty array if OOB loading didn't yield any new members\n * @returns in case the members for this room haven't been stored yet\n */\n public getOutOfBandMembers(roomId: string): Promise<IStateEventWithRoomId[] | null> {\n return Promise.resolve(this.oobMembers.get(roomId) || null);\n }\n\n /**\n * Stores the out-of-band membership events for this room. Note that\n * it still makes sense to store an empty array as the OOB status for the room is\n * marked as fetched, and getOutOfBandMembers will return an empty array instead of null\n * @param membershipEvents - the membership events to store\n * @returns when all members have been stored\n */\n public setOutOfBandMembers(roomId: string, membershipEvents: IStateEventWithRoomId[]): Promise<void> {\n this.oobMembers.set(roomId, membershipEvents);\n return Promise.resolve();\n }\n\n public clearOutOfBandMembers(roomId: string): Promise<void> {\n this.oobMembers.delete(roomId);\n return Promise.resolve();\n }\n\n public getClientOptions(): Promise<IStoredClientOpts | undefined> {\n return Promise.resolve(this.clientOptions);\n }\n\n public storeClientOptions(options: IStoredClientOpts): Promise<void> {\n this.clientOptions = Object.assign({}, options);\n return Promise.resolve();\n }\n\n public async getPendingEvents(roomId: string): Promise<Partial<IEvent>[]> {\n return this.pendingEvents[roomId] ?? [];\n }\n\n public async setPendingEvents(roomId: string, events: Partial<IEvent>[]): Promise<void> {\n this.pendingEvents[roomId] = events;\n }\n\n public saveToDeviceBatches(batches: ToDeviceBatchWithTxnId[]): Promise<void> {\n for (const batch of batches) {\n this.pendingToDeviceBatches.push({\n id: this.nextToDeviceBatchId++,\n eventType: batch.eventType,\n txnId: batch.txnId,\n batch: batch.batch,\n });\n }\n return Promise.resolve();\n }\n\n public async getOldestToDeviceBatch(): Promise<IndexedToDeviceBatch | null> {\n if (this.pendingToDeviceBatches.length === 0) return null;\n return this.pendingToDeviceBatches[0];\n }\n\n public removeToDeviceBatch(id: number): Promise<void> {\n this.pendingToDeviceBatches = this.pendingToDeviceBatches.filter((batch) => batch.id !== id);\n return Promise.resolve();\n }\n\n public async getUserProfile(userId: string): Promise<SyncUserProfile | undefined> {\n return this.userProfiles.get(userId);\n }\n\n public async storeUserProfiles(userProfiles: Map<string, SyncUserProfile>): Promise<void> {\n userProfiles.forEach((profile, userId) => this.userProfiles.set(userId, profile));\n }\n\n public async removeUserProfiles(userIds: string[]): Promise<void> {\n userIds.forEach((userId) => this.userProfiles.delete(userId));\n }\n\n public async destroy(): Promise<void> {\n // Nothing to do\n }\n}\n"],"mappings":"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","ignoreList":[]}
|
package/lib/sync-accumulator.js
CHANGED
|
@@ -68,7 +68,6 @@ function isTaggedEvent(event) {
|
|
|
68
68
|
export class SyncAccumulator {
|
|
69
69
|
constructor() {
|
|
70
70
|
var opts = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
71
|
-
this.opts = opts;
|
|
72
71
|
_defineProperty(this, "accountData", {});
|
|
73
72
|
// $event_type: Object
|
|
74
73
|
_defineProperty(this, "inviteRooms", {});
|
|
@@ -81,6 +80,7 @@ export class SyncAccumulator {
|
|
|
81
80
|
// coherent /sync response and know at what point they should be
|
|
82
81
|
// streaming from without losing events.
|
|
83
82
|
_defineProperty(this, "nextBatch", null);
|
|
83
|
+
this.opts = opts;
|
|
84
84
|
this.opts.maxTimelineEntries = this.opts.maxTimelineEntries || 50;
|
|
85
85
|
}
|
|
86
86
|
accumulate(syncResponse) {
|
|
@@ -380,9 +380,7 @@ export class SyncAccumulator {
|
|
|
380
380
|
// Prune out any events in our stores that have since expired, do this before we
|
|
381
381
|
// insert new events.
|
|
382
382
|
currentData._stickyEvents = currentData._stickyEvents.filter(_ref2 => {
|
|
383
|
-
var
|
|
384
|
-
expiresTs
|
|
385
|
-
} = _ref2;
|
|
383
|
+
var expiresTs = _ref2.expiresTs;
|
|
386
384
|
return expiresTs > now;
|
|
387
385
|
});
|
|
388
386
|
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"sync-accumulator.js","names":["logger","deepCopy","MAX_STICKY_DURATION_MS","UNREAD_THREAD_NOTIFICATIONS","ReceiptAccumulator","Category","isTaggedEvent","event","undefined","SyncAccumulator","constructor","opts","arguments","length","_defineProperty","maxTimelineEntries","accumulate","syncResponse","fromDatabase","accumulateRooms","accumulateAccountData","nextBatch","next_batch","account_data","events","forEach","e","accountData","type","rooms","invite","Object","keys","roomId","accumulateRoom","Invite","join","Join","leave","Leave","knock","Knock","category","data","knockRooms","accumulateInviteState","accumulateKnockState","inviteRooms","accumulateJoinState","joinRooms","error","invite_state","currentData","hasAdded","i","current","state_key","push","knock_state","_ref","_data","_data$ephemeral","_data$state","_data$orgMatrixMsc","_data$timeline","_data$msc4354_sticky","now","Date","_currentState","create","_timeline","_accountData","_unreadNotifications","_unreadThreadNotifications","_summary","_receipts","_stickyEvents","unread_notifications","stable","unstable","summary","_sum$HEROES_KEY","_sum$JOINED_COUNT_KEY","_sum$INVITED_COUNT_KE","HEROES_KEY","INVITED_COUNT_KEY","JOINED_COUNT_KEY","acc","sum","consumeEphemeralEvents","ephemeral","timeline","limited","state","setState","index","_data$timeline$prev_b","transformedEvent","_e$unsigned","assign","unsigned","age","_localTs","token","prev_batch","filter","_ref2","expiresTs","msc4354_sticky","concat","map","cappedDuration","Math","min","duration_ms","createdTs","origin_server_ts","startIndex","slice","getJSON","forDatabase","_roomData$_stickyEven","roomData","roomJson","evType","receiptEvent","buildAccumulatedReceiptEvent","msgData","rollBackState","timelineEvent","prevStateEvent","prev_content","content","prev_sender","sender","stateKey","ev","accData","roomsData","getNextBatchToken","eventMap"],"sources":["../src/sync-accumulator.ts"],"sourcesContent":["/*\nCopyright 2017 - 2023 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\n/**\n * This is an internal module. See {@link SyncAccumulator} for the public class.\n */\n\nimport { logger } from \"./logger.ts\";\nimport { deepCopy } from \"./utils.ts\";\nimport { MAX_STICKY_DURATION_MS, type IContent, type IUnsigned } from \"./models/event.ts\";\nimport { type IRoomSummary } from \"./models/room-summary.ts\";\nimport { type EventType } from \"./@types/event.ts\";\nimport { UNREAD_THREAD_NOTIFICATIONS } from \"./@types/sync.ts\";\nimport { ReceiptAccumulator } from \"./receipt-accumulator.ts\";\nimport { type OlmEncryptionInfo } from \"./crypto-api/index.ts\";\nimport { type SyncUserProfile } from \"./matrix.ts\";\n\ninterface IOpts {\n /**\n * The ideal maximum number of timeline entries to keep in the sync response.\n * This is best-effort, as clients do not always have a back-pagination token for each event,\n * so it's possible there may be slightly *less* than this value. There will never be more.\n * This cannot be 0 or else it makes it impossible to scroll back in a room.\n * Default: 50.\n */\n maxTimelineEntries?: number;\n /**\n * Whether to use the stable or unstable fields for user profiles.\n */\n profileFieldsStable?: boolean;\n}\n\nexport interface IMinimalEvent {\n content: IContent;\n type: EventType | string;\n room_id?: string;\n unsigned?: IUnsigned;\n}\n\nexport interface IEphemeral {\n events: IMinimalEvent[];\n}\n\n/* eslint-disable camelcase */\ninterface UnreadNotificationCounts {\n highlight_count?: number;\n notification_count?: number;\n}\n\nexport interface IRoomEvent extends IMinimalEvent {\n event_id: string;\n sender: string;\n origin_server_ts: number;\n}\n\nexport interface IStateEvent extends IRoomEvent {\n prev_content?: IContent;\n state_key: string;\n}\n\ninterface IState {\n events: IStateEvent[];\n}\n\nexport interface ITimeline {\n events: Array<IRoomEvent | IStateEvent>;\n limited?: boolean;\n prev_batch: string | null;\n}\n\ntype StickyEventFields = {\n msc4354_sticky: { duration_ms: number };\n content: { msc4354_sticky_key?: string };\n};\n\nexport type IStickyEvent = IRoomEvent & StickyEventFields;\n\nexport type IStickyStateEvent = IStateEvent & StickyEventFields;\n\nexport interface ISticky {\n events: Array<IStickyEvent | IStickyStateEvent>;\n}\n\nexport interface IJoinedRoom {\n \"summary\": IRoomSummary;\n // One of `state` or `state_after` is required.\n \"state\"?: IState;\n \"org.matrix.msc4222.state_after\"?: IState; // https://github.com/matrix-org/matrix-spec-proposals/pull/4222\n \"msc4354_sticky\"?: ISticky; // https://github.com/matrix-org/matrix-spec-proposals/pull/4354\n \"timeline\": ITimeline;\n \"ephemeral\": IEphemeral;\n \"account_data\": IAccountData;\n \"unread_notifications\": UnreadNotificationCounts;\n \"unread_thread_notifications\"?: Record<string, UnreadNotificationCounts>;\n \"org.matrix.msc3773.unread_thread_notifications\"?: Record<string, UnreadNotificationCounts>;\n}\n\nexport interface IStrippedState {\n content: IContent;\n state_key: string;\n type: EventType | string;\n sender: string;\n}\n\nexport interface IInviteState {\n events: IStrippedState[];\n}\n\nexport interface IKnockState {\n events: IStrippedState[];\n}\n\nexport interface IInvitedRoom {\n invite_state: IInviteState;\n}\n\nexport interface ILeftRoom {\n // One of `state` or `state_after` is required.\n \"state\"?: IState;\n \"org.matrix.msc4222.state_after\"?: IState;\n \"timeline\": ITimeline;\n \"account_data\": IAccountData;\n}\n\nexport interface IKnockedRoom {\n knock_state: IKnockState;\n}\n\nexport interface IRooms {\n [Category.Join]: Record<string, IJoinedRoom>;\n [Category.Invite]: Record<string, IInvitedRoom>;\n [Category.Leave]: Record<string, ILeftRoom>;\n [Category.Knock]: Record<string, IKnockedRoom>;\n}\n\ninterface IPresence {\n events: IMinimalEvent[];\n}\n\ninterface IAccountData {\n events: IMinimalEvent[];\n}\n\n/** A to-device message as received from the sync. */\nexport interface IToDeviceEvent {\n content: IContent;\n sender: string;\n type: string;\n}\n\n/**\n * A (possibly decrypted) to-device message after it has been successfully processed by the sdk.\n *\n * If the message was encrypted, the `encryptionInfo` field will contain the encryption information.\n * If the message was sent in clear, this field will be null.\n *\n * The `message` field contains the message `type`, `content`, and `sender` as if the message was sent in clear.\n */\nexport interface ReceivedToDeviceMessage {\n /** The message type, content, and sender as if the message was sent in clear. */\n message: IToDeviceEvent;\n /**\n * Information about the encryption of the message.\n * Will be null if the message was sent in clear\n */\n encryptionInfo: OlmEncryptionInfo | null;\n}\n\ninterface IToDevice {\n events: IToDeviceEvent[];\n}\n\nexport interface IDeviceLists {\n changed?: string[];\n left?: string[];\n}\n\n/**\n * The \"users\" section of the sync update which contains extended profile updates.\n */\nexport interface UsersUpdate {\n [userId: string]: {\n profile_updates?: SyncUserProfile | null;\n };\n}\n\nexport interface ISyncResponse {\n \"next_batch\": string;\n \"rooms\": IRooms;\n \"presence\"?: IPresence;\n \"account_data\": IAccountData;\n \"to_device\"?: IToDevice;\n \"device_lists\"?: IDeviceLists;\n \"device_one_time_keys_count\"?: Record<string, number>;\n \"users\"?: UsersUpdate;\n \"org.matrix.msc4429.users\"?: UsersUpdate;\n \"device_unused_fallback_key_types\"?: string[];\n \"org.matrix.msc2732.device_unused_fallback_key_types\"?: string[];\n}\n/* eslint-enable camelcase */\n\nexport enum Category {\n Invite = \"invite\",\n Leave = \"leave\",\n Join = \"join\",\n Knock = \"knock\",\n}\n\ninterface IRoom {\n _currentState: { [eventType: string]: { [stateKey: string]: IStateEvent } };\n _timeline: {\n event: IRoomEvent | IStateEvent;\n token: string | null;\n }[];\n _summary: Partial<IRoomSummary>;\n _accountData: { [eventType: string]: IMinimalEvent };\n _unreadNotifications: Partial<UnreadNotificationCounts>;\n _unreadThreadNotifications?: Record<string, Partial<UnreadNotificationCounts>>;\n _receipts: ReceiptAccumulator;\n _stickyEvents: {\n readonly event: IStickyEvent | IStickyStateEvent;\n /**\n * This is the timestamp at which point it is safe to remove this event from the store.\n * This value is immutable\n */\n readonly expiresTs: number;\n }[];\n}\n\nexport interface ISyncData {\n nextBatch: string;\n accountData: IMinimalEvent[];\n roomsData: IRooms;\n}\n\ntype TaggedEvent = IRoomEvent & { _localTs?: number };\n\nfunction isTaggedEvent(event: IRoomEvent): event is TaggedEvent {\n return \"_localTs\" in event && event[\"_localTs\"] !== undefined;\n}\n\n/**\n * The purpose of this class is to accumulate /sync responses such that a\n * complete \"initial\" JSON response can be returned which accurately represents\n * the sum total of the /sync responses accumulated to date. It only handles\n * room data: that is, everything under the \"rooms\" top-level key.\n *\n * This class is used when persisting room data so a complete /sync response can\n * be loaded from disk and incremental syncs can be performed on the server,\n * rather than asking the server to do an initial sync on startup.\n */\nexport class SyncAccumulator {\n private accountData: Record<string, IMinimalEvent> = {}; // $event_type: Object\n private inviteRooms: Record<string, IInvitedRoom> = {}; // $roomId: { ... sync 'invite' json data ... }\n private knockRooms: Record<string, IKnockedRoom> = {}; // $roomId: { ... sync 'knock' json data ... }\n private joinRooms: { [roomId: string]: IRoom } = {};\n // the /sync token which corresponds to the last time rooms were\n // accumulated. We remember this so that any caller can obtain a\n // coherent /sync response and know at what point they should be\n // streaming from without losing events.\n private nextBatch: string | null = null;\n\n public constructor(private readonly opts: IOpts = {}) {\n this.opts.maxTimelineEntries = this.opts.maxTimelineEntries || 50;\n }\n\n public accumulate(syncResponse: ISyncResponse, fromDatabase = false): void {\n this.accumulateRooms(syncResponse, fromDatabase);\n this.accumulateAccountData(syncResponse);\n this.nextBatch = syncResponse.next_batch;\n }\n\n private accumulateAccountData(syncResponse: ISyncResponse): void {\n if (!syncResponse.account_data || !syncResponse.account_data.events) {\n return;\n }\n // Clobbers based on event type.\n syncResponse.account_data.events.forEach((e) => {\n this.accountData[e.type] = e;\n });\n }\n\n /**\n * Accumulate incremental /sync room data.\n * @param syncResponse - the complete /sync JSON\n * @param fromDatabase - True if the sync response is one saved to the database\n */\n private accumulateRooms(syncResponse: ISyncResponse, fromDatabase = false): void {\n if (!syncResponse.rooms) {\n return;\n }\n if (syncResponse.rooms.invite) {\n Object.keys(syncResponse.rooms.invite).forEach((roomId) => {\n this.accumulateRoom(roomId, Category.Invite, syncResponse.rooms.invite[roomId], fromDatabase);\n });\n }\n if (syncResponse.rooms.join) {\n Object.keys(syncResponse.rooms.join).forEach((roomId) => {\n this.accumulateRoom(roomId, Category.Join, syncResponse.rooms.join[roomId], fromDatabase);\n });\n }\n if (syncResponse.rooms.leave) {\n Object.keys(syncResponse.rooms.leave).forEach((roomId) => {\n this.accumulateRoom(roomId, Category.Leave, syncResponse.rooms.leave[roomId], fromDatabase);\n });\n }\n if (syncResponse.rooms.knock) {\n Object.keys(syncResponse.rooms.knock).forEach((roomId) => {\n this.accumulateRoom(roomId, Category.Knock, syncResponse.rooms.knock[roomId], fromDatabase);\n });\n }\n }\n\n private accumulateRoom(roomId: string, category: Category.Invite, data: IInvitedRoom, fromDatabase: boolean): void;\n private accumulateRoom(roomId: string, category: Category.Join, data: IJoinedRoom, fromDatabase: boolean): void;\n private accumulateRoom(roomId: string, category: Category.Leave, data: ILeftRoom, fromDatabase: boolean): void;\n private accumulateRoom(roomId: string, category: Category.Knock, data: IKnockedRoom, fromDatabase: boolean): void;\n private accumulateRoom(roomId: string, category: Category, data: any, fromDatabase = false): void {\n // Valid /sync state transitions\n // +--------+ <======+ 1: Accept an invite\n // +== | INVITE | | (5) 2: Leave a room\n // | +--------+ =====+ | 3: Join a public room previously\n // |(1) (4) | | left (handle as if new room)\n // V (2) V | 4: Reject an invite\n // +------+ ========> +--------+ 5: Invite to a room previously\n // | JOIN | (3) | LEAVE* | left (handle as if new room)\n // +------+ <======== +--------+\n //\n // * equivalent to \"no state\"\n switch (category) {\n case Category.Invite: // (5)\n if (this.knockRooms[roomId]) {\n // was previously knock, now invite, need to delete knock state\n delete this.knockRooms[roomId];\n }\n this.accumulateInviteState(roomId, data as IInvitedRoom);\n break;\n\n case Category.Knock:\n this.accumulateKnockState(roomId, data as IKnockedRoom);\n break;\n\n case Category.Join:\n if (this.knockRooms[roomId]) {\n // delete knock state on join\n delete this.knockRooms[roomId];\n } else if (this.inviteRooms[roomId]) {\n // (1)\n // was previously invite, now join. We expect /sync to give\n // the entire state and timeline on 'join', so delete previous\n // invite state\n delete this.inviteRooms[roomId];\n }\n // (3)\n this.accumulateJoinState(roomId, data as IJoinedRoom, fromDatabase);\n break;\n\n case Category.Leave:\n if (this.knockRooms[roomId]) {\n // delete knock state on leave\n delete this.knockRooms[roomId];\n } else if (this.inviteRooms[roomId]) {\n // (4)\n delete this.inviteRooms[roomId];\n } else {\n // (2)\n delete this.joinRooms[roomId];\n }\n break;\n\n default:\n logger.error(\"Unknown cateogory: \", category);\n }\n }\n\n private accumulateInviteState(roomId: string, data: IInvitedRoom): void {\n if (!data.invite_state || !data.invite_state.events) {\n // no new data\n return;\n }\n if (!this.inviteRooms[roomId]) {\n this.inviteRooms[roomId] = {\n invite_state: data.invite_state,\n };\n return;\n }\n // accumulate extra keys for invite->invite transitions\n // clobber based on event type / state key\n // We expect invite_state to be small, so just loop over the events\n const currentData = this.inviteRooms[roomId];\n data.invite_state.events.forEach((e) => {\n let hasAdded = false;\n for (let i = 0; i < currentData.invite_state.events.length; i++) {\n const current = currentData.invite_state.events[i];\n if (current.type === e.type && current.state_key == e.state_key) {\n currentData.invite_state.events[i] = e; // update\n hasAdded = true;\n }\n }\n if (!hasAdded) {\n currentData.invite_state.events.push(e);\n }\n });\n }\n\n private accumulateKnockState(roomId: string, data: IKnockedRoom): void {\n if (!data.knock_state || !data.knock_state.events) {\n // no new data\n return;\n }\n if (!this.knockRooms[roomId]) {\n this.knockRooms[roomId] = {\n knock_state: data.knock_state,\n };\n return;\n }\n // accumulate extra keys\n // clobber based on event type / state key\n // We expect knock_state to be small, so just loop over the events\n const currentData = this.knockRooms[roomId];\n data.knock_state.events.forEach((e) => {\n let hasAdded = false;\n for (let i = 0; i < currentData.knock_state.events.length; i++) {\n const current = currentData.knock_state.events[i];\n if (current.type === e.type && current.state_key == e.state_key) {\n currentData.knock_state.events[i] = e; // update\n hasAdded = true;\n }\n }\n if (!hasAdded) {\n currentData.knock_state.events.push(e);\n }\n });\n }\n\n // Accumulate timeline and state events in a room.\n private accumulateJoinState(roomId: string, data: IJoinedRoom, fromDatabase = false): void {\n const now = Date.now();\n // We expect this function to be called a lot (every /sync) so we want\n // this to be fast. /sync stores events in an array but we often want\n // to clobber based on type/state_key. Rather than convert arrays to\n // maps all the time, just keep private maps which contain\n // the actual current accumulated sync state, and array-ify it when\n // getJSON() is called.\n\n // State resolution:\n // The 'state' key is the delta from the previous sync (or start of time\n // if no token was supplied), to the START of the timeline. To obtain\n // the current state, we need to \"roll forward\" state by reading the\n // timeline. We want to store the current state so we can drop events\n // out the end of the timeline based on opts.maxTimelineEntries.\n //\n // 'state' 'timeline' current state\n // |-------x<======================>x\n // T I M E\n //\n // When getJSON() is called, we 'roll back' the current state by the\n // number of entries in the timeline to work out what 'state' should be.\n\n // Back-pagination:\n // On an initial /sync, the server provides a back-pagination token for\n // the start of the timeline. When /sync deltas come down, they also\n // include back-pagination tokens for the start of the timeline. This\n // means not all events in the timeline have back-pagination tokens, as\n // it is only the ones at the START of the timeline which have them.\n // In order for us to have a valid timeline (and back-pagination token\n // to match), we need to make sure that when we remove old timeline\n // events, that we roll forward to an event which has a back-pagination\n // token. This means we can't keep a strict sliding-window based on\n // opts.maxTimelineEntries, and we may have a few less. We should never\n // have more though, provided that the /sync limit is less than or equal\n // to opts.maxTimelineEntries.\n\n if (!this.joinRooms[roomId]) {\n // Create truly empty objects so event types of 'hasOwnProperty' and co\n // don't cause this code to break.\n this.joinRooms[roomId] = {\n _currentState: Object.create(null),\n _timeline: [],\n _accountData: Object.create(null),\n _unreadNotifications: {},\n _unreadThreadNotifications: {},\n _summary: {},\n _receipts: new ReceiptAccumulator(),\n _stickyEvents: [],\n };\n }\n const currentData = this.joinRooms[roomId];\n\n if (data.account_data && data.account_data.events) {\n // clobber based on type\n data.account_data.events.forEach((e) => {\n currentData._accountData[e.type] = e;\n });\n }\n\n // these probably clobber, spec is unclear.\n if (data.unread_notifications) {\n currentData._unreadNotifications = data.unread_notifications;\n }\n currentData._unreadThreadNotifications =\n data[UNREAD_THREAD_NOTIFICATIONS.stable!] ?? data[UNREAD_THREAD_NOTIFICATIONS.unstable!] ?? undefined;\n\n if (data.summary) {\n const HEROES_KEY = \"m.heroes\";\n const INVITED_COUNT_KEY = \"m.invited_member_count\";\n const JOINED_COUNT_KEY = \"m.joined_member_count\";\n\n const acc = currentData._summary;\n const sum = data.summary;\n acc[HEROES_KEY] = sum[HEROES_KEY] ?? acc[HEROES_KEY];\n acc[JOINED_COUNT_KEY] = sum[JOINED_COUNT_KEY] ?? acc[JOINED_COUNT_KEY];\n acc[INVITED_COUNT_KEY] = sum[INVITED_COUNT_KEY] ?? acc[INVITED_COUNT_KEY];\n }\n\n // We purposefully do not persist m.typing events.\n // Technically you could refresh a browser before the timer on a\n // typing event is up, so it'll look like you aren't typing when\n // you really still are. However, the alternative is worse. If\n // we do persist typing events, it will look like people are\n // typing forever until someone really does start typing (which\n // will prompt Synapse to send down an actual m.typing event to\n // clobber the one we persisted).\n\n // Persist the receipts\n currentData._receipts.consumeEphemeralEvents(data.ephemeral?.events);\n\n // if we got a limited sync, we need to remove all timeline entries or else\n // we will have gaps in the timeline.\n if (data.timeline && data.timeline.limited) {\n currentData._timeline = [];\n }\n\n // Work out the current state. The deltas need to be applied in the order:\n // - existing state which didn't come down /sync.\n // - State events under the 'state' key.\n // - State events under the 'state_after' key OR state events in the 'timeline' if 'state_after' is not present.\n data.state?.events?.forEach((e) => {\n setState(currentData._currentState, e);\n });\n data[\"org.matrix.msc4222.state_after\"]?.events?.forEach((e) => {\n setState(currentData._currentState, e);\n });\n data.timeline?.events?.forEach((e, index) => {\n if (!data[\"org.matrix.msc4222.state_after\"]) {\n // this nops if 'e' isn't a state event\n setState(currentData._currentState, e);\n }\n // append the event to the timeline. The back-pagination token\n // corresponds to the first event in the timeline\n let transformedEvent: TaggedEvent;\n if (!fromDatabase) {\n transformedEvent = Object.assign({}, e);\n if (transformedEvent.unsigned !== undefined) {\n transformedEvent.unsigned = Object.assign({}, transformedEvent.unsigned);\n }\n const age = e.unsigned?.age;\n if (age !== undefined) transformedEvent._localTs = Date.now() - age;\n } else {\n transformedEvent = e;\n }\n\n currentData._timeline.push({\n event: transformedEvent,\n token: index === 0 ? (data.timeline.prev_batch ?? null) : null,\n });\n });\n\n // Prune out any events in our stores that have since expired, do this before we\n // insert new events.\n currentData._stickyEvents = currentData._stickyEvents.filter(({ expiresTs }) => expiresTs > now);\n\n // We want this to be fast, so don't worry about duplicate events here. The RoomStickyEventsStore will\n // process these events into the correct mapped order.\n if (data.msc4354_sticky?.events) {\n currentData._stickyEvents = currentData._stickyEvents.concat(\n data.msc4354_sticky.events.map((event) => {\n // If `duration_ms` exceeds the spec limit of a hour, we cap it.\n const cappedDuration = Math.min(event.msc4354_sticky.duration_ms, MAX_STICKY_DURATION_MS);\n // If `origin_server_ts` claims to have been from the future, we still bound it to now.\n const createdTs = Math.min(event.origin_server_ts, now);\n return {\n event,\n expiresTs: cappedDuration + createdTs,\n };\n }),\n );\n }\n\n // attempt to prune the timeline by jumping between events which have\n // pagination tokens.\n if (currentData._timeline.length > this.opts.maxTimelineEntries!) {\n const startIndex = currentData._timeline.length - this.opts.maxTimelineEntries!;\n for (let i = startIndex; i < currentData._timeline.length; i++) {\n if (currentData._timeline[i].token) {\n // keep all events after this, including this one\n currentData._timeline = currentData._timeline.slice(i, currentData._timeline.length);\n break;\n }\n }\n }\n }\n\n /**\n * Return everything under the 'rooms' key from a /sync response which\n * represents all room data that should be stored. This should be paired\n * with the sync token which represents the most recent /sync response\n * provided to accumulate().\n * @param forDatabase - True to generate a sync to be saved to storage\n * @returns An object with a \"nextBatch\", \"roomsData\" and \"accountData\"\n * keys.\n * The \"nextBatch\" key is a string which represents at what point in the\n * /sync stream the accumulator reached. This token should be used when\n * restarting a /sync stream at startup. Failure to do so can lead to missing\n * events. The \"roomsData\" key is an Object which represents the entire\n * /sync response from the 'rooms' key onwards. The \"accountData\" key is\n * a list of raw events which represent global account data.\n */\n public getJSON(forDatabase = false): ISyncData {\n const data: IRooms = {\n join: {},\n invite: {},\n knock: {},\n // always empty. This is set by /sync when a room was previously\n // in 'invite' or 'join'. On fresh startup, the client won't know\n // about any previous room being in 'invite' or 'join' so we can\n // just omit mentioning it at all, even if it has previously come\n // down /sync.\n // The notable exception is when a client is kicked or banned:\n // we may want to hold onto that room so the client can clearly see\n // why their room has disappeared. We don't persist it though because\n // it is unclear *when* we can safely remove the room from the DB.\n // Instead, we assume that if you're loading from the DB, you've\n // refreshed the page, which means you've seen the kick/ban already.\n leave: {},\n };\n Object.keys(this.inviteRooms).forEach((roomId) => {\n data.invite[roomId] = this.inviteRooms[roomId];\n });\n Object.keys(this.knockRooms).forEach((roomId) => {\n data.knock[roomId] = this.knockRooms[roomId];\n });\n Object.keys(this.joinRooms).forEach((roomId) => {\n const roomData = this.joinRooms[roomId];\n const roomJson: IJoinedRoom & {\n // We track both `state` and `state_after` for downgrade compatibility\n \"state\": IState;\n \"org.matrix.msc4222.state_after\": IState;\n } = {\n \"ephemeral\": { events: [] },\n \"account_data\": { events: [] },\n \"state\": { events: [] },\n \"org.matrix.msc4222.state_after\": { events: [] },\n \"timeline\": {\n events: [],\n prev_batch: null,\n },\n \"unread_notifications\": roomData._unreadNotifications,\n \"unread_thread_notifications\": roomData._unreadThreadNotifications,\n \"summary\": roomData._summary as IRoomSummary,\n \"msc4354_sticky\": roomData._stickyEvents?.length\n ? {\n events: roomData._stickyEvents.map((e) => e.event),\n }\n : undefined,\n };\n // Add account data\n Object.keys(roomData._accountData).forEach((evType) => {\n roomJson.account_data.events.push(roomData._accountData[evType]);\n });\n\n const receiptEvent = roomData._receipts.buildAccumulatedReceiptEvent(roomId);\n\n // add only if we have some receipt data\n if (receiptEvent) {\n roomJson.ephemeral.events.push(receiptEvent);\n }\n\n // Add timeline data\n roomData._timeline.forEach((msgData) => {\n if (!roomJson.timeline.prev_batch) {\n // the first event we add to the timeline MUST match up to\n // the prev_batch token.\n if (!msgData.token) {\n return; // this shouldn't happen as we prune constantly.\n }\n roomJson.timeline.prev_batch = msgData.token;\n }\n\n let transformedEvent: (IRoomEvent | IStateEvent) & { _localTs?: number };\n if (!forDatabase && isTaggedEvent(msgData.event)) {\n // This means we have to copy each event, so we can fix it up to\n // set a correct 'age' parameter whilst keeping the local timestamp\n // on our stored event. If this turns out to be a bottleneck, it could\n // be optimised either by doing this in the main process after the data\n // has been structured-cloned to go between the worker & main process,\n // or special-casing data from saved syncs to read the local timestamp\n // directly rather than turning it into age to then immediately be\n // transformed back again into a local timestamp.\n transformedEvent = Object.assign({}, msgData.event);\n if (transformedEvent.unsigned !== undefined) {\n transformedEvent.unsigned = Object.assign({}, transformedEvent.unsigned);\n }\n delete transformedEvent._localTs;\n transformedEvent.unsigned = transformedEvent.unsigned || {};\n transformedEvent.unsigned.age = Date.now() - msgData.event._localTs!;\n } else {\n transformedEvent = msgData.event;\n }\n roomJson.timeline.events.push(transformedEvent);\n });\n\n // Add state data: roll back current state to the start of timeline,\n // by \"reverse clobbering\" from the end of the timeline to the start.\n // Convert maps back into arrays.\n const rollBackState = Object.create(null);\n for (let i = roomJson.timeline.events.length - 1; i >= 0; i--) {\n const timelineEvent = roomJson.timeline.events[i];\n if (\n (timelineEvent as IStateEvent).state_key === null ||\n (timelineEvent as IStateEvent).state_key === undefined\n ) {\n continue; // not a state event\n }\n // since we're going back in time, we need to use the previous\n // state value else we'll break causality. We don't have the\n // complete previous state event, so we need to create one.\n const prevStateEvent = deepCopy(timelineEvent);\n if (prevStateEvent.unsigned) {\n if (prevStateEvent.unsigned.prev_content) {\n prevStateEvent.content = prevStateEvent.unsigned.prev_content;\n }\n if (prevStateEvent.unsigned.prev_sender) {\n prevStateEvent.sender = prevStateEvent.unsigned.prev_sender;\n }\n }\n setState(rollBackState, prevStateEvent);\n }\n Object.keys(roomData._currentState).forEach((evType) => {\n Object.keys(roomData._currentState[evType]).forEach((stateKey) => {\n let ev = roomData._currentState[evType][stateKey];\n // Push to both fields to provide downgrade compatibility in the sync accumulator db\n // the code will prefer `state_after` if it is present\n roomJson[\"org.matrix.msc4222.state_after\"].events.push(ev);\n // Roll the state back to the value at the start of the timeline if it was changed\n if (rollBackState[evType] && rollBackState[evType][stateKey]) {\n ev = rollBackState[evType][stateKey];\n }\n roomJson.state.events.push(ev);\n });\n });\n data.join[roomId] = roomJson;\n });\n\n // Add account data\n const accData: IMinimalEvent[] = [];\n Object.keys(this.accountData).forEach((evType) => {\n accData.push(this.accountData[evType]);\n });\n\n return {\n nextBatch: this.nextBatch!,\n roomsData: data,\n accountData: accData,\n };\n }\n\n public getNextBatchToken(): string {\n return this.nextBatch!;\n }\n}\n\nfunction setState(eventMap: Record<string, Record<string, IStateEvent>>, event: IRoomEvent | IStateEvent): void {\n if ((event as IStateEvent).state_key === null || (event as IStateEvent).state_key === undefined || !event.type) {\n return;\n }\n if (!eventMap[event.type]) {\n eventMap[event.type] = Object.create(null);\n }\n eventMap[event.type][(event as IStateEvent).state_key] = event as IStateEvent;\n}\n"],"mappings":"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","ignoreList":[]}
|
|
1
|
+
{"version":3,"file":"sync-accumulator.js","names":["logger","deepCopy","MAX_STICKY_DURATION_MS","UNREAD_THREAD_NOTIFICATIONS","ReceiptAccumulator","Category","isTaggedEvent","event","undefined","SyncAccumulator","constructor","opts","arguments","length","_defineProperty","maxTimelineEntries","accumulate","syncResponse","fromDatabase","accumulateRooms","accumulateAccountData","nextBatch","next_batch","account_data","events","forEach","e","accountData","type","rooms","invite","Object","keys","roomId","accumulateRoom","Invite","join","Join","leave","Leave","knock","Knock","category","data","knockRooms","accumulateInviteState","accumulateKnockState","inviteRooms","accumulateJoinState","joinRooms","error","invite_state","currentData","hasAdded","i","current","state_key","push","knock_state","_ref","_data","_data$ephemeral","_data$state","_data$orgMatrixMsc","_data$timeline","_data$msc4354_sticky","now","Date","_currentState","create","_timeline","_accountData","_unreadNotifications","_unreadThreadNotifications","_summary","_receipts","_stickyEvents","unread_notifications","stable","unstable","summary","_sum$HEROES_KEY","_sum$JOINED_COUNT_KEY","_sum$INVITED_COUNT_KE","HEROES_KEY","INVITED_COUNT_KEY","JOINED_COUNT_KEY","acc","sum","consumeEphemeralEvents","ephemeral","timeline","limited","state","setState","index","_data$timeline$prev_b","transformedEvent","_e$unsigned","assign","unsigned","age","_localTs","token","prev_batch","filter","_ref2","expiresTs","msc4354_sticky","concat","map","cappedDuration","Math","min","duration_ms","createdTs","origin_server_ts","startIndex","slice","getJSON","forDatabase","_roomData$_stickyEven","roomData","roomJson","evType","receiptEvent","buildAccumulatedReceiptEvent","msgData","rollBackState","timelineEvent","prevStateEvent","prev_content","content","prev_sender","sender","stateKey","ev","accData","roomsData","getNextBatchToken","eventMap"],"sources":["../src/sync-accumulator.ts"],"sourcesContent":["/*\nCopyright 2017 - 2023 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\n/**\n * This is an internal module. See {@link SyncAccumulator} for the public class.\n */\n\nimport { logger } from \"./logger.ts\";\nimport { deepCopy } from \"./utils.ts\";\nimport { MAX_STICKY_DURATION_MS, type IContent, type IUnsigned } from \"./models/event.ts\";\nimport { type IRoomSummary } from \"./models/room-summary.ts\";\nimport { type EventType } from \"./@types/event.ts\";\nimport { UNREAD_THREAD_NOTIFICATIONS } from \"./@types/sync.ts\";\nimport { ReceiptAccumulator } from \"./receipt-accumulator.ts\";\nimport { type OlmEncryptionInfo } from \"./crypto-api/index.ts\";\nimport { type SyncUserProfile } from \"./matrix.ts\";\n\ninterface IOpts {\n /**\n * The ideal maximum number of timeline entries to keep in the sync response.\n * This is best-effort, as clients do not always have a back-pagination token for each event,\n * so it's possible there may be slightly *less* than this value. There will never be more.\n * This cannot be 0 or else it makes it impossible to scroll back in a room.\n * Default: 50.\n */\n maxTimelineEntries?: number;\n /**\n * Whether to use the stable or unstable fields for user profiles.\n */\n profileFieldsStable?: boolean;\n}\n\nexport interface IMinimalEvent {\n content: IContent;\n type: EventType | string;\n room_id?: string;\n unsigned?: IUnsigned;\n}\n\nexport interface IEphemeral {\n events: IMinimalEvent[];\n}\n\n/* eslint-disable camelcase */\ninterface UnreadNotificationCounts {\n highlight_count?: number;\n notification_count?: number;\n}\n\nexport interface IRoomEvent extends IMinimalEvent {\n event_id: string;\n sender: string;\n origin_server_ts: number;\n}\n\nexport interface IStateEvent extends IRoomEvent {\n prev_content?: IContent;\n state_key: string;\n}\n\ninterface IState {\n events: IStateEvent[];\n}\n\nexport interface ITimeline {\n events: Array<IRoomEvent | IStateEvent>;\n limited?: boolean;\n prev_batch: string | null;\n}\n\ntype StickyEventFields = {\n msc4354_sticky: { duration_ms: number };\n content: { msc4354_sticky_key?: string };\n};\n\nexport type IStickyEvent = IRoomEvent & StickyEventFields;\n\nexport type IStickyStateEvent = IStateEvent & StickyEventFields;\n\nexport interface ISticky {\n events: Array<IStickyEvent | IStickyStateEvent>;\n}\n\nexport interface IJoinedRoom {\n \"summary\": IRoomSummary;\n // One of `state` or `state_after` is required.\n \"state\"?: IState;\n \"org.matrix.msc4222.state_after\"?: IState; // https://github.com/matrix-org/matrix-spec-proposals/pull/4222\n \"msc4354_sticky\"?: ISticky; // https://github.com/matrix-org/matrix-spec-proposals/pull/4354\n \"timeline\": ITimeline;\n \"ephemeral\": IEphemeral;\n \"account_data\": IAccountData;\n \"unread_notifications\": UnreadNotificationCounts;\n \"unread_thread_notifications\"?: Record<string, UnreadNotificationCounts>;\n \"org.matrix.msc3773.unread_thread_notifications\"?: Record<string, UnreadNotificationCounts>;\n}\n\nexport interface IStrippedState {\n content: IContent;\n state_key: string;\n type: EventType | string;\n sender: string;\n}\n\nexport interface IInviteState {\n events: IStrippedState[];\n}\n\nexport interface IKnockState {\n events: IStrippedState[];\n}\n\nexport interface IInvitedRoom {\n invite_state: IInviteState;\n}\n\nexport interface ILeftRoom {\n // One of `state` or `state_after` is required.\n \"state\"?: IState;\n \"org.matrix.msc4222.state_after\"?: IState;\n \"timeline\": ITimeline;\n \"account_data\": IAccountData;\n}\n\nexport interface IKnockedRoom {\n knock_state: IKnockState;\n}\n\nexport interface IRooms {\n [Category.Join]: Record<string, IJoinedRoom>;\n [Category.Invite]: Record<string, IInvitedRoom>;\n [Category.Leave]: Record<string, ILeftRoom>;\n [Category.Knock]: Record<string, IKnockedRoom>;\n}\n\ninterface IPresence {\n events: IMinimalEvent[];\n}\n\ninterface IAccountData {\n events: IMinimalEvent[];\n}\n\n/** A to-device message as received from the sync. */\nexport interface IToDeviceEvent {\n content: IContent;\n sender: string;\n type: string;\n}\n\n/**\n * A (possibly decrypted) to-device message after it has been successfully processed by the sdk.\n *\n * If the message was encrypted, the `encryptionInfo` field will contain the encryption information.\n * If the message was sent in clear, this field will be null.\n *\n * The `message` field contains the message `type`, `content`, and `sender` as if the message was sent in clear.\n */\nexport interface ReceivedToDeviceMessage {\n /** The message type, content, and sender as if the message was sent in clear. */\n message: IToDeviceEvent;\n /**\n * Information about the encryption of the message.\n * Will be null if the message was sent in clear\n */\n encryptionInfo: OlmEncryptionInfo | null;\n}\n\ninterface IToDevice {\n events: IToDeviceEvent[];\n}\n\nexport interface IDeviceLists {\n changed?: string[];\n left?: string[];\n}\n\n/**\n * The \"users\" section of the sync update which contains extended profile updates.\n */\nexport interface UsersUpdate {\n [userId: string]: {\n profile_updates?: SyncUserProfile | null;\n };\n}\n\nexport interface ISyncResponse {\n \"next_batch\": string;\n \"rooms\": IRooms;\n \"presence\"?: IPresence;\n \"account_data\": IAccountData;\n \"to_device\"?: IToDevice;\n \"device_lists\"?: IDeviceLists;\n \"device_one_time_keys_count\"?: Record<string, number>;\n \"users\"?: UsersUpdate;\n \"org.matrix.msc4429.users\"?: UsersUpdate;\n \"device_unused_fallback_key_types\"?: string[];\n \"org.matrix.msc2732.device_unused_fallback_key_types\"?: string[];\n}\n/* eslint-enable camelcase */\n\nexport enum Category {\n Invite = \"invite\",\n Leave = \"leave\",\n Join = \"join\",\n Knock = \"knock\",\n}\n\ninterface IRoom {\n _currentState: { [eventType: string]: { [stateKey: string]: IStateEvent } };\n _timeline: {\n event: IRoomEvent | IStateEvent;\n token: string | null;\n }[];\n _summary: Partial<IRoomSummary>;\n _accountData: { [eventType: string]: IMinimalEvent };\n _unreadNotifications: Partial<UnreadNotificationCounts>;\n _unreadThreadNotifications?: Record<string, Partial<UnreadNotificationCounts>>;\n _receipts: ReceiptAccumulator;\n _stickyEvents: {\n readonly event: IStickyEvent | IStickyStateEvent;\n /**\n * This is the timestamp at which point it is safe to remove this event from the store.\n * This value is immutable\n */\n readonly expiresTs: number;\n }[];\n}\n\nexport interface ISyncData {\n nextBatch: string;\n accountData: IMinimalEvent[];\n roomsData: IRooms;\n}\n\ntype TaggedEvent = IRoomEvent & { _localTs?: number };\n\nfunction isTaggedEvent(event: IRoomEvent): event is TaggedEvent {\n return \"_localTs\" in event && event[\"_localTs\"] !== undefined;\n}\n\n/**\n * The purpose of this class is to accumulate /sync responses such that a\n * complete \"initial\" JSON response can be returned which accurately represents\n * the sum total of the /sync responses accumulated to date. It only handles\n * room data: that is, everything under the \"rooms\" top-level key.\n *\n * This class is used when persisting room data so a complete /sync response can\n * be loaded from disk and incremental syncs can be performed on the server,\n * rather than asking the server to do an initial sync on startup.\n */\nexport class SyncAccumulator {\n private accountData: Record<string, IMinimalEvent> = {}; // $event_type: Object\n private inviteRooms: Record<string, IInvitedRoom> = {}; // $roomId: { ... sync 'invite' json data ... }\n private knockRooms: Record<string, IKnockedRoom> = {}; // $roomId: { ... sync 'knock' json data ... }\n private joinRooms: { [roomId: string]: IRoom } = {};\n // the /sync token which corresponds to the last time rooms were\n // accumulated. We remember this so that any caller can obtain a\n // coherent /sync response and know at what point they should be\n // streaming from without losing events.\n private nextBatch: string | null = null;\n\n public constructor(private readonly opts: IOpts = {}) {\n this.opts.maxTimelineEntries = this.opts.maxTimelineEntries || 50;\n }\n\n public accumulate(syncResponse: ISyncResponse, fromDatabase = false): void {\n this.accumulateRooms(syncResponse, fromDatabase);\n this.accumulateAccountData(syncResponse);\n this.nextBatch = syncResponse.next_batch;\n }\n\n private accumulateAccountData(syncResponse: ISyncResponse): void {\n if (!syncResponse.account_data || !syncResponse.account_data.events) {\n return;\n }\n // Clobbers based on event type.\n syncResponse.account_data.events.forEach((e) => {\n this.accountData[e.type] = e;\n });\n }\n\n /**\n * Accumulate incremental /sync room data.\n * @param syncResponse - the complete /sync JSON\n * @param fromDatabase - True if the sync response is one saved to the database\n */\n private accumulateRooms(syncResponse: ISyncResponse, fromDatabase = false): void {\n if (!syncResponse.rooms) {\n return;\n }\n if (syncResponse.rooms.invite) {\n Object.keys(syncResponse.rooms.invite).forEach((roomId) => {\n this.accumulateRoom(roomId, Category.Invite, syncResponse.rooms.invite[roomId], fromDatabase);\n });\n }\n if (syncResponse.rooms.join) {\n Object.keys(syncResponse.rooms.join).forEach((roomId) => {\n this.accumulateRoom(roomId, Category.Join, syncResponse.rooms.join[roomId], fromDatabase);\n });\n }\n if (syncResponse.rooms.leave) {\n Object.keys(syncResponse.rooms.leave).forEach((roomId) => {\n this.accumulateRoom(roomId, Category.Leave, syncResponse.rooms.leave[roomId], fromDatabase);\n });\n }\n if (syncResponse.rooms.knock) {\n Object.keys(syncResponse.rooms.knock).forEach((roomId) => {\n this.accumulateRoom(roomId, Category.Knock, syncResponse.rooms.knock[roomId], fromDatabase);\n });\n }\n }\n\n private accumulateRoom(roomId: string, category: Category.Invite, data: IInvitedRoom, fromDatabase: boolean): void;\n private accumulateRoom(roomId: string, category: Category.Join, data: IJoinedRoom, fromDatabase: boolean): void;\n private accumulateRoom(roomId: string, category: Category.Leave, data: ILeftRoom, fromDatabase: boolean): void;\n private accumulateRoom(roomId: string, category: Category.Knock, data: IKnockedRoom, fromDatabase: boolean): void;\n private accumulateRoom(roomId: string, category: Category, data: any, fromDatabase = false): void {\n // Valid /sync state transitions\n // +--------+ <======+ 1: Accept an invite\n // +== | INVITE | | (5) 2: Leave a room\n // | +--------+ =====+ | 3: Join a public room previously\n // |(1) (4) | | left (handle as if new room)\n // V (2) V | 4: Reject an invite\n // +------+ ========> +--------+ 5: Invite to a room previously\n // | JOIN | (3) | LEAVE* | left (handle as if new room)\n // +------+ <======== +--------+\n //\n // * equivalent to \"no state\"\n switch (category) {\n case Category.Invite: // (5)\n if (this.knockRooms[roomId]) {\n // was previously knock, now invite, need to delete knock state\n delete this.knockRooms[roomId];\n }\n this.accumulateInviteState(roomId, data as IInvitedRoom);\n break;\n\n case Category.Knock:\n this.accumulateKnockState(roomId, data as IKnockedRoom);\n break;\n\n case Category.Join:\n if (this.knockRooms[roomId]) {\n // delete knock state on join\n delete this.knockRooms[roomId];\n } else if (this.inviteRooms[roomId]) {\n // (1)\n // was previously invite, now join. We expect /sync to give\n // the entire state and timeline on 'join', so delete previous\n // invite state\n delete this.inviteRooms[roomId];\n }\n // (3)\n this.accumulateJoinState(roomId, data as IJoinedRoom, fromDatabase);\n break;\n\n case Category.Leave:\n if (this.knockRooms[roomId]) {\n // delete knock state on leave\n delete this.knockRooms[roomId];\n } else if (this.inviteRooms[roomId]) {\n // (4)\n delete this.inviteRooms[roomId];\n } else {\n // (2)\n delete this.joinRooms[roomId];\n }\n break;\n\n default:\n logger.error(\"Unknown cateogory: \", category);\n }\n }\n\n private accumulateInviteState(roomId: string, data: IInvitedRoom): void {\n if (!data.invite_state || !data.invite_state.events) {\n // no new data\n return;\n }\n if (!this.inviteRooms[roomId]) {\n this.inviteRooms[roomId] = {\n invite_state: data.invite_state,\n };\n return;\n }\n // accumulate extra keys for invite->invite transitions\n // clobber based on event type / state key\n // We expect invite_state to be small, so just loop over the events\n const currentData = this.inviteRooms[roomId];\n data.invite_state.events.forEach((e) => {\n let hasAdded = false;\n for (let i = 0; i < currentData.invite_state.events.length; i++) {\n const current = currentData.invite_state.events[i];\n if (current.type === e.type && current.state_key == e.state_key) {\n currentData.invite_state.events[i] = e; // update\n hasAdded = true;\n }\n }\n if (!hasAdded) {\n currentData.invite_state.events.push(e);\n }\n });\n }\n\n private accumulateKnockState(roomId: string, data: IKnockedRoom): void {\n if (!data.knock_state || !data.knock_state.events) {\n // no new data\n return;\n }\n if (!this.knockRooms[roomId]) {\n this.knockRooms[roomId] = {\n knock_state: data.knock_state,\n };\n return;\n }\n // accumulate extra keys\n // clobber based on event type / state key\n // We expect knock_state to be small, so just loop over the events\n const currentData = this.knockRooms[roomId];\n data.knock_state.events.forEach((e) => {\n let hasAdded = false;\n for (let i = 0; i < currentData.knock_state.events.length; i++) {\n const current = currentData.knock_state.events[i];\n if (current.type === e.type && current.state_key == e.state_key) {\n currentData.knock_state.events[i] = e; // update\n hasAdded = true;\n }\n }\n if (!hasAdded) {\n currentData.knock_state.events.push(e);\n }\n });\n }\n\n // Accumulate timeline and state events in a room.\n private accumulateJoinState(roomId: string, data: IJoinedRoom, fromDatabase = false): void {\n const now = Date.now();\n // We expect this function to be called a lot (every /sync) so we want\n // this to be fast. /sync stores events in an array but we often want\n // to clobber based on type/state_key. Rather than convert arrays to\n // maps all the time, just keep private maps which contain\n // the actual current accumulated sync state, and array-ify it when\n // getJSON() is called.\n\n // State resolution:\n // The 'state' key is the delta from the previous sync (or start of time\n // if no token was supplied), to the START of the timeline. To obtain\n // the current state, we need to \"roll forward\" state by reading the\n // timeline. We want to store the current state so we can drop events\n // out the end of the timeline based on opts.maxTimelineEntries.\n //\n // 'state' 'timeline' current state\n // |-------x<======================>x\n // T I M E\n //\n // When getJSON() is called, we 'roll back' the current state by the\n // number of entries in the timeline to work out what 'state' should be.\n\n // Back-pagination:\n // On an initial /sync, the server provides a back-pagination token for\n // the start of the timeline. When /sync deltas come down, they also\n // include back-pagination tokens for the start of the timeline. This\n // means not all events in the timeline have back-pagination tokens, as\n // it is only the ones at the START of the timeline which have them.\n // In order for us to have a valid timeline (and back-pagination token\n // to match), we need to make sure that when we remove old timeline\n // events, that we roll forward to an event which has a back-pagination\n // token. This means we can't keep a strict sliding-window based on\n // opts.maxTimelineEntries, and we may have a few less. We should never\n // have more though, provided that the /sync limit is less than or equal\n // to opts.maxTimelineEntries.\n\n if (!this.joinRooms[roomId]) {\n // Create truly empty objects so event types of 'hasOwnProperty' and co\n // don't cause this code to break.\n this.joinRooms[roomId] = {\n _currentState: Object.create(null),\n _timeline: [],\n _accountData: Object.create(null),\n _unreadNotifications: {},\n _unreadThreadNotifications: {},\n _summary: {},\n _receipts: new ReceiptAccumulator(),\n _stickyEvents: [],\n };\n }\n const currentData = this.joinRooms[roomId];\n\n if (data.account_data && data.account_data.events) {\n // clobber based on type\n data.account_data.events.forEach((e) => {\n currentData._accountData[e.type] = e;\n });\n }\n\n // these probably clobber, spec is unclear.\n if (data.unread_notifications) {\n currentData._unreadNotifications = data.unread_notifications;\n }\n currentData._unreadThreadNotifications =\n data[UNREAD_THREAD_NOTIFICATIONS.stable!] ?? data[UNREAD_THREAD_NOTIFICATIONS.unstable!] ?? undefined;\n\n if (data.summary) {\n const HEROES_KEY = \"m.heroes\";\n const INVITED_COUNT_KEY = \"m.invited_member_count\";\n const JOINED_COUNT_KEY = \"m.joined_member_count\";\n\n const acc = currentData._summary;\n const sum = data.summary;\n acc[HEROES_KEY] = sum[HEROES_KEY] ?? acc[HEROES_KEY];\n acc[JOINED_COUNT_KEY] = sum[JOINED_COUNT_KEY] ?? acc[JOINED_COUNT_KEY];\n acc[INVITED_COUNT_KEY] = sum[INVITED_COUNT_KEY] ?? acc[INVITED_COUNT_KEY];\n }\n\n // We purposefully do not persist m.typing events.\n // Technically you could refresh a browser before the timer on a\n // typing event is up, so it'll look like you aren't typing when\n // you really still are. However, the alternative is worse. If\n // we do persist typing events, it will look like people are\n // typing forever until someone really does start typing (which\n // will prompt Synapse to send down an actual m.typing event to\n // clobber the one we persisted).\n\n // Persist the receipts\n currentData._receipts.consumeEphemeralEvents(data.ephemeral?.events);\n\n // if we got a limited sync, we need to remove all timeline entries or else\n // we will have gaps in the timeline.\n if (data.timeline && data.timeline.limited) {\n currentData._timeline = [];\n }\n\n // Work out the current state. The deltas need to be applied in the order:\n // - existing state which didn't come down /sync.\n // - State events under the 'state' key.\n // - State events under the 'state_after' key OR state events in the 'timeline' if 'state_after' is not present.\n data.state?.events?.forEach((e) => {\n setState(currentData._currentState, e);\n });\n data[\"org.matrix.msc4222.state_after\"]?.events?.forEach((e) => {\n setState(currentData._currentState, e);\n });\n data.timeline?.events?.forEach((e, index) => {\n if (!data[\"org.matrix.msc4222.state_after\"]) {\n // this nops if 'e' isn't a state event\n setState(currentData._currentState, e);\n }\n // append the event to the timeline. The back-pagination token\n // corresponds to the first event in the timeline\n let transformedEvent: TaggedEvent;\n if (!fromDatabase) {\n transformedEvent = Object.assign({}, e);\n if (transformedEvent.unsigned !== undefined) {\n transformedEvent.unsigned = Object.assign({}, transformedEvent.unsigned);\n }\n const age = e.unsigned?.age;\n if (age !== undefined) transformedEvent._localTs = Date.now() - age;\n } else {\n transformedEvent = e;\n }\n\n currentData._timeline.push({\n event: transformedEvent,\n token: index === 0 ? (data.timeline.prev_batch ?? null) : null,\n });\n });\n\n // Prune out any events in our stores that have since expired, do this before we\n // insert new events.\n currentData._stickyEvents = currentData._stickyEvents.filter(({ expiresTs }) => expiresTs > now);\n\n // We want this to be fast, so don't worry about duplicate events here. The RoomStickyEventsStore will\n // process these events into the correct mapped order.\n if (data.msc4354_sticky?.events) {\n currentData._stickyEvents = currentData._stickyEvents.concat(\n data.msc4354_sticky.events.map((event) => {\n // If `duration_ms` exceeds the spec limit of a hour, we cap it.\n const cappedDuration = Math.min(event.msc4354_sticky.duration_ms, MAX_STICKY_DURATION_MS);\n // If `origin_server_ts` claims to have been from the future, we still bound it to now.\n const createdTs = Math.min(event.origin_server_ts, now);\n return {\n event,\n expiresTs: cappedDuration + createdTs,\n };\n }),\n );\n }\n\n // attempt to prune the timeline by jumping between events which have\n // pagination tokens.\n if (currentData._timeline.length > this.opts.maxTimelineEntries!) {\n const startIndex = currentData._timeline.length - this.opts.maxTimelineEntries!;\n for (let i = startIndex; i < currentData._timeline.length; i++) {\n if (currentData._timeline[i].token) {\n // keep all events after this, including this one\n currentData._timeline = currentData._timeline.slice(i, currentData._timeline.length);\n break;\n }\n }\n }\n }\n\n /**\n * Return everything under the 'rooms' key from a /sync response which\n * represents all room data that should be stored. This should be paired\n * with the sync token which represents the most recent /sync response\n * provided to accumulate().\n * @param forDatabase - True to generate a sync to be saved to storage\n * @returns An object with a \"nextBatch\", \"roomsData\" and \"accountData\"\n * keys.\n * The \"nextBatch\" key is a string which represents at what point in the\n * /sync stream the accumulator reached. This token should be used when\n * restarting a /sync stream at startup. Failure to do so can lead to missing\n * events. The \"roomsData\" key is an Object which represents the entire\n * /sync response from the 'rooms' key onwards. The \"accountData\" key is\n * a list of raw events which represent global account data.\n */\n public getJSON(forDatabase = false): ISyncData {\n const data: IRooms = {\n join: {},\n invite: {},\n knock: {},\n // always empty. This is set by /sync when a room was previously\n // in 'invite' or 'join'. On fresh startup, the client won't know\n // about any previous room being in 'invite' or 'join' so we can\n // just omit mentioning it at all, even if it has previously come\n // down /sync.\n // The notable exception is when a client is kicked or banned:\n // we may want to hold onto that room so the client can clearly see\n // why their room has disappeared. We don't persist it though because\n // it is unclear *when* we can safely remove the room from the DB.\n // Instead, we assume that if you're loading from the DB, you've\n // refreshed the page, which means you've seen the kick/ban already.\n leave: {},\n };\n Object.keys(this.inviteRooms).forEach((roomId) => {\n data.invite[roomId] = this.inviteRooms[roomId];\n });\n Object.keys(this.knockRooms).forEach((roomId) => {\n data.knock[roomId] = this.knockRooms[roomId];\n });\n Object.keys(this.joinRooms).forEach((roomId) => {\n const roomData = this.joinRooms[roomId];\n const roomJson: IJoinedRoom & {\n // We track both `state` and `state_after` for downgrade compatibility\n \"state\": IState;\n \"org.matrix.msc4222.state_after\": IState;\n } = {\n \"ephemeral\": { events: [] },\n \"account_data\": { events: [] },\n \"state\": { events: [] },\n \"org.matrix.msc4222.state_after\": { events: [] },\n \"timeline\": {\n events: [],\n prev_batch: null,\n },\n \"unread_notifications\": roomData._unreadNotifications,\n \"unread_thread_notifications\": roomData._unreadThreadNotifications,\n \"summary\": roomData._summary as IRoomSummary,\n \"msc4354_sticky\": roomData._stickyEvents?.length\n ? {\n events: roomData._stickyEvents.map((e) => e.event),\n }\n : undefined,\n };\n // Add account data\n Object.keys(roomData._accountData).forEach((evType) => {\n roomJson.account_data.events.push(roomData._accountData[evType]);\n });\n\n const receiptEvent = roomData._receipts.buildAccumulatedReceiptEvent(roomId);\n\n // add only if we have some receipt data\n if (receiptEvent) {\n roomJson.ephemeral.events.push(receiptEvent);\n }\n\n // Add timeline data\n roomData._timeline.forEach((msgData) => {\n if (!roomJson.timeline.prev_batch) {\n // the first event we add to the timeline MUST match up to\n // the prev_batch token.\n if (!msgData.token) {\n return; // this shouldn't happen as we prune constantly.\n }\n roomJson.timeline.prev_batch = msgData.token;\n }\n\n let transformedEvent: (IRoomEvent | IStateEvent) & { _localTs?: number };\n if (!forDatabase && isTaggedEvent(msgData.event)) {\n // This means we have to copy each event, so we can fix it up to\n // set a correct 'age' parameter whilst keeping the local timestamp\n // on our stored event. If this turns out to be a bottleneck, it could\n // be optimised either by doing this in the main process after the data\n // has been structured-cloned to go between the worker & main process,\n // or special-casing data from saved syncs to read the local timestamp\n // directly rather than turning it into age to then immediately be\n // transformed back again into a local timestamp.\n transformedEvent = Object.assign({}, msgData.event);\n if (transformedEvent.unsigned !== undefined) {\n transformedEvent.unsigned = Object.assign({}, transformedEvent.unsigned);\n }\n delete transformedEvent._localTs;\n transformedEvent.unsigned = transformedEvent.unsigned || {};\n transformedEvent.unsigned.age = Date.now() - msgData.event._localTs!;\n } else {\n transformedEvent = msgData.event;\n }\n roomJson.timeline.events.push(transformedEvent);\n });\n\n // Add state data: roll back current state to the start of timeline,\n // by \"reverse clobbering\" from the end of the timeline to the start.\n // Convert maps back into arrays.\n const rollBackState = Object.create(null);\n for (let i = roomJson.timeline.events.length - 1; i >= 0; i--) {\n const timelineEvent = roomJson.timeline.events[i];\n if (\n (timelineEvent as IStateEvent).state_key === null ||\n (timelineEvent as IStateEvent).state_key === undefined\n ) {\n continue; // not a state event\n }\n // since we're going back in time, we need to use the previous\n // state value else we'll break causality. We don't have the\n // complete previous state event, so we need to create one.\n const prevStateEvent = deepCopy(timelineEvent);\n if (prevStateEvent.unsigned) {\n if (prevStateEvent.unsigned.prev_content) {\n prevStateEvent.content = prevStateEvent.unsigned.prev_content;\n }\n if (prevStateEvent.unsigned.prev_sender) {\n prevStateEvent.sender = prevStateEvent.unsigned.prev_sender;\n }\n }\n setState(rollBackState, prevStateEvent);\n }\n Object.keys(roomData._currentState).forEach((evType) => {\n Object.keys(roomData._currentState[evType]).forEach((stateKey) => {\n let ev = roomData._currentState[evType][stateKey];\n // Push to both fields to provide downgrade compatibility in the sync accumulator db\n // the code will prefer `state_after` if it is present\n roomJson[\"org.matrix.msc4222.state_after\"].events.push(ev);\n // Roll the state back to the value at the start of the timeline if it was changed\n if (rollBackState[evType] && rollBackState[evType][stateKey]) {\n ev = rollBackState[evType][stateKey];\n }\n roomJson.state.events.push(ev);\n });\n });\n data.join[roomId] = roomJson;\n });\n\n // Add account data\n const accData: IMinimalEvent[] = [];\n Object.keys(this.accountData).forEach((evType) => {\n accData.push(this.accountData[evType]);\n });\n\n return {\n nextBatch: this.nextBatch!,\n roomsData: data,\n accountData: accData,\n };\n }\n\n public getNextBatchToken(): string {\n return this.nextBatch!;\n }\n}\n\nfunction setState(eventMap: Record<string, Record<string, IStateEvent>>, event: IRoomEvent | IStateEvent): void {\n if ((event as IStateEvent).state_key === null || (event as IStateEvent).state_key === undefined || !event.type) {\n return;\n }\n if (!eventMap[event.type]) {\n eventMap[event.type] = Object.create(null);\n }\n eventMap[event.type][(event as IStateEvent).state_key] = event as IStateEvent;\n}\n"],"mappings":"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","ignoreList":[]}
|