matrix-js-sdk 41.3.0 → 41.4.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.
Files changed (43) hide show
  1. package/CHANGELOG.md +17 -0
  2. package/lib/@types/event.d.ts +9 -0
  3. package/lib/@types/event.d.ts.map +1 -1
  4. package/lib/@types/event.js.map +1 -1
  5. package/lib/crypto-api/index.d.ts +27 -5
  6. package/lib/crypto-api/index.d.ts.map +1 -1
  7. package/lib/crypto-api/index.js +22 -3
  8. package/lib/crypto-api/index.js.map +1 -1
  9. package/lib/embedded.js +1 -1
  10. package/lib/embedded.js.map +1 -1
  11. package/lib/matrixrtc/MatrixRTCSession.js +2 -1
  12. package/lib/matrixrtc/MatrixRTCSession.js.map +1 -1
  13. package/lib/matrixrtc/membershipData/rtc.d.ts.map +1 -1
  14. package/lib/matrixrtc/membershipData/rtc.js +5 -4
  15. package/lib/matrixrtc/membershipData/rtc.js.map +1 -1
  16. package/lib/oidc/authorize.d.ts +8 -4
  17. package/lib/oidc/authorize.d.ts.map +1 -1
  18. package/lib/oidc/authorize.js +18 -6
  19. package/lib/oidc/authorize.js.map +1 -1
  20. package/lib/oidc/discovery.js +1 -1
  21. package/lib/oidc/discovery.js.map +1 -1
  22. package/lib/rust-crypto/backup.d.ts +6 -2
  23. package/lib/rust-crypto/backup.d.ts.map +1 -1
  24. package/lib/rust-crypto/backup.js +30 -10
  25. package/lib/rust-crypto/backup.js.map +1 -1
  26. package/lib/rust-crypto/rust-crypto.d.ts +16 -11
  27. package/lib/rust-crypto/rust-crypto.d.ts.map +1 -1
  28. package/lib/rust-crypto/rust-crypto.js +38 -21
  29. package/lib/rust-crypto/rust-crypto.js.map +1 -1
  30. package/lib/store/indexeddb-local-backend.d.ts.map +1 -1
  31. package/lib/store/indexeddb-local-backend.js +4 -2
  32. package/lib/store/indexeddb-local-backend.js.map +1 -1
  33. package/package.json +14 -10
  34. package/src/@types/event.ts +14 -2
  35. package/src/crypto-api/index.ts +30 -5
  36. package/src/embedded.ts +1 -1
  37. package/src/matrixrtc/MatrixRTCSession.ts +2 -1
  38. package/src/matrixrtc/membershipData/rtc.ts +5 -4
  39. package/src/oidc/authorize.ts +25 -6
  40. package/src/oidc/discovery.ts +1 -1
  41. package/src/rust-crypto/backup.ts +31 -10
  42. package/src/rust-crypto/rust-crypto.ts +28 -13
  43. package/src/store/indexeddb-local-backend.ts +2 -1
@@ -1 +1 @@
1
- {"version":3,"file":"MatrixRTCSession.js","names":["logger","rootLogger","TypedEventEmitter","EventTimeline","EventType","RelationType","KnownMembership","CallMembership","RoomStateEvent","MembershipManager","StickyEventMembershipManager","logDurationSync","MembershipManagerEvent","RTCEncryptionManager","ToDeviceKeyTransport","TypedReEmitter","RoomStickyEventsEvent","computeSlotId","MatrixRTCSessionEvent","DEFAULT_SESSION_MEMBERSHIPS_FOR_SLOT_OPTS","listenForStickyEvents","listenForMemberStateEvents","MatrixRTCSession","membershipStatus","_this$membershipManag","membershipManager","status","probablyLeft","_this$membershipManag2","delayId","_this$membershipManag3","callId","_this$slotDescription","slotDescription","id","slotId","sessionMembershipsForSlot","room","_arguments","arguments","_asyncToGenerator","options","length","undefined","getChild","concat","roomId","application","callMemberEvents","collectMembersEvents","callMemberships","computeBackendIdentityAndVerifyMemberEvents","sort","a","b","createdTs","debug","map","m","userId","sessionForSlot","client","opts","roomSubset","constructor","calculateMembershipsOpts","_this","this","_defineProperty","counters","roomEventEncryptionKeysSent","roomEventEncryptionKeysReceived","totals","roomEventEncryptionKeysReceivedTotalAge","ensureRecalculateSessionMembers","added","updated","removed","flatMap","v","current","previous","some","e","getType","RTCMembership","recalculateSessionMembers","Promise","resolve","_this$encryptionManag","membershipNeedsRecalculation","oldMemberships","memberships","changed","i","equal","_this$membershipManag4","_this$membershipManag5","info","emit","MembershipsChanged","onRTCSessionMemberUpdate","ownMembership","pendingNotificationToSend","_this$joinConfig","eventId","joinConfig","notificationType","sendCallNotify","callIntent","warn","encryptionManager","onMembershipsUpdate","setExpiryTimer","on","Members","onRoomMemberUpdate","Update","onStickyEventUpdate","initialMembershipCalculated","isJoined","_this$membershipManag6","_this$membershipManag7","stop","_this2","_this2$membershipMana","leave","expiryTimeout","clearTimeout","off","joinRTCSession","ownMembershipIdentity","fociPreferred","multiSfuFocus","_this$joinConfig2","unstableSendStickyEvents","memberId","reEmitter","reEmit","ProbablyLeft","StatusChanged","DelayIdChanged","statistics","transport","keyBin","encryptionKeyIndex","membership","rtcBackendIdentity","EncryptionKeyChanged","join","error","MembershipManagerError","JoinStateChanged","joinRoomSession","deviceId","getUserId","getDeviceId","leaveRoomSession","_arguments2","_this3","timeout","leavePromise","getFocusInUse","oldestMembership","getOldestMembership","getTransport","getConsensusCallIntent","_this$memberships$fin","getFirstCallIntent","find","every","updateCallIntent","_this4","_this4$membershipMana","_this4$membershipMana2","myMembership","Error","reemitEncryptionKeys","_this$encryptionManag2","getEncryptionKeys","forEach","keyRing","key","keyInfo","keyIndex","soonestExpiry","thisExpiry","getMsUntilExpiry","setTimeout","parentEventId","_this5","sendNotificationEvent","_ref3","content","user_ids","event_id","rel_type","Reference","Date","now","response","sendEvent","RTCNotification","apply","then","notification","newResult","_objectSpread","DidSendCallNotification","catch","_ref4","errorLegacy","errorNew","recalculateSessionMembersPromise","finally","_x","_x2","_x3","_x4","_computeBackendIdentityAndVerifyMemberEvents","memberEvent","getContent","quickFilterNonRelevantContents","parseFromEvent","isValidMembership","push","eventKeysCount","Object","keys","filter","k","_membership$userId","JSON","stringify","isExpired","hasMembershipState","Join","_unstable_getStickyEvents","roomState","getLiveTimeline","getState","FORWARDS","callMemberStateEvents","getStateEvents","GroupCallMemberPrefix","callMemberStateEvent","stickyEvent","msc4354_sticky_key","getStateKey"],"sources":["../../src/matrixrtc/MatrixRTCSession.ts"],"sourcesContent":["/*\nCopyright 2023 - 2026 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\nimport { type Logger, logger as rootLogger } from \"../logger.ts\";\nimport { TypedEventEmitter } from \"../models/typed-event-emitter.ts\";\nimport { EventTimeline } from \"../models/event-timeline.ts\";\nimport { type Room } from \"../models/room.ts\";\nimport { type MatrixClient } from \"../client.ts\";\nimport { EventType, RelationType } from \"../@types/event.ts\";\nimport { KnownMembership } from \"../@types/membership.ts\";\nimport { type ISendEventResponse } from \"../@types/requests.ts\";\nimport { CallMembership } from \"./CallMembership.ts\";\nimport { RoomStateEvent } from \"../models/room-state.ts\";\nimport { MembershipManager, StickyEventMembershipManager } from \"./MembershipManager.ts\";\nimport { type CallMembershipIdentityParts, type IEncryptionManager } from \"./EncryptionManager.ts\";\nimport { logDurationSync } from \"../utils.ts\";\nimport type {\n Statistics,\n RTCNotificationType,\n Status,\n IRTCNotificationContent,\n RTCCallIntent,\n Transport,\n SlotDescription,\n} from \"./types.ts\";\nimport {\n MembershipManagerEvent,\n type MembershipManagerEventHandlerMap,\n type IMembershipManager,\n} from \"./IMembershipManager.ts\";\nimport { RTCEncryptionManager } from \"./RTCEncryptionManager.ts\";\nimport { ToDeviceKeyTransport } from \"./ToDeviceKeyTransport.ts\";\nimport { TypedReEmitter } from \"../ReEmitter.ts\";\nimport { type IContent, type MatrixEvent } from \"../models/event.ts\";\nimport { RoomStickyEventsEvent, type RoomStickyEventsMap } from \"../models/room-sticky-events.ts\";\nimport { computeSlotId } from \"./utils.ts\";\n\n/**\n * Events emitted by MatrixRTCSession\n */\nexport enum MatrixRTCSessionEvent {\n // A member joined, left, or updated a property of their membership.\n MembershipsChanged = \"memberships_changed\",\n // We joined or left the session: our own local idea of whether we are joined,\n // separate from MembershipsChanged, ie. independent of whether our member event\n // has successfully gone through.\n JoinStateChanged = \"join_state_changed\",\n // The key used to encrypt media has changed\n EncryptionKeyChanged = \"encryption_key_changed\",\n /** The membership manager had to shut down caused by an unrecoverable error */\n MembershipManagerError = \"membership_manager_error\",\n /** The RTCSession did send a call notification caused by joining the call as the first member */\n DidSendCallNotification = \"did_send_call_notification\",\n}\n\nexport type MatrixRTCSessionEventHandlerMap = {\n [MatrixRTCSessionEvent.MembershipsChanged]: (\n oldMemberships: CallMembership[],\n newMemberships: CallMembership[],\n ) => void;\n [MatrixRTCSessionEvent.JoinStateChanged]: (isJoined: boolean) => void;\n [MatrixRTCSessionEvent.EncryptionKeyChanged]: (\n key: Uint8Array<ArrayBuffer>,\n encryptionKeyIndex: number,\n membership: CallMembershipIdentityParts,\n rtcBackendIdentity: string,\n ) => void;\n [MatrixRTCSessionEvent.MembershipManagerError]: (error: unknown) => void;\n [MatrixRTCSessionEvent.DidSendCallNotification]: (\n notificationContentNew: { event_id: string } & IRTCNotificationContent,\n ) => void;\n};\n\nexport interface SessionConfig {\n /**\n * What kind of notification to send when starting the session.\n * @default `undefined` (no notification)\n */\n notificationType?: RTCNotificationType;\n\n /**\n * Determines the kind of call this will be.\n */\n callIntent?: RTCCallIntent;\n}\n\n// The names follow these principles:\n// - we use the technical term delay if the option is related to delayed events.\n// - we use delayedLeaveEvent if the option is related to the delayed leave event.\n// - we use membershipEvent if the option is related to the rtc member state event.\n// - we use the technical term expiry if the option is related to the expiry field of the membership state event.\n// - we use a `Ms` postfix if the option is a duration to avoid using words like:\n// `time`, `duration`, `delay`, `timeout`... that might be mistaken/confused with technical terms.\nexport interface MembershipConfig {\n /**\n * The timeout (in milliseconds) after we joined the call, that our membership should expire\n * unless we have explicitly updated it.\n *\n * This is what goes into the m.rtc.member event expiry field and is typically set to a number of hours.\n */\n membershipEventExpiryMs?: number;\n\n /**\n * The time in (in milliseconds) which the manager will prematurely send the updated state event before the membership `expires` time to make sure it\n * sends the updated state event early enough.\n *\n * A headroom of 1000ms and a `membershipExpiryTimeout` of 10000ms would result in the first membership event update after 9s and\n * a membership event that would be considered expired after 10s.\n *\n * This value does not have an effect on the value of `SessionMembershipData.expires`.\n */\n membershipEventExpiryHeadroomMs?: number;\n\n /**\n * The timeout (in milliseconds) with which the deleayed leave event on the server is configured.\n * After this time the server will set the event to the disconnected stat if it has not received a keep-alive from the client.\n */\n delayedLeaveEventDelayMs?: number;\n\n /**\n * The interval (in milliseconds) in which the client will send membership keep-alives to the server.\n */\n delayedLeaveEventRestartMs?: number;\n\n /**\n * The maximum number of retries that the manager will do for delayed event sending/updating and state event sending when a server rate limit has been hit.\n */\n maximumRateLimitRetryCount?: number;\n\n /**\n * The maximum number of retries that the manager will do for delayed event sending/updating and state event sending when a network error occurs.\n */\n maximumNetworkErrorRetryCount?: number;\n\n /**\n * The time (in milliseconds) after which we will retry a http request if it\n * failed to send due to a network error. (send membership event, send delayed event, restart delayed event...)\n */\n networkErrorRetryMs?: number;\n\n /**\n * The time (in milliseconds) after which a we consider a delayed event restart http request to have failed.\n * Setting this to a lower value will result in more frequent retries but also a higher chance of failiour.\n *\n * In the presence of network packet loss (hurting TCP connections), the custom delayedEventRestartLocalTimeoutMs\n * helps by keeping more delayed event reset candidates in flight,\n * improving the chances of a successful reset. (its is equivalent to the js-sdk `localTimeout` configuration,\n * but only applies to calls to the `_unstable_restartScheduledDelayedEvent` endpoint\n * or the `_unstable_updateDelayedEvent` endpoint with a body of `{action:\"restart\"}`.)\n */\n delayedLeaveEventRestartLocalTimeoutMs?: number;\n\n /**\n * Send membership using sticky events rather than state events.\n * This also make the client use the new m.rtc.member MSC4354 event format. (instead of m.call.member)\n *\n * **WARNING**: This is an unstable feature and not all clients will support it.\n */\n unstableSendStickyEvents?: boolean;\n}\n\nexport interface EncryptionConfig {\n /**\n * If true, generate and share a media key for this participant,\n * and emit MatrixRTCSessionEvent.EncryptionKeyChanged when\n * media keys for other participants become available.\n */\n manageMediaKeys?: boolean;\n /**\n * The minimum time (in milliseconds) between each attempt to send encryption key(s).\n * e.g. if this is set to 1000, then we will send at most one key event every second.\n * @deprecated - Not used by the new encryption manager.\n */\n updateEncryptionKeyThrottle?: number;\n\n /**\n * Sometimes it is necessary to rotate the encryption key after a membership update.\n * For performance reasons we might not want to rotate the key immediately but allow future memberships to use the same key.\n * If 5 people join in a row in less than 5 seconds, we don't want to rotate the key for each of them.\n * If 5 people leave in a row in less than 5 seconds, we don't want to rotate the key for each of them.\n * So we do share the key which was already used live for <5s to new joiners.\n * This does result in a potential leak up to the configured time of call media.\n * This has to be considered when choosing a value for this property.\n */\n keyRotationGracePeriodMs?: number;\n\n /**\n * The delay (in milliseconds) after a member leaves before we create and publish a new key, because people\n * tend to leave calls at the same time.\n * @deprecated - Not used by the new encryption manager.\n */\n makeKeyDelay?: number;\n /**\n * The delay (in milliseconds) between sending a new key and starting to encrypt with it. This\n * gives others a chance to receive the new key to minimize the chance they get media they can't decrypt.\n *\n * The higher this value is, the better it is for existing members as they will have a smoother experience.\n * But it impacts new joiners: They will always have to wait `useKeyDelay` before being able to decrypt the media\n * (as it will be encrypted with the new key after the delay only), even if the key has already arrived before the delay.\n */\n useKeyDelay?: number;\n}\nexport type JoinSessionConfig = SessionConfig & MembershipConfig & EncryptionConfig;\n\ninterface SessionMembershipsForSlotOpts {\n /**\n * Listen for incoming sticky member events. If disabled, this session will\n * ignore any incoming sticky events.\n */\n listenForStickyEvents: boolean;\n /**\n * Listen for incoming member state events (legacy). If disabled, this session will\n * ignore any incoming state events.\n */\n listenForMemberStateEvents: boolean;\n}\n\nconst DEFAULT_SESSION_MEMBERSHIPS_FOR_SLOT_OPTS: SessionMembershipsForSlotOpts = {\n listenForStickyEvents: true,\n listenForMemberStateEvents: true,\n};\n\n/**\n * A MatrixRTCSession manages the membership & properties of a MatrixRTC session.\n * This class doesn't deal with media at all, just membership & properties of a session.\n */\nexport class MatrixRTCSession extends TypedEventEmitter<\n MatrixRTCSessionEvent | MembershipManagerEvent,\n MatrixRTCSessionEventHandlerMap & MembershipManagerEventHandlerMap\n> {\n private membershipManager?: IMembershipManager;\n private encryptionManager?: IEncryptionManager;\n private joinConfig?: SessionConfig;\n private logger: Logger;\n\n private pendingNotificationToSend: undefined | RTCNotificationType;\n /**\n * This timeout is responsible to track any expiration. We need to know when we have to start\n * to ignore other call members. There is no callback for this. This timeout will always be configured to\n * emit when the next membership expires.\n */\n private expiryTimeout?: ReturnType<typeof setTimeout>;\n\n public memberships: CallMembership[] = [];\n\n /**\n * Resolves when the session has calculated the initial membership of the session.\n */\n public readonly initialMembershipCalculated: Promise<void>;\n /**\n * Does membership need to be recalculated? This is set to false upon\n * recalculation.\n */\n private membershipNeedsRecalculation = false;\n\n /**\n * The statistics for this session.\n */\n public statistics: Statistics = {\n counters: {\n roomEventEncryptionKeysSent: 0,\n roomEventEncryptionKeysReceived: 0,\n },\n totals: {\n roomEventEncryptionKeysReceivedTotalAge: 0,\n },\n };\n\n public get membershipStatus(): Status | undefined {\n return this.membershipManager?.status;\n }\n public get probablyLeft(): boolean | undefined {\n return this.membershipManager?.probablyLeft;\n }\n public get delayId(): string | undefined {\n return this.membershipManager?.delayId;\n }\n\n /**\n * The callId (sessionId) of the call.\n *\n * It can be undefined since the callId is only known once the first membership joins.\n * The callId is the property that, per definition, groups memberships into one call.\n * @deprecated use `slotId` instead.\n */\n public get callId(): string | undefined {\n return this.slotDescription?.id;\n }\n /**\n * The slotId of the call.\n * `{application}#{appSpecificId}`\n * It can be undefined since the slotId is only known once the first membership joins.\n * The slotId is the property that, per definition, groups memberships into one call.\n */\n public get slotId(): string | undefined {\n return computeSlotId(this.slotDescription);\n }\n\n /**\n * Returns all the call memberships for a room that match the provided `sessionDescription`,\n * oldest first.\n *\n * By default, this will return *both* sticky and member state events.\n */\n public static async sessionMembershipsForSlot(\n room: Pick<Room, \"getLiveTimeline\" | \"roomId\" | \"hasMembershipState\" | \"_unstable_getStickyEvents\">,\n slotDescription: SlotDescription,\n // default both true this implied we combine sticky and state events for the final call state\n // (prefer sticky events in case of a duplicate)\n options: SessionMembershipsForSlotOpts = DEFAULT_SESSION_MEMBERSHIPS_FOR_SLOT_OPTS,\n ): Promise<CallMembership[]> {\n const logger = rootLogger.getChild(\n `[MatrixRTCSession ${room.roomId} ${slotDescription.application}#${slotDescription.id}]`,\n );\n const callMemberEvents = collectMembersEvents(room, options, logger);\n\n const callMemberships = await computeBackendIdentityAndVerifyMemberEvents(\n room,\n callMemberEvents,\n slotDescription,\n logger,\n );\n\n callMemberships.sort((a, b) => a.createdTs() - b.createdTs());\n if (callMemberships.length > 1) {\n logger.debug(\n `Call memberships in room ${room.roomId}, in order: `,\n callMemberships.map((m) => [m.createdTs(), m.userId]),\n );\n }\n\n return callMemberships;\n }\n\n /**\n * Return the MatrixRTC session for the room.\n * This returned session can be used to find out if there are active sessions\n * for the requested room and `slotDescription`.\n */\n public static sessionForSlot(\n client: MatrixClient,\n room: Room,\n slotDescription: SlotDescription,\n opts?: SessionMembershipsForSlotOpts,\n ): MatrixRTCSession {\n return new MatrixRTCSession(client, room, slotDescription, opts);\n }\n\n /**\n * WARN: this can in theory only be a subset of the room with the properties required by\n * this class.\n * Outside of tests this most likely will be a full room, however.\n * @deprecated Relying on a full Room object being available here is an anti-pattern. You should be tracking\n * the room object in your own code and passing it in when needed.\n */\n public get room(): Room {\n return this.roomSubset as Room;\n }\n\n /**\n * This constructs a room session. When using MatrixRTC inside the js-sdk this is expected\n * to be used with the MatrixRTCSessionManager exclusively.\n *\n * In cases where you don't use the js-sdk but build on top of another Matrix stack this class can be used standalone\n * to manage a joined MatrixRTC session.\n *\n * @param client A subset of the {@link MatrixClient} that lets the session interact with the Matrix room.\n * @param roomSubset The room this session is attached to. A subset of a js-sdk Room that the session needs.\n * @param slotDescription The slot description is a virtual address where participants are allowed to meet.\n * This session will only manage memberships that match this slot description.Sessions are distinct if any of\n * those properties are distinct: `roomSubset.roomId`, `slotDescription.application`, `slotDescription.id`.\n * @param calculateMembershipsOpts - Options to configure how memberships are calculated for this session.\n */\n public constructor(\n private readonly client: Pick<\n MatrixClient,\n | \"getUserId\"\n | \"getDeviceId\"\n | \"sendEvent\"\n | \"sendStateEvent\"\n | \"_unstable_sendDelayedStateEvent\"\n | \"_unstable_updateDelayedEvent\"\n | \"_unstable_cancelScheduledDelayedEvent\"\n | \"_unstable_restartScheduledDelayedEvent\"\n | \"_unstable_sendScheduledDelayedEvent\"\n | \"_unstable_sendStickyEvent\"\n | \"_unstable_sendStickyDelayedEvent\"\n | \"cancelPendingEvent\"\n | \"encryptAndSendToDevice\"\n | \"off\"\n | \"on\"\n | \"decryptEventIfNeeded\"\n >,\n private roomSubset: Pick<\n Room,\n | \"getLiveTimeline\"\n | \"roomId\"\n | \"getVersion\"\n | \"hasMembershipState\"\n | \"on\"\n | \"off\"\n | \"_unstable_getStickyEvents\"\n >,\n\n public readonly slotDescription: SlotDescription,\n private readonly calculateMembershipsOpts?: SessionMembershipsForSlotOpts,\n ) {\n super();\n this.logger = rootLogger.getChild(\n `[MatrixRTCSession ${roomSubset.roomId} ${slotDescription.application}#${slotDescription.id}]`,\n );\n\n this.roomSubset.on(RoomStateEvent.Members, this.onRoomMemberUpdate);\n this.roomSubset.on(RoomStickyEventsEvent.Update, this.onStickyEventUpdate);\n\n this.initialMembershipCalculated = this.ensureRecalculateSessionMembers();\n this.setExpiryTimer();\n }\n /*\n * Returns true if we intend to be participating in the MatrixRTC session.\n * This is determined by checking if the relativeExpiry has been set.\n */\n public isJoined(): boolean {\n return this.membershipManager?.isJoined() ?? false;\n }\n\n /**\n * Performs cleanup & removes timers for client shutdown\n */\n public async stop(): Promise<void> {\n await this.membershipManager?.leave(1000);\n if (this.expiryTimeout) {\n clearTimeout(this.expiryTimeout);\n this.expiryTimeout = undefined;\n }\n\n this.roomSubset.off(RoomStateEvent.Members, this.onRoomMemberUpdate);\n this.roomSubset.off(RoomStickyEventsEvent.Update, this.onStickyEventUpdate);\n }\n\n private reEmitter = new TypedReEmitter<\n MatrixRTCSessionEvent | MembershipManagerEvent,\n MatrixRTCSessionEventHandlerMap & MembershipManagerEventHandlerMap\n >(this);\n\n /**\n * Announces this user and device as joined to the MatrixRTC session,\n * and continues to update the membership event to keep it valid until\n * leaveRoomSession() is called\n * This will not subscribe to updates: remember to call subscribe() separately if\n * desired.\n * This method will return immediately and the session will be joined in the background.\n * @param ownMembershipIdentity the identity of the user and device joining the session.\n * This will be put into the content.member.\n * @param fociPreferred the list of preferred foci to use in the joined RTC membership event.\n * If multiSfuFocus is set, this is only needed if this client wants to publish to multiple transports simultaneously.\n * @param multiSfuFocus the active focus to use in the joined RTC membership event. Setting this implies the\n * membership manager will operate in a multi-SFU connection mode. If `undefined`, an `oldest_membership`\n * transport selection will be used instead.\n * @param joinConfig - Additional configuration for the joined session.\n */\n public joinRTCSession(\n ownMembershipIdentity: CallMembershipIdentityParts,\n fociPreferred: Transport[],\n multiSfuFocus?: Transport,\n joinConfig?: JoinSessionConfig,\n ): void {\n if (this.isJoined()) {\n this.logger.info(`Already joined to session in room ${this.roomSubset.roomId}: ignoring join call`);\n return;\n } else {\n // Create MembershipManager and pass the RTCSession logger (with room id info)\n this.membershipManager = joinConfig?.unstableSendStickyEvents\n ? new StickyEventMembershipManager(\n joinConfig,\n this.roomSubset,\n this.client,\n this.slotDescription,\n ownMembershipIdentity.memberId,\n this.logger,\n )\n : new MembershipManager(joinConfig, this.roomSubset, this.client, this.slotDescription, this.logger);\n\n this.reEmitter.reEmit(this.membershipManager!, [\n MembershipManagerEvent.ProbablyLeft,\n MembershipManagerEvent.StatusChanged,\n MembershipManagerEvent.DelayIdChanged,\n ]);\n // Create Encryption manager\n const [room, client, statistics] = [this.roomSubset, this.client, this.statistics];\n const transport = new ToDeviceKeyTransport(ownMembershipIdentity, room.roomId, client, statistics);\n this.encryptionManager = new RTCEncryptionManager(\n ownMembershipIdentity,\n () => this.memberships,\n transport,\n (\n keyBin: Uint8Array<ArrayBuffer>,\n encryptionKeyIndex: number,\n membership: CallMembershipIdentityParts,\n rtcBackendIdentity: string,\n ) => {\n this.emit(\n MatrixRTCSessionEvent.EncryptionKeyChanged,\n keyBin,\n encryptionKeyIndex,\n membership,\n rtcBackendIdentity,\n );\n },\n this.logger,\n );\n }\n\n this.joinConfig = joinConfig;\n this.pendingNotificationToSend = this.joinConfig?.notificationType;\n\n // Join!\n this.membershipManager!.join(fociPreferred, multiSfuFocus, (e) => {\n this.logger.error(\"MembershipManager encountered an unrecoverable error: \", e);\n this.emit(MatrixRTCSessionEvent.MembershipManagerError, e);\n this.emit(MatrixRTCSessionEvent.JoinStateChanged, this.isJoined());\n });\n this.encryptionManager!.join(joinConfig);\n\n this.emit(MatrixRTCSessionEvent.JoinStateChanged, true);\n }\n\n /**\n *\n * @param fociPreferred\n * @param multiSfuFocus\n * @param joinConfig\n * @deprecated use the joinRTCSession method instead\n */\n public joinRoomSession(\n fociPreferred: Transport[],\n multiSfuFocus?: Transport,\n joinConfig?: JoinSessionConfig,\n ): void {\n const [userId, deviceId] = [this.client.getUserId()!, this.client.getDeviceId()!];\n // TODO this wants to become a UUID\n const memberId = `${userId}:${deviceId}`;\n this.joinRTCSession({ userId, deviceId, memberId }, fociPreferred, multiSfuFocus, joinConfig);\n }\n\n /**\n * Announces this user and device as having left the MatrixRTC session\n * and stops scheduled updates.\n * This will not unsubscribe from updates: remember to call unsubscribe() separately if\n * desired.\n * The membership update required to leave the session will retry if it fails.\n * Without network connection the promise will never resolve.\n * A timeout can be provided so that there is a guarantee for the promise to resolve.\n * @returns Whether the membership update was attempted and did not time out.\n */\n public async leaveRoomSession(timeout: number | undefined = undefined): Promise<boolean> {\n if (!this.isJoined()) {\n this.logger.info(`Not joined to session in room ${this.roomSubset.roomId}: ignoring leave call`);\n return false;\n }\n\n this.logger.info(`Leaving call session in room ${this.roomSubset.roomId}`);\n\n this.encryptionManager!.leave();\n\n const leavePromise = this.membershipManager!.leave(timeout);\n this.emit(MatrixRTCSessionEvent.JoinStateChanged, false);\n\n return await leavePromise;\n }\n /**\n * This returns the focus in use by the oldest membership.\n * Do not use since this might be just the focus for the oldest membership. others might use a different focus.\n * @deprecated use `member.getTransport(session.getOldestMembership())` instead for the specific member you want to get the focus for.\n */\n public getFocusInUse(): Transport | undefined {\n const oldestMembership = this.getOldestMembership();\n return oldestMembership?.getTransport(oldestMembership);\n }\n\n public getOldestMembership(): CallMembership | undefined {\n return this.memberships[0];\n }\n\n /**\n * Get the call intent for the current call, based on what members are advertising. If one or more\n * members disagree on the current call intent, or nobody specifies one then `undefined` is returned.\n *\n * If all members that specify a call intent agree, that value is returned.\n * @returns A call intent, or `undefined` if no consensus or not given.\n */\n public getConsensusCallIntent(): RTCCallIntent | undefined {\n const getFirstCallIntent = this.memberships.find((m) => !!m.callIntent)?.callIntent;\n if (!getFirstCallIntent) {\n return undefined;\n }\n if (this.memberships.every((m) => !m.callIntent || m.callIntent === getFirstCallIntent)) {\n return getFirstCallIntent;\n }\n return undefined;\n }\n\n public async updateCallIntent(callIntent: RTCCallIntent): Promise<void> {\n const myMembership = this.membershipManager?.ownMembership;\n if (!myMembership) {\n throw Error(\"Not connected yet\");\n }\n await this.membershipManager?.updateCallIntent(callIntent);\n }\n\n /**\n * Re-emit an EncryptionKeyChanged event for each tracked encryption key. This can be used to export\n * the keys.\n */\n public reemitEncryptionKeys(): void {\n this.encryptionManager?.getEncryptionKeys().forEach((keyRing, key) => {\n keyRing.forEach((keyInfo) => {\n this.emit(\n MatrixRTCSessionEvent.EncryptionKeyChanged,\n keyInfo.key,\n keyInfo.keyIndex,\n keyInfo.membership,\n keyInfo.rtcBackendIdentity,\n );\n });\n });\n }\n\n /**\n * Sets a timer for the soonest membership expiry\n */\n private setExpiryTimer(): void {\n if (this.expiryTimeout) {\n clearTimeout(this.expiryTimeout);\n this.expiryTimeout = undefined;\n }\n\n let soonestExpiry;\n for (const membership of this.memberships) {\n const thisExpiry = membership.getMsUntilExpiry();\n // If getMsUntilExpiry is undefined we have a MSC4143 (MatrixRTC) compliant event - it never expires\n // but will be reliably resent on disconnect.\n if (thisExpiry !== undefined && (soonestExpiry === undefined || thisExpiry < soonestExpiry)) {\n soonestExpiry = thisExpiry;\n }\n }\n\n if (soonestExpiry != undefined) {\n this.expiryTimeout = setTimeout(() => void this.ensureRecalculateSessionMembers(), soonestExpiry);\n }\n }\n\n /**\n * Sends notification events to indiciate the call has started.\n * Note: This does not return a promise, instead scheduling the notification events to be sent.\n * @param parentEventId Event id linking to your RTC call membership event.\n * @param notificationType The type of notification to send\n * @param callIntent The type of call this is (e.g. \"audio\").\n */\n private sendCallNotify(\n parentEventId: string,\n notificationType: RTCNotificationType,\n callIntent?: RTCCallIntent,\n ): void {\n const sendNotificationEvent = async (): Promise<{\n response: ISendEventResponse;\n content: IRTCNotificationContent;\n }> => {\n const content: IRTCNotificationContent = {\n \"m.mentions\": { user_ids: [], room: true },\n \"notification_type\": notificationType,\n \"m.relates_to\": {\n event_id: parentEventId,\n rel_type: RelationType.Reference,\n },\n \"sender_ts\": Date.now(),\n \"lifetime\": 30_000, // 30 seconds\n };\n if (callIntent) {\n content[\"m.call.intent\"] = callIntent;\n }\n const response = await this.client.sendEvent(this.roomSubset.roomId, EventType.RTCNotification, content);\n return { response, content };\n };\n\n void sendNotificationEvent()\n .then((notification) => {\n // Join event_id and origin event content\n const newResult = { ...notification.response, ...notification.content };\n this.emit(MatrixRTCSessionEvent.DidSendCallNotification, newResult);\n })\n .catch(([errorLegacy, errorNew]) =>\n this.logger.error(\"Failed to send call notification\", errorLegacy, errorNew),\n );\n }\n\n /**\n * Call this when the Matrix room members have changed.\n */\n private readonly onRoomMemberUpdate = (): void => {\n void this.ensureRecalculateSessionMembers();\n };\n\n /**\n * Call this when a sticky event update has occured.\n */\n private readonly onStickyEventUpdate: RoomStickyEventsMap[RoomStickyEventsEvent.Update] = (\n added,\n updated,\n removed,\n ): void => {\n if (\n [...added, ...removed, ...updated.flatMap((v) => [v.current, v.previous])].some(\n (e) => e.getType() === EventType.RTCMembership,\n )\n ) {\n void this.ensureRecalculateSessionMembers();\n }\n };\n\n /**\n * Call this when something changed that may impacts the current MatrixRTC members in this session.\n */\n // We allow this name schema since this function should only be used for testing purposes.\n // eslint-disable-next-line @typescript-eslint/naming-convention\n public _onRTCSessionMemberUpdate = async (): Promise<void> => {\n await this.recalculateSessionMembers();\n };\n\n // helper variables to make sure we do not have parallel running recalculations.\n private recalculateSessionMembersPromise: Promise<void> = Promise.resolve();\n\n /**\n * Ensures that membership is recalculated when the state of the session may have changed.\n * Also ensures that only one recalculation is made at a time.\n * @returns A promise resolving when the state has been recalculated.\n */\n private ensureRecalculateSessionMembers(): Promise<void> {\n if (this.membershipNeedsRecalculation) {\n // We have already requested recalcuation, don't attempt a new one.\n return this.recalculateSessionMembersPromise;\n }\n this.membershipNeedsRecalculation = true;\n // Chain the recalculation.\n this.recalculateSessionMembersPromise = this.recalculateSessionMembersPromise\n .finally()\n .then(() => this.recalculateSessionMembers());\n return this.recalculateSessionMembersPromise;\n }\n\n /**\n * Call this when anything that could impact rtc memberships has changed: Room Members or RTC members.\n *\n * Examines the latest call memberships and handles any encryption key sending or rotation that is needed.\n *\n * This function should be called when the room members or call memberships might have changed.\n */\n private readonly recalculateSessionMembers = async (): Promise<void> => {\n // Clear the flag.\n this.membershipNeedsRecalculation = false;\n const oldMemberships = this.memberships;\n\n this.memberships = await MatrixRTCSession.sessionMembershipsForSlot(\n this.room,\n this.slotDescription,\n this.calculateMembershipsOpts,\n );\n\n const changed =\n oldMemberships.length != this.memberships.length ||\n oldMemberships.some((m, i) => !CallMembership.equal(m, this.memberships[i]));\n\n if (changed) {\n this.logger.info(\n `Memberships for call in room ${this.roomSubset.roomId} have changed: emitting (${this.memberships.length} members)`,\n );\n logDurationSync(this.logger, \"emit MatrixRTCSessionEvent.MembershipsChanged\", () => {\n this.emit(MatrixRTCSessionEvent.MembershipsChanged, oldMemberships, this.memberships);\n });\n\n void this.membershipManager?.onRTCSessionMemberUpdate(this.memberships);\n // The `ownMembership` will be set when calling `onRTCSessionMemberUpdate`.\n const ownMembership = this.membershipManager?.ownMembership;\n if (this.pendingNotificationToSend && ownMembership && oldMemberships.length === 0) {\n // If we're the first member in the call, we're responsible for\n // sending the notification event\n if (ownMembership.eventId && this.joinConfig?.notificationType) {\n this.sendCallNotify(\n ownMembership.eventId,\n this.joinConfig.notificationType,\n ownMembership.callIntent,\n );\n } else {\n this.logger.warn(\"Own membership eventId is undefined, cannot send call notification\");\n }\n }\n // If anyone else joins the session it is no longer our responsibility to send the notification.\n // (If we were the joiner we already did sent the notification in the block above.)\n if (this.memberships.length > 0) this.pendingNotificationToSend = undefined;\n } else {\n this.logger.debug(`No membership changes detected for room ${this.roomSubset.roomId}`);\n }\n // This also needs to be done if `changed` = false\n // A member might have updated their fingerprint (created_ts)\n void this.encryptionManager?.onMembershipsUpdate(oldMemberships);\n\n this.setExpiryTimer();\n };\n}\n\n/// Private helpers\nasync function computeBackendIdentityAndVerifyMemberEvents(\n room: Pick<Room, \"hasMembershipState\">,\n callMemberEvents: MatrixEvent[],\n slotDescription: SlotDescription,\n logger: Logger,\n): Promise<CallMembership[]> {\n const callMemberships: CallMembership[] = [];\n\n for (const memberEvent of callMemberEvents) {\n const content = memberEvent.getContent();\n\n // Quick filter to avoid unneeded processing of invalid events or left events.\n if (!quickFilterNonRelevantContents(content, logger)) {\n continue;\n }\n\n try {\n const membership = await CallMembership.parseFromEvent(memberEvent);\n\n if (isValidMembership(membership, room, slotDescription, logger)) {\n callMemberships.push(membership);\n }\n } catch (e) {\n logger.warn(\"Couldn't construct call membership: \", e);\n }\n }\n\n return callMemberships;\n}\n\nfunction quickFilterNonRelevantContents(content: IContent, logger: Logger): boolean {\n // Ignore sticky keys for the count\n const eventKeysCount = Object.keys(content).filter((k) => k !== \"msc4354_sticky_key\").length;\n // Don't even bother about empty events (saves us from costly type/\"key in\" checks in bigger rooms)\n if (eventKeysCount === 0) return false;\n\n // We first decide if it's a MSC4143 event (per device state key)\n if (eventKeysCount > 1 && \"application\" in content) {\n // We have a MSC4143 event membership event with a proper joined content\n return true;\n } else if (eventKeysCount === 1 && \"memberships\" in content) {\n logger.warn(`Legacy event found. Those are ignored, they do not contribute to the MatrixRTC session`);\n return false;\n } else {\n // Invalid or left content\n return false;\n }\n}\n\nfunction isValidMembership(\n membership: CallMembership,\n room: Pick<Room, \"hasMembershipState\">,\n slotDescription: SlotDescription,\n logger: Logger,\n): boolean {\n if (membership.slotDescription.id !== slotDescription.id) {\n logger.info(\n `Ignoring membership of user ${membership.userId} for a different slot. Theirs: ${JSON.stringify(membership.slotDescription)}, Expected: ${JSON.stringify(slotDescription)}`,\n );\n return false;\n }\n\n if (membership.isExpired()) {\n logger.info(`Ignoring expired device membership ${membership.userId}/${membership.deviceId}`);\n return false;\n }\n\n if (!room.hasMembershipState(membership.userId ?? \"\", KnownMembership.Join)) {\n logger.info(`Ignoring membership of user ${membership.userId} who is not in the room.`);\n return false;\n }\n\n return true;\n}\n\n/**\n * Collects the raw member events from room state and sticky store.\n */\nfunction collectMembersEvents(\n room: Pick<Room, \"getLiveTimeline\" | \"roomId\" | \"_unstable_getStickyEvents\">,\n options: SessionMembershipsForSlotOpts,\n logger: Logger,\n): MatrixEvent[] {\n const { listenForStickyEvents, listenForMemberStateEvents } = options;\n let callMemberEvents: MatrixEvent[] = [];\n if (listenForStickyEvents) {\n // prefill with sticky events\n callMemberEvents = [...room._unstable_getStickyEvents()].filter((e) => e.getType() === EventType.RTCMembership);\n }\n if (listenForMemberStateEvents) {\n const roomState = room.getLiveTimeline().getState(EventTimeline.FORWARDS);\n if (!roomState) {\n logger.warn(\"Couldn't get state for room \" + room.roomId + \"using empty membership array\");\n return [];\n }\n const callMemberStateEvents = roomState.getStateEvents(EventType.GroupCallMemberPrefix);\n callMemberEvents = callMemberEvents.concat(\n callMemberStateEvents.filter(\n (callMemberStateEvent) =>\n !callMemberEvents.some(\n // only care about state events which have keys which we have not yet seen in the sticky events.\n // TODO: I believe this can discard a joined state event if there is a matching left sticky event.\n (stickyEvent) =>\n stickyEvent.getContent().msc4354_sticky_key === callMemberStateEvent.getStateKey(),\n ),\n ),\n );\n }\n return callMemberEvents;\n}\n"],"mappings":"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","ignoreList":[]}
1
+ {"version":3,"file":"MatrixRTCSession.js","names":["logger","rootLogger","TypedEventEmitter","EventTimeline","EventType","RelationType","KnownMembership","CallMembership","RoomStateEvent","MembershipManager","StickyEventMembershipManager","logDurationSync","MembershipManagerEvent","RTCEncryptionManager","ToDeviceKeyTransport","TypedReEmitter","RoomStickyEventsEvent","computeSlotId","MatrixRTCSessionEvent","DEFAULT_SESSION_MEMBERSHIPS_FOR_SLOT_OPTS","listenForStickyEvents","listenForMemberStateEvents","MatrixRTCSession","membershipStatus","_this$membershipManag","membershipManager","status","probablyLeft","_this$membershipManag2","delayId","_this$membershipManag3","callId","_this$slotDescription","slotDescription","id","slotId","sessionMembershipsForSlot","room","_arguments","arguments","_asyncToGenerator","options","length","undefined","getChild","concat","roomId","application","callMemberEvents","collectMembersEvents","callMemberships","computeBackendIdentityAndVerifyMemberEvents","sort","a","b","createdTs","debug","map","m","userId","sessionForSlot","client","opts","roomSubset","constructor","calculateMembershipsOpts","_this","this","_defineProperty","counters","roomEventEncryptionKeysSent","roomEventEncryptionKeysReceived","totals","roomEventEncryptionKeysReceivedTotalAge","ensureRecalculateSessionMembers","added","updated","removed","flatMap","v","current","previous","some","e","getType","RTCMembership","recalculateSessionMembers","Promise","resolve","_this$encryptionManag","membershipNeedsRecalculation","oldMemberships","memberships","changed","i","equal","_this$membershipManag4","_this$membershipManag5","info","emit","MembershipsChanged","onRTCSessionMemberUpdate","ownMembership","pendingNotificationToSend","_this$joinConfig","eventId","joinConfig","notificationType","sendCallNotify","callIntent","warn","encryptionManager","onMembershipsUpdate","setExpiryTimer","on","Members","onRoomMemberUpdate","Update","onStickyEventUpdate","initialMembershipCalculated","isJoined","_this$membershipManag6","_this$membershipManag7","stop","_this2","_this2$membershipMana","leave","expiryTimeout","clearTimeout","off","joinRTCSession","ownMembershipIdentity","fociPreferred","multiSfuFocus","_this$joinConfig2","unstableSendStickyEvents","memberId","reEmitter","reEmit","ProbablyLeft","StatusChanged","DelayIdChanged","statistics","transport","keyBin","encryptionKeyIndex","membership","rtcBackendIdentity","EncryptionKeyChanged","join","error","MembershipManagerError","JoinStateChanged","joinRoomSession","deviceId","getUserId","getDeviceId","leaveRoomSession","_arguments2","_this3","timeout","leavePromise","getFocusInUse","oldestMembership","getOldestMembership","getTransport","getConsensusCallIntent","_this$memberships$fin","getFirstCallIntent","find","every","updateCallIntent","_this4","_this4$membershipMana","_this4$membershipMana2","myMembership","Error","reemitEncryptionKeys","_this$encryptionManag2","getEncryptionKeys","forEach","keyRing","key","keyInfo","keyIndex","soonestExpiry","thisExpiry","getMsUntilExpiry","setTimeout","parentEventId","_this5","sendNotificationEvent","_ref3","content","user_ids","event_id","rel_type","Reference","Date","now","response","sendEvent","RTCNotification","apply","then","notification","newResult","_objectSpread","DidSendCallNotification","catch","_ref4","errorLegacy","errorNew","recalculateSessionMembersPromise","finally","_x","_x2","_x3","_x4","_computeBackendIdentityAndVerifyMemberEvents","memberEvent","getContent","quickFilterNonRelevantContents","parseFromEvent","isValidMembership","push","eventKeysCount","Object","keys","filter","k","_membership$userId","JSON","stringify","isExpired","hasMembershipState","Join","_unstable_getStickyEvents","roomState","getLiveTimeline","getState","FORWARDS","callMemberStateEvents","getStateEvents","GroupCallMemberPrefix","callMemberStateEvent","stickyEvent","msc4354_sticky_key","getStateKey"],"sources":["../../src/matrixrtc/MatrixRTCSession.ts"],"sourcesContent":["/*\nCopyright 2023 - 2026 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\nimport { type Logger, logger as rootLogger } from \"../logger.ts\";\nimport { TypedEventEmitter } from \"../models/typed-event-emitter.ts\";\nimport { EventTimeline } from \"../models/event-timeline.ts\";\nimport { type Room } from \"../models/room.ts\";\nimport { type MatrixClient } from \"../client.ts\";\nimport { EventType, RelationType } from \"../@types/event.ts\";\nimport { KnownMembership } from \"../@types/membership.ts\";\nimport { type ISendEventResponse } from \"../@types/requests.ts\";\nimport { CallMembership } from \"./CallMembership.ts\";\nimport { RoomStateEvent } from \"../models/room-state.ts\";\nimport { MembershipManager, StickyEventMembershipManager } from \"./MembershipManager.ts\";\nimport { type CallMembershipIdentityParts, type IEncryptionManager } from \"./EncryptionManager.ts\";\nimport { logDurationSync } from \"../utils.ts\";\nimport type {\n Statistics,\n RTCNotificationType,\n Status,\n IRTCNotificationContent,\n RTCCallIntent,\n Transport,\n SlotDescription,\n} from \"./types.ts\";\nimport {\n MembershipManagerEvent,\n type MembershipManagerEventHandlerMap,\n type IMembershipManager,\n} from \"./IMembershipManager.ts\";\nimport { RTCEncryptionManager } from \"./RTCEncryptionManager.ts\";\nimport { ToDeviceKeyTransport } from \"./ToDeviceKeyTransport.ts\";\nimport { TypedReEmitter } from \"../ReEmitter.ts\";\nimport { type IContent, type MatrixEvent } from \"../models/event.ts\";\nimport { RoomStickyEventsEvent, type RoomStickyEventsMap } from \"../models/room-sticky-events.ts\";\nimport { computeSlotId } from \"./utils.ts\";\n\n/**\n * Events emitted by MatrixRTCSession\n */\nexport enum MatrixRTCSessionEvent {\n // A member joined, left, or updated a property of their membership.\n MembershipsChanged = \"memberships_changed\",\n // We joined or left the session: our own local idea of whether we are joined,\n // separate from MembershipsChanged, ie. independent of whether our member event\n // has successfully gone through.\n JoinStateChanged = \"join_state_changed\",\n // The key used to encrypt media has changed\n EncryptionKeyChanged = \"encryption_key_changed\",\n /** The membership manager had to shut down caused by an unrecoverable error */\n MembershipManagerError = \"membership_manager_error\",\n /** The RTCSession did send a call notification caused by joining the call as the first member */\n DidSendCallNotification = \"did_send_call_notification\",\n}\n\nexport type MatrixRTCSessionEventHandlerMap = {\n [MatrixRTCSessionEvent.MembershipsChanged]: (\n oldMemberships: CallMembership[],\n newMemberships: CallMembership[],\n ) => void;\n [MatrixRTCSessionEvent.JoinStateChanged]: (isJoined: boolean) => void;\n [MatrixRTCSessionEvent.EncryptionKeyChanged]: (\n key: Uint8Array<ArrayBuffer>,\n encryptionKeyIndex: number,\n membership: CallMembershipIdentityParts,\n rtcBackendIdentity: string,\n ) => void;\n [MatrixRTCSessionEvent.MembershipManagerError]: (error: unknown) => void;\n [MatrixRTCSessionEvent.DidSendCallNotification]: (\n notificationContentNew: { event_id: string } & IRTCNotificationContent,\n ) => void;\n};\n\nexport interface SessionConfig {\n /**\n * What kind of notification to send when starting the session.\n * @default `undefined` (no notification)\n */\n notificationType?: RTCNotificationType;\n\n /**\n * Determines the kind of call this will be.\n */\n callIntent?: RTCCallIntent;\n}\n\n// The names follow these principles:\n// - we use the technical term delay if the option is related to delayed events.\n// - we use delayedLeaveEvent if the option is related to the delayed leave event.\n// - we use membershipEvent if the option is related to the rtc member state event.\n// - we use the technical term expiry if the option is related to the expiry field of the membership state event.\n// - we use a `Ms` postfix if the option is a duration to avoid using words like:\n// `time`, `duration`, `delay`, `timeout`... that might be mistaken/confused with technical terms.\nexport interface MembershipConfig {\n /**\n * The timeout (in milliseconds) after we joined the call, that our membership should expire\n * unless we have explicitly updated it.\n *\n * This is what goes into the m.rtc.member event expiry field and is typically set to a number of hours.\n */\n membershipEventExpiryMs?: number;\n\n /**\n * The time in (in milliseconds) which the manager will prematurely send the updated state event before the membership `expires` time to make sure it\n * sends the updated state event early enough.\n *\n * A headroom of 1000ms and a `membershipExpiryTimeout` of 10000ms would result in the first membership event update after 9s and\n * a membership event that would be considered expired after 10s.\n *\n * This value does not have an effect on the value of `SessionMembershipData.expires`.\n */\n membershipEventExpiryHeadroomMs?: number;\n\n /**\n * The timeout (in milliseconds) with which the deleayed leave event on the server is configured.\n * After this time the server will set the event to the disconnected stat if it has not received a keep-alive from the client.\n */\n delayedLeaveEventDelayMs?: number;\n\n /**\n * The interval (in milliseconds) in which the client will send membership keep-alives to the server.\n */\n delayedLeaveEventRestartMs?: number;\n\n /**\n * The maximum number of retries that the manager will do for delayed event sending/updating and state event sending when a server rate limit has been hit.\n */\n maximumRateLimitRetryCount?: number;\n\n /**\n * The maximum number of retries that the manager will do for delayed event sending/updating and state event sending when a network error occurs.\n */\n maximumNetworkErrorRetryCount?: number;\n\n /**\n * The time (in milliseconds) after which we will retry a http request if it\n * failed to send due to a network error. (send membership event, send delayed event, restart delayed event...)\n */\n networkErrorRetryMs?: number;\n\n /**\n * The time (in milliseconds) after which a we consider a delayed event restart http request to have failed.\n * Setting this to a lower value will result in more frequent retries but also a higher chance of failiour.\n *\n * In the presence of network packet loss (hurting TCP connections), the custom delayedEventRestartLocalTimeoutMs\n * helps by keeping more delayed event reset candidates in flight,\n * improving the chances of a successful reset. (its is equivalent to the js-sdk `localTimeout` configuration,\n * but only applies to calls to the `_unstable_restartScheduledDelayedEvent` endpoint\n * or the `_unstable_updateDelayedEvent` endpoint with a body of `{action:\"restart\"}`.)\n */\n delayedLeaveEventRestartLocalTimeoutMs?: number;\n\n /**\n * Send membership using sticky events rather than state events.\n * This also make the client use the new m.rtc.member MSC4354 event format. (instead of m.call.member)\n *\n * **WARNING**: This is an unstable feature and not all clients will support it.\n */\n unstableSendStickyEvents?: boolean;\n}\n\nexport interface EncryptionConfig {\n /**\n * If true, generate and share a media key for this participant,\n * and emit MatrixRTCSessionEvent.EncryptionKeyChanged when\n * media keys for other participants become available.\n */\n manageMediaKeys?: boolean;\n /**\n * The minimum time (in milliseconds) between each attempt to send encryption key(s).\n * e.g. if this is set to 1000, then we will send at most one key event every second.\n * @deprecated - Not used by the new encryption manager.\n */\n updateEncryptionKeyThrottle?: number;\n\n /**\n * Sometimes it is necessary to rotate the encryption key after a membership update.\n * For performance reasons we might not want to rotate the key immediately but allow future memberships to use the same key.\n * If 5 people join in a row in less than 5 seconds, we don't want to rotate the key for each of them.\n * If 5 people leave in a row in less than 5 seconds, we don't want to rotate the key for each of them.\n * So we do share the key which was already used live for <5s to new joiners.\n * This does result in a potential leak up to the configured time of call media.\n * This has to be considered when choosing a value for this property.\n */\n keyRotationGracePeriodMs?: number;\n\n /**\n * The delay (in milliseconds) after a member leaves before we create and publish a new key, because people\n * tend to leave calls at the same time.\n * @deprecated - Not used by the new encryption manager.\n */\n makeKeyDelay?: number;\n /**\n * The delay (in milliseconds) between sending a new key and starting to encrypt with it. This\n * gives others a chance to receive the new key to minimize the chance they get media they can't decrypt.\n *\n * The higher this value is, the better it is for existing members as they will have a smoother experience.\n * But it impacts new joiners: They will always have to wait `useKeyDelay` before being able to decrypt the media\n * (as it will be encrypted with the new key after the delay only), even if the key has already arrived before the delay.\n */\n useKeyDelay?: number;\n}\nexport type JoinSessionConfig = SessionConfig & MembershipConfig & EncryptionConfig;\n\ninterface SessionMembershipsForSlotOpts {\n /**\n * Listen for incoming sticky member events. If disabled, this session will\n * ignore any incoming sticky events.\n */\n listenForStickyEvents: boolean;\n /**\n * Listen for incoming member state events (legacy). If disabled, this session will\n * ignore any incoming state events.\n */\n listenForMemberStateEvents: boolean;\n}\n\nconst DEFAULT_SESSION_MEMBERSHIPS_FOR_SLOT_OPTS: SessionMembershipsForSlotOpts = {\n listenForStickyEvents: true,\n listenForMemberStateEvents: true,\n};\n\n/**\n * A MatrixRTCSession manages the membership & properties of a MatrixRTC session.\n * This class doesn't deal with media at all, just membership & properties of a session.\n */\nexport class MatrixRTCSession extends TypedEventEmitter<\n MatrixRTCSessionEvent | MembershipManagerEvent,\n MatrixRTCSessionEventHandlerMap & MembershipManagerEventHandlerMap\n> {\n private membershipManager?: IMembershipManager;\n private encryptionManager?: IEncryptionManager;\n private joinConfig?: SessionConfig;\n private logger: Logger;\n\n private pendingNotificationToSend: undefined | RTCNotificationType;\n /**\n * This timeout is responsible to track any expiration. We need to know when we have to start\n * to ignore other call members. There is no callback for this. This timeout will always be configured to\n * emit when the next membership expires.\n */\n private expiryTimeout?: ReturnType<typeof setTimeout>;\n\n public memberships: CallMembership[] = [];\n\n /**\n * Resolves when the session has calculated the initial membership of the session.\n */\n public readonly initialMembershipCalculated: Promise<void>;\n /**\n * Does membership need to be recalculated? This is set to false upon\n * recalculation.\n */\n private membershipNeedsRecalculation = false;\n\n /**\n * The statistics for this session.\n */\n public statistics: Statistics = {\n counters: {\n roomEventEncryptionKeysSent: 0,\n roomEventEncryptionKeysReceived: 0,\n },\n totals: {\n roomEventEncryptionKeysReceivedTotalAge: 0,\n },\n };\n\n public get membershipStatus(): Status | undefined {\n return this.membershipManager?.status;\n }\n public get probablyLeft(): boolean | undefined {\n return this.membershipManager?.probablyLeft;\n }\n public get delayId(): string | undefined {\n return this.membershipManager?.delayId;\n }\n\n /**\n * The callId (sessionId) of the call.\n *\n * It can be undefined since the callId is only known once the first membership joins.\n * The callId is the property that, per definition, groups memberships into one call.\n * @deprecated use `slotId` instead.\n */\n public get callId(): string | undefined {\n return this.slotDescription?.id;\n }\n /**\n * The slotId of the call.\n * `{application}#{appSpecificId}`\n * It can be undefined since the slotId is only known once the first membership joins.\n * The slotId is the property that, per definition, groups memberships into one call.\n */\n public get slotId(): string | undefined {\n return computeSlotId(this.slotDescription);\n }\n\n /**\n * Returns all the call memberships for a room that match the provided `sessionDescription`,\n * oldest first.\n *\n * By default, this will return *both* sticky and member state events.\n */\n public static async sessionMembershipsForSlot(\n room: Pick<Room, \"getLiveTimeline\" | \"roomId\" | \"hasMembershipState\" | \"_unstable_getStickyEvents\">,\n slotDescription: SlotDescription,\n // default both true this implied we combine sticky and state events for the final call state\n // (prefer sticky events in case of a duplicate)\n options: SessionMembershipsForSlotOpts = DEFAULT_SESSION_MEMBERSHIPS_FOR_SLOT_OPTS,\n ): Promise<CallMembership[]> {\n const logger = rootLogger.getChild(\n `[MatrixRTCSession ${room.roomId} ${slotDescription.application}#${slotDescription.id}]`,\n );\n const callMemberEvents = collectMembersEvents(room, options, logger);\n\n const callMemberships = await computeBackendIdentityAndVerifyMemberEvents(\n room,\n callMemberEvents,\n slotDescription,\n logger,\n );\n\n callMemberships.sort((a, b) => a.createdTs() - b.createdTs());\n if (callMemberships.length > 1) {\n logger.debug(\n `Call memberships in room ${room.roomId}, in order: `,\n callMemberships.map((m) => [m.createdTs(), m.userId]),\n );\n }\n\n return callMemberships;\n }\n\n /**\n * Return the MatrixRTC session for the room.\n * This returned session can be used to find out if there are active sessions\n * for the requested room and `slotDescription`.\n */\n public static sessionForSlot(\n client: MatrixClient,\n room: Room,\n slotDescription: SlotDescription,\n opts?: SessionMembershipsForSlotOpts,\n ): MatrixRTCSession {\n return new MatrixRTCSession(client, room, slotDescription, opts);\n }\n\n /**\n * WARN: this can in theory only be a subset of the room with the properties required by\n * this class.\n * Outside of tests this most likely will be a full room, however.\n * @deprecated Relying on a full Room object being available here is an anti-pattern. You should be tracking\n * the room object in your own code and passing it in when needed.\n */\n public get room(): Room {\n return this.roomSubset as Room;\n }\n\n /**\n * This constructs a room session. When using MatrixRTC inside the js-sdk this is expected\n * to be used with the MatrixRTCSessionManager exclusively.\n *\n * In cases where you don't use the js-sdk but build on top of another Matrix stack this class can be used standalone\n * to manage a joined MatrixRTC session.\n *\n * @param client A subset of the {@link MatrixClient} that lets the session interact with the Matrix room.\n * @param roomSubset The room this session is attached to. A subset of a js-sdk Room that the session needs.\n * @param slotDescription The slot description is a virtual address where participants are allowed to meet.\n * This session will only manage memberships that match this slot description.Sessions are distinct if any of\n * those properties are distinct: `roomSubset.roomId`, `slotDescription.application`, `slotDescription.id`.\n * @param calculateMembershipsOpts - Options to configure how memberships are calculated for this session.\n */\n public constructor(\n private readonly client: Pick<\n MatrixClient,\n | \"getUserId\"\n | \"getDeviceId\"\n | \"sendEvent\"\n | \"sendStateEvent\"\n | \"_unstable_sendDelayedStateEvent\"\n | \"_unstable_updateDelayedEvent\"\n | \"_unstable_cancelScheduledDelayedEvent\"\n | \"_unstable_restartScheduledDelayedEvent\"\n | \"_unstable_sendScheduledDelayedEvent\"\n | \"_unstable_sendStickyEvent\"\n | \"_unstable_sendStickyDelayedEvent\"\n | \"cancelPendingEvent\"\n | \"encryptAndSendToDevice\"\n | \"off\"\n | \"on\"\n | \"decryptEventIfNeeded\"\n >,\n private roomSubset: Pick<\n Room,\n | \"getLiveTimeline\"\n | \"roomId\"\n | \"getVersion\"\n | \"hasMembershipState\"\n | \"on\"\n | \"off\"\n | \"_unstable_getStickyEvents\"\n >,\n\n public readonly slotDescription: SlotDescription,\n private readonly calculateMembershipsOpts?: SessionMembershipsForSlotOpts,\n ) {\n super();\n this.logger = rootLogger.getChild(\n `[MatrixRTCSession ${roomSubset.roomId} ${slotDescription.application}#${slotDescription.id}]`,\n );\n\n this.roomSubset.on(RoomStateEvent.Members, this.onRoomMemberUpdate);\n this.roomSubset.on(RoomStickyEventsEvent.Update, this.onStickyEventUpdate);\n\n this.initialMembershipCalculated = this.ensureRecalculateSessionMembers();\n this.setExpiryTimer();\n }\n /*\n * Returns true if we intend to be participating in the MatrixRTC session.\n * This is determined by checking if the relativeExpiry has been set.\n */\n public isJoined(): boolean {\n return this.membershipManager?.isJoined() ?? false;\n }\n\n /**\n * Performs cleanup & removes timers for client shutdown\n */\n public async stop(): Promise<void> {\n await this.membershipManager?.leave(1000);\n if (this.expiryTimeout) {\n clearTimeout(this.expiryTimeout);\n this.expiryTimeout = undefined;\n }\n\n this.roomSubset.off(RoomStateEvent.Members, this.onRoomMemberUpdate);\n this.roomSubset.off(RoomStickyEventsEvent.Update, this.onStickyEventUpdate);\n }\n\n private reEmitter = new TypedReEmitter<\n MatrixRTCSessionEvent | MembershipManagerEvent,\n MatrixRTCSessionEventHandlerMap & MembershipManagerEventHandlerMap\n >(this);\n\n /**\n * Announces this user and device as joined to the MatrixRTC session,\n * and continues to update the membership event to keep it valid until\n * leaveRoomSession() is called\n * This will not subscribe to updates: remember to call subscribe() separately if\n * desired.\n * This method will return immediately and the session will be joined in the background.\n * @param ownMembershipIdentity the identity of the user and device joining the session.\n * This will be put into the content.member.\n * @param fociPreferred the list of preferred foci to use in the joined RTC membership event.\n * If multiSfuFocus is set, this is only needed if this client wants to publish to multiple transports simultaneously.\n * @param multiSfuFocus the active focus to use in the joined RTC membership event. Setting this implies the\n * membership manager will operate in a multi-SFU connection mode. If `undefined`, an `oldest_membership`\n * transport selection will be used instead.\n * @param joinConfig - Additional configuration for the joined session.\n */\n public joinRTCSession(\n ownMembershipIdentity: CallMembershipIdentityParts,\n fociPreferred: Transport[],\n multiSfuFocus?: Transport,\n joinConfig?: JoinSessionConfig,\n ): void {\n if (this.isJoined()) {\n this.logger.info(`Already joined to session in room ${this.roomSubset.roomId}: ignoring join call`);\n return;\n } else {\n // Create MembershipManager and pass the RTCSession logger (with room id info)\n this.membershipManager = joinConfig?.unstableSendStickyEvents\n ? new StickyEventMembershipManager(\n joinConfig,\n this.roomSubset,\n this.client,\n this.slotDescription,\n ownMembershipIdentity.memberId,\n this.logger,\n )\n : new MembershipManager(joinConfig, this.roomSubset, this.client, this.slotDescription, this.logger);\n\n this.reEmitter.reEmit(this.membershipManager!, [\n MembershipManagerEvent.ProbablyLeft,\n MembershipManagerEvent.StatusChanged,\n MembershipManagerEvent.DelayIdChanged,\n ]);\n // Create Encryption manager\n const [room, client, statistics] = [this.roomSubset, this.client, this.statistics];\n const transport = new ToDeviceKeyTransport(ownMembershipIdentity, room.roomId, client, statistics);\n this.encryptionManager = new RTCEncryptionManager(\n ownMembershipIdentity,\n () => this.memberships,\n transport,\n (\n keyBin: Uint8Array<ArrayBuffer>,\n encryptionKeyIndex: number,\n membership: CallMembershipIdentityParts,\n rtcBackendIdentity: string,\n ) => {\n this.emit(\n MatrixRTCSessionEvent.EncryptionKeyChanged,\n keyBin,\n encryptionKeyIndex,\n membership,\n rtcBackendIdentity,\n );\n },\n this.logger,\n );\n }\n\n this.joinConfig = joinConfig;\n this.pendingNotificationToSend = this.joinConfig?.notificationType;\n\n // Join!\n this.membershipManager!.join(fociPreferred, multiSfuFocus, (e) => {\n this.logger.error(\"MembershipManager encountered an unrecoverable error: \", e);\n this.emit(MatrixRTCSessionEvent.MembershipManagerError, e);\n this.emit(MatrixRTCSessionEvent.JoinStateChanged, this.isJoined());\n });\n this.encryptionManager!.join(joinConfig);\n\n this.emit(MatrixRTCSessionEvent.JoinStateChanged, true);\n }\n\n /**\n *\n * @param fociPreferred\n * @param multiSfuFocus\n * @param joinConfig\n * @deprecated use the joinRTCSession method instead\n */\n public joinRoomSession(\n fociPreferred: Transport[],\n multiSfuFocus?: Transport,\n joinConfig?: JoinSessionConfig,\n ): void {\n const [userId, deviceId] = [this.client.getUserId()!, this.client.getDeviceId()!];\n // TODO this wants to become a UUID\n const memberId = `${userId}:${deviceId}`;\n this.joinRTCSession({ userId, deviceId, memberId }, fociPreferred, multiSfuFocus, joinConfig);\n }\n\n /**\n * Announces this user and device as having left the MatrixRTC session\n * and stops scheduled updates.\n * This will not unsubscribe from updates: remember to call unsubscribe() separately if\n * desired.\n * The membership update required to leave the session will retry if it fails.\n * Without network connection the promise will never resolve.\n * A timeout can be provided so that there is a guarantee for the promise to resolve.\n * @returns Whether the membership update was attempted and did not time out.\n */\n public async leaveRoomSession(timeout: number | undefined = undefined): Promise<boolean> {\n if (!this.isJoined()) {\n this.logger.info(`Not joined to session in room ${this.roomSubset.roomId}: ignoring leave call`);\n return false;\n }\n\n this.logger.info(`Leaving call session in room ${this.roomSubset.roomId}`);\n\n this.encryptionManager!.leave();\n\n const leavePromise = this.membershipManager!.leave(timeout);\n this.emit(MatrixRTCSessionEvent.JoinStateChanged, false);\n\n return await leavePromise;\n }\n /**\n * This returns the focus in use by the oldest membership.\n * Do not use since this might be just the focus for the oldest membership. others might use a different focus.\n * @deprecated use `member.getTransport(session.getOldestMembership())` instead for the specific member you want to get the focus for.\n */\n public getFocusInUse(): Transport | undefined {\n const oldestMembership = this.getOldestMembership();\n return oldestMembership?.getTransport(oldestMembership);\n }\n\n public getOldestMembership(): CallMembership | undefined {\n return this.memberships[0];\n }\n\n /**\n * Get the call intent for the current call, based on what members are advertising. If one or more\n * members disagree on the current call intent, or nobody specifies one then `undefined` is returned.\n *\n * If all members that specify a call intent agree, that value is returned.\n * @returns A call intent, or `undefined` if no consensus or not given.\n */\n public getConsensusCallIntent(): RTCCallIntent | undefined {\n const getFirstCallIntent = this.memberships.find((m) => !!m.callIntent)?.callIntent;\n if (!getFirstCallIntent) {\n return undefined;\n }\n if (this.memberships.every((m) => !m.callIntent || m.callIntent === getFirstCallIntent)) {\n return getFirstCallIntent;\n }\n return undefined;\n }\n\n public async updateCallIntent(callIntent: RTCCallIntent): Promise<void> {\n const myMembership = this.membershipManager?.ownMembership;\n if (!myMembership) {\n throw Error(\"Not connected yet\");\n }\n await this.membershipManager?.updateCallIntent(callIntent);\n }\n\n /**\n * Re-emit an EncryptionKeyChanged event for each tracked encryption key. This can be used to export\n * the keys.\n */\n public reemitEncryptionKeys(): void {\n this.encryptionManager?.getEncryptionKeys().forEach((keyRing, key) => {\n keyRing.forEach((keyInfo) => {\n this.emit(\n MatrixRTCSessionEvent.EncryptionKeyChanged,\n keyInfo.key,\n keyInfo.keyIndex,\n keyInfo.membership,\n keyInfo.rtcBackendIdentity,\n );\n });\n });\n }\n\n /**\n * Sets a timer for the soonest membership expiry\n */\n private setExpiryTimer(): void {\n if (this.expiryTimeout) {\n clearTimeout(this.expiryTimeout);\n this.expiryTimeout = undefined;\n }\n\n let soonestExpiry;\n for (const membership of this.memberships) {\n const thisExpiry = membership.getMsUntilExpiry();\n // If getMsUntilExpiry is undefined we have a MSC4143 (MatrixRTC) compliant event - it never expires\n // but will be reliably resent on disconnect.\n if (thisExpiry !== undefined && (soonestExpiry === undefined || thisExpiry < soonestExpiry)) {\n soonestExpiry = thisExpiry;\n }\n }\n\n if (soonestExpiry != undefined) {\n this.expiryTimeout = setTimeout(() => void this.ensureRecalculateSessionMembers(), soonestExpiry);\n }\n }\n\n /**\n * Sends notification events to indiciate the call has started.\n * Note: This does not return a promise, instead scheduling the notification events to be sent.\n * @param parentEventId Event id linking to your RTC call membership event.\n * @param notificationType The type of notification to send\n * @param callIntent The type of call this is (e.g. \"audio\").\n */\n private sendCallNotify(\n parentEventId: string,\n notificationType: RTCNotificationType,\n callIntent?: RTCCallIntent,\n ): void {\n const sendNotificationEvent = async (): Promise<{\n response: ISendEventResponse;\n content: IRTCNotificationContent;\n }> => {\n const content: IRTCNotificationContent = {\n \"m.mentions\": { user_ids: [], room: true },\n \"notification_type\": notificationType,\n \"m.relates_to\": {\n event_id: parentEventId,\n rel_type: RelationType.Reference,\n },\n \"sender_ts\": Date.now(),\n \"lifetime\": 30_000, // 30 seconds\n };\n if (callIntent) {\n content[\"m.call.intent\"] = callIntent;\n }\n const response = await this.client.sendEvent(this.roomSubset.roomId, EventType.RTCNotification, content);\n return { response, content };\n };\n\n void sendNotificationEvent()\n .then((notification) => {\n // Join event_id and origin event content\n const newResult = { ...notification.response, ...notification.content };\n this.emit(MatrixRTCSessionEvent.DidSendCallNotification, newResult);\n })\n .catch(([errorLegacy, errorNew]) =>\n this.logger.error(\"Failed to send call notification\", errorLegacy, errorNew),\n );\n }\n\n /**\n * Call this when the Matrix room members have changed.\n */\n private readonly onRoomMemberUpdate = (): void => {\n void this.ensureRecalculateSessionMembers();\n };\n\n /**\n * Call this when a sticky event update has occured.\n */\n private readonly onStickyEventUpdate: RoomStickyEventsMap[RoomStickyEventsEvent.Update] = (\n added,\n updated,\n removed,\n ): void => {\n if (\n [...added, ...removed, ...updated.flatMap((v) => [v.current, v.previous])].some(\n (e) => e.getType() === EventType.RTCMembership,\n )\n ) {\n void this.ensureRecalculateSessionMembers();\n }\n };\n\n /**\n * Call this when something changed that may impacts the current MatrixRTC members in this session.\n */\n // We allow this name schema since this function should only be used for testing purposes.\n // eslint-disable-next-line @typescript-eslint/naming-convention\n public _onRTCSessionMemberUpdate = async (): Promise<void> => {\n await this.recalculateSessionMembers();\n };\n\n // helper variables to make sure we do not have parallel running recalculations.\n private recalculateSessionMembersPromise: Promise<void> = Promise.resolve();\n\n /**\n * Ensures that membership is recalculated when the state of the session may have changed.\n * Also ensures that only one recalculation is made at a time.\n * @returns A promise resolving when the state has been recalculated.\n */\n private ensureRecalculateSessionMembers(): Promise<void> {\n if (this.membershipNeedsRecalculation) {\n // We have already requested recalcuation, don't attempt a new one.\n return this.recalculateSessionMembersPromise;\n }\n this.membershipNeedsRecalculation = true;\n // Chain the recalculation.\n this.recalculateSessionMembersPromise = this.recalculateSessionMembersPromise\n .finally()\n .then(() => this.recalculateSessionMembers());\n return this.recalculateSessionMembersPromise;\n }\n\n /**\n * Call this when anything that could impact rtc memberships has changed: Room Members or RTC members.\n *\n * Examines the latest call memberships and handles any encryption key sending or rotation that is needed.\n *\n * This function should be called when the room members or call memberships might have changed.\n */\n private readonly recalculateSessionMembers = async (): Promise<void> => {\n // Clear the flag.\n this.membershipNeedsRecalculation = false;\n const oldMemberships = this.memberships;\n\n this.memberships = await MatrixRTCSession.sessionMembershipsForSlot(\n this.room,\n this.slotDescription,\n this.calculateMembershipsOpts,\n );\n\n const changed =\n oldMemberships.length != this.memberships.length ||\n oldMemberships.some((m, i) => !CallMembership.equal(m, this.memberships[i]));\n\n if (changed) {\n this.logger.info(\n `Memberships for call in room ${this.roomSubset.roomId} have changed: emitting (${this.memberships.length} members)`,\n );\n logDurationSync(this.logger, \"emit MatrixRTCSessionEvent.MembershipsChanged\", () => {\n this.emit(MatrixRTCSessionEvent.MembershipsChanged, oldMemberships, this.memberships);\n });\n\n void this.membershipManager?.onRTCSessionMemberUpdate(this.memberships);\n // The `ownMembership` will be set when calling `onRTCSessionMemberUpdate`.\n const ownMembership = this.membershipManager?.ownMembership;\n if (this.pendingNotificationToSend && ownMembership && oldMemberships.length === 0) {\n // If we're the first member in the call, we're responsible for\n // sending the notification event\n if (ownMembership.eventId && this.joinConfig?.notificationType) {\n this.sendCallNotify(\n ownMembership.eventId,\n this.joinConfig.notificationType,\n ownMembership.callIntent,\n );\n } else {\n this.logger.warn(\"Own membership eventId is undefined, cannot send call notification\");\n }\n }\n // If anyone else joins the session it is no longer our responsibility to send the notification.\n // (If we were the joiner we already did sent the notification in the block above.)\n if (this.memberships.length > 0) this.pendingNotificationToSend = undefined;\n } else {\n this.logger.debug(`No membership changes detected for room ${this.roomSubset.roomId}`);\n }\n // This also needs to be done if `changed` = false\n // A member might have updated their fingerprint (created_ts)\n void this.encryptionManager?.onMembershipsUpdate(oldMemberships);\n\n this.setExpiryTimer();\n };\n}\n\n/// Private helpers\nasync function computeBackendIdentityAndVerifyMemberEvents(\n room: Pick<Room, \"hasMembershipState\">,\n callMemberEvents: MatrixEvent[],\n slotDescription: SlotDescription,\n logger: Logger,\n): Promise<CallMembership[]> {\n const callMemberships: CallMembership[] = [];\n\n for (const memberEvent of callMemberEvents) {\n const content = memberEvent.getContent();\n\n // Quick filter to avoid unneeded processing of invalid events or left events.\n if (!quickFilterNonRelevantContents(content, logger)) {\n continue;\n }\n\n try {\n const membership = await CallMembership.parseFromEvent(memberEvent);\n\n if (isValidMembership(membership, room, slotDescription, logger)) {\n callMemberships.push(membership);\n }\n } catch (e) {\n logger.warn(\"Couldn't construct call membership: \", e);\n }\n }\n\n return callMemberships;\n}\n\nfunction quickFilterNonRelevantContents(content: IContent, logger: Logger): boolean {\n // Ignore sticky keys for the count\n const eventKeysCount = Object.keys(content).filter((k) => k !== \"msc4354_sticky_key\").length;\n // Don't even bother about empty events (saves us from costly type/\"key in\" checks in bigger rooms)\n if (eventKeysCount === 0) return false;\n\n // We first decide if it's a MSC4143 event (per device state key)\n if (eventKeysCount > 1 && \"application\" in content) {\n // We have a MSC4143 event membership event with a proper joined content\n return true;\n } else if (eventKeysCount === 1 && \"memberships\" in content) {\n // Events used to have this format in the past, but are now deprecated.\n // Given that state events ~cannot be deleted, there can be some remaining events in the room, just ignore them.\n return false;\n } else {\n // Invalid or left content\n return false;\n }\n}\n\nfunction isValidMembership(\n membership: CallMembership,\n room: Pick<Room, \"hasMembershipState\">,\n slotDescription: SlotDescription,\n logger: Logger,\n): boolean {\n if (membership.slotDescription.id !== slotDescription.id) {\n logger.info(\n `Ignoring membership of user ${membership.userId} for a different slot. Theirs: ${JSON.stringify(membership.slotDescription)}, Expected: ${JSON.stringify(slotDescription)}`,\n );\n return false;\n }\n\n if (membership.isExpired()) {\n logger.info(`Ignoring expired device membership ${membership.userId}/${membership.deviceId}`);\n return false;\n }\n\n if (!room.hasMembershipState(membership.userId ?? \"\", KnownMembership.Join)) {\n logger.info(`Ignoring membership of user ${membership.userId} who is not in the room.`);\n return false;\n }\n\n return true;\n}\n\n/**\n * Collects the raw member events from room state and sticky store.\n */\nfunction collectMembersEvents(\n room: Pick<Room, \"getLiveTimeline\" | \"roomId\" | \"_unstable_getStickyEvents\">,\n options: SessionMembershipsForSlotOpts,\n logger: Logger,\n): MatrixEvent[] {\n const { listenForStickyEvents, listenForMemberStateEvents } = options;\n let callMemberEvents: MatrixEvent[] = [];\n if (listenForStickyEvents) {\n // prefill with sticky events\n callMemberEvents = [...room._unstable_getStickyEvents()].filter((e) => e.getType() === EventType.RTCMembership);\n }\n if (listenForMemberStateEvents) {\n const roomState = room.getLiveTimeline().getState(EventTimeline.FORWARDS);\n if (!roomState) {\n logger.warn(\"Couldn't get state for room \" + room.roomId + \"using empty membership array\");\n return [];\n }\n const callMemberStateEvents = roomState.getStateEvents(EventType.GroupCallMemberPrefix);\n callMemberEvents = callMemberEvents.concat(\n callMemberStateEvents.filter(\n (callMemberStateEvent) =>\n !callMemberEvents.some(\n // only care about state events which have keys which we have not yet seen in the sticky events.\n // TODO: I believe this can discard a joined state event if there is a matching left sticky event.\n (stickyEvent) =>\n stickyEvent.getContent().msc4354_sticky_key === callMemberStateEvent.getStateKey(),\n ),\n ),\n );\n }\n return callMemberEvents;\n}\n"],"mappings":"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","ignoreList":[]}
@@ -1 +1 @@
1
- {"version":3,"file":"rtc.d.ts","sourceRoot":"","sources":["../../../src/matrixrtc/membershipData/rtc.ts"],"names":[],"mappings":"AAiBA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,uBAAuB,CAAC;AACtD,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AACnD,OAAO,EAAE,KAAK,mBAAmB,EAAE,KAAK,SAAS,EAAE,MAAM,aAAa,CAAC;AAMvE;;GAEG;AACH,MAAM,WAAW,iBAAiB;IAC9B,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE;QACN,OAAO,EAAE,MAAM,CAAC;QAChB,SAAS,EAAE,MAAM,CAAC;QAClB,EAAE,EAAE,MAAM,CAAC;KACd,CAAC;IACF,cAAc,CAAC,EAAE;QACb,QAAQ,EAAE,MAAM,CAAC;QACjB,QAAQ,EAAE,YAAY,CAAC,SAAS,CAAC;KACpC,CAAC;IACF,aAAa,EAAE,mBAAmB,CAAC,aAAa,CAAC,CAAC;IAClD,gBAAgB,EAAE,SAAS,EAAE,CAAC;IAC9B,UAAU,EAAE,MAAM,EAAE,CAAC;IACrB,oBAAoB,CAAC,EAAE,MAAM,CAAC;IAC9B,YAAY,CAAC,EAAE,MAAM,CAAC;CACzB;AAED;;;;;;GAMG;AACH,eAAO,MAAM,sBAAsB,GAAI,MAAM,QAAQ,EAAE,QAAQ,MAAM,KAAG,IAAI,IAAI,iBA+F/E,CAAC;AAEF,wBAAsB,qBAAqB,CAAC,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,CAK/G"}
1
+ {"version":3,"file":"rtc.d.ts","sourceRoot":"","sources":["../../../src/matrixrtc/membershipData/rtc.ts"],"names":[],"mappings":"AAiBA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,uBAAuB,CAAC;AACtD,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AACnD,OAAO,EAAE,KAAK,mBAAmB,EAAE,KAAK,SAAS,EAAE,MAAM,aAAa,CAAC;AAMvE;;GAEG;AACH,MAAM,WAAW,iBAAiB;IAC9B,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE;QACN,OAAO,EAAE,MAAM,CAAC;QAChB,SAAS,EAAE,MAAM,CAAC;QAClB,EAAE,EAAE,MAAM,CAAC;KACd,CAAC;IACF,cAAc,CAAC,EAAE;QACb,QAAQ,EAAE,MAAM,CAAC;QACjB,QAAQ,EAAE,YAAY,CAAC,SAAS,CAAC;KACpC,CAAC;IACF,aAAa,EAAE,mBAAmB,CAAC,aAAa,CAAC,CAAC;IAClD,gBAAgB,EAAE,SAAS,EAAE,CAAC;IAC9B,UAAU,EAAE,MAAM,EAAE,CAAC;IACrB,oBAAoB,CAAC,EAAE,MAAM,CAAC;IAC9B,YAAY,CAAC,EAAE,MAAM,CAAC;CACzB;AAED;;;;;;GAMG;AACH,eAAO,MAAM,sBAAsB,GAAI,MAAM,QAAQ,EAAE,QAAQ,MAAM,KAAG,IAAI,IAAI,iBA+F/E,CAAC;AAEF,wBAAsB,qBAAqB,CAAC,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,CAM/G"}
@@ -18,7 +18,7 @@ limitations under the License.
18
18
  import { MXID_PATTERN } from "../../models/room-member.js";
19
19
  import { MatrixRTCMembershipParseError } from "./common.js";
20
20
  import { sha256 } from "../../digest.js";
21
- import { encodeUnpaddedBase64Url } from "../../base64.js";
21
+ import { encodeUnpaddedBase64 } from "../../base64.js";
22
22
  import { slotIdToDescription } from "../utils.js";
23
23
 
24
24
  /**
@@ -127,9 +127,10 @@ export function computeRtcIdentityRaw(_x, _x2, _x3) {
127
127
  }
128
128
  function _computeRtcIdentityRaw() {
129
129
  _computeRtcIdentityRaw = _asyncToGenerator(function* (userId, deviceId, memberId) {
130
- var hashInput = "".concat(userId, "|").concat(deviceId, "|").concat(memberId);
131
- var hashBuffer = yield sha256(hashInput);
132
- var hashedString = encodeUnpaddedBase64Url(hashBuffer);
130
+ // canonical JSON serialization (Matrix canonical JSON for arrays)
131
+ var jsonStr = JSON.stringify([userId, deviceId, memberId]);
132
+ var hashBuffer = yield sha256(jsonStr);
133
+ var hashedString = encodeUnpaddedBase64(hashBuffer);
133
134
  return hashedString;
134
135
  });
135
136
  return _computeRtcIdentityRaw.apply(this, arguments);
@@ -1 +1 @@
1
- {"version":3,"file":"rtc.js","names":["MXID_PATTERN","MatrixRTCMembershipParseError","sha256","encodeUnpaddedBase64Url","slotIdToDescription","checkRtcMembershipData","data","sender","_data$application","_data$sticky_key","errors","prefix","expectedSlotPrefix","concat","application","type","slot_id","push","startsWith","ex","Error","message","member","user_id","test","device_id","id","includes","rtc_transports","undefined","Array","isArray","t","versions","every","v","sticky_key","msc4354_sticky_key","rel","event_id","rel_type","length","computeRtcIdentityRaw","_x","_x2","_x3","_computeRtcIdentityRaw","apply","arguments","_asyncToGenerator","userId","deviceId","memberId","hashInput","hashBuffer","hashedString"],"sources":["../../../src/matrixrtc/membershipData/rtc.ts"],"sourcesContent":["/*\nCopyright 2026 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\nimport { MXID_PATTERN } from \"../../models/room-member.ts\";\nimport type { IContent } from \"../../models/event.ts\";\nimport type { RelationType } from \"../../types.ts\";\nimport { type RtcSlotEventContent, type Transport } from \"../types.ts\";\nimport { MatrixRTCMembershipParseError } from \"./common.ts\";\nimport { sha256 } from \"../../digest.ts\";\nimport { encodeUnpaddedBase64Url } from \"../../base64.ts\";\nimport { slotIdToDescription } from \"../utils.ts\";\n\n/**\n * Represents the current form of MSC4143, which uses sticky events to store membership.\n */\nexport interface RtcMembershipData {\n \"slot_id\": string;\n \"member\": {\n user_id: string;\n device_id: string;\n id: string;\n };\n \"m.relates_to\"?: {\n event_id: string;\n rel_type: RelationType.Reference;\n };\n \"application\": RtcSlotEventContent[\"application\"];\n \"rtc_transports\": Transport[];\n \"versions\": string[];\n \"msc4354_sticky_key\"?: string;\n \"sticky_key\"?: string;\n}\n\n/**\n * Validates that `data` matches the format expected by MSC4143.\n * @param data The event content.\n * @param sender The sender of the event.\n * @returns true if `data` is valid RtcMembershipData\n * @throws {MatrixRTCMembershipParseError} if the content is not valid\n */\nexport const checkRtcMembershipData = (data: IContent, sender: string): data is RtcMembershipData => {\n const errors: string[] = [];\n const prefix = \" - \";\n const expectedSlotPrefix = `${data?.application?.type}#`;\n\n // required fields\n if (typeof data.slot_id !== \"string\") {\n errors.push(prefix + \"slot_id must be string\");\n } else if (!data.slot_id.startsWith(expectedSlotPrefix)) {\n errors.push(prefix + `slot_id must start with ${expectedSlotPrefix}`);\n } else {\n try {\n slotIdToDescription(data.slot_id);\n } catch (ex) {\n errors.push(prefix + `slot_id was badly formed${ex instanceof Error ? `: ${ex.message}` : \"\"}`);\n }\n }\n\n if (typeof data.member !== \"object\" || data.member === null) {\n errors.push(prefix + \"member must be an object\");\n } else {\n if (typeof data.member.user_id !== \"string\") {\n errors.push(prefix + \"member.user_id must be string\");\n } else if (!MXID_PATTERN.test(data.member.user_id)) {\n errors.push(prefix + \"member.user_id must be a valid mxid\");\n }\n // This is not what the spec enforces but there currently are no rules what power levels are required to\n // send a m.rtc.member event for a other user. So we add this check for simplicity and to avoid possible attacks until there\n // is a proper definition when this is allowed.\n else if (data.member.user_id !== sender) {\n errors.push(prefix + \"member.user_id must match the sender\");\n }\n if (typeof data.member.device_id !== \"string\") {\n errors.push(prefix + \"member.device_id must be string\");\n }\n if (typeof data.member.id !== \"string\") errors.push(prefix + \"member.id must be string\");\n }\n if (typeof data.application !== \"object\" || data.application === null) {\n errors.push(prefix + \"application must be an object\");\n } else {\n if (typeof data.application.type !== \"string\") {\n errors.push(prefix + \"application.type must be a string\");\n } else {\n if (data.application.type.includes(\"#\")) errors.push(prefix + 'application.type must not include \"#\"');\n }\n }\n if (data.rtc_transports === undefined || !Array.isArray(data.rtc_transports)) {\n errors.push(prefix + \"rtc_transports must be an array\");\n } else {\n // validate that each transport has at least a string 'type'\n for (const t of data.rtc_transports) {\n if (typeof t !== \"object\" || t === null || typeof (t as any).type !== \"string\") {\n errors.push(prefix + \"rtc_transports entries must be objects with a string type\");\n break;\n }\n }\n }\n if (data.versions === undefined || !Array.isArray(data.versions)) {\n errors.push(prefix + \"versions must be an array\");\n } else if (!data.versions.every((v) => typeof v === \"string\")) {\n errors.push(prefix + \"versions must be an array of strings\");\n }\n\n // optional fields\n if ((data.sticky_key ?? data.msc4354_sticky_key) === undefined) {\n errors.push(prefix + \"sticky_key or msc4354_sticky_key must be a defined\");\n }\n if (data.sticky_key !== undefined && typeof data.sticky_key !== \"string\") {\n errors.push(prefix + \"sticky_key must be a string\");\n }\n if (data.msc4354_sticky_key !== undefined && typeof data.msc4354_sticky_key !== \"string\") {\n errors.push(prefix + \"msc4354_sticky_key must be a string\");\n }\n if (\n data.sticky_key !== undefined &&\n data.msc4354_sticky_key !== undefined &&\n data.sticky_key !== data.msc4354_sticky_key\n ) {\n errors.push(prefix + \"sticky_key and msc4354_sticky_key must be equal if both are defined\");\n }\n if (data[\"m.relates_to\"] !== undefined) {\n const rel = data[\"m.relates_to\"] as RtcMembershipData[\"m.relates_to\"];\n if (typeof rel !== \"object\" || rel === null) {\n errors.push(prefix + \"m.relates_to must be an object if provided\");\n } else {\n if (typeof rel.event_id !== \"string\") errors.push(prefix + \"m.relates_to.event_id must be a string\");\n if (rel.rel_type !== \"m.reference\") errors.push(prefix + \"m.relates_to.rel_type must be m.reference\");\n }\n }\n\n if (errors.length) {\n throw new MatrixRTCMembershipParseError(\"RtcMembership\", errors);\n }\n\n return true;\n};\n\nexport async function computeRtcIdentityRaw(userId: string, deviceId: string, memberId: string): Promise<string> {\n const hashInput = `${userId}|${deviceId}|${memberId}`;\n const hashBuffer = await sha256(hashInput);\n const hashedString = encodeUnpaddedBase64Url(hashBuffer);\n return hashedString;\n}\n"],"mappings":"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","ignoreList":[]}
1
+ {"version":3,"file":"rtc.js","names":["MXID_PATTERN","MatrixRTCMembershipParseError","sha256","encodeUnpaddedBase64","slotIdToDescription","checkRtcMembershipData","data","sender","_data$application","_data$sticky_key","errors","prefix","expectedSlotPrefix","concat","application","type","slot_id","push","startsWith","ex","Error","message","member","user_id","test","device_id","id","includes","rtc_transports","undefined","Array","isArray","t","versions","every","v","sticky_key","msc4354_sticky_key","rel","event_id","rel_type","length","computeRtcIdentityRaw","_x","_x2","_x3","_computeRtcIdentityRaw","apply","arguments","_asyncToGenerator","userId","deviceId","memberId","jsonStr","JSON","stringify","hashBuffer","hashedString"],"sources":["../../../src/matrixrtc/membershipData/rtc.ts"],"sourcesContent":["/*\nCopyright 2026 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\nimport { MXID_PATTERN } from \"../../models/room-member.ts\";\nimport type { IContent } from \"../../models/event.ts\";\nimport type { RelationType } from \"../../types.ts\";\nimport { type RtcSlotEventContent, type Transport } from \"../types.ts\";\nimport { MatrixRTCMembershipParseError } from \"./common.ts\";\nimport { sha256 } from \"../../digest.ts\";\nimport { encodeUnpaddedBase64 } from \"../../base64.ts\";\nimport { slotIdToDescription } from \"../utils.ts\";\n\n/**\n * Represents the current form of MSC4143, which uses sticky events to store membership.\n */\nexport interface RtcMembershipData {\n \"slot_id\": string;\n \"member\": {\n user_id: string;\n device_id: string;\n id: string;\n };\n \"m.relates_to\"?: {\n event_id: string;\n rel_type: RelationType.Reference;\n };\n \"application\": RtcSlotEventContent[\"application\"];\n \"rtc_transports\": Transport[];\n \"versions\": string[];\n \"msc4354_sticky_key\"?: string;\n \"sticky_key\"?: string;\n}\n\n/**\n * Validates that `data` matches the format expected by MSC4143.\n * @param data The event content.\n * @param sender The sender of the event.\n * @returns true if `data` is valid RtcMembershipData\n * @throws {MatrixRTCMembershipParseError} if the content is not valid\n */\nexport const checkRtcMembershipData = (data: IContent, sender: string): data is RtcMembershipData => {\n const errors: string[] = [];\n const prefix = \" - \";\n const expectedSlotPrefix = `${data?.application?.type}#`;\n\n // required fields\n if (typeof data.slot_id !== \"string\") {\n errors.push(prefix + \"slot_id must be string\");\n } else if (!data.slot_id.startsWith(expectedSlotPrefix)) {\n errors.push(prefix + `slot_id must start with ${expectedSlotPrefix}`);\n } else {\n try {\n slotIdToDescription(data.slot_id);\n } catch (ex) {\n errors.push(prefix + `slot_id was badly formed${ex instanceof Error ? `: ${ex.message}` : \"\"}`);\n }\n }\n\n if (typeof data.member !== \"object\" || data.member === null) {\n errors.push(prefix + \"member must be an object\");\n } else {\n if (typeof data.member.user_id !== \"string\") {\n errors.push(prefix + \"member.user_id must be string\");\n } else if (!MXID_PATTERN.test(data.member.user_id)) {\n errors.push(prefix + \"member.user_id must be a valid mxid\");\n }\n // This is not what the spec enforces but there currently are no rules what power levels are required to\n // send a m.rtc.member event for a other user. So we add this check for simplicity and to avoid possible attacks until there\n // is a proper definition when this is allowed.\n else if (data.member.user_id !== sender) {\n errors.push(prefix + \"member.user_id must match the sender\");\n }\n if (typeof data.member.device_id !== \"string\") {\n errors.push(prefix + \"member.device_id must be string\");\n }\n if (typeof data.member.id !== \"string\") errors.push(prefix + \"member.id must be string\");\n }\n if (typeof data.application !== \"object\" || data.application === null) {\n errors.push(prefix + \"application must be an object\");\n } else {\n if (typeof data.application.type !== \"string\") {\n errors.push(prefix + \"application.type must be a string\");\n } else {\n if (data.application.type.includes(\"#\")) errors.push(prefix + 'application.type must not include \"#\"');\n }\n }\n if (data.rtc_transports === undefined || !Array.isArray(data.rtc_transports)) {\n errors.push(prefix + \"rtc_transports must be an array\");\n } else {\n // validate that each transport has at least a string 'type'\n for (const t of data.rtc_transports) {\n if (typeof t !== \"object\" || t === null || typeof (t as any).type !== \"string\") {\n errors.push(prefix + \"rtc_transports entries must be objects with a string type\");\n break;\n }\n }\n }\n if (data.versions === undefined || !Array.isArray(data.versions)) {\n errors.push(prefix + \"versions must be an array\");\n } else if (!data.versions.every((v) => typeof v === \"string\")) {\n errors.push(prefix + \"versions must be an array of strings\");\n }\n\n // optional fields\n if ((data.sticky_key ?? data.msc4354_sticky_key) === undefined) {\n errors.push(prefix + \"sticky_key or msc4354_sticky_key must be a defined\");\n }\n if (data.sticky_key !== undefined && typeof data.sticky_key !== \"string\") {\n errors.push(prefix + \"sticky_key must be a string\");\n }\n if (data.msc4354_sticky_key !== undefined && typeof data.msc4354_sticky_key !== \"string\") {\n errors.push(prefix + \"msc4354_sticky_key must be a string\");\n }\n if (\n data.sticky_key !== undefined &&\n data.msc4354_sticky_key !== undefined &&\n data.sticky_key !== data.msc4354_sticky_key\n ) {\n errors.push(prefix + \"sticky_key and msc4354_sticky_key must be equal if both are defined\");\n }\n if (data[\"m.relates_to\"] !== undefined) {\n const rel = data[\"m.relates_to\"] as RtcMembershipData[\"m.relates_to\"];\n if (typeof rel !== \"object\" || rel === null) {\n errors.push(prefix + \"m.relates_to must be an object if provided\");\n } else {\n if (typeof rel.event_id !== \"string\") errors.push(prefix + \"m.relates_to.event_id must be a string\");\n if (rel.rel_type !== \"m.reference\") errors.push(prefix + \"m.relates_to.rel_type must be m.reference\");\n }\n }\n\n if (errors.length) {\n throw new MatrixRTCMembershipParseError(\"RtcMembership\", errors);\n }\n\n return true;\n};\n\nexport async function computeRtcIdentityRaw(userId: string, deviceId: string, memberId: string): Promise<string> {\n // canonical JSON serialization (Matrix canonical JSON for arrays)\n const jsonStr = JSON.stringify([userId, deviceId, memberId]);\n const hashBuffer = await sha256(jsonStr);\n const hashedString = encodeUnpaddedBase64(hashBuffer);\n return hashedString;\n}\n"],"mappings":"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","ignoreList":[]}
@@ -1,4 +1,4 @@
1
- import { type IdTokenClaims } from "oidc-client-ts";
1
+ import { type IdTokenClaims, type SigninRequestCreateArgs } from "oidc-client-ts";
2
2
  import { type BearerTokenResponse, type ValidatedAuthMetadata } from "./validate.ts";
3
3
  export type { BearerTokenResponse };
4
4
  /**
@@ -54,9 +54,11 @@ export declare const generateAuthorizationUrl: (authorizationUrl: string, client
54
54
  * @param urlState - value to append to the opaque state identifier to uniquely identify the callback
55
55
  * @param loginHint - value to send as the `login_hint` to the OP, giving a hint about the login identifier the user might use to log in.
56
56
  * See {@link https://openid.net/specs/openid-connect-core-1_0.html#AuthRequest OIDC core 3.1.2.1}.
57
+ * @param responseMode - value to send as the `response_mode` to the OP, selecting how auth is passed back during redirect.
58
+ * See {@link https://openid.net/specs/openid-connect-core-1_0.html#AuthRequest OIDC core 3.1.2.1}.
57
59
  * @returns a Promise with the url as a string
58
60
  */
59
- export declare const generateOidcAuthorizationUrl: ({ metadata, redirectUri, clientId, homeserverUrl, identityServerUrl, nonce, prompt, urlState, loginHint, }: {
61
+ export declare const generateOidcAuthorizationUrl: ({ metadata, redirectUri, clientId, homeserverUrl, identityServerUrl, nonce, prompt, urlState, loginHint, responseMode, }: {
60
62
  clientId: string;
61
63
  metadata: ValidatedAuthMetadata;
62
64
  homeserverUrl: string;
@@ -66,6 +68,7 @@ export declare const generateOidcAuthorizationUrl: ({ metadata, redirectUri, cli
66
68
  prompt?: string;
67
69
  urlState?: string;
68
70
  loginHint?: string;
71
+ responseMode?: SigninRequestCreateArgs["response_mode"];
69
72
  }) => Promise<string>;
70
73
  /**
71
74
  * @experimental
@@ -75,12 +78,13 @@ export declare const generateOidcAuthorizationUrl: ({ metadata, redirectUri, cli
75
78
  * request to the Token Endpoint, to obtain the access token, refresh token, etc.
76
79
  *
77
80
  * @param code - authorization code as returned by OP during authorization
78
- * @param storedAuthorizationParams - stored params from start of oidc login flow
81
+ * @param state - authorization state param as returned by OP during authorization
82
+ * @param responseMode - the response mode used for authentication
79
83
  * @returns valid bearer token response
80
84
  * @throws An `Error` with `message` set to an entry in {@link OidcError},
81
85
  * when the request fails, or the returned token response is invalid.
82
86
  */
83
- export declare const completeAuthorizationCodeGrant: (code: string, state: string) => Promise<{
87
+ export declare const completeAuthorizationCodeGrant: (code: string, state: string, responseMode?: SigninRequestCreateArgs["response_mode"]) => Promise<{
84
88
  oidcClientSettings: {
85
89
  clientId: string;
86
90
  issuer: string;
@@ -1 +1 @@
1
- {"version":3,"file":"authorize.d.ts","sourceRoot":"","sources":["../../src/oidc/authorize.ts"],"names":[],"mappings":"AAgBA,OAAO,EAAE,KAAK,aAAa,EAAsE,MAAM,gBAAgB,CAAC;AAKxH,OAAO,EACH,KAAK,mBAAmB,EAGxB,KAAK,qBAAqB,EAG7B,MAAM,eAAe,CAAC;AAKvB,YAAY,EAAE,mBAAmB,EAAE,CAAC;AAEpC;;;;GAIG;AACH,MAAM,MAAM,mBAAmB,GAAG;IAC9B,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,CAAC;IACd,WAAW,EAAE,MAAM,CAAC;IACpB,YAAY,EAAE,MAAM,CAAC;IACrB,KAAK,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF;;;;GAIG;AACH,eAAO,MAAM,aAAa,GAAI,WAAW,MAAM,KAAG,MAGjD,CAAC;AAcF;;;;;;;GAOG;AACH,eAAO,MAAM,2BAA2B,GAAI,iBAAiB;IAAE,WAAW,EAAE,MAAM,CAAA;CAAE,KAAG,mBAMrF,CAAC;AAEH;;;;;;;;GAQG;AACH,eAAO,MAAM,wBAAwB,GACjC,kBAAkB,MAAM,EACxB,UAAU,MAAM,EAChB,oDAAoD,mBAAmB,KACxE,OAAO,CAAC,MAAM,CAchB,CAAC;AAEF;;;;;;;;;;;;;;;GAeG;AACH,eAAO,MAAM,4BAA4B,GAAU,4GAUhD;IACC,QAAQ,EAAE,MAAM,CAAC;IACjB,QAAQ,EAAE,qBAAqB,CAAC;IAChC,aAAa,EAAE,MAAM,CAAC;IACtB,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC3B,WAAW,EAAE,MAAM,CAAC;IACpB,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,SAAS,CAAC,EAAE,MAAM,CAAC;CACtB,KAAG,OAAO,CAAC,MAAM,CAsBjB,CAAC;AAqBF;;;;;;;;;;;;GAYG;AACH,eAAO,MAAM,8BAA8B,GACvC,MAAM,MAAM,EACZ,OAAO,MAAM,KACd,OAAO,CAAC;IACP,kBAAkB,EAAE;QAAE,QAAQ,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAA;KAAE,CAAC;IACzD,aAAa,EAAE,mBAAmB,CAAC;IACnC,aAAa,EAAE,MAAM,CAAC;IACtB,aAAa,EAAE,aAAa,CAAC;IAC7B,iBAAiB,CAAC,EAAE,MAAM,CAAC;CAC9B,CA8DA,CAAC"}
1
+ {"version":3,"file":"authorize.d.ts","sourceRoot":"","sources":["../../src/oidc/authorize.ts"],"names":[],"mappings":"AAgBA,OAAO,EACH,KAAK,aAAa,EAGlB,KAAK,uBAAuB,EAI/B,MAAM,gBAAgB,CAAC;AAKxB,OAAO,EACH,KAAK,mBAAmB,EAGxB,KAAK,qBAAqB,EAG7B,MAAM,eAAe,CAAC;AAKvB,YAAY,EAAE,mBAAmB,EAAE,CAAC;AAEpC;;;;GAIG;AACH,MAAM,MAAM,mBAAmB,GAAG;IAC9B,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,CAAC;IACd,WAAW,EAAE,MAAM,CAAC;IACpB,YAAY,EAAE,MAAM,CAAC;IACrB,KAAK,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF;;;;GAIG;AACH,eAAO,MAAM,aAAa,GAAI,WAAW,MAAM,KAAG,MAGjD,CAAC;AAcF;;;;;;;GAOG;AACH,eAAO,MAAM,2BAA2B,GAAI,iBAAiB;IAAE,WAAW,EAAE,MAAM,CAAA;CAAE,KAAG,mBAMrF,CAAC;AAEH;;;;;;;;GAQG;AACH,eAAO,MAAM,wBAAwB,GACjC,kBAAkB,MAAM,EACxB,UAAU,MAAM,EAChB,oDAAoD,mBAAmB,KACxE,OAAO,CAAC,MAAM,CAchB,CAAC;AAEF;;;;;;;;;;;;;;;;;GAiBG;AACH,eAAO,MAAM,4BAA4B,GAAU,0HAWhD;IACC,QAAQ,EAAE,MAAM,CAAC;IACjB,QAAQ,EAAE,qBAAqB,CAAC;IAChC,aAAa,EAAE,MAAM,CAAC;IACtB,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC3B,WAAW,EAAE,MAAM,CAAC;IACpB,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,YAAY,CAAC,EAAE,uBAAuB,CAAC,eAAe,CAAC,CAAC;CAC3D,KAAG,OAAO,CAAC,MAAM,CAsBjB,CAAC;AAqBF;;;;;;;;;;;;;GAaG;AACH,eAAO,MAAM,8BAA8B,GACvC,MAAM,MAAM,EACZ,OAAO,MAAM,EACb,eAAc,uBAAuB,CAAC,eAAe,CAAW,KACjE,OAAO,CAAC;IACP,kBAAkB,EAAE;QAAE,QAAQ,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAA;KAAE,CAAC;IACzD,aAAa,EAAE,mBAAmB,CAAC;IACnC,aAAa,EAAE,MAAM,CAAC;IACtB,aAAa,EAAE,aAAa,CAAC;IAC7B,iBAAiB,CAAC,EAAE,MAAM,CAAC;CAC9B,CAmEA,CAAC"}
@@ -130,6 +130,8 @@ export var generateAuthorizationUrl = /*#__PURE__*/function () {
130
130
  * @param urlState - value to append to the opaque state identifier to uniquely identify the callback
131
131
  * @param loginHint - value to send as the `login_hint` to the OP, giving a hint about the login identifier the user might use to log in.
132
132
  * See {@link https://openid.net/specs/openid-connect-core-1_0.html#AuthRequest OIDC core 3.1.2.1}.
133
+ * @param responseMode - value to send as the `response_mode` to the OP, selecting how auth is passed back during redirect.
134
+ * See {@link https://openid.net/specs/openid-connect-core-1_0.html#AuthRequest OIDC core 3.1.2.1}.
133
135
  * @returns a Promise with the url as a string
134
136
  */
135
137
  export var generateOidcAuthorizationUrl = /*#__PURE__*/function () {
@@ -143,14 +145,15 @@ export var generateOidcAuthorizationUrl = /*#__PURE__*/function () {
143
145
  nonce,
144
146
  prompt,
145
147
  urlState,
146
- loginHint
148
+ loginHint,
149
+ responseMode = "query"
147
150
  } = _ref5;
148
151
  var scope = generateScope();
149
152
  var oidcClient = new OidcClient(_objectSpread(_objectSpread({}, metadata), {}, {
150
153
  client_id: clientId,
151
154
  redirect_uri: redirectUri,
152
155
  authority: metadata.issuer,
153
- response_mode: "query",
156
+ response_mode: responseMode,
154
157
  response_type: "code",
155
158
  scope,
156
159
  stateStore: new WebStorageStateStore({
@@ -203,26 +206,35 @@ var normalizeBearerTokenResponseTokenType = response => ({
203
206
  * request to the Token Endpoint, to obtain the access token, refresh token, etc.
204
207
  *
205
208
  * @param code - authorization code as returned by OP during authorization
206
- * @param storedAuthorizationParams - stored params from start of oidc login flow
209
+ * @param state - authorization state param as returned by OP during authorization
210
+ * @param responseMode - the response mode used for authentication
207
211
  * @returns valid bearer token response
208
212
  * @throws An `Error` with `message` set to an entry in {@link OidcError},
209
213
  * when the request fails, or the returned token response is invalid.
210
214
  */
211
215
  export var completeAuthorizationCodeGrant = /*#__PURE__*/function () {
212
216
  var _ref7 = _asyncToGenerator(function* (code, state) {
217
+ var responseMode = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : "query";
213
218
  /**
214
219
  * Element Web strips and changes the url on starting the app
215
220
  * Use the code and state from query params to rebuild a url
216
221
  * so that oidc-client can parse it
217
222
  */
218
223
  var reconstructedUrl = new URL(window.location.origin);
219
- reconstructedUrl.searchParams.append("code", code);
220
- reconstructedUrl.searchParams.append("state", state);
224
+ var params = new URLSearchParams({
225
+ code,
226
+ state
227
+ });
228
+ if (responseMode === "query") {
229
+ reconstructedUrl.search = params.toString();
230
+ } else {
231
+ reconstructedUrl.hash = "#".concat(params.toString());
232
+ }
221
233
 
222
234
  // set oidc-client to use our logger
223
235
  Log.setLogger(logger);
224
236
  try {
225
- var response = new SigninResponse(reconstructedUrl.searchParams);
237
+ var response = new SigninResponse(params);
226
238
  var stateStore = new WebStorageStateStore({
227
239
  prefix: "mx_oidc_",
228
240
  store: window.sessionStorage