From fea6581190d92407400428672d180dcbe31c8e60 Mon Sep 17 00:00:00 2001 From: Joel Tulloch Date: Thu, 12 Oct 2023 14:07:29 -0400 Subject: [PATCH 1/5] device session is persisted --- src/lib/chat/matrix-client.test.ts | 92 ++++++++++++++++++++++++++-- src/lib/chat/matrix-client.ts | 39 +++++++++--- src/lib/chat/session-storage.test.ts | 64 +++++++++++++++++++ src/lib/chat/session-storage.ts | 27 ++++++++ 4 files changed, 210 insertions(+), 12 deletions(-) create mode 100644 src/lib/chat/session-storage.test.ts create mode 100644 src/lib/chat/session-storage.ts diff --git a/src/lib/chat/matrix-client.test.ts b/src/lib/chat/matrix-client.test.ts index c1a038f0f..460133921 100644 --- a/src/lib/chat/matrix-client.test.ts +++ b/src/lib/chat/matrix-client.test.ts @@ -60,26 +60,108 @@ const getSdkClient = (sdkClient = {}) => ({ ...sdkClient, }); -const subject = (props = {}) => { +const subject = (props = {}, sessionStorage = {}) => { const allProps: any = { createClient: (_opts: any) => getSdkClient(), ...props, }; - return new MatrixClient(allProps); + const mockSessionStorage = { + get: () => ({ deviceId: '', accessToken: '', userId: '' }), + set: (_session) => undefined, + ...sessionStorage, + }; + + return new MatrixClient(allProps, mockSessionStorage); }; +function resolveWith(valueToResolve: T) { + let theResolve; + const promise = new Promise((resolve) => { + theResolve = async () => { + resolve(valueToResolve); + await new Promise((resolve) => setImmediate(resolve)); + await promise; + }; + }); + + return { resolve: theResolve, mock: () => promise }; +} + describe('matrix client', () => { describe('createclient', () => { - it('creates SDK client on connect', () => { + it('creates SDK client with existing session on connect', async () => { const sdkClient = getSdkClient(); const createClient = jest.fn(() => sdkClient); + const matrixSession = { + deviceId: 'abc123', + accessToken: 'token-4321', + userId: '@bob:zos-matrix', + }; - const client = subject({ createClient }); + const client = subject({ createClient }, { get: () => matrixSession }); client.connect(null, 'token'); - expect(createClient).toHaveBeenCalledWith(expect.objectContaining({ baseUrl: config.matrix.homeServerUrl })); + await new Promise((resolve) => setImmediate(resolve)); + + expect(createClient).toHaveBeenCalledWith({ + baseUrl: config.matrix.homeServerUrl, + ...matrixSession, + }); + }); + + it('logs in and creates SDK client with new session if none exists', async () => { + const matrixSession = { + deviceId: 'abc123', + accessToken: 'token-4321', + userId: '@bob:zos-matrix', + }; + + const { resolve, mock } = resolveWith({ + device_id: matrixSession.deviceId, + user_id: matrixSession.userId, + access_token: matrixSession.accessToken, + }); + + const createClient = jest.fn(() => getSdkClient({ login: mock })); + + const client = subject({ createClient }, { get: () => null }); + + client.connect(null, 'token'); + + await resolve(); + + expect(createClient).toHaveBeenNthCalledWith(2, { + baseUrl: config.matrix.homeServerUrl, + ...matrixSession, + }); + }); + + it('saves session if none exists', async () => { + const matrixSession = { + deviceId: 'abc123', + accessToken: 'token-4321', + userId: '@bob:zos-matrix', + }; + + const setSession = jest.fn(); + + const { resolve, mock } = resolveWith({ + device_id: matrixSession.deviceId, + user_id: matrixSession.userId, + access_token: matrixSession.accessToken, + }); + + const createClient = jest.fn(() => getSdkClient({ login: mock })); + + const client = subject({ createClient }, { get: () => null, set: setSession }); + + client.connect(null, 'token'); + + await resolve(); + + expect(setSession).toHaveBeenCalledWith(matrixSession); }); it('starts client on connect', async () => { diff --git a/src/lib/chat/matrix-client.ts b/src/lib/chat/matrix-client.ts index f5ab1bea7..ad28362bd 100644 --- a/src/lib/chat/matrix-client.ts +++ b/src/lib/chat/matrix-client.ts @@ -29,6 +29,7 @@ import { MemberNetworks } from '../../store/users/types'; import { ConnectionStatus, MembershipStateType } from './matrix/types'; import { getFilteredMembersForAutoComplete, setAsDM } from './matrix/utils'; import { uploadImage } from '../../store/channels-list/api'; +import { SessionStorage } from './session-storage'; export class MatrixClient implements IChatClient { private matrix: SDKMatrixClient = null; @@ -41,7 +42,7 @@ export class MatrixClient implements IChatClient { private connectionResolver: () => void; private connectionAwaiter: Promise; - constructor(private sdk = { createClient }) { + constructor(private sdk = { createClient }, private sessionStorage = new SessionStorage()) { this.addConnectionAwaiter(); } @@ -303,21 +304,45 @@ export class MatrixClient implements IChatClient { return (data) => console.log('Received Event', name, data); } - private async initializeClient(_userId: string, accessToken: string) { + private async getCredentials(accessToken: string) { + const credentials = this.sessionStorage.get(); + + if (credentials) { + return credentials; + } + + return await this.login(accessToken); + } + + private async login(token: string) { + const tempClient = this.sdk.createClient({ baseUrl: config.matrix.homeServerUrl }); + + const { user_id, device_id, access_token } = await tempClient.login('org.matrix.login.jwt', { token }); + + this.sessionStorage.set({ + userId: user_id, + deviceId: device_id, + accessToken: access_token, + }); + + return { accessToken: access_token, userId: user_id, deviceId: device_id }; + } + + private async initializeClient(_userId: string, ssoToken: string) { if (!this.matrix) { - this.matrix = this.sdk.createClient({ + const opts: any = { baseUrl: config.matrix.homeServerUrl, - }); + ...(await this.getCredentials(ssoToken)), + }; - const loginResult = await this.matrix.login('org.matrix.login.jwt', { token: accessToken }); + this.matrix = this.sdk.createClient(opts); - this.matrix.deviceId = loginResult.device_id; await this.matrix.initCrypto(); await this.matrix.startClient(); await this.waitForSync(); - return loginResult.user_id; + return opts.userId; } } diff --git a/src/lib/chat/session-storage.test.ts b/src/lib/chat/session-storage.test.ts new file mode 100644 index 000000000..ad479b17c --- /dev/null +++ b/src/lib/chat/session-storage.test.ts @@ -0,0 +1,64 @@ +import { SessionStorage } from './session-storage'; + +const setItem = jest.fn(); +const getItem = jest.fn(); + +describe('session storage', () => { + const subject = (mockLocalStorage = {}) => { + return new SessionStorage({ + getItem, + setItem, + ...mockLocalStorage, + } as any); + }; + + it('sets localStorage vars', async () => { + const matrixSession = { + deviceId: 'abc123', + accessToken: 'token-4321', + userId: '@bob:zos-matrix', + }; + + const client = subject(); + + client.set(matrixSession); + + expect(setItem).toHaveBeenCalledWith('mxz_device_id', 'abc123'); + expect(setItem).toHaveBeenCalledWith('mxz_access_token_abc123', 'token-4321'); + expect(setItem).toHaveBeenCalledWith('mxz_user_id', '@bob:zos-matrix'); + }); + + it('gets from localStorage vars', async () => { + const matrixSession = { + deviceId: 'abc123', + accessToken: 'token-4321', + userId: '@bob:zos-matrix', + }; + + const getItem = jest.fn((key) => { + return { + mxz_device_id: 'abc123', + mxz_access_token_abc123: 'token-4321', + mxz_user_id: '@bob:zos-matrix', + }[key]; + }); + + const client = subject({ getItem }); + + expect(client.get()).toEqual(matrixSession); + }); + + it('returns null if deviceId is not set', async () => { + const getItem = jest.fn((key) => { + return { + mxz_device_id: '', + mxz_access_token_abc123: 'token-4321', + mxz_user_id: '@bob:zos-matrix', + }[key]; + }); + + const client = subject({ getItem }); + + expect(client.get()).toBeNull(); + }); +}); diff --git a/src/lib/chat/session-storage.ts b/src/lib/chat/session-storage.ts new file mode 100644 index 000000000..6296d4a14 --- /dev/null +++ b/src/lib/chat/session-storage.ts @@ -0,0 +1,27 @@ +export interface ChatSession { + deviceId: string; + accessToken: string; + userId: string; +} + +export class SessionStorage { + constructor(private storage = localStorage) {} + + set(session: ChatSession) { + this.storage.setItem('mxz_device_id', session.deviceId); + this.storage.setItem(`mxz_access_token_${session.deviceId}`, session.accessToken); + this.storage.setItem('mxz_user_id', session.userId); + } + + get(): ChatSession { + const deviceId = this.storage.getItem('mxz_device_id'); + + if (!deviceId) return null; + + return { + deviceId, + accessToken: this.storage.getItem(`mxz_access_token_${deviceId}`), + userId: this.storage.getItem('mxz_user_id'), + }; + } +} From f9e8a39f491e2853abfeb18d012c55d5c464f2ef Mon Sep 17 00:00:00 2001 From: Joel Tulloch Date: Thu, 12 Oct 2023 15:09:17 -0400 Subject: [PATCH 2/5] encryption enabled for new rooms --- src/lib/chat/matrix-client.ts | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/src/lib/chat/matrix-client.ts b/src/lib/chat/matrix-client.ts index ad28362bd..d33a535be 100644 --- a/src/lib/chat/matrix-client.ts +++ b/src/lib/chat/matrix-client.ts @@ -155,6 +155,7 @@ export class MatrixClient implements IChatClient { const initial_state: any[] = [ { type: EventType.RoomGuestAccess, state_key: '', content: { guest_access: GuestAccess.Forbidden } }, + { type: EventType.RoomEncryption, state_key: '', content: { algorithm: 'm.megolm.v1.aes-sha2' } }, ]; if (coverUrl) { @@ -339,6 +340,11 @@ export class MatrixClient implements IChatClient { await this.matrix.initCrypto(); + // suppsedly the setter is deprecated, but the direct property set doesn't seem to work. + // this is hopefully only a short-term setting anyway, so just leaving for now. + // this.matrix.getCrypto().globalBlacklistUnverifiedDevices = false; + this.matrix.setGlobalErrorOnUnknownDevices(false); + await this.matrix.startClient(); await this.waitForSync(); From 1f94c7f847464d99bee10719dcb0fa7c32ea4842 Mon Sep 17 00:00:00 2001 From: Dale Fukami Date: Thu, 12 Oct 2023 14:26:15 -0600 Subject: [PATCH 3/5] React to Decrypted message events --- src/lib/chat/matrix-client.ts | 13 ++++++++++++- 1 file changed, 12 insertions(+), 1 deletion(-) diff --git a/src/lib/chat/matrix-client.ts b/src/lib/chat/matrix-client.ts index d33a535be..9e92dc2fa 100644 --- a/src/lib/chat/matrix-client.ts +++ b/src/lib/chat/matrix-client.ts @@ -260,7 +260,7 @@ export class MatrixClient implements IChatClient { } if (event.type === EventType.RoomMessage) { - this.events.receiveNewMessage(event.room_id, mapMatrixMessage(event, this.matrix) as any); + this.publishMessageEvent(event); } if (event.type === EventType.RoomCreate) { @@ -278,6 +278,13 @@ export class MatrixClient implements IChatClient { } }); + this.matrix.on(MatrixEventEvent.Decrypted, async (decryptedEvent: MatrixEvent) => { + const event = decryptedEvent.getEffectiveEvent(); + if (event.type === EventType.RoomMessage) { + this.publishMessageEvent(event); + } + }); + this.matrix.on(ClientEvent.AccountData, this.publishConversationListChange); this.matrix.on(ClientEvent.Event, this.publishUserPresenceChange); this.matrix.on(RoomEvent.Name, this.publishRoomNameChange); @@ -374,6 +381,10 @@ export class MatrixClient implements IChatClient { this.events.onUserJoinedChannel(this.mapChannel(this.matrix.getRoom(event.room_id))); } + private publishMessageEvent(event) { + this.events.receiveNewMessage(event.room_id, mapMatrixMessage(event, this.matrix) as any); + } + private publishConversationListChange = (event: MatrixEvent) => { if (event.getType() === EventType.Direct) { const content = event.getContent(); From 31ccd2ce27b2ee5f58165526087039a86d21248d Mon Sep 17 00:00:00 2001 From: Joel Tulloch Date: Thu, 12 Oct 2023 17:26:53 -0400 Subject: [PATCH 4/5] fix failing tests --- src/lib/chat/matrix-client.test.ts | 22 +++++++++++++++++++--- 1 file changed, 19 insertions(+), 3 deletions(-) diff --git a/src/lib/chat/matrix-client.test.ts b/src/lib/chat/matrix-client.test.ts index 460133921..56609442f 100644 --- a/src/lib/chat/matrix-client.test.ts +++ b/src/lib/chat/matrix-client.test.ts @@ -57,6 +57,7 @@ const getSdkClient = (sdkClient = {}) => ({ getRooms: jest.fn(), getAccountData: jest.fn(), getUser: jest.fn(), + setGlobalErrorOnUnknownDevices: () => undefined, ...sdkClient, }); @@ -66,7 +67,7 @@ const subject = (props = {}, sessionStorage = {}) => { ...props, }; - const mockSessionStorage = { + const mockSessionStorage: any = { get: () => ({ deviceId: '', accessToken: '', userId: '' }), set: (_session) => undefined, ...sessionStorage, @@ -429,9 +430,24 @@ describe('matrix client', () => { expect(createRoom).toHaveBeenCalledWith( expect.objectContaining({ - initial_state: [ + initial_state: expect.arrayContaining([ { type: 'm.room.guest_access', state_key: '', content: { guest_access: GuestAccess.Forbidden } }, - ], + ]), + }) + ); + }); + + it('creates encrypted room', async () => { + const createRoom = jest.fn().mockResolvedValue({ room_id: 'new-room-id' }); + const client = await subject({ createRoom }); + + await client.createConversation([{ userId: 'id', matrixId: '@somebody.else' }], null, null, null); + + expect(createRoom).toHaveBeenCalledWith( + expect.objectContaining({ + initial_state: expect.arrayContaining([ + { type: 'm.room.encryption', state_key: '', content: { algorithm: 'm.megolm.v1.aes-sha2' } }, + ]), }) ); }); From 819a310b9ab12f19efef86b26bc38e40bf326e0d Mon Sep 17 00:00:00 2001 From: Joel Tulloch Date: Thu, 12 Oct 2023 18:03:57 -0400 Subject: [PATCH 5/5] matrix session is cleared on disconnect --- src/lib/chat/matrix-client.test.ts | 46 ++++++++++++++++++++++++++++ src/lib/chat/matrix-client.ts | 6 +++- src/lib/chat/session-storage.test.ts | 13 ++++++++ src/lib/chat/session-storage.ts | 8 +++++ 4 files changed, 72 insertions(+), 1 deletion(-) diff --git a/src/lib/chat/matrix-client.test.ts b/src/lib/chat/matrix-client.test.ts index 56609442f..1967fc6c0 100644 --- a/src/lib/chat/matrix-client.test.ts +++ b/src/lib/chat/matrix-client.test.ts @@ -51,6 +51,7 @@ const getSdkClient = (sdkClient = {}) => ({ login: async () => ({}), initCrypto: async () => null, startClient: jest.fn(async () => undefined), + stopClient: jest.fn(), on: jest.fn((topic, callback) => { if (topic === 'sync') callback('PREPARED'); }), @@ -70,6 +71,7 @@ const subject = (props = {}, sessionStorage = {}) => { const mockSessionStorage: any = { get: () => ({ deviceId: '', accessToken: '', userId: '' }), set: (_session) => undefined, + clear: () => undefined, ...sessionStorage, }; @@ -90,6 +92,50 @@ function resolveWith(valueToResolve: T) { } describe('matrix client', () => { + describe('disconnect', () => { + it('stops client on disconnect', async () => { + const sdkClient = getSdkClient(); + const createClient = jest.fn(() => sdkClient); + const matrixSession = { + deviceId: 'abc123', + accessToken: 'token-4321', + userId: '@bob:zos-matrix', + }; + + const client = subject({ createClient }, { get: () => matrixSession }); + + // initializes underlying matrix client + await client.connect(null, 'token'); + + client.disconnect(); + + expect(sdkClient.stopClient).toHaveBeenCalledOnce(); + }); + + it('clears session storage on disconnect', async () => { + const sdkClient = getSdkClient(); + const createClient = jest.fn(() => sdkClient); + const matrixSession = { + deviceId: 'abc123', + accessToken: 'token-4321', + userId: '@bob:zos-matrix', + }; + + const clearSession = jest.fn(); + + const client = subject({ createClient }, { clear: clearSession, get: () => matrixSession }); + + // initializes underlying matrix client + await client.connect(null, 'token'); + + expect(clearSession).not.toHaveBeenCalled(); + + client.disconnect(); + + expect(clearSession).toHaveBeenCalledOnce(); + }); + }); + describe('createclient', () => { it('creates SDK client with existing session on connect', async () => { const sdkClient = getSdkClient(); diff --git a/src/lib/chat/matrix-client.ts b/src/lib/chat/matrix-client.ts index 9e92dc2fa..b8bdf9d49 100644 --- a/src/lib/chat/matrix-client.ts +++ b/src/lib/chat/matrix-client.ts @@ -63,7 +63,11 @@ export class MatrixClient implements IChatClient { return this.userId; } - disconnect: () => void; + disconnect() { + this.matrix.stopClient(); + this.sessionStorage.clear(); + } + reconnect: () => void; async getAccountData(eventType: string) { diff --git a/src/lib/chat/session-storage.test.ts b/src/lib/chat/session-storage.test.ts index ad479b17c..e499e7c0b 100644 --- a/src/lib/chat/session-storage.test.ts +++ b/src/lib/chat/session-storage.test.ts @@ -2,12 +2,14 @@ import { SessionStorage } from './session-storage'; const setItem = jest.fn(); const getItem = jest.fn(); +const removeItem = jest.fn(); describe('session storage', () => { const subject = (mockLocalStorage = {}) => { return new SessionStorage({ getItem, setItem, + removeItem, ...mockLocalStorage, } as any); }; @@ -28,6 +30,17 @@ describe('session storage', () => { expect(setItem).toHaveBeenCalledWith('mxz_user_id', '@bob:zos-matrix'); }); + it('removes localStorage vars on clear', async () => { + const getItem = jest.fn((key) => (key === 'mxz_device_id' ? 'abc123' : '')); + const client = subject({ getItem }); + + client.clear(); + + expect(removeItem).toHaveBeenCalledWith('mxz_device_id'); + expect(removeItem).toHaveBeenCalledWith('mxz_access_token_abc123'); + expect(removeItem).toHaveBeenCalledWith('mxz_user_id'); + }); + it('gets from localStorage vars', async () => { const matrixSession = { deviceId: 'abc123', diff --git a/src/lib/chat/session-storage.ts b/src/lib/chat/session-storage.ts index 6296d4a14..fe9827cc2 100644 --- a/src/lib/chat/session-storage.ts +++ b/src/lib/chat/session-storage.ts @@ -7,6 +7,14 @@ export interface ChatSession { export class SessionStorage { constructor(private storage = localStorage) {} + clear() { + const deviceId = this.storage.getItem('mxz_device_id'); + + this.storage.removeItem('mxz_device_id'); + this.storage.removeItem(`mxz_access_token_${deviceId}`); + this.storage.removeItem('mxz_user_id'); + } + set(session: ChatSession) { this.storage.setItem('mxz_device_id', session.deviceId); this.storage.setItem(`mxz_access_token_${session.deviceId}`, session.accessToken);