Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2022 Savoir-faire Linux Inc. |
| 3 | * |
| 4 | * This program is free software; you can redistribute it and/or modify |
| 5 | * it under the terms of the GNU Affero General Public License as |
| 6 | * published by the Free Software Foundation; either version 3 of the |
| 7 | * License, or (at your option) any later version. |
| 8 | * |
| 9 | * This program is distributed in the hope that it will be useful, |
| 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| 12 | * GNU Affero General Public License for more details. |
| 13 | * |
| 14 | * You should have received a copy of the GNU Affero General Public |
| 15 | * License along with this program. If not, see |
| 16 | * <https://www.gnu.org/licenses/>. |
| 17 | */ |
Misha Krieger-Raynauld | b933fbb | 2022-11-15 15:11:09 -0500 | [diff] [blame] | 18 | import { createRequire } from 'node:module'; |
| 19 | |
Issam E. Maghni | 0432cb7 | 2022-11-12 06:09:26 +0000 | [diff] [blame] | 20 | import { |
| 21 | AccountDetails, |
| 22 | AccountTextMessage, |
| 23 | ConversationMessage, |
| 24 | Message, |
| 25 | VolatileDetails, |
| 26 | WebSocketMessage, |
| 27 | WebSocketMessageType, |
| 28 | } from 'jami-web-common'; |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 29 | import log from 'loglevel'; |
Misha Krieger-Raynauld | 6f9c7ae | 2022-10-28 11:41:45 -0400 | [diff] [blame] | 30 | import { filter, firstValueFrom, map, Subject } from 'rxjs'; |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 31 | import { Service } from 'typedi'; |
| 32 | |
Misha Krieger-Raynauld | b933fbb | 2022-11-15 15:11:09 -0500 | [diff] [blame] | 33 | import { WebSocketServer } from '../websocket/websocket-server.js'; |
Misha Krieger-Raynauld | addd6fe | 2022-10-22 12:46:04 -0400 | [diff] [blame] | 34 | import { JamiSignal } from './jami-signal.js'; |
| 35 | import { |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 36 | AccountDetailsChanged, |
Misha Krieger-Raynauld | cb11bba | 2022-11-11 18:08:33 -0500 | [diff] [blame] | 37 | AccountMessageStatusChanged, |
| 38 | ContactAdded, |
| 39 | ContactRemoved, |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 40 | ConversationLoaded, |
| 41 | ConversationReady, |
| 42 | ConversationRemoved, |
Charlie | b62c678 | 2022-10-30 15:14:56 -0400 | [diff] [blame] | 43 | IncomingAccountMessage, |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 44 | KnownDevicesChanged, |
| 45 | MessageReceived, |
Misha Krieger-Raynauld | addd6fe | 2022-10-22 12:46:04 -0400 | [diff] [blame] | 46 | NameRegistrationEnded, |
| 47 | RegisteredNameFound, |
| 48 | RegistrationStateChanged, |
| 49 | VolatileDetailsChanged, |
| 50 | } from './jami-signal-interfaces.js'; |
Misha Krieger-Raynauld | bfed173 | 2022-11-01 20:49:35 -0400 | [diff] [blame] | 51 | import { JamiSwig, StringMap, stringMapToRecord, stringVectToArray, vectMapToRecordArray } from './jami-swig.js'; |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 52 | |
Misha Krieger-Raynauld | b933fbb | 2022-11-15 15:11:09 -0500 | [diff] [blame] | 53 | const require = createRequire(import.meta.url); |
| 54 | |
Misha Krieger-Raynauld | bfed173 | 2022-11-01 20:49:35 -0400 | [diff] [blame] | 55 | // TODO: Convert Records to interfaces and replace them in common/ (e.g. Contact) |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 56 | |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 57 | @Service() |
| 58 | export class Jamid { |
Misha Krieger-Raynauld | b933fbb | 2022-11-15 15:11:09 -0500 | [diff] [blame] | 59 | private jamiSwig: JamiSwig; |
| 60 | private usernamesToAccountIds = new Map<string, string>(); |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 61 | private readonly events; |
| 62 | |
Misha Krieger-Raynauld | b933fbb | 2022-11-15 15:11:09 -0500 | [diff] [blame] | 63 | constructor(private webSocketServer: WebSocketServer) { |
| 64 | this.jamiSwig = require('../../jamid.node') as JamiSwig; |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 65 | |
| 66 | // Setup signal handlers |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 67 | const handlers: Record<string, unknown> = {}; |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 68 | |
| 69 | // Add default handler for all signals |
| 70 | const createDefaultHandler = (signal: string) => { |
| 71 | return (...args: unknown[]) => log.warn('Unhandled', signal, args); |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 72 | }; |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 73 | for (const signal in JamiSignal) { |
| 74 | handlers[signal] = createDefaultHandler(signal); |
| 75 | } |
| 76 | |
| 77 | // Overwrite handlers for handled signals using RxJS Subjects, converting multiple arguments to objects |
| 78 | const onAccountsChanged = new Subject<void>(); |
| 79 | handlers.AccountsChanged = () => onAccountsChanged.next(); |
| 80 | |
| 81 | const onAccountDetailsChanged = new Subject<AccountDetailsChanged>(); |
| 82 | handlers.AccountDetailsChanged = (accountId: string, details: AccountDetails) => |
| 83 | onAccountDetailsChanged.next({ accountId, details }); |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 84 | |
| 85 | const onVolatileDetailsChanged = new Subject<VolatileDetailsChanged>(); |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 86 | handlers.VolatileDetailsChanged = (accountId: string, details: VolatileDetails) => |
| 87 | onVolatileDetailsChanged.next({ accountId, details }); |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 88 | |
| 89 | const onRegistrationStateChanged = new Subject<RegistrationStateChanged>(); |
| 90 | handlers.RegistrationStateChanged = (accountId: string, state: string, code: number, details: string) => |
| 91 | onRegistrationStateChanged.next({ accountId, state, code, details }); |
| 92 | |
| 93 | const onNameRegistrationEnded = new Subject<NameRegistrationEnded>(); |
| 94 | handlers.NameRegistrationEnded = (accountId: string, state: number, username: string) => |
| 95 | onNameRegistrationEnded.next({ accountId, state, username }); |
| 96 | |
| 97 | const onRegisteredNameFound = new Subject<RegisteredNameFound>(); |
| 98 | handlers.RegisteredNameFound = (accountId: string, state: number, address: string, username: string) => |
| 99 | onRegisteredNameFound.next({ accountId, state, address, username }); |
| 100 | |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 101 | const onKnownDevicesChanged = new Subject<KnownDevicesChanged>(); |
| 102 | handlers.KnownDevicesChanged = (accountId: string, devices: Record<string, string>) => |
| 103 | onKnownDevicesChanged.next({ accountId, devices }); |
| 104 | |
Charlie | b62c678 | 2022-10-30 15:14:56 -0400 | [diff] [blame] | 105 | const onIncomingAccountMessage = new Subject<IncomingAccountMessage>(); |
Misha Krieger-Raynauld | cb11bba | 2022-11-11 18:08:33 -0500 | [diff] [blame] | 106 | handlers.IncomingAccountMessage = (accountId: string, from: string, payload: Record<string, string>) => |
| 107 | onIncomingAccountMessage.next({ accountId, from, payload }); |
| 108 | |
| 109 | const onAccountMessageStatusChanged = new Subject<AccountMessageStatusChanged>(); |
| 110 | handlers.AccountMessageStatusChanged = (accountId: string, messageId: string, peer: string, state: number) => |
| 111 | onAccountMessageStatusChanged.next({ accountId, messageId, peer, state }); |
| 112 | |
| 113 | const onContactAdded = new Subject<ContactAdded>(); |
| 114 | handlers.ContactAdded = (accountId: string, contactId: string, confirmed: boolean) => |
| 115 | onContactAdded.next({ accountId, contactId, confirmed }); |
| 116 | |
| 117 | const onContactRemoved = new Subject<ContactRemoved>(); |
| 118 | handlers.ContactRemoved = (accountId: string, contactId: string, banned: boolean) => |
| 119 | onContactRemoved.next({ accountId, contactId, banned }); |
Charlie | b62c678 | 2022-10-30 15:14:56 -0400 | [diff] [blame] | 120 | |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 121 | const onConversationReady = new Subject<ConversationReady>(); |
| 122 | handlers.ConversationReady = (accountId: string, conversationId: string) => |
| 123 | onConversationReady.next({ accountId, conversationId }); |
| 124 | |
| 125 | const onConversationRemoved = new Subject<ConversationRemoved>(); |
| 126 | handlers.ConversationRemoved = (accountId: string, conversationId: string) => |
| 127 | onConversationRemoved.next({ accountId, conversationId }); |
| 128 | |
| 129 | const onConversationLoaded = new Subject<ConversationLoaded>(); |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 130 | handlers.ConversationLoaded = (id: number, accountId: string, conversationId: string, messages: Message[]) => |
| 131 | onConversationLoaded.next({ id, accountId, conversationId, messages }); |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 132 | |
| 133 | const onMessageReceived = new Subject<MessageReceived>(); |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 134 | handlers.MessageReceived = (accountId: string, conversationId: string, message: Message) => |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 135 | onMessageReceived.next({ accountId, conversationId, message }); |
| 136 | |
| 137 | // Expose all signals in an events object to allow other handlers to subscribe after jamiSwig.init() |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 138 | this.events = { |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 139 | onAccountsChanged: onAccountsChanged.asObservable(), |
| 140 | onAccountDetailsChanged: onAccountDetailsChanged.asObservable(), |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 141 | onVolatileDetailsChanged: onVolatileDetailsChanged.asObservable(), |
| 142 | onRegistrationStateChanged: onRegistrationStateChanged.asObservable(), |
| 143 | onNameRegistrationEnded: onNameRegistrationEnded.asObservable(), |
| 144 | onRegisteredNameFound: onRegisteredNameFound.asObservable(), |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 145 | onKnownDevicesChanged: onKnownDevicesChanged.asObservable(), |
Charlie | b62c678 | 2022-10-30 15:14:56 -0400 | [diff] [blame] | 146 | onIncomingAccountMessage: onIncomingAccountMessage.asObservable(), |
Misha Krieger-Raynauld | cb11bba | 2022-11-11 18:08:33 -0500 | [diff] [blame] | 147 | onAccountMessageStatusChanged: onAccountMessageStatusChanged.asObservable(), |
| 148 | onContactAdded: onContactAdded.asObservable(), |
| 149 | onContactRemoved: onContactRemoved.asObservable(), |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 150 | onConversationReady: onConversationReady.asObservable(), |
| 151 | onConversationRemoved: onConversationRemoved.asObservable(), |
| 152 | onConversationLoaded: onConversationLoaded.asObservable(), |
| 153 | onMessageReceived: onMessageReceived.asObservable(), |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 154 | }; |
| 155 | |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 156 | this.setupSignalHandlers(); |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 157 | |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 158 | // RxJS Subjects are used as signal handlers for the following reasons: |
| 159 | // 1. You cannot change event handlers after calling jamiSwig.init() |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 160 | // 2. You cannot specify multiple handlers for the same event |
| 161 | // 3. You cannot specify a default handler |
Misha Krieger-Raynauld | b6f1c32 | 2022-10-23 20:42:57 -0400 | [diff] [blame] | 162 | this.jamiSwig.init(handlers); |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 163 | } |
| 164 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 165 | stop(): void { |
Misha Krieger-Raynauld | b6f1c32 | 2022-10-23 20:42:57 -0400 | [diff] [blame] | 166 | this.jamiSwig.fini(); |
Misha Krieger-Raynauld | 62a0da9 | 2022-10-22 13:46:59 -0400 | [diff] [blame] | 167 | } |
| 168 | |
Misha Krieger-Raynauld | b6f1c32 | 2022-10-23 20:42:57 -0400 | [diff] [blame] | 169 | getVolatileAccountDetails(accountId: string): VolatileDetails { |
| 170 | return stringMapToRecord(this.jamiSwig.getVolatileAccountDetails(accountId)) as unknown as VolatileDetails; |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 171 | } |
| 172 | |
Misha Krieger-Raynauld | b6f1c32 | 2022-10-23 20:42:57 -0400 | [diff] [blame] | 173 | getAccountDetails(accountId: string): AccountDetails { |
| 174 | return stringMapToRecord(this.jamiSwig.getAccountDetails(accountId)) as unknown as AccountDetails; |
| 175 | } |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 176 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 177 | setAccountDetails(accountId: string, accountDetails: AccountDetails): void { |
simon | 43da57b | 2022-10-26 18:22:22 -0400 | [diff] [blame] | 178 | const accountDetailsStringMap: StringMap = new this.jamiSwig.StringMap(); |
Misha Krieger-Raynauld | b6f1c32 | 2022-10-23 20:42:57 -0400 | [diff] [blame] | 179 | for (const [key, value] of Object.entries(accountDetails)) { |
| 180 | accountDetailsStringMap.set(key, value); |
| 181 | } |
| 182 | this.jamiSwig.setAccountDetails(accountId, accountDetailsStringMap); |
| 183 | } |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 184 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 185 | async addAccount(details: Map<string, string | number | boolean>): Promise<string> { |
simon | 43da57b | 2022-10-26 18:22:22 -0400 | [diff] [blame] | 186 | const detailsStringMap: StringMap = new this.jamiSwig.StringMap(); |
Misha Krieger-Raynauld | b6f1c32 | 2022-10-23 20:42:57 -0400 | [diff] [blame] | 187 | |
| 188 | detailsStringMap.set('Account.type', 'RING'); |
| 189 | for (const [key, value] of details.entries()) { |
| 190 | detailsStringMap.set('Account.' + key, value.toString()); |
| 191 | } |
| 192 | |
| 193 | const accountId = this.jamiSwig.addAccount(detailsStringMap); |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 194 | return firstValueFrom( |
| 195 | this.events.onRegistrationStateChanged.pipe( |
Misha Krieger-Raynauld | 6f9c7ae | 2022-10-28 11:41:45 -0400 | [diff] [blame] | 196 | filter((value) => value.accountId === accountId), |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 197 | // TODO: is it the only state? |
Misha Krieger-Raynauld | 6f9c7ae | 2022-10-28 11:41:45 -0400 | [diff] [blame] | 198 | // TODO: Replace with string enum in common/ |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 199 | filter((value) => value.state === 'REGISTERED'), |
| 200 | map((value) => value.accountId) |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 201 | ) |
| 202 | ); |
| 203 | } |
| 204 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 205 | removeAccount(accountId: string): void { |
Misha Krieger-Raynauld | b6f1c32 | 2022-10-23 20:42:57 -0400 | [diff] [blame] | 206 | this.jamiSwig.removeAccount(accountId); |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 207 | } |
| 208 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 209 | getAccountIds(): string[] { |
Misha Krieger-Raynauld | b6f1c32 | 2022-10-23 20:42:57 -0400 | [diff] [blame] | 210 | return stringVectToArray(this.jamiSwig.getAccountList()); |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 211 | } |
| 212 | |
Charlie | 2bc0d67 | 2022-11-04 11:53:44 -0400 | [diff] [blame] | 213 | sendAccountTextMessage(from: string, to: string, message: string): void { |
Charlie | b62c678 | 2022-10-30 15:14:56 -0400 | [diff] [blame] | 214 | const messageStringMap: StringMap = new this.jamiSwig.StringMap(); |
Charlie | 2bc0d67 | 2022-11-04 11:53:44 -0400 | [diff] [blame] | 215 | messageStringMap.set('application/json', message); |
| 216 | this.jamiSwig.sendAccountTextMessage(from, to, messageStringMap); |
Charlie | b62c678 | 2022-10-30 15:14:56 -0400 | [diff] [blame] | 217 | } |
| 218 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 219 | // TODO: Add interface for returned type |
Misha Krieger-Raynauld | 6f9c7ae | 2022-10-28 11:41:45 -0400 | [diff] [blame] | 220 | async lookupUsername(username: string, accountId?: string) { |
| 221 | const hasRingNs = this.jamiSwig.lookupName(accountId || '', '', username); |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 222 | if (!hasRingNs) { |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 223 | throw new Error('Jami does not have NS'); |
| 224 | } |
Misha Krieger-Raynauld | 6f9c7ae | 2022-10-28 11:41:45 -0400 | [diff] [blame] | 225 | return firstValueFrom( |
| 226 | this.events.onRegisteredNameFound.pipe( |
| 227 | filter((value) => value.username === username), |
| 228 | map(({ accountId: _, ...response }) => response) // Remove accountId from response |
| 229 | ) |
| 230 | ); |
| 231 | } |
| 232 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 233 | // TODO: Add interface for returned type |
Misha Krieger-Raynauld | 6f9c7ae | 2022-10-28 11:41:45 -0400 | [diff] [blame] | 234 | async lookupAddress(address: string, accountId?: string) { |
| 235 | const hasRingNs = this.jamiSwig.lookupAddress(accountId || '', '', address); |
| 236 | if (!hasRingNs) { |
| 237 | throw new Error('Jami does not have NS'); |
| 238 | } |
| 239 | return firstValueFrom( |
| 240 | this.events.onRegisteredNameFound.pipe( |
| 241 | filter((value) => value.address === address), |
| 242 | map(({ accountId: _, ...response }) => response) // Remove accountId from response |
| 243 | ) |
| 244 | ); |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 245 | } |
| 246 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 247 | // TODO: Create enum for state and return that rather than a number |
| 248 | async registerUsername(accountId: string, username: string, password: string): Promise<number> { |
Misha Krieger-Raynauld | b6f1c32 | 2022-10-23 20:42:57 -0400 | [diff] [blame] | 249 | const hasRingNs = this.jamiSwig.registerName(accountId, password, username); |
| 250 | if (!hasRingNs) { |
Misha Krieger-Raynauld | b6f1c32 | 2022-10-23 20:42:57 -0400 | [diff] [blame] | 251 | throw new Error('Jami does not have NS'); |
| 252 | } |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 253 | return firstValueFrom( |
| 254 | this.events.onNameRegistrationEnded.pipe( |
| 255 | filter((value) => value.accountId === accountId), |
| 256 | map((value) => value.state) |
| 257 | ) |
| 258 | ); |
Misha Krieger-Raynauld | b6f1c32 | 2022-10-23 20:42:57 -0400 | [diff] [blame] | 259 | } |
| 260 | |
| 261 | getDevices(accountId: string): Record<string, string> { |
| 262 | return stringMapToRecord(this.jamiSwig.getKnownRingDevices(accountId)); |
| 263 | } |
| 264 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 265 | addContact(accountId: string, contactId: string): void { |
Misha Krieger-Raynauld | bfed173 | 2022-11-01 20:49:35 -0400 | [diff] [blame] | 266 | this.jamiSwig.addContact(accountId, contactId); |
| 267 | } |
| 268 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 269 | removeContact(accountId: string, contactId: string): void { |
Misha Krieger-Raynauld | bfed173 | 2022-11-01 20:49:35 -0400 | [diff] [blame] | 270 | this.jamiSwig.removeContact(accountId, contactId, false); |
| 271 | } |
| 272 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 273 | blockContact(accountId: string, contactId: string): void { |
Misha Krieger-Raynauld | bfed173 | 2022-11-01 20:49:35 -0400 | [diff] [blame] | 274 | this.jamiSwig.removeContact(accountId, contactId, true); |
| 275 | } |
| 276 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 277 | // TODO: Replace Record with interface |
Misha Krieger-Raynauld | bfed173 | 2022-11-01 20:49:35 -0400 | [diff] [blame] | 278 | getContacts(accountId: string): Record<string, string>[] { |
| 279 | return vectMapToRecordArray(this.jamiSwig.getContacts(accountId)); |
| 280 | } |
| 281 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 282 | // TODO: Replace Record with interface |
Misha Krieger-Raynauld | bfed173 | 2022-11-01 20:49:35 -0400 | [diff] [blame] | 283 | getContactDetails(accountId: string, contactId: string): Record<string, string> { |
| 284 | return stringMapToRecord(this.jamiSwig.getContactDetails(accountId, contactId)); |
| 285 | } |
| 286 | |
Misha Krieger-Raynauld | 153a148 | 2022-11-05 12:00:41 -0400 | [diff] [blame] | 287 | getDefaultModeratorUris(accountId: string): string[] { |
Misha Krieger-Raynauld | b6f1c32 | 2022-10-23 20:42:57 -0400 | [diff] [blame] | 288 | return stringVectToArray(this.jamiSwig.getDefaultModerators(accountId)); |
| 289 | } |
| 290 | |
Misha Krieger-Raynauld | 153a148 | 2022-11-05 12:00:41 -0400 | [diff] [blame] | 291 | addDefaultModerator(accountId: string, contactId: string): void { |
| 292 | this.jamiSwig.setDefaultModerator(accountId, contactId, true); |
| 293 | } |
| 294 | |
| 295 | removeDefaultModerator(accountId: string, contactId: string): void { |
| 296 | this.jamiSwig.setDefaultModerator(accountId, contactId, false); |
| 297 | } |
| 298 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 299 | getConversationIds(accountId: string): string[] { |
| 300 | return stringVectToArray(this.jamiSwig.getConversations(accountId)); |
| 301 | } |
| 302 | |
| 303 | // TODO: Replace Record with interface |
| 304 | getConversationInfos(accountId: string, conversationId: string): Record<string, string> { |
| 305 | return stringMapToRecord(this.jamiSwig.conversationInfos(accountId, conversationId)); |
| 306 | } |
| 307 | |
| 308 | // TODO: Replace Record with interface |
| 309 | getConversationMembers(accountId: string, conversationId: string): Record<string, string>[] { |
| 310 | return vectMapToRecordArray(this.jamiSwig.getConversationMembers(accountId, conversationId)); |
| 311 | } |
| 312 | |
| 313 | async getConversationMessages(accountId: string, conversationId: string, fromMessage?: string): Promise<Message[]> { |
| 314 | const requestId = this.jamiSwig.loadConversationMessages(accountId, conversationId, fromMessage || '', 32); |
| 315 | return firstValueFrom( |
| 316 | this.events.onConversationLoaded.pipe( |
| 317 | filter((value) => value.id === requestId), |
| 318 | map((value) => value.messages) |
| 319 | ) |
| 320 | ); |
| 321 | } |
| 322 | |
| 323 | sendConversationMessage(accountId: string, conversationId: string, message: string, replyTo?: string): void { |
| 324 | this.jamiSwig.sendMessage(accountId, conversationId, message, replyTo || ''); |
| 325 | } |
| 326 | |
Misha Krieger-Raynauld | 7f95933 | 2022-11-04 15:12:53 -0400 | [diff] [blame] | 327 | getCallIds(accountId: string): string[] { |
| 328 | return stringVectToArray(this.jamiSwig.getCallList(accountId)); |
| 329 | } |
| 330 | |
| 331 | // TODO: Replace Record with interface |
| 332 | getCallDetails(accountId: string, callId: string): Record<string, string> { |
| 333 | return stringMapToRecord(this.jamiSwig.getCallDetails(accountId, callId)); |
| 334 | } |
| 335 | |
Misha Krieger-Raynauld | b6f1c32 | 2022-10-23 20:42:57 -0400 | [diff] [blame] | 336 | getAccountIdFromUsername(username: string): string | undefined { |
| 337 | return this.usernamesToAccountIds.get(username); |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 338 | } |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 339 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 340 | private setupSignalHandlers(): void { |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 341 | this.events.onAccountsChanged.subscribe(() => { |
| 342 | log.debug('Received AccountsChanged'); |
| 343 | }); |
| 344 | |
| 345 | this.events.onAccountDetailsChanged.subscribe((signal) => { |
| 346 | log.debug('Received AccountsDetailsChanged', JSON.stringify(signal)); |
| 347 | }); |
| 348 | |
| 349 | this.events.onVolatileDetailsChanged.subscribe(({ accountId, details }) => { |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 350 | const username = details['Account.registeredName']; |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 351 | log.debug( |
| 352 | `Received VolatileDetailsChanged: {"accountId":"${accountId}",` + |
| 353 | `"details":{"Account.registeredName":"${username}", ...}}` |
| 354 | ); |
| 355 | // Keep map of usernames to account IDs |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 356 | if (username) { |
| 357 | this.usernamesToAccountIds.set(username, accountId); |
| 358 | } |
| 359 | }); |
| 360 | |
| 361 | this.events.onRegistrationStateChanged.subscribe((signal) => { |
| 362 | log.debug('Received RegistrationStateChanged:', JSON.stringify(signal)); |
| 363 | }); |
| 364 | |
| 365 | this.events.onNameRegistrationEnded.subscribe((signal) => { |
| 366 | log.debug('Received NameRegistrationEnded:', JSON.stringify(signal)); |
| 367 | }); |
| 368 | |
| 369 | this.events.onRegisteredNameFound.subscribe((signal) => { |
| 370 | log.debug('Received RegisteredNameFound:', JSON.stringify(signal)); |
| 371 | }); |
| 372 | |
| 373 | this.events.onKnownDevicesChanged.subscribe(({ accountId }) => { |
| 374 | log.debug(`Received KnownDevicesChanged: {"accountId":"${accountId}", ...}`); |
| 375 | }); |
| 376 | |
| 377 | this.events.onIncomingAccountMessage.subscribe((signal) => { |
| 378 | log.debug('Received IncomingAccountMessage:', JSON.stringify(signal)); |
Issam E. Maghni | 0432cb7 | 2022-11-12 06:09:26 +0000 | [diff] [blame] | 379 | const message: WebSocketMessage<any> = JSON.parse(signal.payload['application/json']); |
Charlie | 2bc0d67 | 2022-11-04 11:53:44 -0400 | [diff] [blame] | 380 | |
| 381 | if (message === undefined) { |
Misha Krieger-Raynauld | b933fbb | 2022-11-15 15:11:09 -0500 | [diff] [blame] | 382 | log.warn('Undefined account message'); |
| 383 | return; |
| 384 | } |
| 385 | |
| 386 | if (message.type === undefined || message.data === undefined) { |
| 387 | log.warn('Account message is not a valid WebSocketMessage (missing type or data fields)'); |
Charlie | 2bc0d67 | 2022-11-04 11:53:44 -0400 | [diff] [blame] | 388 | return; |
| 389 | } |
| 390 | |
| 391 | if (!Object.values(WebSocketMessageType).includes(message.type)) { |
Misha Krieger-Raynauld | b933fbb | 2022-11-15 15:11:09 -0500 | [diff] [blame] | 392 | log.warn(`Invalid WebSocket message type: ${message.type}`); |
Charlie | 2bc0d67 | 2022-11-04 11:53:44 -0400 | [diff] [blame] | 393 | return; |
| 394 | } |
| 395 | |
Issam E. Maghni | 0432cb7 | 2022-11-12 06:09:26 +0000 | [diff] [blame] | 396 | const data: AccountTextMessage<unknown> = { |
| 397 | from: signal.from, |
| 398 | to: signal.accountId, |
| 399 | message: message.data.message, |
| 400 | }; |
Issam E. Maghni | 0432cb7 | 2022-11-12 06:09:26 +0000 | [diff] [blame] | 401 | |
Misha Krieger-Raynauld | b933fbb | 2022-11-15 15:11:09 -0500 | [diff] [blame] | 402 | this.webSocketServer.send(signal.accountId, message.type, data); |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 403 | }); |
| 404 | |
Misha Krieger-Raynauld | cb11bba | 2022-11-11 18:08:33 -0500 | [diff] [blame] | 405 | this.events.onAccountMessageStatusChanged.subscribe((signal) => { |
| 406 | log.debug('Received AccountMessageStatusChanged:', JSON.stringify(signal)); |
| 407 | }); |
| 408 | |
| 409 | this.events.onContactAdded.subscribe((signal) => { |
| 410 | log.debug('Received ContactAdded:', JSON.stringify(signal)); |
| 411 | }); |
| 412 | |
| 413 | this.events.onContactRemoved.subscribe((signal) => { |
| 414 | log.debug('Received ContactRemoved:', JSON.stringify(signal)); |
| 415 | }); |
| 416 | |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 417 | this.events.onConversationReady.subscribe((signal) => { |
| 418 | log.debug('Received ConversationReady:', JSON.stringify(signal)); |
| 419 | }); |
| 420 | |
| 421 | this.events.onConversationRemoved.subscribe((signal) => { |
| 422 | log.debug('Received ConversationRemoved:', JSON.stringify(signal)); |
| 423 | }); |
| 424 | |
Misha Krieger-Raynauld | 8a381da | 2022-11-03 17:37:51 -0400 | [diff] [blame] | 425 | this.events.onConversationLoaded.subscribe(({ id, accountId, conversationId }) => { |
| 426 | log.debug( |
| 427 | `Received ConversationLoaded: {"id":"${id}","accountId":"${accountId}",` + |
| 428 | `"conversationId":"${conversationId}","messages":[...]}` |
| 429 | ); |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 430 | }); |
| 431 | |
| 432 | this.events.onMessageReceived.subscribe((signal) => { |
| 433 | log.debug('Received MessageReceived:', JSON.stringify(signal)); |
Issam E. Maghni | 0432cb7 | 2022-11-12 06:09:26 +0000 | [diff] [blame] | 434 | const data: ConversationMessage = { |
| 435 | conversationId: signal.conversationId, |
| 436 | message: signal.message, |
| 437 | }; |
Misha Krieger-Raynauld | b933fbb | 2022-11-15 15:11:09 -0500 | [diff] [blame] | 438 | this.webSocketServer.send(signal.accountId, WebSocketMessageType.ConversationMessage, data); |
Misha Krieger-Raynauld | 68a9b56 | 2022-10-28 19:47:46 -0400 | [diff] [blame] | 439 | }); |
| 440 | } |
Issam E. Maghni | f796a09 | 2022-10-09 20:25:26 +0000 | [diff] [blame] | 441 | } |