mirror of
https://github.com/discordjs/discord.js.git
synced 2026-03-10 16:43:31 +01:00
177 lines
4.5 KiB
TypeScript
177 lines
4.5 KiB
TypeScript
import { EventEmitter } from 'node:events';
|
|
import { VoiceOpcodes } from 'discord-api-types/voice/v4';
|
|
import WebSocket, { type MessageEvent } from 'ws';
|
|
|
|
export interface VoiceWebSocket extends EventEmitter {
|
|
on(event: 'error', listener: (error: Error) => void): this;
|
|
on(event: 'open', listener: (event: WebSocket.Event) => void): this;
|
|
on(event: 'close', listener: (event: WebSocket.CloseEvent) => void): this;
|
|
/**
|
|
* Debug event for VoiceWebSocket.
|
|
*
|
|
* @eventProperty
|
|
*/
|
|
on(event: 'debug', listener: (message: string) => void): this;
|
|
/**
|
|
* Packet event.
|
|
*
|
|
* @eventProperty
|
|
*/
|
|
on(event: 'packet', listener: (packet: any) => void): this;
|
|
}
|
|
|
|
/**
|
|
* An extension of the WebSocket class to provide helper functionality when interacting
|
|
* with the Discord Voice gateway.
|
|
*/
|
|
export class VoiceWebSocket extends EventEmitter {
|
|
/**
|
|
* The current heartbeat interval, if any.
|
|
*/
|
|
private heartbeatInterval?: NodeJS.Timeout;
|
|
|
|
/**
|
|
* The time (milliseconds since UNIX epoch) that the last heartbeat acknowledgement packet was received.
|
|
* This is set to 0 if an acknowledgement packet hasn't been received yet.
|
|
*/
|
|
private lastHeartbeatAck: number;
|
|
|
|
/**
|
|
* The time (milliseconds since UNIX epoch) that the last heartbeat was sent. This is set to 0 if a heartbeat
|
|
* hasn't been sent yet.
|
|
*/
|
|
private lastHeartbeatSend: number;
|
|
|
|
/**
|
|
* The number of consecutively missed heartbeats.
|
|
*/
|
|
private missedHeartbeats = 0;
|
|
|
|
/**
|
|
* The last recorded ping.
|
|
*/
|
|
public ping?: number;
|
|
|
|
/**
|
|
* The debug logger function, if debugging is enabled.
|
|
*/
|
|
private readonly debug: ((message: string) => void) | null;
|
|
|
|
/**
|
|
* The underlying WebSocket of this wrapper.
|
|
*/
|
|
private readonly ws: WebSocket;
|
|
|
|
/**
|
|
* Creates a new VoiceWebSocket.
|
|
*
|
|
* @param address - The address to connect to
|
|
*/
|
|
public constructor(address: string, debug: boolean) {
|
|
super();
|
|
this.ws = new WebSocket(address);
|
|
this.ws.onmessage = (err) => this.onMessage(err);
|
|
this.ws.onopen = (err) => this.emit('open', err);
|
|
this.ws.onerror = (err: Error | WebSocket.ErrorEvent) => this.emit('error', err instanceof Error ? err : err.error);
|
|
this.ws.onclose = (err) => this.emit('close', err);
|
|
|
|
this.lastHeartbeatAck = 0;
|
|
this.lastHeartbeatSend = 0;
|
|
|
|
this.debug = debug ? (message: string) => this.emit('debug', message) : null;
|
|
}
|
|
|
|
/**
|
|
* Destroys the VoiceWebSocket. The heartbeat interval is cleared, and the connection is closed.
|
|
*/
|
|
public destroy() {
|
|
try {
|
|
this.debug?.('destroyed');
|
|
this.setHeartbeatInterval(-1);
|
|
this.ws.close(1_000);
|
|
} catch (error) {
|
|
const err = error as Error;
|
|
this.emit('error', err);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Handles message events on the WebSocket. Attempts to JSON parse the messages and emit them
|
|
* as packets.
|
|
*
|
|
* @param event - The message event
|
|
*/
|
|
public onMessage(event: MessageEvent) {
|
|
if (typeof event.data !== 'string') return;
|
|
|
|
this.debug?.(`<< ${event.data}`);
|
|
|
|
let packet: any;
|
|
try {
|
|
packet = JSON.parse(event.data);
|
|
} catch (error) {
|
|
const err = error as Error;
|
|
this.emit('error', err);
|
|
return;
|
|
}
|
|
|
|
if (packet.op === VoiceOpcodes.HeartbeatAck) {
|
|
this.lastHeartbeatAck = Date.now();
|
|
this.missedHeartbeats = 0;
|
|
this.ping = this.lastHeartbeatAck - this.lastHeartbeatSend;
|
|
}
|
|
|
|
this.emit('packet', packet);
|
|
}
|
|
|
|
/**
|
|
* Sends a JSON-stringifiable packet over the WebSocket.
|
|
*
|
|
* @param packet - The packet to send
|
|
*/
|
|
public sendPacket(packet: any) {
|
|
try {
|
|
const stringified = JSON.stringify(packet);
|
|
this.debug?.(`>> ${stringified}`);
|
|
this.ws.send(stringified);
|
|
} catch (error) {
|
|
const err = error as Error;
|
|
this.emit('error', err);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Sends a heartbeat over the WebSocket.
|
|
*/
|
|
private sendHeartbeat() {
|
|
this.lastHeartbeatSend = Date.now();
|
|
this.missedHeartbeats++;
|
|
const nonce = this.lastHeartbeatSend;
|
|
this.sendPacket({
|
|
op: VoiceOpcodes.Heartbeat,
|
|
// eslint-disable-next-line id-length
|
|
d: nonce,
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Sets/clears an interval to send heartbeats over the WebSocket.
|
|
*
|
|
* @param ms - The interval in milliseconds. If negative, the interval will be unset
|
|
*/
|
|
public setHeartbeatInterval(ms: number) {
|
|
if (this.heartbeatInterval !== undefined) clearInterval(this.heartbeatInterval);
|
|
if (ms > 0) {
|
|
this.heartbeatInterval = setInterval(() => {
|
|
if (this.lastHeartbeatSend !== 0 && this.missedHeartbeats >= 3) {
|
|
// Missed too many heartbeats - disconnect
|
|
this.ws.close();
|
|
this.setHeartbeatInterval(-1);
|
|
}
|
|
|
|
this.sendHeartbeat();
|
|
}, ms);
|
|
}
|
|
}
|
|
}
|