@jubbio/voice 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +257 -0
- package/dist/AudioPlayer.d.ts +84 -0
- package/dist/AudioPlayer.js +385 -0
- package/dist/AudioResource.d.ts +60 -0
- package/dist/AudioResource.js +150 -0
- package/dist/VoiceConnection.d.ts +66 -0
- package/dist/VoiceConnection.js +191 -0
- package/dist/enums.d.ts +45 -0
- package/dist/enums.js +52 -0
- package/dist/index.d.ts +25 -0
- package/dist/index.js +42 -0
- package/dist/types.d.ts +99 -0
- package/dist/types.js +3 -0
- package/package.json +43 -0
|
@@ -0,0 +1,150 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.AudioResource = void 0;
|
|
4
|
+
exports.createAudioResource = createAudioResource;
|
|
5
|
+
exports.createAudioResourceFromUrl = createAudioResourceFromUrl;
|
|
6
|
+
exports.probeAudioInfo = probeAudioInfo;
|
|
7
|
+
const child_process_1 = require("child_process");
|
|
8
|
+
const enums_1 = require("./enums");
|
|
9
|
+
/**
|
|
10
|
+
* Represents an audio resource that can be played
|
|
11
|
+
*/
|
|
12
|
+
class AudioResource {
|
|
13
|
+
/** Metadata attached to this resource */
|
|
14
|
+
metadata;
|
|
15
|
+
/** Whether playback has started */
|
|
16
|
+
started = false;
|
|
17
|
+
/** Whether playback has ended */
|
|
18
|
+
ended = false;
|
|
19
|
+
/** The input source (URL or file path) */
|
|
20
|
+
inputSource;
|
|
21
|
+
/** Stream type */
|
|
22
|
+
streamType;
|
|
23
|
+
/** Volume (0-1) */
|
|
24
|
+
volume = 1;
|
|
25
|
+
constructor(input, options = {}) {
|
|
26
|
+
this.metadata = options.metadata;
|
|
27
|
+
this.streamType = options.inputType || enums_1.StreamType.Arbitrary;
|
|
28
|
+
if (typeof input === 'string') {
|
|
29
|
+
this.inputSource = input;
|
|
30
|
+
}
|
|
31
|
+
else {
|
|
32
|
+
// For streams, we'd need to handle differently
|
|
33
|
+
// For now, throw an error
|
|
34
|
+
throw new Error('Stream input not yet supported. Use URL or file path.');
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Get the input source for FFmpeg
|
|
39
|
+
* @internal
|
|
40
|
+
*/
|
|
41
|
+
getInputSource() {
|
|
42
|
+
return this.inputSource;
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Set the volume (0-1)
|
|
46
|
+
*/
|
|
47
|
+
setVolume(volume) {
|
|
48
|
+
this.volume = Math.max(0, Math.min(1, volume));
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Get the current volume
|
|
52
|
+
*/
|
|
53
|
+
getVolume() {
|
|
54
|
+
return this.volume;
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
exports.AudioResource = AudioResource;
|
|
58
|
+
/**
|
|
59
|
+
* Create an audio resource from various inputs
|
|
60
|
+
*/
|
|
61
|
+
function createAudioResource(input, options) {
|
|
62
|
+
return new AudioResource(input, options);
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* Create an audio resource from a YouTube/streaming URL
|
|
66
|
+
* Stores the original URL - extraction happens at playback time
|
|
67
|
+
*/
|
|
68
|
+
function createAudioResourceFromUrl(url, options = {}) {
|
|
69
|
+
// Don't extract stream URL here - just store the original URL
|
|
70
|
+
// The AudioPlayer will use yt-dlp at playback time
|
|
71
|
+
return new AudioResource(url, options);
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Check if URL is a streaming service URL
|
|
75
|
+
*/
|
|
76
|
+
function isStreamingUrl(url) {
|
|
77
|
+
const streamingDomains = [
|
|
78
|
+
'youtube.com',
|
|
79
|
+
'youtu.be',
|
|
80
|
+
'soundcloud.com',
|
|
81
|
+
'spotify.com',
|
|
82
|
+
'twitch.tv',
|
|
83
|
+
'vimeo.com'
|
|
84
|
+
];
|
|
85
|
+
return streamingDomains.some(domain => url.includes(domain));
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Check if input is a valid URL
|
|
89
|
+
*/
|
|
90
|
+
function isValidUrl(input) {
|
|
91
|
+
try {
|
|
92
|
+
new URL(input);
|
|
93
|
+
return true;
|
|
94
|
+
}
|
|
95
|
+
catch {
|
|
96
|
+
return input.startsWith('http://') || input.startsWith('https://');
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* Probe audio info from a URL or search query
|
|
101
|
+
* If input is not a URL, it will search YouTube
|
|
102
|
+
*/
|
|
103
|
+
async function probeAudioInfo(input, ytDlpPath = '~/.local/bin/yt-dlp') {
|
|
104
|
+
return new Promise((resolve, reject) => {
|
|
105
|
+
// If not a valid URL, treat as YouTube search
|
|
106
|
+
let searchQuery = input;
|
|
107
|
+
if (!isValidUrl(input)) {
|
|
108
|
+
searchQuery = `ytsearch1:${input}`;
|
|
109
|
+
}
|
|
110
|
+
const ytdlp = (0, child_process_1.spawn)('bash', [
|
|
111
|
+
'-c',
|
|
112
|
+
`${ytDlpPath} --no-playlist --no-warnings -j "${searchQuery}"`
|
|
113
|
+
]);
|
|
114
|
+
let stdout = '';
|
|
115
|
+
let stderr = '';
|
|
116
|
+
ytdlp.stdout.on('data', (data) => {
|
|
117
|
+
stdout += data.toString();
|
|
118
|
+
});
|
|
119
|
+
ytdlp.stderr.on('data', (data) => {
|
|
120
|
+
stderr += data.toString();
|
|
121
|
+
});
|
|
122
|
+
ytdlp.on('close', (code) => {
|
|
123
|
+
if (code !== 0) {
|
|
124
|
+
reject(new Error(`Failed to probe audio info: ${stderr || 'Unknown error'}`));
|
|
125
|
+
return;
|
|
126
|
+
}
|
|
127
|
+
try {
|
|
128
|
+
const info = JSON.parse(stdout);
|
|
129
|
+
resolve({
|
|
130
|
+
title: info.title || 'Unknown',
|
|
131
|
+
duration: info.duration || 0,
|
|
132
|
+
thumbnail: info.thumbnail,
|
|
133
|
+
url: info.webpage_url || info.url || input
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
catch (e) {
|
|
137
|
+
reject(new Error(`Failed to parse audio info: ${e.message}`));
|
|
138
|
+
}
|
|
139
|
+
});
|
|
140
|
+
ytdlp.on('error', (err) => {
|
|
141
|
+
reject(new Error(`Failed to probe audio info: ${err.message}`));
|
|
142
|
+
});
|
|
143
|
+
// Timeout after 30 seconds
|
|
144
|
+
setTimeout(() => {
|
|
145
|
+
ytdlp.kill();
|
|
146
|
+
reject(new Error('Timeout waiting for audio info'));
|
|
147
|
+
}, 30000);
|
|
148
|
+
});
|
|
149
|
+
}
|
|
150
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AudioResource.js","sourceRoot":"","sources":["../src/AudioResource.ts"],"names":[],"mappings":"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","sourcesContent":["import { Readable } from 'stream';\r\nimport { spawn } from 'child_process';\r\nimport { StreamType } from './enums';\r\nimport { CreateAudioResourceOptions, AudioResourceInput } from './types';\r\n\r\n/**\r\n * Represents an audio resource that can be played\r\n */\r\nexport class AudioResource<T = unknown> {\r\n  /** Metadata attached to this resource */\r\n  public readonly metadata: T;\r\n  \r\n  /** Whether playback has started */\r\n  public started = false;\r\n  \r\n  /** Whether playback has ended */\r\n  public ended = false;\r\n  \r\n  /** The input source (URL or file path) */\r\n  private inputSource: string;\r\n  \r\n  /** Stream type */\r\n  private streamType: StreamType;\r\n  \r\n  /** Volume (0-1) */\r\n  private volume = 1;\r\n\r\n  constructor(\r\n    input: AudioResourceInput,\r\n    options: CreateAudioResourceOptions<T> = {}\r\n  ) {\r\n    this.metadata = options.metadata as T;\r\n    this.streamType = options.inputType || StreamType.Arbitrary;\r\n    \r\n    if (typeof input === 'string') {\r\n      this.inputSource = input;\r\n    } else {\r\n      // For streams, we'd need to handle differently\r\n      // For now, throw an error\r\n      throw new Error('Stream input not yet supported. Use URL or file path.');\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Get the input source for FFmpeg\r\n   * @internal\r\n   */\r\n  getInputSource(): string {\r\n    return this.inputSource;\r\n  }\r\n\r\n  /**\r\n   * Set the volume (0-1)\r\n   */\r\n  setVolume(volume: number): void {\r\n    this.volume = Math.max(0, Math.min(1, volume));\r\n  }\r\n\r\n  /**\r\n   * Get the current volume\r\n   */\r\n  getVolume(): number {\r\n    return this.volume;\r\n  }\r\n}\r\n\r\n/**\r\n * Options for creating audio resource from URL\r\n */\r\nexport interface CreateAudioResourceFromUrlOptions<T = unknown> extends CreateAudioResourceOptions<T> {\r\n  /** Use yt-dlp to extract audio URL */\r\n  useYtDlp?: boolean;\r\n  /** Path to yt-dlp binary */\r\n  ytDlpPath?: string;\r\n}\r\n\r\n/**\r\n * Create an audio resource from various inputs\r\n */\r\nexport function createAudioResource<T = unknown>(\r\n  input: AudioResourceInput,\r\n  options?: CreateAudioResourceOptions<T>\r\n): AudioResource<T> {\r\n  return new AudioResource(input, options);\r\n}\r\n\r\n/**\r\n * Create an audio resource from a YouTube/streaming URL\r\n * Stores the original URL - extraction happens at playback time\r\n */\r\nexport function createAudioResourceFromUrl<T = unknown>(\r\n  url: string,\r\n  options: CreateAudioResourceFromUrlOptions<T> = {}\r\n): AudioResource<T> {\r\n  // Don't extract stream URL here - just store the original URL\r\n  // The AudioPlayer will use yt-dlp at playback time\r\n  return new AudioResource(url, options);\r\n}\r\n\r\n/**\r\n * Check if URL is a streaming service URL\r\n */\r\nfunction isStreamingUrl(url: string): boolean {\r\n  const streamingDomains = [\r\n    'youtube.com',\r\n    'youtu.be',\r\n    'soundcloud.com',\r\n    'spotify.com',\r\n    'twitch.tv',\r\n    'vimeo.com'\r\n  ];\r\n  \r\n  return streamingDomains.some(domain => url.includes(domain));\r\n}\r\n\r\n/**\r\n * Check if input is a valid URL\r\n */\r\nfunction isValidUrl(input: string): boolean {\r\n  try {\r\n    new URL(input);\r\n    return true;\r\n  } catch {\r\n    return input.startsWith('http://') || input.startsWith('https://');\r\n  }\r\n}\r\n\r\n/**\r\n * Probe audio info from a URL or search query\r\n * If input is not a URL, it will search YouTube\r\n */\r\nexport async function probeAudioInfo(input: string, ytDlpPath = '~/.local/bin/yt-dlp'): Promise<{\r\n  title: string;\r\n  duration: number;\r\n  thumbnail?: string;\r\n  url: string;\r\n}> {\r\n  return new Promise((resolve, reject) => {\r\n    // If not a valid URL, treat as YouTube search\r\n    let searchQuery = input;\r\n    if (!isValidUrl(input)) {\r\n      searchQuery = `ytsearch1:${input}`;\r\n    }\r\n    \r\n    const ytdlp = spawn('bash', [\r\n      '-c',\r\n      `${ytDlpPath} --no-playlist --no-warnings -j \"${searchQuery}\"`\r\n    ]);\r\n    \r\n    let stdout = '';\r\n    let stderr = '';\r\n    \r\n    ytdlp.stdout.on('data', (data) => {\r\n      stdout += data.toString();\r\n    });\r\n    \r\n    ytdlp.stderr.on('data', (data) => {\r\n      stderr += data.toString();\r\n    });\r\n    \r\n    ytdlp.on('close', (code) => {\r\n      if (code !== 0) {\r\n        reject(new Error(`Failed to probe audio info: ${stderr || 'Unknown error'}`));\r\n        return;\r\n      }\r\n      \r\n      try {\r\n        const info = JSON.parse(stdout);\r\n        resolve({\r\n          title: info.title || 'Unknown',\r\n          duration: info.duration || 0,\r\n          thumbnail: info.thumbnail,\r\n          url: info.webpage_url || info.url || input\r\n        });\r\n      } catch (e) {\r\n        reject(new Error(`Failed to parse audio info: ${(e as Error).message}`));\r\n      }\r\n    });\r\n    \r\n    ytdlp.on('error', (err) => {\r\n      reject(new Error(`Failed to probe audio info: ${err.message}`));\r\n    });\r\n    \r\n    // Timeout after 30 seconds\r\n    setTimeout(() => {\r\n      ytdlp.kill();\r\n      reject(new Error('Timeout waiting for audio info'));\r\n    }, 30000);\r\n  });\r\n}\r\n"]}
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
import { EventEmitter } from 'events';
|
|
2
|
+
import { Room } from '@livekit/rtc-node';
|
|
3
|
+
import { JoinVoiceChannelOptions, VoiceConnectionState } from './types';
|
|
4
|
+
import { AudioPlayer } from './AudioPlayer';
|
|
5
|
+
/**
|
|
6
|
+
* Represents a voice connection to a channel
|
|
7
|
+
*/
|
|
8
|
+
export declare class VoiceConnection extends EventEmitter {
|
|
9
|
+
/** Current connection state */
|
|
10
|
+
state: VoiceConnectionState;
|
|
11
|
+
/** The channel ID this connection is for */
|
|
12
|
+
readonly channelId: string;
|
|
13
|
+
/** The guild ID this connection is for */
|
|
14
|
+
readonly guildId: string;
|
|
15
|
+
/** LiveKit room instance */
|
|
16
|
+
private room;
|
|
17
|
+
/** LiveKit connection info */
|
|
18
|
+
private livekitEndpoint;
|
|
19
|
+
private livekitToken;
|
|
20
|
+
private livekitRoomName;
|
|
21
|
+
/** Subscribed audio player */
|
|
22
|
+
private subscribedPlayer;
|
|
23
|
+
/** Gateway adapter methods */
|
|
24
|
+
private adapterMethods;
|
|
25
|
+
/** Adapter implementer (for sending payloads) */
|
|
26
|
+
private adapter;
|
|
27
|
+
constructor(options: JoinVoiceChannelOptions);
|
|
28
|
+
/**
|
|
29
|
+
* Subscribe an audio player to this connection
|
|
30
|
+
*/
|
|
31
|
+
subscribe(player: AudioPlayer): void;
|
|
32
|
+
/**
|
|
33
|
+
* Unsubscribe the current audio player
|
|
34
|
+
*/
|
|
35
|
+
unsubscribe(): void;
|
|
36
|
+
/**
|
|
37
|
+
* Get the LiveKit room (for audio player to publish tracks)
|
|
38
|
+
*/
|
|
39
|
+
getRoom(): Room | null;
|
|
40
|
+
/**
|
|
41
|
+
* Disconnect from the voice channel
|
|
42
|
+
*/
|
|
43
|
+
disconnect(): boolean;
|
|
44
|
+
/**
|
|
45
|
+
* Destroy the connection completely
|
|
46
|
+
*/
|
|
47
|
+
destroy(): void;
|
|
48
|
+
/**
|
|
49
|
+
* Rejoin the voice channel (after disconnect)
|
|
50
|
+
*/
|
|
51
|
+
rejoin(): boolean;
|
|
52
|
+
private sendVoiceStateUpdate;
|
|
53
|
+
private handleVoiceServerUpdate;
|
|
54
|
+
private handleVoiceStateUpdate;
|
|
55
|
+
private connectToLiveKit;
|
|
56
|
+
private disconnectFromLiveKit;
|
|
57
|
+
private setState;
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* Join a voice channel - main entry point
|
|
61
|
+
*/
|
|
62
|
+
export declare function joinVoiceChannel(options: JoinVoiceChannelOptions): VoiceConnection;
|
|
63
|
+
/**
|
|
64
|
+
* Get an existing voice connection
|
|
65
|
+
*/
|
|
66
|
+
export declare function getVoiceConnection(guildId: string): VoiceConnection | undefined;
|
|
@@ -0,0 +1,191 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.VoiceConnection = void 0;
|
|
4
|
+
exports.joinVoiceChannel = joinVoiceChannel;
|
|
5
|
+
exports.getVoiceConnection = getVoiceConnection;
|
|
6
|
+
const events_1 = require("events");
|
|
7
|
+
const rtc_node_1 = require("@livekit/rtc-node");
|
|
8
|
+
const enums_1 = require("./enums");
|
|
9
|
+
/**
|
|
10
|
+
* Represents a voice connection to a channel
|
|
11
|
+
*/
|
|
12
|
+
class VoiceConnection extends events_1.EventEmitter {
|
|
13
|
+
/** Current connection state */
|
|
14
|
+
state = { status: enums_1.VoiceConnectionStatus.Connecting };
|
|
15
|
+
/** The channel ID this connection is for */
|
|
16
|
+
channelId;
|
|
17
|
+
/** The guild ID this connection is for */
|
|
18
|
+
guildId;
|
|
19
|
+
/** LiveKit room instance */
|
|
20
|
+
room = null;
|
|
21
|
+
/** LiveKit connection info */
|
|
22
|
+
livekitEndpoint = null;
|
|
23
|
+
livekitToken = null;
|
|
24
|
+
livekitRoomName = null;
|
|
25
|
+
/** Subscribed audio player */
|
|
26
|
+
subscribedPlayer = null;
|
|
27
|
+
/** Gateway adapter methods */
|
|
28
|
+
adapterMethods;
|
|
29
|
+
/** Adapter implementer (for sending payloads) */
|
|
30
|
+
adapter = null;
|
|
31
|
+
constructor(options) {
|
|
32
|
+
super();
|
|
33
|
+
this.channelId = options.channelId;
|
|
34
|
+
this.guildId = options.guildId;
|
|
35
|
+
// Create adapter methods that will receive gateway events
|
|
36
|
+
this.adapterMethods = {
|
|
37
|
+
onVoiceServerUpdate: (data) => this.handleVoiceServerUpdate(data),
|
|
38
|
+
onVoiceStateUpdate: (data) => this.handleVoiceStateUpdate(data),
|
|
39
|
+
destroy: () => this.destroy()
|
|
40
|
+
};
|
|
41
|
+
// Get adapter from creator
|
|
42
|
+
this.adapter = options.adapterCreator(this.adapterMethods);
|
|
43
|
+
// Send voice state update to join channel
|
|
44
|
+
this.sendVoiceStateUpdate(options.channelId, options.selfMute, options.selfDeaf);
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Subscribe an audio player to this connection
|
|
48
|
+
*/
|
|
49
|
+
subscribe(player) {
|
|
50
|
+
if (this.subscribedPlayer) {
|
|
51
|
+
this.subscribedPlayer.unsubscribe(this);
|
|
52
|
+
}
|
|
53
|
+
this.subscribedPlayer = player;
|
|
54
|
+
player.subscribe(this);
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Unsubscribe the current audio player
|
|
58
|
+
*/
|
|
59
|
+
unsubscribe() {
|
|
60
|
+
if (this.subscribedPlayer) {
|
|
61
|
+
this.subscribedPlayer.unsubscribe(this);
|
|
62
|
+
this.subscribedPlayer = null;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Get the LiveKit room (for audio player to publish tracks)
|
|
67
|
+
*/
|
|
68
|
+
getRoom() {
|
|
69
|
+
return this.room;
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Disconnect from the voice channel
|
|
73
|
+
*/
|
|
74
|
+
disconnect() {
|
|
75
|
+
this.sendVoiceStateUpdate(null);
|
|
76
|
+
return true;
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Destroy the connection completely
|
|
80
|
+
*/
|
|
81
|
+
destroy() {
|
|
82
|
+
this.unsubscribe();
|
|
83
|
+
this.disconnectFromLiveKit();
|
|
84
|
+
this.adapter?.destroy();
|
|
85
|
+
this.setState({ status: enums_1.VoiceConnectionStatus.Destroyed });
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Rejoin the voice channel (after disconnect)
|
|
89
|
+
*/
|
|
90
|
+
rejoin() {
|
|
91
|
+
this.sendVoiceStateUpdate(this.channelId);
|
|
92
|
+
return true;
|
|
93
|
+
}
|
|
94
|
+
sendVoiceStateUpdate(channelId, selfMute = false, selfDeaf = false) {
|
|
95
|
+
if (!this.adapter)
|
|
96
|
+
return;
|
|
97
|
+
this.adapter.sendPayload({
|
|
98
|
+
op: 4, // VOICE_STATE_UPDATE
|
|
99
|
+
d: {
|
|
100
|
+
guild_id: this.guildId,
|
|
101
|
+
channel_id: channelId,
|
|
102
|
+
self_mute: selfMute,
|
|
103
|
+
self_deaf: selfDeaf
|
|
104
|
+
}
|
|
105
|
+
});
|
|
106
|
+
}
|
|
107
|
+
handleVoiceServerUpdate(data) {
|
|
108
|
+
this.livekitEndpoint = data.endpoint;
|
|
109
|
+
this.livekitToken = data.token;
|
|
110
|
+
this.livekitRoomName = data.room;
|
|
111
|
+
this.setState({ status: enums_1.VoiceConnectionStatus.Signalling });
|
|
112
|
+
this.connectToLiveKit();
|
|
113
|
+
}
|
|
114
|
+
handleVoiceStateUpdate(data) {
|
|
115
|
+
if (data.channel_id === null) {
|
|
116
|
+
// Disconnected
|
|
117
|
+
this.disconnectFromLiveKit();
|
|
118
|
+
this.setState({ status: enums_1.VoiceConnectionStatus.Disconnected });
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
async connectToLiveKit() {
|
|
122
|
+
if (!this.livekitEndpoint || !this.livekitToken) {
|
|
123
|
+
this.emit('error', new Error('Missing LiveKit connection info'));
|
|
124
|
+
return;
|
|
125
|
+
}
|
|
126
|
+
try {
|
|
127
|
+
// Disconnect existing room if any
|
|
128
|
+
await this.disconnectFromLiveKit();
|
|
129
|
+
this.room = new rtc_node_1.Room();
|
|
130
|
+
this.room.on(rtc_node_1.RoomEvent.Disconnected, () => {
|
|
131
|
+
this.setState({ status: enums_1.VoiceConnectionStatus.Disconnected });
|
|
132
|
+
});
|
|
133
|
+
await this.room.connect(this.livekitEndpoint, this.livekitToken);
|
|
134
|
+
this.setState({ status: enums_1.VoiceConnectionStatus.Ready });
|
|
135
|
+
// Notify subscribed player that connection is ready
|
|
136
|
+
if (this.subscribedPlayer) {
|
|
137
|
+
this.subscribedPlayer.onConnectionReady(this);
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
catch (error) {
|
|
141
|
+
this.emit('error', error);
|
|
142
|
+
this.setState({ status: enums_1.VoiceConnectionStatus.Disconnected });
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
async disconnectFromLiveKit() {
|
|
146
|
+
if (this.room) {
|
|
147
|
+
try {
|
|
148
|
+
await this.room.disconnect();
|
|
149
|
+
}
|
|
150
|
+
catch (e) {
|
|
151
|
+
// Ignore disconnect errors
|
|
152
|
+
}
|
|
153
|
+
this.room = null;
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
setState(newState) {
|
|
157
|
+
const oldState = this.state;
|
|
158
|
+
this.state = newState;
|
|
159
|
+
this.emit('stateChange', oldState, newState);
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
exports.VoiceConnection = VoiceConnection;
|
|
163
|
+
// Store active connections
|
|
164
|
+
const connections = new Map();
|
|
165
|
+
/**
|
|
166
|
+
* Join a voice channel - main entry point
|
|
167
|
+
*/
|
|
168
|
+
function joinVoiceChannel(options) {
|
|
169
|
+
const key = `${options.guildId}:${options.channelId}`;
|
|
170
|
+
// Destroy existing connection if any
|
|
171
|
+
const existing = connections.get(key);
|
|
172
|
+
if (existing) {
|
|
173
|
+
existing.destroy();
|
|
174
|
+
connections.delete(key);
|
|
175
|
+
}
|
|
176
|
+
const connection = new VoiceConnection(options);
|
|
177
|
+
connections.set(key, connection);
|
|
178
|
+
return connection;
|
|
179
|
+
}
|
|
180
|
+
/**
|
|
181
|
+
* Get an existing voice connection
|
|
182
|
+
*/
|
|
183
|
+
function getVoiceConnection(guildId) {
|
|
184
|
+
for (const [key, connection] of connections) {
|
|
185
|
+
if (key.startsWith(`${guildId}:`)) {
|
|
186
|
+
return connection;
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
return undefined;
|
|
190
|
+
}
|
|
191
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"VoiceConnection.js","sourceRoot":"","sources":["../src/VoiceConnection.ts"],"names":[],"mappings":"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","sourcesContent":["import { EventEmitter } from 'events';\r\nimport { Room, RoomEvent } from '@livekit/rtc-node';\r\nimport { VoiceConnectionStatus } from './enums';\r\nimport { \r\n  JoinVoiceChannelOptions, \r\n  VoiceConnectionState,\r\n  GatewayAdapterLibraryMethods,\r\n  VoiceServerUpdate\r\n} from './types';\r\nimport { AudioPlayer } from './AudioPlayer';\r\n\r\n/**\r\n * Represents a voice connection to a channel\r\n */\r\nexport class VoiceConnection extends EventEmitter {\r\n  /** Current connection state */\r\n  public state: VoiceConnectionState = { status: VoiceConnectionStatus.Connecting };\r\n  \r\n  /** The channel ID this connection is for */\r\n  public readonly channelId: string;\r\n  \r\n  /** The guild ID this connection is for */\r\n  public readonly guildId: string;\r\n  \r\n  /** LiveKit room instance */\r\n  private room: Room | null = null;\r\n  \r\n  /** LiveKit connection info */\r\n  private livekitEndpoint: string | null = null;\r\n  private livekitToken: string | null = null;\r\n  private livekitRoomName: string | null = null;\r\n  \r\n  /** Subscribed audio player */\r\n  private subscribedPlayer: AudioPlayer | null = null;\r\n  \r\n  /** Gateway adapter methods */\r\n  private adapterMethods: GatewayAdapterLibraryMethods;\r\n  \r\n  /** Adapter implementer (for sending payloads) */\r\n  private adapter: { sendPayload: (payload: any) => boolean; destroy: () => void } | null = null;\r\n\r\n  constructor(options: JoinVoiceChannelOptions) {\r\n    super();\r\n    this.channelId = options.channelId;\r\n    this.guildId = options.guildId;\r\n    \r\n    // Create adapter methods that will receive gateway events\r\n    this.adapterMethods = {\r\n      onVoiceServerUpdate: (data: VoiceServerUpdate) => this.handleVoiceServerUpdate(data),\r\n      onVoiceStateUpdate: (data) => this.handleVoiceStateUpdate(data),\r\n      destroy: () => this.destroy()\r\n    };\r\n    \r\n    // Get adapter from creator\r\n    this.adapter = options.adapterCreator(this.adapterMethods);\r\n    \r\n    // Send voice state update to join channel\r\n    this.sendVoiceStateUpdate(options.channelId, options.selfMute, options.selfDeaf);\r\n  }\r\n\r\n  /**\r\n   * Subscribe an audio player to this connection\r\n   */\r\n  subscribe(player: AudioPlayer): void {\r\n    if (this.subscribedPlayer) {\r\n      this.subscribedPlayer.unsubscribe(this);\r\n    }\r\n    this.subscribedPlayer = player;\r\n    player.subscribe(this);\r\n  }\r\n\r\n  /**\r\n   * Unsubscribe the current audio player\r\n   */\r\n  unsubscribe(): void {\r\n    if (this.subscribedPlayer) {\r\n      this.subscribedPlayer.unsubscribe(this);\r\n      this.subscribedPlayer = null;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Get the LiveKit room (for audio player to publish tracks)\r\n   */\r\n  getRoom(): Room | null {\r\n    return this.room;\r\n  }\r\n\r\n  /**\r\n   * Disconnect from the voice channel\r\n   */\r\n  disconnect(): boolean {\r\n    this.sendVoiceStateUpdate(null);\r\n    return true;\r\n  }\r\n\r\n  /**\r\n   * Destroy the connection completely\r\n   */\r\n  destroy(): void {\r\n    this.unsubscribe();\r\n    this.disconnectFromLiveKit();\r\n    this.adapter?.destroy();\r\n    this.setState({ status: VoiceConnectionStatus.Destroyed });\r\n  }\r\n\r\n  /**\r\n   * Rejoin the voice channel (after disconnect)\r\n   */\r\n  rejoin(): boolean {\r\n    this.sendVoiceStateUpdate(this.channelId);\r\n    return true;\r\n  }\r\n\r\n  private sendVoiceStateUpdate(channelId: string | null, selfMute = false, selfDeaf = false): void {\r\n    if (!this.adapter) return;\r\n    \r\n    this.adapter.sendPayload({\r\n      op: 4, // VOICE_STATE_UPDATE\r\n      d: {\r\n        guild_id: this.guildId,\r\n        channel_id: channelId,\r\n        self_mute: selfMute,\r\n        self_deaf: selfDeaf\r\n      }\r\n    });\r\n  }\r\n\r\n  private handleVoiceServerUpdate(data: VoiceServerUpdate): void {\r\n    this.livekitEndpoint = data.endpoint;\r\n    this.livekitToken = data.token;\r\n    this.livekitRoomName = data.room;\r\n    \r\n    this.setState({ status: VoiceConnectionStatus.Signalling });\r\n    this.connectToLiveKit();\r\n  }\r\n\r\n  private handleVoiceStateUpdate(data: any): void {\r\n    if (data.channel_id === null) {\r\n      // Disconnected\r\n      this.disconnectFromLiveKit();\r\n      this.setState({ status: VoiceConnectionStatus.Disconnected });\r\n    }\r\n  }\r\n\r\n  private async connectToLiveKit(): Promise<void> {\r\n    if (!this.livekitEndpoint || !this.livekitToken) {\r\n      this.emit('error', new Error('Missing LiveKit connection info'));\r\n      return;\r\n    }\r\n\r\n    try {\r\n      // Disconnect existing room if any\r\n      await this.disconnectFromLiveKit();\r\n      \r\n      this.room = new Room();\r\n      \r\n      this.room.on(RoomEvent.Disconnected, () => {\r\n        this.setState({ status: VoiceConnectionStatus.Disconnected });\r\n      });\r\n\r\n      await this.room.connect(this.livekitEndpoint, this.livekitToken);\r\n      \r\n      this.setState({ status: VoiceConnectionStatus.Ready });\r\n      \r\n      // Notify subscribed player that connection is ready\r\n      if (this.subscribedPlayer) {\r\n        this.subscribedPlayer.onConnectionReady(this);\r\n      }\r\n    } catch (error) {\r\n      this.emit('error', error);\r\n      this.setState({ status: VoiceConnectionStatus.Disconnected });\r\n    }\r\n  }\r\n\r\n  private async disconnectFromLiveKit(): Promise<void> {\r\n    if (this.room) {\r\n      try {\r\n        await this.room.disconnect();\r\n      } catch (e) {\r\n        // Ignore disconnect errors\r\n      }\r\n      this.room = null;\r\n    }\r\n  }\r\n\r\n  private setState(newState: VoiceConnectionState): void {\r\n    const oldState = this.state;\r\n    this.state = newState;\r\n    this.emit('stateChange', oldState, newState);\r\n  }\r\n}\r\n\r\n// Store active connections\r\nconst connections = new Map<string, VoiceConnection>();\r\n\r\n/**\r\n * Join a voice channel - main entry point\r\n */\r\nexport function joinVoiceChannel(options: JoinVoiceChannelOptions): VoiceConnection {\r\n  const key = `${options.guildId}:${options.channelId}`;\r\n  \r\n  // Destroy existing connection if any\r\n  const existing = connections.get(key);\r\n  if (existing) {\r\n    existing.destroy();\r\n    connections.delete(key);\r\n  }\r\n  \r\n  const connection = new VoiceConnection(options);\r\n  connections.set(key, connection);\r\n  \r\n  return connection;\r\n}\r\n\r\n/**\r\n * Get an existing voice connection\r\n */\r\nexport function getVoiceConnection(guildId: string): VoiceConnection | undefined {\r\n  for (const [key, connection] of connections) {\r\n    if (key.startsWith(`${guildId}:`)) {\r\n      return connection;\r\n    }\r\n  }\r\n  return undefined;\r\n}\r\n"]}
|
package/dist/enums.d.ts
ADDED
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Voice connection status
|
|
3
|
+
*/
|
|
4
|
+
export declare enum VoiceConnectionStatus {
|
|
5
|
+
/** Connection is being established */
|
|
6
|
+
Connecting = "connecting",
|
|
7
|
+
/** Connection is ready to use */
|
|
8
|
+
Ready = "ready",
|
|
9
|
+
/** Connection is disconnected */
|
|
10
|
+
Disconnected = "disconnected",
|
|
11
|
+
/** Connection is destroyed */
|
|
12
|
+
Destroyed = "destroyed",
|
|
13
|
+
/** Connection is signalling (exchanging info with server) */
|
|
14
|
+
Signalling = "signalling"
|
|
15
|
+
}
|
|
16
|
+
/**
|
|
17
|
+
* Audio player status
|
|
18
|
+
*/
|
|
19
|
+
export declare enum AudioPlayerStatus {
|
|
20
|
+
/** Player is idle (not playing anything) */
|
|
21
|
+
Idle = "idle",
|
|
22
|
+
/** Player is buffering audio */
|
|
23
|
+
Buffering = "buffering",
|
|
24
|
+
/** Player is playing audio */
|
|
25
|
+
Playing = "playing",
|
|
26
|
+
/** Player is paused */
|
|
27
|
+
Paused = "paused",
|
|
28
|
+
/** Player is auto-paused (no subscribers) */
|
|
29
|
+
AutoPaused = "autopaused"
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Audio resource type
|
|
33
|
+
*/
|
|
34
|
+
export declare enum StreamType {
|
|
35
|
+
/** Arbitrary audio stream */
|
|
36
|
+
Arbitrary = "arbitrary",
|
|
37
|
+
/** Raw PCM audio */
|
|
38
|
+
Raw = "raw",
|
|
39
|
+
/** Opus encoded audio */
|
|
40
|
+
Opus = "opus",
|
|
41
|
+
/** OGG/Opus container */
|
|
42
|
+
OggOpus = "ogg/opus",
|
|
43
|
+
/** WebM/Opus container */
|
|
44
|
+
WebmOpus = "webm/opus"
|
|
45
|
+
}
|
package/dist/enums.js
ADDED
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.StreamType = exports.AudioPlayerStatus = exports.VoiceConnectionStatus = void 0;
|
|
4
|
+
/**
|
|
5
|
+
* Voice connection status
|
|
6
|
+
*/
|
|
7
|
+
var VoiceConnectionStatus;
|
|
8
|
+
(function (VoiceConnectionStatus) {
|
|
9
|
+
/** Connection is being established */
|
|
10
|
+
VoiceConnectionStatus["Connecting"] = "connecting";
|
|
11
|
+
/** Connection is ready to use */
|
|
12
|
+
VoiceConnectionStatus["Ready"] = "ready";
|
|
13
|
+
/** Connection is disconnected */
|
|
14
|
+
VoiceConnectionStatus["Disconnected"] = "disconnected";
|
|
15
|
+
/** Connection is destroyed */
|
|
16
|
+
VoiceConnectionStatus["Destroyed"] = "destroyed";
|
|
17
|
+
/** Connection is signalling (exchanging info with server) */
|
|
18
|
+
VoiceConnectionStatus["Signalling"] = "signalling";
|
|
19
|
+
})(VoiceConnectionStatus || (exports.VoiceConnectionStatus = VoiceConnectionStatus = {}));
|
|
20
|
+
/**
|
|
21
|
+
* Audio player status
|
|
22
|
+
*/
|
|
23
|
+
var AudioPlayerStatus;
|
|
24
|
+
(function (AudioPlayerStatus) {
|
|
25
|
+
/** Player is idle (not playing anything) */
|
|
26
|
+
AudioPlayerStatus["Idle"] = "idle";
|
|
27
|
+
/** Player is buffering audio */
|
|
28
|
+
AudioPlayerStatus["Buffering"] = "buffering";
|
|
29
|
+
/** Player is playing audio */
|
|
30
|
+
AudioPlayerStatus["Playing"] = "playing";
|
|
31
|
+
/** Player is paused */
|
|
32
|
+
AudioPlayerStatus["Paused"] = "paused";
|
|
33
|
+
/** Player is auto-paused (no subscribers) */
|
|
34
|
+
AudioPlayerStatus["AutoPaused"] = "autopaused";
|
|
35
|
+
})(AudioPlayerStatus || (exports.AudioPlayerStatus = AudioPlayerStatus = {}));
|
|
36
|
+
/**
|
|
37
|
+
* Audio resource type
|
|
38
|
+
*/
|
|
39
|
+
var StreamType;
|
|
40
|
+
(function (StreamType) {
|
|
41
|
+
/** Arbitrary audio stream */
|
|
42
|
+
StreamType["Arbitrary"] = "arbitrary";
|
|
43
|
+
/** Raw PCM audio */
|
|
44
|
+
StreamType["Raw"] = "raw";
|
|
45
|
+
/** Opus encoded audio */
|
|
46
|
+
StreamType["Opus"] = "opus";
|
|
47
|
+
/** OGG/Opus container */
|
|
48
|
+
StreamType["OggOpus"] = "ogg/opus";
|
|
49
|
+
/** WebM/Opus container */
|
|
50
|
+
StreamType["WebmOpus"] = "webm/opus";
|
|
51
|
+
})(StreamType || (exports.StreamType = StreamType = {}));
|
|
52
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW51bXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvZW51bXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUE7O0dBRUc7QUFDSCxJQUFZLHFCQVdYO0FBWEQsV0FBWSxxQkFBcUI7SUFDL0Isc0NBQXNDO0lBQ3RDLGtEQUF5QixDQUFBO0lBQ3pCLGlDQUFpQztJQUNqQyx3Q0FBZSxDQUFBO0lBQ2YsaUNBQWlDO0lBQ2pDLHNEQUE2QixDQUFBO0lBQzdCLDhCQUE4QjtJQUM5QixnREFBdUIsQ0FBQTtJQUN2Qiw2REFBNkQ7SUFDN0Qsa0RBQXlCLENBQUE7QUFDM0IsQ0FBQyxFQVhXLHFCQUFxQixxQ0FBckIscUJBQXFCLFFBV2hDO0FBRUQ7O0dBRUc7QUFDSCxJQUFZLGlCQVdYO0FBWEQsV0FBWSxpQkFBaUI7SUFDM0IsNENBQTRDO0lBQzVDLGtDQUFhLENBQUE7SUFDYixnQ0FBZ0M7SUFDaEMsNENBQXVCLENBQUE7SUFDdkIsOEJBQThCO0lBQzlCLHdDQUFtQixDQUFBO0lBQ25CLHVCQUF1QjtJQUN2QixzQ0FBaUIsQ0FBQTtJQUNqQiw2Q0FBNkM7SUFDN0MsOENBQXlCLENBQUE7QUFDM0IsQ0FBQyxFQVhXLGlCQUFpQixpQ0FBakIsaUJBQWlCLFFBVzVCO0FBRUQ7O0dBRUc7QUFDSCxJQUFZLFVBV1g7QUFYRCxXQUFZLFVBQVU7SUFDcEIsNkJBQTZCO0lBQzdCLHFDQUF1QixDQUFBO0lBQ3ZCLG9CQUFvQjtJQUNwQix5QkFBVyxDQUFBO0lBQ1gseUJBQXlCO0lBQ3pCLDJCQUFhLENBQUE7SUFDYix5QkFBeUI7SUFDekIsa0NBQW9CLENBQUE7SUFDcEIsMEJBQTBCO0lBQzFCLG9DQUFzQixDQUFBO0FBQ3hCLENBQUMsRUFYVyxVQUFVLDBCQUFWLFVBQVUsUUFXckIiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcclxuICogVm9pY2UgY29ubmVjdGlvbiBzdGF0dXNcclxuICovXHJcbmV4cG9ydCBlbnVtIFZvaWNlQ29ubmVjdGlvblN0YXR1cyB7XHJcbiAgLyoqIENvbm5lY3Rpb24gaXMgYmVpbmcgZXN0YWJsaXNoZWQgKi9cclxuICBDb25uZWN0aW5nID0gJ2Nvbm5lY3RpbmcnLFxyXG4gIC8qKiBDb25uZWN0aW9uIGlzIHJlYWR5IHRvIHVzZSAqL1xyXG4gIFJlYWR5ID0gJ3JlYWR5JyxcclxuICAvKiogQ29ubmVjdGlvbiBpcyBkaXNjb25uZWN0ZWQgKi9cclxuICBEaXNjb25uZWN0ZWQgPSAnZGlzY29ubmVjdGVkJyxcclxuICAvKiogQ29ubmVjdGlvbiBpcyBkZXN0cm95ZWQgKi9cclxuICBEZXN0cm95ZWQgPSAnZGVzdHJveWVkJyxcclxuICAvKiogQ29ubmVjdGlvbiBpcyBzaWduYWxsaW5nIChleGNoYW5naW5nIGluZm8gd2l0aCBzZXJ2ZXIpICovXHJcbiAgU2lnbmFsbGluZyA9ICdzaWduYWxsaW5nJ1xyXG59XHJcblxyXG4vKipcclxuICogQXVkaW8gcGxheWVyIHN0YXR1c1xyXG4gKi9cclxuZXhwb3J0IGVudW0gQXVkaW9QbGF5ZXJTdGF0dXMge1xyXG4gIC8qKiBQbGF5ZXIgaXMgaWRsZSAobm90IHBsYXlpbmcgYW55dGhpbmcpICovXHJcbiAgSWRsZSA9ICdpZGxlJyxcclxuICAvKiogUGxheWVyIGlzIGJ1ZmZlcmluZyBhdWRpbyAqL1xyXG4gIEJ1ZmZlcmluZyA9ICdidWZmZXJpbmcnLFxyXG4gIC8qKiBQbGF5ZXIgaXMgcGxheWluZyBhdWRpbyAqL1xyXG4gIFBsYXlpbmcgPSAncGxheWluZycsXHJcbiAgLyoqIFBsYXllciBpcyBwYXVzZWQgKi9cclxuICBQYXVzZWQgPSAncGF1c2VkJyxcclxuICAvKiogUGxheWVyIGlzIGF1dG8tcGF1c2VkIChubyBzdWJzY3JpYmVycykgKi9cclxuICBBdXRvUGF1c2VkID0gJ2F1dG9wYXVzZWQnXHJcbn1cclxuXHJcbi8qKlxyXG4gKiBBdWRpbyByZXNvdXJjZSB0eXBlXHJcbiAqL1xyXG5leHBvcnQgZW51bSBTdHJlYW1UeXBlIHtcclxuICAvKiogQXJiaXRyYXJ5IGF1ZGlvIHN0cmVhbSAqL1xyXG4gIEFyYml0cmFyeSA9ICdhcmJpdHJhcnknLFxyXG4gIC8qKiBSYXcgUENNIGF1ZGlvICovXHJcbiAgUmF3ID0gJ3JhdycsXHJcbiAgLyoqIE9wdXMgZW5jb2RlZCBhdWRpbyAqL1xyXG4gIE9wdXMgPSAnb3B1cycsXHJcbiAgLyoqIE9HRy9PcHVzIGNvbnRhaW5lciAqL1xyXG4gIE9nZ09wdXMgPSAnb2dnL29wdXMnLFxyXG4gIC8qKiBXZWJNL09wdXMgY29udGFpbmVyICovXHJcbiAgV2VibU9wdXMgPSAnd2VibS9vcHVzJ1xyXG59XHJcbiJdfQ==
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @jubbio/voice - Voice library for Jubbio bots
|
|
3
|
+
*
|
|
4
|
+
* Usage:
|
|
5
|
+
* ```typescript
|
|
6
|
+
* import { joinVoiceChannel, createAudioPlayer, createAudioResource } from '@jubbio/voice';
|
|
7
|
+
*
|
|
8
|
+
* const connection = joinVoiceChannel({
|
|
9
|
+
* channelId: '123456789',
|
|
10
|
+
* guildId: '987654321',
|
|
11
|
+
* adapterCreator: client.guilds.cache.get('987654321')!.voiceAdapterCreator
|
|
12
|
+
* });
|
|
13
|
+
*
|
|
14
|
+
* const player = createAudioPlayer();
|
|
15
|
+
* const resource = createAudioResource('https://example.com/audio.mp3');
|
|
16
|
+
*
|
|
17
|
+
* player.play(resource);
|
|
18
|
+
* connection.subscribe(player);
|
|
19
|
+
* ```
|
|
20
|
+
*/
|
|
21
|
+
export * from './VoiceConnection';
|
|
22
|
+
export * from './AudioPlayer';
|
|
23
|
+
export * from './AudioResource';
|
|
24
|
+
export * from './types';
|
|
25
|
+
export * from './enums';
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* @jubbio/voice - Voice library for Jubbio bots
|
|
4
|
+
*
|
|
5
|
+
* Usage:
|
|
6
|
+
* ```typescript
|
|
7
|
+
* import { joinVoiceChannel, createAudioPlayer, createAudioResource } from '@jubbio/voice';
|
|
8
|
+
*
|
|
9
|
+
* const connection = joinVoiceChannel({
|
|
10
|
+
* channelId: '123456789',
|
|
11
|
+
* guildId: '987654321',
|
|
12
|
+
* adapterCreator: client.guilds.cache.get('987654321')!.voiceAdapterCreator
|
|
13
|
+
* });
|
|
14
|
+
*
|
|
15
|
+
* const player = createAudioPlayer();
|
|
16
|
+
* const resource = createAudioResource('https://example.com/audio.mp3');
|
|
17
|
+
*
|
|
18
|
+
* player.play(resource);
|
|
19
|
+
* connection.subscribe(player);
|
|
20
|
+
* ```
|
|
21
|
+
*/
|
|
22
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
23
|
+
if (k2 === undefined) k2 = k;
|
|
24
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
25
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
26
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
27
|
+
}
|
|
28
|
+
Object.defineProperty(o, k2, desc);
|
|
29
|
+
}) : (function(o, m, k, k2) {
|
|
30
|
+
if (k2 === undefined) k2 = k;
|
|
31
|
+
o[k2] = m[k];
|
|
32
|
+
}));
|
|
33
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
34
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
35
|
+
};
|
|
36
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
37
|
+
__exportStar(require("./VoiceConnection"), exports);
|
|
38
|
+
__exportStar(require("./AudioPlayer"), exports);
|
|
39
|
+
__exportStar(require("./AudioResource"), exports);
|
|
40
|
+
__exportStar(require("./types"), exports);
|
|
41
|
+
__exportStar(require("./enums"), exports);
|
|
42
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBbUJHOzs7Ozs7Ozs7Ozs7Ozs7O0FBRUgsb0RBQWtDO0FBQ2xDLGdEQUE4QjtBQUM5QixrREFBZ0M7QUFDaEMsMENBQXdCO0FBQ3hCLDBDQUF3QiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxyXG4gKiBAanViYmlvL3ZvaWNlIC0gVm9pY2UgbGlicmFyeSBmb3IgSnViYmlvIGJvdHNcclxuICogXHJcbiAqIFVzYWdlOlxyXG4gKiBgYGB0eXBlc2NyaXB0XHJcbiAqIGltcG9ydCB7IGpvaW5Wb2ljZUNoYW5uZWwsIGNyZWF0ZUF1ZGlvUGxheWVyLCBjcmVhdGVBdWRpb1Jlc291cmNlIH0gZnJvbSAnQGp1YmJpby92b2ljZSc7XHJcbiAqIFxyXG4gKiBjb25zdCBjb25uZWN0aW9uID0gam9pblZvaWNlQ2hhbm5lbCh7XHJcbiAqICAgY2hhbm5lbElkOiAnMTIzNDU2Nzg5JyxcclxuICogICBndWlsZElkOiAnOTg3NjU0MzIxJyxcclxuICogICBhZGFwdGVyQ3JlYXRvcjogY2xpZW50Lmd1aWxkcy5jYWNoZS5nZXQoJzk4NzY1NDMyMScpIS52b2ljZUFkYXB0ZXJDcmVhdG9yXHJcbiAqIH0pO1xyXG4gKiBcclxuICogY29uc3QgcGxheWVyID0gY3JlYXRlQXVkaW9QbGF5ZXIoKTtcclxuICogY29uc3QgcmVzb3VyY2UgPSBjcmVhdGVBdWRpb1Jlc291cmNlKCdodHRwczovL2V4YW1wbGUuY29tL2F1ZGlvLm1wMycpO1xyXG4gKiBcclxuICogcGxheWVyLnBsYXkocmVzb3VyY2UpO1xyXG4gKiBjb25uZWN0aW9uLnN1YnNjcmliZShwbGF5ZXIpO1xyXG4gKiBgYGBcclxuICovXHJcblxyXG5leHBvcnQgKiBmcm9tICcuL1ZvaWNlQ29ubmVjdGlvbic7XHJcbmV4cG9ydCAqIGZyb20gJy4vQXVkaW9QbGF5ZXInO1xyXG5leHBvcnQgKiBmcm9tICcuL0F1ZGlvUmVzb3VyY2UnO1xyXG5leHBvcnQgKiBmcm9tICcuL3R5cGVzJztcclxuZXhwb3J0ICogZnJvbSAnLi9lbnVtcyc7XHJcbiJdfQ==
|