@jubbio/voice 1.0.6 → 1.0.7

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.
@@ -112,8 +112,11 @@ async function probeAudioInfo(input, ytDlpPath) {
112
112
  }
113
113
  let ytdlp;
114
114
  if (isWindows) {
115
- // Windows: use shell with quoted command
116
- const cmd = `${ytdlpBin} --no-playlist --no-warnings -j "${searchQuery}"`;
115
+ // Windows: escape special characters and use shell
116
+ // Replace double quotes with escaped version for cmd
117
+ const escapedQuery = searchQuery.replace(/"/g, '\\"');
118
+ const cmd = `${ytdlpBin} --no-playlist --no-warnings -j "${escapedQuery}"`;
119
+ console.log('[probeAudioInfo] Running:', cmd);
117
120
  ytdlp = (0, child_process_1.spawn)(cmd, [], { shell: true });
118
121
  }
119
122
  else {
@@ -133,7 +136,11 @@ async function probeAudioInfo(input, ytDlpPath) {
133
136
  });
134
137
  ytdlp.on('close', (code) => {
135
138
  if (code !== 0) {
136
- reject(new Error(`Failed to probe audio info: ${stderr || 'Unknown error'}`));
139
+ reject(new Error(`yt-dlp failed (code ${code}): ${stderr || 'Unknown error'}`));
140
+ return;
141
+ }
142
+ if (!stdout.trim()) {
143
+ reject(new Error(`yt-dlp returned empty response. stderr: ${stderr}`));
137
144
  return;
138
145
  }
139
146
  try {
@@ -146,7 +153,7 @@ async function probeAudioInfo(input, ytDlpPath) {
146
153
  });
147
154
  }
148
155
  catch (e) {
149
- reject(new Error(`Failed to parse audio info: ${e.message}`));
156
+ reject(new Error(`Failed to parse yt-dlp output: ${stdout.substring(0, 200)}`));
150
157
  }
151
158
  });
152
159
  ytdlp.on('error', (err) => {
@@ -159,4 +166,4 @@ async function probeAudioInfo(input, ytDlpPath) {
159
166
  }, 30000);
160
167
  });
161
168
  }
162
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AudioResource.js","sourceRoot":"","sources":["../src/AudioResource.ts"],"names":[],"mappings":";;;AA+EA,kDAKC;AAMD,gEAOC;AAkCD,wCAuEC;AAzMD,iDAAsC;AACtC,mCAAqC;AAGrC;;GAEG;AACH,MAAa,aAAa;IACxB,yCAAyC;IACzB,QAAQ,CAAI;IAE5B,mCAAmC;IAC5B,OAAO,GAAG,KAAK,CAAC;IAEvB,iCAAiC;IAC1B,KAAK,GAAG,KAAK,CAAC;IAErB,0CAA0C;IAClC,WAAW,CAAS;IAE5B,kBAAkB;IACV,UAAU,CAAa;IAE/B,mBAAmB;IACX,MAAM,GAAG,CAAC,CAAC;IAEnB,YACE,KAAyB,EACzB,UAAyC,EAAE;QAE3C,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC,QAAa,CAAC;QACtC,IAAI,CAAC,UAAU,GAAG,OAAO,CAAC,SAAS,IAAI,kBAAU,CAAC,SAAS,CAAC;QAE5D,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;YAC9B,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;QAC3B,CAAC;aAAM,CAAC;YACN,+CAA+C;YAC/C,0BAA0B;YAC1B,MAAM,IAAI,KAAK,CAAC,uDAAuD,CAAC,CAAC;QAC3E,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,cAAc;QACZ,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;IAED;;OAEG;IACH,SAAS,CAAC,MAAc;QACtB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC,CAAC;IACjD,CAAC;IAED;;OAEG;IACH,SAAS;QACP,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;CACF;AAxDD,sCAwDC;AAYD;;GAEG;AACH,SAAgB,mBAAmB,CACjC,KAAyB,EACzB,OAAuC;IAEvC,OAAO,IAAI,aAAa,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;AAC3C,CAAC;AAED;;;GAGG;AACH,SAAgB,0BAA0B,CACxC,GAAW,EACX,UAAgD,EAAE;IAElD,8DAA8D;IAC9D,mDAAmD;IACnD,OAAO,IAAI,aAAa,CAAC,GAAG,EAAE,OAAO,CAAC,CAAC;AACzC,CAAC;AAED;;GAEG;AACH,SAAS,cAAc,CAAC,GAAW;IACjC,MAAM,gBAAgB,GAAG;QACvB,aAAa;QACb,UAAU;QACV,gBAAgB;QAChB,aAAa;QACb,WAAW;QACX,WAAW;KACZ,CAAC;IAEF,OAAO,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,GAAG,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC;AAC/D,CAAC;AAED;;GAEG;AACH,SAAS,UAAU,CAAC,KAAa;IAC/B,IAAI,CAAC;QACH,IAAI,GAAG,CAAC,KAAK,CAAC,CAAC;QACf,OAAO,IAAI,CAAC;IACd,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC,UAAU,CAAC,SAAS,CAAC,IAAI,KAAK,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC;IACrE,CAAC;AACH,CAAC;AAED;;;GAGG;AACI,KAAK,UAAU,cAAc,CAAC,KAAa,EAAE,SAAkB;IAMpE,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QACrC,MAAM,SAAS,GAAG,OAAO,CAAC,QAAQ,KAAK,OAAO,CAAC;QAC/C,MAAM,gBAAgB,GAAG,SAAS,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,qBAAqB,CAAC;QACtE,MAAM,QAAQ,GAAG,SAAS,IAAI,gBAAgB,CAAC;QAE/C,8CAA8C;QAC9C,IAAI,WAAW,GAAG,KAAK,CAAC;QACxB,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC;YACvB,WAAW,GAAG,aAAa,KAAK,EAAE,CAAC;QACrC,CAAC;QAED,IAAI,KAA+B,CAAC;QAEpC,IAAI,SAAS,EAAE,CAAC;YACd,yCAAyC;YACzC,MAAM,GAAG,GAAG,GAAG,QAAQ,oCAAoC,WAAW,GAAG,CAAC;YAC1E,KAAK,GAAG,IAAA,qBAAK,EAAC,GAAG,EAAE,EAAE,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;QAC1C,CAAC;aAAM,CAAC;YACN,uCAAuC;YACvC,KAAK,GAAG,IAAA,qBAAK,EAAC,MAAM,EAAE;gBACpB,IAAI;gBACJ,GAAG,QAAQ,oCAAoC,WAAW,GAAG;aAC9D,CAAC,CAAC;QACL,CAAC;QAED,IAAI,MAAM,GAAG,EAAE,CAAC;QAChB,IAAI,MAAM,GAAG,EAAE,CAAC;QAEhB,KAAK,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE;YAChC,MAAM,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;QAC5B,CAAC,CAAC,CAAC;QAEH,KAAK,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE;YAChC,MAAM,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;QAC5B,CAAC,CAAC,CAAC;QAEH,KAAK,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,IAAI,EAAE,EAAE;YACzB,IAAI,IAAI,KAAK,CAAC,EAAE,CAAC;gBACf,MAAM,CAAC,IAAI,KAAK,CAAC,+BAA+B,MAAM,IAAI,eAAe,EAAE,CAAC,CAAC,CAAC;gBAC9E,OAAO;YACT,CAAC;YAED,IAAI,CAAC;gBACH,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;gBAChC,OAAO,CAAC;oBACN,KAAK,EAAE,IAAI,CAAC,KAAK,IAAI,SAAS;oBAC9B,QAAQ,EAAE,IAAI,CAAC,QAAQ,IAAI,CAAC;oBAC5B,SAAS,EAAE,IAAI,CAAC,SAAS;oBACzB,GAAG,EAAE,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,GAAG,IAAI,KAAK;iBAC3C,CAAC,CAAC;YACL,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,MAAM,CAAC,IAAI,KAAK,CAAC,+BAAgC,CAAW,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;YAC3E,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,KAAK,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;YACxB,MAAM,CAAC,IAAI,KAAK,CAAC,+BAA+B,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;QAClE,CAAC,CAAC,CAAC;QAEH,2BAA2B;QAC3B,UAAU,CAAC,GAAG,EAAE;YACd,KAAK,CAAC,IAAI,EAAE,CAAC;YACb,MAAM,CAAC,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC,CAAC;QACtD,CAAC,EAAE,KAAK,CAAC,CAAC;IACZ,CAAC,CAAC,CAAC;AACL,CAAC","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?: string): 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    const isWindows = process.platform === 'win32';\r\n    const defaultYtDlpPath = isWindows ? 'yt-dlp' : '~/.local/bin/yt-dlp';\r\n    const ytdlpBin = ytDlpPath || defaultYtDlpPath;\r\n    \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    let ytdlp: ReturnType<typeof spawn>;\r\n    \r\n    if (isWindows) {\r\n      // Windows: use shell with quoted command\r\n      const cmd = `${ytdlpBin} --no-playlist --no-warnings -j \"${searchQuery}\"`;\r\n      ytdlp = spawn(cmd, [], { shell: true });\r\n    } else {\r\n      // Unix: use bash -c with quoted string\r\n      ytdlp = spawn('bash', [\r\n        '-c',\r\n        `${ytdlpBin} --no-playlist --no-warnings -j \"${searchQuery}\"`\r\n      ]);\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"]}
169
+ //# 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?: string): 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    const isWindows = process.platform === 'win32';\r\n    const defaultYtDlpPath = isWindows ? 'yt-dlp' : '~/.local/bin/yt-dlp';\r\n    const ytdlpBin = ytDlpPath || defaultYtDlpPath;\r\n    \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    let ytdlp: ReturnType<typeof spawn>;\r\n    \r\n    if (isWindows) {\r\n      // Windows: escape special characters and use shell\r\n      // Replace double quotes with escaped version for cmd\r\n      const escapedQuery = searchQuery.replace(/\"/g, '\\\\\"');\r\n      const cmd = `${ytdlpBin} --no-playlist --no-warnings -j \"${escapedQuery}\"`;\r\n      console.log('[probeAudioInfo] Running:', cmd);\r\n      ytdlp = spawn(cmd, [], { shell: true });\r\n    } else {\r\n      // Unix: use bash -c with quoted string\r\n      ytdlp = spawn('bash', [\r\n        '-c',\r\n        `${ytdlpBin} --no-playlist --no-warnings -j \"${searchQuery}\"`\r\n      ]);\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(`yt-dlp failed (code ${code}): ${stderr || 'Unknown error'}`));\r\n        return;\r\n      }\r\n      \r\n      if (!stdout.trim()) {\r\n        reject(new Error(`yt-dlp returned empty response. stderr: ${stderr}`));\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 yt-dlp output: ${stdout.substring(0, 200)}`));\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"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@jubbio/voice",
3
- "version": "1.0.6",
3
+ "version": "1.0.7",
4
4
  "description": "Voice library for Jubbio bots with LiveKit backend",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",