diff --git a/shared/js/ConnectionHandler.ts b/shared/js/ConnectionHandler.ts index 961944e3..626831d7 100644 --- a/shared/js/ConnectionHandler.ts +++ b/shared/js/ConnectionHandler.ts @@ -8,7 +8,7 @@ import {LocalClientEntry} from "tc-shared/ui/client"; import {ConnectionProfile} from "tc-shared/profiles/ConnectionProfile"; import {ServerAddress} from "tc-shared/ui/server"; import * as log from "tc-shared/log"; -import {LogCategory, logError} from "tc-shared/log"; +import {LogCategory, logError, logInfo} from "tc-shared/log"; import {createErrorModal, createInfoModal, createInputModal, Modal} from "tc-shared/ui/elements/Modal"; import {hashPassword} from "tc-shared/utils/helpers"; import {HandshakeHandler} from "tc-shared/connection/HandshakeHandler"; @@ -35,8 +35,9 @@ import {ServerEventLog} from "tc-shared/ui/frames/log/ServerEventLog"; import {EventType} from "tc-shared/ui/frames/log/Definitions"; import {PluginCmdRegistry} from "tc-shared/connection/PluginCmdHandler"; import {W2GPluginCmdHandler} from "tc-shared/video-viewer/W2GPlugin"; -import {VoiceConnectionStatus} from "tc-shared/connection/VoiceConnection"; +import {VoiceConnectionStatus, WhisperSessionInitializeData} from "tc-shared/connection/VoiceConnection"; import {getServerConnectionFactory} from "tc-shared/connection/ConnectionFactory"; +import {WhisperSession} from "tc-shared/voice/Whisper"; export enum InputHardwareState { MISSING, @@ -199,6 +200,8 @@ export class ConnectionHandler { }); this.serverConnection.getVoiceConnection().events.on("notify_connection_status_changed", () => this.update_voice_status()); + this.serverConnection.getVoiceConnection().setWhisperSessionInitializer(this.initializeWhisperSession.bind(this)); + this.channelTree = new ChannelTree(this); this.fileManager = new FileManager(this); this.permissions = new PermissionManager(this); @@ -701,8 +704,8 @@ export class ConnectionHandler { const vconnection = this.serverConnection.getVoiceConnection(); - const codecEncodeSupported = !targetChannel || vconnection.encoding_supported(targetChannel.properties.channel_codec); - const codecDecodeSupported = !targetChannel || vconnection.decoding_supported(targetChannel.properties.channel_codec); + const codecEncodeSupported = !targetChannel || vconnection.encodingSupported(targetChannel.properties.channel_codec); + const codecDecodeSupported = !targetChannel || vconnection.decodingSupported(targetChannel.properties.channel_codec); const property_update = { client_input_muted: this.client_status.input_muted, @@ -711,7 +714,7 @@ export class ConnectionHandler { /* update the encoding codec */ if(codecEncodeSupported && targetChannel) { - vconnection.set_encoder_codec(targetChannel.properties.channel_codec); + vconnection.setEncoderCodec(targetChannel.properties.channel_codec); } if(!this.serverConnection.connected() || vconnection.getConnectionState() !== VoiceConnectionStatus.Connected) { @@ -720,10 +723,10 @@ export class ConnectionHandler { } else { const recording_supported = this.getInputHardwareState() === InputHardwareState.VALID && - (!targetChannel || vconnection.encoding_supported(targetChannel.properties.channel_codec)) && + (!targetChannel || vconnection.encodingSupported(targetChannel.properties.channel_codec)) && vconnection.getConnectionState() === VoiceConnectionStatus.Connected; - const playback_supported = this.hasOutputHardware() && (!targetChannel || vconnection.decoding_supported(targetChannel.properties.channel_codec)); + const playback_supported = this.hasOutputHardware() && (!targetChannel || vconnection.decodingSupported(targetChannel.properties.channel_codec)); property_update["client_input_hardware"] = recording_supported; property_update["client_output_hardware"] = playback_supported; @@ -778,7 +781,7 @@ export class ConnectionHandler { const enableRecording = !this.client_status.input_muted && !this.client_status.output_muted; /* No need to start the microphone when we're not even connected */ - const input = vconnection.voice_recorder()?.input; + const input = vconnection.voiceRecorder()?.input; if(input) { if(enableRecording && this.serverConnection.connected()) { if(this.getInputHardwareState() !== InputHardwareState.START_FAILED) @@ -821,7 +824,7 @@ export class ConnectionHandler { let recorder: RecorderProfile = default_recorder; try { - await this.serverConnection.getVoiceConnection().acquire_voice_recorder(recorder); + await this.serverConnection.getVoiceConnection().acquireVoiceRecorder(recorder); } catch (error) { logError(LogCategory.AUDIO, tr("Failed to acquire recorder: %o"), error); createErrorModal(tr("Failed to acquire recorder"), tr("Failed to acquire recorder.\nLookup the console for more details.")).open(); @@ -879,7 +882,7 @@ export class ConnectionHandler { } } - getVoiceRecorder() : RecorderProfile | undefined { return this.serverConnection.getVoiceConnection().voice_recorder(); } + getVoiceRecorder() : RecorderProfile | undefined { return this.serverConnection.getVoiceConnection().voiceRecorder(); } reconnect_properties(profile?: ConnectionProfile) : ConnectParameters { const name = (this.getClient() ? this.getClient().clientNickName() : "") || @@ -972,6 +975,23 @@ export class ConnectionHandler { }); } + private async initializeWhisperSession(session: WhisperSession) : Promise { + /* TODO: Try to load the clients unique via a clientgetuidfromclid */ + if(!session.getClientUniqueId()) + throw "missing clients unique id"; + + logInfo(LogCategory.CLIENT, tr("Initializing a whisper session for client %d (%s | %s)"), session.getClientId(), session.getClientUniqueId(), session.getClientName()); + return { + clientName: session.getClientName(), + clientUniqueId: session.getClientUniqueId(), + + blocked: false, + volume: 1, + + sessionTimeout: 60 * 1000 + } + } + destroy() { this.event_registry.unregister_handler(this); this.cancel_reconnect(true); diff --git a/shared/js/connection/DummyVoiceConnection.ts b/shared/js/connection/DummyVoiceConnection.ts index b3276cd8..b2b573ee 100644 --- a/shared/js/connection/DummyVoiceConnection.ts +++ b/shared/js/connection/DummyVoiceConnection.ts @@ -66,7 +66,7 @@ export class DummyVoiceConnection extends AbstractVoiceConnection { super(connection); } - async acquire_voice_recorder(recorder: RecorderProfile | undefined): Promise { + async acquireVoiceRecorder(recorder: RecorderProfile | undefined): Promise { if(this.recorder === recorder) return; @@ -88,15 +88,15 @@ export class DummyVoiceConnection extends AbstractVoiceConnection { this.events.fire("notify_recorder_changed", {}); } - available_clients(): VoiceClient[] { + availableClients(): VoiceClient[] { return this.voiceClients; } - decoding_supported(codec: number): boolean { + decodingSupported(codec: number): boolean { return false; } - encoding_supported(codec: number): boolean { + encodingSupported(codec: number): boolean { return false; } @@ -104,23 +104,23 @@ export class DummyVoiceConnection extends AbstractVoiceConnection { return VoiceConnectionStatus.ClientUnsupported; } - get_encoder_codec(): number { + getEncoderCodec(): number { return 0; } - register_client(clientId: number): VoiceClient { + registerClient(clientId: number): VoiceClient { const client = new DummyVoiceClient(clientId); this.voiceClients.push(client); return client; } - set_encoder_codec(codec: number) {} + setEncoderCodec(codec: number) {} async unregister_client(client: VoiceClient): Promise { this.voiceClients.remove(client as any); } - voice_recorder(): RecorderProfile { + voiceRecorder(): RecorderProfile { return this.recorder; } diff --git a/shared/js/connection/VoiceConnection.ts b/shared/js/connection/VoiceConnection.ts index cab64190..2da8f76b 100644 --- a/shared/js/connection/VoiceConnection.ts +++ b/shared/js/connection/VoiceConnection.ts @@ -1,6 +1,7 @@ import {RecorderProfile} from "tc-shared/voice/RecorderProfile"; import {AbstractServerConnection} from "tc-shared/connection/ConnectionBase"; import {Registry} from "tc-shared/events"; +import {WhisperSession} from "tc-shared/voice/Whisper"; export enum PlayerState { PREBUFFERING, @@ -55,9 +56,31 @@ export interface VoiceConnectionEvents { newStatus: VoiceConnectionStatus }, - "notify_recorder_changed": {} + "notify_recorder_changed": {}, + + "notify_whisper_created": { + session: WhisperSession + }, + "notify_whisper_initialized": { + session: WhisperSession + }, + "notify_whisper_destroyed": { + session: WhisperSession + } } +export type WhisperSessionInitializeData = { + clientName: string, + clientUniqueId: string, + + sessionTimeout: number, + + blocked: boolean, + volume: number +}; + +export type WhisperSessionInitializer = (session: WhisperSession) => Promise; + export abstract class AbstractVoiceConnection { readonly events: Registry; readonly connection: AbstractServerConnection; @@ -69,16 +92,23 @@ export abstract class AbstractVoiceConnection { abstract getConnectionState() : VoiceConnectionStatus; - abstract encoding_supported(codec: number) : boolean; - abstract decoding_supported(codec: number) : boolean; + abstract encodingSupported(codec: number) : boolean; + abstract decodingSupported(codec: number) : boolean; - abstract register_client(client_id: number) : VoiceClient; - abstract available_clients() : VoiceClient[]; + abstract registerClient(client_id: number) : VoiceClient; + abstract availableClients() : VoiceClient[]; abstract unregister_client(client: VoiceClient) : Promise; - abstract voice_recorder() : RecorderProfile; - abstract acquire_voice_recorder(recorder: RecorderProfile | undefined) : Promise; + abstract voiceRecorder() : RecorderProfile; + abstract acquireVoiceRecorder(recorder: RecorderProfile | undefined) : Promise; - abstract get_encoder_codec() : number; - abstract set_encoder_codec(codec: number); + abstract getEncoderCodec() : number; + abstract setEncoderCodec(codec: number); + + /* the whisper API */ + abstract getWhisperSessions() : WhisperSession[]; + abstract dropWhisperSession(session: WhisperSession); + + abstract setWhisperSessionInitializer(initializer: WhisperSessionInitializer | undefined); + abstract getWhisperSessionInitializer() : WhisperSessionInitializer | undefined; } \ No newline at end of file diff --git a/shared/js/profiles/identities/TeamSpeakIdentity.ts b/shared/js/profiles/identities/TeamSpeakIdentity.ts index ca950edc..35ab8994 100644 --- a/shared/js/profiles/identities/TeamSpeakIdentity.ts +++ b/shared/js/profiles/identities/TeamSpeakIdentity.ts @@ -25,7 +25,7 @@ export namespace CryptoHelper { return str.replace(/-/g, '+').replace(/_/g, '/'); } - export function arraybuffer_to_string(buf) { + export function arraybuffer_to_string(buf) : string { return String.fromCharCode.apply(null, new Uint16Array(buf)); } diff --git a/shared/js/ui/client.ts b/shared/js/ui/client.ts index 774d76da..0f756293 100644 --- a/shared/js/ui/client.ts +++ b/shared/js/ui/client.ts @@ -2,7 +2,7 @@ import * as contextmenu from "tc-shared/ui/elements/ContextMenu"; import {Registry} from "tc-shared/events"; import {ChannelTree} from "tc-shared/ui/view"; import * as log from "tc-shared/log"; -import {LogCategory, LogType} from "tc-shared/log"; +import {LogCategory, logInfo, LogType} from "tc-shared/log"; import {Settings, settings} from "tc-shared/settings"; import {Sound} from "tc-shared/sound/Sounds"; import {Group, GroupManager, GroupTarget, GroupType} from "tc-shared/permission/GroupManager"; @@ -19,7 +19,7 @@ import {spawnChangeLatency} from "tc-shared/ui/modal/ModalChangeLatency"; import {formatMessage} from "tc-shared/ui/frames/chat"; import {spawnYesNo} from "tc-shared/ui/modal/ModalYesNo"; import * as hex from "tc-shared/crypto/hex"; -import { ClientEntry as ClientEntryView } from "./tree/Client"; +import {ClientEntry as ClientEntryView} from "./tree/Client"; import * as React from "react"; import {ChannelTreeEntry, ChannelTreeEntryEvents} from "tc-shared/ui/TreeEntry"; import {spawnClientVolumeChange, spawnMusicBotVolumeChange} from "tc-shared/ui/modal/ModalChangeVolumeNew"; @@ -27,7 +27,7 @@ import {spawnPermissionEditorModal} from "tc-shared/ui/modal/permission/ModalPer import {EventClient, EventType} from "tc-shared/ui/frames/log/Definitions"; import {W2GPluginCmdHandler} from "tc-shared/video-viewer/W2GPlugin"; import {global_client_actions} from "tc-shared/events/GlobalEvents"; -import { ClientIcon } from "svg-sprites/client-icons"; +import {ClientIcon} from "svg-sprites/client-icons"; import {VoiceClient} from "tc-shared/connection/VoiceConnection"; export enum ClientType { @@ -279,11 +279,11 @@ export class ClientEntry extends ChannelTreeEntry { if(flag) { this.channelTree.client.serverConnection.send_command('clientmute', { clid: this.clientId() - }); + }).then(() => {}); } else if(this._audio_muted) { this.channelTree.client.serverConnection.send_command('clientunmute', { clid: this.clientId() - }); + }).then(() => {}); } this._audio_muted = flag; @@ -393,7 +393,7 @@ export class ClientEntry extends ChannelTreeEntry { this.channelTree.client.serverConnection.send_command("servergroupdelclient", { sgid: group.id, cldbid: this.properties.client_database_id - }); + }).then(() => {}); }; entry.disabled = !this.channelTree.client.permissions.neededPermission(PermissionType.I_GROUP_MEMBER_ADD_POWER).granted(group.requiredMemberRemovePower); } else { @@ -401,7 +401,7 @@ export class ClientEntry extends ChannelTreeEntry { this.channelTree.client.serverConnection.send_command("servergroupaddclient", { sgid: group.id, cldbid: this.properties.client_database_id - }); + }).then(() => {}); }; entry.disabled = !this.channelTree.client.permissions.neededPermission(PermissionType.I_GROUP_MEMBER_REMOVE_POWER).granted(group.requiredMemberAddPower); } @@ -424,7 +424,7 @@ export class ClientEntry extends ChannelTreeEntry { cldbid: this.properties.client_database_id, cgid: group.id, cid: this.currentChannel().channelId - }); + }).then(() => {}); }; entry.disabled = !this.channelTree.client.permissions.neededPermission(PermissionType.I_GROUP_MEMBER_ADD_POWER).granted(group.requiredMemberRemovePower); entry.type = contextmenu.MenuEntryType.CHECKBOX; @@ -482,20 +482,20 @@ export class ClientEntry extends ChannelTreeEntry { return this.channelTree.client.serverConnection.send_command("servergroupaddclient", { sgid: groups[0], cldbid: this.properties.client_database_id - }).then(result => true); + }).then(() => true); } else return this.channelTree.client.serverConnection.send_command("servergroupdelclient", { sgid: groups[0], cldbid: this.properties.client_database_id - }).then(result => true); + }).then(() => true); } else { const data = groups.map(e => { return {sgid: e}; }); data[0]["cldbid"] = this.properties.client_database_id; if(flag) { - return this.channelTree.client.serverConnection.send_command("clientaddservergroup", data, {flagset: ["continueonerror"]}).then(result => true); + return this.channelTree.client.serverConnection.send_command("clientaddservergroup", data, {flagset: ["continueonerror"]}).then(() => true); } else - return this.channelTree.client.serverConnection.send_command("clientdelservergroup", data, {flagset: ["continueonerror"]}).then(result => true); + return this.channelTree.client.serverConnection.send_command("clientdelservergroup", data, {flagset: ["continueonerror"]}).then(() => true); } }); } @@ -548,7 +548,7 @@ export class ClientEntry extends ChannelTreeEntry { icon_class: ClientIcon.Poke, name: tr("Poke client"), callback: () => { - createInputModal(tr("Poke client"), tr("Poke message:
"), text => true, result => { + createInputModal(tr("Poke client"), tr("Poke message:
"), () => true, result => { if(typeof(result) === "string") { this.channelTree.client.serverConnection.send_command("clientpoke", { clid: this.clientId(), @@ -567,14 +567,14 @@ export class ClientEntry extends ChannelTreeEntry { icon_class: ClientIcon.Edit, name: tr("Change description"), callback: () => { - createInputModal(tr("Change client description"), tr("New description:
"), text => true, result => { + createInputModal(tr("Change client description"), tr("New description:
"), () => true, result => { if(typeof(result) === "string") { //TODO tr console.log("Changing " + this.clientNickName() + "'s description to " + result); this.channelTree.client.serverConnection.send_command("clientedit", { clid: this.clientId(), client_description: result - }); + }).then(() => {}); } }, { width: 400, maxLength: 1024 }).open(); @@ -590,22 +590,21 @@ export class ClientEntry extends ChannelTreeEntry { this.channelTree.client.serverConnection.send_command("clientmove", { clid: this.clientId(), cid: this.channelTree.client.getClient().currentChannel().getChannelId() - }); + }).then(() => {}); } }, { type: contextmenu.MenuEntryType.ENTRY, icon_class: ClientIcon.KickChannel, name: tr("Kick client from channel"), callback: () => { - createInputModal(tr("Kick client from channel"), tr("Kick reason:
"), text => true, result => { + createInputModal(tr("Kick client from channel"), tr("Kick reason:
"), () => true, result => { if(typeof(result) !== 'boolean' || result) { - //TODO tr - console.log("Kicking client " + this.clientNickName() + " from channel with reason " + result); + logInfo(LogCategory.CLIENT, tr("Kicking client %s from channel with reason %s"), this.clientNickName(), result); this.channelTree.client.serverConnection.send_command("clientkick", { clid: this.clientId(), reasonid: ViewReasonId.VREASON_CHANNEL_KICK, reasonmsg: result - }); + }).then(() => {}); } }, { width: 400, maxLength: 255 }).open(); @@ -615,16 +614,14 @@ export class ClientEntry extends ChannelTreeEntry { icon_class: ClientIcon.KickServer, name: tr("Kick client fom server"), callback: () => { - createInputModal(tr("Kick client from server"), tr("Kick reason:
"), text => true, result => { + createInputModal(tr("Kick client from server"), tr("Kick reason:
"), () => true, result => { if(typeof(result) !== 'boolean' || result) { - //TODO tr - console.log("Kicking client " + this.clientNickName() + " from server with reason " + result); + logInfo(LogCategory.CLIENT, tr("Kicking client %s from server with reason %s"), this.clientNickName(), result); this.channelTree.client.serverConnection.send_command("clientkick", { clid: this.clientId(), reasonid: ViewReasonId.VREASON_SERVER_KICK, reasonmsg: result - }); - + }).then(() => {}); } }, { width: 400, maxLength: 255 }).open(); } @@ -945,16 +942,12 @@ export class ClientEntry extends ChannelTreeEntry { export class LocalClientEntry extends ClientEntry { handle: ConnectionHandler; - private renaming: boolean; - constructor(handle: ConnectionHandler) { super(0, "local client"); this.handle = handle; } showContextMenu(x: number, y: number, on_close: () => void = undefined): void { - const _self = this; - contextmenu.spawn_context_menu(x, y, ...this.contextmenu_info(), { @@ -962,19 +955,19 @@ export class LocalClientEntry extends ClientEntry { tr("Change name") + (contextmenu.get_provider().html_format_enabled() ? "" : ""), icon_class: "client-change_nickname", - callback: () =>_self.openRename(), + callback: () => this.openRename(), type: contextmenu.MenuEntryType.ENTRY }, { name: tr("Change description"), icon_class: "client-edit", callback: () => { - createInputModal(tr("Change own description"), tr("New description:
"), text => true, result => { + createInputModal(tr("Change own description"), tr("New description:
"), () => true, result => { if(result) { - console.log(tr("Changing own description to %s"), result); - _self.channelTree.client.serverConnection.send_command("clientedit", { - clid: _self.clientId(), + logInfo(LogCategory.CLIENT, tr("Changing own description to %s"), result); + this.channelTree.client.serverConnection.send_command("clientedit", { + clid: this.clientId(), client_description: result - }); + }).then(() => {}); } }, { width: 400, maxLength: 1024 }).open(); @@ -994,7 +987,7 @@ export class LocalClientEntry extends ClientEntry { renameSelf(new_name: string) : Promise { const old_name = this.properties.client_nickname; this.updateVariables({ key: "client_nickname", value: new_name }); /* change it locally */ - return this.handle.serverConnection.send_command("clientupdate", { client_nickname: new_name }).then((e) => { + return this.handle.serverConnection.send_command("clientupdate", { client_nickname: new_name }).then(() => { settings.changeGlobal(Settings.KEY_CONNECT_USERNAME, new_name); this.channelTree.client.log.log(EventType.CLIENT_NICKNAME_CHANGED_OWN, { client: this.log_data(), @@ -1122,8 +1115,7 @@ export class MusicClientEntry extends ClientEntry { this.channelTree.client.serverConnection.send_command("clientedit", { clid: this.clientId(), client_nickname: result - }); - + }).then(() => {}); } }, { width: "40em", min_width: "10em", maxLength: 255 }).open(); }, @@ -1133,13 +1125,12 @@ export class MusicClientEntry extends ClientEntry { icon_class: "client-edit", disabled: false, callback: () => { - createInputModal(tr("Change music bots description"), tr("New description:
"), text => true, result => { + createInputModal(tr("Change music bots description"), tr("New description:
"), () => true, result => { if(typeof(result) === 'string') { this.channelTree.client.serverConnection.send_command("clientedit", { clid: this.clientId(), client_description: result - }); - + }).then(() => {}); } }, { width: "60em", min_width: "10em", maxLength: 255 }).open(); }, @@ -1159,7 +1150,7 @@ export class MusicClientEntry extends ClientEntry { icon_class: "client-edit", disabled: false, callback: () => { - createInputModal(tr("Please enter the URL"), tr("URL:"), text => true, result => { + createInputModal(tr("Please enter the URL"), tr("URL:"), () => true, result => { if(result) { this.channelTree.client.serverConnection.send_command("musicbotqueueadd", { bot_id: this.properties.client_database_id, @@ -1187,21 +1178,21 @@ export class MusicClientEntry extends ClientEntry { this.channelTree.client.serverConnection.send_command("clientmove", { clid: this.clientId(), cid: this.channelTree.client.getClient().currentChannel().getChannelId() - }); + }).then(() => {}); } }, { type: contextmenu.MenuEntryType.ENTRY, icon_class: "client-kick_channel", name: tr("Kick client from channel"), callback: () => { - createInputModal(tr("Kick client from channel"), tr("Kick reason:
"), text => true, result => { + createInputModal(tr("Kick client from channel"), tr("Kick reason:
"), () => true, result => { if(typeof(result) !== 'boolean' || result) { - console.log(tr("Kicking client %o from channel with reason %o"), this.clientNickName(), result); + logInfo(LogCategory.CLIENT, tr("Kicking client %o from channel with reason %o"), this.clientNickName(), result); this.channelTree.client.serverConnection.send_command("clientkick", { clid: this.clientId(), reasonid: ViewReasonId.VREASON_CHANNEL_KICK, reasonmsg: result - }); + }).then(() => {}); } }, { width: 400, maxLength: 255 }).open(); } @@ -1249,7 +1240,7 @@ export class MusicClientEntry extends ClientEntry { if(result) { this.channelTree.client.serverConnection.send_command("musicbotdelete", { bot_id: this.properties.client_database_id - }); + }).then(() => {}); } }); }, @@ -1282,7 +1273,7 @@ export class MusicClientEntry extends ClientEntry { this._info_promise_resolve = resolve; }); - this.channelTree.client.serverConnection.send_command("musicbotplayerinfo", {bot_id: this.properties.client_database_id }); + this.channelTree.client.serverConnection.send_command("musicbotplayerinfo", {bot_id: this.properties.client_database_id }).then(() => {}); return this._info_promise; } } \ No newline at end of file diff --git a/shared/js/ui/tree/Channel.tsx b/shared/js/ui/tree/Channel.tsx index 5cacf8d0..3e312f2d 100644 --- a/shared/js/ui/tree/Channel.tsx +++ b/shared/js/ui/tree/Channel.tsx @@ -147,7 +147,7 @@ class ChannelEntryIcons extends ReactComponentBase; + + /* get information about the whisperer */ + getClientId() : number; + + /* only ensured to be valid if session has been initialized */ + getClientName() : string | undefined; + + /* only ensured to be valid if session has been initialized */ + getClientUniqueId() : string | undefined; + + isBlocked() : boolean; + setBlocked(flag: boolean); + + getSessionTimeout() : number; + setSessionTimeout() : number; + + getLastWhisperTimestamp() : number; + + setVolume(volume: number); + getVolume() : number; +} \ No newline at end of file diff --git a/web/app/voice/VoiceHandler.ts b/web/app/voice/VoiceHandler.ts index df75b411..30c226f9 100644 --- a/web/app/voice/VoiceHandler.ts +++ b/web/app/voice/VoiceHandler.ts @@ -6,14 +6,20 @@ import {RecorderProfile} from "tc-shared/voice/RecorderProfile"; import {VoiceClientController} from "./VoiceClient"; import {settings, ValuedSettingsKey} from "tc-shared/settings"; import {tr} from "tc-shared/i18n/localize"; -import {AbstractVoiceConnection, VoiceClient, VoiceConnectionStatus} from "tc-shared/connection/VoiceConnection"; +import { + AbstractVoiceConnection, + VoiceClient, + VoiceConnectionStatus, + WhisperSessionInitializer +} from "tc-shared/connection/VoiceConnection"; import {codecPool} from "./CodecConverter"; import {createErrorModal} from "tc-shared/ui/elements/Modal"; import {ServerConnectionEvents} from "tc-shared/connection/ConnectionBase"; import {ConnectionState} from "tc-shared/ConnectionHandler"; -import {VoiceBridge, VoicePacket} from "./bridge/VoiceBridge"; +import {VoiceBridge, VoicePacket, VoiceWhisperPacket} from "./bridge/VoiceBridge"; import {NativeWebRTCVoiceBridge} from "./bridge/NativeWebRTCVoiceBridge"; import {EventType} from "tc-shared/ui/frames/log/Definitions"; +import {kUnknownWhisperClientUniqueId, WhisperSession} from "tc-shared/voice/Whisper"; export enum VoiceEncodeType { JS_ENCODE, @@ -46,6 +52,9 @@ export class VoiceConnection extends AbstractVoiceConnection { private currentAudioSource: RecorderProfile; private voiceClients: VoiceClientController[] = []; + private whisperSessionInitializer: WhisperSessionInitializer; + private whisperSessions: {[key: number]: WhisperSession} = {}; + private voiceBridge: VoiceBridge; private encoderCodec: number = 5; @@ -53,6 +62,8 @@ export class VoiceConnection extends AbstractVoiceConnection { constructor(connection: ServerConnection) { super(connection); + this.setWhisperSessionInitializer(undefined); + this.connectionState = VoiceConnectionStatus.Disconnected; this.connection = connection; @@ -69,7 +80,7 @@ export class VoiceConnection extends AbstractVoiceConnection { destroy() { this.connection.events.off(this.serverConnectionStateListener); this.dropVoiceBridge(); - this.acquire_voice_recorder(undefined, true).catch(error => { + this.acquireVoiceRecorder(undefined, true).catch(error => { log.warn(LogCategory.VOICE, tr("Failed to release voice recorder: %o"), error); }).then(() => { for(const client of this.voiceClients) { @@ -84,7 +95,7 @@ export class VoiceConnection extends AbstractVoiceConnection { this.events.destroy(); } - async acquire_voice_recorder(recorder: RecorderProfile | undefined, enforce?: boolean) { + async acquireVoiceRecorder(recorder: RecorderProfile | undefined, enforce?: boolean) { if(this.currentAudioSource === recorder && !enforce) return; @@ -151,6 +162,7 @@ export class VoiceConnection extends AbstractVoiceConnection { this.voiceBridge = new NativeWebRTCVoiceBridge(); this.voiceBridge.callback_incoming_voice = packet => this.handleVoicePacket(packet); + this.voiceBridge.callback_incoming_whisper = packet => this.handleWhisperPacket(packet); this.voiceBridge.callback_send_control_data = (request, payload) => { this.connection.sendData(JSON.stringify(Object.assign({ type: "WebRTC", @@ -176,7 +188,7 @@ export class VoiceConnection extends AbstractVoiceConnection { this.connectAttemptCounter = 0; this.connection.client.log.log(EventType.CONNECTION_VOICE_CONNECT_SUCCEEDED, { }); - const currentInput = this.voice_recorder()?.input; + const currentInput = this.voiceRecorder()?.input; if(currentInput) { this.voiceBridge.setInput(currentInput).catch(error => { createErrorModal(tr("Input recorder attechment failed"), tr("Failed to apply the current microphone recorder to the voice sender.")).open(); @@ -284,7 +296,7 @@ export class VoiceConnection extends AbstractVoiceConnection { private handleRecorderUnmount() { log.info(LogCategory.VOICE, "Lost recorder!"); this.currentAudioSource = undefined; - this.acquire_voice_recorder(undefined, true); /* we can ignore the promise because we should finish this directly */ + this.acquireVoiceRecorder(undefined, true); /* we can ignore the promise because we should finish this directly */ } private setConnectionState(state: VoiceConnectionStatus) { @@ -304,11 +316,11 @@ export class VoiceConnection extends AbstractVoiceConnection { } } - voice_recorder(): RecorderProfile { + voiceRecorder(): RecorderProfile { return this.currentAudioSource; } - available_clients(): VoiceClient[] { + availableClients(): VoiceClient[] { return this.voiceClients; } @@ -327,27 +339,61 @@ export class VoiceConnection extends AbstractVoiceConnection { return Promise.resolve(); } - register_client(client_id: number): VoiceClient { + registerClient(client_id: number): VoiceClient { const client = new VoiceClientController(client_id); this.voiceClients.push(client); return client; } - decoding_supported(codec: number): boolean { + decodingSupported(codec: number): boolean { return VoiceConnection.codecSupported(codec); } - encoding_supported(codec: number): boolean { + encodingSupported(codec: number): boolean { return VoiceConnection.codecSupported(codec); } - get_encoder_codec(): number { + getEncoderCodec(): number { return this.encoderCodec; } - set_encoder_codec(codec: number) { + setEncoderCodec(codec: number) { this.encoderCodec = codec; } + + protected handleWhisperPacket(packet: VoiceWhisperPacket) { + console.error("Received voice whisper packet: %o", packet); + } + + getWhisperSessions(): WhisperSession[] { + return Object.values(this.whisperSessions); + } + + dropWhisperSession(session: WhisperSession) { + throw "this is currently not supported"; + } + + setWhisperSessionInitializer(initializer: WhisperSessionInitializer | undefined) { + this.whisperSessionInitializer = initializer; + if(!this.whisperSessionInitializer) { + this.whisperSessionInitializer = async session => { + logWarn(LogCategory.VOICE, tr("Missing whisper session initializer. Blocking whisper from %d (%s)"), session.getClientId(), session.getClientUniqueId()); + return { + clientName: session.getClientName() || tr("Unknown client"), + clientUniqueId: session.getClientUniqueId() || kUnknownWhisperClientUniqueId, + + blocked: true, + volume: 1, + + sessionTimeout: 60 * 1000 + } + } + } + } + + getWhisperSessionInitializer(): WhisperSessionInitializer | undefined { + return this.whisperSessionInitializer; + } } /* funny fact that typescript dosn't find this */ diff --git a/web/app/voice/bridge/NativeWebRTCVoiceBridge.ts b/web/app/voice/bridge/NativeWebRTCVoiceBridge.ts index 4bbef0a6..2e4ca56e 100644 --- a/web/app/voice/bridge/NativeWebRTCVoiceBridge.ts +++ b/web/app/voice/bridge/NativeWebRTCVoiceBridge.ts @@ -4,6 +4,9 @@ import * as log from "tc-shared/log"; import {LogCategory} from "tc-shared/log"; import {tr} from "tc-shared/i18n/localize"; import {WebRTCVoiceBridge} from "./WebRTCVoiceBridge"; +import {VoiceWhisperPacket} from "tc-backend/web/voice/bridge/VoiceBridge"; +import {CryptoHelper} from "tc-shared/profiles/identities/TeamSpeakIdentity"; +import arraybuffer_to_string = CryptoHelper.arraybuffer_to_string; export class NativeWebRTCVoiceBridge extends WebRTCVoiceBridge { static isSupported(): boolean { @@ -40,8 +43,8 @@ export class NativeWebRTCVoiceBridge extends WebRTCVoiceBridge { connection.addStream(this.localAudioDestinationNode.stream); } - protected handleMainDataChannelMessage(message: MessageEvent) { - super.handleMainDataChannelMessage(message); + protected handleVoiceDataChannelMessage(message: MessageEvent) { + super.handleVoiceDataChannelMessage(message); let bin = new Uint8Array(message.data); let clientId = bin[2] << 8 | bin[3]; @@ -56,6 +59,33 @@ export class NativeWebRTCVoiceBridge extends WebRTCVoiceBridge { }); } + protected handleWhisperDataChannelMessage(message: MessageEvent) { + super.handleWhisperDataChannelMessage(message); + + let payload = new Uint8Array(message.data); + let payload_offset = 0; + + const flags = payload[payload_offset++]; + + let packet = {} as VoiceWhisperPacket; + if((flags & 0x01) === 1) { + packet.clientUniqueId = arraybuffer_to_string(payload.subarray(payload_offset, payload_offset + 28)); + payload_offset += 28; + + packet.clientNickname = arraybuffer_to_string(payload.subarray(payload_offset + 1, payload_offset + 1 + payload[payload_offset])); + payload_offset += payload[payload_offset] + 1; + } + packet.voiceId = payload[payload_offset] << 8 | payload[payload_offset + 1]; + payload_offset += 2; + + packet.clientId = payload[payload_offset] << 8 | payload[payload_offset + 1]; + payload_offset += 2; + + packet.codec = payload[payload_offset]; + + this.callback_incoming_whisper(packet); + } + getInput(): AbstractInput | undefined { return this.currentInput; } @@ -104,4 +134,10 @@ export class NativeWebRTCVoiceBridge extends WebRTCVoiceBridge { channel.send(packet); } + + startWhisper() { + } + + stopWhisper() { + } } \ No newline at end of file diff --git a/web/app/voice/bridge/VoiceBridge.ts b/web/app/voice/bridge/VoiceBridge.ts index 3b04f4f4..1d928673 100644 --- a/web/app/voice/bridge/VoiceBridge.ts +++ b/web/app/voice/bridge/VoiceBridge.ts @@ -17,11 +17,17 @@ export interface VoicePacket { payload: Uint8Array; } +export interface VoiceWhisperPacket extends VoicePacket { + clientUniqueId?: string; + clientNickname?: string; +} + export abstract class VoiceBridge { protected muted: boolean; callback_send_control_data: (request: string, payload: any) => void; callback_incoming_voice: (packet: VoicePacket) => void; + callback_incoming_whisper: (packet: VoiceWhisperPacket) => void; callback_disconnect: () => void; @@ -36,11 +42,9 @@ export abstract class VoiceBridge { handleControlData(request: string, payload: any) { } abstract connect(): Promise; - abstract disconnect(); abstract getInput(): AbstractInput | undefined; - abstract setInput(input: AbstractInput | undefined): Promise; abstract sendStopSignal(codec: number); diff --git a/web/app/voice/bridge/WebRTCVoiceBridge.ts b/web/app/voice/bridge/WebRTCVoiceBridge.ts index 481287bc..448fc090 100644 --- a/web/app/voice/bridge/WebRTCVoiceBridge.ts +++ b/web/app/voice/bridge/WebRTCVoiceBridge.ts @@ -10,7 +10,9 @@ export abstract class WebRTCVoiceBridge extends VoiceBridge { private connectionState: "unconnected" | "connecting" | "connected"; private rtcConnection: RTCPeerConnection; - private mainDataChannel: RTCDataChannel; + private voiceDataChannel: RTCDataChannel; + private whisperDataChannel: RTCDataChannel; + private cachedIceCandidates: RTCIceCandidateInit[]; private callbackRtcAnswer: (answer: any) => void; @@ -18,7 +20,7 @@ export abstract class WebRTCVoiceBridge extends VoiceBridge { private callbackConnectCanceled: (() => void)[] = []; private callbackRtcConnected: () => void; private callbackRtcConnectFailed: (error: any) => void; - private callbackMainDatachannelOpened: (() => void)[] = []; + private callbackVoiceDataChannelOpened: (() => void)[] = []; private allowReconnect: boolean; @@ -90,15 +92,23 @@ export abstract class WebRTCVoiceBridge extends VoiceBridge { this.initializeRtpConnection(this.rtcConnection); } - (window as any).dropVoice = () => this.callback_disconnect(); { const dataChannelConfig = { ordered: false, maxRetransmits: 0 }; - this.mainDataChannel = this.rtcConnection.createDataChannel('main', dataChannelConfig); - this.mainDataChannel.onmessage = this.handleMainDataChannelMessage.bind(this); - this.mainDataChannel.onopen = this.handleMainDataChannelOpen.bind(this); - this.mainDataChannel.binaryType = "arraybuffer"; + this.voiceDataChannel = this.rtcConnection.createDataChannel('main', dataChannelConfig); + this.voiceDataChannel.onmessage = this.handleVoiceDataChannelMessage.bind(this); + this.voiceDataChannel.onopen = this.handleVoiceDataChannelOpen.bind(this); + this.voiceDataChannel.binaryType = "arraybuffer"; + } + + { + const dataChannelConfig = { ordered: false, maxRetransmits: 0 }; + + this.whisperDataChannel = this.rtcConnection.createDataChannel('voice-whisper', dataChannelConfig); + this.whisperDataChannel.onmessage = this.handleWhisperDataChannelMessage.bind(this); + this.whisperDataChannel.onopen = this.handleWhisperDataChannelOpen.bind(this); + this.whisperDataChannel.binaryType = "arraybuffer"; } let offer: RTCSessionDescriptionInit; @@ -218,10 +228,10 @@ export abstract class WebRTCVoiceBridge extends VoiceBridge { } private cleanupRtcResources() { - if(this.mainDataChannel) { - this.mainDataChannel.onclose = undefined; - this.mainDataChannel.close(); - this.mainDataChannel = undefined; + if(this.voiceDataChannel) { + this.voiceDataChannel.onclose = undefined; + this.voiceDataChannel.close(); + this.voiceDataChannel = undefined; } if(this.rtcConnection) { @@ -240,15 +250,15 @@ export abstract class WebRTCVoiceBridge extends VoiceBridge { } protected async awaitMainChannelOpened(timeout: number) { - if(typeof this.mainDataChannel === "undefined") + if(typeof this.voiceDataChannel === "undefined") throw tr("missing main data channel"); - if(this.mainDataChannel.readyState === "open") + if(this.voiceDataChannel.readyState === "open") return; await new Promise((resolve, reject) => { const id = setTimeout(reject, timeout); - this.callbackMainDatachannelOpened.push(() => { + this.callbackVoiceDataChannelOpened.push(() => { clearTimeout(id); resolve(); }); @@ -332,13 +342,19 @@ export abstract class WebRTCVoiceBridge extends VoiceBridge { } } - protected handleMainDataChannelOpen() { - logDebug(LogCategory.WEBRTC, tr("Main data channel is open now")); - while(this.callbackMainDatachannelOpened.length > 0) - this.callbackMainDatachannelOpened.pop()(); + protected handleVoiceDataChannelOpen() { + logDebug(LogCategory.WEBRTC, tr("Voice data channel is open now")); + while(this.callbackVoiceDataChannelOpened.length > 0) + this.callbackVoiceDataChannelOpened.pop()(); } - protected handleMainDataChannelMessage(message: MessageEvent) { } + protected handleVoiceDataChannelMessage(message: MessageEvent) { } + + protected handleWhisperDataChannelOpen() { + logDebug(LogCategory.WEBRTC, tr("Whisper data channel is open now")); + } + + protected handleWhisperDataChannelMessage(message: MessageEvent) { } handleControlData(request: string, payload: any) { super.handleControlData(request, payload); @@ -368,7 +384,7 @@ export abstract class WebRTCVoiceBridge extends VoiceBridge { } public getMainDataChannel() : RTCDataChannel { - return this.mainDataChannel; + return this.voiceDataChannel; } protected abstract initializeRtpConnection(connection: RTCPeerConnection);