@tagea/capacitor-matrix 0.2.0 → 0.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +14 -0
- package/dist/docs.json +16 -0
- package/dist/esm/definitions.d.ts +3 -0
- package/dist/esm/definitions.js.map +1 -1
- package/dist/esm/web.d.ts +3 -0
- package/dist/esm/web.js +30 -1
- package/dist/esm/web.js.map +1 -1
- package/dist/plugin.cjs.js +30 -1
- package/dist/plugin.cjs.js.map +1 -1
- package/dist/plugin.js +30 -1
- package/dist/plugin.js.map +1 -1
- package/package.json +2 -1
package/README.md
CHANGED
|
@@ -349,6 +349,7 @@ The full API reference is auto-generated below from the TypeScript definitions.
|
|
|
349
349
|
* [`getPresence(...)`](#getpresence)
|
|
350
350
|
* [`getDevices()`](#getdevices)
|
|
351
351
|
* [`deleteDevice(...)`](#deletedevice)
|
|
352
|
+
* [`verifyDevice(...)`](#verifydevice)
|
|
352
353
|
* [`setPusher(...)`](#setpusher)
|
|
353
354
|
* [`initializeCrypto()`](#initializecrypto)
|
|
354
355
|
* [`getEncryptionStatus()`](#getencryptionstatus)
|
|
@@ -871,6 +872,19 @@ deleteDevice(options: { deviceId: string; auth?: Record<string, unknown>; }) =>
|
|
|
871
872
|
--------------------
|
|
872
873
|
|
|
873
874
|
|
|
875
|
+
### verifyDevice(...)
|
|
876
|
+
|
|
877
|
+
```typescript
|
|
878
|
+
verifyDevice(options: { deviceId: string; }) => Promise<void>
|
|
879
|
+
```
|
|
880
|
+
|
|
881
|
+
| Param | Type |
|
|
882
|
+
| ------------- | ---------------------------------- |
|
|
883
|
+
| **`options`** | <code>{ deviceId: string; }</code> |
|
|
884
|
+
|
|
885
|
+
--------------------
|
|
886
|
+
|
|
887
|
+
|
|
874
888
|
### setPusher(...)
|
|
875
889
|
|
|
876
890
|
```typescript
|
package/dist/docs.json
CHANGED
|
@@ -592,6 +592,22 @@
|
|
|
592
592
|
],
|
|
593
593
|
"slug": "deletedevice"
|
|
594
594
|
},
|
|
595
|
+
{
|
|
596
|
+
"name": "verifyDevice",
|
|
597
|
+
"signature": "(options: { deviceId: string; }) => Promise<void>",
|
|
598
|
+
"parameters": [
|
|
599
|
+
{
|
|
600
|
+
"name": "options",
|
|
601
|
+
"docs": "",
|
|
602
|
+
"type": "{ deviceId: string; }"
|
|
603
|
+
}
|
|
604
|
+
],
|
|
605
|
+
"returns": "Promise<void>",
|
|
606
|
+
"tags": [],
|
|
607
|
+
"docs": "",
|
|
608
|
+
"complexTypes": [],
|
|
609
|
+
"slug": "verifydevice"
|
|
610
|
+
},
|
|
595
611
|
{
|
|
596
612
|
"name": "setPusher",
|
|
597
613
|
"signature": "(options: PusherOptions) => Promise<void>",
|
|
@@ -333,6 +333,9 @@ export interface MatrixPlugin {
|
|
|
333
333
|
deviceId: string;
|
|
334
334
|
auth?: Record<string, unknown>;
|
|
335
335
|
}): Promise<void>;
|
|
336
|
+
verifyDevice(options: {
|
|
337
|
+
deviceId: string;
|
|
338
|
+
}): Promise<void>;
|
|
336
339
|
setPusher(options: PusherOptions): Promise<void>;
|
|
337
340
|
initializeCrypto(): Promise<void>;
|
|
338
341
|
getEncryptionStatus(): Promise<EncryptionStatus>;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"definitions.js","sourceRoot":"","sources":["../../src/definitions.ts"],"names":[],"mappings":"","sourcesContent":["import type { PluginListenerHandle } from '@capacitor/core';\n\n// Auth & Session\n\nexport interface LoginOptions {\n homeserverUrl: string;\n userId: string;\n password: string;\n}\n\nexport interface LoginWithTokenOptions {\n homeserverUrl: string;\n accessToken: string;\n userId: string;\n deviceId: string;\n}\n\nexport interface SessionInfo {\n accessToken: string;\n userId: string;\n deviceId: string;\n homeserverUrl: string;\n}\n\n// Messaging\n\nexport interface SendMessageOptions {\n roomId: string;\n body: string;\n msgtype?: 'm.text' | 'm.notice' | 'm.emote' | 'm.image' | 'm.audio' | 'm.video' | 'm.file';\n fileUri?: string;\n fileName?: string;\n mimeType?: string;\n fileSize?: number;\n /** Audio/video duration in milliseconds (sets info.duration per Matrix spec) */\n duration?: number;\n /** Image/video width in pixels (sets info.w per Matrix spec) */\n width?: number;\n /** Image/video height in pixels (sets info.h per Matrix spec) */\n height?: number;\n}\n\n// Presence\n\nexport interface PresenceInfo {\n presence: 'online' | 'offline' | 'unavailable';\n statusMsg?: string;\n lastActiveAgo?: number;\n}\n\n// Typing\n\nexport interface TypingEvent {\n roomId: string;\n userIds: string[];\n}\n\nexport interface ReceiptReceivedEvent {\n roomId: string;\n /** The event that was read */\n eventId: string;\n /** The user who sent the read receipt */\n userId: string;\n}\n\nexport interface PresenceChangedEvent {\n userId: string;\n presence: PresenceInfo;\n}\n\n// Edit & Reply\n\nexport interface EditMessageOptions {\n roomId: string;\n eventId: string;\n newBody: string;\n /** Required when editing a media message; must match the original msgtype */\n msgtype?: 'm.text' | 'm.notice' | 'm.emote' | 'm.image' | 'm.audio' | 'm.video' | 'm.file';\n /** New file to replace the media content (optional for caption-only edits) */\n fileUri?: string;\n fileName?: string;\n mimeType?: string;\n fileSize?: number;\n /** Audio/video duration in milliseconds */\n duration?: number;\n /** Image/video width in pixels */\n width?: number;\n /** Image/video height in pixels */\n height?: number;\n}\n\nexport interface SendReplyOptions {\n roomId: string;\n body: string;\n replyToEventId: string;\n msgtype?: 'm.text' | 'm.notice' | 'm.emote' | 'm.image' | 'm.audio' | 'm.video' | 'm.file';\n fileUri?: string;\n fileName?: string;\n mimeType?: string;\n fileSize?: number;\n /** Audio/video duration in milliseconds (sets info.duration per Matrix spec) */\n duration?: number;\n /** Image/video width in pixels (sets info.w per Matrix spec) */\n width?: number;\n /** Image/video height in pixels (sets info.h per Matrix spec) */\n height?: number;\n}\n\n// Upload\n\nexport interface UploadContentOptions {\n fileUri: string;\n fileName: string;\n mimeType: string;\n}\n\nexport interface UploadContentResult {\n contentUri: string;\n}\n\n// Thumbnail\n\nexport interface ThumbnailUrlOptions {\n mxcUrl: string;\n width: number;\n height: number;\n method?: 'scale' | 'crop';\n}\n\nexport interface MatrixEvent {\n eventId: string;\n roomId: string;\n senderId: string;\n type: string;\n content: Record<string, unknown>;\n originServerTs: number;\n /** Delivery/read status for own messages: 'sending' | 'sent' | 'delivered' | 'read' */\n status?: 'sending' | 'sent' | 'delivered' | 'read';\n /** User IDs that have read this event */\n readBy?: string[];\n /** Unsigned data (e.g. m.relations for edits, transaction_id for local echo) */\n unsigned?: Record<string, unknown>;\n}\n\n// Rooms\n\nexport interface RoomSummary {\n roomId: string;\n name: string;\n topic?: string;\n memberCount: number;\n isEncrypted: boolean;\n unreadCount: number;\n lastEventTs?: number;\n membership?: 'join' | 'invite' | 'leave' | 'ban';\n avatarUrl?: string;\n isDirect?: boolean;\n}\n\nexport interface RoomMember {\n userId: string;\n displayName?: string;\n membership: 'join' | 'invite' | 'leave' | 'ban';\n avatarUrl?: string;\n}\n\n// Device Management\n\nexport interface DeviceInfo {\n deviceId: string;\n displayName?: string;\n lastSeenTs?: number;\n lastSeenIp?: string;\n /** Whether this device is verified via cross-signing */\n isCrossSigningVerified?: boolean;\n}\n\n// Pusher\n\nexport interface PusherOptions {\n pushkey: string;\n kind: string | null;\n appId: string;\n appDisplayName: string;\n deviceDisplayName: string;\n lang: string;\n data: { url: string; format?: string };\n}\n\n// User Discovery\n\nexport interface UserProfile {\n userId: string;\n displayName?: string;\n avatarUrl?: string;\n}\n\n// Encryption\n\nexport interface CrossSigningStatus {\n hasMaster: boolean;\n hasSelfSigning: boolean;\n hasUserSigning: boolean;\n isReady: boolean;\n}\n\nexport interface KeyBackupStatus {\n exists: boolean;\n version?: string;\n enabled: boolean;\n}\n\nexport interface RecoveryKeyInfo {\n recoveryKey: string;\n}\n\nexport interface EncryptionStatus {\n isCrossSigningReady: boolean;\n crossSigningStatus: CrossSigningStatus;\n isKeyBackupEnabled: boolean;\n keyBackupVersion?: string;\n isSecretStorageReady: boolean;\n}\n\n// Events & Sync\n\nexport type SyncState = 'INITIAL' | 'SYNCING' | 'ERROR' | 'STOPPED';\n\nexport interface SyncStateChangeEvent {\n state: SyncState;\n error?: string;\n}\n\nexport interface MessageReceivedEvent {\n event: MatrixEvent;\n}\n\nexport interface RoomUpdatedEvent {\n roomId: string;\n summary: RoomSummary;\n}\n\n// Plugin Interface\n\nexport interface MatrixPlugin {\n // Auth\n login(options: LoginOptions): Promise<SessionInfo>;\n loginWithToken(options: LoginWithTokenOptions): Promise<SessionInfo>;\n logout(): Promise<void>;\n getSession(): Promise<SessionInfo | null>;\n\n // Sync\n startSync(): Promise<void>;\n stopSync(): Promise<void>;\n getSyncState(): Promise<{ state: SyncState }>;\n\n // Rooms\n createRoom(options: {\n name?: string;\n topic?: string;\n isEncrypted?: boolean;\n isDirect?: boolean;\n invite?: string[];\n preset?: 'private_chat' | 'trusted_private_chat' | 'public_chat';\n historyVisibility?: 'invited' | 'joined' | 'shared' | 'world_readable';\n }): Promise<{ roomId: string }>;\n getRooms(): Promise<{ rooms: RoomSummary[] }>;\n getRoomMembers(options: { roomId: string }): Promise<{ members: RoomMember[] }>;\n joinRoom(options: { roomIdOrAlias: string }): Promise<{ roomId: string }>;\n leaveRoom(options: { roomId: string }): Promise<void>;\n forgetRoom(options: { roomId: string }): Promise<void>;\n\n // Messaging\n sendMessage(options: SendMessageOptions): Promise<{ eventId: string }>;\n editMessage(options: EditMessageOptions): Promise<{ eventId: string }>;\n sendReply(options: SendReplyOptions): Promise<{ eventId: string }>;\n getRoomMessages(options: {\n roomId: string;\n limit?: number;\n from?: string;\n }): Promise<{ events: MatrixEvent[]; nextBatch?: string }>;\n markRoomAsRead(options: {\n roomId: string;\n eventId: string;\n }): Promise<void>;\n refreshEventStatuses(options: {\n roomId: string;\n eventIds: string[];\n }): Promise<{ events: MatrixEvent[] }>;\n redactEvent(options: {\n roomId: string;\n eventId: string;\n reason?: string;\n }): Promise<void>;\n sendReaction(options: {\n roomId: string;\n eventId: string;\n key: string;\n }): Promise<{ eventId: string }>;\n\n // Room Management\n setRoomName(options: { roomId: string; name: string }): Promise<void>;\n setRoomTopic(options: { roomId: string; topic: string }): Promise<void>;\n setRoomAvatar(options: { roomId: string; mxcUrl: string }): Promise<void>;\n inviteUser(options: { roomId: string; userId: string }): Promise<void>;\n kickUser(options: { roomId: string; userId: string; reason?: string }): Promise<void>;\n banUser(options: { roomId: string; userId: string; reason?: string }): Promise<void>;\n unbanUser(options: { roomId: string; userId: string }): Promise<void>;\n\n // Typing\n sendTyping(options: {\n roomId: string;\n isTyping: boolean;\n timeout?: number;\n }): Promise<void>;\n\n // Media\n getMediaUrl(options: { mxcUrl: string }): Promise<{ httpUrl: string }>;\n getThumbnailUrl(options: ThumbnailUrlOptions): Promise<{ httpUrl: string }>;\n uploadContent(options: UploadContentOptions): Promise<UploadContentResult>;\n\n // User Discovery\n searchUsers(options: {\n searchTerm: string;\n limit?: number;\n }): Promise<{ results: UserProfile[]; limited: boolean }>;\n\n // Presence\n setPresence(options: {\n presence: 'online' | 'offline' | 'unavailable';\n statusMsg?: string;\n }): Promise<void>;\n getPresence(options: { userId: string }): Promise<PresenceInfo>;\n\n // Device Management\n getDevices(): Promise<{ devices: DeviceInfo[] }>;\n deleteDevice(options: { deviceId: string; auth?: Record<string, unknown> }): Promise<void>;\n\n // Push\n setPusher(options: PusherOptions): Promise<void>;\n\n // Encryption\n initializeCrypto(): Promise<void>;\n getEncryptionStatus(): Promise<EncryptionStatus>;\n bootstrapCrossSigning(): Promise<void>;\n setupKeyBackup(): Promise<KeyBackupStatus>;\n getKeyBackupStatus(): Promise<KeyBackupStatus>;\n restoreKeyBackup(options?: {\n recoveryKey?: string;\n }): Promise<{ importedKeys: number }>;\n setupRecovery(options?: {\n passphrase?: string;\n /**\n * Passphrase for the *existing* secret storage key, used by\n * bootstrapSecretStorage to decrypt and migrate the current cross-signing\n * and backup secrets into the newly created SSSS. Only needed on web;\n * native platforms (Rust SDK) handle the migration internally.\n */\n existingPassphrase?: string;\n }): Promise<RecoveryKeyInfo>;\n /** Wipe all local Matrix state (crypto DB, session, caches). */\n clearAllData(): Promise<void>;\n isRecoveryEnabled(): Promise<{ enabled: boolean }>;\n recoverAndSetup(options: {\n recoveryKey?: string;\n passphrase?: string;\n }): Promise<void>;\n resetRecoveryKey(options?: {\n passphrase?: string;\n }): Promise<RecoveryKeyInfo>;\n exportRoomKeys(options: {\n passphrase: string;\n }): Promise<{ data: string }>;\n importRoomKeys(options: {\n data: string;\n passphrase: string;\n }): Promise<{ importedKeys: number }>;\n\n // Listeners\n addListener(\n event: 'syncStateChange',\n listenerFunc: (data: SyncStateChangeEvent) => void,\n ): Promise<PluginListenerHandle>;\n addListener(\n event: 'messageReceived',\n listenerFunc: (data: MessageReceivedEvent) => void,\n ): Promise<PluginListenerHandle>;\n addListener(\n event: 'roomUpdated',\n listenerFunc: (data: RoomUpdatedEvent) => void,\n ): Promise<PluginListenerHandle>;\n addListener(\n event: 'typingChanged',\n listenerFunc: (data: TypingEvent) => void,\n ): Promise<PluginListenerHandle>;\n addListener(\n event: 'receiptReceived',\n listenerFunc: (data: ReceiptReceivedEvent) => void,\n ): Promise<PluginListenerHandle>;\n addListener(\n event: 'presenceChanged',\n listenerFunc: (data: PresenceChangedEvent) => void,\n ): Promise<PluginListenerHandle>;\n removeAllListeners(): Promise<void>;\n}\n"]}
|
|
1
|
+
{"version":3,"file":"definitions.js","sourceRoot":"","sources":["../../src/definitions.ts"],"names":[],"mappings":"","sourcesContent":["import type { PluginListenerHandle } from '@capacitor/core';\n\n// Auth & Session\n\nexport interface LoginOptions {\n homeserverUrl: string;\n userId: string;\n password: string;\n}\n\nexport interface LoginWithTokenOptions {\n homeserverUrl: string;\n accessToken: string;\n userId: string;\n deviceId: string;\n}\n\nexport interface SessionInfo {\n accessToken: string;\n userId: string;\n deviceId: string;\n homeserverUrl: string;\n}\n\n// Messaging\n\nexport interface SendMessageOptions {\n roomId: string;\n body: string;\n msgtype?: 'm.text' | 'm.notice' | 'm.emote' | 'm.image' | 'm.audio' | 'm.video' | 'm.file';\n fileUri?: string;\n fileName?: string;\n mimeType?: string;\n fileSize?: number;\n /** Audio/video duration in milliseconds (sets info.duration per Matrix spec) */\n duration?: number;\n /** Image/video width in pixels (sets info.w per Matrix spec) */\n width?: number;\n /** Image/video height in pixels (sets info.h per Matrix spec) */\n height?: number;\n}\n\n// Presence\n\nexport interface PresenceInfo {\n presence: 'online' | 'offline' | 'unavailable';\n statusMsg?: string;\n lastActiveAgo?: number;\n}\n\n// Typing\n\nexport interface TypingEvent {\n roomId: string;\n userIds: string[];\n}\n\nexport interface ReceiptReceivedEvent {\n roomId: string;\n /** The event that was read */\n eventId: string;\n /** The user who sent the read receipt */\n userId: string;\n}\n\nexport interface PresenceChangedEvent {\n userId: string;\n presence: PresenceInfo;\n}\n\n// Edit & Reply\n\nexport interface EditMessageOptions {\n roomId: string;\n eventId: string;\n newBody: string;\n /** Required when editing a media message; must match the original msgtype */\n msgtype?: 'm.text' | 'm.notice' | 'm.emote' | 'm.image' | 'm.audio' | 'm.video' | 'm.file';\n /** New file to replace the media content (optional for caption-only edits) */\n fileUri?: string;\n fileName?: string;\n mimeType?: string;\n fileSize?: number;\n /** Audio/video duration in milliseconds */\n duration?: number;\n /** Image/video width in pixels */\n width?: number;\n /** Image/video height in pixels */\n height?: number;\n}\n\nexport interface SendReplyOptions {\n roomId: string;\n body: string;\n replyToEventId: string;\n msgtype?: 'm.text' | 'm.notice' | 'm.emote' | 'm.image' | 'm.audio' | 'm.video' | 'm.file';\n fileUri?: string;\n fileName?: string;\n mimeType?: string;\n fileSize?: number;\n /** Audio/video duration in milliseconds (sets info.duration per Matrix spec) */\n duration?: number;\n /** Image/video width in pixels (sets info.w per Matrix spec) */\n width?: number;\n /** Image/video height in pixels (sets info.h per Matrix spec) */\n height?: number;\n}\n\n// Upload\n\nexport interface UploadContentOptions {\n fileUri: string;\n fileName: string;\n mimeType: string;\n}\n\nexport interface UploadContentResult {\n contentUri: string;\n}\n\n// Thumbnail\n\nexport interface ThumbnailUrlOptions {\n mxcUrl: string;\n width: number;\n height: number;\n method?: 'scale' | 'crop';\n}\n\nexport interface MatrixEvent {\n eventId: string;\n roomId: string;\n senderId: string;\n type: string;\n content: Record<string, unknown>;\n originServerTs: number;\n /** Delivery/read status for own messages: 'sending' | 'sent' | 'delivered' | 'read' */\n status?: 'sending' | 'sent' | 'delivered' | 'read';\n /** User IDs that have read this event */\n readBy?: string[];\n /** Unsigned data (e.g. m.relations for edits, transaction_id for local echo) */\n unsigned?: Record<string, unknown>;\n}\n\n// Rooms\n\nexport interface RoomSummary {\n roomId: string;\n name: string;\n topic?: string;\n memberCount: number;\n isEncrypted: boolean;\n unreadCount: number;\n lastEventTs?: number;\n membership?: 'join' | 'invite' | 'leave' | 'ban';\n avatarUrl?: string;\n isDirect?: boolean;\n}\n\nexport interface RoomMember {\n userId: string;\n displayName?: string;\n membership: 'join' | 'invite' | 'leave' | 'ban';\n avatarUrl?: string;\n}\n\n// Device Management\n\nexport interface DeviceInfo {\n deviceId: string;\n displayName?: string;\n lastSeenTs?: number;\n lastSeenIp?: string;\n /** Whether this device is verified via cross-signing */\n isCrossSigningVerified?: boolean;\n}\n\n// Pusher\n\nexport interface PusherOptions {\n pushkey: string;\n kind: string | null;\n appId: string;\n appDisplayName: string;\n deviceDisplayName: string;\n lang: string;\n data: { url: string; format?: string };\n}\n\n// User Discovery\n\nexport interface UserProfile {\n userId: string;\n displayName?: string;\n avatarUrl?: string;\n}\n\n// Encryption\n\nexport interface CrossSigningStatus {\n hasMaster: boolean;\n hasSelfSigning: boolean;\n hasUserSigning: boolean;\n isReady: boolean;\n}\n\nexport interface KeyBackupStatus {\n exists: boolean;\n version?: string;\n enabled: boolean;\n}\n\nexport interface RecoveryKeyInfo {\n recoveryKey: string;\n}\n\nexport interface EncryptionStatus {\n isCrossSigningReady: boolean;\n crossSigningStatus: CrossSigningStatus;\n isKeyBackupEnabled: boolean;\n keyBackupVersion?: string;\n isSecretStorageReady: boolean;\n}\n\n// Events & Sync\n\nexport type SyncState = 'INITIAL' | 'SYNCING' | 'ERROR' | 'STOPPED';\n\nexport interface SyncStateChangeEvent {\n state: SyncState;\n error?: string;\n}\n\nexport interface MessageReceivedEvent {\n event: MatrixEvent;\n}\n\nexport interface RoomUpdatedEvent {\n roomId: string;\n summary: RoomSummary;\n}\n\n// Plugin Interface\n\nexport interface MatrixPlugin {\n // Auth\n login(options: LoginOptions): Promise<SessionInfo>;\n loginWithToken(options: LoginWithTokenOptions): Promise<SessionInfo>;\n logout(): Promise<void>;\n getSession(): Promise<SessionInfo | null>;\n\n // Sync\n startSync(): Promise<void>;\n stopSync(): Promise<void>;\n getSyncState(): Promise<{ state: SyncState }>;\n\n // Rooms\n createRoom(options: {\n name?: string;\n topic?: string;\n isEncrypted?: boolean;\n isDirect?: boolean;\n invite?: string[];\n preset?: 'private_chat' | 'trusted_private_chat' | 'public_chat';\n historyVisibility?: 'invited' | 'joined' | 'shared' | 'world_readable';\n }): Promise<{ roomId: string }>;\n getRooms(): Promise<{ rooms: RoomSummary[] }>;\n getRoomMembers(options: { roomId: string }): Promise<{ members: RoomMember[] }>;\n joinRoom(options: { roomIdOrAlias: string }): Promise<{ roomId: string }>;\n leaveRoom(options: { roomId: string }): Promise<void>;\n forgetRoom(options: { roomId: string }): Promise<void>;\n\n // Messaging\n sendMessage(options: SendMessageOptions): Promise<{ eventId: string }>;\n editMessage(options: EditMessageOptions): Promise<{ eventId: string }>;\n sendReply(options: SendReplyOptions): Promise<{ eventId: string }>;\n getRoomMessages(options: {\n roomId: string;\n limit?: number;\n from?: string;\n }): Promise<{ events: MatrixEvent[]; nextBatch?: string }>;\n markRoomAsRead(options: {\n roomId: string;\n eventId: string;\n }): Promise<void>;\n refreshEventStatuses(options: {\n roomId: string;\n eventIds: string[];\n }): Promise<{ events: MatrixEvent[] }>;\n redactEvent(options: {\n roomId: string;\n eventId: string;\n reason?: string;\n }): Promise<void>;\n sendReaction(options: {\n roomId: string;\n eventId: string;\n key: string;\n }): Promise<{ eventId: string }>;\n\n // Room Management\n setRoomName(options: { roomId: string; name: string }): Promise<void>;\n setRoomTopic(options: { roomId: string; topic: string }): Promise<void>;\n setRoomAvatar(options: { roomId: string; mxcUrl: string }): Promise<void>;\n inviteUser(options: { roomId: string; userId: string }): Promise<void>;\n kickUser(options: { roomId: string; userId: string; reason?: string }): Promise<void>;\n banUser(options: { roomId: string; userId: string; reason?: string }): Promise<void>;\n unbanUser(options: { roomId: string; userId: string }): Promise<void>;\n\n // Typing\n sendTyping(options: {\n roomId: string;\n isTyping: boolean;\n timeout?: number;\n }): Promise<void>;\n\n // Media\n getMediaUrl(options: { mxcUrl: string }): Promise<{ httpUrl: string }>;\n getThumbnailUrl(options: ThumbnailUrlOptions): Promise<{ httpUrl: string }>;\n uploadContent(options: UploadContentOptions): Promise<UploadContentResult>;\n\n // User Discovery\n searchUsers(options: {\n searchTerm: string;\n limit?: number;\n }): Promise<{ results: UserProfile[]; limited: boolean }>;\n\n // Presence\n setPresence(options: {\n presence: 'online' | 'offline' | 'unavailable';\n statusMsg?: string;\n }): Promise<void>;\n getPresence(options: { userId: string }): Promise<PresenceInfo>;\n\n // Device Management\n getDevices(): Promise<{ devices: DeviceInfo[] }>;\n deleteDevice(options: { deviceId: string; auth?: Record<string, unknown> }): Promise<void>;\n verifyDevice(options: { deviceId: string }): Promise<void>;\n\n // Push\n setPusher(options: PusherOptions): Promise<void>;\n\n // Encryption\n initializeCrypto(): Promise<void>;\n getEncryptionStatus(): Promise<EncryptionStatus>;\n bootstrapCrossSigning(): Promise<void>;\n setupKeyBackup(): Promise<KeyBackupStatus>;\n getKeyBackupStatus(): Promise<KeyBackupStatus>;\n restoreKeyBackup(options?: {\n recoveryKey?: string;\n }): Promise<{ importedKeys: number }>;\n setupRecovery(options?: {\n passphrase?: string;\n /**\n * Passphrase for the *existing* secret storage key, used by\n * bootstrapSecretStorage to decrypt and migrate the current cross-signing\n * and backup secrets into the newly created SSSS. Only needed on web;\n * native platforms (Rust SDK) handle the migration internally.\n */\n existingPassphrase?: string;\n }): Promise<RecoveryKeyInfo>;\n /** Wipe all local Matrix state (crypto DB, session, caches). */\n clearAllData(): Promise<void>;\n isRecoveryEnabled(): Promise<{ enabled: boolean }>;\n recoverAndSetup(options: {\n recoveryKey?: string;\n passphrase?: string;\n }): Promise<void>;\n resetRecoveryKey(options?: {\n passphrase?: string;\n }): Promise<RecoveryKeyInfo>;\n exportRoomKeys(options: {\n passphrase: string;\n }): Promise<{ data: string }>;\n importRoomKeys(options: {\n data: string;\n passphrase: string;\n }): Promise<{ importedKeys: number }>;\n\n // Listeners\n addListener(\n event: 'syncStateChange',\n listenerFunc: (data: SyncStateChangeEvent) => void,\n ): Promise<PluginListenerHandle>;\n addListener(\n event: 'messageReceived',\n listenerFunc: (data: MessageReceivedEvent) => void,\n ): Promise<PluginListenerHandle>;\n addListener(\n event: 'roomUpdated',\n listenerFunc: (data: RoomUpdatedEvent) => void,\n ): Promise<PluginListenerHandle>;\n addListener(\n event: 'typingChanged',\n listenerFunc: (data: TypingEvent) => void,\n ): Promise<PluginListenerHandle>;\n addListener(\n event: 'receiptReceived',\n listenerFunc: (data: ReceiptReceivedEvent) => void,\n ): Promise<PluginListenerHandle>;\n addListener(\n event: 'presenceChanged',\n listenerFunc: (data: PresenceChangedEvent) => void,\n ): Promise<PluginListenerHandle>;\n removeAllListeners(): Promise<void>;\n}\n"]}
|
package/dist/esm/web.d.ts
CHANGED
|
@@ -167,6 +167,9 @@ export declare class MatrixWeb extends WebPlugin implements MatrixPlugin {
|
|
|
167
167
|
recoveryKey?: string;
|
|
168
168
|
passphrase?: string;
|
|
169
169
|
}): Promise<void>;
|
|
170
|
+
verifyDevice(options: {
|
|
171
|
+
deviceId: string;
|
|
172
|
+
}): Promise<void>;
|
|
170
173
|
resetRecoveryKey(options?: {
|
|
171
174
|
passphrase?: string;
|
|
172
175
|
}): Promise<RecoveryKeyInfo>;
|
package/dist/esm/web.js
CHANGED
|
@@ -742,7 +742,6 @@ export class MatrixWeb extends WebPlugin {
|
|
|
742
742
|
async bootstrapCrossSigning() {
|
|
743
743
|
const crypto = await this.ensureCrypto();
|
|
744
744
|
await crypto.bootstrapCrossSigning({
|
|
745
|
-
setupNewCrossSigning: true,
|
|
746
745
|
authUploadDeviceSigningKeys: async (makeRequest) => {
|
|
747
746
|
var _a;
|
|
748
747
|
// UIA flow: attempt with dummy auth, fall back to session-based retry
|
|
@@ -871,6 +870,36 @@ export class MatrixWeb extends WebPlugin {
|
|
|
871
870
|
}
|
|
872
871
|
// Now that the key is stored locally, activate backup in the running client
|
|
873
872
|
await crypto.checkKeyBackupAndEnable();
|
|
873
|
+
// Restore cross-signing trust for the current device.
|
|
874
|
+
// With the SSSS key now available (via getSecretStorageKey callback),
|
|
875
|
+
// bootstrapCrossSigning downloads the existing keys from secret storage
|
|
876
|
+
// and self-verifies this device.
|
|
877
|
+
try {
|
|
878
|
+
await crypto.bootstrapCrossSigning({
|
|
879
|
+
authUploadDeviceSigningKeys: async (makeRequest) => {
|
|
880
|
+
var _a;
|
|
881
|
+
try {
|
|
882
|
+
await makeRequest({ type: 'm.login.dummy' });
|
|
883
|
+
}
|
|
884
|
+
catch (e) {
|
|
885
|
+
const session = (_a = e === null || e === void 0 ? void 0 : e.data) === null || _a === void 0 ? void 0 : _a.session;
|
|
886
|
+
if (session) {
|
|
887
|
+
await makeRequest({ type: 'm.login.dummy', session });
|
|
888
|
+
}
|
|
889
|
+
else {
|
|
890
|
+
throw e;
|
|
891
|
+
}
|
|
892
|
+
}
|
|
893
|
+
},
|
|
894
|
+
});
|
|
895
|
+
}
|
|
896
|
+
catch (_a) {
|
|
897
|
+
// Non-fatal: cross-signing restore failed, device remains unverified
|
|
898
|
+
}
|
|
899
|
+
}
|
|
900
|
+
async verifyDevice(options) {
|
|
901
|
+
const crypto = await this.ensureCrypto();
|
|
902
|
+
await crypto.crossSignDevice(options.deviceId);
|
|
874
903
|
}
|
|
875
904
|
async resetRecoveryKey(options) {
|
|
876
905
|
return this.setupRecovery(options);
|
package/dist/esm/web.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"web.js","sourceRoot":"","sources":["../../src/web.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AAC5C,OAAO,EACL,YAAY,EACZ,WAAW,EACX,SAAS,EACT,eAAe,EACf,SAAS,EACT,OAAO,EACP,SAAS,EACT,YAAY,EACZ,SAAS,GACV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,iBAAiB,EAAE,MAAM,2CAA2C,CAAC;AAC9E,OAAO,EAAE,+BAA+B,EAAE,MAAM,6CAA6C,CAAC;AA2B9F,MAAM,WAAW,GAAG,gBAAgB,CAAC;AAErC,MAAM,OAAO,SAAU,SAAQ,SAAS;IAAxC;;QAOmB,qBAAgB,GAAG;YAClC,mBAAmB,EAAE,KAAK,EACxB,IAAuC,EACY,EAAE;;gBACrD,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;gBACxC,IAAI,CAAC,KAAK;oBAAE,OAAO,IAAI,CAAC;gBAExB,kFAAkF;gBAClF,+EAA+E;gBAC/E,qFAAqF;gBACrF,IAAI,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,kBAAkB,KAAK,KAAK,EAAE,CAAC;oBAC/D,OAAO,CAAC,KAAK,EAAE,IAAI,CAAC,gBAAgB,CAAC,CAAC;gBACxC,CAAC;gBAED,kEAAkE;gBAClE,IAAI,IAAI,CAAC,kBAAkB,EAAE,CAAC;oBAC5B,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAE9B,CAAC;oBACF,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,UAAU,EAAE,CAAC;wBACxB,MAAM,OAAO,GAAG,MAAM,+BAA+B,CACnD,IAAI,CAAC,kBAAkB,EACvB,OAAO,CAAC,UAAU,CAAC,IAAI,EACvB,OAAO,CAAC,UAAU,CAAC,UAAU,EAC7B,MAAA,OAAO,CAAC,UAAU,CAAC,IAAI,mCAAI,GAAG,CAC/B,CAAC;wBACF,qDAAqD;wBACrD,IAAI,CAAC,gBAAgB,GAAG,OAAO,CAAC;wBAChC,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;wBAChC,OAAO,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;oBAC1B,CAAC;gBACH,CAAC;gBAED,0EAA0E;gBAC1E,qEAAqE;gBACrE,IAAI,IAAI,CAAC,kBAAkB,EAAE,CAAC;oBAC5B,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAE9B,CAAC;oBACF,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,UAAU,EAAE,CAAC;wBACxB,MAAM,OAAO,GAAG,MAAM,+BAA+B,CACnD,IAAI,CAAC,kBAAkB,EACvB,OAAO,CAAC,UAAU,CAAC,IAAI,EACvB,OAAO,CAAC,UAAU,CAAC,UAAU,EAC7B,MAAA,OAAO,CAAC,UAAU,CAAC,IAAI,mCAAI,GAAG,CAC/B,CAAC;wBACF,OAAO,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;oBAC1B,CAAC;gBACH,CAAC;gBAED,OAAO,IAAI,CAAC;YACd,CAAC;YACD,qBAAqB,EAAE,CACrB,KAAa,EACb,QAAiB,EACjB,GAA4B,EACtB,EAAE;gBACR,IAAI,CAAC,gBAAgB,GAAG,GAAG,CAAC;gBAC5B,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;YAClC,CAAC;SACF,CAAC;IAkqCJ,CAAC;IAhqCC,yDAAyD;IAEzD,KAAK,CAAC,KAAK,CAAC,OAAqB;QAC/B,MAAM,SAAS,GAAG,YAAY,CAAC,EAAE,OAAO,EAAE,OAAO,CAAC,aAAa,EAAE,CAAC,CAAC;QACnE,MAAM,GAAG,GAAG,MAAM,SAAS,CAAC,iBAAiB,CAAC,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,QAAQ,CAAC,CAAC;QAEhF,IAAI,CAAC,MAAM,GAAG,YAAY,CAAC;YACzB,OAAO,EAAE,OAAO,CAAC,aAAa;YAC9B,WAAW,EAAE,GAAG,CAAC,YAAY;YAC7B,MAAM,EAAE,GAAG,CAAC,OAAO;YACnB,QAAQ,EAAE,GAAG,CAAC,SAAS;YACvB,eAAe,EAAE,IAAI,CAAC,gBAAgB;SACvC,CAAC,CAAC;QAEH,MAAM,OAAO,GAAgB;YAC3B,WAAW,EAAE,GAAG,CAAC,YAAY;YAC7B,MAAM,EAAE,GAAG,CAAC,OAAO;YACnB,QAAQ,EAAE,GAAG,CAAC,SAAS;YACvB,aAAa,EAAE,OAAO,CAAC,aAAa;SACrC,CAAC;QAEF,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC;QAC7B,OAAO,OAAO,CAAC;IACjB,CAAC;IAED,KAAK,CAAC,cAAc,CAAC,OAA8B;QACjD,sEAAsE;QACtE,uDAAuD;QACvD,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE,CAAC;YACzB,IAAI,CAAC,MAAM,GAAG,SAAS,CAAC;QAC1B,CAAC;QAED,IAAI,CAAC,MAAM,GAAG,YAAY,CAAC;YACzB,OAAO,EAAE,OAAO,CAAC,aAAa;YAC9B,WAAW,EAAE,OAAO,CAAC,WAAW;YAChC,MAAM,EAAE,OAAO,CAAC,MAAM;YACtB,QAAQ,EAAE,OAAO,CAAC,QAAQ;YAC1B,eAAe,EAAE,IAAI,CAAC,gBAAgB;SACvC,CAAC,CAAC;QAEH,MAAM,OAAO,GAAgB;YAC3B,WAAW,EAAE,OAAO,CAAC,WAAW;YAChC,MAAM,EAAE,OAAO,CAAC,MAAM;YACtB,QAAQ,EAAE,OAAO,CAAC,QAAQ;YAC1B,aAAa,EAAE,OAAO,CAAC,aAAa;SACrC,CAAC;QAEF,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC;QAC7B,OAAO,OAAO,CAAC;IACjB,CAAC;IAED,KAAK,CAAC,MAAM;QACV,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE,CAAC;YACzB,IAAI,CAAC;gBACH,MAAM,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;YACjC,CAAC;YAAC,WAAM,CAAC;gBACP,wDAAwD;YAC1D,CAAC;YACD,IAAI,CAAC,MAAM,GAAG,SAAS,CAAC;QAC1B,CAAC;QACD,YAAY,CAAC,UAAU,CAAC,WAAW,CAAC,CAAC;IACvC,CAAC;IAED,KAAK,CAAC,YAAY;QAChB,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE,CAAC;YACzB,IAAI,CAAC,MAAM,GAAG,SAAS,CAAC;QAC1B,CAAC;QAED,gCAAgC;QAChC,IAAI,CAAC,gBAAgB,GAAG,SAAS,CAAC;QAClC,IAAI,CAAC,kBAAkB,GAAG,SAAS,CAAC;QACpC,IAAI,CAAC,kBAAkB,GAAG,SAAS,CAAC;QACpC,IAAI,CAAC,kBAAkB,GAAG,SAAS,CAAC;QAEpC,YAAY,CAAC,UAAU,CAAC,WAAW,CAAC,CAAC;QAErC,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;IACjC,CAAC;IAED,KAAK,CAAC,UAAU;QACd,MAAM,GAAG,GAAG,YAAY,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;QAC9C,IAAI,CAAC,GAAG;YAAE,OAAO,IAAI,CAAC;QACtB,IAAI,CAAC;YACH,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAgB,CAAC;QACxC,CAAC;QAAC,WAAM,CAAC;YACP,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED,yDAAyD;IAEzD,KAAK,CAAC,SAAS;QACb,IAAI,CAAC,aAAa,EAAE,CAAC;QAErB,IAAI,CAAC,MAAO,CAAC,EAAE,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC,KAAK,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE;;YACvD,MAAM,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;YACxC,IAAI,CAAC,eAAe,CAAC,iBAAiB,EAAE;gBACtC,KAAK,EAAE,MAAM;gBACb,KAAK,EAAE,MAAA,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,KAAK,0CAAE,OAAO;aAC5B,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,MAAO,CAAC,EAAE,CAAC,SAAS,CAAC,QAAQ,EAAE,CAAC,KAAqB,EAAE,IAAsB,EAAE,EAAE;;YACpF,IAAI,CAAC,eAAe,CAAC,iBAAiB,EAAE;gBACtC,KAAK,EAAE,IAAI,CAAC,cAAc,CAAC,KAAK,EAAE,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,MAAM,CAAC;aAChD,CAAC,CAAC;YACH,uEAAuE;YACvE,IAAI,KAAK,CAAC,gBAAgB,EAAE,IAAI,KAAK,CAAC,OAAO,EAAE,KAAK,kBAAkB,EAAE,CAAC;gBACvE,KAAK,CAAC,IAAI,CAAC,iBAAwB,EAAE,GAAG,EAAE;oBACxC,IAAI,CAAC,eAAe,CAAC,iBAAiB,EAAE;wBACtC,KAAK,EAAE,IAAI,CAAC,cAAc,CAAC,KAAK,EAAE,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,MAAM,CAAC;qBAChD,CAAC,CAAC;gBACL,CAAC,CAAC,CAAC;YACL,CAAC;YACD,mGAAmG;YACnG,IAAI,KAAK,CAAC,OAAO,EAAE,KAAK,SAAS,CAAC,QAAQ,IAAI,KAAK,CAAC,OAAO,EAAE,KAAK,SAAS,CAAC,aAAa,EAAE,CAAC;gBAC1F,MAAM,GAAG,GAAG,MAAA,KAAK,CAAC,UAAU,EAAE,0CAAG,cAAc,CAAC,CAAC;gBACjD,MAAM,QAAQ,GAAG,CAAA,GAAG,aAAH,GAAG,uBAAH,GAAG,CAAE,QAAQ,KAAI,KAAK,CAAC,eAAe,EAAE,CAAC;gBAC1D,IAAI,QAAQ,IAAI,IAAI,EAAE,CAAC;oBACrB,MAAM,WAAW,GAAG,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;oBACjD,IAAI,WAAW,EAAE,CAAC;wBAChB,gDAAgD;wBAChD,UAAU,CAAC,GAAG,EAAE;4BACd,IAAI,CAAC,eAAe,CAAC,iBAAiB,EAAE;gCACtC,KAAK,EAAE,IAAI,CAAC,cAAc,CAAC,WAAW,EAAE,IAAI,CAAC,MAAM,CAAC;6BACrD,CAAC,CAAC;wBACL,CAAC,EAAE,GAAG,CAAC,CAAC;oBACV,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,MAAO,CAAC,EAAE,CAAC,SAAS,CAAC,OAAO,EAAE,CAAC,KAAqB,EAAE,IAAU,EAAE,EAAE;;YACvE,MAAM,cAAc,GAAG,KAAK,CAAC,UAAU,EAA6D,CAAC;YACrG,KAAK,MAAM,CAAC,OAAO,EAAE,YAAY,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,cAAc,CAAC,EAAE,CAAC;gBACrE,MAAM,KAAK,GAAG,MAAA,YAAY,CAAC,QAAQ,CAAC,mCAAI,EAAE,CAAC;gBAC3C,KAAK,MAAM,MAAM,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;oBACxC,IAAI,CAAC,eAAe,CAAC,iBAAiB,EAAE;wBACtC,MAAM,EAAE,IAAI,CAAC,MAAM;wBACnB,OAAO;wBACP,MAAM;qBACP,CAAC,CAAC;gBACL,CAAC;YACH,CAAC;YACD,qDAAqD;YACrD,MAAM,QAAQ,GAAG,MAAA,IAAI,CAAC,MAAM,0CAAE,SAAS,EAAE,CAAC;YAC1C,IAAI,QAAQ,EAAE,CAAC;gBACb,MAAM,QAAQ,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC,SAAS,EAAE,CAAC;gBACpD,+EAA+E;gBAC/E,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;gBAC5C,KAAK,IAAI,CAAC,GAAG,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC,IAAI,QAAQ,CAAC,MAAM,GAAG,KAAK,EAAE,CAAC,EAAE,EAAE,CAAC;oBACpE,MAAM,GAAG,GAAG,QAAQ,CAAC,CAAC,CAAC,CAAC;oBACxB,IAAI,GAAG,CAAC,SAAS,EAAE,KAAK,QAAQ;wBAAE,SAAS;oBAC3C,MAAM,UAAU,GAAG,IAAI,CAAC,cAAc,CAAC,GAAG,EAAE,IAAI,CAAC,MAAM,CAAC,CAAC;oBACzD,IAAI,UAAU,CAAC,MAAM,KAAK,MAAM,EAAE,CAAC;wBACjC,IAAI,CAAC,eAAe,CAAC,iBAAiB,EAAE,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC,CAAC;oBACjE,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,MAAO,CAAC,EAAE,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,IAAU,EAAE,EAAE;YAC7C,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE;gBAClC,MAAM,EAAE,IAAI,CAAC,MAAM;gBACnB,OAAO,EAAE,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC;aAClC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,MAAO,CAAC,EAAE,CAAC,eAAe,CAAC,MAAM,EAAE,CAAC,MAAsB,EAAE,MAAW,EAAE,EAAE;YAC9E,MAAM,MAAM,GAAG,MAAM,aAAN,MAAM,uBAAN,MAAM,CAAE,MAAM,CAAC;YAC9B,IAAI,MAAM,EAAE,CAAC;gBACX,MAAM,IAAI,GAAG,IAAI,CAAC,MAAO,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;gBAC1C,IAAI,IAAI,EAAE,CAAC;oBACT,MAAM,OAAO,GAAa,IAAI;yBAC3B,UAAU,EAAE;yBACZ,MAAM,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC;yBAC5B,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;oBAC7B,IAAI,CAAC,eAAe,CAAC,eAAe,EAAE,EAAE,MAAM,EAAE,OAAO,EAAE,CAAC,CAAC;gBAC7D,CAAC;YACH,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,MAAO,CAAC,EAAE,CAAC,SAAS,CAAC,QAAQ,EAAE,CAAC,MAAkC,EAAE,IAAU,EAAE,EAAE;;YACrF,IAAI,CAAC,eAAe,CAAC,iBAAiB,EAAE;gBACtC,MAAM,EAAE,IAAI,CAAC,MAAM;gBACnB,QAAQ,EAAE;oBACR,QAAQ,EAAE,MAAC,IAAI,CAAC,QAAqC,mCAAI,SAAS;oBAClE,SAAS,EAAE,MAAA,IAAI,CAAC,iBAAiB,mCAAI,SAAS;oBAC9C,aAAa,EAAE,MAAA,IAAI,CAAC,aAAa,mCAAI,SAAS;iBAC/C;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,MAAM,IAAI,CAAC,MAAO,CAAC,WAAW,CAAC,EAAE,gBAAgB,EAAE,EAAE,EAAE,CAAC,CAAC;IAC3D,CAAC;IAED,KAAK,CAAC,QAAQ;QACZ,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,IAAI,CAAC,MAAO,CAAC,UAAU,EAAE,CAAC;IAC5B,CAAC;IAED,KAAK,CAAC,YAAY;QAChB,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,GAAG,GAAG,IAAI,CAAC,MAAO,CAAC,YAAY,EAAE,CAAC;QACxC,OAAO,EAAE,KAAK,EAAE,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC;IAC3C,CAAC;IAED,yDAAyD;IAEzD,KAAK,CAAC,UAAU,CAAC,OAQhB;;QACC,IAAI,CAAC,aAAa,EAAE,CAAC;QAErB,MAAM,UAAU,GAA4B;YAC1C,UAAU,EAAE,SAAkB;SAC/B,CAAC;QACF,IAAI,OAAO,CAAC,IAAI;YAAE,UAAU,CAAC,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC;QACjD,IAAI,OAAO,CAAC,KAAK;YAAE,UAAU,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC;QACpD,IAAI,MAAA,OAAO,CAAC,MAAM,0CAAE,MAAM;YAAE,UAAU,CAAC,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;QAC/D,IAAI,OAAO,CAAC,MAAM;YAAE,UAAU,CAAC,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;QACvD,IAAI,OAAO,CAAC,QAAQ;YAAE,UAAU,CAAC,SAAS,GAAG,IAAI,CAAC;QAElD,MAAM,YAAY,GAA8B,EAAE,CAAC;QACnD,IAAI,OAAO,CAAC,WAAW,EAAE,CAAC;YACxB,YAAY,CAAC,IAAI,CAAC;gBAChB,IAAI,EAAE,mBAAmB;gBACzB,SAAS,EAAE,EAAE;gBACb,OAAO,EAAE,EAAE,SAAS,EAAE,sBAAsB,EAAE;aAC/C,CAAC,CAAC;QACL,CAAC;QACD,IAAI,OAAO,CAAC,iBAAiB,EAAE,CAAC;YAC9B,YAAY,CAAC,IAAI,CAAC;gBAChB,IAAI,EAAE,2BAA2B;gBACjC,SAAS,EAAE,EAAE;gBACb,OAAO,EAAE,EAAE,kBAAkB,EAAE,OAAO,CAAC,iBAAiB,EAAE;aAC3D,CAAC,CAAC;QACL,CAAC;QACD,IAAI,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC5B,UAAU,CAAC,aAAa,GAAG,YAAY,CAAC;QAC1C,CAAC;QAED,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,MAAO,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC;QACtD,OAAO,EAAE,MAAM,EAAE,GAAG,CAAC,OAAO,EAAE,CAAC;IACjC,CAAC;IAED,KAAK,CAAC,QAAQ;QACZ,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,KAAK,GAAG,IAAI,CAAC,MAAO,CAAC,QAAQ,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;QACxE,OAAO,EAAE,KAAK,EAAE,CAAC;IACnB,CAAC;IAED,KAAK,CAAC,cAAc,CAAC,OAA2B;QAC9C,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,GAAG,IAAI,CAAC,MAAO,CAAC,OAAO,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;QAClD,IAAI,CAAC,IAAI;YAAE,MAAM,IAAI,KAAK,CAAC,QAAQ,OAAO,CAAC,MAAM,YAAY,CAAC,CAAC;QAE/D,MAAM,IAAI,CAAC,mBAAmB,EAAE,CAAC;QACjC,MAAM,OAAO,GAAiB,IAAI,CAAC,UAAU,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE;;YAAC,OAAA,CAAC;gBAC1D,MAAM,EAAE,CAAC,CAAC,MAAM;gBAChB,WAAW,EAAE,MAAA,CAAC,CAAC,IAAI,mCAAI,SAAS;gBAChC,UAAU,EAAE,CAAC,CAAC,UAAsC;gBACpD,SAAS,EAAE,MAAA,CAAC,CAAC,eAAe,EAAE,mCAAI,SAAS;aAC5C,CAAC,CAAA;SAAA,CAAC,CAAC;QAEJ,OAAO,EAAE,OAAO,EAAE,CAAC;IACrB,CAAC;IAED,KAAK,CAAC,QAAQ,CAAC,OAAkC;QAC/C,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,GAAG,MAAM,IAAI,CAAC,MAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;QAChE,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE,CAAC;IACjC,CAAC;IAED,KAAK,CAAC,SAAS,CAAC,OAA2B;QACzC,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,CAAC,MAAO,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;IAC3C,CAAC;IAED,KAAK,CAAC,UAAU,CAAC,OAA2B;QAC1C,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,CAAC,MAAO,CAAC,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;IAC5C,CAAC;IAED,yDAAyD;IAEzD,KAAK,CAAC,WAAW,CAAC,OAA2B;;QAC3C,IAAI,CAAC,aAAa,EAAE,CAAC;QAErB,MAAM,OAAO,GAAG,MAAA,OAAO,CAAC,OAAO,mCAAI,QAAQ,CAAC;QAC5C,MAAM,UAAU,GAAG,CAAC,SAAS,EAAE,SAAS,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;QAE/D,IAAI,UAAU,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpD,uCAAuC;YACvC,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;YAC9C,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;YACnC,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,MAAO,CAAC,aAAa,CAAC,IAAI,EAAE;gBACvD,IAAI,EAAE,OAAO,CAAC,QAAQ;gBACtB,IAAI,EAAE,OAAO,CAAC,QAAQ;aACvB,CAAC,CAAC;YACH,MAAM,MAAM,GAAG,SAAS,CAAC,WAAW,CAAC;YACrC,MAAM,OAAO,GAA4B;gBACvC,OAAO;gBACP,IAAI,EAAE,OAAO,CAAC,IAAI,IAAI,OAAO,CAAC,QAAQ,IAAI,MAAM;gBAChD,GAAG,EAAE,MAAM;gBACX,IAAI,8CACF,QAAQ,EAAE,OAAO,CAAC,QAAQ,EAC1B,IAAI,EAAE,MAAA,OAAO,CAAC,QAAQ,mCAAI,IAAI,CAAC,IAAI,IAChC,CAAC,OAAO,CAAC,QAAQ,KAAK,SAAS,IAAI,EAAE,QAAQ,EAAE,OAAO,CAAC,QAAQ,EAAE,CAAC,GAClE,CAAC,OAAO,CAAC,KAAK,KAAK,SAAS,IAAI,EAAE,CAAC,EAAE,OAAO,CAAC,KAAK,EAAE,CAAC,GACrD,CAAC,OAAO,CAAC,MAAM,KAAK,SAAS,IAAI,EAAE,CAAC,EAAE,OAAO,CAAC,MAAM,EAAE,CAAC,CAC3D;aACF,CAAC;YACF,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,MAAO,CAAC,WAAW,CAAC,OAAO,CAAC,MAAM,EAAE,OAAc,CAAC,CAAC;YAC3E,OAAO,EAAE,OAAO,EAAE,GAAG,CAAC,QAAQ,EAAE,CAAC;QACnC,CAAC;QAED,eAAe;QACf,MAAM,UAAU,GAAG;YACjB,QAAQ,EAAE,OAAO,CAAC,IAAI;YACtB,UAAU,EAAE,OAAO,CAAC,MAAM;YAC1B,SAAS,EAAE,OAAO,CAAC,KAAK;SAChB,CAAC;QACX,MAAM,UAAU,GAAG,MAAA,UAAU,CAAC,OAAkC,CAAC,mCAAI,OAAO,CAAC,IAAI,CAAC;QAClF,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,MAAO,CAAC,WAAW,CAAC,OAAO,CAAC,MAAM,EAAE;YACzD,OAAO,EAAE,UAAU;YACnB,IAAI,EAAE,OAAO,CAAC,IAAI;SACnB,CAAC,CAAC;QACH,OAAO,EAAE,OAAO,EAAE,GAAG,CAAC,QAAQ,EAAE,CAAC;IACnC,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,OAA2B;;QAC3C,IAAI,CAAC,aAAa,EAAE,CAAC;QAErB,MAAM,OAAO,GAAG,MAAA,OAAO,CAAC,OAAO,mCAAI,QAAQ,CAAC;QAC5C,MAAM,UAAU,GAAG,CAAC,SAAS,EAAE,SAAS,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;QAE/D,IAAI,UAAmC,CAAC;QAExC,IAAI,UAAU,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpD,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;YAC9C,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;YACnC,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,MAAO,CAAC,aAAa,CAAC,IAAI,EAAE;gBACvD,IAAI,EAAE,OAAO,CAAC,QAAQ;gBACtB,IAAI,EAAE,OAAO,CAAC,QAAQ;aACvB,CAAC,CAAC;YACH,UAAU,GAAG;gBACX,OAAO;gBACP,IAAI,EAAE,OAAO,CAAC,OAAO,IAAI,OAAO,CAAC,QAAQ,IAAI,MAAM;gBACnD,GAAG,EAAE,SAAS,CAAC,WAAW;gBAC1B,IAAI,8CACF,QAAQ,EAAE,OAAO,CAAC,QAAQ,EAC1B,IAAI,EAAE,MAAA,OAAO,CAAC,QAAQ,mCAAI,IAAI,CAAC,IAAI,IAChC,CAAC,OAAO,CAAC,QAAQ,KAAK,SAAS,IAAI,EAAE,QAAQ,EAAE,OAAO,CAAC,QAAQ,EAAE,CAAC,GAClE,CAAC,OAAO,CAAC,KAAK,KAAK,SAAS,IAAI,EAAE,CAAC,EAAE,OAAO,CAAC,KAAK,EAAE,CAAC,GACrD,CAAC,OAAO,CAAC,MAAM,KAAK,SAAS,IAAI,EAAE,CAAC,EAAE,OAAO,CAAC,MAAM,EAAE,CAAC,CAC3D;aACF,CAAC;QACJ,CAAC;aAAM,CAAC;YACN,UAAU,GAAG;gBACX,OAAO;gBACP,IAAI,EAAE,OAAO,CAAC,OAAO;aACtB,CAAC;QACJ,CAAC;QAED,MAAM,OAAO,mCACR,UAAU,KACb,IAAI,EAAE,KAAK,OAAO,CAAC,OAAO,EAAE,EAC5B,eAAe,EAAE,UAAU,EAC3B,cAAc,EAAE;gBACd,QAAQ,EAAE,WAAW;gBACrB,QAAQ,EAAE,OAAO,CAAC,OAAO;aAC1B,GACF,CAAC;QACF,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,MAAO,CAAC,WAAW,CAAC,OAAO,CAAC,MAAM,EAAE,OAAc,CAAC,CAAC;QAC3E,OAAO,EAAE,OAAO,EAAE,GAAG,CAAC,QAAQ,EAAE,CAAC;IACnC,CAAC;IAED,KAAK,CAAC,SAAS,CAAC,OAAyB;;QACvC,IAAI,CAAC,aAAa,EAAE,CAAC;QAErB,MAAM,OAAO,GAAG,MAAA,OAAO,CAAC,OAAO,mCAAI,QAAQ,CAAC;QAC5C,MAAM,UAAU,GAAG,CAAC,SAAS,EAAE,SAAS,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;QAE/D,IAAI,OAAgC,CAAC;QAErC,IAAI,UAAU,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpD,yDAAyD;YACzD,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;YAC9C,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;YACnC,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,MAAO,CAAC,aAAa,CAAC,IAAI,EAAE;gBACvD,IAAI,EAAE,OAAO,CAAC,QAAQ;gBACtB,IAAI,EAAE,OAAO,CAAC,QAAQ;aACvB,CAAC,CAAC;YACH,OAAO,GAAG;gBACR,OAAO;gBACP,IAAI,EAAE,OAAO,CAAC,IAAI,IAAI,OAAO,CAAC,QAAQ,IAAI,MAAM;gBAChD,GAAG,EAAE,SAAS,CAAC,WAAW;gBAC1B,IAAI,8CACF,QAAQ,EAAE,OAAO,CAAC,QAAQ,EAC1B,IAAI,EAAE,MAAA,OAAO,CAAC,QAAQ,mCAAI,IAAI,CAAC,IAAI,IAChC,CAAC,OAAO,CAAC,QAAQ,KAAK,SAAS,IAAI,EAAE,QAAQ,EAAE,OAAO,CAAC,QAAQ,EAAE,CAAC,GAClE,CAAC,OAAO,CAAC,KAAK,KAAK,SAAS,IAAI,EAAE,CAAC,EAAE,OAAO,CAAC,KAAK,EAAE,CAAC,GACrD,CAAC,OAAO,CAAC,MAAM,KAAK,SAAS,IAAI,EAAE,CAAC,EAAE,OAAO,CAAC,MAAM,EAAE,CAAC,CAC3D;gBACD,cAAc,EAAE;oBACd,eAAe,EAAE;wBACf,QAAQ,EAAE,OAAO,CAAC,cAAc;qBACjC;iBACF;aACF,CAAC;QACJ,CAAC;aAAM,CAAC;YACN,aAAa;YACb,OAAO,GAAG;gBACR,OAAO,EAAE,OAAO,CAAC,IAAI;gBACrB,IAAI,EAAE,OAAO,CAAC,IAAI;gBAClB,cAAc,EAAE;oBACd,eAAe,EAAE;wBACf,QAAQ,EAAE,OAAO,CAAC,cAAc;qBACjC;iBACF;aACF,CAAC;QACJ,CAAC;QAED,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,MAAO,CAAC,WAAW,CAAC,OAAO,CAAC,MAAM,EAAE,OAAc,CAAC,CAAC;QAC3E,OAAO,EAAE,OAAO,EAAE,GAAG,CAAC,QAAQ,EAAE,CAAC;IACnC,CAAC;IAED,KAAK,CAAC,eAAe,CACnB,OAA0D;;QAE1D,IAAI,CAAC,aAAa,EAAE,CAAC;QAErB,MAAM,KAAK,GAAG,MAAA,OAAO,CAAC,KAAK,mCAAI,EAAE,CAAC;QAClC,MAAM,IAAI,GAAG,IAAI,CAAC,MAAO,CAAC,OAAO,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;QAElD,0EAA0E;QAC1E,IAAI,CAAC,OAAO,CAAC,IAAI,IAAI,IAAI,EAAE,CAAC;YAC1B,0EAA0E;YAC1E,IAAI,CAAC;gBACH,MAAM,IAAI,CAAC,MAAO,CAAC,UAAU,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;YAC7C,CAAC;YAAC,WAAM,CAAC;gBACP,iDAAiD;YACnD,CAAC;YAED,MAAM,QAAQ,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;YACxC,MAAM,cAAc,GAAG,QAAQ,CAAC,SAAS,EAAE,CAAC;YAC5C,6FAA6F;YAC7F,MAAM,iBAAiB,GAAG,cAAc,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE;gBACpD,MAAM,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,CAAC;gBACtB,OAAO,CAAC,KAAK,SAAS,CAAC,QAAQ,IAAI,CAAC,KAAK,SAAS,CAAC,aAAa,CAAC;YACnE,CAAC,CAAC,CAAC;YACH,MAAM,MAAM,GAAkB,iBAAiB;iBAC5C,KAAK,CAAC,CAAC,KAAK,CAAC;iBACb,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;iBAClD,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,cAAc,GAAG,CAAC,CAAC,cAAc,CAAC,CAAC;YACvD,MAAM,SAAS,GAAG,MAAA,QAAQ,CAAC,kBAAkB,CAAC,SAAS,CAAC,QAAQ,CAAC,mCAAI,SAAS,CAAC;YAC/E,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,SAAS,EAAE,CAAC;QAC1C,CAAC;QAED,wCAAwC;QACxC,MAAM,SAAS,GAAG,MAAA,OAAO,CAAC,IAAI,mCAAI,IAAI,CAAC;QACvC,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,MAAO,CAAC,qBAAqB,CAClD,OAAO,CAAC,MAAM,EACd,SAAS,EACT,KAAK,EACL,SAAS,CAAC,QAAQ,CACnB,CAAC;QAEF,MAAM,MAAM,GAAkB,CAAC,MAAA,GAAG,CAAC,KAAK,mCAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE;;YAAC,OAAA,CAAC;gBAC1D,OAAO,EAAE,CAAC,CAAC,QAAQ;gBACnB,MAAM,EAAE,OAAO,CAAC,MAAM;gBACtB,QAAQ,EAAE,CAAC,CAAC,MAAM;gBAClB,IAAI,EAAE,CAAC,CAAC,IAAI;gBACZ,OAAO,EAAE,CAAC,MAAA,CAAC,CAAC,OAAO,mCAAI,EAAE,CAA4B;gBACrD,cAAc,EAAE,CAAC,CAAC,gBAAgB;aACnC,CAAC,CAAA;SAAA,CAAC,CAAC;QAEJ,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,MAAA,GAAG,CAAC,GAAG,mCAAI,SAAS,EAAE,CAAC;IACrD,CAAC;IAED,KAAK,CAAC,cAAc,CAAC,OAA4C;QAC/D,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,GAAG,IAAI,CAAC,MAAO,CAAC,OAAO,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;QAClD,IAAI,IAAI,EAAE,CAAC;YACT,MAAM,KAAK,GAAG,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;YAClD,IAAI,KAAK,EAAE,CAAC;gBACV,MAAM,IAAI,CAAC,MAAO,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;gBAC1C,OAAO;YACT,CAAC;QACH,CAAC;QACD,sDAAsD;QACtD,MAAM,IAAI,CAAC,MAAO,CAAC,6BAA6B,CAC9C,OAAO,CAAC,MAAM,EACd,OAAO,CAAC,OAAO,EACf,OAAO,CAAC,OAAO,CAChB,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,oBAAoB,CAAC,OAA+C;QACxE,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,GAAG,IAAI,CAAC,MAAO,CAAC,OAAO,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;QAClD,IAAI,CAAC,IAAI;YAAE,OAAO,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC;QACjC,MAAM,MAAM,GAA0C,EAAE,CAAC;QACzD,KAAK,MAAM,GAAG,IAAI,OAAO,CAAC,QAAQ,EAAE,CAAC;YACnC,MAAM,KAAK,GAAG,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC;YACtC,IAAI,KAAK,EAAE,CAAC;gBACV,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,KAAK,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC;YAC1D,CAAC;QACH,CAAC;QACD,OAAO,EAAE,MAAM,EAAE,CAAC;IACpB,CAAC;IAED,4DAA4D;IAE5D,KAAK,CAAC,WAAW,CAAC,OAA6D;QAC7E,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,CAAC,MAAO,CAAC,WAAW,CAAC,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,OAAO,EAAE,SAAS,EAAE;YACzE,MAAM,EAAE,OAAO,CAAC,MAAM;SAChB,CAAC,CAAC;IACZ,CAAC;IAED,KAAK,CAAC,YAAY,CAAC,OAAyD;QAC1E,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,QAAQ,GAAG,IAAI,CAAC,MAAO,CAAC,SAAS,EAAE,CAAC;QAE1C,+EAA+E;QAC/E,MAAM,IAAI,GAAG,IAAI,CAAC,MAAO,CAAC,OAAO,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;QAClD,IAAI,IAAI,IAAI,QAAQ,EAAE,CAAC;YACrB,IAAI,CAAC;gBACH,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,sBAAsB,CACrD,OAAO,CAAC,OAAO,EACf,YAAY,CAAC,UAAU,EACvB,SAAS,CAAC,QAAQ,CACnB,CAAC;gBACF,IAAI,SAAS,EAAE,CAAC;oBACd,MAAM,QAAQ,GAAG,SAAS,CAAC,YAAY,EAAE,CAAC,IAAI,CAC5C,CAAC,CAAC,EAAE,EAAE,eAAC,OAAA,CAAC,CAAC,SAAS,EAAE,KAAK,QAAQ,IAAI,CAAA,MAAA,MAAA,CAAC,CAAC,UAAU,EAAE,0CAAG,cAAc,CAAC,0CAAE,GAAG,MAAK,OAAO,CAAC,GAAG,CAAA,EAAA,CAC3F,CAAC;oBACF,IAAI,QAAQ,EAAE,CAAC;wBACb,MAAM,UAAU,GAAG,QAAQ,CAAC,KAAK,EAAE,CAAC;wBACpC,IAAI,UAAU,EAAE,CAAC;4BACf,MAAM,IAAI,CAAC,MAAO,CAAC,WAAW,CAAC,OAAO,CAAC,MAAM,EAAE,UAAU,CAAC,CAAC;4BAC3D,OAAO,EAAE,OAAO,EAAE,UAAU,EAAE,CAAC;wBACjC,CAAC;oBACH,CAAC;gBACH,CAAC;YACH,CAAC;YAAC,WAAM,CAAC;gBACP,uBAAuB;YACzB,CAAC;QACH,CAAC;QAED,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,MAAO,CAAC,SAAS,CAAC,OAAO,CAAC,MAAM,EAAE,SAAS,CAAC,QAAQ,EAAE;YAC3E,cAAc,EAAE;gBACd,QAAQ,EAAE,YAAY,CAAC,UAAU;gBACjC,QAAQ,EAAE,OAAO,CAAC,OAAO;gBACzB,GAAG,EAAE,OAAO,CAAC,GAAG;aACjB;SACF,CAAC,CAAC;QACH,OAAO,EAAE,OAAO,EAAE,GAAG,CAAC,QAAQ,EAAE,CAAC;IACnC,CAAC;IAED,0DAA0D;IAE1D,KAAK,CAAC,WAAW,CAAC,OAAyC;QACzD,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,CAAC,MAAO,CAAC,WAAW,CAAC,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,IAAI,CAAC,CAAC;IAC/D,CAAC;IAED,KAAK,CAAC,YAAY,CAAC,OAA0C;QAC3D,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,CAAC,MAAO,CAAC,YAAY,CAAC,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC;IACjE,CAAC;IAED,KAAK,CAAC,aAAa,CAAC,OAA2C;QAC7D,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,CAAC,MAAO,CAAC,cAAc,CAAC,OAAO,CAAC,MAAM,EAAE,eAAsB,EAAE,EAAE,GAAG,EAAE,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC;IACrG,CAAC;IAED,KAAK,CAAC,UAAU,CAAC,OAA2C;QAC1D,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,CAAC,MAAO,CAAC,MAAM,CAAC,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;IAC5D,CAAC;IAED,KAAK,CAAC,QAAQ,CAAC,OAA4D;QACzE,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,CAAC,MAAO,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;IAC1E,CAAC;IAED,KAAK,CAAC,OAAO,CAAC,OAA4D;QACxE,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,CAAC,MAAO,CAAC,GAAG,CAAC,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;IACzE,CAAC;IAED,KAAK,CAAC,SAAS,CAAC,OAA2C;QACzD,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,CAAC,MAAO,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;IAC3D,CAAC;IAED,0DAA0D;IAE1D,KAAK,CAAC,UAAU,CAAC,OAAgE;;QAC/E,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,CAAC,MAAO,CAAC,UAAU,CAAC,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,QAAQ,EAAE,MAAA,OAAO,CAAC,OAAO,mCAAI,KAAK,CAAC,CAAC;IAC5F,CAAC;IAED,0DAA0D;IAE1D,KAAK,CAAC,WAAW,CAAC,OAA2B;QAC3C,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,uDAAuD;QACvD,MAAM,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;QACrD,MAAM,OAAO,GAAG,IAAI,CAAC,MAAO,CAAC,gBAAgB,EAAE,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QACnE,MAAM,WAAW,GAAG,IAAI,CAAC,MAAO,CAAC,cAAc,EAAE,CAAC;QAClD,MAAM,OAAO,GAAG,GAAG,OAAO,qCAAqC,OAAO,iBAAiB,WAAW,EAAE,CAAC;QACrG,OAAO,EAAE,OAAO,EAAE,CAAC;IACrB,CAAC;IAED,KAAK,CAAC,eAAe,CAAC,OAA4B;;QAChD,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;QACrD,MAAM,OAAO,GAAG,IAAI,CAAC,MAAO,CAAC,gBAAgB,EAAE,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QACnE,MAAM,WAAW,GAAG,IAAI,CAAC,MAAO,CAAC,cAAc,EAAE,CAAC;QAClD,MAAM,MAAM,GAAG,MAAA,OAAO,CAAC,MAAM,mCAAI,OAAO,CAAC;QACzC,MAAM,OAAO,GAAG,GAAG,OAAO,sCAAsC,OAAO,UAAU,OAAO,CAAC,KAAK,WAAW,OAAO,CAAC,MAAM,WAAW,MAAM,iBAAiB,WAAW,EAAE,CAAC;QACvK,OAAO,EAAE,OAAO,EAAE,CAAC;IACrB,CAAC;IAED,KAAK,CAAC,aAAa,CAAC,OAA6B;QAC/C,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QAC9C,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QACnC,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,MAAO,CAAC,aAAa,CAAC,IAAI,EAAE;YACjD,IAAI,EAAE,OAAO,CAAC,QAAQ;YACtB,IAAI,EAAE,OAAO,CAAC,QAAQ;SACvB,CAAC,CAAC;QACH,OAAO,EAAE,UAAU,EAAE,GAAG,CAAC,WAAW,EAAE,CAAC;IACzC,CAAC;IAED,0DAA0D;IAE1D,KAAK,CAAC,WAAW,CAAC,OAA+E;QAC/F,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,CAAC,MAAO,CAAC,WAAW,CAAC;YAC7B,QAAQ,EAAE,OAAO,CAAC,QAAQ;YAC1B,UAAU,EAAE,OAAO,CAAC,SAAS;SAC9B,CAAC,CAAC;IACL,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,OAA2B;;QAC3C,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,GAAG,IAAI,CAAC,MAAO,CAAC,OAAO,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;QAClD,OAAO;YACL,QAAQ,EAAE,MAAC,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,QAAqC,mCAAI,SAAS;YACnE,SAAS,EAAE,MAAA,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,iBAAiB,mCAAI,SAAS;YAC/C,aAAa,EAAE,MAAA,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,aAAa,mCAAI,SAAS;SAChD,CAAC;IACJ,CAAC;IAED,0DAA0D;IAE1D,KAAK,CAAC,UAAU;;QACd,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,MAAO,CAAC,UAAU,EAAE,CAAC;QAC5C,MAAM,MAAM,GAAG,IAAI,CAAC,MAAO,CAAC,SAAS,EAAE,CAAC;QACxC,MAAM,QAAQ,GAAG,MAAA,IAAI,CAAC,MAAO,CAAC,SAAS,EAAE,mCAAI,EAAE,CAAC;QAChD,MAAM,OAAO,GAAiB,MAAM,OAAO,CAAC,GAAG,CAC7C,CAAC,MAAA,GAAG,CAAC,OAAO,mCAAI,EAAE,CAAC,CAAC,GAAG,CAAC,KAAK,EAAE,CAAM,EAAE,EAAE;;YACvC,IAAI,sBAA2C,CAAC;YAChD,IAAI,MAAM,EAAE,CAAC;gBACX,IAAI,CAAC;oBACH,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,2BAA2B,CAAC,QAAQ,EAAE,CAAC,CAAC,SAAS,CAAC,CAAC;oBAC/E,sBAAsB,GAAG,MAAA,MAAM,aAAN,MAAM,uBAAN,MAAM,CAAE,oBAAoB,mCAAI,KAAK,CAAC;gBACjE,CAAC;gBAAC,WAAM,CAAC;oBACP,mCAAmC;gBACrC,CAAC;YACH,CAAC;YACD,OAAO;gBACL,QAAQ,EAAE,CAAC,CAAC,SAAS;gBACrB,WAAW,EAAE,MAAA,CAAC,CAAC,YAAY,mCAAI,SAAS;gBACxC,UAAU,EAAE,MAAA,CAAC,CAAC,YAAY,mCAAI,SAAS;gBACvC,UAAU,EAAE,MAAA,CAAC,CAAC,YAAY,mCAAI,SAAS;gBACvC,sBAAsB;aACvB,CAAC;QACJ,CAAC,CAAC,CACH,CAAC;QACF,OAAO,EAAE,OAAO,EAAE,CAAC;IACrB,CAAC;IAED,KAAK,CAAC,YAAY,CAAC,OAA6D;QAC9E,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,CAAC,MAAO,CAAC,YAAY,CAAC,OAAO,CAAC,QAAQ,EAAE,OAAO,CAAC,IAAW,CAAC,CAAC;IACzE,CAAC;IAED,yDAAyD;IAEzD,KAAK,CAAC,SAAS,CAAC,OAAsB;;QACpC,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,CAAC,MAAO,CAAC,SAAS,CAAC;YAC3B,OAAO,EAAE,OAAO,CAAC,OAAO;YACxB,IAAI,EAAE,MAAA,OAAO,CAAC,IAAI,mCAAI,SAAS;YAC/B,MAAM,EAAE,OAAO,CAAC,KAAK;YACrB,gBAAgB,EAAE,OAAO,CAAC,cAAc;YACxC,mBAAmB,EAAE,OAAO,CAAC,iBAAiB;YAC9C,IAAI,EAAE,OAAO,CAAC,IAAI;YAClB,IAAI,EAAE,OAAO,CAAC,IAAI;SACZ,CAAC,CAAC;IACZ,CAAC;IAED,2DAA2D;IAE3D,KAAK,CAAC,gBAAgB;;QACpB,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,UAAU,GAAG,EAAE,oBAAoB,EAAE,eAAe,EAAE,CAAC;QAE7D,IAAI,CAAC;YACH,MAAM,IAAI,CAAC,MAAO,CAAC,cAAc,CAAC,UAAU,CAAC,CAAC;QAChD,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YAChB,oEAAoE;YACpE,8DAA8D;YAC9D,kEAAkE;YAClE,IAAI,MAAA,CAAC,aAAD,CAAC,uBAAD,CAAC,CAAE,OAAO,0CAAE,QAAQ,CAAC,oCAAoC,CAAC,EAAE,CAAC;gBAC/D,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;gBAC/B,MAAM,IAAI,CAAC,MAAO,CAAC,cAAc,CAAC,UAAU,CAAC,CAAC;YAChD,CAAC;iBAAM,CAAC;gBACN,MAAM,CAAC,CAAC;YACV,CAAC;QACH,CAAC;QAED,sEAAsE;QACtE,yEAAyE;QACzE,sEAAsE;QACtE,MAAM,MAAM,GAAG,IAAI,CAAC,MAAO,CAAC,SAAS,EAAS,CAAC;QAC/C,IAAI,MAAA,MAAM,aAAN,MAAM,uBAAN,MAAM,CAAE,uBAAuB,0CAAE,yBAAyB,EAAE,CAAC;YAC/D,MAAM,MAAM,CAAC,uBAAuB,CAAC,yBAAyB,EAAE,CAAC;QACnE,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,iBAAiB;QAC7B,IAAI,OAAO,SAAS,KAAK,WAAW;YAAE,OAAO;QAC7C,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,MAAM,SAAS,CAAC,SAAS,EAAE,CAAC;YACxC,MAAM,OAAO,CAAC,GAAG,CACf,GAAG;iBACA,MAAM,CAAC,CAAC,EAAE,EAAE,EAAE,WAAC,OAAA,MAAA,EAAE,CAAC,IAAI,0CAAE,UAAU,CAAC,eAAe,CAAC,CAAA,EAAA,CAAC;iBACpD,GAAG,CACF,CAAC,EAAE,EAAE,EAAE,CACL,IAAI,OAAO,CAAO,CAAC,OAAO,EAAE,EAAE;gBAC5B,MAAM,GAAG,GAAG,SAAS,CAAC,cAAc,CAAC,EAAE,CAAC,IAAK,CAAC,CAAC;gBAC/C,GAAG,CAAC,SAAS,GAAG,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC;gBAChC,GAAG,CAAC,OAAO,GAAG,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC;YAChC,CAAC,CAAC,CACL,CACJ,CAAC;QACJ,CAAC;QAAC,WAAM,CAAC;YACP,0DAA0D;QAC5D,CAAC;IACH,CAAC;IAED,KAAK,CAAC,mBAAmB;QACvB,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,MAAM,GAAG,IAAI,CAAC,MAAO,CAAC,SAAS,EAAE,CAAC;QACxC,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,OAAO;gBACL,mBAAmB,EAAE,KAAK;gBAC1B,kBAAkB,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,cAAc,EAAE,KAAK,EAAE,cAAc,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE;gBACtG,kBAAkB,EAAE,KAAK;gBACzB,oBAAoB,EAAE,KAAK;aAC5B,CAAC;QACJ,CAAC;QAED,MAAM,OAAO,GAAG,MAAM,MAAM,CAAC,mBAAmB,EAAE,CAAC;QACnD,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,qBAAqB,EAAE,CAAC;QACtD,MAAM,aAAa,GAAG,MAAM,MAAM,CAAC,6BAA6B,EAAE,CAAC;QAEnE,2EAA2E;QAC3E,2EAA2E;QAC3E,0DAA0D;QAC1D,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,sBAAsB,EAAE,CAAC;QACvD,MAAM,QAAQ,GAAG,QAAQ,CAAC,YAAY,KAAK,IAAI,CAAC;QAEhD,OAAO;YACL,mBAAmB,EAAE,OAAO;YAC5B,kBAAkB,EAAE;gBAClB,SAAS,EAAE,QAAQ,CAAC,kBAAkB;gBACtC,cAAc,EAAE,QAAQ,CAAC,wBAAwB,CAAC,cAAc;gBAChE,cAAc,EAAE,QAAQ,CAAC,wBAAwB,CAAC,cAAc;gBAChE,OAAO,EAAE,OAAO;aACjB;YACD,kBAAkB,EAAE,aAAa,KAAK,IAAI;YAC1C,gBAAgB,EAAE,aAAa,aAAb,aAAa,cAAb,aAAa,GAAI,SAAS;YAC5C,oBAAoB,EAAE,QAAQ;SAC/B,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,qBAAqB;QACzB,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,YAAY,EAAE,CAAC;QACzC,MAAM,MAAM,CAAC,qBAAqB,CAAC;YACjC,oBAAoB,EAAE,IAAI;YAC1B,2BAA2B,EAAE,KAAK,EAAE,WAAW,EAAE,EAAE;;gBACjD,sEAAsE;gBACtE,IAAI,CAAC;oBACH,MAAM,WAAW,CAAC,EAAE,IAAI,EAAE,eAAe,EAAE,CAAC,CAAC;gBAC/C,CAAC;gBAAC,OAAO,CAAM,EAAE,CAAC;oBAChB,MAAM,OAAO,GAAG,MAAA,CAAC,aAAD,CAAC,uBAAD,CAAC,CAAE,IAAI,0CAAE,OAAO,CAAC;oBACjC,IAAI,OAAO,EAAE,CAAC;wBACZ,MAAM,WAAW,CAAC,EAAE,IAAI,EAAE,eAAe,EAAE,OAAO,EAAE,CAAC,CAAC;oBACxD,CAAC;yBAAM,CAAC;wBACN,MAAM,CAAC,CAAC;oBACV,CAAC;gBACH,CAAC;YACH,CAAC;SACF,CAAC,CAAC;IACL,CAAC;IAED,KAAK,CAAC,cAAc;QAClB,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,YAAY,EAAE,CAAC;QACzC,MAAM,MAAM,CAAC,cAAc,EAAE,CAAC;QAC9B,MAAM,OAAO,GAAG,MAAM,MAAM,CAAC,6BAA6B,EAAE,CAAC;QAC7D,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,OAAO,EAAE,OAAO,aAAP,OAAO,cAAP,OAAO,GAAI,SAAS,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC;IACxE,CAAC;IAED,KAAK,CAAC,kBAAkB;QACtB,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,MAAM,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;QACpC,MAAM,OAAO,GAAG,MAAM,MAAM,CAAC,6BAA6B,EAAE,CAAC;QAC7D,OAAO;YACL,MAAM,EAAE,OAAO,KAAK,IAAI;YACxB,OAAO,EAAE,OAAO,aAAP,OAAO,cAAP,OAAO,GAAI,SAAS;YAC7B,OAAO,EAAE,OAAO,KAAK,IAAI;SAC1B,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,gBAAgB,CAAC,QAAmC;;QACxD,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,MAAM,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;QAEpC,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,gBAAgB,EAAE,CAAC;QAC/C,OAAO,EAAE,YAAY,EAAE,MAAA,MAAM,aAAN,MAAM,uBAAN,MAAM,CAAE,QAAQ,mCAAI,CAAC,EAAE,CAAC;IACjD,CAAC;IAED,KAAK,CAAC,aAAa,CAAC,OAGnB;;QACC,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,YAAY,EAAE,CAAC;QAEzC,MAAM,OAAO,GAAG,MAAM,MAAM,CAAC,+BAA+B,CAAC,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,UAAU,CAAC,CAAC;QAClF,iEAAiE;QACjE,uEAAuE;QACvE,oCAAoC;QACpC,IAAI,CAAC,gBAAgB,GAAG,OAAO,CAAC,UAAU,CAAC;QAC3C,IAAI,CAAC,kBAAkB,GAAG,SAAS,CAAC;QAEpC,gEAAgE;QAChE,sDAAsD;QACtD,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,UAAU,EAAE,CAAC;YACxB,IAAI,CAAC,kBAAkB,GAAG,OAAO,CAAC,UAAU,CAAC;QAC/C,CAAC;QACD,2EAA2E;QAC3E,gEAAgE;QAChE,wEAAwE;QACxE,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,kBAAkB,EAAE,CAAC;YAChC,IAAI,CAAC,kBAAkB,GAAG,OAAO,CAAC,kBAAkB,CAAC;QACvD,CAAC;QAED,IAAI,CAAC;YACH,MAAM,gBAAgB,GAAG,MAAM,CAAC,sBAAsB,CAAC;gBACrD,sBAAsB,EAAE,KAAK,IAAI,EAAE,CAAC,OAAO;gBAC3C,qBAAqB,EAAE,IAAI;gBAC3B,iBAAiB,EAAE,IAAI;aACxB,CAAC,CAAC;YAEH,oEAAoE;YACpE,MAAM,cAAc,GAAG,IAAI,OAAO,CAAQ,CAAC,CAAC,EAAE,MAAM,EAAE,EAAE;gBACtD,UAAU,CACR,GAAG,EAAE,CAAC,MAAM,CAAC,IAAI,KAAK,CAAC,4EAA4E,CAAC,CAAC,EACrG,KAAM,CACP,CAAC;YACJ,CAAC,CAAC,CAAC;YAEH,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC,gBAAgB,EAAE,cAAc,CAAC,CAAC,CAAC;QACzD,CAAC;gBAAS,CAAC;YACT,iFAAiF;YACjF,IAAI,CAAC,kBAAkB,GAAG,SAAS,CAAC;YACpC,IAAI,CAAC,kBAAkB,GAAG,SAAS,CAAC;QACtC,CAAC;QAED,OAAO,EAAE,WAAW,EAAE,MAAA,OAAO,CAAC,iBAAiB,mCAAI,EAAE,EAAE,CAAC;IAC1D,CAAC;IAED,KAAK,CAAC,iBAAiB;QACrB,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,YAAY,EAAE,CAAC;QACzC,MAAM,KAAK,GAAG,MAAM,MAAM,CAAC,oBAAoB,EAAE,CAAC;QAClD,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;IAC5B,CAAC;IAED,KAAK,CAAC,eAAe,CAAC,OAAsD;QAC1E,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,YAAY,EAAE,CAAC;QAEzC,uCAAuC;QACvC,IAAI,OAAO,CAAC,WAAW,EAAE,CAAC;YACxB,IAAI,CAAC,gBAAgB,GAAG,iBAAiB,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;QACjE,CAAC;aAAM,IAAI,OAAO,CAAC,UAAU,EAAE,CAAC;YAC9B,kEAAkE;YAClE,qEAAqE;YACrE,IAAI,CAAC,kBAAkB,GAAG,OAAO,CAAC,UAAU,CAAC;YAC7C,IAAI,CAAC,gBAAgB,GAAG,SAAS,CAAC,CAAC,0BAA0B;QAC/D,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,KAAK,CAAC,mDAAmD,CAAC,CAAC;QACvE,CAAC;QAED,iFAAiF;QACjF,kDAAkD;QAClD,IAAI,CAAC;YACH,MAAM,MAAM,CAAC,4CAA4C,EAAE,CAAC;QAC9D,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,MAAM,GAAG,GAAG,CAAC,YAAY,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;YACvD,IAAI,GAAG,CAAC,QAAQ,CAAC,+BAA+B,CAAC,EAAE,CAAC;gBAClD,sEAAsE;gBACtE,0EAA0E;gBAC1E,sEAAsE;gBACtE,4EAA4E;gBAC5E,8DAA8D;gBAC9D,8DAA8D;gBAC9D,MAAM,MAAM,CAAC,sBAAsB,CAAC;oBAClC,iBAAiB,EAAE,IAAI;iBACxB,CAAC,CAAC;gBACH,MAAM,MAAM,CAAC,uBAAuB,EAAE,CAAC;gBACvC,OAAO;YACT,CAAC;YACD,0CAA0C;YAC1C,IAAI,CAAC,gBAAgB,GAAG,SAAS,CAAC;YAClC,IAAI,CAAC,kBAAkB,GAAG,SAAS,CAAC;YACpC,IAAI,CAAC,kBAAkB,GAAG,SAAS,CAAC;YACpC,MAAM,CAAC,CAAC;QACV,CAAC;QAED,4EAA4E;QAC5E,MAAM,MAAM,CAAC,uBAAuB,EAAE,CAAC;IACzC,CAAC;IAED,KAAK,CAAC,gBAAgB,CAAC,OAAiC;QACtD,OAAO,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;IACrC,CAAC;IAED,KAAK,CAAC,cAAc,CAAC,OAA+B;QAClD,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,MAAM,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;QACpC,MAAM,IAAI,GAAG,MAAM,MAAM,CAAC,oBAAoB,EAAE,CAAC;QACjD,2EAA2E;QAC3E,yDAAyD;QACzD,KAAK,OAAO,CAAC,UAAU,CAAC,CAAC,kEAAkE;QAC3F,OAAO,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC;IACxB,CAAC;IAED,KAAK,CAAC,cAAc,CAAC,OAA6C;QAChE,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,MAAM,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;QACpC,KAAK,OAAO,CAAC,UAAU,CAAC,CAAC,+DAA+D;QACxF,MAAM,MAAM,CAAC,oBAAoB,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QAChD,OAAO,EAAE,YAAY,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,gDAAgD;IAC/E,CAAC;IAED,yDAAyD;IAEjD,aAAa;QACnB,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;YACjB,MAAM,IAAI,KAAK,CAAC,wDAAwD,CAAC,CAAC;QAC5E,CAAC;IACH,CAAC;IAEO,aAAa;QACnB,MAAM,MAAM,GAAG,IAAI,CAAC,MAAO,CAAC,SAAS,EAAE,CAAC;QACxC,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,MAAM,IAAI,KAAK,CAAC,wDAAwD,CAAC,CAAC;QAC5E,CAAC;QACD,OAAO,MAAM,CAAC;IAChB,CAAC;IAEO,KAAK,CAAC,YAAY;QACxB,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,IAAI,CAAC,IAAI,CAAC,MAAO,CAAC,SAAS,EAAE,EAAE,CAAC;YAC9B,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAChC,CAAC;QACD,OAAO,IAAI,CAAC,aAAa,EAAE,CAAC;IAC9B,CAAC;IAEO,cAAc,CAAC,OAAoB;QACzC,YAAY,CAAC,OAAO,CAAC,WAAW,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC;IAC7D,CAAC;IAEO,cAAc,CAAC,KAAqB,EAAE,cAAuB;;QACnE,MAAM,MAAM,GAAG,MAAA,MAAA,KAAK,CAAC,SAAS,EAAE,mCAAI,cAAc,mCAAI,EAAE,CAAC;QAEzD,2CAA2C;QAC3C,IAAI,KAAK,CAAC,UAAU,EAAE,EAAE,CAAC;YACvB,OAAO;gBACL,OAAO,EAAE,MAAA,KAAK,CAAC,KAAK,EAAE,mCAAI,EAAE;gBAC5B,MAAM;gBACN,QAAQ,EAAE,MAAA,KAAK,CAAC,SAAS,EAAE,mCAAI,EAAE;gBACjC,IAAI,EAAE,kBAAkB;gBACxB,OAAO,EAAE,EAAE,IAAI,EAAE,iBAAiB,EAAE;gBACpC,cAAc,EAAE,KAAK,CAAC,KAAK,EAAE;aAC9B,CAAC;QACJ,CAAC;QAED,MAAM,OAAO,GAAG,kBAAK,CAAC,MAAA,KAAK,CAAC,UAAU,EAAE,mCAAI,EAAE,CAAC,CAA6B,CAAC;QAE7E,mEAAmE;QACnE,MAAM,OAAO,GAAG,KAAK,CAAC,KAAK,EAAE,CAAC;QAC9B,IAAI,OAAO,IAAI,MAAM,EAAE,CAAC;YACtB,MAAM,IAAI,GAAG,MAAA,IAAI,CAAC,MAAM,0CAAE,OAAO,CAAC,MAAM,CAAC,CAAC;YAC1C,IAAI,IAAI,EAAE,CAAC;gBACT,IAAI,CAAC;oBACH,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,sBAAsB,CACrD,OAAO,EACP,YAAY,CAAC,UAAU,EACvB,SAAS,CAAC,QAAQ,CACnB,CAAC;oBACF,IAAI,SAAS,EAAE,CAAC;wBACd,MAAM,MAAM,GAAG,SAAS,CAAC,yBAAyB,EAAE,CAAC;wBACrD,IAAI,MAAM,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;4BAChC,OAAO,CAAC,SAAS,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,MAAM,CAAC,EAAE,EAAE,CAAC,CAAC;gCACjD,GAAG;gCACH,KAAK,EAAE,MAAM,CAAC,IAAI;gCAClB,OAAO,EAAE,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,SAAS,EAAE,CAAC;6BACtD,CAAC,CAAC,CAAC;wBACN,CAAC;oBACH,CAAC;gBACH,CAAC;gBAAC,WAAM,CAAC;oBACP,iCAAiC;gBACnC,CAAC;YACH,CAAC;QACH,CAAC;QAED,iCAAiC;QACjC,IAAI,MAA6B,CAAC;QAClC,MAAM,MAAM,GAAa,EAAE,CAAC;QAC5B,MAAM,QAAQ,GAAG,MAAA,IAAI,CAAC,MAAM,0CAAE,SAAS,EAAE,CAAC;QAC1C,MAAM,MAAM,GAAG,KAAK,CAAC,SAAS,EAAE,CAAC;QACjC,MAAM,IAAI,GAAG,OAAO,IAAI,MAAM,CAAC,CAAC,CAAC,MAAA,IAAI,CAAC,MAAM,0CAAE,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QAE1E,IAAI,MAAM,KAAK,QAAQ,IAAI,OAAO,EAAE,CAAC;YACnC,sCAAsC;YACtC,MAAM,SAAS,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,gDAAgD;YAChF,IAAI,SAAS,KAAK,SAAS,IAAI,SAAS,KAAK,YAAY,IAAI,SAAS,KAAK,QAAQ,EAAE,CAAC;gBACpF,MAAM,GAAG,SAAS,CAAC;YACrB,CAAC;iBAAM,CAAC;gBACN,sDAAsD;gBACtD,IAAI,IAAI,EAAE,CAAC;oBACT,IAAI,CAAC;wBACH,MAAM,OAAO,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;wBACxC,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;4BAC7B,IAAI,MAAM,CAAC,MAAM,KAAK,QAAQ;gCAAE,SAAS;4BACzC,IAAI,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,CAAC;gCAClD,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;4BAC7B,CAAC;wBACH,CAAC;oBACH,CAAC;oBAAC,WAAM,CAAC;wBACP,gBAAgB;oBAClB,CAAC;gBACH,CAAC;gBACD,MAAM,GAAG,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC;YAC/C,CAAC;QACH,CAAC;aAAM,IAAI,OAAO,IAAI,IAAI,EAAE,CAAC;YAC3B,4CAA4C;YAC5C,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;gBACxC,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;oBAC7B,IAAI,MAAM,CAAC,MAAM,KAAK,MAAM;wBAAE,SAAS;oBACvC,IAAI,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,CAAC;wBAClD,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;oBAC7B,CAAC;gBACH,CAAC;YACH,CAAC;YAAC,WAAM,CAAC;gBACP,SAAS;YACX,CAAC;QACH,CAAC;QAED,oFAAoF;QACpF,MAAM,YAAY,GAAG,MAAA,KAAK,CAAC,WAAW,qDAAI,CAAC;QAC3C,MAAM,QAAQ,GAAG,YAAY,IAAI,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,MAAM,GAAG,CAAC;YACnE,CAAC,CAAE,YAAwC;YAC3C,CAAC,CAAC,SAAS,CAAC;QAEd,OAAO;YACL,OAAO,EAAE,OAAO,aAAP,OAAO,cAAP,OAAO,GAAI,EAAE;YACtB,MAAM;YACN,QAAQ,EAAE,MAAM,aAAN,MAAM,cAAN,MAAM,GAAI,EAAE;YACtB,IAAI,EAAE,KAAK,CAAC,OAAO,EAAE;YACrB,OAAO;YACP,cAAc,EAAE,KAAK,CAAC,KAAK,EAAE;YAC7B,MAAM;YACN,MAAM,EAAE,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,SAAS;YAC9C,QAAQ;SACT,CAAC;IACJ,CAAC;IAEO,aAAa,CAAC,IAAU;;QAC9B,kEAAkE;QAClE,IAAI,QAAQ,GAAG,KAAK,CAAC;QACrB,IAAI,CAAC;YACH,MAAM,WAAW,GAAG,MAAA,IAAI,CAAC,MAAM,0CAAE,cAAc,CAAC,UAAiB,CAAC,CAAC;YACnE,IAAI,WAAW,EAAE,CAAC;gBAChB,MAAM,aAAa,GAAG,WAAW,CAAC,UAAU,EAA8B,CAAC;gBAC3E,KAAK,MAAM,OAAO,IAAI,MAAM,CAAC,MAAM,CAAC,aAAa,CAAC,EAAE,CAAC;oBACnD,IAAI,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC;wBAClC,QAAQ,GAAG,IAAI,CAAC;wBAChB,MAAM;oBACR,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;QAAC,WAAM,CAAC;YACP,SAAS;QACX,CAAC;QAED,iBAAiB;QACjB,IAAI,SAA6B,CAAC;QAClC,MAAM,WAAW,GAAG,IAAI,CAAC,YAAY,CAAC,cAAc,CAAC,eAAe,EAAE,EAAE,CAAC,CAAC;QAC1E,IAAI,WAAW,EAAE,CAAC;YAChB,MAAM,MAAM,GAAG,MAAA,WAAW,CAAC,UAAU,EAAE,0CAAE,GAAyB,CAAC;YACnE,IAAI,MAAM,EAAE,CAAC;gBACX,SAAS,GAAG,MAAM,CAAC;YACrB,CAAC;QACH,CAAC;QAED,OAAO;YACL,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,KAAK,EAAE,MAAA,MAAA,MAAA,IAAI,CAAC,YAAY,CAAC,cAAc,CAAC,cAAc,EAAE,EAAE,CAAC,0CAAE,UAAU,EAAE,0CAAE,KAAK,mCAAI,SAAS;YAC7F,WAAW,EAAE,IAAI,CAAC,oBAAoB,EAAE;YACxC,WAAW,EAAE,IAAI,CAAC,uBAAuB,EAAE;YAC3C,WAAW,EAAE,MAAA,IAAI,CAAC,0BAA0B,EAAE,mCAAI,CAAC;YACnD,WAAW,EAAE,IAAI,CAAC,sBAAsB,EAAE,IAAI,SAAS;YACvD,UAAU,EAAE,IAAI,CAAC,eAAe,EAA+B;YAC/D,SAAS;YACT,QAAQ;SACT,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,OAGjB;;QACC,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,MAAM,IAAI,GAAG,MAAM,IAAI,CAAC,MAAO,CAAC,mBAAmB,CAAC;YAClD,IAAI,EAAE,OAAO,CAAC,UAAU;YACxB,KAAK,EAAE,MAAA,OAAO,CAAC,KAAK,mCAAI,EAAE;SAC3B,CAAC,CAAC;QACH,OAAO;YACL,OAAO,EAAE,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAkE,EAAE,EAAE,CAAC,CAAC;gBACjG,MAAM,EAAE,CAAC,CAAC,OAAO;gBACjB,WAAW,EAAE,CAAC,CAAC,YAAY;gBAC3B,SAAS,EAAE,CAAC,CAAC,UAAU;aACxB,CAAC,CAAC;YACH,OAAO,EAAE,IAAI,CAAC,OAAO;SACtB,CAAC;IACJ,CAAC;IAEO,YAAY,CAAC,KAAoB;QACvC,QAAQ,KAAK,EAAE,CAAC;YACd,KAAK,UAAU,CAAC;YAChB,KAAK,SAAS,CAAC;YACf,KAAK,SAAS,CAAC;YACf,KAAK,cAAc;gBACjB,OAAO,SAAS,CAAC;YACnB,KAAK,OAAO;gBACV,OAAO,OAAO,CAAC;YACjB,KAAK,SAAS;gBACZ,OAAO,SAAS,CAAC;YACnB;gBACE,OAAO,SAAS,CAAC;QACrB,CAAC;IACH,CAAC;CACF","sourcesContent":["import { WebPlugin } from '@capacitor/core';\nimport {\n createClient,\n ClientEvent,\n RoomEvent,\n RoomMemberEvent,\n Direction,\n MsgType,\n EventType,\n RelationType,\n UserEvent,\n} from 'matrix-js-sdk';\nimport { decodeRecoveryKey } from 'matrix-js-sdk/lib/crypto-api/recovery-key';\nimport { deriveRecoveryKeyFromPassphrase } from 'matrix-js-sdk/lib/crypto-api/key-passphrase';\nimport type { MatrixClient, Room, MatrixEvent as SdkMatrixEvent, User } from 'matrix-js-sdk';\n\nimport type {\n MatrixPlugin,\n LoginOptions,\n LoginWithTokenOptions,\n SessionInfo,\n SendMessageOptions,\n EditMessageOptions,\n SendReplyOptions,\n UploadContentOptions,\n UploadContentResult,\n ThumbnailUrlOptions,\n PusherOptions,\n DeviceInfo,\n MatrixEvent,\n RoomSummary,\n RoomMember,\n SyncState,\n EncryptionStatus,\n KeyBackupStatus,\n RecoveryKeyInfo,\n PresenceInfo,\n UserProfile,\n} from './definitions';\n\nconst SESSION_KEY = 'matrix_session';\n\nexport class MatrixWeb extends WebPlugin implements MatrixPlugin {\n private client?: MatrixClient;\n private secretStorageKey?: Uint8Array<ArrayBuffer>;\n private secretStorageKeyId?: string; // key ID the cached bytes belong to\n private recoveryPassphrase?: string;\n private fallbackPassphrase?: string; // old passphrase for SSSS migration in setupRecovery\n\n private readonly _cryptoCallbacks = {\n getSecretStorageKey: async (\n opts: { keys: Record<string, unknown> },\n ): Promise<[string, Uint8Array<ArrayBuffer>] | null> => {\n const keyId = Object.keys(opts.keys)[0];\n if (!keyId) return null;\n\n // Exact match: only return the cached raw key for the key ID it was cached under.\n // (bootstrapSecretStorage uses createSecretStorageKey for the new key, so this\n // path is only reached for an already-established key — e.g. after recoverAndSetup.)\n if (this.secretStorageKey && this.secretStorageKeyId === keyId) {\n return [keyId, this.secretStorageKey];\n }\n\n // Derive from the current passphrase (set during recoverAndSetup)\n if (this.recoveryPassphrase) {\n const keyInfo = opts.keys[keyId] as {\n passphrase?: { salt: string; iterations: number; bits?: number };\n };\n if (keyInfo?.passphrase) {\n const derived = await deriveRecoveryKeyFromPassphrase(\n this.recoveryPassphrase,\n keyInfo.passphrase.salt,\n keyInfo.passphrase.iterations,\n keyInfo.passphrase.bits ?? 256,\n );\n // Cache with the correct key ID for subsequent calls\n this.secretStorageKey = derived;\n this.secretStorageKeyId = keyId;\n return [keyId, derived];\n }\n }\n\n // Fallback: derive from the OLD passphrase when bootstrapSecretStorage is\n // migrating existing cross-signing / backup secrets into a new SSSS.\n if (this.fallbackPassphrase) {\n const keyInfo = opts.keys[keyId] as {\n passphrase?: { salt: string; iterations: number; bits?: number };\n };\n if (keyInfo?.passphrase) {\n const derived = await deriveRecoveryKeyFromPassphrase(\n this.fallbackPassphrase,\n keyInfo.passphrase.salt,\n keyInfo.passphrase.iterations,\n keyInfo.passphrase.bits ?? 256,\n );\n return [keyId, derived];\n }\n }\n\n return null;\n },\n cacheSecretStorageKey: (\n keyId: string,\n _keyInfo: unknown,\n key: Uint8Array<ArrayBuffer>,\n ): void => {\n this.secretStorageKey = key;\n this.secretStorageKeyId = keyId;\n },\n };\n\n // ── Auth ──────────────────────────────────────────────\n\n async login(options: LoginOptions): Promise<SessionInfo> {\n const tmpClient = createClient({ baseUrl: options.homeserverUrl });\n const res = await tmpClient.loginWithPassword(options.userId, options.password);\n\n this.client = createClient({\n baseUrl: options.homeserverUrl,\n accessToken: res.access_token,\n userId: res.user_id,\n deviceId: res.device_id,\n cryptoCallbacks: this._cryptoCallbacks,\n });\n\n const session: SessionInfo = {\n accessToken: res.access_token,\n userId: res.user_id,\n deviceId: res.device_id,\n homeserverUrl: options.homeserverUrl,\n };\n\n this.persistSession(session);\n return session;\n }\n\n async loginWithToken(options: LoginWithTokenOptions): Promise<SessionInfo> {\n // Stop any previously running client to avoid parallel instances that\n // would deadlock on the shared IndexedDB crypto store.\n if (this.client) {\n this.client.stopClient();\n this.client = undefined;\n }\n\n this.client = createClient({\n baseUrl: options.homeserverUrl,\n accessToken: options.accessToken,\n userId: options.userId,\n deviceId: options.deviceId,\n cryptoCallbacks: this._cryptoCallbacks,\n });\n\n const session: SessionInfo = {\n accessToken: options.accessToken,\n userId: options.userId,\n deviceId: options.deviceId,\n homeserverUrl: options.homeserverUrl,\n };\n\n this.persistSession(session);\n return session;\n }\n\n async logout(): Promise<void> {\n if (this.client) {\n this.client.stopClient();\n try {\n await this.client.logout(true);\n } catch {\n // ignore logout errors (e.g. token already invalidated)\n }\n this.client = undefined;\n }\n localStorage.removeItem(SESSION_KEY);\n }\n\n async clearAllData(): Promise<void> {\n if (this.client) {\n this.client.stopClient();\n this.client = undefined;\n }\n\n // Reset all cached crypto state\n this.secretStorageKey = undefined;\n this.secretStorageKeyId = undefined;\n this.recoveryPassphrase = undefined;\n this.fallbackPassphrase = undefined;\n\n localStorage.removeItem(SESSION_KEY);\n\n await this.deleteCryptoStore();\n }\n\n async getSession(): Promise<SessionInfo | null> {\n const raw = localStorage.getItem(SESSION_KEY);\n if (!raw) return null;\n try {\n return JSON.parse(raw) as SessionInfo;\n } catch {\n return null;\n }\n }\n\n // ── Sync ──────────────────────────────────────────────\n\n async startSync(): Promise<void> {\n this.requireClient();\n\n this.client!.on(ClientEvent.Sync, (state, _prev, data) => {\n const mapped = this.mapSyncState(state);\n this.notifyListeners('syncStateChange', {\n state: mapped,\n error: data?.error?.message,\n });\n });\n\n this.client!.on(RoomEvent.Timeline, (event: SdkMatrixEvent, room: Room | undefined) => {\n this.notifyListeners('messageReceived', {\n event: this.serializeEvent(event, room?.roomId),\n });\n // When an encrypted event arrives, listen for decryption and re-notify\n if (event.isBeingDecrypted() || event.getType() === 'm.room.encrypted') {\n event.once('Event.decrypted' as any, () => {\n this.notifyListeners('messageReceived', {\n event: this.serializeEvent(event, room?.roomId),\n });\n });\n }\n // When a reaction or redaction arrives, re-emit the parent event with updated aggregated reactions\n if (event.getType() === EventType.Reaction || event.getType() === EventType.RoomRedaction) {\n const rel = event.getContent()?.['m.relates_to'];\n const targetId = rel?.event_id || event.getAssociatedId();\n if (targetId && room) {\n const targetEvent = room.findEventById(targetId);\n if (targetEvent) {\n // Small delay to let the SDK finish aggregation\n setTimeout(() => {\n this.notifyListeners('messageReceived', {\n event: this.serializeEvent(targetEvent, room.roomId),\n });\n }, 100);\n }\n }\n }\n });\n\n this.client!.on(RoomEvent.Receipt, (event: SdkMatrixEvent, room: Room) => {\n const receiptContent = event.getContent() as Record<string, Record<string, Record<string, unknown>>>;\n for (const [eventId, receiptTypes] of Object.entries(receiptContent)) {\n const mRead = receiptTypes['m.read'] ?? {};\n for (const userId of Object.keys(mRead)) {\n this.notifyListeners('receiptReceived', {\n roomId: room.roomId,\n eventId,\n userId,\n });\n }\n }\n // Re-emit own sent messages with updated read status\n const myUserId = this.client?.getUserId();\n if (myUserId) {\n const timeline = room.getLiveTimeline().getEvents();\n // Walk backwards through recent events; stop after checking a reasonable batch\n const limit = Math.min(timeline.length, 50);\n for (let i = timeline.length - 1; i >= timeline.length - limit; i--) {\n const evt = timeline[i];\n if (evt.getSender() !== myUserId) continue;\n const serialized = this.serializeEvent(evt, room.roomId);\n if (serialized.status === 'read') {\n this.notifyListeners('messageReceived', { event: serialized });\n }\n }\n }\n });\n\n this.client!.on(RoomEvent.Name, (room: Room) => {\n this.notifyListeners('roomUpdated', {\n roomId: room.roomId,\n summary: this.serializeRoom(room),\n });\n });\n\n this.client!.on(RoomMemberEvent.Typing, (_event: SdkMatrixEvent, member: any) => {\n const roomId = member?.roomId;\n if (roomId) {\n const room = this.client!.getRoom(roomId);\n if (room) {\n const userIds: string[] = room\n .getMembers()\n .filter((m: any) => m.typing)\n .map((m: any) => m.userId);\n this.notifyListeners('typingChanged', { roomId, userIds });\n }\n }\n });\n\n this.client!.on(UserEvent.Presence, (_event: SdkMatrixEvent | undefined, user: User) => {\n this.notifyListeners('presenceChanged', {\n userId: user.userId,\n presence: {\n presence: (user.presence as PresenceInfo['presence']) ?? 'offline',\n statusMsg: user.presenceStatusMsg ?? undefined,\n lastActiveAgo: user.lastActiveAgo ?? undefined,\n },\n });\n });\n\n await this.client!.startClient({ initialSyncLimit: 20 });\n }\n\n async stopSync(): Promise<void> {\n this.requireClient();\n this.client!.stopClient();\n }\n\n async getSyncState(): Promise<{ state: SyncState }> {\n this.requireClient();\n const raw = this.client!.getSyncState();\n return { state: this.mapSyncState(raw) };\n }\n\n // ── Rooms ─────────────────────────────────────────────\n\n async createRoom(options: {\n name?: string;\n topic?: string;\n isEncrypted?: boolean;\n isDirect?: boolean;\n invite?: string[];\n preset?: 'private_chat' | 'trusted_private_chat' | 'public_chat';\n historyVisibility?: 'invited' | 'joined' | 'shared' | 'world_readable';\n }): Promise<{ roomId: string }> {\n this.requireClient();\n\n const createOpts: Record<string, unknown> = {\n visibility: 'private' as const,\n };\n if (options.name) createOpts.name = options.name;\n if (options.topic) createOpts.topic = options.topic;\n if (options.invite?.length) createOpts.invite = options.invite;\n if (options.preset) createOpts.preset = options.preset;\n if (options.isDirect) createOpts.is_direct = true;\n\n const initialState: Record<string, unknown>[] = [];\n if (options.isEncrypted) {\n initialState.push({\n type: 'm.room.encryption',\n state_key: '',\n content: { algorithm: 'm.megolm.v1.aes-sha2' },\n });\n }\n if (options.historyVisibility) {\n initialState.push({\n type: 'm.room.history_visibility',\n state_key: '',\n content: { history_visibility: options.historyVisibility },\n });\n }\n if (initialState.length > 0) {\n createOpts.initial_state = initialState;\n }\n\n const res = await this.client!.createRoom(createOpts);\n return { roomId: res.room_id };\n }\n\n async getRooms(): Promise<{ rooms: RoomSummary[] }> {\n this.requireClient();\n const rooms = this.client!.getRooms().map((r) => this.serializeRoom(r));\n return { rooms };\n }\n\n async getRoomMembers(options: { roomId: string }): Promise<{ members: RoomMember[] }> {\n this.requireClient();\n const room = this.client!.getRoom(options.roomId);\n if (!room) throw new Error(`Room ${options.roomId} not found`);\n\n await room.loadMembersIfNeeded();\n const members: RoomMember[] = room.getMembers().map((m) => ({\n userId: m.userId,\n displayName: m.name ?? undefined,\n membership: m.membership as RoomMember['membership'],\n avatarUrl: m.getMxcAvatarUrl() ?? undefined,\n }));\n\n return { members };\n }\n\n async joinRoom(options: { roomIdOrAlias: string }): Promise<{ roomId: string }> {\n this.requireClient();\n const room = await this.client!.joinRoom(options.roomIdOrAlias);\n return { roomId: room.roomId };\n }\n\n async leaveRoom(options: { roomId: string }): Promise<void> {\n this.requireClient();\n await this.client!.leave(options.roomId);\n }\n\n async forgetRoom(options: { roomId: string }): Promise<void> {\n this.requireClient();\n await this.client!.forget(options.roomId);\n }\n\n // ── Messaging ─────────────────────────────────────────\n\n async sendMessage(options: SendMessageOptions): Promise<{ eventId: string }> {\n this.requireClient();\n\n const msgtype = options.msgtype ?? 'm.text';\n const mediaTypes = ['m.image', 'm.audio', 'm.video', 'm.file'];\n\n if (mediaTypes.includes(msgtype) && options.fileUri) {\n // Media message: upload file then send\n const response = await fetch(options.fileUri);\n const blob = await response.blob();\n const uploadRes = await this.client!.uploadContent(blob, {\n name: options.fileName,\n type: options.mimeType,\n });\n const mxcUrl = uploadRes.content_uri;\n const content: Record<string, unknown> = {\n msgtype,\n body: options.body || options.fileName || 'file',\n url: mxcUrl,\n info: {\n mimetype: options.mimeType,\n size: options.fileSize ?? blob.size,\n ...(options.duration !== undefined && { duration: options.duration }),\n ...(options.width !== undefined && { w: options.width }),\n ...(options.height !== undefined && { h: options.height }),\n },\n };\n const res = await this.client!.sendMessage(options.roomId, content as any);\n return { eventId: res.event_id };\n }\n\n // Text message\n const msgtypeMap = {\n 'm.text': MsgType.Text,\n 'm.notice': MsgType.Notice,\n 'm.emote': MsgType.Emote,\n } as const;\n const mappedType = msgtypeMap[msgtype as keyof typeof msgtypeMap] ?? MsgType.Text;\n const res = await this.client!.sendMessage(options.roomId, {\n msgtype: mappedType,\n body: options.body,\n });\n return { eventId: res.event_id };\n }\n\n async editMessage(options: EditMessageOptions): Promise<{ eventId: string }> {\n this.requireClient();\n\n const msgtype = options.msgtype ?? 'm.text';\n const mediaTypes = ['m.image', 'm.audio', 'm.video', 'm.file'];\n\n let newContent: Record<string, unknown>;\n\n if (mediaTypes.includes(msgtype) && options.fileUri) {\n const response = await fetch(options.fileUri);\n const blob = await response.blob();\n const uploadRes = await this.client!.uploadContent(blob, {\n name: options.fileName,\n type: options.mimeType,\n });\n newContent = {\n msgtype,\n body: options.newBody || options.fileName || 'file',\n url: uploadRes.content_uri,\n info: {\n mimetype: options.mimeType,\n size: options.fileSize ?? blob.size,\n ...(options.duration !== undefined && { duration: options.duration }),\n ...(options.width !== undefined && { w: options.width }),\n ...(options.height !== undefined && { h: options.height }),\n },\n };\n } else {\n newContent = {\n msgtype,\n body: options.newBody,\n };\n }\n\n const content: Record<string, unknown> = {\n ...newContent,\n body: `* ${options.newBody}`,\n 'm.new_content': newContent,\n 'm.relates_to': {\n rel_type: 'm.replace',\n event_id: options.eventId,\n },\n };\n const res = await this.client!.sendMessage(options.roomId, content as any);\n return { eventId: res.event_id };\n }\n\n async sendReply(options: SendReplyOptions): Promise<{ eventId: string }> {\n this.requireClient();\n\n const msgtype = options.msgtype ?? 'm.text';\n const mediaTypes = ['m.image', 'm.audio', 'm.video', 'm.file'];\n\n let content: Record<string, unknown>;\n\n if (mediaTypes.includes(msgtype) && options.fileUri) {\n // Media reply: upload file then send with reply relation\n const response = await fetch(options.fileUri);\n const blob = await response.blob();\n const uploadRes = await this.client!.uploadContent(blob, {\n name: options.fileName,\n type: options.mimeType,\n });\n content = {\n msgtype,\n body: options.body || options.fileName || 'file',\n url: uploadRes.content_uri,\n info: {\n mimetype: options.mimeType,\n size: options.fileSize ?? blob.size,\n ...(options.duration !== undefined && { duration: options.duration }),\n ...(options.width !== undefined && { w: options.width }),\n ...(options.height !== undefined && { h: options.height }),\n },\n 'm.relates_to': {\n 'm.in_reply_to': {\n event_id: options.replyToEventId,\n },\n },\n };\n } else {\n // Text reply\n content = {\n msgtype: MsgType.Text,\n body: options.body,\n 'm.relates_to': {\n 'm.in_reply_to': {\n event_id: options.replyToEventId,\n },\n },\n };\n }\n\n const res = await this.client!.sendMessage(options.roomId, content as any);\n return { eventId: res.event_id };\n }\n\n async getRoomMessages(\n options: { roomId: string; limit?: number; from?: string },\n ): Promise<{ events: MatrixEvent[]; nextBatch?: string }> {\n this.requireClient();\n\n const limit = options.limit ?? 20;\n const room = this.client!.getRoom(options.roomId);\n\n // If no explicit pagination token, return events from the synced timeline\n if (!options.from && room) {\n // Paginate backwards so we have enough events (initial sync may be small)\n try {\n await this.client!.scrollback(room, limit);\n } catch {\n // scrollback may fail if there's no more history\n }\n\n const timeline = room.getLiveTimeline();\n const timelineEvents = timeline.getEvents();\n // Filter out reactions and redactions before slicing — they're aggregated into parent events\n const displayableEvents = timelineEvents.filter((e) => {\n const t = e.getType();\n return t !== EventType.Reaction && t !== EventType.RoomRedaction;\n });\n const events: MatrixEvent[] = displayableEvents\n .slice(-limit)\n .map((e) => this.serializeEvent(e, options.roomId))\n .sort((a, b) => a.originServerTs - b.originServerTs);\n const backToken = timeline.getPaginationToken(Direction.Backward) ?? undefined;\n return { events, nextBatch: backToken };\n }\n\n // Paginate further back using the token\n const fromToken = options.from ?? null;\n const res = await this.client!.createMessagesRequest(\n options.roomId,\n fromToken,\n limit,\n Direction.Backward,\n );\n\n const events: MatrixEvent[] = (res.chunk ?? []).map((e) => ({\n eventId: e.event_id,\n roomId: options.roomId,\n senderId: e.sender,\n type: e.type,\n content: (e.content ?? {}) as Record<string, unknown>,\n originServerTs: e.origin_server_ts,\n }));\n\n return { events, nextBatch: res.end ?? undefined };\n }\n\n async markRoomAsRead(options: { roomId: string; eventId: string }): Promise<void> {\n this.requireClient();\n const room = this.client!.getRoom(options.roomId);\n if (room) {\n const event = room.findEventById(options.eventId);\n if (event) {\n await this.client!.sendReadReceipt(event);\n return;\n }\n }\n // Fallback to HTTP request if event not found locally\n await this.client!.setRoomReadMarkersHttpRequest(\n options.roomId,\n options.eventId,\n options.eventId,\n );\n }\n\n async refreshEventStatuses(options: { roomId: string; eventIds: string[] }): Promise<{ events: import('./definitions').MatrixEvent[] }> {\n this.requireClient();\n const room = this.client!.getRoom(options.roomId);\n if (!room) return { events: [] };\n const events: import('./definitions').MatrixEvent[] = [];\n for (const eid of options.eventIds) {\n const event = room.findEventById(eid);\n if (event) {\n events.push(this.serializeEvent(event, options.roomId));\n }\n }\n return { events };\n }\n\n // ── Redactions & Reactions ───────────────────────────────\n\n async redactEvent(options: { roomId: string; eventId: string; reason?: string }): Promise<void> {\n this.requireClient();\n await this.client!.redactEvent(options.roomId, options.eventId, undefined, {\n reason: options.reason,\n } as any);\n }\n\n async sendReaction(options: { roomId: string; eventId: string; key: string }): Promise<{ eventId: string }> {\n this.requireClient();\n const myUserId = this.client!.getUserId();\n\n // Check if the user already reacted with this key — if so, toggle off (redact)\n const room = this.client!.getRoom(options.roomId);\n if (room && myUserId) {\n try {\n const relations = room.relations.getChildEventsForEvent(\n options.eventId,\n RelationType.Annotation,\n EventType.Reaction,\n );\n if (relations) {\n const existing = relations.getRelations().find(\n (e) => e.getSender() === myUserId && e.getContent()?.['m.relates_to']?.key === options.key,\n );\n if (existing) {\n const existingId = existing.getId();\n if (existingId) {\n await this.client!.redactEvent(options.roomId, existingId);\n return { eventId: existingId };\n }\n }\n }\n } catch {\n // fall through to send\n }\n }\n\n const res = await this.client!.sendEvent(options.roomId, EventType.Reaction, {\n 'm.relates_to': {\n rel_type: RelationType.Annotation,\n event_id: options.eventId,\n key: options.key,\n },\n });\n return { eventId: res.event_id };\n }\n\n // ── Room Management ────────────────────────────────────\n\n async setRoomName(options: { roomId: string; name: string }): Promise<void> {\n this.requireClient();\n await this.client!.setRoomName(options.roomId, options.name);\n }\n\n async setRoomTopic(options: { roomId: string; topic: string }): Promise<void> {\n this.requireClient();\n await this.client!.setRoomTopic(options.roomId, options.topic);\n }\n\n async setRoomAvatar(options: { roomId: string; mxcUrl: string }): Promise<void> {\n this.requireClient();\n await this.client!.sendStateEvent(options.roomId, 'm.room.avatar' as any, { url: options.mxcUrl });\n }\n\n async inviteUser(options: { roomId: string; userId: string }): Promise<void> {\n this.requireClient();\n await this.client!.invite(options.roomId, options.userId);\n }\n\n async kickUser(options: { roomId: string; userId: string; reason?: string }): Promise<void> {\n this.requireClient();\n await this.client!.kick(options.roomId, options.userId, options.reason);\n }\n\n async banUser(options: { roomId: string; userId: string; reason?: string }): Promise<void> {\n this.requireClient();\n await this.client!.ban(options.roomId, options.userId, options.reason);\n }\n\n async unbanUser(options: { roomId: string; userId: string }): Promise<void> {\n this.requireClient();\n await this.client!.unban(options.roomId, options.userId);\n }\n\n // ── Typing ─────────────────────────────────────────────\n\n async sendTyping(options: { roomId: string; isTyping: boolean; timeout?: number }): Promise<void> {\n this.requireClient();\n await this.client!.sendTyping(options.roomId, options.isTyping, options.timeout ?? 30000);\n }\n\n // ── Media ──────────────────────────────────────────────\n\n async getMediaUrl(options: { mxcUrl: string }): Promise<{ httpUrl: string }> {\n this.requireClient();\n // Use the authenticated media endpoint (Matrix v1.11+)\n const mxcPath = options.mxcUrl.replace('mxc://', '');\n const baseUrl = this.client!.getHomeserverUrl().replace(/\\/$/, '');\n const accessToken = this.client!.getAccessToken();\n const httpUrl = `${baseUrl}/_matrix/client/v1/media/download/${mxcPath}?access_token=${accessToken}`;\n return { httpUrl };\n }\n\n async getThumbnailUrl(options: ThumbnailUrlOptions): Promise<{ httpUrl: string }> {\n this.requireClient();\n const mxcPath = options.mxcUrl.replace('mxc://', '');\n const baseUrl = this.client!.getHomeserverUrl().replace(/\\/$/, '');\n const accessToken = this.client!.getAccessToken();\n const method = options.method ?? 'scale';\n const httpUrl = `${baseUrl}/_matrix/client/v1/media/thumbnail/${mxcPath}?width=${options.width}&height=${options.height}&method=${method}&access_token=${accessToken}`;\n return { httpUrl };\n }\n\n async uploadContent(options: UploadContentOptions): Promise<UploadContentResult> {\n this.requireClient();\n const response = await fetch(options.fileUri);\n const blob = await response.blob();\n const res = await this.client!.uploadContent(blob, {\n name: options.fileName,\n type: options.mimeType,\n });\n return { contentUri: res.content_uri };\n }\n\n // ── Presence ───────────────────────────────────────────\n\n async setPresence(options: { presence: 'online' | 'offline' | 'unavailable'; statusMsg?: string }): Promise<void> {\n this.requireClient();\n await this.client!.setPresence({\n presence: options.presence,\n status_msg: options.statusMsg,\n });\n }\n\n async getPresence(options: { userId: string }): Promise<PresenceInfo> {\n this.requireClient();\n const user = this.client!.getUser(options.userId);\n return {\n presence: (user?.presence as PresenceInfo['presence']) ?? 'offline',\n statusMsg: user?.presenceStatusMsg ?? undefined,\n lastActiveAgo: user?.lastActiveAgo ?? undefined,\n };\n }\n\n // ── Device Management ──────────────────────────────────\n\n async getDevices(): Promise<{ devices: DeviceInfo[] }> {\n this.requireClient();\n const res = await this.client!.getDevices();\n const crypto = this.client!.getCrypto();\n const myUserId = this.client!.getUserId() ?? '';\n const devices: DeviceInfo[] = await Promise.all(\n (res.devices ?? []).map(async (d: any) => {\n let isCrossSigningVerified: boolean | undefined;\n if (crypto) {\n try {\n const status = await crypto.getDeviceVerificationStatus(myUserId, d.device_id);\n isCrossSigningVerified = status?.crossSigningVerified ?? false;\n } catch {\n // ignore — crypto may not be ready\n }\n }\n return {\n deviceId: d.device_id,\n displayName: d.display_name ?? undefined,\n lastSeenTs: d.last_seen_ts ?? undefined,\n lastSeenIp: d.last_seen_ip ?? undefined,\n isCrossSigningVerified,\n };\n }),\n );\n return { devices };\n }\n\n async deleteDevice(options: { deviceId: string; auth?: Record<string, unknown> }): Promise<void> {\n this.requireClient();\n await this.client!.deleteDevice(options.deviceId, options.auth as any);\n }\n\n // ── Push ──────────────────────────────────────────────\n\n async setPusher(options: PusherOptions): Promise<void> {\n this.requireClient();\n await this.client!.setPusher({\n pushkey: options.pushkey,\n kind: options.kind ?? undefined,\n app_id: options.appId,\n app_display_name: options.appDisplayName,\n device_display_name: options.deviceDisplayName,\n lang: options.lang,\n data: options.data,\n } as any);\n }\n\n // ── Encryption ──────────────────────────────────────────\n\n async initializeCrypto(): Promise<void> {\n this.requireClient();\n const cryptoOpts = { cryptoDatabasePrefix: 'matrix-js-sdk' };\n\n try {\n await this.client!.initRustCrypto(cryptoOpts);\n } catch (e: any) {\n // After logout + re-login the server issues a new deviceId, but the\n // shared IndexedDB crypto store still references the old one.\n // Delete the stale store and retry so crypto initialises cleanly.\n if (e?.message?.includes(\"account in the store doesn't match\")) {\n await this.deleteCryptoStore();\n await this.client!.initRustCrypto(cryptoOpts);\n } else {\n throw e;\n }\n }\n\n // Flush the initial /keys/query request that initRustCrypto enqueues.\n // Without this, any call to getIdentity (e.g. via getCrossSigningStatus)\n // will spin-wait and emit periodic WARN logs until sync processes it.\n const crypto = this.client!.getCrypto() as any;\n if (crypto?.outgoingRequestsManager?.doProcessOutgoingRequests) {\n await crypto.outgoingRequestsManager.doProcessOutgoingRequests();\n }\n }\n\n private async deleteCryptoStore(): Promise<void> {\n if (typeof indexedDB === 'undefined') return;\n try {\n const dbs = await indexedDB.databases();\n await Promise.all(\n dbs\n .filter((db) => db.name?.startsWith('matrix-js-sdk'))\n .map(\n (db) =>\n new Promise<void>((resolve) => {\n const req = indexedDB.deleteDatabase(db.name!);\n req.onsuccess = () => resolve();\n req.onerror = () => resolve();\n }),\n ),\n );\n } catch {\n // indexedDB.databases() not available in all environments\n }\n }\n\n async getEncryptionStatus(): Promise<EncryptionStatus> {\n this.requireClient();\n const crypto = this.client!.getCrypto();\n if (!crypto) {\n return {\n isCrossSigningReady: false,\n crossSigningStatus: { hasMaster: false, hasSelfSigning: false, hasUserSigning: false, isReady: false },\n isKeyBackupEnabled: false,\n isSecretStorageReady: false,\n };\n }\n\n const csReady = await crypto.isCrossSigningReady();\n const csStatus = await crypto.getCrossSigningStatus();\n const backupVersion = await crypto.getActiveSessionBackupVersion();\n\n // Use getSecretStorageStatus().defaultKeyId to check if secret storage was\n // set up at all, rather than isSecretStorageReady() which also checks that\n // cross-signing keys are stored (too strict for Phase 1).\n const ssStatus = await crypto.getSecretStorageStatus();\n const ssHasKey = ssStatus.defaultKeyId !== null;\n\n return {\n isCrossSigningReady: csReady,\n crossSigningStatus: {\n hasMaster: csStatus.publicKeysOnDevice,\n hasSelfSigning: csStatus.privateKeysCachedLocally.selfSigningKey,\n hasUserSigning: csStatus.privateKeysCachedLocally.userSigningKey,\n isReady: csReady,\n },\n isKeyBackupEnabled: backupVersion !== null,\n keyBackupVersion: backupVersion ?? undefined,\n isSecretStorageReady: ssHasKey,\n };\n }\n\n async bootstrapCrossSigning(): Promise<void> {\n const crypto = await this.ensureCrypto();\n await crypto.bootstrapCrossSigning({\n setupNewCrossSigning: true,\n authUploadDeviceSigningKeys: async (makeRequest) => {\n // UIA flow: attempt with dummy auth, fall back to session-based retry\n try {\n await makeRequest({ type: 'm.login.dummy' });\n } catch (e: any) {\n const session = e?.data?.session;\n if (session) {\n await makeRequest({ type: 'm.login.dummy', session });\n } else {\n throw e;\n }\n }\n },\n });\n }\n\n async setupKeyBackup(): Promise<KeyBackupStatus> {\n const crypto = await this.ensureCrypto();\n await crypto.resetKeyBackup();\n const version = await crypto.getActiveSessionBackupVersion();\n return { exists: true, version: version ?? undefined, enabled: true };\n }\n\n async getKeyBackupStatus(): Promise<KeyBackupStatus> {\n this.requireClient();\n const crypto = this.requireCrypto();\n const version = await crypto.getActiveSessionBackupVersion();\n return {\n exists: version !== null,\n version: version ?? undefined,\n enabled: version !== null,\n };\n }\n\n async restoreKeyBackup(_options?: { recoveryKey?: string }): Promise<{ importedKeys: number }> {\n this.requireClient();\n const crypto = this.requireCrypto();\n\n const result = await crypto.restoreKeyBackup();\n return { importedKeys: result?.imported ?? 0 };\n }\n\n async setupRecovery(options?: {\n passphrase?: string;\n existingPassphrase?: string;\n }): Promise<RecoveryKeyInfo> {\n const crypto = await this.ensureCrypto();\n\n const keyInfo = await crypto.createRecoveryKeyFromPassphrase(options?.passphrase);\n // Pre-cache the new key bytes. secretStorageKeyId will be set by\n // cacheSecretStorageKey once bootstrapSecretStorage writes the new key\n // into SSSS and the SDK calls back.\n this.secretStorageKey = keyInfo.privateKey;\n this.secretStorageKeyId = undefined;\n\n // If the caller provides the same or old passphrase, keep it so\n // getSecretStorageKey can derive the key for the SDK.\n if (options?.passphrase) {\n this.recoveryPassphrase = options.passphrase;\n }\n // If the caller knows the OLD passphrase, keep it as fallbackPassphrase so\n // that getSecretStorageKey can decrypt the existing SSSS during\n // bootstrapSecretStorage's migration of cross-signing / backup secrets.\n if (options?.existingPassphrase) {\n this.fallbackPassphrase = options.existingPassphrase;\n }\n\n try {\n const bootstrapPromise = crypto.bootstrapSecretStorage({\n createSecretStorageKey: async () => keyInfo,\n setupNewSecretStorage: true,\n setupNewKeyBackup: true,\n });\n\n // Guard against SDK hanging when it can't retrieve the old SSSS key\n const timeoutPromise = new Promise<never>((_, reject) => {\n setTimeout(\n () => reject(new Error('bootstrapSecretStorage timed out — the old SSSS key could not be retrieved')),\n 30_000,\n );\n });\n\n await Promise.race([bootstrapPromise, timeoutPromise]);\n } finally {\n // Always clear transient crypto state so it doesn't bleed into subsequent calls.\n this.fallbackPassphrase = undefined;\n this.recoveryPassphrase = undefined;\n }\n\n return { recoveryKey: keyInfo.encodedPrivateKey ?? '' };\n }\n\n async isRecoveryEnabled(): Promise<{ enabled: boolean }> {\n const crypto = await this.ensureCrypto();\n const ready = await crypto.isSecretStorageReady();\n return { enabled: ready };\n }\n\n async recoverAndSetup(options: { recoveryKey?: string; passphrase?: string }): Promise<void> {\n const crypto = await this.ensureCrypto();\n\n // Derive/decode the secret storage key\n if (options.recoveryKey) {\n this.secretStorageKey = decodeRecoveryKey(options.recoveryKey);\n } else if (options.passphrase) {\n // Store passphrase — the getSecretStorageKey callback will derive\n // the key using the server's stored PBKDF2 params (salt, iterations)\n this.recoveryPassphrase = options.passphrase;\n this.secretStorageKey = undefined; // Clear any stale raw key\n } else {\n throw new Error('Either recoveryKey or passphrase must be provided');\n }\n\n // Load the backup decryption key from secret storage into the Rust crypto store.\n // This triggers the getSecretStorageKey callback.\n try {\n await crypto.loadSessionBackupPrivateKeyFromSecretStorage();\n } catch (e) {\n const msg = e instanceof Error ? e.message : String(e);\n if (msg.includes('decryption key does not match')) {\n // The passphrase is correct (SSSS decrypted fine), but the backup key\n // stored in SSSS doesn't match the server's current backup. This happens\n // when another client re-created the backup without updating SSSS, or\n // vice-versa. Auto-fix by creating a new backup that matches the SSSS key.\n // recoveryPassphrase / secretStorageKey are still set, so the\n // getSecretStorageKey callback can decrypt the existing SSSS.\n await crypto.bootstrapSecretStorage({\n setupNewKeyBackup: true,\n });\n await crypto.checkKeyBackupAndEnable();\n return;\n }\n // Different error — clear state and throw\n this.secretStorageKey = undefined;\n this.secretStorageKeyId = undefined;\n this.recoveryPassphrase = undefined;\n throw e;\n }\n\n // Now that the key is stored locally, activate backup in the running client\n await crypto.checkKeyBackupAndEnable();\n }\n\n async resetRecoveryKey(options?: { passphrase?: string }): Promise<RecoveryKeyInfo> {\n return this.setupRecovery(options);\n }\n\n async exportRoomKeys(options: { passphrase: string }): Promise<{ data: string }> {\n this.requireClient();\n const crypto = this.requireCrypto();\n const keys = await crypto.exportRoomKeysAsJson();\n // The exported JSON is not encrypted by default; for passphrase encryption\n // the caller should handle it, or we return the raw JSON\n void options.passphrase; // passphrase encryption is handled natively; on web we return raw\n return { data: keys };\n }\n\n async importRoomKeys(options: { data: string; passphrase: string }): Promise<{ importedKeys: number }> {\n this.requireClient();\n const crypto = this.requireCrypto();\n void options.passphrase; // passphrase decryption handled natively; on web we import raw\n await crypto.importRoomKeysAsJson(options.data);\n return { importedKeys: -1 }; // count not available from importRoomKeysAsJson\n }\n\n // ── Helpers ───────────────────────────────────────────\n\n private requireClient(): void {\n if (!this.client) {\n throw new Error('Not logged in. Call login() or loginWithToken() first.');\n }\n }\n\n private requireCrypto() {\n const crypto = this.client!.getCrypto();\n if (!crypto) {\n throw new Error('Crypto not initialized. Call initializeCrypto() first.');\n }\n return crypto;\n }\n\n private async ensureCrypto() {\n this.requireClient();\n if (!this.client!.getCrypto()) {\n await this.initializeCrypto();\n }\n return this.requireCrypto();\n }\n\n private persistSession(session: SessionInfo): void {\n localStorage.setItem(SESSION_KEY, JSON.stringify(session));\n }\n\n private serializeEvent(event: SdkMatrixEvent, fallbackRoomId?: string): MatrixEvent {\n const roomId = event.getRoomId() ?? fallbackRoomId ?? '';\n\n // Redacted events should be marked clearly\n if (event.isRedacted()) {\n return {\n eventId: event.getId() ?? '',\n roomId,\n senderId: event.getSender() ?? '',\n type: 'm.room.redaction',\n content: { body: 'Message deleted' },\n originServerTs: event.getTs(),\n };\n }\n\n const content = { ...(event.getContent() ?? {}) } as Record<string, unknown>;\n\n // Include aggregated reactions from the room's relations container\n const eventId = event.getId();\n if (eventId && roomId) {\n const room = this.client?.getRoom(roomId);\n if (room) {\n try {\n const relations = room.relations.getChildEventsForEvent(\n eventId,\n RelationType.Annotation,\n EventType.Reaction,\n );\n if (relations) {\n const sorted = relations.getSortedAnnotationsByKey();\n if (sorted && sorted.length > 0) {\n content.reactions = sorted.map(([key, events]) => ({\n key,\n count: events.size,\n senders: Array.from(events).map((e) => e.getSender()),\n }));\n }\n }\n } catch {\n // relations may not be available\n }\n }\n }\n\n // Determine delivery/read status\n let status: MatrixEvent['status'];\n const readBy: string[] = [];\n const myUserId = this.client?.getUserId();\n const sender = event.getSender();\n const room = eventId && roomId ? this.client?.getRoom(roomId) : undefined;\n\n if (sender === myUserId && eventId) {\n // Own message — check delivery status\n const evtStatus = event.status; // null = sent & echoed, 'sending', 'sent', etc.\n if (evtStatus === 'sending' || evtStatus === 'encrypting' || evtStatus === 'queued') {\n status = 'sending';\n } else {\n // Event is at least sent; check if anyone has read it\n if (room) {\n try {\n const members = room.getJoinedMembers();\n for (const member of members) {\n if (member.userId === myUserId) continue;\n if (room.hasUserReadEvent(member.userId, eventId)) {\n readBy.push(member.userId);\n }\n }\n } catch {\n // ignore errors\n }\n }\n status = readBy.length > 0 ? 'read' : 'sent';\n }\n } else if (eventId && room) {\n // Other's message — collect who has read it\n try {\n const members = room.getJoinedMembers();\n for (const member of members) {\n if (member.userId === sender) continue;\n if (room.hasUserReadEvent(member.userId, eventId)) {\n readBy.push(member.userId);\n }\n }\n } catch {\n // ignore\n }\n }\n\n // Include unsigned data (e.g. m.relations for edits, transaction_id for local echo)\n const unsignedData = event.getUnsigned?.();\n const unsigned = unsignedData && Object.keys(unsignedData).length > 0\n ? (unsignedData as Record<string, unknown>)\n : undefined;\n\n return {\n eventId: eventId ?? '',\n roomId,\n senderId: sender ?? '',\n type: event.getType(),\n content,\n originServerTs: event.getTs(),\n status,\n readBy: readBy.length > 0 ? readBy : undefined,\n unsigned,\n };\n }\n\n private serializeRoom(room: Room): RoomSummary {\n // Detect DM: check m.direct account data or guess from room state\n let isDirect = false;\n try {\n const directEvent = this.client?.getAccountData('m.direct' as any);\n if (directEvent) {\n const directContent = directEvent.getContent() as Record<string, string[]>;\n for (const roomIds of Object.values(directContent)) {\n if (roomIds.includes(room.roomId)) {\n isDirect = true;\n break;\n }\n }\n }\n } catch {\n // ignore\n }\n\n // Get avatar URL\n let avatarUrl: string | undefined;\n const avatarEvent = room.currentState.getStateEvents('m.room.avatar', '');\n if (avatarEvent) {\n const mxcUrl = avatarEvent.getContent()?.url as string | undefined;\n if (mxcUrl) {\n avatarUrl = mxcUrl;\n }\n }\n\n return {\n roomId: room.roomId,\n name: room.name,\n topic: room.currentState.getStateEvents('m.room.topic', '')?.getContent()?.topic ?? undefined,\n memberCount: room.getJoinedMemberCount(),\n isEncrypted: room.hasEncryptionStateEvent(),\n unreadCount: room.getUnreadNotificationCount() ?? 0,\n lastEventTs: room.getLastActiveTimestamp() || undefined,\n membership: room.getMyMembership() as RoomSummary['membership'],\n avatarUrl,\n isDirect,\n };\n }\n\n async searchUsers(options: {\n searchTerm: string;\n limit?: number;\n }): Promise<{ results: UserProfile[]; limited: boolean }> {\n this.requireClient();\n const resp = await this.client!.searchUserDirectory({\n term: options.searchTerm,\n limit: options.limit ?? 10,\n });\n return {\n results: resp.results.map((u: { user_id: string; display_name?: string; avatar_url?: string }) => ({\n userId: u.user_id,\n displayName: u.display_name,\n avatarUrl: u.avatar_url,\n })),\n limited: resp.limited,\n };\n }\n\n private mapSyncState(state: string | null): SyncState {\n switch (state) {\n case 'PREPARED':\n case 'SYNCING':\n case 'CATCHUP':\n case 'RECONNECTING':\n return 'SYNCING';\n case 'ERROR':\n return 'ERROR';\n case 'STOPPED':\n return 'STOPPED';\n default:\n return 'INITIAL';\n }\n }\n}\n"]}
|
|
1
|
+
{"version":3,"file":"web.js","sourceRoot":"","sources":["../../src/web.ts"],"names":[],"mappings":"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","sourcesContent":["import { WebPlugin } from '@capacitor/core';\nimport {\n createClient,\n ClientEvent,\n RoomEvent,\n RoomMemberEvent,\n Direction,\n MsgType,\n EventType,\n RelationType,\n UserEvent,\n} from 'matrix-js-sdk';\nimport { decodeRecoveryKey } from 'matrix-js-sdk/lib/crypto-api/recovery-key';\nimport { deriveRecoveryKeyFromPassphrase } from 'matrix-js-sdk/lib/crypto-api/key-passphrase';\nimport type { MatrixClient, Room, MatrixEvent as SdkMatrixEvent, User } from 'matrix-js-sdk';\n\nimport type {\n MatrixPlugin,\n LoginOptions,\n LoginWithTokenOptions,\n SessionInfo,\n SendMessageOptions,\n EditMessageOptions,\n SendReplyOptions,\n UploadContentOptions,\n UploadContentResult,\n ThumbnailUrlOptions,\n PusherOptions,\n DeviceInfo,\n MatrixEvent,\n RoomSummary,\n RoomMember,\n SyncState,\n EncryptionStatus,\n KeyBackupStatus,\n RecoveryKeyInfo,\n PresenceInfo,\n UserProfile,\n} from './definitions';\n\nconst SESSION_KEY = 'matrix_session';\n\nexport class MatrixWeb extends WebPlugin implements MatrixPlugin {\n private client?: MatrixClient;\n private secretStorageKey?: Uint8Array<ArrayBuffer>;\n private secretStorageKeyId?: string; // key ID the cached bytes belong to\n private recoveryPassphrase?: string;\n private fallbackPassphrase?: string; // old passphrase for SSSS migration in setupRecovery\n\n private readonly _cryptoCallbacks = {\n getSecretStorageKey: async (\n opts: { keys: Record<string, unknown> },\n ): Promise<[string, Uint8Array<ArrayBuffer>] | null> => {\n const keyId = Object.keys(opts.keys)[0];\n if (!keyId) return null;\n\n // Exact match: only return the cached raw key for the key ID it was cached under.\n // (bootstrapSecretStorage uses createSecretStorageKey for the new key, so this\n // path is only reached for an already-established key — e.g. after recoverAndSetup.)\n if (this.secretStorageKey && this.secretStorageKeyId === keyId) {\n return [keyId, this.secretStorageKey];\n }\n\n // Derive from the current passphrase (set during recoverAndSetup)\n if (this.recoveryPassphrase) {\n const keyInfo = opts.keys[keyId] as {\n passphrase?: { salt: string; iterations: number; bits?: number };\n };\n if (keyInfo?.passphrase) {\n const derived = await deriveRecoveryKeyFromPassphrase(\n this.recoveryPassphrase,\n keyInfo.passphrase.salt,\n keyInfo.passphrase.iterations,\n keyInfo.passphrase.bits ?? 256,\n );\n // Cache with the correct key ID for subsequent calls\n this.secretStorageKey = derived;\n this.secretStorageKeyId = keyId;\n return [keyId, derived];\n }\n }\n\n // Fallback: derive from the OLD passphrase when bootstrapSecretStorage is\n // migrating existing cross-signing / backup secrets into a new SSSS.\n if (this.fallbackPassphrase) {\n const keyInfo = opts.keys[keyId] as {\n passphrase?: { salt: string; iterations: number; bits?: number };\n };\n if (keyInfo?.passphrase) {\n const derived = await deriveRecoveryKeyFromPassphrase(\n this.fallbackPassphrase,\n keyInfo.passphrase.salt,\n keyInfo.passphrase.iterations,\n keyInfo.passphrase.bits ?? 256,\n );\n return [keyId, derived];\n }\n }\n\n return null;\n },\n cacheSecretStorageKey: (\n keyId: string,\n _keyInfo: unknown,\n key: Uint8Array<ArrayBuffer>,\n ): void => {\n this.secretStorageKey = key;\n this.secretStorageKeyId = keyId;\n },\n };\n\n // ── Auth ──────────────────────────────────────────────\n\n async login(options: LoginOptions): Promise<SessionInfo> {\n const tmpClient = createClient({ baseUrl: options.homeserverUrl });\n const res = await tmpClient.loginWithPassword(options.userId, options.password);\n\n this.client = createClient({\n baseUrl: options.homeserverUrl,\n accessToken: res.access_token,\n userId: res.user_id,\n deviceId: res.device_id,\n cryptoCallbacks: this._cryptoCallbacks,\n });\n\n const session: SessionInfo = {\n accessToken: res.access_token,\n userId: res.user_id,\n deviceId: res.device_id,\n homeserverUrl: options.homeserverUrl,\n };\n\n this.persistSession(session);\n return session;\n }\n\n async loginWithToken(options: LoginWithTokenOptions): Promise<SessionInfo> {\n // Stop any previously running client to avoid parallel instances that\n // would deadlock on the shared IndexedDB crypto store.\n if (this.client) {\n this.client.stopClient();\n this.client = undefined;\n }\n\n this.client = createClient({\n baseUrl: options.homeserverUrl,\n accessToken: options.accessToken,\n userId: options.userId,\n deviceId: options.deviceId,\n cryptoCallbacks: this._cryptoCallbacks,\n });\n\n const session: SessionInfo = {\n accessToken: options.accessToken,\n userId: options.userId,\n deviceId: options.deviceId,\n homeserverUrl: options.homeserverUrl,\n };\n\n this.persistSession(session);\n return session;\n }\n\n async logout(): Promise<void> {\n if (this.client) {\n this.client.stopClient();\n try {\n await this.client.logout(true);\n } catch {\n // ignore logout errors (e.g. token already invalidated)\n }\n this.client = undefined;\n }\n localStorage.removeItem(SESSION_KEY);\n }\n\n async clearAllData(): Promise<void> {\n if (this.client) {\n this.client.stopClient();\n this.client = undefined;\n }\n\n // Reset all cached crypto state\n this.secretStorageKey = undefined;\n this.secretStorageKeyId = undefined;\n this.recoveryPassphrase = undefined;\n this.fallbackPassphrase = undefined;\n\n localStorage.removeItem(SESSION_KEY);\n\n await this.deleteCryptoStore();\n }\n\n async getSession(): Promise<SessionInfo | null> {\n const raw = localStorage.getItem(SESSION_KEY);\n if (!raw) return null;\n try {\n return JSON.parse(raw) as SessionInfo;\n } catch {\n return null;\n }\n }\n\n // ── Sync ──────────────────────────────────────────────\n\n async startSync(): Promise<void> {\n this.requireClient();\n\n this.client!.on(ClientEvent.Sync, (state, _prev, data) => {\n const mapped = this.mapSyncState(state);\n this.notifyListeners('syncStateChange', {\n state: mapped,\n error: data?.error?.message,\n });\n });\n\n this.client!.on(RoomEvent.Timeline, (event: SdkMatrixEvent, room: Room | undefined) => {\n this.notifyListeners('messageReceived', {\n event: this.serializeEvent(event, room?.roomId),\n });\n // When an encrypted event arrives, listen for decryption and re-notify\n if (event.isBeingDecrypted() || event.getType() === 'm.room.encrypted') {\n event.once('Event.decrypted' as any, () => {\n this.notifyListeners('messageReceived', {\n event: this.serializeEvent(event, room?.roomId),\n });\n });\n }\n // When a reaction or redaction arrives, re-emit the parent event with updated aggregated reactions\n if (event.getType() === EventType.Reaction || event.getType() === EventType.RoomRedaction) {\n const rel = event.getContent()?.['m.relates_to'];\n const targetId = rel?.event_id || event.getAssociatedId();\n if (targetId && room) {\n const targetEvent = room.findEventById(targetId);\n if (targetEvent) {\n // Small delay to let the SDK finish aggregation\n setTimeout(() => {\n this.notifyListeners('messageReceived', {\n event: this.serializeEvent(targetEvent, room.roomId),\n });\n }, 100);\n }\n }\n }\n });\n\n this.client!.on(RoomEvent.Receipt, (event: SdkMatrixEvent, room: Room) => {\n const receiptContent = event.getContent() as Record<string, Record<string, Record<string, unknown>>>;\n for (const [eventId, receiptTypes] of Object.entries(receiptContent)) {\n const mRead = receiptTypes['m.read'] ?? {};\n for (const userId of Object.keys(mRead)) {\n this.notifyListeners('receiptReceived', {\n roomId: room.roomId,\n eventId,\n userId,\n });\n }\n }\n // Re-emit own sent messages with updated read status\n const myUserId = this.client?.getUserId();\n if (myUserId) {\n const timeline = room.getLiveTimeline().getEvents();\n // Walk backwards through recent events; stop after checking a reasonable batch\n const limit = Math.min(timeline.length, 50);\n for (let i = timeline.length - 1; i >= timeline.length - limit; i--) {\n const evt = timeline[i];\n if (evt.getSender() !== myUserId) continue;\n const serialized = this.serializeEvent(evt, room.roomId);\n if (serialized.status === 'read') {\n this.notifyListeners('messageReceived', { event: serialized });\n }\n }\n }\n });\n\n this.client!.on(RoomEvent.Name, (room: Room) => {\n this.notifyListeners('roomUpdated', {\n roomId: room.roomId,\n summary: this.serializeRoom(room),\n });\n });\n\n this.client!.on(RoomMemberEvent.Typing, (_event: SdkMatrixEvent, member: any) => {\n const roomId = member?.roomId;\n if (roomId) {\n const room = this.client!.getRoom(roomId);\n if (room) {\n const userIds: string[] = room\n .getMembers()\n .filter((m: any) => m.typing)\n .map((m: any) => m.userId);\n this.notifyListeners('typingChanged', { roomId, userIds });\n }\n }\n });\n\n this.client!.on(UserEvent.Presence, (_event: SdkMatrixEvent | undefined, user: User) => {\n this.notifyListeners('presenceChanged', {\n userId: user.userId,\n presence: {\n presence: (user.presence as PresenceInfo['presence']) ?? 'offline',\n statusMsg: user.presenceStatusMsg ?? undefined,\n lastActiveAgo: user.lastActiveAgo ?? undefined,\n },\n });\n });\n\n await this.client!.startClient({ initialSyncLimit: 20 });\n }\n\n async stopSync(): Promise<void> {\n this.requireClient();\n this.client!.stopClient();\n }\n\n async getSyncState(): Promise<{ state: SyncState }> {\n this.requireClient();\n const raw = this.client!.getSyncState();\n return { state: this.mapSyncState(raw) };\n }\n\n // ── Rooms ─────────────────────────────────────────────\n\n async createRoom(options: {\n name?: string;\n topic?: string;\n isEncrypted?: boolean;\n isDirect?: boolean;\n invite?: string[];\n preset?: 'private_chat' | 'trusted_private_chat' | 'public_chat';\n historyVisibility?: 'invited' | 'joined' | 'shared' | 'world_readable';\n }): Promise<{ roomId: string }> {\n this.requireClient();\n\n const createOpts: Record<string, unknown> = {\n visibility: 'private' as const,\n };\n if (options.name) createOpts.name = options.name;\n if (options.topic) createOpts.topic = options.topic;\n if (options.invite?.length) createOpts.invite = options.invite;\n if (options.preset) createOpts.preset = options.preset;\n if (options.isDirect) createOpts.is_direct = true;\n\n const initialState: Record<string, unknown>[] = [];\n if (options.isEncrypted) {\n initialState.push({\n type: 'm.room.encryption',\n state_key: '',\n content: { algorithm: 'm.megolm.v1.aes-sha2' },\n });\n }\n if (options.historyVisibility) {\n initialState.push({\n type: 'm.room.history_visibility',\n state_key: '',\n content: { history_visibility: options.historyVisibility },\n });\n }\n if (initialState.length > 0) {\n createOpts.initial_state = initialState;\n }\n\n const res = await this.client!.createRoom(createOpts);\n return { roomId: res.room_id };\n }\n\n async getRooms(): Promise<{ rooms: RoomSummary[] }> {\n this.requireClient();\n const rooms = this.client!.getRooms().map((r) => this.serializeRoom(r));\n return { rooms };\n }\n\n async getRoomMembers(options: { roomId: string }): Promise<{ members: RoomMember[] }> {\n this.requireClient();\n const room = this.client!.getRoom(options.roomId);\n if (!room) throw new Error(`Room ${options.roomId} not found`);\n\n await room.loadMembersIfNeeded();\n const members: RoomMember[] = room.getMembers().map((m) => ({\n userId: m.userId,\n displayName: m.name ?? undefined,\n membership: m.membership as RoomMember['membership'],\n avatarUrl: m.getMxcAvatarUrl() ?? undefined,\n }));\n\n return { members };\n }\n\n async joinRoom(options: { roomIdOrAlias: string }): Promise<{ roomId: string }> {\n this.requireClient();\n const room = await this.client!.joinRoom(options.roomIdOrAlias);\n return { roomId: room.roomId };\n }\n\n async leaveRoom(options: { roomId: string }): Promise<void> {\n this.requireClient();\n await this.client!.leave(options.roomId);\n }\n\n async forgetRoom(options: { roomId: string }): Promise<void> {\n this.requireClient();\n await this.client!.forget(options.roomId);\n }\n\n // ── Messaging ─────────────────────────────────────────\n\n async sendMessage(options: SendMessageOptions): Promise<{ eventId: string }> {\n this.requireClient();\n\n const msgtype = options.msgtype ?? 'm.text';\n const mediaTypes = ['m.image', 'm.audio', 'm.video', 'm.file'];\n\n if (mediaTypes.includes(msgtype) && options.fileUri) {\n // Media message: upload file then send\n const response = await fetch(options.fileUri);\n const blob = await response.blob();\n const uploadRes = await this.client!.uploadContent(blob, {\n name: options.fileName,\n type: options.mimeType,\n });\n const mxcUrl = uploadRes.content_uri;\n const content: Record<string, unknown> = {\n msgtype,\n body: options.body || options.fileName || 'file',\n url: mxcUrl,\n info: {\n mimetype: options.mimeType,\n size: options.fileSize ?? blob.size,\n ...(options.duration !== undefined && { duration: options.duration }),\n ...(options.width !== undefined && { w: options.width }),\n ...(options.height !== undefined && { h: options.height }),\n },\n };\n const res = await this.client!.sendMessage(options.roomId, content as any);\n return { eventId: res.event_id };\n }\n\n // Text message\n const msgtypeMap = {\n 'm.text': MsgType.Text,\n 'm.notice': MsgType.Notice,\n 'm.emote': MsgType.Emote,\n } as const;\n const mappedType = msgtypeMap[msgtype as keyof typeof msgtypeMap] ?? MsgType.Text;\n const res = await this.client!.sendMessage(options.roomId, {\n msgtype: mappedType,\n body: options.body,\n });\n return { eventId: res.event_id };\n }\n\n async editMessage(options: EditMessageOptions): Promise<{ eventId: string }> {\n this.requireClient();\n\n const msgtype = options.msgtype ?? 'm.text';\n const mediaTypes = ['m.image', 'm.audio', 'm.video', 'm.file'];\n\n let newContent: Record<string, unknown>;\n\n if (mediaTypes.includes(msgtype) && options.fileUri) {\n const response = await fetch(options.fileUri);\n const blob = await response.blob();\n const uploadRes = await this.client!.uploadContent(blob, {\n name: options.fileName,\n type: options.mimeType,\n });\n newContent = {\n msgtype,\n body: options.newBody || options.fileName || 'file',\n url: uploadRes.content_uri,\n info: {\n mimetype: options.mimeType,\n size: options.fileSize ?? blob.size,\n ...(options.duration !== undefined && { duration: options.duration }),\n ...(options.width !== undefined && { w: options.width }),\n ...(options.height !== undefined && { h: options.height }),\n },\n };\n } else {\n newContent = {\n msgtype,\n body: options.newBody,\n };\n }\n\n const content: Record<string, unknown> = {\n ...newContent,\n body: `* ${options.newBody}`,\n 'm.new_content': newContent,\n 'm.relates_to': {\n rel_type: 'm.replace',\n event_id: options.eventId,\n },\n };\n const res = await this.client!.sendMessage(options.roomId, content as any);\n return { eventId: res.event_id };\n }\n\n async sendReply(options: SendReplyOptions): Promise<{ eventId: string }> {\n this.requireClient();\n\n const msgtype = options.msgtype ?? 'm.text';\n const mediaTypes = ['m.image', 'm.audio', 'm.video', 'm.file'];\n\n let content: Record<string, unknown>;\n\n if (mediaTypes.includes(msgtype) && options.fileUri) {\n // Media reply: upload file then send with reply relation\n const response = await fetch(options.fileUri);\n const blob = await response.blob();\n const uploadRes = await this.client!.uploadContent(blob, {\n name: options.fileName,\n type: options.mimeType,\n });\n content = {\n msgtype,\n body: options.body || options.fileName || 'file',\n url: uploadRes.content_uri,\n info: {\n mimetype: options.mimeType,\n size: options.fileSize ?? blob.size,\n ...(options.duration !== undefined && { duration: options.duration }),\n ...(options.width !== undefined && { w: options.width }),\n ...(options.height !== undefined && { h: options.height }),\n },\n 'm.relates_to': {\n 'm.in_reply_to': {\n event_id: options.replyToEventId,\n },\n },\n };\n } else {\n // Text reply\n content = {\n msgtype: MsgType.Text,\n body: options.body,\n 'm.relates_to': {\n 'm.in_reply_to': {\n event_id: options.replyToEventId,\n },\n },\n };\n }\n\n const res = await this.client!.sendMessage(options.roomId, content as any);\n return { eventId: res.event_id };\n }\n\n async getRoomMessages(\n options: { roomId: string; limit?: number; from?: string },\n ): Promise<{ events: MatrixEvent[]; nextBatch?: string }> {\n this.requireClient();\n\n const limit = options.limit ?? 20;\n const room = this.client!.getRoom(options.roomId);\n\n // If no explicit pagination token, return events from the synced timeline\n if (!options.from && room) {\n // Paginate backwards so we have enough events (initial sync may be small)\n try {\n await this.client!.scrollback(room, limit);\n } catch {\n // scrollback may fail if there's no more history\n }\n\n const timeline = room.getLiveTimeline();\n const timelineEvents = timeline.getEvents();\n // Filter out reactions and redactions before slicing — they're aggregated into parent events\n const displayableEvents = timelineEvents.filter((e) => {\n const t = e.getType();\n return t !== EventType.Reaction && t !== EventType.RoomRedaction;\n });\n const events: MatrixEvent[] = displayableEvents\n .slice(-limit)\n .map((e) => this.serializeEvent(e, options.roomId))\n .sort((a, b) => a.originServerTs - b.originServerTs);\n const backToken = timeline.getPaginationToken(Direction.Backward) ?? undefined;\n return { events, nextBatch: backToken };\n }\n\n // Paginate further back using the token\n const fromToken = options.from ?? null;\n const res = await this.client!.createMessagesRequest(\n options.roomId,\n fromToken,\n limit,\n Direction.Backward,\n );\n\n const events: MatrixEvent[] = (res.chunk ?? []).map((e) => ({\n eventId: e.event_id,\n roomId: options.roomId,\n senderId: e.sender,\n type: e.type,\n content: (e.content ?? {}) as Record<string, unknown>,\n originServerTs: e.origin_server_ts,\n }));\n\n return { events, nextBatch: res.end ?? undefined };\n }\n\n async markRoomAsRead(options: { roomId: string; eventId: string }): Promise<void> {\n this.requireClient();\n const room = this.client!.getRoom(options.roomId);\n if (room) {\n const event = room.findEventById(options.eventId);\n if (event) {\n await this.client!.sendReadReceipt(event);\n return;\n }\n }\n // Fallback to HTTP request if event not found locally\n await this.client!.setRoomReadMarkersHttpRequest(\n options.roomId,\n options.eventId,\n options.eventId,\n );\n }\n\n async refreshEventStatuses(options: { roomId: string; eventIds: string[] }): Promise<{ events: import('./definitions').MatrixEvent[] }> {\n this.requireClient();\n const room = this.client!.getRoom(options.roomId);\n if (!room) return { events: [] };\n const events: import('./definitions').MatrixEvent[] = [];\n for (const eid of options.eventIds) {\n const event = room.findEventById(eid);\n if (event) {\n events.push(this.serializeEvent(event, options.roomId));\n }\n }\n return { events };\n }\n\n // ── Redactions & Reactions ───────────────────────────────\n\n async redactEvent(options: { roomId: string; eventId: string; reason?: string }): Promise<void> {\n this.requireClient();\n await this.client!.redactEvent(options.roomId, options.eventId, undefined, {\n reason: options.reason,\n } as any);\n }\n\n async sendReaction(options: { roomId: string; eventId: string; key: string }): Promise<{ eventId: string }> {\n this.requireClient();\n const myUserId = this.client!.getUserId();\n\n // Check if the user already reacted with this key — if so, toggle off (redact)\n const room = this.client!.getRoom(options.roomId);\n if (room && myUserId) {\n try {\n const relations = room.relations.getChildEventsForEvent(\n options.eventId,\n RelationType.Annotation,\n EventType.Reaction,\n );\n if (relations) {\n const existing = relations.getRelations().find(\n (e) => e.getSender() === myUserId && e.getContent()?.['m.relates_to']?.key === options.key,\n );\n if (existing) {\n const existingId = existing.getId();\n if (existingId) {\n await this.client!.redactEvent(options.roomId, existingId);\n return { eventId: existingId };\n }\n }\n }\n } catch {\n // fall through to send\n }\n }\n\n const res = await this.client!.sendEvent(options.roomId, EventType.Reaction, {\n 'm.relates_to': {\n rel_type: RelationType.Annotation,\n event_id: options.eventId,\n key: options.key,\n },\n });\n return { eventId: res.event_id };\n }\n\n // ── Room Management ────────────────────────────────────\n\n async setRoomName(options: { roomId: string; name: string }): Promise<void> {\n this.requireClient();\n await this.client!.setRoomName(options.roomId, options.name);\n }\n\n async setRoomTopic(options: { roomId: string; topic: string }): Promise<void> {\n this.requireClient();\n await this.client!.setRoomTopic(options.roomId, options.topic);\n }\n\n async setRoomAvatar(options: { roomId: string; mxcUrl: string }): Promise<void> {\n this.requireClient();\n await this.client!.sendStateEvent(options.roomId, 'm.room.avatar' as any, { url: options.mxcUrl });\n }\n\n async inviteUser(options: { roomId: string; userId: string }): Promise<void> {\n this.requireClient();\n await this.client!.invite(options.roomId, options.userId);\n }\n\n async kickUser(options: { roomId: string; userId: string; reason?: string }): Promise<void> {\n this.requireClient();\n await this.client!.kick(options.roomId, options.userId, options.reason);\n }\n\n async banUser(options: { roomId: string; userId: string; reason?: string }): Promise<void> {\n this.requireClient();\n await this.client!.ban(options.roomId, options.userId, options.reason);\n }\n\n async unbanUser(options: { roomId: string; userId: string }): Promise<void> {\n this.requireClient();\n await this.client!.unban(options.roomId, options.userId);\n }\n\n // ── Typing ─────────────────────────────────────────────\n\n async sendTyping(options: { roomId: string; isTyping: boolean; timeout?: number }): Promise<void> {\n this.requireClient();\n await this.client!.sendTyping(options.roomId, options.isTyping, options.timeout ?? 30000);\n }\n\n // ── Media ──────────────────────────────────────────────\n\n async getMediaUrl(options: { mxcUrl: string }): Promise<{ httpUrl: string }> {\n this.requireClient();\n // Use the authenticated media endpoint (Matrix v1.11+)\n const mxcPath = options.mxcUrl.replace('mxc://', '');\n const baseUrl = this.client!.getHomeserverUrl().replace(/\\/$/, '');\n const accessToken = this.client!.getAccessToken();\n const httpUrl = `${baseUrl}/_matrix/client/v1/media/download/${mxcPath}?access_token=${accessToken}`;\n return { httpUrl };\n }\n\n async getThumbnailUrl(options: ThumbnailUrlOptions): Promise<{ httpUrl: string }> {\n this.requireClient();\n const mxcPath = options.mxcUrl.replace('mxc://', '');\n const baseUrl = this.client!.getHomeserverUrl().replace(/\\/$/, '');\n const accessToken = this.client!.getAccessToken();\n const method = options.method ?? 'scale';\n const httpUrl = `${baseUrl}/_matrix/client/v1/media/thumbnail/${mxcPath}?width=${options.width}&height=${options.height}&method=${method}&access_token=${accessToken}`;\n return { httpUrl };\n }\n\n async uploadContent(options: UploadContentOptions): Promise<UploadContentResult> {\n this.requireClient();\n const response = await fetch(options.fileUri);\n const blob = await response.blob();\n const res = await this.client!.uploadContent(blob, {\n name: options.fileName,\n type: options.mimeType,\n });\n return { contentUri: res.content_uri };\n }\n\n // ── Presence ───────────────────────────────────────────\n\n async setPresence(options: { presence: 'online' | 'offline' | 'unavailable'; statusMsg?: string }): Promise<void> {\n this.requireClient();\n await this.client!.setPresence({\n presence: options.presence,\n status_msg: options.statusMsg,\n });\n }\n\n async getPresence(options: { userId: string }): Promise<PresenceInfo> {\n this.requireClient();\n const user = this.client!.getUser(options.userId);\n return {\n presence: (user?.presence as PresenceInfo['presence']) ?? 'offline',\n statusMsg: user?.presenceStatusMsg ?? undefined,\n lastActiveAgo: user?.lastActiveAgo ?? undefined,\n };\n }\n\n // ── Device Management ──────────────────────────────────\n\n async getDevices(): Promise<{ devices: DeviceInfo[] }> {\n this.requireClient();\n const res = await this.client!.getDevices();\n const crypto = this.client!.getCrypto();\n const myUserId = this.client!.getUserId() ?? '';\n const devices: DeviceInfo[] = await Promise.all(\n (res.devices ?? []).map(async (d: any) => {\n let isCrossSigningVerified: boolean | undefined;\n if (crypto) {\n try {\n const status = await crypto.getDeviceVerificationStatus(myUserId, d.device_id);\n isCrossSigningVerified = status?.crossSigningVerified ?? false;\n } catch {\n // ignore — crypto may not be ready\n }\n }\n return {\n deviceId: d.device_id,\n displayName: d.display_name ?? undefined,\n lastSeenTs: d.last_seen_ts ?? undefined,\n lastSeenIp: d.last_seen_ip ?? undefined,\n isCrossSigningVerified,\n };\n }),\n );\n return { devices };\n }\n\n async deleteDevice(options: { deviceId: string; auth?: Record<string, unknown> }): Promise<void> {\n this.requireClient();\n await this.client!.deleteDevice(options.deviceId, options.auth as any);\n }\n\n // ── Push ──────────────────────────────────────────────\n\n async setPusher(options: PusherOptions): Promise<void> {\n this.requireClient();\n await this.client!.setPusher({\n pushkey: options.pushkey,\n kind: options.kind ?? undefined,\n app_id: options.appId,\n app_display_name: options.appDisplayName,\n device_display_name: options.deviceDisplayName,\n lang: options.lang,\n data: options.data,\n } as any);\n }\n\n // ── Encryption ──────────────────────────────────────────\n\n async initializeCrypto(): Promise<void> {\n this.requireClient();\n const cryptoOpts = { cryptoDatabasePrefix: 'matrix-js-sdk' };\n\n try {\n await this.client!.initRustCrypto(cryptoOpts);\n } catch (e: any) {\n // After logout + re-login the server issues a new deviceId, but the\n // shared IndexedDB crypto store still references the old one.\n // Delete the stale store and retry so crypto initialises cleanly.\n if (e?.message?.includes(\"account in the store doesn't match\")) {\n await this.deleteCryptoStore();\n await this.client!.initRustCrypto(cryptoOpts);\n } else {\n throw e;\n }\n }\n\n // Flush the initial /keys/query request that initRustCrypto enqueues.\n // Without this, any call to getIdentity (e.g. via getCrossSigningStatus)\n // will spin-wait and emit periodic WARN logs until sync processes it.\n const crypto = this.client!.getCrypto() as any;\n if (crypto?.outgoingRequestsManager?.doProcessOutgoingRequests) {\n await crypto.outgoingRequestsManager.doProcessOutgoingRequests();\n }\n }\n\n private async deleteCryptoStore(): Promise<void> {\n if (typeof indexedDB === 'undefined') return;\n try {\n const dbs = await indexedDB.databases();\n await Promise.all(\n dbs\n .filter((db) => db.name?.startsWith('matrix-js-sdk'))\n .map(\n (db) =>\n new Promise<void>((resolve) => {\n const req = indexedDB.deleteDatabase(db.name!);\n req.onsuccess = () => resolve();\n req.onerror = () => resolve();\n }),\n ),\n );\n } catch {\n // indexedDB.databases() not available in all environments\n }\n }\n\n async getEncryptionStatus(): Promise<EncryptionStatus> {\n this.requireClient();\n const crypto = this.client!.getCrypto();\n if (!crypto) {\n return {\n isCrossSigningReady: false,\n crossSigningStatus: { hasMaster: false, hasSelfSigning: false, hasUserSigning: false, isReady: false },\n isKeyBackupEnabled: false,\n isSecretStorageReady: false,\n };\n }\n\n const csReady = await crypto.isCrossSigningReady();\n const csStatus = await crypto.getCrossSigningStatus();\n const backupVersion = await crypto.getActiveSessionBackupVersion();\n\n // Use getSecretStorageStatus().defaultKeyId to check if secret storage was\n // set up at all, rather than isSecretStorageReady() which also checks that\n // cross-signing keys are stored (too strict for Phase 1).\n const ssStatus = await crypto.getSecretStorageStatus();\n const ssHasKey = ssStatus.defaultKeyId !== null;\n\n return {\n isCrossSigningReady: csReady,\n crossSigningStatus: {\n hasMaster: csStatus.publicKeysOnDevice,\n hasSelfSigning: csStatus.privateKeysCachedLocally.selfSigningKey,\n hasUserSigning: csStatus.privateKeysCachedLocally.userSigningKey,\n isReady: csReady,\n },\n isKeyBackupEnabled: backupVersion !== null,\n keyBackupVersion: backupVersion ?? undefined,\n isSecretStorageReady: ssHasKey,\n };\n }\n\n async bootstrapCrossSigning(): Promise<void> {\n const crypto = await this.ensureCrypto();\n await crypto.bootstrapCrossSigning({\n authUploadDeviceSigningKeys: async (makeRequest) => {\n // UIA flow: attempt with dummy auth, fall back to session-based retry\n try {\n await makeRequest({ type: 'm.login.dummy' });\n } catch (e: any) {\n const session = e?.data?.session;\n if (session) {\n await makeRequest({ type: 'm.login.dummy', session });\n } else {\n throw e;\n }\n }\n },\n });\n }\n\n async setupKeyBackup(): Promise<KeyBackupStatus> {\n const crypto = await this.ensureCrypto();\n await crypto.resetKeyBackup();\n const version = await crypto.getActiveSessionBackupVersion();\n return { exists: true, version: version ?? undefined, enabled: true };\n }\n\n async getKeyBackupStatus(): Promise<KeyBackupStatus> {\n this.requireClient();\n const crypto = this.requireCrypto();\n const version = await crypto.getActiveSessionBackupVersion();\n return {\n exists: version !== null,\n version: version ?? undefined,\n enabled: version !== null,\n };\n }\n\n async restoreKeyBackup(_options?: { recoveryKey?: string }): Promise<{ importedKeys: number }> {\n this.requireClient();\n const crypto = this.requireCrypto();\n\n const result = await crypto.restoreKeyBackup();\n return { importedKeys: result?.imported ?? 0 };\n }\n\n async setupRecovery(options?: {\n passphrase?: string;\n existingPassphrase?: string;\n }): Promise<RecoveryKeyInfo> {\n const crypto = await this.ensureCrypto();\n\n const keyInfo = await crypto.createRecoveryKeyFromPassphrase(options?.passphrase);\n // Pre-cache the new key bytes. secretStorageKeyId will be set by\n // cacheSecretStorageKey once bootstrapSecretStorage writes the new key\n // into SSSS and the SDK calls back.\n this.secretStorageKey = keyInfo.privateKey;\n this.secretStorageKeyId = undefined;\n\n // If the caller provides the same or old passphrase, keep it so\n // getSecretStorageKey can derive the key for the SDK.\n if (options?.passphrase) {\n this.recoveryPassphrase = options.passphrase;\n }\n // If the caller knows the OLD passphrase, keep it as fallbackPassphrase so\n // that getSecretStorageKey can decrypt the existing SSSS during\n // bootstrapSecretStorage's migration of cross-signing / backup secrets.\n if (options?.existingPassphrase) {\n this.fallbackPassphrase = options.existingPassphrase;\n }\n\n try {\n const bootstrapPromise = crypto.bootstrapSecretStorage({\n createSecretStorageKey: async () => keyInfo,\n setupNewSecretStorage: true,\n setupNewKeyBackup: true,\n });\n\n // Guard against SDK hanging when it can't retrieve the old SSSS key\n const timeoutPromise = new Promise<never>((_, reject) => {\n setTimeout(\n () => reject(new Error('bootstrapSecretStorage timed out — the old SSSS key could not be retrieved')),\n 30_000,\n );\n });\n\n await Promise.race([bootstrapPromise, timeoutPromise]);\n } finally {\n // Always clear transient crypto state so it doesn't bleed into subsequent calls.\n this.fallbackPassphrase = undefined;\n this.recoveryPassphrase = undefined;\n }\n\n return { recoveryKey: keyInfo.encodedPrivateKey ?? '' };\n }\n\n async isRecoveryEnabled(): Promise<{ enabled: boolean }> {\n const crypto = await this.ensureCrypto();\n const ready = await crypto.isSecretStorageReady();\n return { enabled: ready };\n }\n\n async recoverAndSetup(options: { recoveryKey?: string; passphrase?: string }): Promise<void> {\n const crypto = await this.ensureCrypto();\n\n // Derive/decode the secret storage key\n if (options.recoveryKey) {\n this.secretStorageKey = decodeRecoveryKey(options.recoveryKey);\n } else if (options.passphrase) {\n // Store passphrase — the getSecretStorageKey callback will derive\n // the key using the server's stored PBKDF2 params (salt, iterations)\n this.recoveryPassphrase = options.passphrase;\n this.secretStorageKey = undefined; // Clear any stale raw key\n } else {\n throw new Error('Either recoveryKey or passphrase must be provided');\n }\n\n // Load the backup decryption key from secret storage into the Rust crypto store.\n // This triggers the getSecretStorageKey callback.\n try {\n await crypto.loadSessionBackupPrivateKeyFromSecretStorage();\n } catch (e) {\n const msg = e instanceof Error ? e.message : String(e);\n if (msg.includes('decryption key does not match')) {\n // The passphrase is correct (SSSS decrypted fine), but the backup key\n // stored in SSSS doesn't match the server's current backup. This happens\n // when another client re-created the backup without updating SSSS, or\n // vice-versa. Auto-fix by creating a new backup that matches the SSSS key.\n // recoveryPassphrase / secretStorageKey are still set, so the\n // getSecretStorageKey callback can decrypt the existing SSSS.\n await crypto.bootstrapSecretStorage({\n setupNewKeyBackup: true,\n });\n await crypto.checkKeyBackupAndEnable();\n return;\n }\n // Different error — clear state and throw\n this.secretStorageKey = undefined;\n this.secretStorageKeyId = undefined;\n this.recoveryPassphrase = undefined;\n throw e;\n }\n\n // Now that the key is stored locally, activate backup in the running client\n await crypto.checkKeyBackupAndEnable();\n\n // Restore cross-signing trust for the current device.\n // With the SSSS key now available (via getSecretStorageKey callback),\n // bootstrapCrossSigning downloads the existing keys from secret storage\n // and self-verifies this device.\n try {\n await crypto.bootstrapCrossSigning({\n authUploadDeviceSigningKeys: async (makeRequest) => {\n try {\n await makeRequest({ type: 'm.login.dummy' });\n } catch (e: any) {\n const session = e?.data?.session;\n if (session) {\n await makeRequest({ type: 'm.login.dummy', session });\n } else {\n throw e;\n }\n }\n },\n });\n } catch {\n // Non-fatal: cross-signing restore failed, device remains unverified\n }\n }\n\n async verifyDevice(options: { deviceId: string }): Promise<void> {\n const crypto = await this.ensureCrypto();\n await crypto.crossSignDevice(options.deviceId);\n }\n\n async resetRecoveryKey(options?: { passphrase?: string }): Promise<RecoveryKeyInfo> {\n return this.setupRecovery(options);\n }\n\n async exportRoomKeys(options: { passphrase: string }): Promise<{ data: string }> {\n this.requireClient();\n const crypto = this.requireCrypto();\n const keys = await crypto.exportRoomKeysAsJson();\n // The exported JSON is not encrypted by default; for passphrase encryption\n // the caller should handle it, or we return the raw JSON\n void options.passphrase; // passphrase encryption is handled natively; on web we return raw\n return { data: keys };\n }\n\n async importRoomKeys(options: { data: string; passphrase: string }): Promise<{ importedKeys: number }> {\n this.requireClient();\n const crypto = this.requireCrypto();\n void options.passphrase; // passphrase decryption handled natively; on web we import raw\n await crypto.importRoomKeysAsJson(options.data);\n return { importedKeys: -1 }; // count not available from importRoomKeysAsJson\n }\n\n // ── Helpers ───────────────────────────────────────────\n\n private requireClient(): void {\n if (!this.client) {\n throw new Error('Not logged in. Call login() or loginWithToken() first.');\n }\n }\n\n private requireCrypto() {\n const crypto = this.client!.getCrypto();\n if (!crypto) {\n throw new Error('Crypto not initialized. Call initializeCrypto() first.');\n }\n return crypto;\n }\n\n private async ensureCrypto() {\n this.requireClient();\n if (!this.client!.getCrypto()) {\n await this.initializeCrypto();\n }\n return this.requireCrypto();\n }\n\n private persistSession(session: SessionInfo): void {\n localStorage.setItem(SESSION_KEY, JSON.stringify(session));\n }\n\n private serializeEvent(event: SdkMatrixEvent, fallbackRoomId?: string): MatrixEvent {\n const roomId = event.getRoomId() ?? fallbackRoomId ?? '';\n\n // Redacted events should be marked clearly\n if (event.isRedacted()) {\n return {\n eventId: event.getId() ?? '',\n roomId,\n senderId: event.getSender() ?? '',\n type: 'm.room.redaction',\n content: { body: 'Message deleted' },\n originServerTs: event.getTs(),\n };\n }\n\n const content = { ...(event.getContent() ?? {}) } as Record<string, unknown>;\n\n // Include aggregated reactions from the room's relations container\n const eventId = event.getId();\n if (eventId && roomId) {\n const room = this.client?.getRoom(roomId);\n if (room) {\n try {\n const relations = room.relations.getChildEventsForEvent(\n eventId,\n RelationType.Annotation,\n EventType.Reaction,\n );\n if (relations) {\n const sorted = relations.getSortedAnnotationsByKey();\n if (sorted && sorted.length > 0) {\n content.reactions = sorted.map(([key, events]) => ({\n key,\n count: events.size,\n senders: Array.from(events).map((e) => e.getSender()),\n }));\n }\n }\n } catch {\n // relations may not be available\n }\n }\n }\n\n // Determine delivery/read status\n let status: MatrixEvent['status'];\n const readBy: string[] = [];\n const myUserId = this.client?.getUserId();\n const sender = event.getSender();\n const room = eventId && roomId ? this.client?.getRoom(roomId) : undefined;\n\n if (sender === myUserId && eventId) {\n // Own message — check delivery status\n const evtStatus = event.status; // null = sent & echoed, 'sending', 'sent', etc.\n if (evtStatus === 'sending' || evtStatus === 'encrypting' || evtStatus === 'queued') {\n status = 'sending';\n } else {\n // Event is at least sent; check if anyone has read it\n if (room) {\n try {\n const members = room.getJoinedMembers();\n for (const member of members) {\n if (member.userId === myUserId) continue;\n if (room.hasUserReadEvent(member.userId, eventId)) {\n readBy.push(member.userId);\n }\n }\n } catch {\n // ignore errors\n }\n }\n status = readBy.length > 0 ? 'read' : 'sent';\n }\n } else if (eventId && room) {\n // Other's message — collect who has read it\n try {\n const members = room.getJoinedMembers();\n for (const member of members) {\n if (member.userId === sender) continue;\n if (room.hasUserReadEvent(member.userId, eventId)) {\n readBy.push(member.userId);\n }\n }\n } catch {\n // ignore\n }\n }\n\n // Include unsigned data (e.g. m.relations for edits, transaction_id for local echo)\n const unsignedData = event.getUnsigned?.();\n const unsigned = unsignedData && Object.keys(unsignedData).length > 0\n ? (unsignedData as Record<string, unknown>)\n : undefined;\n\n return {\n eventId: eventId ?? '',\n roomId,\n senderId: sender ?? '',\n type: event.getType(),\n content,\n originServerTs: event.getTs(),\n status,\n readBy: readBy.length > 0 ? readBy : undefined,\n unsigned,\n };\n }\n\n private serializeRoom(room: Room): RoomSummary {\n // Detect DM: check m.direct account data or guess from room state\n let isDirect = false;\n try {\n const directEvent = this.client?.getAccountData('m.direct' as any);\n if (directEvent) {\n const directContent = directEvent.getContent() as Record<string, string[]>;\n for (const roomIds of Object.values(directContent)) {\n if (roomIds.includes(room.roomId)) {\n isDirect = true;\n break;\n }\n }\n }\n } catch {\n // ignore\n }\n\n // Get avatar URL\n let avatarUrl: string | undefined;\n const avatarEvent = room.currentState.getStateEvents('m.room.avatar', '');\n if (avatarEvent) {\n const mxcUrl = avatarEvent.getContent()?.url as string | undefined;\n if (mxcUrl) {\n avatarUrl = mxcUrl;\n }\n }\n\n return {\n roomId: room.roomId,\n name: room.name,\n topic: room.currentState.getStateEvents('m.room.topic', '')?.getContent()?.topic ?? undefined,\n memberCount: room.getJoinedMemberCount(),\n isEncrypted: room.hasEncryptionStateEvent(),\n unreadCount: room.getUnreadNotificationCount() ?? 0,\n lastEventTs: room.getLastActiveTimestamp() || undefined,\n membership: room.getMyMembership() as RoomSummary['membership'],\n avatarUrl,\n isDirect,\n };\n }\n\n async searchUsers(options: {\n searchTerm: string;\n limit?: number;\n }): Promise<{ results: UserProfile[]; limited: boolean }> {\n this.requireClient();\n const resp = await this.client!.searchUserDirectory({\n term: options.searchTerm,\n limit: options.limit ?? 10,\n });\n return {\n results: resp.results.map((u: { user_id: string; display_name?: string; avatar_url?: string }) => ({\n userId: u.user_id,\n displayName: u.display_name,\n avatarUrl: u.avatar_url,\n })),\n limited: resp.limited,\n };\n }\n\n private mapSyncState(state: string | null): SyncState {\n switch (state) {\n case 'PREPARED':\n case 'SYNCING':\n case 'CATCHUP':\n case 'RECONNECTING':\n return 'SYNCING';\n case 'ERROR':\n return 'ERROR';\n case 'STOPPED':\n return 'STOPPED';\n default:\n return 'INITIAL';\n }\n }\n}\n"]}
|