@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.
@@ -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":";;;AAuMA,4CAcC;AAKD,gDAOC;AAjOD,mCAAsC;AACtC,gDAAoD;AACpD,mCAAgD;AAShD;;GAEG;AACH,MAAa,eAAgB,SAAQ,qBAAY;IAC/C,+BAA+B;IACxB,KAAK,GAAyB,EAAE,MAAM,EAAE,6BAAqB,CAAC,UAAU,EAAE,CAAC;IAElF,4CAA4C;IAC5B,SAAS,CAAS;IAElC,0CAA0C;IAC1B,OAAO,CAAS;IAEhC,4BAA4B;IACpB,IAAI,GAAgB,IAAI,CAAC;IAEjC,8BAA8B;IACtB,eAAe,GAAkB,IAAI,CAAC;IACtC,YAAY,GAAkB,IAAI,CAAC;IACnC,eAAe,GAAkB,IAAI,CAAC;IAE9C,8BAA8B;IACtB,gBAAgB,GAAuB,IAAI,CAAC;IAEpD,8BAA8B;IACtB,cAAc,CAA+B;IAErD,iDAAiD;IACzC,OAAO,GAA2E,IAAI,CAAC;IAE/F,YAAY,OAAgC;QAC1C,KAAK,EAAE,CAAC;QACR,IAAI,CAAC,SAAS,GAAG,OAAO,CAAC,SAAS,CAAC;QACnC,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC;QAE/B,0DAA0D;QAC1D,IAAI,CAAC,cAAc,GAAG;YACpB,mBAAmB,EAAE,CAAC,IAAuB,EAAE,EAAE,CAAC,IAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC;YACpF,kBAAkB,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,sBAAsB,CAAC,IAAI,CAAC;YAC/D,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE;SAC9B,CAAC;QAEF,2BAA2B;QAC3B,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC,cAAc,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QAE3D,0CAA0C;QAC1C,IAAI,CAAC,oBAAoB,CAAC,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,QAAQ,EAAE,OAAO,CAAC,QAAQ,CAAC,CAAC;IACnF,CAAC;IAED;;OAEG;IACH,SAAS,CAAC,MAAmB;QAC3B,IAAI,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC1B,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QAC1C,CAAC;QACD,IAAI,CAAC,gBAAgB,GAAG,MAAM,CAAC;QAC/B,MAAM,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;IACzB,CAAC;IAED;;OAEG;IACH,WAAW;QACT,IAAI,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC1B,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;YACxC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;QAC/B,CAAC;IACH,CAAC;IAED;;OAEG;IACH,OAAO;QACL,OAAO,IAAI,CAAC,IAAI,CAAC;IACnB,CAAC;IAED;;OAEG;IACH,UAAU;QACR,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,CAAC;QAChC,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;OAEG;IACH,OAAO;QACL,IAAI,CAAC,WAAW,EAAE,CAAC;QACnB,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC7B,IAAI,CAAC,OAAO,EAAE,OAAO,EAAE,CAAC;QACxB,IAAI,CAAC,QAAQ,CAAC,EAAE,MAAM,EAAE,6BAAqB,CAAC,SAAS,EAAE,CAAC,CAAC;IAC7D,CAAC;IAED;;OAEG;IACH,MAAM;QACJ,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAC1C,OAAO,IAAI,CAAC;IACd,CAAC;IAEO,oBAAoB,CAAC,SAAwB,EAAE,QAAQ,GAAG,KAAK,EAAE,QAAQ,GAAG,KAAK;QACvF,IAAI,CAAC,IAAI,CAAC,OAAO;YAAE,OAAO;QAE1B,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC;YACvB,EAAE,EAAE,CAAC,EAAE,qBAAqB;YAC5B,CAAC,EAAE;gBACD,QAAQ,EAAE,IAAI,CAAC,OAAO;gBACtB,UAAU,EAAE,SAAS;gBACrB,SAAS,EAAE,QAAQ;gBACnB,SAAS,EAAE,QAAQ;aACpB;SACF,CAAC,CAAC;IACL,CAAC;IAEO,uBAAuB,CAAC,IAAuB;QACrD,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,QAAQ,CAAC;QACrC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC;QAC/B,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,IAAI,CAAC;QAEjC,IAAI,CAAC,QAAQ,CAAC,EAAE,MAAM,EAAE,6BAAqB,CAAC,UAAU,EAAE,CAAC,CAAC;QAC5D,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAC1B,CAAC;IAEO,sBAAsB,CAAC,IAAS;QACtC,IAAI,IAAI,CAAC,UAAU,KAAK,IAAI,EAAE,CAAC;YAC7B,eAAe;YACf,IAAI,CAAC,qBAAqB,EAAE,CAAC;YAC7B,IAAI,CAAC,QAAQ,CAAC,EAAE,MAAM,EAAE,6BAAqB,CAAC,YAAY,EAAE,CAAC,CAAC;QAChE,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,gBAAgB;QAC5B,IAAI,CAAC,IAAI,CAAC,eAAe,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC;YAChD,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,KAAK,CAAC,iCAAiC,CAAC,CAAC,CAAC;YACjE,OAAO;QACT,CAAC;QAED,IAAI,CAAC;YACH,kCAAkC;YAClC,MAAM,IAAI,CAAC,qBAAqB,EAAE,CAAC;YAEnC,IAAI,CAAC,IAAI,GAAG,IAAI,eAAI,EAAE,CAAC;YAEvB,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,oBAAS,CAAC,YAAY,EAAE,GAAG,EAAE;gBACxC,IAAI,CAAC,QAAQ,CAAC,EAAE,MAAM,EAAE,6BAAqB,CAAC,YAAY,EAAE,CAAC,CAAC;YAChE,CAAC,CAAC,CAAC;YAEH,MAAM,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,eAAe,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC;YAEjE,IAAI,CAAC,QAAQ,CAAC,EAAE,MAAM,EAAE,6BAAqB,CAAC,KAAK,EAAE,CAAC,CAAC;YAEvD,oDAAoD;YACpD,IAAI,IAAI,CAAC,gBAAgB,EAAE,CAAC;gBAC1B,IAAI,CAAC,gBAAgB,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;YAChD,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;YAC1B,IAAI,CAAC,QAAQ,CAAC,EAAE,MAAM,EAAE,6BAAqB,CAAC,YAAY,EAAE,CAAC,CAAC;QAChE,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,qBAAqB;QACjC,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC;YACd,IAAI,CAAC;gBACH,MAAM,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC;YAC/B,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,2BAA2B;YAC7B,CAAC;YACD,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACnB,CAAC;IACH,CAAC;IAEO,QAAQ,CAAC,QAA8B;QAC7C,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC;QAC5B,IAAI,CAAC,KAAK,GAAG,QAAQ,CAAC;QACtB,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,QAAQ,EAAE,QAAQ,CAAC,CAAC;IAC/C,CAAC;CACF;AAjLD,0CAiLC;AAED,2BAA2B;AAC3B,MAAM,WAAW,GAAG,IAAI,GAAG,EAA2B,CAAC;AAEvD;;GAEG;AACH,SAAgB,gBAAgB,CAAC,OAAgC;IAC/D,MAAM,GAAG,GAAG,GAAG,OAAO,CAAC,OAAO,IAAI,OAAO,CAAC,SAAS,EAAE,CAAC;IAEtD,qCAAqC;IACrC,MAAM,QAAQ,GAAG,WAAW,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;IACtC,IAAI,QAAQ,EAAE,CAAC;QACb,QAAQ,CAAC,OAAO,EAAE,CAAC;QACnB,WAAW,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;IAC1B,CAAC;IAED,MAAM,UAAU,GAAG,IAAI,eAAe,CAAC,OAAO,CAAC,CAAC;IAChD,WAAW,CAAC,GAAG,CAAC,GAAG,EAAE,UAAU,CAAC,CAAC;IAEjC,OAAO,UAAU,CAAC;AACpB,CAAC;AAED;;GAEG;AACH,SAAgB,kBAAkB,CAAC,OAAe;IAChD,KAAK,MAAM,CAAC,GAAG,EAAE,UAAU,CAAC,IAAI,WAAW,EAAE,CAAC;QAC5C,IAAI,GAAG,CAAC,UAAU,CAAC,GAAG,OAAO,GAAG,CAAC,EAAE,CAAC;YAClC,OAAO,UAAU,CAAC;QACpB,CAAC;IACH,CAAC;IACD,OAAO,SAAS,CAAC;AACnB,CAAC","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"]}
@@ -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==
@@ -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==