blob: 00583d255cd7e605307107c577329e96fb1686b3 [file] [log] [blame]
Adrien Béraud6ecaa402021-04-06 17:37:25 -04001/*
2 * Copyright (c) 2017-2021 Savoir-faire Linux Inc.
3 *
4 * Author: Adrien Béraud <adrien.beraud@savoirfairelinux.com>
5 * Author: Asad Salman <me@asad.co>
6 *
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 3 of the License, or
10 * (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program. If not, see <https://www.gnu.org/licenses/>.
19 */
Adrien Béraude74741b2021-04-19 13:22:54 -040020"use strict"
Adrien Béraud947e8792021-04-15 18:32:44 -040021
Adrien Béraude74741b2021-04-19 13:22:54 -040022import Account from './model/Account.js'
23import Conversation from './model/Conversation.js'
24import { createRequire } from 'module';
25const require = createRequire(import.meta.url);
Adrien Béraud6ecaa402021-04-06 17:37:25 -040026
Adrien Béraud6ecaa402021-04-06 17:37:25 -040027class JamiDaemon {
28 constructor() {
29 this.accounts = []
30 this.dring = require("./dring.node")
31 this.dring.init({
32 "AccountsChanged": () => {
33 console.log("AccountsChanged")
34 const newAccounts = []
Adrien Béraud35e7d7c2021-04-13 03:28:39 -040035 JamiDaemon.vectToJs(this.dring.getAccountList()).forEach(accountId => {
Adrien Béraud6ecaa402021-04-06 17:37:25 -040036 for (const account in this.accounts) {
37 if (account.id === accountId) {
38 newAccounts.push(account)
39 return
40 }
41 }
42 newAccounts.push(new Account(accountId,
Adrien Béraud35e7d7c2021-04-13 03:28:39 -040043 JamiDaemon.mapToJs(this.dring.getAccountDetails(accountId)),
44 JamiDaemon.mapToJs(this.dring.getVolatileAccountDetails(accountId))
Adrien Béraud0cb76c92021-04-07 19:59:08 -040045 ))
Adrien Béraud6ecaa402021-04-06 17:37:25 -040046 })
47 this.accounts = newAccounts
48 },
49 "AccountDetailsChanged": (accountId, details) => {
50 console.log(`AccountDetailsChanged ${accountId}`)
51 const account = this.getAccount(accountId)
52 if (!account) {
53 console.log(`Unknown account ${accountId}`)
54 return
55 }
56 account.details = details
57 },
58 "VolatileDetailsChanged": (accountId, details) => {
59 console.log(`VolatileDetailsChanged ${accountId}`)
60 const account = this.getAccount(accountId)
61 if (!account) {
62 console.log(`Unknown account ${accountId}`)
63 return
64 }
65 account.volatileDetails = details
66 },
67 "IncomingAccountMessage": (accountId, from, message) => {
68 console.log(`Received message: ${accountId} ${from} ${message["text/plain"]}`)
69/*
70 if (parser.validate(message["text/plain"]) === true) {
Adrien Béraude74741b2021-04-19 13:22:54 -040071 console.log(message["text/plain"])
Adrien Béraud6ecaa402021-04-06 17:37:25 -040072 } else {
73
Adrien Béraude74741b2021-04-19 13:22:54 -040074 user = connectedUsers[accountId]
Adrien Béraud6ecaa402021-04-06 17:37:25 -040075 console.log(user.socketId)
Adrien Béraude74741b2021-04-19 13:22:54 -040076 io.to(user.socketId).emit('receivedMessage', message["text/plain"])
77 //io.emit('receivedMessage', message["text/plain"])
Adrien Béraud6ecaa402021-04-06 17:37:25 -040078 }*/
79 },
80 "RegistrationStateChanged": (accountId, state, /*int*/ code, detail) => {
81 const account = this.getAccount(accountId)
82 if (!account) {
83 console.log(`Unknown account ${accountId}`)
84 return
85 }
86 account.registrationState = state
87 console.log("RegistrationStateChanged: " + accountId + " " + state + " " + code + " " + detail)
88 if (state === "REGISTERED") {
89 /*if (tempAccounts[accountId]) {
90
91 const ctx = tempAccounts[accountId]
92 ctx.newUser.accountId = accountId
93 ctx.newUser.jamiId = jami.dring.getAccountDetails(accountId).get("Account.username")
94 //connectedUsers[accountId] = ctx.newUser
95 ctx.done(null, ctx.newUser)
96 delete tempAccounts[accountId]
97 }*/
98 } else if (state === "ERROR_AUTH") {
99 //done(null, false)
100 //remove account
101 }
102 },
103 "RegisteredNameFound": (accountId, state, address, name) => {
Adrien Béraud0cb76c92021-04-07 19:59:08 -0400104 console.log(`RegisteredNameFound: ${accountId} ${state} ${address} ${name}`)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400105 const account = this.getAccount(accountId)
106 if (!account) {
107 console.log(`Unknown account ${accountId}`)
108 return
109 }
110 if (state == 0) {
111 const contact = account.getContactFromCache(address)
Adrien Béraud150b4782021-04-21 19:40:59 -0400112 if (!contact.isRegisteredNameResolved())
113 contact.setRegisteredName(name)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400114 }
Adrien Béraude74741b2021-04-19 13:22:54 -0400115 let index = account.lookups.length - 1
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400116 while (index >= 0) {
117 const lookup = account.lookups[index]
118 if ((lookup.address && lookup.address === address) || (lookup.name && lookup.name === name)) {
119 lookup.resolve({address, name, state})
Adrien Béraude74741b2021-04-19 13:22:54 -0400120 account.lookups.splice(index, 1)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400121 }
Adrien Béraude74741b2021-04-19 13:22:54 -0400122 index -= 1
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400123 }
Adrien Béraud0cb76c92021-04-07 19:59:08 -0400124 },
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400125 "ConversationReady": (accountId, conversationId) => {
Adrien Béraud0cb76c92021-04-07 19:59:08 -0400126 console.log(`conversationReady: ${accountId} ${conversationId}`)
127 const account = this.getAccount(accountId)
128 if (!account) {
129 console.log(`Unknown account ${accountId}`)
130 return
131 }
132 let conversation = account.getConversation(conversationId)
133 if (!conversation) {
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400134 const members = JamiDaemon.vectMapToJs(this.dring.getConversationMembers(accountId, conversationId))
135 members.forEach(member => member.contact = account.getContactFromCache(member.uri))
136 conversation = new Conversation(conversationId, accountId, members)
Adrien Béraud0cb76c92021-04-07 19:59:08 -0400137 account.addConversation(conversation)
138 }
139 },
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400140 "ConversationRemoved": (accountId, conversationId) => {
Adrien Béraud0cb76c92021-04-07 19:59:08 -0400141 console.log(`conversationRemoved: ${accountId} ${conversationId}`)
142 const account = this.getAccount(accountId)
143 if (!account) {
144 console.log(`Unknown account ${accountId}`)
145 return
146 }
147 account.removeConversation(conversationId)
148 },
Adrien Béraud5e9e19b2021-04-22 01:38:53 -0400149 "ConversationLoaded": (accountId, conversationId, messages) => {
Adrien Béraud0cb76c92021-04-07 19:59:08 -0400150 console.log(`conversationLoaded: ${accountId} ${conversationId}`)
Adrien Béraud5e9e19b2021-04-22 01:38:53 -0400151 console.log(messages)
Adrien Béraud0cb76c92021-04-07 19:59:08 -0400152 const account = this.getAccount(accountId)
153 if (!account) {
154 console.log(`Unknown account ${accountId}`)
155 return
156 }
157 const conversation = account.getConversation(conversationId)
Adrien Béraud5e9e19b2021-04-22 01:38:53 -0400158 if (conversation) {
159 conversation.addLoadedMessages(message)
160 }
161 },
162 "MessageReceived": (accountId, conversationId, message) => {
163 console.log(`messageReceived: ${accountId} ${conversationId}`)
164 console.log(message)
165 const account = this.getAccount(accountId)
166 if (!account) {
167 console.log(`Unknown account ${accountId}`)
168 return
169 }
170 const conversation = account.getConversation(conversationId)
171 if (conversation) {
Adrien Béraud0cb76c92021-04-07 19:59:08 -0400172 conversation.addMessage(message)
173 }
174 },
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400175 "ConversationRequestReceived": (accountId, conversationId, request) => {
Adrien Béraud0cb76c92021-04-07 19:59:08 -0400176 console.log(`conversationRequestReceived: ${accountId} ${conversationId}`)
177 const account = this.getAccount(accountId)
178 if (!account) {
179 console.log(`Unknown account ${accountId}`)
180 return
181 }
182 },
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400183 "ConversationMemberEvent": (accountId, conversationId, member, event) => {
Adrien Béraud0cb76c92021-04-07 19:59:08 -0400184 console.log(`conversationMemberEvent: ${accountId} ${conversationId}`)
185 const account = this.getAccount(accountId)
186 if (!account) {
187 console.log(`Unknown account ${accountId}`)
188 return
189 }
190 },
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400191 "OnConversationError": (accountId, conversationId, code, what) => {
Adrien Béraud0cb76c92021-04-07 19:59:08 -0400192 console.log(`onConversationError: ${accountId} ${conversationId}`)
193 const account = this.getAccount(accountId)
194 if (!account) {
195 console.log(`Unknown account ${accountId}`)
196 return
197 }
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400198 }
199 })
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400200
201 JamiDaemon.vectToJs(this.dring.getAccountList()).forEach(accountId => {
202 const account = new Account(accountId,
203 JamiDaemon.mapToJs(this.dring.getAccountDetails(accountId)),
204 JamiDaemon.mapToJs(this.dring.getVolatileAccountDetails(accountId))
205 )
206 JamiDaemon.vectToJs(this.dring.getConversations(accountId)).forEach(conversationId => {
207 const members = JamiDaemon.vectMapToJs(this.dring.getConversationMembers(accountId, conversationId))
Adrien Béraud150b4782021-04-21 19:40:59 -0400208 members.forEach(member => {
209 member.contact = account.getContactFromCache(member.uri)
210 if (!member.contact.isRegisteredNameResolved()) {
211 if (!member.uri) return
212 console.log(`lookupAddress ${accountId} ${member.uri}`)
213 member.contact.setRegisteredName(new Promise((resolve, reject) =>
214 account.lookups.push({address: member.uri, resolve, reject})
215 ).then(result => {
216 if (result.state == 0)
217 return result.name
218 else if (result.state == 1)
219 return undefined
220 else
221 return null
222 }))
223 this.dring.lookupAddress(accountId, "", member.uri)
224 }
225 })
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400226 const conversation = new Conversation(conversationId, accountId, members)
227 account.addConversation(conversation)
228 })
229 this.accounts.push(account)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400230 })
231 }
232
233 addAccount(account) {
234 const params = accountDetailsToNative(account)
235 params.set("Account.type", "RING")
236 return this.dring.addAccount(params)
237 }
238 getAccount(accountId) {
239 for (let i = 0; i < this.accounts.length; i++) {
240 const account = this.accounts[i]
241 if (account.getId() === accountId)
242 return account
243 }
244 return undefined
245 }
246 getAccountList() {
247 return this.accounts
248 }
249 /*getAccountDetails(accountId) {
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400250 return this.mapToJs(this.dring.getAccountDetails(accountId))
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400251 }*/
252 setAccountDetails(accountId, details) {
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400253 this.dring.setAccountDetails(accountId, mapToNative(details))
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400254 }
255 getAudioOutputDeviceList() {
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400256 return JamiDaemon.vectToJs(this.dring.getAudioOutputDeviceList())
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400257 }
258 getVolume(deviceName) {
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400259 return this.dring.getVolume(deviceName)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400260 }
261 setVolume(deviceName, volume) {
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400262 return this.dring.setVolume(deviceName, volume)
263 }
264
265 lookupName(accountId, name) {
266 const p = new Promise((resolve, reject) => {
267 const account = this.getAccount(accountId)
268 if (!account) {
269 reject(new Error("Can't find account"))
270 } else {
271 account.lookups.push({name, resolve, reject})
272 }
273 })
274 this.dring.lookupName(accountId, "", name)
275 return p
276 }
277
278 lookupAddress(accountId, address) {
279 console.log(`lookupAddress ${accountId} ${address}`)
280 const p = new Promise((resolve, reject) => {
281 const account = this.getAccount(accountId)
282 if (!account) {
283 reject(new Error("Can't find account"))
284 } else {
285 account.lookups.push({address, resolve, reject})
286 }
287 })
288 this.dring.lookupAddress(accountId, "", address)
289 return p
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400290 }
291
292 stop() {
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400293 this.dring.fini()
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400294 }
295
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400296 addContact(accountId, contactId) {
297 this.dring.addContact(accountId, contactId)
298 const details = JamiDaemon.mapToJs(this.dring.getContactDetails(accountId, contactId))
299 if (details.conversationId) {
300 const account = this.getAccount(accountId)
301 if (account) {
302 let conversation = account.getConversation(details.conversationId)
303 if (!conversation) {
304 const members = JamiDaemon.vectMapToJs(this.dring.getConversationMembers(accountId, details.conversationId))
305 members.forEach(member => member.contact = account.getContactFromCache(member.uri))
306 conversation = new Conversation(details.conversationId, accountId, members)
307 account.addConversation(conversation)
308 }
309 }
310 }
311 return details
312 }
313
Adrien Béraud150b4782021-04-21 19:40:59 -0400314 getDefaultModerators(accountId) {
315 const account = this.getAccount(accountId)
316 if (!account) {
317 console.log(`Unknown account ${accountId}`)
318 return {}
319 }
320 return JamiDaemon.vectToJs(this.dring.getDefaultModerators(accountId))
321 .map(contactId => account.getContactFromCache(contctId))
322 }
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400323
Adrien Béraud150b4782021-04-21 19:40:59 -0400324 setDefaultModerators(accountId, moderators) {
325
326 }
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400327
Adrien Béraud5e9e19b2021-04-22 01:38:53 -0400328 sendMessage(accountId, conversationId, message) {
329 this.dring.sendMessage(accountId, conversationId, message, "")
330 }
331
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400332 boolToStr(bool) {
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400333 return bool ? "true" : "false"
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400334 }
335
336 accountDetailsToNative(account) {
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400337 const params = new this.dring.StringMap()
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400338 if (account.managerUri)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400339 params.set("Account.managerUri", account.managerUri)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400340 if (account.managerUsername)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400341 params.set("Account.managerUsername", account.managerUsername)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400342 if (account.archivePassword) {
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400343 params.set("Account.archivePassword", account.archivePassword)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400344 } else {
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400345 console.log("archivePassword required")
Adrien Béraude74741b2021-04-19 13:22:54 -0400346 return
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400347 }
348 if (account.alias)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400349 params.set("Account.alias", account.alias)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400350 if (account.displayName)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400351 params.set("Account.displayName", account.displayName)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400352 if (account.enable)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400353 params.set("Account.enable", this.boolToStr(account.enable))
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400354 if (account.autoAnswer)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400355 params.set("Account.autoAnswer", this.boolToStr(account.autoAnswer))
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400356 if (account.ringtonePath)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400357 params.set("Account.ringtonePath", account.ringtonePath)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400358 if (account.ringtoneEnabled)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400359 params.set("Account.ringtoneEnabled", this.boolToStr(account.ringtoneEnabled))
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400360 if (account.videoEnabled)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400361 params.set("Account.videoEnabled", this.boolToStr(account.videoEnabled))
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400362 if (account.useragent) {
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400363 params.set("Account.useragent", account.useragent)
364 params.set("Account.hasCustomUserAgent", "TRUE")
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400365 } else {
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400366 params.set("Account.hasCustomUserAgent", "FALSE")
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400367 }
368 if (account.audioPortMin)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400369 params.set("Account.audioPortMin", account.audioPortMin)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400370 if (account.audioPortMax)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400371 params.set("Account.audioPortMax", account.audioPortMax)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400372 if (account.videoPortMin)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400373 params.set("Account.videoPortMin", account.videoPortMin)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400374 if (account.videoPortMax)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400375 params.set("Account.videoPortMax", account.videoPortMax)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400376 if (account.localInterface)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400377 params.set("Account.localInterface", account.localInterface)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400378 if (account.publishedSameAsLocal)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400379 params.set("Account.publishedSameAsLocal", this.boolToStr(account.publishedSameAsLocal))
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400380 if (account.localPort)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400381 params.set("Account.localPort", account.localPort)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400382 if (account.publishedPort)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400383 params.set("Account.publishedPort", account.publishedPort)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400384 if (account.publishedAddress)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400385 params.set("Account.publishedAddress", account.publishedAddress)
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400386 if (account.upnpEnabled)
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400387 params.set("Account.upnpEnabled", this.boolToStr(account.upnpEnabled))
388 return params
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400389 }
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400390 static vectToJs(vect) {
391 const len = vect.size()
392 const outputArr = new Array(len)
393 for (let i = 0; i < len; i++)
394 outputArr[i] = vect.get(i)
395 return outputArr
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400396 }
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400397 static mapToJs(m) {
398 const outputObj = {}
399 JamiDaemon.vectToJs(m.keys())
400 .forEach(k => outputObj[k] = m.get(k))
401 return outputObj
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400402 }
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400403 static vectMapToJs(vectMap) {
404 const len = vectMap.size()
405 const outputArr = new Array(len)
406 for (let i = 0; i < len; i++)
407 outputArr[i] = JamiDaemon.mapToJs(vectMap.get(i))
408 return outputArr
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400409 }
410
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400411 mapToNative(map){
412 const ret = new this.dring.StringMap()
413 map.forEach((value, key) => ret.set(key, value))
Adrien Béraude74741b2021-04-19 13:22:54 -0400414 return ret
Adrien Béraud35e7d7c2021-04-13 03:28:39 -0400415 }
Adrien Béraud6ecaa402021-04-06 17:37:25 -0400416}
417
Adrien Béraude74741b2021-04-19 13:22:54 -0400418export default JamiDaemon