@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,385 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.AudioPlayer = void 0;
4
+ exports.createAudioPlayer = createAudioPlayer;
5
+ const events_1 = require("events");
6
+ const child_process_1 = require("child_process");
7
+ const process_1 = require("process");
8
+ const rtc_node_1 = require("@livekit/rtc-node");
9
+ const enums_1 = require("./enums");
10
+ // Audio settings for LiveKit (48kHz stereo)
11
+ const SAMPLE_RATE = 48000;
12
+ const CHANNELS = 2;
13
+ const FRAME_DURATION_MS = 20;
14
+ const SAMPLES_PER_FRAME = (SAMPLE_RATE * FRAME_DURATION_MS) / 1000; // 960
15
+ // Jitter buffer settings
16
+ const FRAME_INTERVAL_NS = BigInt(20_000_000); // 20ms in nanoseconds
17
+ const TARGET_BUFFER_FRAMES = 150; // ~3 seconds - target buffer size
18
+ const MIN_BUFFER_FRAMES = 75; // ~1.5 seconds - minimum before we start
19
+ const MAX_BUFFER_FRAMES = 500; // ~10 seconds - max buffer to prevent memory issues
20
+ const LOW_BUFFER_THRESHOLD = 50; // ~1 second - when to slow down playback
21
+ /**
22
+ * Audio player for playing audio resources
23
+ */
24
+ class AudioPlayer extends events_1.EventEmitter {
25
+ /** Current player state */
26
+ state = { status: enums_1.AudioPlayerStatus.Idle };
27
+ /** Player options */
28
+ options;
29
+ /** Subscribed voice connections */
30
+ subscriptions = new Set();
31
+ /** Current audio resource */
32
+ currentResource = null;
33
+ /** FFmpeg process */
34
+ ffmpegProcess = null;
35
+ /** LiveKit audio source and track */
36
+ audioSource = null;
37
+ audioTrack = null;
38
+ /** Frame queue and playback state */
39
+ frameQueue = [];
40
+ playbackTimeout = null;
41
+ leftoverBuffer = null;
42
+ isPublished = false;
43
+ /** High-resolution timing */
44
+ nextFrameTime = BigInt(0);
45
+ isPlaybackLoopRunning = false;
46
+ ffmpegDone = false;
47
+ /** Buffer statistics */
48
+ bufferUnderruns = 0;
49
+ framesPlayed = 0;
50
+ constructor(options = {}) {
51
+ super();
52
+ this.options = {
53
+ behaviors: {
54
+ noSubscriber: 'pause',
55
+ maxMissedFrames: 5,
56
+ ...options.behaviors
57
+ }
58
+ };
59
+ }
60
+ /**
61
+ * Play an audio resource
62
+ */
63
+ play(resource) {
64
+ // Stop current playback
65
+ this.stop();
66
+ this.currentResource = resource;
67
+ this.setState({ status: enums_1.AudioPlayerStatus.Buffering, resource });
68
+ // Start playback if we have a ready connection
69
+ for (const connection of this.subscriptions) {
70
+ if (connection.getRoom()) {
71
+ this.startPlayback(connection);
72
+ break;
73
+ }
74
+ }
75
+ }
76
+ /**
77
+ * Pause playback
78
+ */
79
+ pause() {
80
+ if (this.state.status !== enums_1.AudioPlayerStatus.Playing) {
81
+ return false;
82
+ }
83
+ this.setState({ status: enums_1.AudioPlayerStatus.Paused, resource: this.currentResource });
84
+ return true;
85
+ }
86
+ /**
87
+ * Unpause playback
88
+ */
89
+ unpause() {
90
+ if (this.state.status !== enums_1.AudioPlayerStatus.Paused) {
91
+ return false;
92
+ }
93
+ this.setState({ status: enums_1.AudioPlayerStatus.Playing, resource: this.currentResource });
94
+ return true;
95
+ }
96
+ /**
97
+ * Stop playback
98
+ */
99
+ stop(force = false) {
100
+ if (this.state.status === enums_1.AudioPlayerStatus.Idle && !force) {
101
+ return false;
102
+ }
103
+ this.cleanup();
104
+ this.currentResource = null;
105
+ this.setState({ status: enums_1.AudioPlayerStatus.Idle });
106
+ return true;
107
+ }
108
+ /**
109
+ * Subscribe a voice connection to this player
110
+ * @internal
111
+ */
112
+ subscribe(connection) {
113
+ this.subscriptions.add(connection);
114
+ }
115
+ /**
116
+ * Unsubscribe a voice connection from this player
117
+ * @internal
118
+ */
119
+ unsubscribe(connection) {
120
+ this.subscriptions.delete(connection);
121
+ // Auto-pause if no subscribers
122
+ if (this.subscriptions.size === 0 && this.options.behaviors?.noSubscriber === 'pause') {
123
+ if (this.state.status === enums_1.AudioPlayerStatus.Playing) {
124
+ this.setState({ status: enums_1.AudioPlayerStatus.AutoPaused, resource: this.currentResource });
125
+ }
126
+ }
127
+ }
128
+ /**
129
+ * Called when a connection becomes ready
130
+ * @internal
131
+ */
132
+ onConnectionReady(connection) {
133
+ // If we have a resource waiting, start playback
134
+ if (this.currentResource && this.state.status === enums_1.AudioPlayerStatus.Buffering) {
135
+ this.startPlayback(connection);
136
+ }
137
+ }
138
+ async startPlayback(connection) {
139
+ const room = connection.getRoom();
140
+ if (!room || !this.currentResource)
141
+ return;
142
+ try {
143
+ // Create audio source and track
144
+ await this.setupAudioTrack(room);
145
+ // Start FFmpeg to decode audio - this will set state to Playing when ready
146
+ await this.startFFmpeg();
147
+ }
148
+ catch (error) {
149
+ this.emit('error', { message: error.message, resource: this.currentResource });
150
+ this.stop();
151
+ }
152
+ }
153
+ async setupAudioTrack(room) {
154
+ if (this.isPublished)
155
+ return;
156
+ this.audioSource = new rtc_node_1.AudioSource(SAMPLE_RATE, CHANNELS);
157
+ this.audioTrack = rtc_node_1.LocalAudioTrack.createAudioTrack('music', this.audioSource);
158
+ const options = new rtc_node_1.TrackPublishOptions();
159
+ options.source = rtc_node_1.TrackSource.SOURCE_MICROPHONE;
160
+ if (room.localParticipant) {
161
+ await room.localParticipant.publishTrack(this.audioTrack, options);
162
+ }
163
+ this.isPublished = true;
164
+ }
165
+ async startFFmpeg() {
166
+ if (!this.currentResource)
167
+ return;
168
+ const inputSource = this.currentResource.getInputSource();
169
+ console.log(`FFmpeg input source: ${inputSource.substring(0, 100)}...`);
170
+ // Check if this is a streaming URL that needs yt-dlp
171
+ const needsYtDlp = inputSource.includes('youtube.com') ||
172
+ inputSource.includes('youtu.be') ||
173
+ inputSource.includes('soundcloud.com') ||
174
+ inputSource.includes('twitch.tv') ||
175
+ inputSource.startsWith('ytsearch:');
176
+ if (needsYtDlp) {
177
+ // Use yt-dlp to pipe audio directly to FFmpeg
178
+ console.log('Using yt-dlp pipe mode');
179
+ this.ffmpegProcess = (0, child_process_1.spawn)('bash', [
180
+ '-c',
181
+ `~/.local/bin/yt-dlp -f "bestaudio/best" -o - --no-playlist --no-warnings "${inputSource}" | ffmpeg -i pipe:0 -f s16le -ar ${SAMPLE_RATE} -ac ${CHANNELS} -acodec pcm_s16le -`
182
+ ], { stdio: ['pipe', 'pipe', 'pipe'] });
183
+ }
184
+ else {
185
+ console.log('Using direct FFmpeg mode');
186
+ this.ffmpegProcess = (0, child_process_1.spawn)('ffmpeg', [
187
+ '-reconnect', '1',
188
+ '-reconnect_streamed', '1',
189
+ '-reconnect_delay_max', '5',
190
+ '-i', inputSource,
191
+ '-f', 's16le',
192
+ '-ar', String(SAMPLE_RATE),
193
+ '-ac', String(CHANNELS),
194
+ '-acodec', 'pcm_s16le',
195
+ '-'
196
+ ], { stdio: ['pipe', 'pipe', 'pipe'] });
197
+ }
198
+ const frameSize = SAMPLES_PER_FRAME * CHANNELS * 2;
199
+ this.ffmpegDone = false;
200
+ let hasReceivedData = false;
201
+ this.ffmpegProcess.stdout?.on('data', (chunk) => {
202
+ if (this.state.status !== enums_1.AudioPlayerStatus.Playing &&
203
+ this.state.status !== enums_1.AudioPlayerStatus.Buffering)
204
+ return;
205
+ hasReceivedData = true;
206
+ // Handle leftover from previous chunk
207
+ if (this.leftoverBuffer && this.leftoverBuffer.length > 0) {
208
+ chunk = Buffer.concat([this.leftoverBuffer, chunk]);
209
+ this.leftoverBuffer = null;
210
+ }
211
+ let offset = 0;
212
+ while (offset + frameSize <= chunk.length) {
213
+ const frame = chunk.slice(offset, offset + frameSize);
214
+ const int16Array = new Int16Array(SAMPLES_PER_FRAME * CHANNELS);
215
+ for (let i = 0; i < int16Array.length; i++) {
216
+ int16Array[i] = frame.readInt16LE(i * 2);
217
+ }
218
+ this.frameQueue.push(int16Array);
219
+ offset += frameSize;
220
+ }
221
+ // Save leftover
222
+ if (offset < chunk.length) {
223
+ this.leftoverBuffer = chunk.slice(offset);
224
+ }
225
+ });
226
+ let stderrOutput = '';
227
+ this.ffmpegProcess.stderr?.on('data', (data) => {
228
+ stderrOutput += data.toString();
229
+ });
230
+ this.ffmpegProcess.on('close', (code) => {
231
+ this.ffmpegDone = true;
232
+ this.ffmpegProcess = null;
233
+ if (code !== 0) {
234
+ console.error(`FFmpeg stderr:\n${stderrOutput}`);
235
+ }
236
+ console.log(`[AudioPlayer] FFmpeg closed with code ${code}, hasReceivedData: ${hasReceivedData}, queue: ${this.frameQueue.length}`);
237
+ });
238
+ this.ffmpegProcess.on('error', (err) => {
239
+ console.error('FFmpeg process error:', err.message);
240
+ this.emit('error', { message: err.message, resource: this.currentResource });
241
+ });
242
+ // Wait for initial buffer with timeout
243
+ const bufferTimeout = 10000; // 10 seconds for initial buffer
244
+ const startTime = Date.now();
245
+ while (this.frameQueue.length < MIN_BUFFER_FRAMES && Date.now() - startTime < bufferTimeout) {
246
+ await new Promise(r => setTimeout(r, 100));
247
+ // Check if FFmpeg failed early
248
+ if (this.ffmpegDone && this.frameQueue.length === 0) {
249
+ throw new Error('FFmpeg failed to produce audio data');
250
+ }
251
+ }
252
+ if (this.frameQueue.length === 0) {
253
+ throw new Error('Timeout waiting for audio data');
254
+ }
255
+ console.log(`[AudioPlayer] Starting playback with ${this.frameQueue.length} frames buffered (target: ${TARGET_BUFFER_FRAMES})`);
256
+ // Mark ready for playback - setState will trigger the loop
257
+ this.isPlaybackLoopRunning = true;
258
+ this.nextFrameTime = process_1.hrtime.bigint();
259
+ console.log(`[AudioPlayer] Playback ready, audioSource exists: ${!!this.audioSource}`);
260
+ // Set state to playing - this will trigger scheduleNextFrame via setState
261
+ this.setState({ status: enums_1.AudioPlayerStatus.Playing, resource: this.currentResource });
262
+ }
263
+ /**
264
+ * High-resolution frame scheduling using hrtime
265
+ * This provides much more accurate timing than setInterval
266
+ */
267
+ scheduleNextFrame() {
268
+ if (!this.isPlaybackLoopRunning || this.state.status !== enums_1.AudioPlayerStatus.Playing) {
269
+ console.log(`[AudioPlayer] scheduleNextFrame skipped: loopRunning=${this.isPlaybackLoopRunning}, status=${this.state.status}`);
270
+ return;
271
+ }
272
+ const now = process_1.hrtime.bigint();
273
+ const delayNs = this.nextFrameTime - now;
274
+ const delayMs = Number(delayNs) / 1_000_000;
275
+ if (this.framesPlayed === 0) {
276
+ console.log(`[AudioPlayer] First frame scheduling: delayMs=${delayMs.toFixed(2)}`);
277
+ }
278
+ // Schedule next frame
279
+ if (delayMs > 1) {
280
+ this.playbackTimeout = setTimeout(() => this.processFrame(), Math.max(1, delayMs - 1));
281
+ }
282
+ else {
283
+ // We're behind, process immediately
284
+ setImmediate(() => this.processFrame());
285
+ }
286
+ }
287
+ /**
288
+ * Process and send a single audio frame
289
+ */
290
+ async processFrame() {
291
+ if (!this.isPlaybackLoopRunning || this.state.status !== enums_1.AudioPlayerStatus.Playing) {
292
+ if (this.framesPlayed === 0) {
293
+ console.log(`[AudioPlayer] processFrame skipped: loopRunning=${this.isPlaybackLoopRunning}, status=${this.state.status}`);
294
+ }
295
+ return;
296
+ }
297
+ // Check buffer status
298
+ const bufferSize = this.frameQueue.length;
299
+ if (bufferSize > 0 && this.audioSource) {
300
+ const int16Array = this.frameQueue.shift();
301
+ const audioFrame = new rtc_node_1.AudioFrame(int16Array, SAMPLE_RATE, CHANNELS, SAMPLES_PER_FRAME);
302
+ try {
303
+ await this.audioSource.captureFrame(audioFrame);
304
+ this.framesPlayed++;
305
+ // Log progress every 500 frames (~10 seconds)
306
+ if (this.framesPlayed % 500 === 0) {
307
+ console.log(`[AudioPlayer] Progress: ${this.framesPlayed} frames played, buffer: ${bufferSize}`);
308
+ }
309
+ }
310
+ catch (e) {
311
+ console.error(`[AudioPlayer] Frame error:`, e.message);
312
+ }
313
+ // Update timing for next frame
314
+ this.nextFrameTime += FRAME_INTERVAL_NS;
315
+ // Adaptive timing: if buffer is low, slow down slightly to let it recover
316
+ if (bufferSize < LOW_BUFFER_THRESHOLD && !this.ffmpegDone) {
317
+ // Add 1ms delay to let buffer recover
318
+ this.nextFrameTime += BigInt(1_000_000);
319
+ this.bufferUnderruns++;
320
+ if (this.bufferUnderruns % 50 === 0) {
321
+ console.log(`[AudioPlayer] Buffer low: ${bufferSize} frames, ${this.bufferUnderruns} underruns`);
322
+ }
323
+ }
324
+ // Schedule next frame
325
+ this.scheduleNextFrame();
326
+ }
327
+ else if (this.ffmpegDone && bufferSize === 0) {
328
+ // Playback finished
329
+ console.log('[AudioPlayer] Playback finished - queue empty and FFmpeg done');
330
+ this.stop();
331
+ }
332
+ else if (bufferSize === 0) {
333
+ // Buffer underrun - wait for more data
334
+ this.bufferUnderruns++;
335
+ console.log(`[AudioPlayer] Buffer underrun #${this.bufferUnderruns}, waiting for data...`);
336
+ // Wait a bit and try again
337
+ this.nextFrameTime = process_1.hrtime.bigint() + BigInt(50_000_000); // 50ms
338
+ this.scheduleNextFrame();
339
+ }
340
+ }
341
+ cleanup() {
342
+ // Stop playback loop
343
+ this.isPlaybackLoopRunning = false;
344
+ if (this.playbackTimeout) {
345
+ clearTimeout(this.playbackTimeout);
346
+ this.playbackTimeout = null;
347
+ }
348
+ // Kill FFmpeg
349
+ if (this.ffmpegProcess) {
350
+ this.ffmpegProcess.kill('SIGKILL');
351
+ this.ffmpegProcess = null;
352
+ }
353
+ // Clear frame queue
354
+ this.frameQueue = [];
355
+ this.leftoverBuffer = null;
356
+ // Reset timing and state
357
+ this.nextFrameTime = BigInt(0);
358
+ this.ffmpegDone = false;
359
+ // Log stats
360
+ if (this.framesPlayed > 0) {
361
+ console.log(`[AudioPlayer] Playback stats: ${this.framesPlayed} frames, ${this.bufferUnderruns} underruns`);
362
+ }
363
+ this.bufferUnderruns = 0;
364
+ this.framesPlayed = 0;
365
+ // Note: We don't unpublish the track - it stays published for next play
366
+ }
367
+ setState(newState) {
368
+ const oldState = this.state;
369
+ this.state = newState;
370
+ this.emit('stateChange', oldState, newState);
371
+ // Start playback loop when transitioning to Playing
372
+ if (newState.status === enums_1.AudioPlayerStatus.Playing && oldState.status !== enums_1.AudioPlayerStatus.Playing) {
373
+ console.log(`[AudioPlayer] State changed to Playing, starting playback loop`);
374
+ this.scheduleNextFrame();
375
+ }
376
+ }
377
+ }
378
+ exports.AudioPlayer = AudioPlayer;
379
+ /**
380
+ * Create an audio player
381
+ */
382
+ function createAudioPlayer(options) {
383
+ return new AudioPlayer(options);
384
+ }
385
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AudioPlayer.js","sourceRoot":"","sources":["../src/AudioPlayer.ts"],"names":[],"mappings":"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","sourcesContent":["import { EventEmitter } from 'events';\r\nimport { spawn, ChildProcess } from 'child_process';\r\nimport { hrtime } from 'process';\r\nimport { \r\n  Room, \r\n  LocalAudioTrack, \r\n  AudioSource, \r\n  TrackPublishOptions, \r\n  TrackSource,\r\n  AudioFrame \r\n} from '@livekit/rtc-node';\r\nimport { AudioPlayerStatus } from './enums';\r\nimport { CreateAudioPlayerOptions, AudioPlayerState } from './types';\r\nimport { AudioResource } from './AudioResource';\r\nimport { VoiceConnection } from './VoiceConnection';\r\n\r\n// Audio settings for LiveKit (48kHz stereo)\r\nconst SAMPLE_RATE = 48000;\r\nconst CHANNELS = 2;\r\nconst FRAME_DURATION_MS = 20;\r\nconst SAMPLES_PER_FRAME = (SAMPLE_RATE * FRAME_DURATION_MS) / 1000; // 960\r\n\r\n// Jitter buffer settings\r\nconst FRAME_INTERVAL_NS = BigInt(20_000_000); // 20ms in nanoseconds\r\nconst TARGET_BUFFER_FRAMES = 150; // ~3 seconds - target buffer size\r\nconst MIN_BUFFER_FRAMES = 75;    // ~1.5 seconds - minimum before we start\r\nconst MAX_BUFFER_FRAMES = 500;   // ~10 seconds - max buffer to prevent memory issues\r\nconst LOW_BUFFER_THRESHOLD = 50; // ~1 second - when to slow down playback\r\n\r\n/**\r\n * Audio player for playing audio resources\r\n */\r\nexport class AudioPlayer extends EventEmitter {\r\n  /** Current player state */\r\n  public state: AudioPlayerState = { status: AudioPlayerStatus.Idle };\r\n  \r\n  /** Player options */\r\n  private options: CreateAudioPlayerOptions;\r\n  \r\n  /** Subscribed voice connections */\r\n  private subscriptions: Set<VoiceConnection> = new Set();\r\n  \r\n  /** Current audio resource */\r\n  private currentResource: AudioResource | null = null;\r\n  \r\n  /** FFmpeg process */\r\n  private ffmpegProcess: ChildProcess | null = null;\r\n  \r\n  /** LiveKit audio source and track */\r\n  private audioSource: AudioSource | null = null;\r\n  private audioTrack: LocalAudioTrack | null = null;\r\n  \r\n  /** Frame queue and playback state */\r\n  private frameQueue: Int16Array[] = [];\r\n  private playbackTimeout: NodeJS.Timeout | null = null;\r\n  private leftoverBuffer: Buffer | null = null;\r\n  private isPublished = false;\r\n  \r\n  /** High-resolution timing */\r\n  private nextFrameTime: bigint = BigInt(0);\r\n  private isPlaybackLoopRunning = false;\r\n  private ffmpegDone = false;\r\n  \r\n  /** Buffer statistics */\r\n  private bufferUnderruns = 0;\r\n  private framesPlayed = 0;\r\n\r\n  constructor(options: CreateAudioPlayerOptions = {}) {\r\n    super();\r\n    this.options = {\r\n      behaviors: {\r\n        noSubscriber: 'pause',\r\n        maxMissedFrames: 5,\r\n        ...options.behaviors\r\n      }\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Play an audio resource\r\n   */\r\n  play(resource: AudioResource): void {\r\n    // Stop current playback\r\n    this.stop();\r\n    \r\n    this.currentResource = resource;\r\n    this.setState({ status: AudioPlayerStatus.Buffering, resource });\r\n    \r\n    // Start playback if we have a ready connection\r\n    for (const connection of this.subscriptions) {\r\n      if (connection.getRoom()) {\r\n        this.startPlayback(connection);\r\n        break;\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Pause playback\r\n   */\r\n  pause(): boolean {\r\n    if (this.state.status !== AudioPlayerStatus.Playing) {\r\n      return false;\r\n    }\r\n    this.setState({ status: AudioPlayerStatus.Paused, resource: this.currentResource });\r\n    return true;\r\n  }\r\n\r\n  /**\r\n   * Unpause playback\r\n   */\r\n  unpause(): boolean {\r\n    if (this.state.status !== AudioPlayerStatus.Paused) {\r\n      return false;\r\n    }\r\n    this.setState({ status: AudioPlayerStatus.Playing, resource: this.currentResource });\r\n    return true;\r\n  }\r\n\r\n  /**\r\n   * Stop playback\r\n   */\r\n  stop(force = false): boolean {\r\n    if (this.state.status === AudioPlayerStatus.Idle && !force) {\r\n      return false;\r\n    }\r\n    this.cleanup();\r\n    this.currentResource = null;\r\n    this.setState({ status: AudioPlayerStatus.Idle });\r\n    return true;\r\n  }\r\n\r\n  /**\r\n   * Subscribe a voice connection to this player\r\n   * @internal\r\n   */\r\n  subscribe(connection: VoiceConnection): void {\r\n    this.subscriptions.add(connection);\r\n  }\r\n\r\n  /**\r\n   * Unsubscribe a voice connection from this player\r\n   * @internal\r\n   */\r\n  unsubscribe(connection: VoiceConnection): void {\r\n    this.subscriptions.delete(connection);\r\n    \r\n    // Auto-pause if no subscribers\r\n    if (this.subscriptions.size === 0 && this.options.behaviors?.noSubscriber === 'pause') {\r\n      if (this.state.status === AudioPlayerStatus.Playing) {\r\n        this.setState({ status: AudioPlayerStatus.AutoPaused, resource: this.currentResource });\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Called when a connection becomes ready\r\n   * @internal\r\n   */\r\n  onConnectionReady(connection: VoiceConnection): void {\r\n    // If we have a resource waiting, start playback\r\n    if (this.currentResource && this.state.status === AudioPlayerStatus.Buffering) {\r\n      this.startPlayback(connection);\r\n    }\r\n  }\r\n\r\n  private async startPlayback(connection: VoiceConnection): Promise<void> {\r\n    const room = connection.getRoom();\r\n    if (!room || !this.currentResource) return;\r\n\r\n    try {\r\n      // Create audio source and track\r\n      await this.setupAudioTrack(room);\r\n      \r\n      // Start FFmpeg to decode audio - this will set state to Playing when ready\r\n      await this.startFFmpeg();\r\n    } catch (error) {\r\n      this.emit('error', { message: (error as Error).message, resource: this.currentResource });\r\n      this.stop();\r\n    }\r\n  }\r\n\r\n  private async setupAudioTrack(room: Room): Promise<void> {\r\n    if (this.isPublished) return;\r\n    \r\n    this.audioSource = new AudioSource(SAMPLE_RATE, CHANNELS);\r\n    this.audioTrack = LocalAudioTrack.createAudioTrack('music', this.audioSource);\r\n    \r\n    const options = new TrackPublishOptions();\r\n    options.source = TrackSource.SOURCE_MICROPHONE;\r\n    \r\n    if (room.localParticipant) {\r\n      await room.localParticipant.publishTrack(this.audioTrack, options);\r\n    }\r\n    this.isPublished = true;\r\n  }\r\n\r\n  private async startFFmpeg(): Promise<void> {\r\n    if (!this.currentResource) return;\r\n    \r\n    const inputSource = this.currentResource.getInputSource();\r\n    console.log(`FFmpeg input source: ${inputSource.substring(0, 100)}...`);\r\n    \r\n    // Check if this is a streaming URL that needs yt-dlp\r\n    const needsYtDlp = inputSource.includes('youtube.com') || \r\n                       inputSource.includes('youtu.be') ||\r\n                       inputSource.includes('soundcloud.com') ||\r\n                       inputSource.includes('twitch.tv') ||\r\n                       inputSource.startsWith('ytsearch:');\r\n    \r\n    if (needsYtDlp) {\r\n      // Use yt-dlp to pipe audio directly to FFmpeg\r\n      console.log('Using yt-dlp pipe mode');\r\n      this.ffmpegProcess = spawn('bash', [\r\n        '-c',\r\n        `~/.local/bin/yt-dlp -f \"bestaudio/best\" -o - --no-playlist --no-warnings \"${inputSource}\" | ffmpeg -i pipe:0 -f s16le -ar ${SAMPLE_RATE} -ac ${CHANNELS} -acodec pcm_s16le -`\r\n      ], { stdio: ['pipe', 'pipe', 'pipe'] });\r\n    } else {\r\n      console.log('Using direct FFmpeg mode');\r\n      this.ffmpegProcess = spawn('ffmpeg', [\r\n        '-reconnect', '1',\r\n        '-reconnect_streamed', '1',\r\n        '-reconnect_delay_max', '5',\r\n        '-i', inputSource,\r\n        '-f', 's16le',\r\n        '-ar', String(SAMPLE_RATE),\r\n        '-ac', String(CHANNELS),\r\n        '-acodec', 'pcm_s16le',\r\n        '-'\r\n      ], { stdio: ['pipe', 'pipe', 'pipe'] });\r\n    }\r\n\r\n    const frameSize = SAMPLES_PER_FRAME * CHANNELS * 2;\r\n    this.ffmpegDone = false;\r\n    let hasReceivedData = false;\r\n\r\n    this.ffmpegProcess.stdout?.on('data', (chunk: Buffer) => {\r\n      if (this.state.status !== AudioPlayerStatus.Playing && \r\n          this.state.status !== AudioPlayerStatus.Buffering) return;\r\n      \r\n      hasReceivedData = true;\r\n      \r\n      // Handle leftover from previous chunk\r\n      if (this.leftoverBuffer && this.leftoverBuffer.length > 0) {\r\n        chunk = Buffer.concat([this.leftoverBuffer, chunk]);\r\n        this.leftoverBuffer = null;\r\n      }\r\n      \r\n      let offset = 0;\r\n      while (offset + frameSize <= chunk.length) {\r\n        const frame = chunk.slice(offset, offset + frameSize);\r\n        const int16Array = new Int16Array(SAMPLES_PER_FRAME * CHANNELS);\r\n        \r\n        for (let i = 0; i < int16Array.length; i++) {\r\n          int16Array[i] = frame.readInt16LE(i * 2);\r\n        }\r\n        \r\n        this.frameQueue.push(int16Array);\r\n        offset += frameSize;\r\n      }\r\n      \r\n      // Save leftover\r\n      if (offset < chunk.length) {\r\n        this.leftoverBuffer = chunk.slice(offset);\r\n      }\r\n    });\r\n\r\n    let stderrOutput = '';\r\n    this.ffmpegProcess.stderr?.on('data', (data: Buffer) => {\r\n      stderrOutput += data.toString();\r\n    });\r\n\r\n    this.ffmpegProcess.on('close', (code) => {\r\n      this.ffmpegDone = true;\r\n      this.ffmpegProcess = null;\r\n      if (code !== 0) {\r\n        console.error(`FFmpeg stderr:\\n${stderrOutput}`);\r\n      }\r\n      console.log(`[AudioPlayer] FFmpeg closed with code ${code}, hasReceivedData: ${hasReceivedData}, queue: ${this.frameQueue.length}`);\r\n    });\r\n\r\n    this.ffmpegProcess.on('error', (err) => {\r\n      console.error('FFmpeg process error:', err.message);\r\n      this.emit('error', { message: err.message, resource: this.currentResource });\r\n    });\r\n\r\n    // Wait for initial buffer with timeout\r\n    const bufferTimeout = 10000; // 10 seconds for initial buffer\r\n    const startTime = Date.now();\r\n    \r\n    while (this.frameQueue.length < MIN_BUFFER_FRAMES && Date.now() - startTime < bufferTimeout) {\r\n      await new Promise(r => setTimeout(r, 100));\r\n      \r\n      // Check if FFmpeg failed early\r\n      if (this.ffmpegDone && this.frameQueue.length === 0) {\r\n        throw new Error('FFmpeg failed to produce audio data');\r\n      }\r\n    }\r\n    \r\n    if (this.frameQueue.length === 0) {\r\n      throw new Error('Timeout waiting for audio data');\r\n    }\r\n    \r\n    console.log(`[AudioPlayer] Starting playback with ${this.frameQueue.length} frames buffered (target: ${TARGET_BUFFER_FRAMES})`);\r\n\r\n    // Mark ready for playback - setState will trigger the loop\r\n    this.isPlaybackLoopRunning = true;\r\n    this.nextFrameTime = hrtime.bigint();\r\n    console.log(`[AudioPlayer] Playback ready, audioSource exists: ${!!this.audioSource}`);\r\n    \r\n    // Set state to playing - this will trigger scheduleNextFrame via setState\r\n    this.setState({ status: AudioPlayerStatus.Playing, resource: this.currentResource });\r\n  }\r\n\r\n  /**\r\n   * High-resolution frame scheduling using hrtime\r\n   * This provides much more accurate timing than setInterval\r\n   */\r\n  private scheduleNextFrame(): void {\r\n    if (!this.isPlaybackLoopRunning || this.state.status !== AudioPlayerStatus.Playing) {\r\n      console.log(`[AudioPlayer] scheduleNextFrame skipped: loopRunning=${this.isPlaybackLoopRunning}, status=${this.state.status}`);\r\n      return;\r\n    }\r\n\r\n    const now = hrtime.bigint();\r\n    const delayNs = this.nextFrameTime - now;\r\n    const delayMs = Number(delayNs) / 1_000_000;\r\n\r\n    if (this.framesPlayed === 0) {\r\n      console.log(`[AudioPlayer] First frame scheduling: delayMs=${delayMs.toFixed(2)}`);\r\n    }\r\n\r\n    // Schedule next frame\r\n    if (delayMs > 1) {\r\n      this.playbackTimeout = setTimeout(() => this.processFrame(), Math.max(1, delayMs - 1));\r\n    } else {\r\n      // We're behind, process immediately\r\n      setImmediate(() => this.processFrame());\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Process and send a single audio frame\r\n   */\r\n  private async processFrame(): Promise<void> {\r\n    if (!this.isPlaybackLoopRunning || this.state.status !== AudioPlayerStatus.Playing) {\r\n      if (this.framesPlayed === 0) {\r\n        console.log(`[AudioPlayer] processFrame skipped: loopRunning=${this.isPlaybackLoopRunning}, status=${this.state.status}`);\r\n      }\r\n      return;\r\n    }\r\n\r\n    // Check buffer status\r\n    const bufferSize = this.frameQueue.length;\r\n    \r\n    if (bufferSize > 0 && this.audioSource) {\r\n      const int16Array = this.frameQueue.shift()!;\r\n      const audioFrame = new AudioFrame(int16Array, SAMPLE_RATE, CHANNELS, SAMPLES_PER_FRAME);\r\n      \r\n      try {\r\n        await this.audioSource.captureFrame(audioFrame);\r\n        this.framesPlayed++;\r\n        \r\n        // Log progress every 500 frames (~10 seconds)\r\n        if (this.framesPlayed % 500 === 0) {\r\n          console.log(`[AudioPlayer] Progress: ${this.framesPlayed} frames played, buffer: ${bufferSize}`);\r\n        }\r\n      } catch (e) {\r\n        console.error(`[AudioPlayer] Frame error:`, (e as Error).message);\r\n      }\r\n      \r\n      // Update timing for next frame\r\n      this.nextFrameTime += FRAME_INTERVAL_NS;\r\n      \r\n      // Adaptive timing: if buffer is low, slow down slightly to let it recover\r\n      if (bufferSize < LOW_BUFFER_THRESHOLD && !this.ffmpegDone) {\r\n        // Add 1ms delay to let buffer recover\r\n        this.nextFrameTime += BigInt(1_000_000);\r\n        this.bufferUnderruns++;\r\n        \r\n        if (this.bufferUnderruns % 50 === 0) {\r\n          console.log(`[AudioPlayer] Buffer low: ${bufferSize} frames, ${this.bufferUnderruns} underruns`);\r\n        }\r\n      }\r\n      \r\n      // Schedule next frame\r\n      this.scheduleNextFrame();\r\n      \r\n    } else if (this.ffmpegDone && bufferSize === 0) {\r\n      // Playback finished\r\n      console.log('[AudioPlayer] Playback finished - queue empty and FFmpeg done');\r\n      this.stop();\r\n    } else if (bufferSize === 0) {\r\n      // Buffer underrun - wait for more data\r\n      this.bufferUnderruns++;\r\n      console.log(`[AudioPlayer] Buffer underrun #${this.bufferUnderruns}, waiting for data...`);\r\n      \r\n      // Wait a bit and try again\r\n      this.nextFrameTime = hrtime.bigint() + BigInt(50_000_000); // 50ms\r\n      this.scheduleNextFrame();\r\n    }\r\n  }\r\n\r\n  private cleanup(): void {\r\n    // Stop playback loop\r\n    this.isPlaybackLoopRunning = false;\r\n    if (this.playbackTimeout) {\r\n      clearTimeout(this.playbackTimeout);\r\n      this.playbackTimeout = null;\r\n    }\r\n    \r\n    // Kill FFmpeg\r\n    if (this.ffmpegProcess) {\r\n      this.ffmpegProcess.kill('SIGKILL');\r\n      this.ffmpegProcess = null;\r\n    }\r\n    \r\n    // Clear frame queue\r\n    this.frameQueue = [];\r\n    this.leftoverBuffer = null;\r\n    \r\n    // Reset timing and state\r\n    this.nextFrameTime = BigInt(0);\r\n    this.ffmpegDone = false;\r\n    \r\n    // Log stats\r\n    if (this.framesPlayed > 0) {\r\n      console.log(`[AudioPlayer] Playback stats: ${this.framesPlayed} frames, ${this.bufferUnderruns} underruns`);\r\n    }\r\n    this.bufferUnderruns = 0;\r\n    this.framesPlayed = 0;\r\n    \r\n    // Note: We don't unpublish the track - it stays published for next play\r\n  }\r\n\r\n  private setState(newState: AudioPlayerState): void {\r\n    const oldState = this.state;\r\n    this.state = newState;\r\n    this.emit('stateChange', oldState, newState);\r\n    \r\n    // Start playback loop when transitioning to Playing\r\n    if (newState.status === AudioPlayerStatus.Playing && oldState.status !== AudioPlayerStatus.Playing) {\r\n      console.log(`[AudioPlayer] State changed to Playing, starting playback loop`);\r\n      this.scheduleNextFrame();\r\n    }\r\n  }\r\n}\r\n\r\n/**\r\n * Create an audio player\r\n */\r\nexport function createAudioPlayer(options?: CreateAudioPlayerOptions): AudioPlayer {\r\n  return new AudioPlayer(options);\r\n}\r\n"]}
@@ -0,0 +1,60 @@
1
+ import { CreateAudioResourceOptions, AudioResourceInput } from './types';
2
+ /**
3
+ * Represents an audio resource that can be played
4
+ */
5
+ export declare class AudioResource<T = unknown> {
6
+ /** Metadata attached to this resource */
7
+ readonly metadata: T;
8
+ /** Whether playback has started */
9
+ started: boolean;
10
+ /** Whether playback has ended */
11
+ ended: boolean;
12
+ /** The input source (URL or file path) */
13
+ private inputSource;
14
+ /** Stream type */
15
+ private streamType;
16
+ /** Volume (0-1) */
17
+ private volume;
18
+ constructor(input: AudioResourceInput, options?: CreateAudioResourceOptions<T>);
19
+ /**
20
+ * Get the input source for FFmpeg
21
+ * @internal
22
+ */
23
+ getInputSource(): string;
24
+ /**
25
+ * Set the volume (0-1)
26
+ */
27
+ setVolume(volume: number): void;
28
+ /**
29
+ * Get the current volume
30
+ */
31
+ getVolume(): number;
32
+ }
33
+ /**
34
+ * Options for creating audio resource from URL
35
+ */
36
+ export interface CreateAudioResourceFromUrlOptions<T = unknown> extends CreateAudioResourceOptions<T> {
37
+ /** Use yt-dlp to extract audio URL */
38
+ useYtDlp?: boolean;
39
+ /** Path to yt-dlp binary */
40
+ ytDlpPath?: string;
41
+ }
42
+ /**
43
+ * Create an audio resource from various inputs
44
+ */
45
+ export declare function createAudioResource<T = unknown>(input: AudioResourceInput, options?: CreateAudioResourceOptions<T>): AudioResource<T>;
46
+ /**
47
+ * Create an audio resource from a YouTube/streaming URL
48
+ * Stores the original URL - extraction happens at playback time
49
+ */
50
+ export declare function createAudioResourceFromUrl<T = unknown>(url: string, options?: CreateAudioResourceFromUrlOptions<T>): AudioResource<T>;
51
+ /**
52
+ * Probe audio info from a URL or search query
53
+ * If input is not a URL, it will search YouTube
54
+ */
55
+ export declare function probeAudioInfo(input: string, ytDlpPath?: string): Promise<{
56
+ title: string;
57
+ duration: number;
58
+ thumbnail?: string;
59
+ url: string;
60
+ }>;