Starting with the whisper system
This commit is contained in:
parent
f80c7c4e50
commit
a98503285f
12 changed files with 306 additions and 115 deletions
|
@ -8,7 +8,7 @@ import {LocalClientEntry} from "tc-shared/ui/client";
|
||||||
import {ConnectionProfile} from "tc-shared/profiles/ConnectionProfile";
|
import {ConnectionProfile} from "tc-shared/profiles/ConnectionProfile";
|
||||||
import {ServerAddress} from "tc-shared/ui/server";
|
import {ServerAddress} from "tc-shared/ui/server";
|
||||||
import * as log from "tc-shared/log";
|
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 {createErrorModal, createInfoModal, createInputModal, Modal} from "tc-shared/ui/elements/Modal";
|
||||||
import {hashPassword} from "tc-shared/utils/helpers";
|
import {hashPassword} from "tc-shared/utils/helpers";
|
||||||
import {HandshakeHandler} from "tc-shared/connection/HandshakeHandler";
|
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 {EventType} from "tc-shared/ui/frames/log/Definitions";
|
||||||
import {PluginCmdRegistry} from "tc-shared/connection/PluginCmdHandler";
|
import {PluginCmdRegistry} from "tc-shared/connection/PluginCmdHandler";
|
||||||
import {W2GPluginCmdHandler} from "tc-shared/video-viewer/W2GPlugin";
|
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 {getServerConnectionFactory} from "tc-shared/connection/ConnectionFactory";
|
||||||
|
import {WhisperSession} from "tc-shared/voice/Whisper";
|
||||||
|
|
||||||
export enum InputHardwareState {
|
export enum InputHardwareState {
|
||||||
MISSING,
|
MISSING,
|
||||||
|
@ -199,6 +200,8 @@ export class ConnectionHandler {
|
||||||
});
|
});
|
||||||
this.serverConnection.getVoiceConnection().events.on("notify_connection_status_changed", () => this.update_voice_status());
|
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.channelTree = new ChannelTree(this);
|
||||||
this.fileManager = new FileManager(this);
|
this.fileManager = new FileManager(this);
|
||||||
this.permissions = new PermissionManager(this);
|
this.permissions = new PermissionManager(this);
|
||||||
|
@ -701,8 +704,8 @@ export class ConnectionHandler {
|
||||||
|
|
||||||
const vconnection = this.serverConnection.getVoiceConnection();
|
const vconnection = this.serverConnection.getVoiceConnection();
|
||||||
|
|
||||||
const codecEncodeSupported = !targetChannel || vconnection.encoding_supported(targetChannel.properties.channel_codec);
|
const codecEncodeSupported = !targetChannel || vconnection.encodingSupported(targetChannel.properties.channel_codec);
|
||||||
const codecDecodeSupported = !targetChannel || vconnection.decoding_supported(targetChannel.properties.channel_codec);
|
const codecDecodeSupported = !targetChannel || vconnection.decodingSupported(targetChannel.properties.channel_codec);
|
||||||
|
|
||||||
const property_update = {
|
const property_update = {
|
||||||
client_input_muted: this.client_status.input_muted,
|
client_input_muted: this.client_status.input_muted,
|
||||||
|
@ -711,7 +714,7 @@ export class ConnectionHandler {
|
||||||
|
|
||||||
/* update the encoding codec */
|
/* update the encoding codec */
|
||||||
if(codecEncodeSupported && targetChannel) {
|
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) {
|
if(!this.serverConnection.connected() || vconnection.getConnectionState() !== VoiceConnectionStatus.Connected) {
|
||||||
|
@ -720,10 +723,10 @@ export class ConnectionHandler {
|
||||||
} else {
|
} else {
|
||||||
const recording_supported =
|
const recording_supported =
|
||||||
this.getInputHardwareState() === InputHardwareState.VALID &&
|
this.getInputHardwareState() === InputHardwareState.VALID &&
|
||||||
(!targetChannel || vconnection.encoding_supported(targetChannel.properties.channel_codec)) &&
|
(!targetChannel || vconnection.encodingSupported(targetChannel.properties.channel_codec)) &&
|
||||||
vconnection.getConnectionState() === VoiceConnectionStatus.Connected;
|
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_input_hardware"] = recording_supported;
|
||||||
property_update["client_output_hardware"] = playback_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;
|
const enableRecording = !this.client_status.input_muted && !this.client_status.output_muted;
|
||||||
/* No need to start the microphone when we're not even connected */
|
/* 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(input) {
|
||||||
if(enableRecording && this.serverConnection.connected()) {
|
if(enableRecording && this.serverConnection.connected()) {
|
||||||
if(this.getInputHardwareState() !== InputHardwareState.START_FAILED)
|
if(this.getInputHardwareState() !== InputHardwareState.START_FAILED)
|
||||||
|
@ -821,7 +824,7 @@ export class ConnectionHandler {
|
||||||
let recorder: RecorderProfile = default_recorder;
|
let recorder: RecorderProfile = default_recorder;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await this.serverConnection.getVoiceConnection().acquire_voice_recorder(recorder);
|
await this.serverConnection.getVoiceConnection().acquireVoiceRecorder(recorder);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logError(LogCategory.AUDIO, tr("Failed to acquire recorder: %o"), 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();
|
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 {
|
reconnect_properties(profile?: ConnectionProfile) : ConnectParameters {
|
||||||
const name = (this.getClient() ? this.getClient().clientNickName() : "") ||
|
const name = (this.getClient() ? this.getClient().clientNickName() : "") ||
|
||||||
|
@ -972,6 +975,23 @@ export class ConnectionHandler {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async initializeWhisperSession(session: WhisperSession) : Promise<WhisperSessionInitializeData> {
|
||||||
|
/* 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() {
|
destroy() {
|
||||||
this.event_registry.unregister_handler(this);
|
this.event_registry.unregister_handler(this);
|
||||||
this.cancel_reconnect(true);
|
this.cancel_reconnect(true);
|
||||||
|
|
|
@ -66,7 +66,7 @@ export class DummyVoiceConnection extends AbstractVoiceConnection {
|
||||||
super(connection);
|
super(connection);
|
||||||
}
|
}
|
||||||
|
|
||||||
async acquire_voice_recorder(recorder: RecorderProfile | undefined): Promise<void> {
|
async acquireVoiceRecorder(recorder: RecorderProfile | undefined): Promise<void> {
|
||||||
if(this.recorder === recorder)
|
if(this.recorder === recorder)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
@ -88,15 +88,15 @@ export class DummyVoiceConnection extends AbstractVoiceConnection {
|
||||||
this.events.fire("notify_recorder_changed", {});
|
this.events.fire("notify_recorder_changed", {});
|
||||||
}
|
}
|
||||||
|
|
||||||
available_clients(): VoiceClient[] {
|
availableClients(): VoiceClient[] {
|
||||||
return this.voiceClients;
|
return this.voiceClients;
|
||||||
}
|
}
|
||||||
|
|
||||||
decoding_supported(codec: number): boolean {
|
decodingSupported(codec: number): boolean {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
encoding_supported(codec: number): boolean {
|
encodingSupported(codec: number): boolean {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -104,23 +104,23 @@ export class DummyVoiceConnection extends AbstractVoiceConnection {
|
||||||
return VoiceConnectionStatus.ClientUnsupported;
|
return VoiceConnectionStatus.ClientUnsupported;
|
||||||
}
|
}
|
||||||
|
|
||||||
get_encoder_codec(): number {
|
getEncoderCodec(): number {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
register_client(clientId: number): VoiceClient {
|
registerClient(clientId: number): VoiceClient {
|
||||||
const client = new DummyVoiceClient(clientId);
|
const client = new DummyVoiceClient(clientId);
|
||||||
this.voiceClients.push(client);
|
this.voiceClients.push(client);
|
||||||
return client;
|
return client;
|
||||||
}
|
}
|
||||||
|
|
||||||
set_encoder_codec(codec: number) {}
|
setEncoderCodec(codec: number) {}
|
||||||
|
|
||||||
async unregister_client(client: VoiceClient): Promise<void> {
|
async unregister_client(client: VoiceClient): Promise<void> {
|
||||||
this.voiceClients.remove(client as any);
|
this.voiceClients.remove(client as any);
|
||||||
}
|
}
|
||||||
|
|
||||||
voice_recorder(): RecorderProfile {
|
voiceRecorder(): RecorderProfile {
|
||||||
return this.recorder;
|
return this.recorder;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
import {RecorderProfile} from "tc-shared/voice/RecorderProfile";
|
import {RecorderProfile} from "tc-shared/voice/RecorderProfile";
|
||||||
import {AbstractServerConnection} from "tc-shared/connection/ConnectionBase";
|
import {AbstractServerConnection} from "tc-shared/connection/ConnectionBase";
|
||||||
import {Registry} from "tc-shared/events";
|
import {Registry} from "tc-shared/events";
|
||||||
|
import {WhisperSession} from "tc-shared/voice/Whisper";
|
||||||
|
|
||||||
export enum PlayerState {
|
export enum PlayerState {
|
||||||
PREBUFFERING,
|
PREBUFFERING,
|
||||||
|
@ -55,9 +56,31 @@ export interface VoiceConnectionEvents {
|
||||||
newStatus: VoiceConnectionStatus
|
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<WhisperSessionInitializeData>;
|
||||||
|
|
||||||
export abstract class AbstractVoiceConnection {
|
export abstract class AbstractVoiceConnection {
|
||||||
readonly events: Registry<VoiceConnectionEvents>;
|
readonly events: Registry<VoiceConnectionEvents>;
|
||||||
readonly connection: AbstractServerConnection;
|
readonly connection: AbstractServerConnection;
|
||||||
|
@ -69,16 +92,23 @@ export abstract class AbstractVoiceConnection {
|
||||||
|
|
||||||
abstract getConnectionState() : VoiceConnectionStatus;
|
abstract getConnectionState() : VoiceConnectionStatus;
|
||||||
|
|
||||||
abstract encoding_supported(codec: number) : boolean;
|
abstract encodingSupported(codec: number) : boolean;
|
||||||
abstract decoding_supported(codec: number) : boolean;
|
abstract decodingSupported(codec: number) : boolean;
|
||||||
|
|
||||||
abstract register_client(client_id: number) : VoiceClient;
|
abstract registerClient(client_id: number) : VoiceClient;
|
||||||
abstract available_clients() : VoiceClient[];
|
abstract availableClients() : VoiceClient[];
|
||||||
abstract unregister_client(client: VoiceClient) : Promise<void>;
|
abstract unregister_client(client: VoiceClient) : Promise<void>;
|
||||||
|
|
||||||
abstract voice_recorder() : RecorderProfile;
|
abstract voiceRecorder() : RecorderProfile;
|
||||||
abstract acquire_voice_recorder(recorder: RecorderProfile | undefined) : Promise<void>;
|
abstract acquireVoiceRecorder(recorder: RecorderProfile | undefined) : Promise<void>;
|
||||||
|
|
||||||
abstract get_encoder_codec() : number;
|
abstract getEncoderCodec() : number;
|
||||||
abstract set_encoder_codec(codec: number);
|
abstract setEncoderCodec(codec: number);
|
||||||
|
|
||||||
|
/* the whisper API */
|
||||||
|
abstract getWhisperSessions() : WhisperSession[];
|
||||||
|
abstract dropWhisperSession(session: WhisperSession);
|
||||||
|
|
||||||
|
abstract setWhisperSessionInitializer(initializer: WhisperSessionInitializer | undefined);
|
||||||
|
abstract getWhisperSessionInitializer() : WhisperSessionInitializer | undefined;
|
||||||
}
|
}
|
|
@ -25,7 +25,7 @@ export namespace CryptoHelper {
|
||||||
return str.replace(/-/g, '+').replace(/_/g, '/');
|
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));
|
return String.fromCharCode.apply(null, new Uint16Array(buf));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -2,7 +2,7 @@ import * as contextmenu from "tc-shared/ui/elements/ContextMenu";
|
||||||
import {Registry} from "tc-shared/events";
|
import {Registry} from "tc-shared/events";
|
||||||
import {ChannelTree} from "tc-shared/ui/view";
|
import {ChannelTree} from "tc-shared/ui/view";
|
||||||
import * as log from "tc-shared/log";
|
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 {Settings, settings} from "tc-shared/settings";
|
||||||
import {Sound} from "tc-shared/sound/Sounds";
|
import {Sound} from "tc-shared/sound/Sounds";
|
||||||
import {Group, GroupManager, GroupTarget, GroupType} from "tc-shared/permission/GroupManager";
|
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 {formatMessage} from "tc-shared/ui/frames/chat";
|
||||||
import {spawnYesNo} from "tc-shared/ui/modal/ModalYesNo";
|
import {spawnYesNo} from "tc-shared/ui/modal/ModalYesNo";
|
||||||
import * as hex from "tc-shared/crypto/hex";
|
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 * as React from "react";
|
||||||
import {ChannelTreeEntry, ChannelTreeEntryEvents} from "tc-shared/ui/TreeEntry";
|
import {ChannelTreeEntry, ChannelTreeEntryEvents} from "tc-shared/ui/TreeEntry";
|
||||||
import {spawnClientVolumeChange, spawnMusicBotVolumeChange} from "tc-shared/ui/modal/ModalChangeVolumeNew";
|
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 {EventClient, EventType} from "tc-shared/ui/frames/log/Definitions";
|
||||||
import {W2GPluginCmdHandler} from "tc-shared/video-viewer/W2GPlugin";
|
import {W2GPluginCmdHandler} from "tc-shared/video-viewer/W2GPlugin";
|
||||||
import {global_client_actions} from "tc-shared/events/GlobalEvents";
|
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";
|
import {VoiceClient} from "tc-shared/connection/VoiceConnection";
|
||||||
|
|
||||||
export enum ClientType {
|
export enum ClientType {
|
||||||
|
@ -279,11 +279,11 @@ export class ClientEntry extends ChannelTreeEntry<ClientEvents> {
|
||||||
if(flag) {
|
if(flag) {
|
||||||
this.channelTree.client.serverConnection.send_command('clientmute', {
|
this.channelTree.client.serverConnection.send_command('clientmute', {
|
||||||
clid: this.clientId()
|
clid: this.clientId()
|
||||||
});
|
}).then(() => {});
|
||||||
} else if(this._audio_muted) {
|
} else if(this._audio_muted) {
|
||||||
this.channelTree.client.serverConnection.send_command('clientunmute', {
|
this.channelTree.client.serverConnection.send_command('clientunmute', {
|
||||||
clid: this.clientId()
|
clid: this.clientId()
|
||||||
});
|
}).then(() => {});
|
||||||
}
|
}
|
||||||
this._audio_muted = flag;
|
this._audio_muted = flag;
|
||||||
|
|
||||||
|
@ -393,7 +393,7 @@ export class ClientEntry extends ChannelTreeEntry<ClientEvents> {
|
||||||
this.channelTree.client.serverConnection.send_command("servergroupdelclient", {
|
this.channelTree.client.serverConnection.send_command("servergroupdelclient", {
|
||||||
sgid: group.id,
|
sgid: group.id,
|
||||||
cldbid: this.properties.client_database_id
|
cldbid: this.properties.client_database_id
|
||||||
});
|
}).then(() => {});
|
||||||
};
|
};
|
||||||
entry.disabled = !this.channelTree.client.permissions.neededPermission(PermissionType.I_GROUP_MEMBER_ADD_POWER).granted(group.requiredMemberRemovePower);
|
entry.disabled = !this.channelTree.client.permissions.neededPermission(PermissionType.I_GROUP_MEMBER_ADD_POWER).granted(group.requiredMemberRemovePower);
|
||||||
} else {
|
} else {
|
||||||
|
@ -401,7 +401,7 @@ export class ClientEntry extends ChannelTreeEntry<ClientEvents> {
|
||||||
this.channelTree.client.serverConnection.send_command("servergroupaddclient", {
|
this.channelTree.client.serverConnection.send_command("servergroupaddclient", {
|
||||||
sgid: group.id,
|
sgid: group.id,
|
||||||
cldbid: this.properties.client_database_id
|
cldbid: this.properties.client_database_id
|
||||||
});
|
}).then(() => {});
|
||||||
};
|
};
|
||||||
entry.disabled = !this.channelTree.client.permissions.neededPermission(PermissionType.I_GROUP_MEMBER_REMOVE_POWER).granted(group.requiredMemberAddPower);
|
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<ClientEvents> {
|
||||||
cldbid: this.properties.client_database_id,
|
cldbid: this.properties.client_database_id,
|
||||||
cgid: group.id,
|
cgid: group.id,
|
||||||
cid: this.currentChannel().channelId
|
cid: this.currentChannel().channelId
|
||||||
});
|
}).then(() => {});
|
||||||
};
|
};
|
||||||
entry.disabled = !this.channelTree.client.permissions.neededPermission(PermissionType.I_GROUP_MEMBER_ADD_POWER).granted(group.requiredMemberRemovePower);
|
entry.disabled = !this.channelTree.client.permissions.neededPermission(PermissionType.I_GROUP_MEMBER_ADD_POWER).granted(group.requiredMemberRemovePower);
|
||||||
entry.type = contextmenu.MenuEntryType.CHECKBOX;
|
entry.type = contextmenu.MenuEntryType.CHECKBOX;
|
||||||
|
@ -482,20 +482,20 @@ export class ClientEntry extends ChannelTreeEntry<ClientEvents> {
|
||||||
return this.channelTree.client.serverConnection.send_command("servergroupaddclient", {
|
return this.channelTree.client.serverConnection.send_command("servergroupaddclient", {
|
||||||
sgid: groups[0],
|
sgid: groups[0],
|
||||||
cldbid: this.properties.client_database_id
|
cldbid: this.properties.client_database_id
|
||||||
}).then(result => true);
|
}).then(() => true);
|
||||||
} else
|
} else
|
||||||
return this.channelTree.client.serverConnection.send_command("servergroupdelclient", {
|
return this.channelTree.client.serverConnection.send_command("servergroupdelclient", {
|
||||||
sgid: groups[0],
|
sgid: groups[0],
|
||||||
cldbid: this.properties.client_database_id
|
cldbid: this.properties.client_database_id
|
||||||
}).then(result => true);
|
}).then(() => true);
|
||||||
} else {
|
} else {
|
||||||
const data = groups.map(e => { return {sgid: e}; });
|
const data = groups.map(e => { return {sgid: e}; });
|
||||||
data[0]["cldbid"] = this.properties.client_database_id;
|
data[0]["cldbid"] = this.properties.client_database_id;
|
||||||
|
|
||||||
if(flag) {
|
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
|
} 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<ClientEvents> {
|
||||||
icon_class: ClientIcon.Poke,
|
icon_class: ClientIcon.Poke,
|
||||||
name: tr("Poke client"),
|
name: tr("Poke client"),
|
||||||
callback: () => {
|
callback: () => {
|
||||||
createInputModal(tr("Poke client"), tr("Poke message:<br>"), text => true, result => {
|
createInputModal(tr("Poke client"), tr("Poke message:<br>"), () => true, result => {
|
||||||
if(typeof(result) === "string") {
|
if(typeof(result) === "string") {
|
||||||
this.channelTree.client.serverConnection.send_command("clientpoke", {
|
this.channelTree.client.serverConnection.send_command("clientpoke", {
|
||||||
clid: this.clientId(),
|
clid: this.clientId(),
|
||||||
|
@ -567,14 +567,14 @@ export class ClientEntry extends ChannelTreeEntry<ClientEvents> {
|
||||||
icon_class: ClientIcon.Edit,
|
icon_class: ClientIcon.Edit,
|
||||||
name: tr("Change description"),
|
name: tr("Change description"),
|
||||||
callback: () => {
|
callback: () => {
|
||||||
createInputModal(tr("Change client description"), tr("New description:<br>"), text => true, result => {
|
createInputModal(tr("Change client description"), tr("New description:<br>"), () => true, result => {
|
||||||
if(typeof(result) === "string") {
|
if(typeof(result) === "string") {
|
||||||
//TODO tr
|
//TODO tr
|
||||||
console.log("Changing " + this.clientNickName() + "'s description to " + result);
|
console.log("Changing " + this.clientNickName() + "'s description to " + result);
|
||||||
this.channelTree.client.serverConnection.send_command("clientedit", {
|
this.channelTree.client.serverConnection.send_command("clientedit", {
|
||||||
clid: this.clientId(),
|
clid: this.clientId(),
|
||||||
client_description: result
|
client_description: result
|
||||||
});
|
}).then(() => {});
|
||||||
|
|
||||||
}
|
}
|
||||||
}, { width: 400, maxLength: 1024 }).open();
|
}, { width: 400, maxLength: 1024 }).open();
|
||||||
|
@ -590,22 +590,21 @@ export class ClientEntry extends ChannelTreeEntry<ClientEvents> {
|
||||||
this.channelTree.client.serverConnection.send_command("clientmove", {
|
this.channelTree.client.serverConnection.send_command("clientmove", {
|
||||||
clid: this.clientId(),
|
clid: this.clientId(),
|
||||||
cid: this.channelTree.client.getClient().currentChannel().getChannelId()
|
cid: this.channelTree.client.getClient().currentChannel().getChannelId()
|
||||||
});
|
}).then(() => {});
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
type: contextmenu.MenuEntryType.ENTRY,
|
type: contextmenu.MenuEntryType.ENTRY,
|
||||||
icon_class: ClientIcon.KickChannel,
|
icon_class: ClientIcon.KickChannel,
|
||||||
name: tr("Kick client from channel"),
|
name: tr("Kick client from channel"),
|
||||||
callback: () => {
|
callback: () => {
|
||||||
createInputModal(tr("Kick client from channel"), tr("Kick reason:<br>"), text => true, result => {
|
createInputModal(tr("Kick client from channel"), tr("Kick reason:<br>"), () => true, result => {
|
||||||
if(typeof(result) !== 'boolean' || result) {
|
if(typeof(result) !== 'boolean' || result) {
|
||||||
//TODO tr
|
logInfo(LogCategory.CLIENT, tr("Kicking client %s from channel with reason %s"), this.clientNickName(), result);
|
||||||
console.log("Kicking client " + this.clientNickName() + " from channel with reason " + result);
|
|
||||||
this.channelTree.client.serverConnection.send_command("clientkick", {
|
this.channelTree.client.serverConnection.send_command("clientkick", {
|
||||||
clid: this.clientId(),
|
clid: this.clientId(),
|
||||||
reasonid: ViewReasonId.VREASON_CHANNEL_KICK,
|
reasonid: ViewReasonId.VREASON_CHANNEL_KICK,
|
||||||
reasonmsg: result
|
reasonmsg: result
|
||||||
});
|
}).then(() => {});
|
||||||
|
|
||||||
}
|
}
|
||||||
}, { width: 400, maxLength: 255 }).open();
|
}, { width: 400, maxLength: 255 }).open();
|
||||||
|
@ -615,16 +614,14 @@ export class ClientEntry extends ChannelTreeEntry<ClientEvents> {
|
||||||
icon_class: ClientIcon.KickServer,
|
icon_class: ClientIcon.KickServer,
|
||||||
name: tr("Kick client fom server"),
|
name: tr("Kick client fom server"),
|
||||||
callback: () => {
|
callback: () => {
|
||||||
createInputModal(tr("Kick client from server"), tr("Kick reason:<br>"), text => true, result => {
|
createInputModal(tr("Kick client from server"), tr("Kick reason:<br>"), () => true, result => {
|
||||||
if(typeof(result) !== 'boolean' || result) {
|
if(typeof(result) !== 'boolean' || result) {
|
||||||
//TODO tr
|
logInfo(LogCategory.CLIENT, tr("Kicking client %s from server with reason %s"), this.clientNickName(), result);
|
||||||
console.log("Kicking client " + this.clientNickName() + " from server with reason " + result);
|
|
||||||
this.channelTree.client.serverConnection.send_command("clientkick", {
|
this.channelTree.client.serverConnection.send_command("clientkick", {
|
||||||
clid: this.clientId(),
|
clid: this.clientId(),
|
||||||
reasonid: ViewReasonId.VREASON_SERVER_KICK,
|
reasonid: ViewReasonId.VREASON_SERVER_KICK,
|
||||||
reasonmsg: result
|
reasonmsg: result
|
||||||
});
|
}).then(() => {});
|
||||||
|
|
||||||
}
|
}
|
||||||
}, { width: 400, maxLength: 255 }).open();
|
}, { width: 400, maxLength: 255 }).open();
|
||||||
}
|
}
|
||||||
|
@ -945,16 +942,12 @@ export class ClientEntry extends ChannelTreeEntry<ClientEvents> {
|
||||||
export class LocalClientEntry extends ClientEntry {
|
export class LocalClientEntry extends ClientEntry {
|
||||||
handle: ConnectionHandler;
|
handle: ConnectionHandler;
|
||||||
|
|
||||||
private renaming: boolean;
|
|
||||||
|
|
||||||
constructor(handle: ConnectionHandler) {
|
constructor(handle: ConnectionHandler) {
|
||||||
super(0, "local client");
|
super(0, "local client");
|
||||||
this.handle = handle;
|
this.handle = handle;
|
||||||
}
|
}
|
||||||
|
|
||||||
showContextMenu(x: number, y: number, on_close: () => void = undefined): void {
|
showContextMenu(x: number, y: number, on_close: () => void = undefined): void {
|
||||||
const _self = this;
|
|
||||||
|
|
||||||
contextmenu.spawn_context_menu(x, y,
|
contextmenu.spawn_context_menu(x, y,
|
||||||
...this.contextmenu_info(), {
|
...this.contextmenu_info(), {
|
||||||
|
|
||||||
|
@ -962,19 +955,19 @@ export class LocalClientEntry extends ClientEntry {
|
||||||
tr("Change name") +
|
tr("Change name") +
|
||||||
(contextmenu.get_provider().html_format_enabled() ? "</b>" : ""),
|
(contextmenu.get_provider().html_format_enabled() ? "</b>" : ""),
|
||||||
icon_class: "client-change_nickname",
|
icon_class: "client-change_nickname",
|
||||||
callback: () =>_self.openRename(),
|
callback: () => this.openRename(),
|
||||||
type: contextmenu.MenuEntryType.ENTRY
|
type: contextmenu.MenuEntryType.ENTRY
|
||||||
}, {
|
}, {
|
||||||
name: tr("Change description"),
|
name: tr("Change description"),
|
||||||
icon_class: "client-edit",
|
icon_class: "client-edit",
|
||||||
callback: () => {
|
callback: () => {
|
||||||
createInputModal(tr("Change own description"), tr("New description:<br>"), text => true, result => {
|
createInputModal(tr("Change own description"), tr("New description:<br>"), () => true, result => {
|
||||||
if(result) {
|
if(result) {
|
||||||
console.log(tr("Changing own description to %s"), result);
|
logInfo(LogCategory.CLIENT, tr("Changing own description to %s"), result);
|
||||||
_self.channelTree.client.serverConnection.send_command("clientedit", {
|
this.channelTree.client.serverConnection.send_command("clientedit", {
|
||||||
clid: _self.clientId(),
|
clid: this.clientId(),
|
||||||
client_description: result
|
client_description: result
|
||||||
});
|
}).then(() => {});
|
||||||
|
|
||||||
}
|
}
|
||||||
}, { width: 400, maxLength: 1024 }).open();
|
}, { width: 400, maxLength: 1024 }).open();
|
||||||
|
@ -994,7 +987,7 @@ export class LocalClientEntry extends ClientEntry {
|
||||||
renameSelf(new_name: string) : Promise<boolean> {
|
renameSelf(new_name: string) : Promise<boolean> {
|
||||||
const old_name = this.properties.client_nickname;
|
const old_name = this.properties.client_nickname;
|
||||||
this.updateVariables({ key: "client_nickname", value: new_name }); /* change it locally */
|
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);
|
settings.changeGlobal(Settings.KEY_CONNECT_USERNAME, new_name);
|
||||||
this.channelTree.client.log.log(EventType.CLIENT_NICKNAME_CHANGED_OWN, {
|
this.channelTree.client.log.log(EventType.CLIENT_NICKNAME_CHANGED_OWN, {
|
||||||
client: this.log_data(),
|
client: this.log_data(),
|
||||||
|
@ -1122,8 +1115,7 @@ export class MusicClientEntry extends ClientEntry {
|
||||||
this.channelTree.client.serverConnection.send_command("clientedit", {
|
this.channelTree.client.serverConnection.send_command("clientedit", {
|
||||||
clid: this.clientId(),
|
clid: this.clientId(),
|
||||||
client_nickname: result
|
client_nickname: result
|
||||||
});
|
}).then(() => {});
|
||||||
|
|
||||||
}
|
}
|
||||||
}, { width: "40em", min_width: "10em", maxLength: 255 }).open();
|
}, { width: "40em", min_width: "10em", maxLength: 255 }).open();
|
||||||
},
|
},
|
||||||
|
@ -1133,13 +1125,12 @@ export class MusicClientEntry extends ClientEntry {
|
||||||
icon_class: "client-edit",
|
icon_class: "client-edit",
|
||||||
disabled: false,
|
disabled: false,
|
||||||
callback: () => {
|
callback: () => {
|
||||||
createInputModal(tr("Change music bots description"), tr("New description:<br>"), text => true, result => {
|
createInputModal(tr("Change music bots description"), tr("New description:<br>"), () => true, result => {
|
||||||
if(typeof(result) === 'string') {
|
if(typeof(result) === 'string') {
|
||||||
this.channelTree.client.serverConnection.send_command("clientedit", {
|
this.channelTree.client.serverConnection.send_command("clientedit", {
|
||||||
clid: this.clientId(),
|
clid: this.clientId(),
|
||||||
client_description: result
|
client_description: result
|
||||||
});
|
}).then(() => {});
|
||||||
|
|
||||||
}
|
}
|
||||||
}, { width: "60em", min_width: "10em", maxLength: 255 }).open();
|
}, { width: "60em", min_width: "10em", maxLength: 255 }).open();
|
||||||
},
|
},
|
||||||
|
@ -1159,7 +1150,7 @@ export class MusicClientEntry extends ClientEntry {
|
||||||
icon_class: "client-edit",
|
icon_class: "client-edit",
|
||||||
disabled: false,
|
disabled: false,
|
||||||
callback: () => {
|
callback: () => {
|
||||||
createInputModal(tr("Please enter the URL"), tr("URL:"), text => true, result => {
|
createInputModal(tr("Please enter the URL"), tr("URL:"), () => true, result => {
|
||||||
if(result) {
|
if(result) {
|
||||||
this.channelTree.client.serverConnection.send_command("musicbotqueueadd", {
|
this.channelTree.client.serverConnection.send_command("musicbotqueueadd", {
|
||||||
bot_id: this.properties.client_database_id,
|
bot_id: this.properties.client_database_id,
|
||||||
|
@ -1187,21 +1178,21 @@ export class MusicClientEntry extends ClientEntry {
|
||||||
this.channelTree.client.serverConnection.send_command("clientmove", {
|
this.channelTree.client.serverConnection.send_command("clientmove", {
|
||||||
clid: this.clientId(),
|
clid: this.clientId(),
|
||||||
cid: this.channelTree.client.getClient().currentChannel().getChannelId()
|
cid: this.channelTree.client.getClient().currentChannel().getChannelId()
|
||||||
});
|
}).then(() => {});
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
type: contextmenu.MenuEntryType.ENTRY,
|
type: contextmenu.MenuEntryType.ENTRY,
|
||||||
icon_class: "client-kick_channel",
|
icon_class: "client-kick_channel",
|
||||||
name: tr("Kick client from channel"),
|
name: tr("Kick client from channel"),
|
||||||
callback: () => {
|
callback: () => {
|
||||||
createInputModal(tr("Kick client from channel"), tr("Kick reason:<br>"), text => true, result => {
|
createInputModal(tr("Kick client from channel"), tr("Kick reason:<br>"), () => true, result => {
|
||||||
if(typeof(result) !== 'boolean' || 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", {
|
this.channelTree.client.serverConnection.send_command("clientkick", {
|
||||||
clid: this.clientId(),
|
clid: this.clientId(),
|
||||||
reasonid: ViewReasonId.VREASON_CHANNEL_KICK,
|
reasonid: ViewReasonId.VREASON_CHANNEL_KICK,
|
||||||
reasonmsg: result
|
reasonmsg: result
|
||||||
});
|
}).then(() => {});
|
||||||
}
|
}
|
||||||
}, { width: 400, maxLength: 255 }).open();
|
}, { width: 400, maxLength: 255 }).open();
|
||||||
}
|
}
|
||||||
|
@ -1249,7 +1240,7 @@ export class MusicClientEntry extends ClientEntry {
|
||||||
if(result) {
|
if(result) {
|
||||||
this.channelTree.client.serverConnection.send_command("musicbotdelete", {
|
this.channelTree.client.serverConnection.send_command("musicbotdelete", {
|
||||||
bot_id: this.properties.client_database_id
|
bot_id: this.properties.client_database_id
|
||||||
});
|
}).then(() => {});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
@ -1282,7 +1273,7 @@ export class MusicClientEntry extends ClientEntry {
|
||||||
this._info_promise_resolve = resolve;
|
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;
|
return this._info_promise;
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -147,7 +147,7 @@ class ChannelEntryIcons extends ReactComponentBase<ChannelEntryIconsProperties,
|
||||||
|
|
||||||
switch (voiceState) {
|
switch (voiceState) {
|
||||||
case VoiceConnectionStatus.Connected:
|
case VoiceConnectionStatus.Connected:
|
||||||
state.is_codec_supported = voiceConnection.decoding_supported(currentCodec);
|
state.is_codec_supported = voiceConnection.decodingSupported(currentCodec);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
|
|
@ -505,7 +505,7 @@ export class ChannelTree {
|
||||||
|
|
||||||
const voice_connection = this.client.serverConnection.getVoiceConnection();
|
const voice_connection = this.client.serverConnection.getVoiceConnection();
|
||||||
if(voice_connection)
|
if(voice_connection)
|
||||||
client.set_audio_handle(voice_connection.register_client(client.clientId()));
|
client.set_audio_handle(voice_connection.registerClient(client.clientId()));
|
||||||
}
|
}
|
||||||
|
|
||||||
unregisterClient(client: ClientEntry) {
|
unregisterClient(client: ClientEntry) {
|
||||||
|
|
|
@ -0,0 +1,48 @@
|
||||||
|
import {Registry} from "tc-shared/events";
|
||||||
|
|
||||||
|
export interface WhisperSessionEvents {
|
||||||
|
notify_state_changed: { oldState: WhisperSessionState, newState: WhisperSessionState }
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum WhisperSessionState {
|
||||||
|
/* the sesston is getting initialized, not all variables may be set */
|
||||||
|
INITIALIZING,
|
||||||
|
|
||||||
|
/* there is currently no whispering */
|
||||||
|
PAUSED,
|
||||||
|
|
||||||
|
/* we're currently buffering */
|
||||||
|
BUFFERING,
|
||||||
|
|
||||||
|
/* we're replaying some whisper */
|
||||||
|
PLAYING,
|
||||||
|
|
||||||
|
/* we're currently receiving a whisper, but it has been blocked */
|
||||||
|
BLOCKED
|
||||||
|
}
|
||||||
|
|
||||||
|
export const kUnknownWhisperClientUniqueId = "unknown";
|
||||||
|
|
||||||
|
export interface WhisperSession {
|
||||||
|
readonly events: Registry<WhisperSessionEvents>;
|
||||||
|
|
||||||
|
/* 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;
|
||||||
|
}
|
|
@ -6,14 +6,20 @@ import {RecorderProfile} from "tc-shared/voice/RecorderProfile";
|
||||||
import {VoiceClientController} from "./VoiceClient";
|
import {VoiceClientController} from "./VoiceClient";
|
||||||
import {settings, ValuedSettingsKey} from "tc-shared/settings";
|
import {settings, ValuedSettingsKey} from "tc-shared/settings";
|
||||||
import {tr} from "tc-shared/i18n/localize";
|
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 {codecPool} from "./CodecConverter";
|
||||||
import {createErrorModal} from "tc-shared/ui/elements/Modal";
|
import {createErrorModal} from "tc-shared/ui/elements/Modal";
|
||||||
import {ServerConnectionEvents} from "tc-shared/connection/ConnectionBase";
|
import {ServerConnectionEvents} from "tc-shared/connection/ConnectionBase";
|
||||||
import {ConnectionState} from "tc-shared/ConnectionHandler";
|
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 {NativeWebRTCVoiceBridge} from "./bridge/NativeWebRTCVoiceBridge";
|
||||||
import {EventType} from "tc-shared/ui/frames/log/Definitions";
|
import {EventType} from "tc-shared/ui/frames/log/Definitions";
|
||||||
|
import {kUnknownWhisperClientUniqueId, WhisperSession} from "tc-shared/voice/Whisper";
|
||||||
|
|
||||||
export enum VoiceEncodeType {
|
export enum VoiceEncodeType {
|
||||||
JS_ENCODE,
|
JS_ENCODE,
|
||||||
|
@ -46,6 +52,9 @@ export class VoiceConnection extends AbstractVoiceConnection {
|
||||||
private currentAudioSource: RecorderProfile;
|
private currentAudioSource: RecorderProfile;
|
||||||
private voiceClients: VoiceClientController[] = [];
|
private voiceClients: VoiceClientController[] = [];
|
||||||
|
|
||||||
|
private whisperSessionInitializer: WhisperSessionInitializer;
|
||||||
|
private whisperSessions: {[key: number]: WhisperSession} = {};
|
||||||
|
|
||||||
private voiceBridge: VoiceBridge;
|
private voiceBridge: VoiceBridge;
|
||||||
|
|
||||||
private encoderCodec: number = 5;
|
private encoderCodec: number = 5;
|
||||||
|
@ -53,6 +62,8 @@ export class VoiceConnection extends AbstractVoiceConnection {
|
||||||
constructor(connection: ServerConnection) {
|
constructor(connection: ServerConnection) {
|
||||||
super(connection);
|
super(connection);
|
||||||
|
|
||||||
|
this.setWhisperSessionInitializer(undefined);
|
||||||
|
|
||||||
this.connectionState = VoiceConnectionStatus.Disconnected;
|
this.connectionState = VoiceConnectionStatus.Disconnected;
|
||||||
|
|
||||||
this.connection = connection;
|
this.connection = connection;
|
||||||
|
@ -69,7 +80,7 @@ export class VoiceConnection extends AbstractVoiceConnection {
|
||||||
destroy() {
|
destroy() {
|
||||||
this.connection.events.off(this.serverConnectionStateListener);
|
this.connection.events.off(this.serverConnectionStateListener);
|
||||||
this.dropVoiceBridge();
|
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);
|
log.warn(LogCategory.VOICE, tr("Failed to release voice recorder: %o"), error);
|
||||||
}).then(() => {
|
}).then(() => {
|
||||||
for(const client of this.voiceClients) {
|
for(const client of this.voiceClients) {
|
||||||
|
@ -84,7 +95,7 @@ export class VoiceConnection extends AbstractVoiceConnection {
|
||||||
this.events.destroy();
|
this.events.destroy();
|
||||||
}
|
}
|
||||||
|
|
||||||
async acquire_voice_recorder(recorder: RecorderProfile | undefined, enforce?: boolean) {
|
async acquireVoiceRecorder(recorder: RecorderProfile | undefined, enforce?: boolean) {
|
||||||
if(this.currentAudioSource === recorder && !enforce)
|
if(this.currentAudioSource === recorder && !enforce)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
@ -151,6 +162,7 @@ export class VoiceConnection extends AbstractVoiceConnection {
|
||||||
|
|
||||||
this.voiceBridge = new NativeWebRTCVoiceBridge();
|
this.voiceBridge = new NativeWebRTCVoiceBridge();
|
||||||
this.voiceBridge.callback_incoming_voice = packet => this.handleVoicePacket(packet);
|
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.voiceBridge.callback_send_control_data = (request, payload) => {
|
||||||
this.connection.sendData(JSON.stringify(Object.assign({
|
this.connection.sendData(JSON.stringify(Object.assign({
|
||||||
type: "WebRTC",
|
type: "WebRTC",
|
||||||
|
@ -176,7 +188,7 @@ export class VoiceConnection extends AbstractVoiceConnection {
|
||||||
this.connectAttemptCounter = 0;
|
this.connectAttemptCounter = 0;
|
||||||
|
|
||||||
this.connection.client.log.log(EventType.CONNECTION_VOICE_CONNECT_SUCCEEDED, { });
|
this.connection.client.log.log(EventType.CONNECTION_VOICE_CONNECT_SUCCEEDED, { });
|
||||||
const currentInput = this.voice_recorder()?.input;
|
const currentInput = this.voiceRecorder()?.input;
|
||||||
if(currentInput) {
|
if(currentInput) {
|
||||||
this.voiceBridge.setInput(currentInput).catch(error => {
|
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();
|
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() {
|
private handleRecorderUnmount() {
|
||||||
log.info(LogCategory.VOICE, "Lost recorder!");
|
log.info(LogCategory.VOICE, "Lost recorder!");
|
||||||
this.currentAudioSource = undefined;
|
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) {
|
private setConnectionState(state: VoiceConnectionStatus) {
|
||||||
|
@ -304,11 +316,11 @@ export class VoiceConnection extends AbstractVoiceConnection {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
voice_recorder(): RecorderProfile {
|
voiceRecorder(): RecorderProfile {
|
||||||
return this.currentAudioSource;
|
return this.currentAudioSource;
|
||||||
}
|
}
|
||||||
|
|
||||||
available_clients(): VoiceClient[] {
|
availableClients(): VoiceClient[] {
|
||||||
return this.voiceClients;
|
return this.voiceClients;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -327,27 +339,61 @@ export class VoiceConnection extends AbstractVoiceConnection {
|
||||||
return Promise.resolve();
|
return Promise.resolve();
|
||||||
}
|
}
|
||||||
|
|
||||||
register_client(client_id: number): VoiceClient {
|
registerClient(client_id: number): VoiceClient {
|
||||||
const client = new VoiceClientController(client_id);
|
const client = new VoiceClientController(client_id);
|
||||||
this.voiceClients.push(client);
|
this.voiceClients.push(client);
|
||||||
return client;
|
return client;
|
||||||
}
|
}
|
||||||
|
|
||||||
decoding_supported(codec: number): boolean {
|
decodingSupported(codec: number): boolean {
|
||||||
return VoiceConnection.codecSupported(codec);
|
return VoiceConnection.codecSupported(codec);
|
||||||
}
|
}
|
||||||
|
|
||||||
encoding_supported(codec: number): boolean {
|
encodingSupported(codec: number): boolean {
|
||||||
return VoiceConnection.codecSupported(codec);
|
return VoiceConnection.codecSupported(codec);
|
||||||
}
|
}
|
||||||
|
|
||||||
get_encoder_codec(): number {
|
getEncoderCodec(): number {
|
||||||
return this.encoderCodec;
|
return this.encoderCodec;
|
||||||
}
|
}
|
||||||
|
|
||||||
set_encoder_codec(codec: number) {
|
setEncoderCodec(codec: number) {
|
||||||
this.encoderCodec = codec;
|
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 */
|
/* funny fact that typescript dosn't find this */
|
||||||
|
|
|
@ -4,6 +4,9 @@ import * as log from "tc-shared/log";
|
||||||
import {LogCategory} from "tc-shared/log";
|
import {LogCategory} from "tc-shared/log";
|
||||||
import {tr} from "tc-shared/i18n/localize";
|
import {tr} from "tc-shared/i18n/localize";
|
||||||
import {WebRTCVoiceBridge} from "./WebRTCVoiceBridge";
|
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 {
|
export class NativeWebRTCVoiceBridge extends WebRTCVoiceBridge {
|
||||||
static isSupported(): boolean {
|
static isSupported(): boolean {
|
||||||
|
@ -40,8 +43,8 @@ export class NativeWebRTCVoiceBridge extends WebRTCVoiceBridge {
|
||||||
connection.addStream(this.localAudioDestinationNode.stream);
|
connection.addStream(this.localAudioDestinationNode.stream);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected handleMainDataChannelMessage(message: MessageEvent) {
|
protected handleVoiceDataChannelMessage(message: MessageEvent) {
|
||||||
super.handleMainDataChannelMessage(message);
|
super.handleVoiceDataChannelMessage(message);
|
||||||
|
|
||||||
let bin = new Uint8Array(message.data);
|
let bin = new Uint8Array(message.data);
|
||||||
let clientId = bin[2] << 8 | bin[3];
|
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 {
|
getInput(): AbstractInput | undefined {
|
||||||
return this.currentInput;
|
return this.currentInput;
|
||||||
}
|
}
|
||||||
|
@ -104,4 +134,10 @@ export class NativeWebRTCVoiceBridge extends WebRTCVoiceBridge {
|
||||||
|
|
||||||
channel.send(packet);
|
channel.send(packet);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
startWhisper() {
|
||||||
|
}
|
||||||
|
|
||||||
|
stopWhisper() {
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -17,11 +17,17 @@ export interface VoicePacket {
|
||||||
payload: Uint8Array;
|
payload: Uint8Array;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface VoiceWhisperPacket extends VoicePacket {
|
||||||
|
clientUniqueId?: string;
|
||||||
|
clientNickname?: string;
|
||||||
|
}
|
||||||
|
|
||||||
export abstract class VoiceBridge {
|
export abstract class VoiceBridge {
|
||||||
protected muted: boolean;
|
protected muted: boolean;
|
||||||
|
|
||||||
callback_send_control_data: (request: string, payload: any) => void;
|
callback_send_control_data: (request: string, payload: any) => void;
|
||||||
callback_incoming_voice: (packet: VoicePacket) => void;
|
callback_incoming_voice: (packet: VoicePacket) => void;
|
||||||
|
callback_incoming_whisper: (packet: VoiceWhisperPacket) => void;
|
||||||
|
|
||||||
callback_disconnect: () => void;
|
callback_disconnect: () => void;
|
||||||
|
|
||||||
|
@ -36,11 +42,9 @@ export abstract class VoiceBridge {
|
||||||
handleControlData(request: string, payload: any) { }
|
handleControlData(request: string, payload: any) { }
|
||||||
|
|
||||||
abstract connect(): Promise<VoiceBridgeConnectResult>;
|
abstract connect(): Promise<VoiceBridgeConnectResult>;
|
||||||
|
|
||||||
abstract disconnect();
|
abstract disconnect();
|
||||||
|
|
||||||
abstract getInput(): AbstractInput | undefined;
|
abstract getInput(): AbstractInput | undefined;
|
||||||
|
|
||||||
abstract setInput(input: AbstractInput | undefined): Promise<void>;
|
abstract setInput(input: AbstractInput | undefined): Promise<void>;
|
||||||
|
|
||||||
abstract sendStopSignal(codec: number);
|
abstract sendStopSignal(codec: number);
|
||||||
|
|
|
@ -10,7 +10,9 @@ export abstract class WebRTCVoiceBridge extends VoiceBridge {
|
||||||
private connectionState: "unconnected" | "connecting" | "connected";
|
private connectionState: "unconnected" | "connecting" | "connected";
|
||||||
|
|
||||||
private rtcConnection: RTCPeerConnection;
|
private rtcConnection: RTCPeerConnection;
|
||||||
private mainDataChannel: RTCDataChannel;
|
private voiceDataChannel: RTCDataChannel;
|
||||||
|
private whisperDataChannel: RTCDataChannel;
|
||||||
|
|
||||||
private cachedIceCandidates: RTCIceCandidateInit[];
|
private cachedIceCandidates: RTCIceCandidateInit[];
|
||||||
|
|
||||||
private callbackRtcAnswer: (answer: any) => void;
|
private callbackRtcAnswer: (answer: any) => void;
|
||||||
|
@ -18,7 +20,7 @@ export abstract class WebRTCVoiceBridge extends VoiceBridge {
|
||||||
private callbackConnectCanceled: (() => void)[] = [];
|
private callbackConnectCanceled: (() => void)[] = [];
|
||||||
private callbackRtcConnected: () => void;
|
private callbackRtcConnected: () => void;
|
||||||
private callbackRtcConnectFailed: (error: any) => void;
|
private callbackRtcConnectFailed: (error: any) => void;
|
||||||
private callbackMainDatachannelOpened: (() => void)[] = [];
|
private callbackVoiceDataChannelOpened: (() => void)[] = [];
|
||||||
|
|
||||||
private allowReconnect: boolean;
|
private allowReconnect: boolean;
|
||||||
|
|
||||||
|
@ -90,15 +92,23 @@ export abstract class WebRTCVoiceBridge extends VoiceBridge {
|
||||||
|
|
||||||
this.initializeRtpConnection(this.rtcConnection);
|
this.initializeRtpConnection(this.rtcConnection);
|
||||||
}
|
}
|
||||||
(window as any).dropVoice = () => this.callback_disconnect();
|
|
||||||
|
|
||||||
{
|
{
|
||||||
const dataChannelConfig = { ordered: false, maxRetransmits: 0 };
|
const dataChannelConfig = { ordered: false, maxRetransmits: 0 };
|
||||||
|
|
||||||
this.mainDataChannel = this.rtcConnection.createDataChannel('main', dataChannelConfig);
|
this.voiceDataChannel = this.rtcConnection.createDataChannel('main', dataChannelConfig);
|
||||||
this.mainDataChannel.onmessage = this.handleMainDataChannelMessage.bind(this);
|
this.voiceDataChannel.onmessage = this.handleVoiceDataChannelMessage.bind(this);
|
||||||
this.mainDataChannel.onopen = this.handleMainDataChannelOpen.bind(this);
|
this.voiceDataChannel.onopen = this.handleVoiceDataChannelOpen.bind(this);
|
||||||
this.mainDataChannel.binaryType = "arraybuffer";
|
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;
|
let offer: RTCSessionDescriptionInit;
|
||||||
|
@ -218,10 +228,10 @@ export abstract class WebRTCVoiceBridge extends VoiceBridge {
|
||||||
}
|
}
|
||||||
|
|
||||||
private cleanupRtcResources() {
|
private cleanupRtcResources() {
|
||||||
if(this.mainDataChannel) {
|
if(this.voiceDataChannel) {
|
||||||
this.mainDataChannel.onclose = undefined;
|
this.voiceDataChannel.onclose = undefined;
|
||||||
this.mainDataChannel.close();
|
this.voiceDataChannel.close();
|
||||||
this.mainDataChannel = undefined;
|
this.voiceDataChannel = undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(this.rtcConnection) {
|
if(this.rtcConnection) {
|
||||||
|
@ -240,15 +250,15 @@ export abstract class WebRTCVoiceBridge extends VoiceBridge {
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async awaitMainChannelOpened(timeout: number) {
|
protected async awaitMainChannelOpened(timeout: number) {
|
||||||
if(typeof this.mainDataChannel === "undefined")
|
if(typeof this.voiceDataChannel === "undefined")
|
||||||
throw tr("missing main data channel");
|
throw tr("missing main data channel");
|
||||||
|
|
||||||
if(this.mainDataChannel.readyState === "open")
|
if(this.voiceDataChannel.readyState === "open")
|
||||||
return;
|
return;
|
||||||
|
|
||||||
await new Promise((resolve, reject) => {
|
await new Promise((resolve, reject) => {
|
||||||
const id = setTimeout(reject, timeout);
|
const id = setTimeout(reject, timeout);
|
||||||
this.callbackMainDatachannelOpened.push(() => {
|
this.callbackVoiceDataChannelOpened.push(() => {
|
||||||
clearTimeout(id);
|
clearTimeout(id);
|
||||||
resolve();
|
resolve();
|
||||||
});
|
});
|
||||||
|
@ -332,13 +342,19 @@ export abstract class WebRTCVoiceBridge extends VoiceBridge {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected handleMainDataChannelOpen() {
|
protected handleVoiceDataChannelOpen() {
|
||||||
logDebug(LogCategory.WEBRTC, tr("Main data channel is open now"));
|
logDebug(LogCategory.WEBRTC, tr("Voice data channel is open now"));
|
||||||
while(this.callbackMainDatachannelOpened.length > 0)
|
while(this.callbackVoiceDataChannelOpened.length > 0)
|
||||||
this.callbackMainDatachannelOpened.pop()();
|
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) {
|
handleControlData(request: string, payload: any) {
|
||||||
super.handleControlData(request, payload);
|
super.handleControlData(request, payload);
|
||||||
|
@ -368,7 +384,7 @@ export abstract class WebRTCVoiceBridge extends VoiceBridge {
|
||||||
}
|
}
|
||||||
|
|
||||||
public getMainDataChannel() : RTCDataChannel {
|
public getMainDataChannel() : RTCDataChannel {
|
||||||
return this.mainDataChannel;
|
return this.voiceDataChannel;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected abstract initializeRtpConnection(connection: RTCPeerConnection);
|
protected abstract initializeRtpConnection(connection: RTCPeerConnection);
|
||||||
|
|
Loading…
Add table
Reference in a new issue