TeaWeb/shared/js/ui/frames/footer/StatusController.ts

223 lines
No EOL
9.1 KiB
TypeScript

import {ConnectionHandler, ConnectionState} from "tc-shared/ConnectionHandler";
import {
ConnectionComponent,
ConnectionStatus,
ConnectionStatusEvents
} from "./StatusDefinitions";
import {Registry} from "tc-shared/events";
import {VoiceConnectionStatus} from "tc-shared/connection/VoiceConnection";
import {VideoConnectionStatus} from "tc-shared/connection/VideoConnection";
export class StatusController {
private readonly events: Registry<ConnectionStatusEvents>;
private currentConnectionHandler: ConnectionHandler;
private listenerHandler: (() => void)[];
private detailedInfoOpen = false;
private detailUpdateTimer: number;
constructor(events: Registry<ConnectionStatusEvents>) {
this.events = events;
this.events.on("query_component_detail_state", () => this.events.fire_react("notify_component_detail_state", { shown: this.detailedInfoOpen }));
this.events.on("query_component_status", event => this.notifyComponentStatus(event.component));
this.events.on("query_connection_status", () => this.notifyConnectionStatus());
this.events.on("action_toggle_component_detail", event => this.setDetailsShown(typeof event.shown === "boolean" ? event.shown : !this.detailedInfoOpen));
}
getEvents() : Registry<ConnectionStatusEvents> {
return this.events;
}
setConnectionHandler(handler: ConnectionHandler | undefined) {
if(this.currentConnectionHandler === handler) { return; }
this.unregisterHandlerEvents();
this.currentConnectionHandler = handler;
this.registerHandlerEvents(handler);
this.notifyConnectionStatus();
}
setDetailsShown(flag: boolean) {
if(this.detailedInfoOpen === flag) { return; }
this.detailedInfoOpen = flag;
this.events.fire_react("notify_component_detail_state", { shown: this.detailedInfoOpen });
if(this.detailedInfoOpen) {
this.detailUpdateTimer = setInterval(() => {
this.notifyComponentStatus("video");
this.notifyComponentStatus("signaling");
this.notifyComponentStatus("voice");
}, 1000);
} else {
clearInterval(this.detailUpdateTimer);
}
}
private registerHandlerEvents(handler: ConnectionHandler) {
this.unregisterHandlerEvents();
const events = this.listenerHandler = [];
events.push(handler.events().on("notify_connection_state_changed", () => {
this.notifyConnectionStatus();
if(this.detailedInfoOpen) {
this.notifyComponentStatus("signaling");
}
}));
events.push(handler.getServerConnection().getVoiceConnection().events.on("notify_connection_status_changed", () => {
this.notifyConnectionStatus();
if(this.detailedInfoOpen) {
this.notifyComponentStatus("voice");
}
}));
events.push(handler.getServerConnection().getVideoConnection().getEvents().on("notify_status_changed", () => {
this.notifyConnectionStatus();
if(this.detailedInfoOpen) {
this.notifyComponentStatus("video");
}
}));
}
private unregisterHandlerEvents() {
this.listenerHandler?.forEach(callback => callback());
}
private async getComponentStatus(component: ConnectionComponent, detailed: boolean) : Promise<ConnectionStatus> {
if(!this.currentConnectionHandler) {
return { type: "disconnected" };
}
switch (component) {
case "video":
const videoConnection = this.currentConnectionHandler.getServerConnection().getVideoConnection();
switch (videoConnection.getStatus()) {
case VideoConnectionStatus.Failed:
/* FIXME: Reason! */
return { type: "unhealthy", reason: tr("Unknown") };
case VideoConnectionStatus.Connected:
if(detailed) {
const statistics = await videoConnection.getConnectionStats();
return {
type: "healthy",
bytesSend: statistics.bytesSend,
bytesReceived: statistics.bytesReceived
};
} else {
return {
type: "healthy"
};
}
case VideoConnectionStatus.Disconnected:
return {
type: "disconnected"
};
case VideoConnectionStatus.Connecting:
return { type: "connecting-video" };
case VideoConnectionStatus.Unsupported:
return { type: "unsupported", side: "server" };
}
break;
case "voice":
const voiceConnection = this.currentConnectionHandler.getServerConnection().getVoiceConnection();
switch (voiceConnection.getConnectionState()) {
case VoiceConnectionStatus.Failed:
return { type: "unhealthy", reason: voiceConnection.getFailedMessage() };
case VoiceConnectionStatus.Connected:
if(detailed) {
const statistics = await voiceConnection.getConnectionStats();
return {
type: "healthy",
bytesSend: statistics.bytesSend,
bytesReceived: statistics.bytesReceived
};
} else {
return {
type: "healthy"
};
}
case VoiceConnectionStatus.Disconnecting:
case VoiceConnectionStatus.Disconnected:
return { type: "disconnected" };
case VoiceConnectionStatus.Connecting:
return { type: "connecting-voice" };
case VoiceConnectionStatus.ServerUnsupported:
return { type: "unsupported", side: "server" };
case VoiceConnectionStatus.ClientUnsupported:
return { type: "unsupported", side: "client" };
}
break;
case "signaling":
switch (this.currentConnectionHandler.connection_state) {
case ConnectionState.INITIALISING:
return { type: "connecting-signalling", state: "initializing" };
case ConnectionState.CONNECTING:
return { type: "connecting-signalling", state: "connecting" };
case ConnectionState.AUTHENTICATING:
return { type: "connecting-signalling", state: "authentication" };
case ConnectionState.CONNECTED:
if(detailed) {
const statistics = this.currentConnectionHandler.getServerConnection().getControlStatistics();
return {
type: "healthy",
bytesSend: statistics.bytesSend,
bytesReceived: statistics.bytesReceived
};
} else {
return {
type: "healthy"
};
}
case ConnectionState.UNCONNECTED:
case ConnectionState.DISCONNECTING:
return { type: "disconnected" };
}
break;
}
}
async notifyComponentStatus(component: ConnectionComponent) {
this.events.fire_react("notify_component_status", { component: component, status: await this.getComponentStatus(component, true) });
}
async notifyConnectionStatus() {
let status: ConnectionStatus = { type: "healthy" };
for(const component of ["signaling", "voice", "video"] as ConnectionComponent[]) {
let componentState = await this.getComponentStatus(component, false);
if(componentState.type === "healthy" || componentState.type === "unsupported") {
continue;
} else if(componentState.type === "disconnected" && component !== "signaling") {
switch (component) {
case "voice":
componentState = { type: "unhealthy", reason: tr("No voice connection") };
break;
case "video":
componentState = { type: "unhealthy", reason: tr("No video connection") };
break;
}
}
status = componentState;
break;
}
this.events.fire_react("notify_connection_status", { status: status });
}
}