@livekit/agents 1.0.44 → 1.0.46

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.
Files changed (157) hide show
  1. package/dist/ipc/supervised_proc.cjs +1 -1
  2. package/dist/ipc/supervised_proc.cjs.map +1 -1
  3. package/dist/ipc/supervised_proc.js +1 -1
  4. package/dist/ipc/supervised_proc.js.map +1 -1
  5. package/dist/llm/llm.cjs +1 -1
  6. package/dist/llm/llm.cjs.map +1 -1
  7. package/dist/llm/llm.js +1 -1
  8. package/dist/llm/llm.js.map +1 -1
  9. package/dist/log.cjs +13 -9
  10. package/dist/log.cjs.map +1 -1
  11. package/dist/log.d.cts +1 -1
  12. package/dist/log.d.ts +1 -1
  13. package/dist/log.d.ts.map +1 -1
  14. package/dist/log.js +13 -9
  15. package/dist/log.js.map +1 -1
  16. package/dist/stream/index.cjs +3 -0
  17. package/dist/stream/index.cjs.map +1 -1
  18. package/dist/stream/index.d.cts +1 -0
  19. package/dist/stream/index.d.ts +1 -0
  20. package/dist/stream/index.d.ts.map +1 -1
  21. package/dist/stream/index.js +2 -0
  22. package/dist/stream/index.js.map +1 -1
  23. package/dist/stream/multi_input_stream.cjs +139 -0
  24. package/dist/stream/multi_input_stream.cjs.map +1 -0
  25. package/dist/stream/multi_input_stream.d.cts +55 -0
  26. package/dist/stream/multi_input_stream.d.ts +55 -0
  27. package/dist/stream/multi_input_stream.d.ts.map +1 -0
  28. package/dist/stream/multi_input_stream.js +115 -0
  29. package/dist/stream/multi_input_stream.js.map +1 -0
  30. package/dist/stream/multi_input_stream.test.cjs +340 -0
  31. package/dist/stream/multi_input_stream.test.cjs.map +1 -0
  32. package/dist/stream/multi_input_stream.test.js +339 -0
  33. package/dist/stream/multi_input_stream.test.js.map +1 -0
  34. package/dist/stt/stt.cjs +2 -2
  35. package/dist/stt/stt.cjs.map +1 -1
  36. package/dist/stt/stt.js +2 -2
  37. package/dist/stt/stt.js.map +1 -1
  38. package/dist/telemetry/trace_types.cjs +42 -0
  39. package/dist/telemetry/trace_types.cjs.map +1 -1
  40. package/dist/telemetry/trace_types.d.cts +14 -0
  41. package/dist/telemetry/trace_types.d.ts +14 -0
  42. package/dist/telemetry/trace_types.d.ts.map +1 -1
  43. package/dist/telemetry/trace_types.js +28 -0
  44. package/dist/telemetry/trace_types.js.map +1 -1
  45. package/dist/tts/fallback_adapter.cjs +466 -0
  46. package/dist/tts/fallback_adapter.cjs.map +1 -0
  47. package/dist/tts/fallback_adapter.d.cts +110 -0
  48. package/dist/tts/fallback_adapter.d.ts +110 -0
  49. package/dist/tts/fallback_adapter.d.ts.map +1 -0
  50. package/dist/tts/fallback_adapter.js +442 -0
  51. package/dist/tts/fallback_adapter.js.map +1 -0
  52. package/dist/tts/index.cjs +3 -0
  53. package/dist/tts/index.cjs.map +1 -1
  54. package/dist/tts/index.d.cts +1 -0
  55. package/dist/tts/index.d.ts +1 -0
  56. package/dist/tts/index.d.ts.map +1 -1
  57. package/dist/tts/index.js +2 -0
  58. package/dist/tts/index.js.map +1 -1
  59. package/dist/tts/tts.cjs +2 -2
  60. package/dist/tts/tts.cjs.map +1 -1
  61. package/dist/tts/tts.js +2 -2
  62. package/dist/tts/tts.js.map +1 -1
  63. package/dist/utils.cjs +13 -0
  64. package/dist/utils.cjs.map +1 -1
  65. package/dist/utils.d.cts +1 -0
  66. package/dist/utils.d.ts +1 -0
  67. package/dist/utils.d.ts.map +1 -1
  68. package/dist/utils.js +13 -0
  69. package/dist/utils.js.map +1 -1
  70. package/dist/vad.cjs +11 -10
  71. package/dist/vad.cjs.map +1 -1
  72. package/dist/vad.d.cts +5 -3
  73. package/dist/vad.d.ts +5 -3
  74. package/dist/vad.d.ts.map +1 -1
  75. package/dist/vad.js +11 -10
  76. package/dist/vad.js.map +1 -1
  77. package/dist/voice/agent_activity.cjs +35 -10
  78. package/dist/voice/agent_activity.cjs.map +1 -1
  79. package/dist/voice/agent_activity.d.cts +1 -0
  80. package/dist/voice/agent_activity.d.ts +1 -0
  81. package/dist/voice/agent_activity.d.ts.map +1 -1
  82. package/dist/voice/agent_activity.js +35 -10
  83. package/dist/voice/agent_activity.js.map +1 -1
  84. package/dist/voice/agent_session.cjs +19 -7
  85. package/dist/voice/agent_session.cjs.map +1 -1
  86. package/dist/voice/agent_session.d.cts +3 -2
  87. package/dist/voice/agent_session.d.ts +3 -2
  88. package/dist/voice/agent_session.d.ts.map +1 -1
  89. package/dist/voice/agent_session.js +19 -7
  90. package/dist/voice/agent_session.js.map +1 -1
  91. package/dist/voice/audio_recognition.cjs +85 -36
  92. package/dist/voice/audio_recognition.cjs.map +1 -1
  93. package/dist/voice/audio_recognition.d.cts +22 -1
  94. package/dist/voice/audio_recognition.d.ts +22 -1
  95. package/dist/voice/audio_recognition.d.ts.map +1 -1
  96. package/dist/voice/audio_recognition.js +89 -36
  97. package/dist/voice/audio_recognition.js.map +1 -1
  98. package/dist/voice/audio_recognition_span.test.cjs +233 -0
  99. package/dist/voice/audio_recognition_span.test.cjs.map +1 -0
  100. package/dist/voice/audio_recognition_span.test.js +232 -0
  101. package/dist/voice/audio_recognition_span.test.js.map +1 -0
  102. package/dist/voice/io.cjs +6 -3
  103. package/dist/voice/io.cjs.map +1 -1
  104. package/dist/voice/io.d.cts +3 -2
  105. package/dist/voice/io.d.ts +3 -2
  106. package/dist/voice/io.d.ts.map +1 -1
  107. package/dist/voice/io.js +6 -3
  108. package/dist/voice/io.js.map +1 -1
  109. package/dist/voice/recorder_io/recorder_io.cjs +3 -1
  110. package/dist/voice/recorder_io/recorder_io.cjs.map +1 -1
  111. package/dist/voice/recorder_io/recorder_io.d.ts.map +1 -1
  112. package/dist/voice/recorder_io/recorder_io.js +3 -1
  113. package/dist/voice/recorder_io/recorder_io.js.map +1 -1
  114. package/dist/voice/room_io/_input.cjs +23 -20
  115. package/dist/voice/room_io/_input.cjs.map +1 -1
  116. package/dist/voice/room_io/_input.d.cts +2 -2
  117. package/dist/voice/room_io/_input.d.ts +2 -2
  118. package/dist/voice/room_io/_input.d.ts.map +1 -1
  119. package/dist/voice/room_io/_input.js +13 -9
  120. package/dist/voice/room_io/_input.js.map +1 -1
  121. package/dist/voice/room_io/room_io.cjs +9 -0
  122. package/dist/voice/room_io/room_io.cjs.map +1 -1
  123. package/dist/voice/room_io/room_io.d.cts +3 -1
  124. package/dist/voice/room_io/room_io.d.ts +3 -1
  125. package/dist/voice/room_io/room_io.d.ts.map +1 -1
  126. package/dist/voice/room_io/room_io.js +9 -0
  127. package/dist/voice/room_io/room_io.js.map +1 -1
  128. package/dist/voice/utils.cjs +47 -0
  129. package/dist/voice/utils.cjs.map +1 -0
  130. package/dist/voice/utils.d.cts +4 -0
  131. package/dist/voice/utils.d.ts +4 -0
  132. package/dist/voice/utils.d.ts.map +1 -0
  133. package/dist/voice/utils.js +23 -0
  134. package/dist/voice/utils.js.map +1 -0
  135. package/package.json +1 -1
  136. package/src/ipc/supervised_proc.ts +1 -1
  137. package/src/llm/llm.ts +1 -1
  138. package/src/log.ts +22 -11
  139. package/src/stream/index.ts +1 -0
  140. package/src/stream/multi_input_stream.test.ts +540 -0
  141. package/src/stream/multi_input_stream.ts +172 -0
  142. package/src/stt/stt.ts +2 -2
  143. package/src/telemetry/trace_types.ts +18 -0
  144. package/src/tts/fallback_adapter.ts +579 -0
  145. package/src/tts/index.ts +1 -0
  146. package/src/tts/tts.ts +2 -2
  147. package/src/utils.ts +16 -0
  148. package/src/vad.ts +12 -11
  149. package/src/voice/agent_activity.ts +25 -0
  150. package/src/voice/agent_session.ts +17 -11
  151. package/src/voice/audio_recognition.ts +114 -38
  152. package/src/voice/audio_recognition_span.test.ts +261 -0
  153. package/src/voice/io.ts +7 -4
  154. package/src/voice/recorder_io/recorder_io.ts +2 -1
  155. package/src/voice/room_io/_input.ts +16 -10
  156. package/src/voice/room_io/room_io.ts +12 -0
  157. package/src/voice/utils.ts +29 -0
@@ -0,0 +1,466 @@
1
+ "use strict";
2
+ var __defProp = Object.defineProperty;
3
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
+ var __getOwnPropNames = Object.getOwnPropertyNames;
5
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
6
+ var __export = (target, all) => {
7
+ for (var name in all)
8
+ __defProp(target, name, { get: all[name], enumerable: true });
9
+ };
10
+ var __copyProps = (to, from, except, desc) => {
11
+ if (from && typeof from === "object" || typeof from === "function") {
12
+ for (let key of __getOwnPropNames(from))
13
+ if (!__hasOwnProp.call(to, key) && key !== except)
14
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
15
+ }
16
+ return to;
17
+ };
18
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
19
+ var fallback_adapter_exports = {};
20
+ __export(fallback_adapter_exports, {
21
+ FallbackAdapter: () => FallbackAdapter
22
+ });
23
+ module.exports = __toCommonJS(fallback_adapter_exports);
24
+ var import_rtc_node = require("@livekit/rtc-node");
25
+ var import_exceptions = require("../_exceptions.cjs");
26
+ var import_log = require("../log.cjs");
27
+ var import_tokenize = require("../tokenize/index.cjs");
28
+ var import_types = require("../types.cjs");
29
+ var import_utils = require("../utils.cjs");
30
+ var import_stream_adapter = require("./stream_adapter.cjs");
31
+ var import_tts = require("./tts.cjs");
32
+ const DEFAULT_FALLBACK_API_CONNECT_OPTIONS = {
33
+ maxRetry: 0,
34
+ timeoutMs: import_types.DEFAULT_API_CONNECT_OPTIONS.timeoutMs,
35
+ retryIntervalMs: import_types.DEFAULT_API_CONNECT_OPTIONS.retryIntervalMs
36
+ };
37
+ const FORWARD_POLL_MS = 10;
38
+ class FallbackAdapter extends import_tts.TTS {
39
+ /** The list of TTS instances used for fallback (in priority order). */
40
+ ttsInstances;
41
+ /** Number of retries per TTS instance before falling back to the next one. */
42
+ maxRetryPerTTS;
43
+ /** Delay in milliseconds before attempting to recover a failed TTS instance. */
44
+ recoveryDelayMs;
45
+ _status = [];
46
+ _logger = (0, import_log.log)();
47
+ _recoveryTimeouts = /* @__PURE__ */ new Map();
48
+ label = `tts.FallbackAdapter`;
49
+ constructor(opts) {
50
+ if (!opts.ttsInstances || opts.ttsInstances.length < 1) {
51
+ throw new Error("at least one TTS instance must be provided.");
52
+ }
53
+ const numChannels = opts.ttsInstances[0].numChannels;
54
+ const allNumChannelsMatch = opts.ttsInstances.every((tts) => tts.numChannels === numChannels);
55
+ if (!allNumChannelsMatch) {
56
+ throw new Error("All TTS instances should have the same number of channels");
57
+ }
58
+ const sampleRate = Math.max(...opts.ttsInstances.map((t) => t.sampleRate));
59
+ const capabilities = FallbackAdapter.aggregateCapabilities(opts.ttsInstances);
60
+ super(sampleRate, numChannels, capabilities);
61
+ this.ttsInstances = opts.ttsInstances;
62
+ this.maxRetryPerTTS = opts.maxRetryPerTTS ?? 2;
63
+ this.recoveryDelayMs = opts.recoveryDelayMs ?? 1e3;
64
+ this._status = opts.ttsInstances.map(() => ({
65
+ available: true,
66
+ recoveringTask: null
67
+ }));
68
+ this.setupEventForwarding();
69
+ }
70
+ static aggregateCapabilities(instances) {
71
+ const streaming = instances.some((tts) => tts.capabilities.streaming);
72
+ const alignedTranscript = instances.every((tts) => tts.capabilities.alignedTranscript === true);
73
+ return { streaming, alignedTranscript };
74
+ }
75
+ setupEventForwarding() {
76
+ this.ttsInstances.forEach((tts) => {
77
+ tts.on("metrics_collected", (metrics) => {
78
+ this.emit("metrics_collected", metrics);
79
+ });
80
+ tts.on("error", (error) => {
81
+ this.emit("error", error);
82
+ });
83
+ });
84
+ }
85
+ /**
86
+ * Returns the current status of all TTS instances, including availability and recovery state.
87
+ */
88
+ get status() {
89
+ return this._status;
90
+ }
91
+ getStreamingInstance(index) {
92
+ const tts = this.ttsInstances[index];
93
+ if (tts.capabilities.streaming) {
94
+ return tts;
95
+ }
96
+ return new import_stream_adapter.StreamAdapter(tts, new import_tokenize.basic.SentenceTokenizer());
97
+ }
98
+ /**
99
+ * Creates a new AudioResampler for the given TTS index if needed.
100
+ * Returns null if the TTS sample rate matches the adapter's output rate.
101
+ * Each stream should create its own resampler to avoid concurrency issues.
102
+ * @internal
103
+ */
104
+ createResamplerForTTS(index) {
105
+ const tts = this.ttsInstances[index];
106
+ if (this.sampleRate !== tts.sampleRate) {
107
+ this._logger.debug(
108
+ `resampling ${tts.label} from ${tts.sampleRate}Hz to ${this.sampleRate}Hz`
109
+ );
110
+ return new import_rtc_node.AudioResampler(tts.sampleRate, this.sampleRate, tts.numChannels);
111
+ }
112
+ return null;
113
+ }
114
+ emitAvailabilityChanged(tts, available) {
115
+ const event = { tts, available };
116
+ this.emit(
117
+ "tts_availability_changed",
118
+ event
119
+ );
120
+ }
121
+ tryRecovery(index) {
122
+ const status = this._status[index];
123
+ const tts = this.ttsInstances[index];
124
+ if (status.recoveringTask && !status.recoveringTask.done) {
125
+ return;
126
+ }
127
+ status.recoveringTask = import_utils.Task.from(async (controller) => {
128
+ try {
129
+ const testStream = tts.synthesize(
130
+ "Hello world, this is a recovery test.",
131
+ {
132
+ maxRetry: 0,
133
+ timeoutMs: 1e4,
134
+ retryIntervalMs: 1e3
135
+ },
136
+ controller.signal
137
+ );
138
+ let audioReceived = false;
139
+ for await (const _ of testStream) {
140
+ audioReceived = true;
141
+ }
142
+ if (!audioReceived) {
143
+ throw new Error("Recovery test completed but no audio was received");
144
+ }
145
+ status.available = true;
146
+ status.recoveringTask = null;
147
+ this._logger.info({ tts: tts.label }, "TTS recovered");
148
+ this.emitAvailabilityChanged(tts, true);
149
+ } catch (error) {
150
+ status.recoveringTask = null;
151
+ if (controller.signal.aborted) {
152
+ return;
153
+ }
154
+ this._logger.debug({ tts: tts.label, error }, "TTS recovery failed, will retry");
155
+ const timeoutId = setTimeout(() => {
156
+ this._recoveryTimeouts.delete(index);
157
+ this.tryRecovery(index);
158
+ }, this.recoveryDelayMs);
159
+ this._recoveryTimeouts.set(index, timeoutId);
160
+ }
161
+ });
162
+ }
163
+ markUnAvailable(index) {
164
+ const status = this._status[index];
165
+ if (status.recoveringTask && !status.recoveringTask.done) {
166
+ return;
167
+ }
168
+ if (status.available) {
169
+ status.available = false;
170
+ this.emitAvailabilityChanged(this.ttsInstances[index], false);
171
+ }
172
+ this.tryRecovery(index);
173
+ }
174
+ /**
175
+ * Receives text and returns synthesis in the form of a {@link ChunkedStream}
176
+ */
177
+ synthesize(text, connOptions, abortSignal) {
178
+ return new FallbackChunkedStream(
179
+ this,
180
+ text,
181
+ connOptions ?? DEFAULT_FALLBACK_API_CONNECT_OPTIONS,
182
+ abortSignal
183
+ );
184
+ }
185
+ /**
186
+ * Returns a {@link SynthesizeStream} that can be used to push text and receive audio data
187
+ *
188
+ * @param options - Optional configuration including connection options
189
+ */
190
+ stream(options) {
191
+ return new FallbackSynthesizeStream(
192
+ this,
193
+ (options == null ? void 0 : options.connOptions) ?? DEFAULT_FALLBACK_API_CONNECT_OPTIONS
194
+ );
195
+ }
196
+ /**
197
+ * Close the FallbackAdapter and all underlying TTS instances.
198
+ * This cancels any ongoing recovery tasks and cleans up resources.
199
+ */
200
+ async close() {
201
+ this._recoveryTimeouts.forEach((timeoutId) => {
202
+ clearTimeout(timeoutId);
203
+ });
204
+ this._recoveryTimeouts.clear();
205
+ const recoveryTasks = this._status.map((s) => s.recoveringTask).filter((t) => t !== null);
206
+ if (recoveryTasks.length > 0) {
207
+ await (0, import_utils.cancelAndWait)(recoveryTasks, 1e3);
208
+ }
209
+ for (const tts of this.ttsInstances) {
210
+ tts.removeAllListeners("metrics_collected");
211
+ tts.removeAllListeners("error");
212
+ }
213
+ await Promise.all(this.ttsInstances.map((tts) => tts.close()));
214
+ }
215
+ }
216
+ class FallbackChunkedStream extends import_tts.ChunkedStream {
217
+ adapter;
218
+ connOptions;
219
+ _logger = (0, import_log.log)();
220
+ label = "tts.FallbackChunkedStream";
221
+ constructor(adapter, text, connOptions, abortSignal) {
222
+ super(text, adapter, connOptions, abortSignal);
223
+ this.adapter = adapter;
224
+ this.connOptions = connOptions;
225
+ }
226
+ async run() {
227
+ const allTTSFailed = this.adapter.status.every((s) => !s.available);
228
+ let lastRequestId = "";
229
+ let lastSegmentId = "";
230
+ if (allTTSFailed) {
231
+ this._logger.warn("All fallback TTS instances failed, retrying from first...");
232
+ }
233
+ for (let i = 0; i < this.adapter.ttsInstances.length; i++) {
234
+ const tts = this.adapter.ttsInstances[i];
235
+ const status = this.adapter.status[i];
236
+ if (!status.available && !allTTSFailed) {
237
+ this.adapter.markUnAvailable(i);
238
+ continue;
239
+ }
240
+ try {
241
+ this._logger.debug({ tts: tts.label }, "attempting TTS synthesis");
242
+ const connOptions = {
243
+ ...this.connOptions,
244
+ maxRetry: this.adapter.maxRetryPerTTS
245
+ };
246
+ const stream = tts.synthesize(this.inputText, connOptions, this.abortSignal);
247
+ let audioReceived = false;
248
+ const resampler = this.adapter.createResamplerForTTS(i);
249
+ for await (const audio of stream) {
250
+ if (this.abortController.signal.aborted) {
251
+ stream.close();
252
+ return;
253
+ }
254
+ if (resampler) {
255
+ for (const frame of resampler.push(audio.frame)) {
256
+ this.queue.put({
257
+ ...audio,
258
+ frame
259
+ });
260
+ audioReceived = true;
261
+ }
262
+ } else {
263
+ this.queue.put(audio);
264
+ audioReceived = true;
265
+ }
266
+ lastRequestId = audio.requestId;
267
+ lastSegmentId = audio.segmentId;
268
+ }
269
+ if (resampler) {
270
+ for (const frame of resampler.flush()) {
271
+ this.queue.put({
272
+ requestId: lastRequestId || "",
273
+ segmentId: lastSegmentId || "",
274
+ frame,
275
+ final: true
276
+ });
277
+ audioReceived = true;
278
+ }
279
+ }
280
+ if (!audioReceived) {
281
+ throw new import_exceptions.APIConnectionError({
282
+ message: "TTS synthesis completed but no audio was received"
283
+ });
284
+ }
285
+ this._logger.debug({ tts: tts.label }, "TTS synthesis succeeded");
286
+ return;
287
+ } catch (error) {
288
+ if (error instanceof import_exceptions.APIError || error instanceof import_exceptions.APIConnectionError) {
289
+ this._logger.warn({ tts: tts.label, error }, "TTS failed, switching to next instance");
290
+ this.adapter.markUnAvailable(i);
291
+ } else {
292
+ throw error;
293
+ }
294
+ }
295
+ }
296
+ const labels = this.adapter.ttsInstances.map((t) => t.label).join(", ");
297
+ throw new import_exceptions.APIConnectionError({
298
+ message: `all TTS instances failed (${labels})`
299
+ });
300
+ }
301
+ }
302
+ class FallbackSynthesizeStream extends import_tts.SynthesizeStream {
303
+ adapter;
304
+ tokenBuffer = [];
305
+ audioPushed = false;
306
+ _logger = (0, import_log.log)();
307
+ label = "tts.FallbackSynthesizeStream";
308
+ constructor(adapter, connOptions) {
309
+ super(adapter, connOptions);
310
+ this.adapter = adapter;
311
+ }
312
+ async run() {
313
+ const allTTSFailed = this.adapter.status.every((s) => !s.available);
314
+ if (allTTSFailed) {
315
+ this._logger.warn("All fallback TTS instances failed, retrying from first...");
316
+ }
317
+ const readInputLLMStream = (async () => {
318
+ try {
319
+ for await (const input of this.input) {
320
+ if (this.abortController.signal.aborted) break;
321
+ this.tokenBuffer.push(input);
322
+ }
323
+ } catch (error) {
324
+ this._logger.debug({ error }, "Error reading input LLM stream");
325
+ throw error;
326
+ } finally {
327
+ this.tokenBuffer.push(import_tts.SynthesizeStream.END_OF_STREAM);
328
+ }
329
+ })();
330
+ for (let i = 0; i < this.adapter.ttsInstances.length; i++) {
331
+ const tts = this.adapter.getStreamingInstance(i);
332
+ const originalTts = this.adapter.ttsInstances[i];
333
+ const status = this.adapter.status[i];
334
+ let lastRequestId = "";
335
+ let lastSegmentId = "";
336
+ if (!status.available && !allTTSFailed) {
337
+ this.adapter.markUnAvailable(i);
338
+ continue;
339
+ }
340
+ try {
341
+ this._logger.debug({ tts: originalTts.label }, "attempting TTS stream");
342
+ const connOptions = {
343
+ ...this.connOptions,
344
+ maxRetry: this.adapter.maxRetryPerTTS
345
+ };
346
+ const stream = tts.stream({ connOptions });
347
+ const resampler = this.adapter.createResamplerForTTS(i);
348
+ let bufferIndex = 0;
349
+ let streamOutputCompleted = false;
350
+ const forwardBufferToTTS = async () => {
351
+ while (true) {
352
+ while (bufferIndex < this.tokenBuffer.length) {
353
+ const token = this.tokenBuffer[bufferIndex++];
354
+ if (token === import_tts.SynthesizeStream.FLUSH_SENTINEL) {
355
+ stream.flush();
356
+ } else if (token === import_tts.SynthesizeStream.END_OF_STREAM) {
357
+ stream.endInput();
358
+ return;
359
+ } else {
360
+ stream.pushText(token);
361
+ }
362
+ }
363
+ await new Promise((resolve) => setTimeout(resolve, FORWARD_POLL_MS));
364
+ if (this.abortController.signal.aborted || streamOutputCompleted) {
365
+ stream.endInput();
366
+ return;
367
+ }
368
+ }
369
+ };
370
+ const processOutput = async () => {
371
+ try {
372
+ for await (const audio of stream) {
373
+ if (this.abortController.signal.aborted) {
374
+ stream.close();
375
+ return;
376
+ }
377
+ if (audio === import_tts.SynthesizeStream.END_OF_STREAM) {
378
+ continue;
379
+ }
380
+ if (resampler) {
381
+ for (const frame of resampler.push(audio.frame)) {
382
+ this.queue.put({
383
+ ...audio,
384
+ frame
385
+ });
386
+ this.audioPushed = true;
387
+ }
388
+ } else {
389
+ this.queue.put(audio);
390
+ this.audioPushed = true;
391
+ }
392
+ lastRequestId = audio.requestId;
393
+ lastSegmentId = audio.segmentId;
394
+ }
395
+ if (resampler) {
396
+ for (const frame of resampler.flush()) {
397
+ this.queue.put({
398
+ requestId: lastRequestId || "",
399
+ segmentId: lastSegmentId || "",
400
+ frame,
401
+ final: true
402
+ });
403
+ this.audioPushed = true;
404
+ }
405
+ }
406
+ } finally {
407
+ streamOutputCompleted = true;
408
+ }
409
+ };
410
+ const [outputResult, forwardBufferResult] = await Promise.allSettled([
411
+ processOutput(),
412
+ forwardBufferToTTS().catch((err) => {
413
+ stream.close();
414
+ throw err;
415
+ })
416
+ ]);
417
+ if (outputResult.status === "rejected") {
418
+ stream.close();
419
+ throw outputResult.reason;
420
+ }
421
+ if (forwardBufferResult.status === "rejected") {
422
+ stream.close();
423
+ throw forwardBufferResult.reason;
424
+ }
425
+ if (!this.audioPushed) {
426
+ throw new import_exceptions.APIConnectionError({
427
+ message: "TTS stream completed but no audio was received"
428
+ });
429
+ }
430
+ this.queue.put(import_tts.SynthesizeStream.END_OF_STREAM);
431
+ this._logger.debug({ tts: originalTts.label }, "TTS stream succeeded");
432
+ await readInputLLMStream.catch(() => {
433
+ });
434
+ return;
435
+ } catch (error) {
436
+ if (this.audioPushed) {
437
+ this._logger.error(
438
+ { tts: originalTts.label },
439
+ "TTS failed after audio pushed, cannot fallback mid-utterance"
440
+ );
441
+ throw error;
442
+ }
443
+ if (error instanceof import_exceptions.APIError || error instanceof import_exceptions.APIConnectionError) {
444
+ this._logger.warn(
445
+ { tts: originalTts.label, error },
446
+ "TTS failed, switching to next instance"
447
+ );
448
+ this.adapter.markUnAvailable(i);
449
+ } else {
450
+ throw error;
451
+ }
452
+ }
453
+ }
454
+ await readInputLLMStream.catch(() => {
455
+ });
456
+ const labels = this.adapter.ttsInstances.map((t) => t.label).join(", ");
457
+ throw new import_exceptions.APIConnectionError({
458
+ message: `all TTS instances failed (${labels})`
459
+ });
460
+ }
461
+ }
462
+ // Annotate the CommonJS export names for ESM import in node:
463
+ 0 && (module.exports = {
464
+ FallbackAdapter
465
+ });
466
+ //# sourceMappingURL=fallback_adapter.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/tts/fallback_adapter.ts"],"sourcesContent":["// SPDX-FileCopyrightText: 2025 LiveKit, Inc.\n//\n// SPDX-License-Identifier: Apache-2.0\nimport { AudioResampler } from '@livekit/rtc-node';\nimport { APIConnectionError, APIError } from '../_exceptions.js';\nimport { log } from '../log.js';\nimport { basic } from '../tokenize/index.js';\nimport { type APIConnectOptions, DEFAULT_API_CONNECT_OPTIONS } from '../types.js';\nimport { Task, cancelAndWait } from '../utils.js';\nimport { StreamAdapter } from './stream_adapter.js';\nimport { ChunkedStream, SynthesizeStream, TTS, type TTSCapabilities } from './tts.js';\n\n/**\n * Internal status tracking for each TTS instance.\n * @internal\n */\ninterface TTSStatus {\n available: boolean;\n recoveringTask: Task<void> | null;\n}\n\n/**\n * Options for creating a FallbackAdapter.\n */\nexport interface FallbackAdapterOptions {\n /** List of TTS instances to use for fallback (in priority order). At least one is required. */\n ttsInstances: TTS[];\n /** Number of internal retries per TTS instance before moving to the next one. Defaults to 2. */\n maxRetryPerTTS?: number;\n /** Delay in milliseconds before attempting to recover a failed TTS instance. Defaults to 1000. */\n recoveryDelayMs?: number;\n}\n\n/**\n * Event emitted when a TTS instance's availability changes.\n */\nexport interface AvailabilityChangedEvent {\n /** The TTS instance whose availability changed. */\n tts: TTS;\n /** Whether the TTS instance is now available. */\n available: boolean;\n}\n\nconst DEFAULT_FALLBACK_API_CONNECT_OPTIONS: APIConnectOptions = {\n maxRetry: 0,\n timeoutMs: DEFAULT_API_CONNECT_OPTIONS.timeoutMs,\n retryIntervalMs: DEFAULT_API_CONNECT_OPTIONS.retryIntervalMs,\n};\n\nconst FORWARD_POLL_MS = 10;\n\n/**\n * FallbackAdapter is a TTS wrapper that provides automatic failover between multiple TTS providers.\n *\n * When the primary TTS fails, it automatically switches to the next available provider in the list.\n * Failed providers are monitored in the background and restored when they recover.\n *\n * Features:\n * - Automatic failover to backup TTS providers on failure\n * - Background health checks to restore recovered providers\n * - Automatic audio resampling when TTS providers have different sample rates\n * - Support for both streaming and non-streaming TTS providers\n *\n * @example\n * ```typescript\n * import { FallbackAdapter } from '@livekit/agents';\n * import { TTS as OpenAITTS } from '@livekit/agents-plugin-openai';\n * import { TTS as ElevenLabsTTS } from '@livekit/agents-plugin-elevenlabs';\n *\n * const fallbackTTS = new FallbackAdapter({\n * ttsInstances: [\n * new OpenAITTS(), // Primary\n * new ElevenLabsTTS(), // Fallback\n * ],\n * maxRetryPerTTS: 2, // Retry each TTS twice before moving to next\n * recoveryDelayMs: 1000, // Check recovery every 1 second\n * });\n *\n * ```\n */\nexport class FallbackAdapter extends TTS {\n /** The list of TTS instances used for fallback (in priority order). */\n readonly ttsInstances: TTS[];\n /** Number of retries per TTS instance before falling back to the next one. */\n readonly maxRetryPerTTS: number;\n /** Delay in milliseconds before attempting to recover a failed TTS instance. */\n readonly recoveryDelayMs: number;\n\n private _status: TTSStatus[] = [];\n private _logger = log();\n private _recoveryTimeouts: Map<number, NodeJS.Timeout> = new Map();\n\n label: string = `tts.FallbackAdapter`;\n\n constructor(opts: FallbackAdapterOptions) {\n if (!opts.ttsInstances || opts.ttsInstances.length < 1) {\n throw new Error('at least one TTS instance must be provided.');\n }\n const numChannels = opts.ttsInstances[0]!.numChannels;\n const allNumChannelsMatch = opts.ttsInstances.every((tts) => tts.numChannels === numChannels);\n if (!allNumChannelsMatch) {\n throw new Error('All TTS instances should have the same number of channels');\n }\n const sampleRate = Math.max(...opts.ttsInstances.map((t) => t.sampleRate));\n const capabilities = FallbackAdapter.aggregateCapabilities(opts.ttsInstances);\n super(sampleRate, numChannels, capabilities);\n this.ttsInstances = opts.ttsInstances;\n this.maxRetryPerTTS = opts.maxRetryPerTTS ?? 2;\n this.recoveryDelayMs = opts.recoveryDelayMs ?? 1000;\n this._status = opts.ttsInstances.map(() => ({\n available: true,\n recoveringTask: null,\n }));\n this.setupEventForwarding();\n }\n private static aggregateCapabilities(instances: TTS[]): TTSCapabilities {\n const streaming = instances.some((tts) => tts.capabilities.streaming);\n const alignedTranscript = instances.every((tts) => tts.capabilities.alignedTranscript === true);\n return { streaming, alignedTranscript };\n }\n\n private setupEventForwarding(): void {\n this.ttsInstances.forEach((tts) => {\n tts.on('metrics_collected', (metrics) => {\n this.emit('metrics_collected', metrics);\n });\n tts.on('error', (error) => {\n this.emit('error', error);\n });\n });\n }\n\n /**\n * Returns the current status of all TTS instances, including availability and recovery state.\n */\n get status(): TTSStatus[] {\n return this._status;\n }\n\n getStreamingInstance(index: number): TTS {\n const tts = this.ttsInstances[index]!;\n if (tts.capabilities.streaming) {\n return tts;\n }\n // Wrap non-streaming TTS with StreamAdapter\n return new StreamAdapter(tts, new basic.SentenceTokenizer());\n }\n\n /**\n * Creates a new AudioResampler for the given TTS index if needed.\n * Returns null if the TTS sample rate matches the adapter's output rate.\n * Each stream should create its own resampler to avoid concurrency issues.\n * @internal\n */\n createResamplerForTTS(index: number): AudioResampler | null {\n const tts = this.ttsInstances[index]!;\n if (this.sampleRate !== tts.sampleRate) {\n this._logger.debug(\n `resampling ${tts.label} from ${tts.sampleRate}Hz to ${this.sampleRate}Hz`,\n );\n return new AudioResampler(tts.sampleRate, this.sampleRate, tts.numChannels);\n }\n return null;\n }\n\n private emitAvailabilityChanged(tts: TTS, available: boolean): void {\n const event: AvailabilityChangedEvent = { tts, available };\n (this as unknown as { emit: (event: string, data: AvailabilityChangedEvent) => void }).emit(\n 'tts_availability_changed',\n event,\n );\n }\n\n private tryRecovery(index: number): void {\n const status = this._status[index]!;\n const tts = this.ttsInstances[index]!;\n if (status.recoveringTask && !status.recoveringTask.done) {\n return;\n }\n status.recoveringTask = Task.from(async (controller) => {\n try {\n const testStream = tts.synthesize(\n 'Hello world, this is a recovery test.',\n {\n maxRetry: 0,\n timeoutMs: 10000,\n retryIntervalMs: 1000,\n },\n controller.signal,\n );\n let audioReceived = false;\n for await (const _ of testStream) {\n audioReceived = true;\n }\n if (!audioReceived) {\n throw new Error('Recovery test completed but no audio was received');\n }\n\n status.available = true;\n status.recoveringTask = null;\n this._logger.info({ tts: tts.label }, 'TTS recovered');\n this.emitAvailabilityChanged(tts, true);\n } catch (error) {\n status.recoveringTask = null;\n // Don't schedule retry if we're shutting down\n if (controller.signal.aborted) {\n return;\n }\n this._logger.debug({ tts: tts.label, error }, 'TTS recovery failed, will retry');\n // Retry recovery after delay (matches Python's retry behavior)\n const timeoutId = setTimeout(() => {\n this._recoveryTimeouts.delete(index);\n this.tryRecovery(index);\n }, this.recoveryDelayMs);\n this._recoveryTimeouts.set(index, timeoutId);\n }\n });\n }\n\n markUnAvailable(index: number): void {\n const status = this._status[index]!;\n if (status.recoveringTask && !status.recoveringTask.done) {\n return;\n }\n if (status.available) {\n status.available = false;\n this.emitAvailabilityChanged(this.ttsInstances[index]!, false);\n }\n this.tryRecovery(index);\n }\n\n /**\n * Receives text and returns synthesis in the form of a {@link ChunkedStream}\n */\n synthesize(\n text: string,\n connOptions?: APIConnectOptions,\n abortSignal?: AbortSignal,\n ): ChunkedStream {\n return new FallbackChunkedStream(\n this,\n text,\n connOptions ?? DEFAULT_FALLBACK_API_CONNECT_OPTIONS,\n abortSignal,\n );\n }\n\n /**\n * Returns a {@link SynthesizeStream} that can be used to push text and receive audio data\n *\n * @param options - Optional configuration including connection options\n */\n stream(options?: { connOptions?: APIConnectOptions }): SynthesizeStream {\n return new FallbackSynthesizeStream(\n this,\n options?.connOptions ?? DEFAULT_FALLBACK_API_CONNECT_OPTIONS,\n );\n }\n\n /**\n * Close the FallbackAdapter and all underlying TTS instances.\n * This cancels any ongoing recovery tasks and cleans up resources.\n */\n async close(): Promise<void> {\n // clear all recovery timeouts so that it does not cause issue\n this._recoveryTimeouts.forEach((timeoutId) => {\n clearTimeout(timeoutId);\n });\n this._recoveryTimeouts.clear();\n\n // Cancel all recovery tasks\n const recoveryTasks = this._status\n .map((s) => s.recoveringTask)\n .filter((t): t is Task<void> => t !== null);\n\n if (recoveryTasks.length > 0) {\n await cancelAndWait(recoveryTasks, 1000);\n }\n\n // Remove event listeners\n for (const tts of this.ttsInstances) {\n tts.removeAllListeners('metrics_collected');\n tts.removeAllListeners('error');\n }\n\n // Close all TTS instances\n await Promise.all(this.ttsInstances.map((tts) => tts.close()));\n }\n}\n\nclass FallbackChunkedStream extends ChunkedStream {\n private adapter: FallbackAdapter;\n private connOptions: APIConnectOptions;\n private _logger = log();\n\n label: string = 'tts.FallbackChunkedStream';\n\n constructor(\n adapter: FallbackAdapter,\n text: string,\n connOptions: APIConnectOptions,\n abortSignal?: AbortSignal,\n ) {\n super(text, adapter, connOptions, abortSignal);\n this.adapter = adapter;\n this.connOptions = connOptions;\n }\n\n protected async run(): Promise<void> {\n const allTTSFailed = this.adapter.status.every((s) => !s.available);\n let lastRequestId: string = '';\n let lastSegmentId: string = '';\n if (allTTSFailed) {\n this._logger.warn('All fallback TTS instances failed, retrying from first...');\n }\n for (let i = 0; i < this.adapter.ttsInstances.length; i++) {\n const tts = this.adapter.ttsInstances[i]!;\n const status = this.adapter.status[i]!;\n if (!status.available && !allTTSFailed) {\n this.adapter.markUnAvailable(i);\n continue;\n }\n try {\n this._logger.debug({ tts: tts.label }, 'attempting TTS synthesis');\n const connOptions: APIConnectOptions = {\n ...this.connOptions,\n maxRetry: this.adapter.maxRetryPerTTS,\n };\n const stream = tts.synthesize(this.inputText, connOptions, this.abortSignal);\n let audioReceived = false;\n const resampler = this.adapter.createResamplerForTTS(i);\n for await (const audio of stream) {\n if (this.abortController.signal.aborted) {\n stream.close();\n return;\n }\n\n if (resampler) {\n for (const frame of resampler.push(audio.frame)) {\n this.queue.put({\n ...audio,\n frame,\n });\n audioReceived = true;\n }\n } else {\n this.queue.put(audio);\n audioReceived = true;\n }\n lastRequestId = audio.requestId;\n lastSegmentId = audio.segmentId;\n }\n\n // Flush any remaining resampled frames\n if (resampler) {\n for (const frame of resampler.flush()) {\n this.queue.put({\n requestId: lastRequestId || '',\n segmentId: lastSegmentId || '',\n frame,\n final: true,\n });\n audioReceived = true;\n }\n }\n\n // Verify audio was actually received - silent failures should trigger fallback\n if (!audioReceived) {\n throw new APIConnectionError({\n message: 'TTS synthesis completed but no audio was received',\n });\n }\n\n this._logger.debug({ tts: tts.label }, 'TTS synthesis succeeded');\n return;\n } catch (error) {\n if (error instanceof APIError || error instanceof APIConnectionError) {\n this._logger.warn({ tts: tts.label, error }, 'TTS failed, switching to next instance');\n this.adapter.markUnAvailable(i);\n } else {\n throw error;\n }\n }\n }\n const labels = this.adapter.ttsInstances.map((t) => t.label).join(', ');\n throw new APIConnectionError({\n message: `all TTS instances failed (${labels})`,\n });\n }\n}\n\nclass FallbackSynthesizeStream extends SynthesizeStream {\n private adapter: FallbackAdapter;\n private tokenBuffer: (\n | string\n | typeof SynthesizeStream.FLUSH_SENTINEL\n | typeof SynthesizeStream.END_OF_STREAM\n )[] = [];\n private audioPushed = false;\n private _logger = log();\n\n label: string = 'tts.FallbackSynthesizeStream';\n\n constructor(adapter: FallbackAdapter, connOptions: APIConnectOptions) {\n super(adapter, connOptions);\n this.adapter = adapter;\n }\n\n protected async run(): Promise<void> {\n const allTTSFailed = this.adapter.status.every((s) => !s.available);\n if (allTTSFailed) {\n this._logger.warn('All fallback TTS instances failed, retrying from first...');\n }\n const readInputLLMStream = (async () => {\n try {\n for await (const input of this.input) {\n if (this.abortController.signal.aborted) break;\n this.tokenBuffer.push(input);\n }\n } catch (error) {\n this._logger.debug({ error }, 'Error reading input LLM stream');\n throw error;\n } finally {\n this.tokenBuffer.push(SynthesizeStream.END_OF_STREAM);\n }\n })();\n\n for (let i = 0; i < this.adapter.ttsInstances.length; i++) {\n const tts = this.adapter.getStreamingInstance(i);\n const originalTts = this.adapter.ttsInstances[i]!;\n const status = this.adapter.status[i]!;\n let lastRequestId: string = '';\n let lastSegmentId: string = '';\n\n if (!status.available && !allTTSFailed) {\n this.adapter.markUnAvailable(i);\n continue;\n }\n\n try {\n this._logger.debug({ tts: originalTts.label }, 'attempting TTS stream');\n\n const connOptions: APIConnectOptions = {\n ...this.connOptions,\n maxRetry: this.adapter.maxRetryPerTTS,\n };\n\n const stream = tts.stream({ connOptions });\n const resampler = this.adapter.createResamplerForTTS(i);\n let bufferIndex = 0;\n let streamOutputCompleted = false;\n const forwardBufferToTTS = async () => {\n while (true) {\n while (bufferIndex < this.tokenBuffer.length) {\n const token = this.tokenBuffer[bufferIndex++]!;\n if (token === SynthesizeStream.FLUSH_SENTINEL) {\n stream.flush();\n } else if (token === SynthesizeStream.END_OF_STREAM) {\n stream.endInput();\n return;\n } else {\n stream.pushText(token);\n }\n }\n await new Promise((resolve) => setTimeout(resolve, FORWARD_POLL_MS));\n if (this.abortController.signal.aborted || streamOutputCompleted) {\n stream.endInput();\n return;\n }\n }\n };\n\n const processOutput = async () => {\n try {\n for await (const audio of stream) {\n if (this.abortController.signal.aborted) {\n stream.close();\n return;\n }\n\n if (audio === SynthesizeStream.END_OF_STREAM) {\n // Don't forward END_OF_STREAM yet — only emit after we verify audio\n // was received. Otherwise a silent failure would signal completion\n // to consumers before fallback can try the next TTS.\n continue;\n }\n\n if (resampler) {\n for (const frame of resampler.push(audio.frame)) {\n this.queue.put({\n ...audio,\n frame,\n });\n this.audioPushed = true;\n }\n } else {\n this.queue.put(audio);\n this.audioPushed = true;\n }\n lastRequestId = audio.requestId;\n lastSegmentId = audio.segmentId;\n }\n\n // Flush resampler\n if (resampler) {\n for (const frame of resampler.flush()) {\n this.queue.put({\n requestId: lastRequestId || '',\n segmentId: lastSegmentId || '',\n frame,\n final: true,\n });\n this.audioPushed = true;\n }\n }\n } finally {\n // processOutput and forwardBufferToTTS run in parallel.\n // forwardBufferToTTS polls tokenBuffer and only exits when it sees END_OF_STREAM.\n // But END_OF_STREAM is only added when the LLM finishes streaming (line 417).\n // If the TTS fails while the LLM is still streaming, forwardBufferToTTS would\n // keep polling indefinitely, blocking fallback to the next TTS.\n // This flag tells it to exit early.\n streamOutputCompleted = true;\n }\n };\n const [outputResult, forwardBufferResult] = await Promise.allSettled([\n processOutput(),\n forwardBufferToTTS().catch((err) => {\n stream.close(); // Close stream so processOutput can exit\n throw err;\n }),\n ]);\n if (outputResult.status === 'rejected') {\n stream.close();\n throw outputResult.reason;\n }\n if (forwardBufferResult.status === 'rejected') {\n stream.close();\n throw forwardBufferResult.reason;\n }\n\n // Verify audio was actually received - if not, the TTS failed silently\n if (!this.audioPushed) {\n throw new APIConnectionError({\n message: 'TTS stream completed but no audio was received',\n });\n }\n\n this.queue.put(SynthesizeStream.END_OF_STREAM);\n this._logger.debug({ tts: originalTts.label }, 'TTS stream succeeded');\n await readInputLLMStream.catch(() => {});\n return;\n } catch (error) {\n if (this.audioPushed) {\n this._logger.error(\n { tts: originalTts.label },\n 'TTS failed after audio pushed, cannot fallback mid-utterance',\n );\n throw error;\n }\n\n if (error instanceof APIError || error instanceof APIConnectionError) {\n this._logger.warn(\n { tts: originalTts.label, error },\n 'TTS failed, switching to next instance',\n );\n this.adapter.markUnAvailable(i);\n } else {\n throw error;\n }\n }\n }\n await readInputLLMStream.catch(() => {});\n const labels = this.adapter.ttsInstances.map((t) => t.label).join(', ');\n throw new APIConnectionError({\n message: `all TTS instances failed (${labels})`,\n });\n }\n}\n"],"mappings":"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","names":[]}
@@ -0,0 +1,110 @@
1
+ import { AudioResampler } from '@livekit/rtc-node';
2
+ import { type APIConnectOptions } from '../types.js';
3
+ import { Task } from '../utils.js';
4
+ import { ChunkedStream, SynthesizeStream, TTS } from './tts.js';
5
+ /**
6
+ * Internal status tracking for each TTS instance.
7
+ * @internal
8
+ */
9
+ interface TTSStatus {
10
+ available: boolean;
11
+ recoveringTask: Task<void> | null;
12
+ }
13
+ /**
14
+ * Options for creating a FallbackAdapter.
15
+ */
16
+ export interface FallbackAdapterOptions {
17
+ /** List of TTS instances to use for fallback (in priority order). At least one is required. */
18
+ ttsInstances: TTS[];
19
+ /** Number of internal retries per TTS instance before moving to the next one. Defaults to 2. */
20
+ maxRetryPerTTS?: number;
21
+ /** Delay in milliseconds before attempting to recover a failed TTS instance. Defaults to 1000. */
22
+ recoveryDelayMs?: number;
23
+ }
24
+ /**
25
+ * Event emitted when a TTS instance's availability changes.
26
+ */
27
+ export interface AvailabilityChangedEvent {
28
+ /** The TTS instance whose availability changed. */
29
+ tts: TTS;
30
+ /** Whether the TTS instance is now available. */
31
+ available: boolean;
32
+ }
33
+ /**
34
+ * FallbackAdapter is a TTS wrapper that provides automatic failover between multiple TTS providers.
35
+ *
36
+ * When the primary TTS fails, it automatically switches to the next available provider in the list.
37
+ * Failed providers are monitored in the background and restored when they recover.
38
+ *
39
+ * Features:
40
+ * - Automatic failover to backup TTS providers on failure
41
+ * - Background health checks to restore recovered providers
42
+ * - Automatic audio resampling when TTS providers have different sample rates
43
+ * - Support for both streaming and non-streaming TTS providers
44
+ *
45
+ * @example
46
+ * ```typescript
47
+ * import { FallbackAdapter } from '@livekit/agents';
48
+ * import { TTS as OpenAITTS } from '@livekit/agents-plugin-openai';
49
+ * import { TTS as ElevenLabsTTS } from '@livekit/agents-plugin-elevenlabs';
50
+ *
51
+ * const fallbackTTS = new FallbackAdapter({
52
+ * ttsInstances: [
53
+ * new OpenAITTS(), // Primary
54
+ * new ElevenLabsTTS(), // Fallback
55
+ * ],
56
+ * maxRetryPerTTS: 2, // Retry each TTS twice before moving to next
57
+ * recoveryDelayMs: 1000, // Check recovery every 1 second
58
+ * });
59
+ *
60
+ * ```
61
+ */
62
+ export declare class FallbackAdapter extends TTS {
63
+ /** The list of TTS instances used for fallback (in priority order). */
64
+ readonly ttsInstances: TTS[];
65
+ /** Number of retries per TTS instance before falling back to the next one. */
66
+ readonly maxRetryPerTTS: number;
67
+ /** Delay in milliseconds before attempting to recover a failed TTS instance. */
68
+ readonly recoveryDelayMs: number;
69
+ private _status;
70
+ private _logger;
71
+ private _recoveryTimeouts;
72
+ label: string;
73
+ constructor(opts: FallbackAdapterOptions);
74
+ private static aggregateCapabilities;
75
+ private setupEventForwarding;
76
+ /**
77
+ * Returns the current status of all TTS instances, including availability and recovery state.
78
+ */
79
+ get status(): TTSStatus[];
80
+ getStreamingInstance(index: number): TTS;
81
+ /**
82
+ * Creates a new AudioResampler for the given TTS index if needed.
83
+ * Returns null if the TTS sample rate matches the adapter's output rate.
84
+ * Each stream should create its own resampler to avoid concurrency issues.
85
+ * @internal
86
+ */
87
+ createResamplerForTTS(index: number): AudioResampler | null;
88
+ private emitAvailabilityChanged;
89
+ private tryRecovery;
90
+ markUnAvailable(index: number): void;
91
+ /**
92
+ * Receives text and returns synthesis in the form of a {@link ChunkedStream}
93
+ */
94
+ synthesize(text: string, connOptions?: APIConnectOptions, abortSignal?: AbortSignal): ChunkedStream;
95
+ /**
96
+ * Returns a {@link SynthesizeStream} that can be used to push text and receive audio data
97
+ *
98
+ * @param options - Optional configuration including connection options
99
+ */
100
+ stream(options?: {
101
+ connOptions?: APIConnectOptions;
102
+ }): SynthesizeStream;
103
+ /**
104
+ * Close the FallbackAdapter and all underlying TTS instances.
105
+ * This cancels any ongoing recovery tasks and cleans up resources.
106
+ */
107
+ close(): Promise<void>;
108
+ }
109
+ export {};
110
+ //# sourceMappingURL=fallback_adapter.d.ts.map