discord.js-selfbot-v13/src/util/Voice.js

2444 lines
71 KiB
JavaScript
Raw Normal View History

2022-05-21 14:02:00 +00:00
'use strict';
/* eslint-disable */
var __create = Object.create;
var __defProp = Object.defineProperty;
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
var __getOwnPropNames = Object.getOwnPropertyNames;
var __getProtoOf = Object.getPrototypeOf;
var __hasOwnProp = Object.prototype.hasOwnProperty;
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
var __name = (target, value) => __defProp(target, "name", { value, configurable: true });
var __commonJS = (cb, mod) => function __require()
{
return mod || (0, cb[__getOwnPropNames(cb)[0]])((mod = { exports: {} }).exports, mod), mod.exports;
};
var __export = (target, all) =>
{
for (var name in all)
__defProp(target, name, { get: all[name], enumerable: true });
2022-05-21 14:02:00 +00:00
};
var __copyProps = (to, from, except, desc) =>
{
if (from && typeof from === "object" || typeof from === "function")
{
2022-06-26 02:40:04 +00:00
for (let key of __getOwnPropNames(from))
if (!__hasOwnProp.call(to, key) && key !== except)
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
2022-05-21 14:02:00 +00:00
}
return to;
};
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target, mod));
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
var __publicField = (obj, key, value) =>
{
__defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
2022-06-26 02:40:04 +00:00
return value;
};
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// package.json
2022-05-21 14:02:00 +00:00
var require_package = __commonJS({
"package.json"(exports, module2)
{
2022-05-21 14:02:00 +00:00
module2.exports = {
name: "@discordjs/voice",
version: "0.11.0",
description: "Implementation of the Discord Voice API for node.js",
2022-05-21 14:02:00 +00:00
scripts: {
build: "tsup && node scripts/postbuild.mjs",
test: "jest --coverage",
lint: "prettier --check . && eslint src __tests__ --ext mjs,js,ts",
format: "prettier --write . && eslint src __tests__ --ext mjs,js,ts --fix",
docs: "docgen -i src/index.ts -c docs/index.json -o docs/docs.json --typescript && api-extractor run --local",
prepack: "yarn build && yarn lint && yarn test",
changelog: "git cliff --prepend ./CHANGELOG.md -u -c ./cliff.toml -r ../../ --include-path 'packages/voice/*'",
release: "cliff-jumper"
2022-05-21 14:02:00 +00:00
},
main: "./dist/index.js",
module: "./dist/index.mjs",
typings: "./dist/index.d.ts",
2022-05-21 14:02:00 +00:00
exports: {
import: "./dist/index.mjs",
require: "./dist/index.js",
types: "./dist/index.d.ts"
2022-05-21 14:02:00 +00:00
},
directories: {
lib: "src",
test: "__tests__"
2022-05-21 14:02:00 +00:00
},
files: [
"dist"
],
2022-05-21 14:02:00 +00:00
contributors: [
"Crawl <icrawltogo@gmail.com>",
"Amish Shah <amishshah.2k@gmail.com>",
"SpaceEEC <spaceeec@yahoo.com>",
"Vlad Frangu <kingdgrizzle@gmail.com>",
"Antonio Roman <kyradiscord@gmail.com>"
],
license: "Apache-2.0",
keywords: [
"discord",
"discord.js",
"audio",
"voice",
"streaming"
2022-05-21 14:02:00 +00:00
],
repository: {
type: "git",
url: "git+https://github.com/discordjs/discord.js.git"
2022-05-21 14:02:00 +00:00
},
bugs: {
url: "https://github.com/discordjs/discord.js/issues"
2022-05-21 14:02:00 +00:00
},
homepage: "https://discord.js.org",
2022-05-21 14:02:00 +00:00
dependencies: {
"@types/ws": "^8.5.3",
"discord-api-types": "^0.36.2",
"prism-media": "^1.3.4",
tslib: "^2.4.0",
ws: "^8.8.1"
2022-05-21 14:02:00 +00:00
},
devDependencies: {
"@babel/core": "^7.18.6",
"@babel/preset-env": "^7.18.6",
"@babel/preset-typescript": "^7.18.6",
"@discordjs/docgen": "workspace:^",
"@discordjs/scripts": "workspace:^",
"@favware/cliff-jumper": "^1.8.5",
"@microsoft/api-extractor": "^7.28.4",
"@types/jest": "^28.1.6",
"@types/node": "^16.11.45",
eslint: "^8.20.0",
jest: "^28.1.3",
"jest-websocket-mock": "^2.3.0",
"mock-socket": "^9.1.5",
prettier: "^2.7.1",
tsup: "^6.1.3",
tweetnacl: "^1.0.3",
typescript: "^4.7.4"
2022-05-21 14:02:00 +00:00
},
engines: {
node: ">=16.9.0"
2022-05-21 14:02:00 +00:00
},
publishConfig: {
access: "public"
}
2022-05-21 14:02:00 +00:00
};
}
2022-05-21 14:02:00 +00:00
});
2022-06-26 02:40:04 +00:00
// src/index.ts
2022-05-21 14:02:00 +00:00
var src_exports = {};
__export(src_exports, {
AudioPlayer: () => AudioPlayer,
AudioPlayerError: () => AudioPlayerError,
AudioPlayerStatus: () => AudioPlayerStatus,
AudioReceiveStream: () => AudioReceiveStream,
AudioResource: () => AudioResource,
EndBehaviorType: () => EndBehaviorType,
NoSubscriberBehavior: () => NoSubscriberBehavior,
PlayerSubscription: () => PlayerSubscription,
SSRCMap: () => SSRCMap,
SpeakingMap: () => SpeakingMap,
StreamType: () => StreamType,
VoiceConnection: () => VoiceConnection2,
VoiceConnectionDisconnectReason: () => VoiceConnectionDisconnectReason,
VoiceConnectionStatus: () => VoiceConnectionStatus,
VoiceReceiver: () => VoiceReceiver,
createAudioPlayer: () => createAudioPlayer,
createAudioResource: () => createAudioResource,
createDefaultAudioReceiveStreamOptions: () => createDefaultAudioReceiveStreamOptions,
demuxProbe: () => demuxProbe,
entersState: () => entersState,
generateDependencyReport: () => generateDependencyReport,
getGroups: () => getGroups,
getVoiceConnection: () => getVoiceConnection,
getVoiceConnections: () => getVoiceConnections,
joinVoiceChannel: () => joinVoiceChannel,
2022-06-29 03:28:29 +00:00
validateDiscordOpusHead: () => validateDiscordOpusHead,
Networking: () => Networking,
2022-05-21 14:02:00 +00:00
});
module.exports = __toCommonJS(src_exports);
2022-06-26 02:40:04 +00:00
// src/VoiceConnection.ts
var import_node_events7 = require("events");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/DataStore.ts
var import_v10 = require("discord-api-types/v10");
function createJoinVoiceChannelPayload(config)
{
2022-05-21 14:02:00 +00:00
return {
op: import_v10.GatewayOpcodes.VoiceStateUpdate,
d: {
guild_id: config.guildId,
channel_id: config.channelId,
self_deaf: config.selfDeaf,
self_mute: config.selfMute
}
2022-05-21 14:02:00 +00:00
};
}
__name(createJoinVoiceChannelPayload, "createJoinVoiceChannelPayload");
2022-05-21 14:02:00 +00:00
var groups = /* @__PURE__ */ new Map();
groups.set("default", /* @__PURE__ */ new Map());
function getOrCreateGroup(group)
{
2022-05-21 14:02:00 +00:00
const existing = groups.get(group);
if (existing)
return existing;
2022-05-21 14:02:00 +00:00
const map = /* @__PURE__ */ new Map();
groups.set(group, map);
return map;
}
__name(getOrCreateGroup, "getOrCreateGroup");
function getGroups()
{
2022-05-21 14:02:00 +00:00
return groups;
}
__name(getGroups, "getGroups");
function getVoiceConnections(group = "default")
{
2022-05-21 14:02:00 +00:00
return groups.get(group);
}
__name(getVoiceConnections, "getVoiceConnections");
function getVoiceConnection(guildId, group = "default")
{
2022-05-21 14:02:00 +00:00
return getVoiceConnections(group)?.get(guildId);
}
__name(getVoiceConnection, "getVoiceConnection");
function untrackVoiceConnection(voiceConnection)
{
2022-05-21 14:02:00 +00:00
return getVoiceConnections(voiceConnection.joinConfig.group)?.delete(voiceConnection.joinConfig.guildId);
}
__name(untrackVoiceConnection, "untrackVoiceConnection");
function trackVoiceConnection(voiceConnection)
{
2022-05-21 14:02:00 +00:00
return getOrCreateGroup(voiceConnection.joinConfig.group).set(voiceConnection.joinConfig.guildId, voiceConnection);
}
__name(trackVoiceConnection, "trackVoiceConnection");
2022-05-21 14:02:00 +00:00
var FRAME_LENGTH = 20;
var audioCycleInterval;
var nextTime = -1;
var audioPlayers = [];
function audioCycleStep()
{
if (nextTime === -1)
return;
2022-05-21 14:02:00 +00:00
nextTime += FRAME_LENGTH;
const available = audioPlayers.filter((player) => player.checkPlayable());
available.forEach((player) => player["_stepDispatch"]());
2022-05-21 14:02:00 +00:00
prepareNextAudioFrame(available);
}
__name(audioCycleStep, "audioCycleStep");
function prepareNextAudioFrame(players)
{
2022-05-21 14:02:00 +00:00
const nextPlayer = players.shift();
if (!nextPlayer)
{
if (nextTime !== -1)
{
2022-05-21 14:02:00 +00:00
audioCycleInterval = setTimeout(() => audioCycleStep(), nextTime - Date.now());
}
return;
}
nextPlayer["_stepPrepare"]();
2022-05-21 14:02:00 +00:00
setImmediate(() => prepareNextAudioFrame(players));
}
__name(prepareNextAudioFrame, "prepareNextAudioFrame");
function hasAudioPlayer(target)
{
2022-05-21 14:02:00 +00:00
return audioPlayers.includes(target);
}
__name(hasAudioPlayer, "hasAudioPlayer");
function addAudioPlayer(player)
{
if (hasAudioPlayer(player))
return player;
2022-05-21 14:02:00 +00:00
audioPlayers.push(player);
if (audioPlayers.length === 1)
{
2022-05-21 14:02:00 +00:00
nextTime = Date.now();
setImmediate(() => audioCycleStep());
}
return player;
}
__name(addAudioPlayer, "addAudioPlayer");
function deleteAudioPlayer(player)
{
2022-05-21 14:02:00 +00:00
const index = audioPlayers.indexOf(player);
if (index === -1)
return;
2022-05-21 14:02:00 +00:00
audioPlayers.splice(index, 1);
if (audioPlayers.length === 0)
{
2022-05-21 14:02:00 +00:00
nextTime = -1;
if (typeof audioCycleInterval !== "undefined")
clearTimeout(audioCycleInterval);
2022-05-21 14:02:00 +00:00
}
}
__name(deleteAudioPlayer, "deleteAudioPlayer");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/networking/Networking.ts
var import_node_events3 = require("events");
var import_v42 = require("discord-api-types/voice/v4");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/networking/VoiceUDPSocket.ts
var import_node_dgram = require("dgram");
var import_node_events = require("events");
var import_node_net = require("net");
function parseLocalPacket(message)
{
2022-05-21 14:02:00 +00:00
const packet = Buffer.from(message);
const ip = packet.slice(8, packet.indexOf(0, 8)).toString("utf-8");
if (!(0, import_node_net.isIPv4)(ip))
{
throw new Error("Malformed IP address");
2022-05-21 14:02:00 +00:00
}
const port = packet.readUInt16BE(packet.length - 2);
return { ip, port };
}
__name(parseLocalPacket, "parseLocalPacket");
2022-05-21 14:02:00 +00:00
var KEEP_ALIVE_INTERVAL = 5e3;
var KEEP_ALIVE_LIMIT = 12;
var MAX_COUNTER_VALUE = 2 ** 32 - 1;
var VoiceUDPSocket = class extends import_node_events.EventEmitter
{
constructor(remote, debug = false)
{
2022-05-21 14:02:00 +00:00
super();
__publicField(this, "socket");
__publicField(this, "remote");
__publicField(this, "keepAlives");
__publicField(this, "keepAliveCounter", 0);
__publicField(this, "keepAliveBuffer");
__publicField(this, "keepAliveInterval");
__publicField(this, "ping");
__publicField(this, "debug");
this.socket = (0, import_node_dgram.createSocket)("udp4");
this.socket.on("error", (error) => this.emit("error", error));
this.socket.on("message", (buffer) => this.onMessage(buffer));
this.socket.on("close", () => this.emit("close"));
2022-05-21 14:02:00 +00:00
this.remote = remote;
this.keepAlives = [];
this.keepAliveBuffer = Buffer.alloc(8);
this.keepAliveInterval = setInterval(() => this.keepAlive(), KEEP_ALIVE_INTERVAL);
setImmediate(() => this.keepAlive());
this.debug = debug ? (message) => this.emit("debug", message) : null;
2022-05-21 14:02:00 +00:00
}
onMessage(buffer)
{
if (buffer.length === 8)
{
2022-05-21 14:02:00 +00:00
const counter = buffer.readUInt32LE(0);
const index = this.keepAlives.findIndex(({ value }) => value === counter);
if (index === -1)
return;
2022-05-21 14:02:00 +00:00
this.ping = Date.now() - this.keepAlives[index].timestamp;
this.keepAlives.splice(0, index);
}
this.emit("message", buffer);
2022-05-21 14:02:00 +00:00
}
keepAlive()
{
if (this.keepAlives.length >= KEEP_ALIVE_LIMIT)
{
this.debug?.("UDP socket has not received enough responses from Discord - closing socket");
2022-05-21 14:02:00 +00:00
this.destroy();
return;
}
this.keepAliveBuffer.writeUInt32LE(this.keepAliveCounter, 0);
this.send(this.keepAliveBuffer);
this.keepAlives.push({
value: this.keepAliveCounter,
timestamp: Date.now()
2022-05-21 14:02:00 +00:00
});
this.keepAliveCounter++;
if (this.keepAliveCounter > MAX_COUNTER_VALUE)
{
2022-05-21 14:02:00 +00:00
this.keepAliveCounter = 0;
}
}
send(buffer)
{
2022-05-21 14:02:00 +00:00
return this.socket.send(buffer, this.remote.port, this.remote.ip);
}
destroy()
{
try
{
2022-05-21 14:02:00 +00:00
this.socket.close();
} catch {
}
2022-05-21 14:02:00 +00:00
clearInterval(this.keepAliveInterval);
}
performIPDiscovery(ssrc)
{
return new Promise((resolve2, reject) =>
{
const listener = /* @__PURE__ */ __name((message) =>
{
try
{
if (message.readUInt16BE(0) !== 2)
return;
2022-05-21 14:02:00 +00:00
const packet = parseLocalPacket(message);
this.socket.off("message", listener);
2022-05-21 14:02:00 +00:00
resolve2(packet);
} catch {
}
}, "listener");
this.socket.on("message", listener);
this.socket.once("close", () => reject(new Error("Cannot perform IP discovery - socket closed")));
2022-05-21 14:02:00 +00:00
const discoveryBuffer = Buffer.alloc(74);
discoveryBuffer.writeUInt16BE(1, 0);
discoveryBuffer.writeUInt16BE(70, 2);
discoveryBuffer.writeUInt32BE(ssrc, 4);
this.send(discoveryBuffer);
});
}
};
__name(VoiceUDPSocket, "VoiceUDPSocket");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/networking/VoiceWebSocket.ts
var import_node_events2 = require("events");
var import_v4 = require("discord-api-types/voice/v4");
var import_ws = __toESM(require("ws"));
var VoiceWebSocket = class extends import_node_events2.EventEmitter
{
constructor(address, debug)
{
2022-05-21 14:02:00 +00:00
super();
__publicField(this, "heartbeatInterval");
__publicField(this, "lastHeartbeatAck");
__publicField(this, "lastHeartbeatSend");
__publicField(this, "missedHeartbeats", 0);
__publicField(this, "ping");
__publicField(this, "debug");
__publicField(this, "ws");
2022-05-21 14:02:00 +00:00
this.ws = new import_ws.default(address);
this.ws.onmessage = (e) => this.onMessage(e);
this.ws.onopen = (e) => this.emit("open", e);
this.ws.onerror = (e) => this.emit("error", e instanceof Error ? e : e.error);
this.ws.onclose = (e) => this.emit("close", e);
2022-05-21 14:02:00 +00:00
this.lastHeartbeatAck = 0;
this.lastHeartbeatSend = 0;
this.debug = debug ? (message) => this.emit("debug", message) : null;
2022-05-21 14:02:00 +00:00
}
destroy()
{
try
{
this.debug?.("destroyed");
2022-05-21 14:02:00 +00:00
this.setHeartbeatInterval(-1);
this.ws.close(1e3);
} catch (error)
{
2022-05-21 14:02:00 +00:00
const e = error;
this.emit("error", e);
2022-05-21 14:02:00 +00:00
}
}
onMessage(event)
{
if (typeof event.data !== "string")
return;
2022-05-21 14:02:00 +00:00
this.debug?.(`<< ${event.data}`);
let packet;
try
{
2022-05-21 14:02:00 +00:00
packet = JSON.parse(event.data);
} catch (error)
{
2022-05-21 14:02:00 +00:00
const e = error;
this.emit("error", e);
2022-05-21 14:02:00 +00:00
return;
}
if (packet.op === import_v4.VoiceOpcodes.HeartbeatAck)
{
2022-05-21 14:02:00 +00:00
this.lastHeartbeatAck = Date.now();
this.missedHeartbeats = 0;
this.ping = this.lastHeartbeatAck - this.lastHeartbeatSend;
}
this.emit("packet", packet);
2022-05-21 14:02:00 +00:00
}
sendPacket(packet)
{
try
{
2022-05-21 14:02:00 +00:00
const stringified = JSON.stringify(packet);
this.debug?.(`>> ${stringified}`);
return this.ws.send(stringified);
} catch (error)
{
2022-05-21 14:02:00 +00:00
const e = error;
this.emit("error", e);
2022-05-21 14:02:00 +00:00
}
}
sendHeartbeat()
{
2022-05-21 14:02:00 +00:00
this.lastHeartbeatSend = Date.now();
this.missedHeartbeats++;
const nonce2 = this.lastHeartbeatSend;
return this.sendPacket({
op: import_v4.VoiceOpcodes.Heartbeat,
d: nonce2
2022-05-21 14:02:00 +00:00
});
}
setHeartbeatInterval(ms)
{
if (typeof this.heartbeatInterval !== "undefined")
clearInterval(this.heartbeatInterval);
if (ms > 0)
{
this.heartbeatInterval = setInterval(() =>
{
if (this.lastHeartbeatSend !== 0 && this.missedHeartbeats >= 3)
{
2022-05-21 14:02:00 +00:00
this.ws.close();
this.setHeartbeatInterval(-1);
}
this.sendHeartbeat();
}, ms);
}
}
};
__name(VoiceWebSocket, "VoiceWebSocket");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/util/Secretbox.ts
2022-05-21 14:02:00 +00:00
var libs = {
"sodium-native": (sodium) => ({
open: (buffer, nonce2, secretKey) =>
{
if (buffer)
{
2022-05-21 14:02:00 +00:00
const output = Buffer.allocUnsafe(buffer.length - sodium.crypto_box_MACBYTES);
if (sodium.crypto_secretbox_open_easy(output, buffer, nonce2, secretKey))
return output;
2022-05-21 14:02:00 +00:00
}
return null;
},
close: (opusPacket, nonce2, secretKey) =>
{
2022-05-21 14:02:00 +00:00
const output = Buffer.allocUnsafe(opusPacket.length + sodium.crypto_box_MACBYTES);
sodium.crypto_secretbox_easy(output, opusPacket, nonce2, secretKey);
return output;
},
random: (n, buffer = Buffer.allocUnsafe(n)) =>
{
2022-05-21 14:02:00 +00:00
sodium.randombytes_buf(buffer);
return buffer;
}
2022-05-21 14:02:00 +00:00
}),
sodium: (sodium) => ({
2022-05-21 14:02:00 +00:00
open: sodium.api.crypto_secretbox_open_easy,
close: sodium.api.crypto_secretbox_easy,
random: (n, buffer = Buffer.allocUnsafe(n)) =>
{
2022-05-21 14:02:00 +00:00
sodium.api.randombytes_buf(buffer);
return buffer;
}
2022-05-21 14:02:00 +00:00
}),
"libsodium-wrappers": (sodium) => ({
2022-05-21 14:02:00 +00:00
open: sodium.crypto_secretbox_open_easy,
close: sodium.crypto_secretbox_easy,
random: sodium.randombytes_buf
2022-05-21 14:02:00 +00:00
}),
tweetnacl: (tweetnacl) => ({
2022-05-21 14:02:00 +00:00
open: tweetnacl.secretbox.open,
close: tweetnacl.secretbox,
random: tweetnacl.randomBytes
})
2022-05-21 14:02:00 +00:00
};
var fallbackError = /* @__PURE__ */ __name(() =>
{
2022-05-21 14:02:00 +00:00
throw new Error(`Cannot play audio as no valid encryption package is installed.
- Install sodium, libsodium-wrappers, or tweetnacl.
- Use the generateDependencyReport() function for more information.
`);
}, "fallbackError");
2022-05-21 14:02:00 +00:00
var methods = {
open: fallbackError,
close: fallbackError,
random: fallbackError
2022-05-21 14:02:00 +00:00
};
void (async () =>
{
for (const libName of Object.keys(libs))
{
try
{
2022-05-21 14:02:00 +00:00
const lib = require(libName);
if (libName === "libsodium-wrappers" && lib.ready)
await lib.ready;
2022-05-21 14:02:00 +00:00
Object.assign(methods, libs[libName](lib));
break;
} catch {
}
2022-05-21 14:02:00 +00:00
}
})();
2022-06-26 02:40:04 +00:00
// src/util/util.ts
var noop = /* @__PURE__ */ __name(() =>
{
}, "noop");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/networking/Networking.ts
2022-05-21 14:02:00 +00:00
var CHANNELS = 2;
var TIMESTAMP_INC = 48e3 / 100 * CHANNELS;
2022-05-21 14:02:00 +00:00
var MAX_NONCE_SIZE = 2 ** 32 - 1;
var SUPPORTED_ENCRYPTION_MODES = ["xsalsa20_poly1305_lite", "xsalsa20_poly1305_suffix", "xsalsa20_poly1305"];
2022-05-21 14:02:00 +00:00
var nonce = Buffer.alloc(24);
function stringifyState(state)
{
2022-05-21 14:02:00 +00:00
return JSON.stringify({
...state,
ws: Reflect.has(state, "ws"),
udp: Reflect.has(state, "udp")
2022-05-21 14:02:00 +00:00
});
}
__name(stringifyState, "stringifyState");
function chooseEncryptionMode(options)
{
const option = options.find((option2) => SUPPORTED_ENCRYPTION_MODES.includes(option2));
if (!option)
{
throw new Error(`No compatible encryption modes. Available include: ${options.join(", ")}`);
2022-05-21 14:02:00 +00:00
}
return option;
}
__name(chooseEncryptionMode, "chooseEncryptionMode");
function randomNBit(n)
{
2022-05-21 14:02:00 +00:00
return Math.floor(Math.random() * 2 ** n);
}
__name(randomNBit, "randomNBit");
var Networking = class extends import_node_events3.EventEmitter
{
constructor(options, debug)
{
2022-05-21 14:02:00 +00:00
super();
__publicField(this, "_state");
__publicField(this, "debug");
2022-05-21 14:02:00 +00:00
this.onWsOpen = this.onWsOpen.bind(this);
this.onChildError = this.onChildError.bind(this);
this.onWsPacket = this.onWsPacket.bind(this);
this.onWsClose = this.onWsClose.bind(this);
this.onWsDebug = this.onWsDebug.bind(this);
this.onUdpDebug = this.onUdpDebug.bind(this);
this.onUdpClose = this.onUdpClose.bind(this);
this.debug = debug ? (message) => this.emit("debug", message) : null;
2022-05-21 14:02:00 +00:00
this._state = {
code: 0 /* OpeningWs */,
ws: this.createWebSocket(options.endpoint),
connectionOptions: options
2022-05-21 14:02:00 +00:00
};
}
destroy()
{
2022-05-21 14:02:00 +00:00
this.state = {
code: 6 /* Closed */
2022-05-21 14:02:00 +00:00
};
}
get state()
{
2022-05-21 14:02:00 +00:00
return this._state;
}
set state(newState)
{
const oldWs = Reflect.get(this._state, "ws");
const newWs = Reflect.get(newState, "ws");
if (oldWs && oldWs !== newWs)
{
oldWs.off("debug", this.onWsDebug);
oldWs.on("error", noop);
oldWs.off("error", this.onChildError);
oldWs.off("open", this.onWsOpen);
oldWs.off("packet", this.onWsPacket);
oldWs.off("close", this.onWsClose);
2022-05-21 14:02:00 +00:00
oldWs.destroy();
}
const oldUdp = Reflect.get(this._state, "udp");
const newUdp = Reflect.get(newState, "udp");
if (oldUdp && oldUdp !== newUdp)
{
oldUdp.on("error", noop);
oldUdp.off("error", this.onChildError);
oldUdp.off("close", this.onUdpClose);
oldUdp.off("debug", this.onUdpDebug);
2022-05-21 14:02:00 +00:00
oldUdp.destroy();
}
const oldState = this._state;
this._state = newState;
this.emit("stateChange", oldState, newState);
2022-05-21 14:02:00 +00:00
this.debug?.(`state change:
from ${stringifyState(oldState)}
to ${stringifyState(newState)}`);
}
createWebSocket(endpoint)
{
2022-05-21 14:02:00 +00:00
const ws = new VoiceWebSocket(`wss://${endpoint}?v=4`, Boolean(this.debug));
ws.on("error", this.onChildError);
ws.once("open", this.onWsOpen);
ws.on("packet", this.onWsPacket);
ws.once("close", this.onWsClose);
ws.on("debug", this.onWsDebug);
2022-05-21 14:02:00 +00:00
return ws;
}
onChildError(error)
{
this.emit("error", error);
2022-05-21 14:02:00 +00:00
}
onWsOpen()
{
if (this.state.code === 0 /* OpeningWs */)
{
2022-05-21 14:02:00 +00:00
const packet = {
op: import_v42.VoiceOpcodes.Identify,
d: {
server_id: this.state.connectionOptions.serverId,
user_id: this.state.connectionOptions.userId,
session_id: this.state.connectionOptions.sessionId,
token: this.state.connectionOptions.token
}
2022-05-21 14:02:00 +00:00
};
this.state.ws.sendPacket(packet);
this.state = {
...this.state,
code: 1 /* Identifying */
2022-05-21 14:02:00 +00:00
};
} else if (this.state.code === 5 /* Resuming */)
{
2022-05-21 14:02:00 +00:00
const packet = {
op: import_v42.VoiceOpcodes.Resume,
d: {
server_id: this.state.connectionOptions.serverId,
session_id: this.state.connectionOptions.sessionId,
token: this.state.connectionOptions.token
}
2022-05-21 14:02:00 +00:00
};
this.state.ws.sendPacket(packet);
}
}
onWsClose({ code })
{
2022-05-21 14:02:00 +00:00
const canResume = code === 4015 || code < 4e3;
if (canResume && this.state.code === 4 /* Ready */)
{
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
code: 5 /* Resuming */,
ws: this.createWebSocket(this.state.connectionOptions.endpoint)
2022-05-21 14:02:00 +00:00
};
} else if (this.state.code !== 6 /* Closed */)
{
2022-05-21 14:02:00 +00:00
this.destroy();
this.emit("close", code);
2022-05-21 14:02:00 +00:00
}
}
onUdpClose()
{
if (this.state.code === 4 /* Ready */)
{
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
code: 5 /* Resuming */,
ws: this.createWebSocket(this.state.connectionOptions.endpoint)
2022-05-21 14:02:00 +00:00
};
}
}
onWsPacket(packet)
{
if (packet.op === import_v42.VoiceOpcodes.Hello && this.state.code !== 6 /* Closed */)
{
2022-05-21 14:02:00 +00:00
this.state.ws.setHeartbeatInterval(packet.d.heartbeat_interval);
} else if (packet.op === import_v42.VoiceOpcodes.Ready && this.state.code === 1 /* Identifying */)
{
2022-05-21 14:02:00 +00:00
const { ip, port, ssrc, modes } = packet.d;
const udp = new VoiceUDPSocket({ ip, port });
udp.on("error", this.onChildError);
udp.on("debug", this.onUdpDebug);
udp.once("close", this.onUdpClose);
udp.performIPDiscovery(ssrc).then((localConfig) =>
{
if (this.state.code !== 2 /* UdpHandshaking */)
return;
this.state.ws.sendPacket({
op: import_v42.VoiceOpcodes.SelectProtocol,
d: {
protocol: "udp",
data: {
address: localConfig.ip,
port: localConfig.port,
mode: chooseEncryptionMode(modes)
}
}
});
this.state = {
...this.state,
code: 3 /* SelectingProtocol */
};
}).catch((error) => this.emit("error", error));
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
code: 2 /* UdpHandshaking */,
udp,
connectionData: {
ssrc
}
2022-05-21 14:02:00 +00:00
};
} else if (packet.op === import_v42.VoiceOpcodes.SessionDescription && this.state.code === 3 /* SelectingProtocol */)
{
2022-05-21 14:02:00 +00:00
const { mode: encryptionMode, secret_key: secretKey } = packet.d;
this.state = {
...this.state,
code: 4 /* Ready */,
connectionData: {
...this.state.connectionData,
encryptionMode,
secretKey: new Uint8Array(secretKey),
sequence: randomNBit(16),
timestamp: randomNBit(32),
nonce: 0,
nonceBuffer: Buffer.alloc(24),
speaking: false,
packetsPlayed: 0
}
2022-05-21 14:02:00 +00:00
};
} else if (packet.op === import_v42.VoiceOpcodes.Resumed && this.state.code === 5 /* Resuming */)
{
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
code: 4 /* Ready */
2022-05-21 14:02:00 +00:00
};
this.state.connectionData.speaking = false;
}
}
onWsDebug(message)
{
2022-05-21 14:02:00 +00:00
this.debug?.(`[WS] ${message}`);
}
onUdpDebug(message)
{
2022-05-21 14:02:00 +00:00
this.debug?.(`[UDP] ${message}`);
}
prepareAudioPacket(opusPacket)
{
2022-05-21 14:02:00 +00:00
const state = this.state;
if (state.code !== 4 /* Ready */)
return;
2022-05-21 14:02:00 +00:00
state.preparedPacket = this.createAudioPacket(opusPacket, state.connectionData);
return state.preparedPacket;
}
dispatchAudio()
{
2022-05-21 14:02:00 +00:00
const state = this.state;
if (state.code !== 4 /* Ready */)
return false;
if (typeof state.preparedPacket !== "undefined")
{
2022-05-21 14:02:00 +00:00
this.playAudioPacket(state.preparedPacket);
state.preparedPacket = void 0;
return true;
}
return false;
}
playAudioPacket(audioPacket)
{
2022-05-21 14:02:00 +00:00
const state = this.state;
if (state.code !== 4 /* Ready */)
return;
2022-05-21 14:02:00 +00:00
const { connectionData } = state;
connectionData.packetsPlayed++;
connectionData.sequence++;
connectionData.timestamp += TIMESTAMP_INC;
if (connectionData.sequence >= 2 ** 16)
connectionData.sequence = 0;
if (connectionData.timestamp >= 2 ** 32)
connectionData.timestamp = 0;
2022-05-21 14:02:00 +00:00
this.setSpeaking(true);
state.udp.send(audioPacket);
}
setSpeaking(speaking)
{
2022-05-21 14:02:00 +00:00
const state = this.state;
if (state.code !== 4 /* Ready */)
return;
if (state.connectionData.speaking === speaking)
return;
2022-05-21 14:02:00 +00:00
state.connectionData.speaking = speaking;
state.ws.sendPacket({
op: import_v42.VoiceOpcodes.Speaking,
d: {
speaking: speaking ? 1 : 0,
delay: 0,
ssrc: state.connectionData.ssrc
}
2022-05-21 14:02:00 +00:00
});
}
createAudioPacket(opusPacket, connectionData)
{
2022-05-21 14:02:00 +00:00
const packetBuffer = Buffer.alloc(12);
packetBuffer[0] = 128;
packetBuffer[1] = 120;
const { sequence, timestamp, ssrc } = connectionData;
packetBuffer.writeUIntBE(sequence, 2, 2);
packetBuffer.writeUIntBE(timestamp, 4, 4);
packetBuffer.writeUIntBE(ssrc, 8, 4);
packetBuffer.copy(nonce, 0, 0, 12);
return Buffer.concat([packetBuffer, ...this.encryptOpusPacket(opusPacket, connectionData)]);
}
encryptOpusPacket(opusPacket, connectionData)
{
2022-05-21 14:02:00 +00:00
const { secretKey, encryptionMode } = connectionData;
if (encryptionMode === "xsalsa20_poly1305_lite")
{
2022-05-21 14:02:00 +00:00
connectionData.nonce++;
if (connectionData.nonce > MAX_NONCE_SIZE)
connectionData.nonce = 0;
2022-05-21 14:02:00 +00:00
connectionData.nonceBuffer.writeUInt32BE(connectionData.nonce, 0);
return [
methods.close(opusPacket, connectionData.nonceBuffer, secretKey),
connectionData.nonceBuffer.slice(0, 4)
];
} else if (encryptionMode === "xsalsa20_poly1305_suffix")
{
2022-05-21 14:02:00 +00:00
const random = methods.random(24, connectionData.nonceBuffer);
return [methods.close(opusPacket, random, secretKey), random];
}
return [methods.close(opusPacket, nonce, secretKey)];
}
};
__name(Networking, "Networking");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/receive/VoiceReceiver.ts
var import_v43 = require("discord-api-types/voice/v4");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/receive/AudioReceiveStream.ts
var import_node_stream = require("stream");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/audio/AudioPlayer.ts
var import_node_events4 = __toESM(require("events"));
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/audio/AudioPlayerError.ts
var AudioPlayerError = class extends Error
{
constructor(error, resource)
{
2022-05-21 14:02:00 +00:00
super(error.message);
__publicField(this, "resource");
2022-05-21 14:02:00 +00:00
this.resource = resource;
this.name = error.name;
this.stack = error.stack;
}
};
__name(AudioPlayerError, "AudioPlayerError");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/audio/PlayerSubscription.ts
var PlayerSubscription = class
{
constructor(connection, player)
{
__publicField(this, "connection");
__publicField(this, "player");
2022-05-21 14:02:00 +00:00
this.connection = connection;
this.player = player;
}
unsubscribe()
{
this.connection["onSubscriptionRemoved"](this);
this.player["unsubscribe"](this);
2022-05-21 14:02:00 +00:00
}
};
__name(PlayerSubscription, "PlayerSubscription");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/audio/AudioPlayer.ts
2022-05-21 14:02:00 +00:00
var SILENCE_FRAME = Buffer.from([248, 255, 254]);
var NoSubscriberBehavior = /* @__PURE__ */ ((NoSubscriberBehavior2) =>
{
NoSubscriberBehavior2["Pause"] = "pause";
NoSubscriberBehavior2["Play"] = "play";
NoSubscriberBehavior2["Stop"] = "stop";
2022-05-21 14:02:00 +00:00
return NoSubscriberBehavior2;
})(NoSubscriberBehavior || {});
var AudioPlayerStatus = /* @__PURE__ */ ((AudioPlayerStatus2) =>
{
AudioPlayerStatus2["Idle"] = "idle";
AudioPlayerStatus2["Buffering"] = "buffering";
AudioPlayerStatus2["Paused"] = "paused";
AudioPlayerStatus2["Playing"] = "playing";
AudioPlayerStatus2["AutoPaused"] = "autopaused";
2022-05-21 14:02:00 +00:00
return AudioPlayerStatus2;
})(AudioPlayerStatus || {});
function stringifyState2(state)
{
2022-05-21 14:02:00 +00:00
return JSON.stringify({
...state,
resource: Reflect.has(state, "resource"),
stepTimeout: Reflect.has(state, "stepTimeout")
2022-05-21 14:02:00 +00:00
});
}
__name(stringifyState2, "stringifyState");
var AudioPlayer = class extends import_node_events4.default
{
constructor(options = {})
{
2022-05-21 14:02:00 +00:00
super();
__publicField(this, "_state");
__publicField(this, "subscribers", []);
__publicField(this, "behaviors");
__publicField(this, "debug");
this._state = { status: "idle" /* Idle */ };
2022-05-21 14:02:00 +00:00
this.behaviors = {
noSubscriber: "pause" /* Pause */,
2022-05-21 14:02:00 +00:00
maxMissedFrames: 5,
...options.behaviors
2022-05-21 14:02:00 +00:00
};
this.debug = options.debug === false ? null : (message) => this.emit("debug", message);
2022-05-21 14:02:00 +00:00
}
get playable()
{
return this.subscribers.filter(({ connection }) => connection.state.status === "ready" /* Ready */).map(({ connection }) => connection);
2022-05-21 14:02:00 +00:00
}
subscribe(connection)
{
const existingSubscription = this.subscribers.find((subscription) => subscription.connection === connection);
if (!existingSubscription)
{
2022-05-21 14:02:00 +00:00
const subscription = new PlayerSubscription(connection, this);
this.subscribers.push(subscription);
setImmediate(() => this.emit("subscribe", subscription));
2022-05-21 14:02:00 +00:00
return subscription;
}
return existingSubscription;
}
unsubscribe(subscription)
{
2022-05-21 14:02:00 +00:00
const index = this.subscribers.indexOf(subscription);
const exists = index !== -1;
if (exists)
{
2022-05-21 14:02:00 +00:00
this.subscribers.splice(index, 1);
subscription.connection.setSpeaking(false);
this.emit("unsubscribe", subscription);
2022-05-21 14:02:00 +00:00
}
return exists;
}
get state()
{
2022-05-21 14:02:00 +00:00
return this._state;
}
set state(newState)
{
2022-05-21 14:02:00 +00:00
const oldState = this._state;
const newResource = Reflect.get(newState, "resource");
if (oldState.status !== "idle" /* Idle */ && oldState.resource !== newResource)
{
oldState.resource.playStream.on("error", noop);
oldState.resource.playStream.off("error", oldState.onStreamError);
2022-05-21 14:02:00 +00:00
oldState.resource.audioPlayer = void 0;
oldState.resource.playStream.destroy();
oldState.resource.playStream.read();
}
if (oldState.status === "buffering" /* Buffering */ && (newState.status !== "buffering" /* Buffering */ || newState.resource !== oldState.resource))
{
oldState.resource.playStream.off("end", oldState.onFailureCallback);
oldState.resource.playStream.off("close", oldState.onFailureCallback);
oldState.resource.playStream.off("finish", oldState.onFailureCallback);
oldState.resource.playStream.off("readable", oldState.onReadableCallback);
2022-05-21 14:02:00 +00:00
}
if (newState.status === "idle" /* Idle */)
{
2022-05-21 14:02:00 +00:00
this._signalStopSpeaking();
deleteAudioPlayer(this);
}
if (newResource)
{
2022-05-21 14:02:00 +00:00
addAudioPlayer(this);
}
const didChangeResources = oldState.status !== "idle" /* Idle */ && newState.status === "playing" /* Playing */ && oldState.resource !== newState.resource;
2022-05-21 14:02:00 +00:00
this._state = newState;
this.emit("stateChange", oldState, this._state);
if (oldState.status !== newState.status || didChangeResources)
{
2022-05-21 14:02:00 +00:00
this.emit(newState.status, oldState, this._state);
}
this.debug?.(`state change:
from ${stringifyState2(oldState)}
to ${stringifyState2(newState)}`);
}
play(resource)
{
if (resource.ended)
{
throw new Error("Cannot play a resource that has already ended.");
2022-05-21 14:02:00 +00:00
}
if (resource.audioPlayer)
{
if (resource.audioPlayer === this)
{
2022-05-21 14:02:00 +00:00
return;
}
throw new Error("Resource is already being played by another audio player.");
2022-05-21 14:02:00 +00:00
}
resource.audioPlayer = this;
const onStreamError = /* @__PURE__ */ __name((error) =>
{
if (this.state.status !== "idle" /* Idle */)
{
this.emit("error", new AudioPlayerError(error, this.state.resource));
2022-05-21 14:02:00 +00:00
}
if (this.state.status !== "idle" /* Idle */ && this.state.resource === resource)
{
2022-05-21 14:02:00 +00:00
this.state = {
status: "idle" /* Idle */
2022-05-21 14:02:00 +00:00
};
}
}, "onStreamError");
resource.playStream.once("error", onStreamError);
if (resource.started)
{
2022-05-21 14:02:00 +00:00
this.state = {
status: "playing" /* Playing */,
2022-05-21 14:02:00 +00:00
missedFrames: 0,
playbackDuration: 0,
resource,
onStreamError
2022-05-21 14:02:00 +00:00
};
} else
{
const onReadableCallback = /* @__PURE__ */ __name(() =>
{
if (this.state.status === "buffering" /* Buffering */ && this.state.resource === resource)
{
2022-05-21 14:02:00 +00:00
this.state = {
status: "playing" /* Playing */,
2022-05-21 14:02:00 +00:00
missedFrames: 0,
playbackDuration: 0,
resource,
onStreamError
2022-05-21 14:02:00 +00:00
};
}
}, "onReadableCallback");
const onFailureCallback = /* @__PURE__ */ __name(() =>
{
if (this.state.status === "buffering" /* Buffering */ && this.state.resource === resource)
{
2022-05-21 14:02:00 +00:00
this.state = {
status: "idle" /* Idle */
2022-05-21 14:02:00 +00:00
};
}
}, "onFailureCallback");
resource.playStream.once("readable", onReadableCallback);
resource.playStream.once("end", onFailureCallback);
resource.playStream.once("close", onFailureCallback);
resource.playStream.once("finish", onFailureCallback);
2022-05-21 14:02:00 +00:00
this.state = {
status: "buffering" /* Buffering */,
2022-05-21 14:02:00 +00:00
resource,
onReadableCallback,
onFailureCallback,
onStreamError
2022-05-21 14:02:00 +00:00
};
}
}
pause(interpolateSilence = true)
{
if (this.state.status !== "playing" /* Playing */)
return false;
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
status: "paused" /* Paused */,
silencePacketsRemaining: interpolateSilence ? 5 : 0
2022-05-21 14:02:00 +00:00
};
return true;
}
unpause()
{
if (this.state.status !== "paused" /* Paused */)
return false;
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
status: "playing" /* Playing */,
missedFrames: 0
2022-05-21 14:02:00 +00:00
};
return true;
}
stop(force = false)
{
if (this.state.status === "idle" /* Idle */)
return false;
if (force || this.state.resource.silencePaddingFrames === 0)
{
2022-05-21 14:02:00 +00:00
this.state = {
status: "idle" /* Idle */
2022-05-21 14:02:00 +00:00
};
} else if (this.state.resource.silenceRemaining === -1)
{
2022-05-21 14:02:00 +00:00
this.state.resource.silenceRemaining = this.state.resource.silencePaddingFrames;
}
return true;
}
checkPlayable()
{
2022-05-21 14:02:00 +00:00
const state = this._state;
if (state.status === "idle" /* Idle */ || state.status === "buffering" /* Buffering */)
return false;
if (!state.resource.readable)
{
2022-05-21 14:02:00 +00:00
this.state = {
status: "idle" /* Idle */
2022-05-21 14:02:00 +00:00
};
return false;
}
return true;
}
_stepDispatch()
{
2022-05-21 14:02:00 +00:00
const state = this._state;
if (state.status === "idle" /* Idle */ || state.status === "buffering" /* Buffering */)
return;
this.playable.forEach((connection) => connection.dispatchAudio());
2022-05-21 14:02:00 +00:00
}
_stepPrepare()
{
2022-05-21 14:02:00 +00:00
const state = this._state;
if (state.status === "idle" /* Idle */ || state.status === "buffering" /* Buffering */)
return;
2022-05-21 14:02:00 +00:00
const playable = this.playable;
if (state.status === "autopaused" /* AutoPaused */ && playable.length > 0)
{
2022-05-21 14:02:00 +00:00
this.state = {
...state,
status: "playing" /* Playing */,
missedFrames: 0
2022-05-21 14:02:00 +00:00
};
}
if (state.status === "paused" /* Paused */ || state.status === "autopaused" /* AutoPaused */)
{
if (state.silencePacketsRemaining > 0)
{
2022-05-21 14:02:00 +00:00
state.silencePacketsRemaining--;
this._preparePacket(SILENCE_FRAME, playable, state);
if (state.silencePacketsRemaining === 0)
{
2022-05-21 14:02:00 +00:00
this._signalStopSpeaking();
}
}
return;
}
if (playable.length === 0)
{
if (this.behaviors.noSubscriber === "pause" /* Pause */)
{
2022-05-21 14:02:00 +00:00
this.state = {
...state,
status: "autopaused" /* AutoPaused */,
silencePacketsRemaining: 5
2022-05-21 14:02:00 +00:00
};
return;
} else if (this.behaviors.noSubscriber === "stop" /* Stop */)
{
2022-05-21 14:02:00 +00:00
this.stop(true);
}
}
const packet = state.resource.read();
if (state.status === "playing" /* Playing */)
{
if (packet)
{
2022-05-21 14:02:00 +00:00
this._preparePacket(packet, playable, state);
state.missedFrames = 0;
} else
{
2022-05-21 14:02:00 +00:00
this._preparePacket(SILENCE_FRAME, playable, state);
state.missedFrames++;
if (state.missedFrames >= this.behaviors.maxMissedFrames)
{
2022-05-21 14:02:00 +00:00
this.stop();
}
}
}
}
_signalStopSpeaking()
{
2022-05-21 14:02:00 +00:00
return this.subscribers.forEach(({ connection }) => connection.setSpeaking(false));
}
_preparePacket(packet, receivers, state)
{
2022-05-21 14:02:00 +00:00
state.playbackDuration += 20;
receivers.forEach((connection) => connection.prepareAudioPacket(packet));
2022-05-21 14:02:00 +00:00
}
};
__name(AudioPlayer, "AudioPlayer");
function createAudioPlayer(options)
{
2022-05-21 14:02:00 +00:00
return new AudioPlayer(options);
}
__name(createAudioPlayer, "createAudioPlayer");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/receive/AudioReceiveStream.ts
var EndBehaviorType = /* @__PURE__ */ ((EndBehaviorType2) =>
{
EndBehaviorType2[EndBehaviorType2["Manual"] = 0] = "Manual";
EndBehaviorType2[EndBehaviorType2["AfterSilence"] = 1] = "AfterSilence";
EndBehaviorType2[EndBehaviorType2["AfterInactivity"] = 2] = "AfterInactivity";
2022-05-21 14:02:00 +00:00
return EndBehaviorType2;
})(EndBehaviorType || {});
function createDefaultAudioReceiveStreamOptions()
{
2022-05-21 14:02:00 +00:00
return {
end: {
behavior: 0 /* Manual */
}
2022-05-21 14:02:00 +00:00
};
}
__name(createDefaultAudioReceiveStreamOptions, "createDefaultAudioReceiveStreamOptions");
var AudioReceiveStream = class extends import_node_stream.Readable
{
constructor({ end, ...options })
{
2022-05-21 14:02:00 +00:00
super({
...options,
objectMode: true
2022-05-21 14:02:00 +00:00
});
__publicField(this, "end");
__publicField(this, "endTimeout");
2022-05-21 14:02:00 +00:00
this.end = end;
}
push(buffer)
{
if (buffer)
{
if (this.end.behavior === 2 /* AfterInactivity */ || this.end.behavior === 1 /* AfterSilence */ && (buffer.compare(SILENCE_FRAME) !== 0 || typeof this.endTimeout === "undefined"))
{
2022-05-21 14:02:00 +00:00
this.renewEndTimeout(this.end);
}
}
return super.push(buffer);
}
renewEndTimeout(end)
{
if (this.endTimeout)
{
2022-05-21 14:02:00 +00:00
clearTimeout(this.endTimeout);
}
this.endTimeout = setTimeout(() => this.push(null), end.duration);
}
_read()
{
}
2022-05-21 14:02:00 +00:00
};
__name(AudioReceiveStream, "AudioReceiveStream");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/receive/SSRCMap.ts
var import_node_events5 = require("events");
var SSRCMap = class extends import_node_events5.EventEmitter
{
constructor()
{
2022-05-21 14:02:00 +00:00
super();
__publicField(this, "map");
2022-05-21 14:02:00 +00:00
this.map = /* @__PURE__ */ new Map();
}
update(data)
{
2022-05-21 14:02:00 +00:00
const existing = this.map.get(data.audioSSRC);
const newValue = {
...this.map.get(data.audioSSRC),
...data
2022-05-21 14:02:00 +00:00
};
this.map.set(data.audioSSRC, newValue);
if (!existing)
this.emit("create", newValue);
this.emit("update", existing, newValue);
}
get(target)
{
if (typeof target === "number")
{
2022-05-21 14:02:00 +00:00
return this.map.get(target);
}
for (const data of this.map.values())
{
if (data.userId === target)
{
2022-05-21 14:02:00 +00:00
return data;
}
}
return void 0;
}
delete(target)
{
if (typeof target === "number")
{
2022-05-21 14:02:00 +00:00
const existing = this.map.get(target);
if (existing)
{
2022-05-21 14:02:00 +00:00
this.map.delete(target);
this.emit("delete", existing);
2022-05-21 14:02:00 +00:00
}
return existing;
}
for (const [audioSSRC, data] of this.map.entries())
{
if (data.userId === target)
{
2022-05-21 14:02:00 +00:00
this.map.delete(audioSSRC);
this.emit("delete", data);
2022-05-21 14:02:00 +00:00
return data;
}
}
return void 0;
}
};
__name(SSRCMap, "SSRCMap");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/receive/SpeakingMap.ts
var import_node_events6 = require("events");
var _SpeakingMap = class extends import_node_events6.EventEmitter
{
constructor()
{
2022-05-21 14:02:00 +00:00
super();
__publicField(this, "users");
__publicField(this, "speakingTimeouts");
2022-05-21 14:02:00 +00:00
this.users = /* @__PURE__ */ new Map();
this.speakingTimeouts = /* @__PURE__ */ new Map();
}
onPacket(userId)
{
2022-05-21 14:02:00 +00:00
const timeout = this.speakingTimeouts.get(userId);
if (timeout)
{
2022-05-21 14:02:00 +00:00
clearTimeout(timeout);
} else
{
2022-05-21 14:02:00 +00:00
this.users.set(userId, Date.now());
this.emit("start", userId);
2022-05-21 14:02:00 +00:00
}
this.startTimeout(userId);
}
startTimeout(userId)
{
this.speakingTimeouts.set(userId, setTimeout(() =>
{
this.emit("end", userId);
this.speakingTimeouts.delete(userId);
this.users.delete(userId);
}, _SpeakingMap.DELAY));
2022-05-21 14:02:00 +00:00
}
};
var SpeakingMap = _SpeakingMap;
__name(SpeakingMap, "SpeakingMap");
__publicField(SpeakingMap, "DELAY", 100);
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/receive/VoiceReceiver.ts
var VoiceReceiver = class
{
constructor(voiceConnection)
{
__publicField(this, "voiceConnection");
__publicField(this, "ssrcMap");
__publicField(this, "subscriptions");
__publicField(this, "connectionData");
__publicField(this, "speaking");
2022-05-21 14:02:00 +00:00
this.voiceConnection = voiceConnection;
this.ssrcMap = new SSRCMap();
this.speaking = new SpeakingMap();
this.subscriptions = /* @__PURE__ */ new Map();
this.connectionData = {};
this.onWsPacket = this.onWsPacket.bind(this);
this.onUdpMessage = this.onUdpMessage.bind(this);
}
onWsPacket(packet)
{
if (packet.op === import_v43.VoiceOpcodes.ClientDisconnect && typeof packet.d?.user_id === "string")
{
2022-05-21 14:02:00 +00:00
this.ssrcMap.delete(packet.d.user_id);
} else if (packet.op === import_v43.VoiceOpcodes.Speaking && typeof packet.d?.user_id === "string" && typeof packet.d?.ssrc === "number")
{
2022-05-21 14:02:00 +00:00
this.ssrcMap.update({ userId: packet.d.user_id, audioSSRC: packet.d.ssrc });
} else if (packet.op === import_v43.VoiceOpcodes.ClientConnect && typeof packet.d?.user_id === "string" && typeof packet.d?.audio_ssrc === "number")
{
2022-05-21 14:02:00 +00:00
this.ssrcMap.update({
userId: packet.d.user_id,
audioSSRC: packet.d.audio_ssrc,
videoSSRC: packet.d.video_ssrc === 0 ? void 0 : packet.d.video_ssrc
2022-05-21 14:02:00 +00:00
});
}
}
decrypt(buffer, mode, nonce2, secretKey)
{
2022-05-21 14:02:00 +00:00
let end;
if (mode === "xsalsa20_poly1305_lite")
{
2022-05-21 14:02:00 +00:00
buffer.copy(nonce2, 0, buffer.length - 4);
end = buffer.length - 4;
} else if (mode === "xsalsa20_poly1305_suffix")
{
2022-05-21 14:02:00 +00:00
buffer.copy(nonce2, 0, buffer.length - 24);
end = buffer.length - 24;
} else
{
2022-05-21 14:02:00 +00:00
buffer.copy(nonce2, 0, 0, 12);
}
const decrypted = methods.open(buffer.slice(12, end), nonce2, secretKey);
if (!decrypted)
return;
2022-05-21 14:02:00 +00:00
return Buffer.from(decrypted);
}
parsePacket(buffer, mode, nonce2, secretKey)
{
2022-05-21 14:02:00 +00:00
let packet = this.decrypt(buffer, mode, nonce2, secretKey);
if (!packet)
return;
if (packet[0] === 190 && packet[1] === 222)
{
2022-05-21 14:02:00 +00:00
const headerExtensionLength = packet.readUInt16BE(2);
packet = packet.subarray(4 + 4 * headerExtensionLength);
2022-05-21 14:02:00 +00:00
}
return packet;
}
onUdpMessage(msg)
{
if (msg.length <= 8)
return;
2022-05-21 14:02:00 +00:00
const ssrc = msg.readUInt32BE(8);
const userData = this.ssrcMap.get(ssrc);
if (!userData)
return;
2022-05-21 14:02:00 +00:00
this.speaking.onPacket(userData.userId);
const stream = this.subscriptions.get(userData.userId);
if (!stream)
return;
if (this.connectionData.encryptionMode && this.connectionData.nonceBuffer && this.connectionData.secretKey)
{
const packet = this.parsePacket(msg, this.connectionData.encryptionMode, this.connectionData.nonceBuffer, this.connectionData.secretKey);
if (packet)
{
2022-05-21 14:02:00 +00:00
stream.push(packet);
} else
{
stream.destroy(new Error("Failed to parse packet"));
2022-05-21 14:02:00 +00:00
}
}
}
subscribe(userId, options)
{
2022-05-21 14:02:00 +00:00
const existing = this.subscriptions.get(userId);
if (existing)
return existing;
2022-05-21 14:02:00 +00:00
const stream = new AudioReceiveStream({
...createDefaultAudioReceiveStreamOptions(),
...options
2022-05-21 14:02:00 +00:00
});
stream.once("close", () => this.subscriptions.delete(userId));
2022-05-21 14:02:00 +00:00
this.subscriptions.set(userId, stream);
return stream;
}
};
__name(VoiceReceiver, "VoiceReceiver");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/VoiceConnection.ts
var VoiceConnectionStatus = /* @__PURE__ */ ((VoiceConnectionStatus2) =>
{
VoiceConnectionStatus2["Signalling"] = "signalling";
VoiceConnectionStatus2["Connecting"] = "connecting";
VoiceConnectionStatus2["Ready"] = "ready";
VoiceConnectionStatus2["Disconnected"] = "disconnected";
VoiceConnectionStatus2["Destroyed"] = "destroyed";
2022-05-21 14:02:00 +00:00
return VoiceConnectionStatus2;
})(VoiceConnectionStatus || {});
var VoiceConnectionDisconnectReason = /* @__PURE__ */ ((VoiceConnectionDisconnectReason2) =>
{
VoiceConnectionDisconnectReason2[VoiceConnectionDisconnectReason2["WebSocketClose"] = 0] = "WebSocketClose";
VoiceConnectionDisconnectReason2[VoiceConnectionDisconnectReason2["AdapterUnavailable"] = 1] = "AdapterUnavailable";
VoiceConnectionDisconnectReason2[VoiceConnectionDisconnectReason2["EndpointRemoved"] = 2] = "EndpointRemoved";
VoiceConnectionDisconnectReason2[VoiceConnectionDisconnectReason2["Manual"] = 3] = "Manual";
2022-05-21 14:02:00 +00:00
return VoiceConnectionDisconnectReason2;
})(VoiceConnectionDisconnectReason || {});
var VoiceConnection2 = class extends import_node_events7.EventEmitter
{
constructor(joinConfig, { debug, adapterCreator })
{
2022-05-21 14:02:00 +00:00
super();
__publicField(this, "rejoinAttempts");
__publicField(this, "_state");
__publicField(this, "joinConfig");
__publicField(this, "packets");
__publicField(this, "receiver");
__publicField(this, "debug");
this.debug = debug ? (message) => this.emit("debug", message) : null;
2022-05-21 14:02:00 +00:00
this.rejoinAttempts = 0;
this.receiver = new VoiceReceiver(this);
this.onNetworkingClose = this.onNetworkingClose.bind(this);
this.onNetworkingStateChange = this.onNetworkingStateChange.bind(this);
this.onNetworkingError = this.onNetworkingError.bind(this);
this.onNetworkingDebug = this.onNetworkingDebug.bind(this);
const adapter = adapterCreator({
onVoiceServerUpdate: (data) => this.addServerPacket(data),
onVoiceStateUpdate: (data) => this.addStatePacket(data),
destroy: () => this.destroy(false)
2022-05-21 14:02:00 +00:00
});
this._state = { status: "signalling" /* Signalling */, adapter };
2022-05-21 14:02:00 +00:00
this.packets = {
server: void 0,
state: void 0
2022-05-21 14:02:00 +00:00
};
this.joinConfig = joinConfig;
}
get state()
{
2022-05-21 14:02:00 +00:00
return this._state;
}
set state(newState)
{
2022-05-21 14:02:00 +00:00
const oldState = this._state;
const oldNetworking = Reflect.get(oldState, "networking");
const newNetworking = Reflect.get(newState, "networking");
const oldSubscription = Reflect.get(oldState, "subscription");
const newSubscription = Reflect.get(newState, "subscription");
if (oldNetworking !== newNetworking)
{
if (oldNetworking)
{
oldNetworking.on("error", noop);
oldNetworking.off("debug", this.onNetworkingDebug);
oldNetworking.off("error", this.onNetworkingError);
oldNetworking.off("close", this.onNetworkingClose);
oldNetworking.off("stateChange", this.onNetworkingStateChange);
2022-05-21 14:02:00 +00:00
oldNetworking.destroy();
}
if (newNetworking)
this.updateReceiveBindings(newNetworking.state, oldNetworking?.state);
2022-05-21 14:02:00 +00:00
}
if (newState.status === "ready" /* Ready */)
{
2022-05-21 14:02:00 +00:00
this.rejoinAttempts = 0;
} else if (newState.status === "destroyed" /* Destroyed */)
{
for (const stream of this.receiver.subscriptions.values())
{
if (!stream.destroyed)
stream.destroy();
2022-05-21 14:02:00 +00:00
}
}
if (oldState.status !== "destroyed" /* Destroyed */ && newState.status === "destroyed" /* Destroyed */)
{
2022-05-21 14:02:00 +00:00
oldState.adapter.destroy();
}
this._state = newState;
if (oldSubscription && oldSubscription !== newSubscription)
{
2022-05-21 14:02:00 +00:00
oldSubscription.unsubscribe();
}
this.emit("stateChange", oldState, newState);
if (oldState.status !== newState.status)
{
2022-05-21 14:02:00 +00:00
this.emit(newState.status, oldState, newState);
}
}
addServerPacket(packet)
{
2022-05-21 14:02:00 +00:00
this.packets.server = packet;
if (packet.endpoint)
{
2022-05-21 14:02:00 +00:00
this.configureNetworking();
} else if (this.state.status !== "destroyed" /* Destroyed */)
{
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
status: "disconnected" /* Disconnected */,
reason: 2 /* EndpointRemoved */
2022-05-21 14:02:00 +00:00
};
}
}
addStatePacket(packet)
{
2022-05-21 14:02:00 +00:00
this.packets.state = packet;
if (typeof packet.self_deaf !== "undefined")
this.joinConfig.selfDeaf = packet.self_deaf;
if (typeof packet.self_mute !== "undefined")
this.joinConfig.selfMute = packet.self_mute;
if (packet.channel_id)
this.joinConfig.channelId = packet.channel_id;
}
updateReceiveBindings(newState, oldState)
{
const oldWs = Reflect.get(oldState ?? {}, "ws");
const newWs = Reflect.get(newState, "ws");
const oldUdp = Reflect.get(oldState ?? {}, "udp");
const newUdp = Reflect.get(newState, "udp");
if (oldWs !== newWs)
{
oldWs?.off("packet", this.receiver.onWsPacket);
newWs?.on("packet", this.receiver.onWsPacket);
}
if (oldUdp !== newUdp)
{
oldUdp?.off("message", this.receiver.onUdpMessage);
newUdp?.on("message", this.receiver.onUdpMessage);
}
this.receiver.connectionData = Reflect.get(newState, "connectionData") ?? {};
}
configureNetworking()
{
2022-05-21 14:02:00 +00:00
const { server, state } = this.packets;
if (!server || !state || this.state.status === "destroyed" /* Destroyed */ || !server.endpoint)
return;
const networking = new Networking({
endpoint: server.endpoint,
serverId: server.guild_id,
token: server.token,
sessionId: state.session_id,
userId: state.user_id
}, Boolean(this.debug));
networking.once("close", this.onNetworkingClose);
networking.on("stateChange", this.onNetworkingStateChange);
networking.on("error", this.onNetworkingError);
networking.on("debug", this.onNetworkingDebug);
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
status: "connecting" /* Connecting */,
networking
2022-05-21 14:02:00 +00:00
};
}
onNetworkingClose(code)
{
if (this.state.status === "destroyed" /* Destroyed */)
return;
if (code === 4014)
{
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
status: "disconnected" /* Disconnected */,
2022-05-21 14:02:00 +00:00
reason: 0 /* WebSocketClose */,
closeCode: code
2022-05-21 14:02:00 +00:00
};
} else
{
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
status: "signalling" /* Signalling */
2022-05-21 14:02:00 +00:00
};
this.rejoinAttempts++;
if (!this.state.adapter.sendPayload(createJoinVoiceChannelPayload(this.joinConfig)))
{
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
status: "disconnected" /* Disconnected */,
reason: 1 /* AdapterUnavailable */
2022-05-21 14:02:00 +00:00
};
}
}
}
onNetworkingStateChange(oldState, newState)
{
2022-05-21 14:02:00 +00:00
this.updateReceiveBindings(newState, oldState);
if (oldState.code === newState.code)
return;
if (this.state.status !== "connecting" /* Connecting */ && this.state.status !== "ready" /* Ready */)
return;
if (newState.code === 4 /* Ready */)
{
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
status: "ready" /* Ready */
2022-05-21 14:02:00 +00:00
};
} else if (newState.code !== 6 /* Closed */)
{
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
status: "connecting" /* Connecting */
2022-05-21 14:02:00 +00:00
};
}
}
onNetworkingError(error)
{
this.emit("error", error);
2022-05-21 14:02:00 +00:00
}
onNetworkingDebug(message)
{
2022-05-21 14:02:00 +00:00
this.debug?.(`[NW] ${message}`);
}
prepareAudioPacket(buffer)
{
2022-05-21 14:02:00 +00:00
const state = this.state;
if (state.status !== "ready" /* Ready */)
return;
2022-05-21 14:02:00 +00:00
return state.networking.prepareAudioPacket(buffer);
}
dispatchAudio()
{
2022-05-21 14:02:00 +00:00
const state = this.state;
if (state.status !== "ready" /* Ready */)
return;
2022-05-21 14:02:00 +00:00
return state.networking.dispatchAudio();
}
playOpusPacket(buffer)
{
2022-05-21 14:02:00 +00:00
const state = this.state;
if (state.status !== "ready" /* Ready */)
return;
2022-05-21 14:02:00 +00:00
state.networking.prepareAudioPacket(buffer);
return state.networking.dispatchAudio();
}
destroy(adapterAvailable = true)
{
if (this.state.status === "destroyed" /* Destroyed */)
{
throw new Error("Cannot destroy VoiceConnection - it has already been destroyed");
2022-05-21 14:02:00 +00:00
}
if (getVoiceConnection(this.joinConfig.guildId, this.joinConfig.group) === this)
{
2022-05-21 14:02:00 +00:00
untrackVoiceConnection(this);
}
if (adapterAvailable)
{
2022-05-21 14:02:00 +00:00
this.state.adapter.sendPayload(createJoinVoiceChannelPayload({ ...this.joinConfig, channelId: null }));
}
this.state = {
status: "destroyed" /* Destroyed */
2022-05-21 14:02:00 +00:00
};
}
disconnect()
{
if (this.state.status === "destroyed" /* Destroyed */ || this.state.status === "signalling" /* Signalling */)
{
2022-05-21 14:02:00 +00:00
return false;
}
this.joinConfig.channelId = null;
if (!this.state.adapter.sendPayload(createJoinVoiceChannelPayload(this.joinConfig)))
{
2022-05-21 14:02:00 +00:00
this.state = {
adapter: this.state.adapter,
subscription: this.state.subscription,
status: "disconnected" /* Disconnected */,
reason: 1 /* AdapterUnavailable */
2022-05-21 14:02:00 +00:00
};
return false;
}
this.state = {
adapter: this.state.adapter,
reason: 3 /* Manual */,
status: "disconnected" /* Disconnected */
2022-05-21 14:02:00 +00:00
};
return true;
}
rejoin(joinConfig)
{
if (this.state.status === "destroyed" /* Destroyed */)
{
2022-05-21 14:02:00 +00:00
return false;
}
const notReady = this.state.status !== "ready" /* Ready */;
if (notReady)
this.rejoinAttempts++;
2022-05-21 14:02:00 +00:00
Object.assign(this.joinConfig, joinConfig);
if (this.state.adapter.sendPayload(createJoinVoiceChannelPayload(this.joinConfig)))
{
if (notReady)
{
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
status: "signalling" /* Signalling */
2022-05-21 14:02:00 +00:00
};
}
return true;
}
this.state = {
adapter: this.state.adapter,
subscription: this.state.subscription,
status: "disconnected" /* Disconnected */,
reason: 1 /* AdapterUnavailable */
2022-05-21 14:02:00 +00:00
};
return false;
}
setSpeaking(enabled)
{
if (this.state.status !== "ready" /* Ready */)
return false;
2022-05-21 14:02:00 +00:00
return this.state.networking.setSpeaking(enabled);
}
subscribe(player)
{
if (this.state.status === "destroyed" /* Destroyed */)
return;
const subscription = player["subscribe"](this);
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
subscription
2022-05-21 14:02:00 +00:00
};
return subscription;
}
get ping()
{
if (this.state.status === "ready" /* Ready */ && this.state.networking.state.code === 4 /* Ready */)
{
2022-05-21 14:02:00 +00:00
return {
ws: this.state.networking.state.ws.ping,
udp: this.state.networking.state.udp.ping
2022-05-21 14:02:00 +00:00
};
}
return {
ws: void 0,
udp: void 0
2022-05-21 14:02:00 +00:00
};
}
onSubscriptionRemoved(subscription)
{
if (this.state.status !== "destroyed" /* Destroyed */ && this.state.subscription === subscription)
{
2022-05-21 14:02:00 +00:00
this.state = {
...this.state,
subscription: void 0
2022-05-21 14:02:00 +00:00
};
}
}
};
__name(VoiceConnection2, "VoiceConnection");
function createVoiceConnection(joinConfig, options)
{
2022-05-21 14:02:00 +00:00
const payload = createJoinVoiceChannelPayload(joinConfig);
const existing = getVoiceConnection(joinConfig.guildId, joinConfig.group);
if (existing && existing.state.status !== "destroyed" /* Destroyed */)
{
if (existing.state.status === "disconnected" /* Disconnected */)
{
2022-05-21 14:02:00 +00:00
existing.rejoin({
channelId: joinConfig.channelId,
selfDeaf: joinConfig.selfDeaf,
selfMute: joinConfig.selfMute
2022-05-21 14:02:00 +00:00
});
} else if (!existing.state.adapter.sendPayload(payload))
{
2022-05-21 14:02:00 +00:00
existing.state = {
...existing.state,
status: "disconnected" /* Disconnected */,
reason: 1 /* AdapterUnavailable */
2022-05-21 14:02:00 +00:00
};
}
return existing;
}
const voiceConnection = new VoiceConnection2(joinConfig, options);
trackVoiceConnection(voiceConnection);
if (voiceConnection.state.status !== "destroyed" /* Destroyed */)
{
if (!voiceConnection.state.adapter.sendPayload(payload))
{
2022-05-21 14:02:00 +00:00
voiceConnection.state = {
...voiceConnection.state,
status: "disconnected" /* Disconnected */,
reason: 1 /* AdapterUnavailable */
2022-05-21 14:02:00 +00:00
};
}
}
return voiceConnection;
}
__name(createVoiceConnection, "createVoiceConnection");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/joinVoiceChannel.ts
function joinVoiceChannel(options)
{
2022-05-21 14:02:00 +00:00
const joinConfig = {
selfDeaf: true,
selfMute: false,
group: "default",
...options
2022-05-21 14:02:00 +00:00
};
return createVoiceConnection(joinConfig, {
adapterCreator: options.adapterCreator,
debug: options.debug
2022-05-21 14:02:00 +00:00
});
}
__name(joinVoiceChannel, "joinVoiceChannel");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/audio/AudioResource.ts
var import_node_stream2 = require("stream");
var import_prism_media2 = __toESM(require("prism-media"));
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/audio/TransformerGraph.ts
var import_prism_media = __toESM(require("prism-media"));
var FFMPEG_PCM_ARGUMENTS = ["-analyzeduration", "0", "-loglevel", "0", "-f", "s16le", "-ar", "48000", "-ac", "2"];
2022-05-21 14:02:00 +00:00
var FFMPEG_OPUS_ARGUMENTS = [
"-analyzeduration",
"0",
"-loglevel",
"0",
"-acodec",
"libopus",
"-f",
"opus",
"-ar",
"48000",
"-ac",
"2"
2022-05-21 14:02:00 +00:00
];
var StreamType = /* @__PURE__ */ ((StreamType2) =>
{
StreamType2["Arbitrary"] = "arbitrary";
StreamType2["Raw"] = "raw";
StreamType2["OggOpus"] = "ogg/opus";
StreamType2["WebmOpus"] = "webm/opus";
StreamType2["Opus"] = "opus";
2022-05-21 14:02:00 +00:00
return StreamType2;
})(StreamType || {});
var Node = class
{
constructor(type)
{
__publicField(this, "edges", []);
__publicField(this, "type");
2022-05-21 14:02:00 +00:00
this.type = type;
}
addEdge(edge)
{
2022-05-21 14:02:00 +00:00
this.edges.push({ ...edge, from: this });
}
};
__name(Node, "Node");
2022-05-21 14:02:00 +00:00
var NODES = /* @__PURE__ */ new Map();
for (const streamType of Object.values(StreamType))
{
2022-05-21 14:02:00 +00:00
NODES.set(streamType, new Node(streamType));
}
function getNode(type)
{
2022-05-21 14:02:00 +00:00
const node = NODES.get(type);
if (!node)
throw new Error(`Node type '${type}' does not exist!`);
2022-05-21 14:02:00 +00:00
return node;
}
__name(getNode, "getNode");
getNode("raw" /* Raw */).addEdge({
type: "opus encoder" /* OpusEncoder */,
to: getNode("opus" /* Opus */),
2022-05-21 14:02:00 +00:00
cost: 1.5,
transformer: () => new import_prism_media.default.opus.Encoder({ rate: 48e3, channels: 2, frameSize: 960 })
2022-05-21 14:02:00 +00:00
});
getNode("opus" /* Opus */).addEdge({
type: "opus decoder" /* OpusDecoder */,
to: getNode("raw" /* Raw */),
2022-05-21 14:02:00 +00:00
cost: 1.5,
transformer: () => new import_prism_media.default.opus.Decoder({ rate: 48e3, channels: 2, frameSize: 960 })
2022-05-21 14:02:00 +00:00
});
getNode("ogg/opus" /* OggOpus */).addEdge({
type: "ogg/opus demuxer" /* OggOpusDemuxer */,
to: getNode("opus" /* Opus */),
2022-05-21 14:02:00 +00:00
cost: 1,
transformer: () => new import_prism_media.default.opus.OggDemuxer()
2022-05-21 14:02:00 +00:00
});
getNode("webm/opus" /* WebmOpus */).addEdge({
type: "webm/opus demuxer" /* WebmOpusDemuxer */,
to: getNode("opus" /* Opus */),
2022-05-21 14:02:00 +00:00
cost: 1,
transformer: () => new import_prism_media.default.opus.WebmDemuxer()
2022-05-21 14:02:00 +00:00
});
var FFMPEG_PCM_EDGE = {
type: "ffmpeg pcm" /* FFmpegPCM */,
to: getNode("raw" /* Raw */),
2022-05-21 14:02:00 +00:00
cost: 2,
transformer: (input) => new import_prism_media.default.FFmpeg({
args: typeof input === "string" ? ["-i", input, ...FFMPEG_PCM_ARGUMENTS] : FFMPEG_PCM_ARGUMENTS
})
2022-05-21 14:02:00 +00:00
};
getNode("arbitrary" /* Arbitrary */).addEdge(FFMPEG_PCM_EDGE);
getNode("ogg/opus" /* OggOpus */).addEdge(FFMPEG_PCM_EDGE);
getNode("webm/opus" /* WebmOpus */).addEdge(FFMPEG_PCM_EDGE);
getNode("raw" /* Raw */).addEdge({
type: "volume transformer" /* InlineVolume */,
to: getNode("raw" /* Raw */),
2022-05-21 14:02:00 +00:00
cost: 0.5,
transformer: () => new import_prism_media.default.VolumeTransformer({ type: "s16le" })
2022-05-21 14:02:00 +00:00
});
function canEnableFFmpegOptimizations()
{
try
{
return import_prism_media.default.FFmpeg.getInfo().output.includes("--enable-libopus");
} catch {
}
2022-05-21 14:02:00 +00:00
return false;
}
__name(canEnableFFmpegOptimizations, "canEnableFFmpegOptimizations");
if (canEnableFFmpegOptimizations())
{
2022-05-21 14:02:00 +00:00
const FFMPEG_OGG_EDGE = {
type: "ffmpeg ogg" /* FFmpegOgg */,
to: getNode("ogg/opus" /* OggOpus */),
2022-05-21 14:02:00 +00:00
cost: 2,
transformer: (input) => new import_prism_media.default.FFmpeg({
args: typeof input === "string" ? ["-i", input, ...FFMPEG_OPUS_ARGUMENTS] : FFMPEG_OPUS_ARGUMENTS
})
2022-05-21 14:02:00 +00:00
};
getNode("arbitrary" /* Arbitrary */).addEdge(FFMPEG_OGG_EDGE);
getNode("ogg/opus" /* OggOpus */).addEdge(FFMPEG_OGG_EDGE);
getNode("webm/opus" /* WebmOpus */).addEdge(FFMPEG_OGG_EDGE);
2022-05-21 14:02:00 +00:00
}
function findPath(from, constraints, goal = getNode("opus" /* Opus */), path = [], depth = 5)
{
if (from === goal && constraints(path))
{
2022-05-21 14:02:00 +00:00
return { cost: 0 };
} else if (depth === 0)
{
2022-05-21 14:02:00 +00:00
return { cost: Infinity };
}
let currentBest = void 0;
for (const edge of from.edges)
{
if (currentBest && edge.cost > currentBest.cost)
continue;
2022-05-21 14:02:00 +00:00
const next = findPath(edge.to, constraints, goal, [...path, edge], depth - 1);
const cost = edge.cost + next.cost;
if (!currentBest || cost < currentBest.cost)
{
2022-05-21 14:02:00 +00:00
currentBest = { cost, edge, next };
}
}
return currentBest ?? { cost: Infinity };
}
__name(findPath, "findPath");
function constructPipeline(step)
{
2022-05-21 14:02:00 +00:00
const edges = [];
let current = step;
while (current?.edge)
{
2022-05-21 14:02:00 +00:00
edges.push(current.edge);
current = current.next;
}
return edges;
}
__name(constructPipeline, "constructPipeline");
function findPipeline(from, constraint)
{
2022-05-21 14:02:00 +00:00
return constructPipeline(findPath(getNode(from), constraint));
}
__name(findPipeline, "findPipeline");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/audio/AudioResource.ts
var AudioResource = class
{
constructor(edges, streams, metadata, silencePaddingFrames)
{
__publicField(this, "playStream");
__publicField(this, "edges");
__publicField(this, "metadata");
__publicField(this, "volume");
__publicField(this, "encoder");
__publicField(this, "audioPlayer");
__publicField(this, "playbackDuration", 0);
__publicField(this, "started", false);
__publicField(this, "silencePaddingFrames");
__publicField(this, "silenceRemaining", -1);
2022-05-21 14:02:00 +00:00
this.edges = edges;
this.playStream = streams.length > 1 ? (0, import_node_stream2.pipeline)(streams, noop) : streams[0];
this.metadata = metadata;
this.silencePaddingFrames = silencePaddingFrames;
for (const stream of streams)
{
if (stream instanceof import_prism_media2.default.VolumeTransformer)
{
2022-05-21 14:02:00 +00:00
this.volume = stream;
} else if (stream instanceof import_prism_media2.default.opus.Encoder)
{
2022-05-21 14:02:00 +00:00
this.encoder = stream;
}
}
this.playStream.once("readable", () => this.started = true);
2022-05-21 14:02:00 +00:00
}
get readable()
{
if (this.silenceRemaining === 0)
return false;
2022-05-21 14:02:00 +00:00
const real = this.playStream.readable;
if (!real)
{
if (this.silenceRemaining === -1)
this.silenceRemaining = this.silencePaddingFrames;
2022-05-21 14:02:00 +00:00
return this.silenceRemaining !== 0;
}
return real;
}
get ended()
{
2022-05-21 14:02:00 +00:00
return this.playStream.readableEnded || this.playStream.destroyed || this.silenceRemaining === 0;
}
read()
{
if (this.silenceRemaining === 0)
{
2022-05-21 14:02:00 +00:00
return null;
} else if (this.silenceRemaining > 0)
{
2022-05-21 14:02:00 +00:00
this.silenceRemaining--;
return SILENCE_FRAME;
}
const packet = this.playStream.read();
if (packet)
{
2022-05-21 14:02:00 +00:00
this.playbackDuration += 20;
}
return packet;
}
};
__name(AudioResource, "AudioResource");
var VOLUME_CONSTRAINT = /* @__PURE__ */ __name((path) => path.some((edge) => edge.type === "volume transformer" /* InlineVolume */), "VOLUME_CONSTRAINT");
var NO_CONSTRAINT = /* @__PURE__ */ __name(() => true, "NO_CONSTRAINT");
function inferStreamType(stream)
{
if (stream instanceof import_prism_media2.default.opus.Encoder)
{
return { streamType: "opus" /* Opus */, hasVolume: false };
} else if (stream instanceof import_prism_media2.default.opus.Decoder)
{
return { streamType: "raw" /* Raw */, hasVolume: false };
} else if (stream instanceof import_prism_media2.default.VolumeTransformer)
{
return { streamType: "raw" /* Raw */, hasVolume: true };
} else if (stream instanceof import_prism_media2.default.opus.OggDemuxer)
{
return { streamType: "opus" /* Opus */, hasVolume: false };
} else if (stream instanceof import_prism_media2.default.opus.WebmDemuxer)
{
return { streamType: "opus" /* Opus */, hasVolume: false };
}
return { streamType: "arbitrary" /* Arbitrary */, hasVolume: false };
2022-05-21 14:02:00 +00:00
}
__name(inferStreamType, "inferStreamType");
function createAudioResource(input, options = {})
{
2022-05-21 14:02:00 +00:00
let inputType = options.inputType;
let needsInlineVolume = Boolean(options.inlineVolume);
if (typeof input === "string")
{
inputType = "arbitrary" /* Arbitrary */;
} else if (typeof inputType === "undefined")
{
2022-05-21 14:02:00 +00:00
const analysis = inferStreamType(input);
inputType = analysis.streamType;
needsInlineVolume = needsInlineVolume && !analysis.hasVolume;
}
const transformerPipeline = findPipeline(inputType, needsInlineVolume ? VOLUME_CONSTRAINT : NO_CONSTRAINT);
if (transformerPipeline.length === 0)
{
if (typeof input === "string")
throw new Error(`Invalid pipeline constructed for string resource '${input}'`);
2022-05-21 14:02:00 +00:00
return new AudioResource([], [input], options.metadata ?? null, options.silencePaddingFrames ?? 5);
}
const streams = transformerPipeline.map((edge) => edge.transformer(input));
if (typeof input !== "string")
streams.unshift(input);
2022-05-21 14:02:00 +00:00
return new AudioResource(transformerPipeline, streams, options.metadata ?? null, options.silencePaddingFrames ?? 5);
}
__name(createAudioResource, "createAudioResource");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/util/generateDependencyReport.ts
var import_node_path = require("path");
var import_prism_media3 = __toESM(require("prism-media"));
function findPackageJSON(dir, packageName, depth)
{
if (depth === 0)
return void 0;
const attemptedPath = (0, import_node_path.resolve)(dir, "./package.json");
try
{
2022-05-21 14:02:00 +00:00
const pkg = require(attemptedPath);
if (pkg.name !== packageName)
throw new Error("package.json does not match");
2022-05-21 14:02:00 +00:00
return pkg;
} catch (err)
{
return findPackageJSON((0, import_node_path.resolve)(dir, ".."), packageName, depth - 1);
2022-05-21 14:02:00 +00:00
}
}
__name(findPackageJSON, "findPackageJSON");
function version(name)
{
try
{
const pkg = name === "@discordjs/voice" ? require_package() : findPackageJSON((0, import_node_path.dirname)(require.resolve(name)), name, 3);
return pkg?.version ?? "not found";
} catch (err)
{
return "not found";
2022-05-21 14:02:00 +00:00
}
}
__name(version, "version");
function generateDependencyReport()
{
2022-05-21 14:02:00 +00:00
const report = [];
const addVersion = /* @__PURE__ */ __name((name) => report.push(`- ${name}: ${version(name)}`), "addVersion");
report.push("Core Dependencies");
addVersion("@discordjs/voice");
addVersion("prism-media");
report.push("");
report.push("Opus Libraries");
addVersion("@discordjs/opus");
addVersion("opusscript");
report.push("");
report.push("Encryption Libraries");
addVersion("sodium-native");
addVersion("sodium");
addVersion("libsodium-wrappers");
addVersion("tweetnacl");
report.push("");
report.push("FFmpeg");
try
{
2022-05-21 14:02:00 +00:00
const info = import_prism_media3.default.FFmpeg.getInfo();
report.push(`- version: ${info.version}`);
report.push(`- libopus: ${info.output.includes("--enable-libopus") ? "yes" : "no"}`);
} catch (err)
{
report.push("- not found");
2022-05-21 14:02:00 +00:00
}
return ["-".repeat(50), ...report, "-".repeat(50)].join("\n");
2022-05-21 14:02:00 +00:00
}
__name(generateDependencyReport, "generateDependencyReport");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/util/entersState.ts
var import_node_events8 = require("events");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/util/abortAfter.ts
function abortAfter(delay)
{
2022-05-21 14:02:00 +00:00
const ac = new AbortController();
const timeout = setTimeout(() => ac.abort(), delay);
ac.signal.addEventListener("abort", () => clearTimeout(timeout));
2022-05-21 14:02:00 +00:00
return [ac, ac.signal];
}
__name(abortAfter, "abortAfter");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/util/entersState.ts
async function entersState(target, status, timeoutOrSignal)
{
if (target.state.status !== status)
{
const [ac, signal] = typeof timeoutOrSignal === "number" ? abortAfter(timeoutOrSignal) : [void 0, timeoutOrSignal];
try
{
await (0, import_node_events8.once)(target, status, { signal });
} finally
{
2022-05-21 14:02:00 +00:00
ac?.abort();
}
}
return target;
}
__name(entersState, "entersState");
2022-05-21 14:02:00 +00:00
2022-06-26 02:40:04 +00:00
// src/util/demuxProbe.ts
var import_node_stream3 = require("stream");
var import_prism_media4 = __toESM(require("prism-media"));
function validateDiscordOpusHead(opusHead)
{
2022-05-21 14:02:00 +00:00
const channels = opusHead.readUInt8(9);
const sampleRate = opusHead.readUInt32LE(12);
return channels === 2 && sampleRate === 48e3;
}
__name(validateDiscordOpusHead, "validateDiscordOpusHead");
function demuxProbe(stream, probeSize = 1024, validator = validateDiscordOpusHead)
{
return new Promise((resolve2, reject) =>
{
if (stream.readableObjectMode)
return reject(new Error("Cannot probe a readable stream in object mode"));
if (stream.readableEnded)
return reject(new Error("Cannot probe a stream that has ended"));
2022-05-21 14:02:00 +00:00
let readBuffer = Buffer.alloc(0);
let resolved = void 0;
const finish = /* @__PURE__ */ __name((type) =>
{
stream.off("data", onData);
stream.off("close", onClose);
stream.off("end", onClose);
2022-05-21 14:02:00 +00:00
stream.pause();
resolved = type;
if (stream.readableEnded)
{
2022-05-21 14:02:00 +00:00
resolve2({
stream: import_node_stream3.Readable.from(readBuffer),
type
2022-05-21 14:02:00 +00:00
});
} else
{
if (readBuffer.length > 0)
{
2022-05-21 14:02:00 +00:00
stream.push(readBuffer);
}
resolve2({
stream,
type
2022-05-21 14:02:00 +00:00
});
}
}, "finish");
const foundHead = /* @__PURE__ */ __name((type) => (head) =>
{
if (validator(head))
{
finish(type);
}
}, "foundHead");
2022-05-21 14:02:00 +00:00
const webm = new import_prism_media4.default.opus.WebmDemuxer();
webm.once("error", noop);
webm.on("head", foundHead("webm/opus" /* WebmOpus */));
2022-05-21 14:02:00 +00:00
const ogg = new import_prism_media4.default.opus.OggDemuxer();
ogg.once("error", noop);
ogg.on("head", foundHead("ogg/opus" /* OggOpus */));
const onClose = /* @__PURE__ */ __name(() =>
{
if (!resolved)
{
finish("arbitrary" /* Arbitrary */);
2022-05-21 14:02:00 +00:00
}
}, "onClose");
const onData = /* @__PURE__ */ __name((buffer) =>
{
2022-05-21 14:02:00 +00:00
readBuffer = Buffer.concat([readBuffer, buffer]);
webm.write(buffer);
ogg.write(buffer);
if (readBuffer.length >= probeSize)
{
stream.off("data", onData);
2022-05-21 14:02:00 +00:00
stream.pause();
process.nextTick(onClose);
}
}, "onData");
stream.once("error", reject);
stream.on("data", onData);
stream.once("close", onClose);
stream.once("end", onClose);
2022-05-21 14:02:00 +00:00
});
}
__name(demuxProbe, "demuxProbe");
2022-05-21 14:02:00 +00:00
// Annotate the CommonJS export names for ESM import in node:
0 && (module.exports = {
AudioPlayer,
AudioPlayerError,
AudioPlayerStatus,
AudioReceiveStream,
AudioResource,
EndBehaviorType,
NoSubscriberBehavior,
PlayerSubscription,
SSRCMap,
SpeakingMap,
StreamType,
VoiceConnection,
VoiceConnectionDisconnectReason,
VoiceConnectionStatus,
VoiceReceiver,
createAudioPlayer,
createAudioResource,
createDefaultAudioReceiveStreamOptions,
demuxProbe,
entersState,
generateDependencyReport,
getGroups,
getVoiceConnection,
getVoiceConnections,
joinVoiceChannel,
validateDiscordOpusHead,
Networking,
});
//# sourceMappingURL=index.js.map