Demo Bot: Complete code refactoring. WIP.
This commit is contained in:
		
							
								
								
									
										21
									
								
								bots/bots.ts
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								bots/bots.ts
									
									
									
									
									
								
							| @ -1,14 +1,14 @@ | |||||||
| import { BotsConfig } from './lib/config' | import { BotsConfig } from './lib/config' | ||||||
| import { logger } from './lib/logger' | import { logger } from './lib/logger' | ||||||
| import { ComponentBot } from './lib/bot/component' | import { BotComponent } from './lib/bot/component' | ||||||
| import { DemoBot } from './lib/bot/demobot' | import { BotHandlerDemo } from './lib/bot/handlers/demo' | ||||||
|  |  | ||||||
| if (!process.argv[2]) { | if (!process.argv[2]) { | ||||||
|   throw new Error('Missing parameter: the demobot configuration file path') |   throw new Error('Missing parameter: the demobot configuration file path') | ||||||
| } | } | ||||||
| const botsConfig = new BotsConfig(process.argv[2]) | const botsConfig = new BotsConfig(process.argv[2]) | ||||||
|  |  | ||||||
| const runningBots: ComponentBot[] = [] | const runningBots: BotComponent[] = [] | ||||||
|  |  | ||||||
| async function start (botsConfig: BotsConfig): Promise<void> { | async function start (botsConfig: BotsConfig): Promise<void> { | ||||||
|   await botsConfig.load() |   await botsConfig.load() | ||||||
| @ -19,18 +19,23 @@ async function start (botsConfig: BotsConfig): Promise<void> { | |||||||
|     logger.info('Starting DemoBot...') |     logger.info('Starting DemoBot...') | ||||||
|  |  | ||||||
|     const config = botsConfig.getDemoBotConfig() |     const config = botsConfig.getDemoBotConfig() | ||||||
|     const instance = new DemoBot( |     const instance = new BotComponent( | ||||||
|       'DemoBot', |       'DemoBot', | ||||||
|       { |       { | ||||||
|         service: config.service, |         service: config.service, | ||||||
|         domain: config.domain, |         domain: config.domain, | ||||||
|         password: config.password |         password: config.password | ||||||
|       }, |       }, | ||||||
|       config.rooms, |       config.mucDomain | ||||||
|       'DemoBot' // FIXME: handle the case where the nick is already taken. |  | ||||||
|     ) |     ) | ||||||
|     runningBots.push(instance) |     runningBots.push(instance) | ||||||
|     instance.connect().catch(err => { throw err }) |  | ||||||
|  |     instance.connect().then(async () => { | ||||||
|  |       for (const roomId of config.rooms) { | ||||||
|  |         const room = await instance.joinRoom(roomId, 'DemoBot') | ||||||
|  |         room.attachHandler(new BotHandlerDemo(room)) | ||||||
|  |       } | ||||||
|  |     }).catch(err => { throw err }) | ||||||
|   } |   } | ||||||
|   if (!atLeastOne) { |   if (!atLeastOne) { | ||||||
|     logger.info('No bot to launch, exiting.') |     logger.info('No bot to launch, exiting.') | ||||||
| @ -42,7 +47,7 @@ async function shutdown (): Promise<void> { | |||||||
|   logger.info('Shutdown...') |   logger.info('Shutdown...') | ||||||
|   for (const bot of runningBots) { |   for (const bot of runningBots) { | ||||||
|     logger.info('Stopping the bot ' + bot.botName + '...') |     logger.info('Stopping the bot ' + bot.botName + '...') | ||||||
|     await bot.stop() |     await bot.disconnect() | ||||||
|   } |   } | ||||||
|   process.exit(0) |   process.exit(0) | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,23 +1,20 @@ | |||||||
| /* eslint-disable no-void */ | import type { XMPPStanza, XMPPStanzaType } from './types' | ||||||
|  | import type { Node } from '@xmpp/xml' | ||||||
| import { logger } from '../logger' | import { logger } from '../logger' | ||||||
| import type { XMPPStanza } from './types' | import { component, xml, Component, Options } from '@xmpp/component' | ||||||
| import { component, xml, Component } from '@xmpp/component' | import { parse, JID } from '@xmpp/jid' | ||||||
| import type { JID } from '@xmpp/jid' | import { BotRoom } from './room' | ||||||
|  |  | ||||||
| interface ComponentConnectionConfig { | class BotComponent { | ||||||
|   service: string |  | ||||||
|   domain: string |  | ||||||
|   password: string |  | ||||||
| } |  | ||||||
|  |  | ||||||
| abstract class ComponentBot { |  | ||||||
|   protected xmpp?: Component |   protected xmpp?: Component | ||||||
|   protected address?: JID |   protected address?: JID | ||||||
|   protected xml = xml |   public readonly xml = xml | ||||||
|  |   protected rooms: Map<string, BotRoom> = new Map() | ||||||
|  |  | ||||||
|   constructor ( |   constructor ( | ||||||
|     public readonly botName: string, |     public readonly botName: string, | ||||||
|     protected readonly connectionConfig: ComponentConnectionConfig |     protected readonly connectionConfig: Options, | ||||||
|  |     protected readonly mucDomain: string | ||||||
|   ) {} |   ) {} | ||||||
|  |  | ||||||
|   public async connect (): Promise<void> { |   public async connect (): Promise<void> { | ||||||
| @ -36,46 +33,78 @@ abstract class ComponentBot { | |||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     this.xmpp.on('stanza', (stanza: XMPPStanza) => { |     this.xmpp.on('stanza', (stanza: XMPPStanza) => { | ||||||
|       logger.debug('stanza received' + (stanza?.toString ? ': ' + stanza.toString() : '')) |       logger.debug('stanza received' + stanza.toString()) | ||||||
|       if (stanza.is('message')) { |       if (!stanza.attrs.from) { return } | ||||||
|         void this.onMessage(stanza) |       const jid = parse(stanza.attrs.from) | ||||||
|       } |       const roomJid = jid.bare() // removing the «resource» part of the jid. | ||||||
|       if (stanza.is('presence')) { |       const room = this.rooms.get(roomJid.toString()) | ||||||
|         void this.onPresence(stanza) |       if (!room) { | ||||||
|       } |         return | ||||||
|       if (stanza.is('iq')) { |  | ||||||
|         void this.onIq(stanza) |  | ||||||
|       } |       } | ||||||
|  |       room.emit('stanza', stanza, jid.getResource()) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     this.xmpp.on('online', (address) => { |     this.xmpp.on('online', (address) => { | ||||||
|       logger.debug('Online with address' + address.toString()) |       logger.debug('Online with address' + address.toString()) | ||||||
|  |  | ||||||
|       this.address = address |       this.address = address | ||||||
|       void this.onOnline() |  | ||||||
|  |       // 'online' is emitted at reconnection, so we must reset rooms rosters | ||||||
|  |       this.rooms.forEach(room => room.emit('reset')) | ||||||
|  |     }) | ||||||
|  |  | ||||||
|  |     this.xmpp.on('offline', () => { | ||||||
|  |       logger.info(`Stoppping process: ${this.botName} is now offline.`) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await this.xmpp.start() |     await this.xmpp.start() | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public async stop (): Promise<any> { |   public async disconnect (): Promise<any> { | ||||||
|     const p = new Promise((resolve) => { |     for (const [roomId, room] of this.rooms) { | ||||||
|       this.xmpp?.on('offline', () => { |       logger.debug(`Leaving room ${roomId}...`) | ||||||
|         logger.info(`Stoppping process: ${this.botName} is now offline.`) |       await room.part() | ||||||
|         resolve(true) |     } | ||||||
|       }) |  | ||||||
|     }) |  | ||||||
|     await this.xmpp?.stop() |     await this.xmpp?.stop() | ||||||
|     await p |     this.xmpp = undefined | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   protected async onMessage (_stanza: XMPPStanza): Promise<void> {} |   public async sendStanza ( | ||||||
|   protected async onIq (_stanza: XMPPStanza): Promise<void> {} |     type: XMPPStanzaType, | ||||||
|   protected async onPresence (_stanza: XMPPStanza): Promise<void> {} |     attrs: object, | ||||||
|   protected async onOnline (): Promise<void> {} |     ...children: Node[] | ||||||
|  |   ): Promise<void> { | ||||||
|  |     attrs = Object.assign({ | ||||||
|  |       from: this.address?.toString() | ||||||
|  |     }, attrs) | ||||||
|  |  | ||||||
|  |     const stanza = this.xml(type, attrs, ...children) | ||||||
|  |     logger.debug('stanza to emit: ' + stanza.toString()) | ||||||
|  |     await this.xmpp?.send(stanza) | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public async joinRoom (roomId: string, nick: string): Promise<BotRoom> { | ||||||
|  |     const roomJID = new JID(roomId, this.mucDomain) | ||||||
|  |     const roomJIDstr = roomJID.toString() | ||||||
|  |     let room: BotRoom | undefined = this.rooms.get(roomJIDstr) | ||||||
|  |     if (!room) { | ||||||
|  |       room = new BotRoom(this, roomJID) | ||||||
|  |       this.rooms.set(roomJIDstr, room) | ||||||
|  |     } | ||||||
|  |     await room.join(nick) | ||||||
|  |     return room | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public async partRoom (roomId: string): Promise<void> { | ||||||
|  |     const roomJID = new JID(roomId, this.mucDomain) | ||||||
|  |     const room = this.rooms.get(roomJID.toString()) | ||||||
|  |     if (!room) { | ||||||
|  |       return | ||||||
|  |     } | ||||||
|  |     await room.part() | ||||||
|  |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| export { | export { | ||||||
|   ComponentConnectionConfig, |   BotComponent | ||||||
|   ComponentBot |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,11 +0,0 @@ | |||||||
| import { RoomComponentBot } from './room' |  | ||||||
|  |  | ||||||
| class DemoBot extends RoomComponentBot { |  | ||||||
|   protected async onRoomJoin (roomId: string, nick: string): Promise<void> { |  | ||||||
|     await this.sendGroupchat(roomId, `Hello ${nick}! I'm the DemoBot, I'm here to demonstrate the chatroom.`) |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export { |  | ||||||
|   DemoBot |  | ||||||
| } |  | ||||||
							
								
								
									
										11
									
								
								bots/lib/bot/handlers/base.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								bots/lib/bot/handlers/base.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | import type { BotRoom } from '../room' | ||||||
|  |  | ||||||
|  | export abstract class BotHandler { | ||||||
|  |   constructor ( | ||||||
|  |     protected readonly room: BotRoom | ||||||
|  |   ) { | ||||||
|  |     this.init() | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   protected abstract init (): void | ||||||
|  | } | ||||||
							
								
								
									
										19
									
								
								bots/lib/bot/handlers/demo.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								bots/lib/bot/handlers/demo.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,19 @@ | |||||||
|  | import type { XMPPUser } from '../types' | ||||||
|  | import { BotHandler } from './base' | ||||||
|  |  | ||||||
|  | export class BotHandlerDemo extends BotHandler { | ||||||
|  |   protected init (): void { | ||||||
|  |     const room = this.room | ||||||
|  |     room.on('room_join', (user: XMPPUser) => { | ||||||
|  |       if (user.isMe) { | ||||||
|  |         return | ||||||
|  |       } | ||||||
|  |       if (!room.isOnline()) { | ||||||
|  |         return | ||||||
|  |       } | ||||||
|  |       room.sendGroupchat( | ||||||
|  |         `Hello ${user.nick}! I'm the DemoBot, I'm here to demonstrate the chatroom.` | ||||||
|  |       ).catch(() => {}) | ||||||
|  |     }) | ||||||
|  |   } | ||||||
|  | } | ||||||
| @ -1,132 +1,130 @@ | |||||||
| import { ComponentBot, ComponentConnectionConfig } from './component' | import type { BotComponent } from './component' | ||||||
| import { XMPPStanza } from './types' | import type { BotHandler } from './handlers/base' | ||||||
|  | import type { XMPPStanza, XMPPUser } from './types' | ||||||
|  | import EventEmitter from 'events' | ||||||
|  | import { JID } from '@xmpp/jid' | ||||||
| import { logger } from '../logger' | import { logger } from '../logger' | ||||||
|  |  | ||||||
| interface RoomComponentBotRoomDescription { | export class BotRoom extends EventEmitter { | ||||||
|   jid: string |   protected state: 'offline' | 'online' = 'offline' | ||||||
|   nick: string |   protected userJID: JID | undefined | ||||||
|   users: Map<string, { |   protected readonly roster: Map<string, XMPPUser> = new Map() | ||||||
|     // TODO: add the current user status somewhere. |  | ||||||
|     nick: string |  | ||||||
|   }> |  | ||||||
| } |  | ||||||
|  |  | ||||||
| abstract class RoomComponentBot extends ComponentBot { |   protected readonly handlers: BotHandler[] = [] | ||||||
|   protected readonly rooms: {[jid: string]: RoomComponentBotRoomDescription} = {} |  | ||||||
|  |  | ||||||
|   constructor ( |   constructor ( | ||||||
|     botName: string, |     protected readonly component: BotComponent, | ||||||
|     connectionConfig: ComponentConnectionConfig, |     protected readonly roomJID: JID | ||||||
|     roomIds: string[], |  | ||||||
|     protected readonly nick: string |  | ||||||
|   ) { |   ) { | ||||||
|     super(botName, connectionConfig) |     super() | ||||||
|     for (const roomId of roomIds) { |  | ||||||
|       this.rooms[roomId] = { |     this.on('reset', () => { | ||||||
|         jid: roomId, |       this.state = 'offline' | ||||||
|         nick: nick, |       this.roster.clear() | ||||||
|         users: new Map() |     }) | ||||||
|       } |     this.on('stanza', (stanza: XMPPStanza, resource?: string) => { | ||||||
|     } |       this.receiveStanza(stanza, resource) | ||||||
|  |     }) | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   async onOnline (): Promise<void> { |   public isOnline (): boolean { | ||||||
|     for (const roomId in this.rooms) { |     return this.state === 'online' | ||||||
|       const room = this.rooms[roomId] |  | ||||||
|       logger.debug(`Connecting to room ${room.jid}...`) |  | ||||||
|       const presence = this.xml( |  | ||||||
|         'presence', |  | ||||||
|         { |  | ||||||
|           from: this.address?.toString(), |  | ||||||
|           to: room.jid + '/' + room.nick |  | ||||||
|         }, |  | ||||||
|         this.xml('x', { |  | ||||||
|           xmlns: 'http://jabber.org/protocol/muc' |  | ||||||
|         }) |  | ||||||
|       ) |  | ||||||
|       await this.xmpp?.send(presence) |  | ||||||
|     } |  | ||||||
|     await super.onOnline() |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   protected async onPresence (stanza: XMPPStanza): Promise<void> { |   public async join (nick: string): Promise<void> { | ||||||
|     const [stanzaRoomId, stanzaNick] = stanza.attrs?.from.split('/') |     this.userJID = new JID(this.roomJID.getLocal(), this.roomJID.getDomain(), nick) | ||||||
|     if (this.rooms[stanzaRoomId]) { |     logger.debug(`Emitting a presence for room ${this.roomJID.toString()}...`) | ||||||
|       await this.onRoomPresence(stanzaRoomId, stanza, stanzaNick) |     await this.component.sendStanza('presence', | ||||||
|     } |       { | ||||||
|  |         to: this.userJID.toString() | ||||||
|  |       }, | ||||||
|  |       this.component.xml('x', { | ||||||
|  |         xmlns: 'http://jabber.org/protocol/muc' | ||||||
|  |       }) | ||||||
|  |     ) | ||||||
|  |     // FIXME: should wait for a presence stanza from the server. | ||||||
|  |     // FIXME: should handle used nick errors. | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public async sendGroupchat (roomId: string, msg: string): Promise<void> { |   public async part (): Promise<void> { | ||||||
|     const room = this.rooms[roomId] |     if (!this.userJID) { return } | ||||||
|     if (!room) { |     logger.debug(`Emitting a presence=unavailable for room ${this.roomJID.toString()}...`) | ||||||
|       logger.error('Trying to send a groupchat on an unknown room: ' + roomId) |     await this.component.sendStanza('presence', { | ||||||
|       return |       to: this.userJID.toString(), | ||||||
|     } |       type: 'unavailable' | ||||||
|     const message = this.xml( |     }) | ||||||
|  |     // FIXME: should wait for a presence stanza from the server. | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public async sendGroupchat (msg: string): Promise<void> { | ||||||
|  |     if (!this.userJID) { return } | ||||||
|  |     logger.debug(`Emitting a groupchat message for room ${this.roomJID.toString()}...`) | ||||||
|  |     await this.component.sendStanza( | ||||||
|       'message', |       'message', | ||||||
|       { |       { | ||||||
|         type: 'groupchat', |         type: 'groupchat', | ||||||
|         to: room.jid, |         to: this.roomJID.toString() | ||||||
|         from: this.address?.toString() |  | ||||||
|       }, |       }, | ||||||
|       this.xml('body', {}, msg) |       this.component.xml('body', {}, msg) | ||||||
|     ) |     ) | ||||||
|     logger.debug('Sending message...: ' + (message.toString() as string)) |  | ||||||
|     await this.xmpp?.send(message) |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public async stop (): Promise<any> { |   public receiveStanza (stanza: XMPPStanza, fromResource?: string): void { | ||||||
|     for (const roomId in this.rooms) { |     if (stanza.name === 'presence') { | ||||||
|       const room = this.rooms[roomId] |       this.receivePresenceStanza(stanza, fromResource) | ||||||
|       logger.debug(`Leaving room ${room.jid}...`) |  | ||||||
|       const presence = this.xml( |  | ||||||
|         'presence', |  | ||||||
|         { |  | ||||||
|           from: this.address?.toString(), |  | ||||||
|           to: room.jid + '/' + room.nick, |  | ||||||
|           type: 'unavailable' |  | ||||||
|         } |  | ||||||
|       ) |  | ||||||
|       // FIXME: should wait for a presence stanza from the server. |  | ||||||
|       await this.xmpp?.send(presence) |  | ||||||
|     } |     } | ||||||
|     await super.stop() |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   protected async onRoomPresence ( |   public receivePresenceStanza (stanza: XMPPStanza, fromResource?: string): void { | ||||||
|     roomId: string, |     if (!fromResource) { | ||||||
|     stanza: XMPPStanza, |  | ||||||
|     nick?: string |  | ||||||
|   ): Promise<any> { |  | ||||||
|     const room = this.rooms[roomId] |  | ||||||
|     if (!room) { |  | ||||||
|       return |       return | ||||||
|     } |     } | ||||||
|     if (!nick) { |  | ||||||
|       return |     const isPresent = stanza.attrs.type !== 'unavailable' | ||||||
|     } |  | ||||||
|     const isPresent = stanza.attrs?.type !== 'unavailable' |     const statusElems = stanza.getChild('x')?.getChildren('status') | ||||||
|     // FIXME: selfPresence should better be tested by searching status=110 |     const statusCodes = [] | ||||||
|     const selfPresence = room.nick === nick |     if (statusElems) { | ||||||
|     if (!isPresent) { |       for (const s of statusElems) { | ||||||
|       room.users.delete(nick) |         statusCodes.push(parseInt(s.attrs.code)) | ||||||
|       if (!selfPresence) { |  | ||||||
|         await this.onRoomPart(roomId, nick) |  | ||||||
|       } |       } | ||||||
|       return |  | ||||||
|     } |     } | ||||||
|     room.users.set(nick, { |     const isMe = statusCodes.includes(110) // status 110 means that is concern the current user. | ||||||
|       nick |  | ||||||
|     }) |     let user = this.roster.get(fromResource) | ||||||
|     if (!selfPresence) { |     const previousState = user?.state | ||||||
|       await this.onRoomJoin(roomId, nick) |     if (!isPresent) { | ||||||
|  |       if (!user) { | ||||||
|  |         return | ||||||
|  |       } | ||||||
|  |       user.state = 'offline' | ||||||
|  |       if (isMe) { | ||||||
|  |         this.state = 'offline' | ||||||
|  |       } | ||||||
|  |       if (previousState === 'online') { | ||||||
|  |         this.emit('room_part', user) | ||||||
|  |       } | ||||||
|  |     } else { | ||||||
|  |       if (!user) { | ||||||
|  |         user = { | ||||||
|  |           state: 'online', | ||||||
|  |           nick: fromResource, | ||||||
|  |           isMe: isMe | ||||||
|  |         } | ||||||
|  |         this.roster.set(fromResource, user) | ||||||
|  |       } else { | ||||||
|  |         user.state = 'online' | ||||||
|  |       } | ||||||
|  |       if (isMe) { | ||||||
|  |         this.state = 'online' | ||||||
|  |       } | ||||||
|  |       if (previousState !== 'online') { | ||||||
|  |         this.emit('room_join', user) | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   protected async onRoomJoin (_roomId: string, _nick: string): Promise<void> {} |   public attachHandler (handler: BotHandler): void { | ||||||
|   protected async onRoomPart (_roomId: string, _nick: string): Promise<void> {} |     this.handlers.push(handler) | ||||||
| } |   } | ||||||
|  |  | ||||||
| export { |  | ||||||
|   RoomComponentBot |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -5,3 +5,9 @@ export type XMPPStanzaType = 'message' | 'iq' | 'presence' | |||||||
| export interface XMPPStanza extends Element { | export interface XMPPStanza extends Element { | ||||||
|   name: XMPPStanzaType |   name: XMPPStanzaType | ||||||
| } | } | ||||||
|  |  | ||||||
|  | export interface XMPPUser { | ||||||
|  |   state: 'offline' | 'online' | ||||||
|  |   nick: string | ||||||
|  |   isMe: boolean | ||||||
|  | } | ||||||
|  | |||||||
| @ -32,6 +32,7 @@ | |||||||
|   ], |   ], | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@xmpp/component": "^0.13.0", |     "@xmpp/component": "^0.13.0", | ||||||
|  |     "@xmpp/jid": "^0.13.0", | ||||||
|     "async": "^3.2.2", |     "async": "^3.2.2", | ||||||
|     "body-parser": "^1.19.0", |     "body-parser": "^1.19.0", | ||||||
|     "decache": "^4.6.0", |     "decache": "^4.6.0", | ||||||
| @ -50,6 +51,7 @@ | |||||||
|     "@types/node": "^16.11.6", |     "@types/node": "^16.11.6", | ||||||
|     "@types/winston": "^2.4.4", |     "@types/winston": "^2.4.4", | ||||||
|     "@types/xmpp__component": "^0.13.0", |     "@types/xmpp__component": "^0.13.0", | ||||||
|  |     "@types/xmpp__jid": "^1.3.2", | ||||||
|     "@typescript-eslint/eslint-plugin": "^4.29.0", |     "@typescript-eslint/eslint-plugin": "^4.29.0", | ||||||
|     "@typescript-eslint/parser": "^4.29.0", |     "@typescript-eslint/parser": "^4.29.0", | ||||||
|     "eslint": "^7.32.0", |     "eslint": "^7.32.0", | ||||||
|  | |||||||
| @ -206,7 +206,7 @@ async function getProsodyConfig (options: RegisterServerOptions): Promise<Prosod | |||||||
|     config.useDemoBot(componentSecret) |     config.useDemoBot(componentSecret) | ||||||
|     bots.demobot = demoBotUUIDs |     bots.demobot = demoBotUUIDs | ||||||
|     demoBotContentObj = JSON.stringify({ |     demoBotContentObj = JSON.stringify({ | ||||||
|       rooms: demoBotUUIDs.map((uuid) => `${uuid}@room.${prosodyDomain}`), |       rooms: demoBotUUIDs, | ||||||
|       service: 'xmpp://127.0.0.1:' + externalComponentsPort, |       service: 'xmpp://127.0.0.1:' + externalComponentsPort, | ||||||
|       domain: 'demobot.' + prosodyDomain, |       domain: 'demobot.' + prosodyDomain, | ||||||
|       mucDomain: 'room.' + prosodyDomain, |       mucDomain: 'room.' + prosodyDomain, | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user