@livekit/agents 1.0.45 → 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.
- package/dist/stream/index.cjs +3 -0
- package/dist/stream/index.cjs.map +1 -1
- package/dist/stream/index.d.cts +1 -0
- package/dist/stream/index.d.ts +1 -0
- package/dist/stream/index.d.ts.map +1 -1
- package/dist/stream/index.js +2 -0
- package/dist/stream/index.js.map +1 -1
- package/dist/stream/multi_input_stream.cjs +139 -0
- package/dist/stream/multi_input_stream.cjs.map +1 -0
- package/dist/stream/multi_input_stream.d.cts +55 -0
- package/dist/stream/multi_input_stream.d.ts +55 -0
- package/dist/stream/multi_input_stream.d.ts.map +1 -0
- package/dist/stream/multi_input_stream.js +115 -0
- package/dist/stream/multi_input_stream.js.map +1 -0
- package/dist/stream/multi_input_stream.test.cjs +340 -0
- package/dist/stream/multi_input_stream.test.cjs.map +1 -0
- package/dist/stream/multi_input_stream.test.js +339 -0
- package/dist/stream/multi_input_stream.test.js.map +1 -0
- package/dist/telemetry/trace_types.cjs +42 -0
- package/dist/telemetry/trace_types.cjs.map +1 -1
- package/dist/telemetry/trace_types.d.cts +14 -0
- package/dist/telemetry/trace_types.d.ts +14 -0
- package/dist/telemetry/trace_types.d.ts.map +1 -1
- package/dist/telemetry/trace_types.js +28 -0
- package/dist/telemetry/trace_types.js.map +1 -1
- package/dist/utils.cjs +13 -0
- package/dist/utils.cjs.map +1 -1
- package/dist/utils.d.cts +1 -0
- package/dist/utils.d.ts +1 -0
- package/dist/utils.d.ts.map +1 -1
- package/dist/utils.js +13 -0
- package/dist/utils.js.map +1 -1
- package/dist/voice/agent_activity.cjs +35 -10
- package/dist/voice/agent_activity.cjs.map +1 -1
- package/dist/voice/agent_activity.d.cts +1 -0
- package/dist/voice/agent_activity.d.ts +1 -0
- package/dist/voice/agent_activity.d.ts.map +1 -1
- package/dist/voice/agent_activity.js +35 -10
- package/dist/voice/agent_activity.js.map +1 -1
- package/dist/voice/agent_session.cjs +19 -7
- package/dist/voice/agent_session.cjs.map +1 -1
- package/dist/voice/agent_session.d.cts +3 -2
- package/dist/voice/agent_session.d.ts +3 -2
- package/dist/voice/agent_session.d.ts.map +1 -1
- package/dist/voice/agent_session.js +19 -7
- package/dist/voice/agent_session.js.map +1 -1
- package/dist/voice/audio_recognition.cjs +85 -36
- package/dist/voice/audio_recognition.cjs.map +1 -1
- package/dist/voice/audio_recognition.d.cts +22 -1
- package/dist/voice/audio_recognition.d.ts +22 -1
- package/dist/voice/audio_recognition.d.ts.map +1 -1
- package/dist/voice/audio_recognition.js +89 -36
- package/dist/voice/audio_recognition.js.map +1 -1
- package/dist/voice/audio_recognition_span.test.cjs +233 -0
- package/dist/voice/audio_recognition_span.test.cjs.map +1 -0
- package/dist/voice/audio_recognition_span.test.js +232 -0
- package/dist/voice/audio_recognition_span.test.js.map +1 -0
- package/dist/voice/io.cjs +6 -3
- package/dist/voice/io.cjs.map +1 -1
- package/dist/voice/io.d.cts +3 -2
- package/dist/voice/io.d.ts +3 -2
- package/dist/voice/io.d.ts.map +1 -1
- package/dist/voice/io.js +6 -3
- package/dist/voice/io.js.map +1 -1
- package/dist/voice/recorder_io/recorder_io.cjs +3 -1
- package/dist/voice/recorder_io/recorder_io.cjs.map +1 -1
- package/dist/voice/recorder_io/recorder_io.d.ts.map +1 -1
- package/dist/voice/recorder_io/recorder_io.js +3 -1
- package/dist/voice/recorder_io/recorder_io.js.map +1 -1
- package/dist/voice/room_io/_input.cjs +17 -17
- package/dist/voice/room_io/_input.cjs.map +1 -1
- package/dist/voice/room_io/_input.d.cts +2 -2
- package/dist/voice/room_io/_input.d.ts +2 -2
- package/dist/voice/room_io/_input.d.ts.map +1 -1
- package/dist/voice/room_io/_input.js +7 -6
- package/dist/voice/room_io/_input.js.map +1 -1
- package/dist/voice/room_io/room_io.cjs +9 -0
- package/dist/voice/room_io/room_io.cjs.map +1 -1
- package/dist/voice/room_io/room_io.d.cts +3 -1
- package/dist/voice/room_io/room_io.d.ts +3 -1
- package/dist/voice/room_io/room_io.d.ts.map +1 -1
- package/dist/voice/room_io/room_io.js +9 -0
- package/dist/voice/room_io/room_io.js.map +1 -1
- package/dist/voice/utils.cjs +47 -0
- package/dist/voice/utils.cjs.map +1 -0
- package/dist/voice/utils.d.cts +4 -0
- package/dist/voice/utils.d.ts +4 -0
- package/dist/voice/utils.d.ts.map +1 -0
- package/dist/voice/utils.js +23 -0
- package/dist/voice/utils.js.map +1 -0
- package/package.json +1 -1
- package/src/stream/index.ts +1 -0
- package/src/stream/multi_input_stream.test.ts +540 -0
- package/src/stream/multi_input_stream.ts +172 -0
- package/src/telemetry/trace_types.ts +18 -0
- package/src/utils.ts +16 -0
- package/src/voice/agent_activity.ts +25 -0
- package/src/voice/agent_session.ts +17 -11
- package/src/voice/audio_recognition.ts +114 -38
- package/src/voice/audio_recognition_span.test.ts +261 -0
- package/src/voice/io.ts +7 -4
- package/src/voice/recorder_io/recorder_io.ts +2 -1
- package/src/voice/room_io/_input.ts +11 -7
- package/src/voice/room_io/room_io.ts +12 -0
- package/src/voice/utils.ts +29 -0
|
@@ -0,0 +1,340 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var import_web = require("node:stream/web");
|
|
3
|
+
var import_vitest = require("vitest");
|
|
4
|
+
var import_utils = require("../utils.cjs");
|
|
5
|
+
var import_multi_input_stream = require("./multi_input_stream.cjs");
|
|
6
|
+
function streamFrom(values) {
|
|
7
|
+
return new import_web.ReadableStream({
|
|
8
|
+
start(controller) {
|
|
9
|
+
for (const v of values) controller.enqueue(v);
|
|
10
|
+
controller.close();
|
|
11
|
+
}
|
|
12
|
+
});
|
|
13
|
+
}
|
|
14
|
+
(0, import_vitest.describe)("MultiInputStream", () => {
|
|
15
|
+
(0, import_vitest.it)("should create a readable output stream", () => {
|
|
16
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
17
|
+
(0, import_vitest.expect)(multi.stream).toBeInstanceOf(import_web.ReadableStream);
|
|
18
|
+
(0, import_vitest.expect)(multi.inputCount).toBe(0);
|
|
19
|
+
(0, import_vitest.expect)(multi.isClosed).toBe(false);
|
|
20
|
+
});
|
|
21
|
+
(0, import_vitest.it)("should read data from a single input stream", async () => {
|
|
22
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
23
|
+
const reader = multi.stream.getReader();
|
|
24
|
+
multi.addInputStream(streamFrom(["a", "b", "c"]));
|
|
25
|
+
const results = [];
|
|
26
|
+
for (let i = 0; i < 3; i++) {
|
|
27
|
+
const { value } = await reader.read();
|
|
28
|
+
results.push(value);
|
|
29
|
+
}
|
|
30
|
+
(0, import_vitest.expect)(results).toEqual(["a", "b", "c"]);
|
|
31
|
+
reader.releaseLock();
|
|
32
|
+
await multi.close();
|
|
33
|
+
});
|
|
34
|
+
(0, import_vitest.it)("should merge data from multiple input streams", async () => {
|
|
35
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
36
|
+
const reader = multi.stream.getReader();
|
|
37
|
+
multi.addInputStream(streamFrom([1, 2]));
|
|
38
|
+
multi.addInputStream(streamFrom([3, 4]));
|
|
39
|
+
const results = [];
|
|
40
|
+
for (let i = 0; i < 4; i++) {
|
|
41
|
+
const { value } = await reader.read();
|
|
42
|
+
results.push(value);
|
|
43
|
+
}
|
|
44
|
+
(0, import_vitest.expect)(results.sort()).toEqual([1, 2, 3, 4]);
|
|
45
|
+
reader.releaseLock();
|
|
46
|
+
await multi.close();
|
|
47
|
+
});
|
|
48
|
+
(0, import_vitest.it)("should allow adding inputs dynamically while reading", async () => {
|
|
49
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
50
|
+
const reader = multi.stream.getReader();
|
|
51
|
+
multi.addInputStream(streamFrom(["first"]));
|
|
52
|
+
const r1 = await reader.read();
|
|
53
|
+
(0, import_vitest.expect)(r1.value).toBe("first");
|
|
54
|
+
multi.addInputStream(streamFrom(["second"]));
|
|
55
|
+
const r2 = await reader.read();
|
|
56
|
+
(0, import_vitest.expect)(r2.value).toBe("second");
|
|
57
|
+
reader.releaseLock();
|
|
58
|
+
await multi.close();
|
|
59
|
+
});
|
|
60
|
+
(0, import_vitest.it)("should continue reading from remaining inputs after removing one", async () => {
|
|
61
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
62
|
+
const reader = multi.stream.getReader();
|
|
63
|
+
const slowSource = new import_web.ReadableStream({
|
|
64
|
+
async start(controller) {
|
|
65
|
+
controller.enqueue("slow-1");
|
|
66
|
+
await (0, import_utils.delay)(50);
|
|
67
|
+
controller.enqueue("slow-2");
|
|
68
|
+
await (0, import_utils.delay)(50);
|
|
69
|
+
controller.enqueue("slow-3");
|
|
70
|
+
controller.close();
|
|
71
|
+
}
|
|
72
|
+
});
|
|
73
|
+
const slowId = multi.addInputStream(slowSource);
|
|
74
|
+
const r1 = await reader.read();
|
|
75
|
+
(0, import_vitest.expect)(r1.value).toBe("slow-1");
|
|
76
|
+
await multi.removeInputStream(slowId);
|
|
77
|
+
multi.addInputStream(streamFrom(["fast-1", "fast-2"]));
|
|
78
|
+
const r2 = await reader.read();
|
|
79
|
+
(0, import_vitest.expect)(r2.value).toBe("fast-1");
|
|
80
|
+
const r3 = await reader.read();
|
|
81
|
+
(0, import_vitest.expect)(r3.value).toBe("fast-2");
|
|
82
|
+
reader.releaseLock();
|
|
83
|
+
await multi.close();
|
|
84
|
+
});
|
|
85
|
+
(0, import_vitest.it)("should handle swapping inputs (remove then add)", async () => {
|
|
86
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
87
|
+
const reader = multi.stream.getReader();
|
|
88
|
+
const id1 = multi.addInputStream(streamFrom(["from-A"]));
|
|
89
|
+
const r1 = await reader.read();
|
|
90
|
+
(0, import_vitest.expect)(r1.value).toBe("from-A");
|
|
91
|
+
await multi.removeInputStream(id1);
|
|
92
|
+
const id2 = multi.addInputStream(streamFrom(["from-B"]));
|
|
93
|
+
const r2 = await reader.read();
|
|
94
|
+
(0, import_vitest.expect)(r2.value).toBe("from-B");
|
|
95
|
+
await multi.removeInputStream(id2);
|
|
96
|
+
reader.releaseLock();
|
|
97
|
+
await multi.close();
|
|
98
|
+
});
|
|
99
|
+
(0, import_vitest.it)("should keep reader awaiting until an input is added", async () => {
|
|
100
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
101
|
+
const reader = multi.stream.getReader();
|
|
102
|
+
let readCompleted = false;
|
|
103
|
+
const readPromise = reader.read().then((result2) => {
|
|
104
|
+
readCompleted = true;
|
|
105
|
+
return result2;
|
|
106
|
+
});
|
|
107
|
+
await (0, import_utils.delay)(50);
|
|
108
|
+
(0, import_vitest.expect)(readCompleted).toBe(false);
|
|
109
|
+
multi.addInputStream(streamFrom(["hello"]));
|
|
110
|
+
const result = await readPromise;
|
|
111
|
+
(0, import_vitest.expect)(readCompleted).toBe(true);
|
|
112
|
+
(0, import_vitest.expect)(result.value).toBe("hello");
|
|
113
|
+
reader.releaseLock();
|
|
114
|
+
await multi.close();
|
|
115
|
+
});
|
|
116
|
+
(0, import_vitest.it)("should handle empty input streams without closing the output", async () => {
|
|
117
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
118
|
+
const reader = multi.stream.getReader();
|
|
119
|
+
multi.addInputStream(streamFrom([]));
|
|
120
|
+
await (0, import_utils.delay)(20);
|
|
121
|
+
multi.addInputStream(streamFrom(["data"]));
|
|
122
|
+
const result = await reader.read();
|
|
123
|
+
(0, import_vitest.expect)(result.value).toBe("data");
|
|
124
|
+
reader.releaseLock();
|
|
125
|
+
await multi.close();
|
|
126
|
+
});
|
|
127
|
+
(0, import_vitest.it)("should remove errored input without killing the output", async () => {
|
|
128
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
129
|
+
const reader = multi.stream.getReader();
|
|
130
|
+
const errorSource = new import_web.ReadableStream({
|
|
131
|
+
async start(controller) {
|
|
132
|
+
controller.enqueue("before-error");
|
|
133
|
+
await (0, import_utils.delay)(20);
|
|
134
|
+
controller.error(new Error("boom"));
|
|
135
|
+
}
|
|
136
|
+
});
|
|
137
|
+
multi.addInputStream(errorSource);
|
|
138
|
+
const r1 = await reader.read();
|
|
139
|
+
(0, import_vitest.expect)(r1.value).toBe("before-error");
|
|
140
|
+
await (0, import_utils.delay)(50);
|
|
141
|
+
(0, import_vitest.expect)(multi.inputCount).toBe(0);
|
|
142
|
+
multi.addInputStream(streamFrom(["after-error"]));
|
|
143
|
+
const r2 = await reader.read();
|
|
144
|
+
(0, import_vitest.expect)(r2.value).toBe("after-error");
|
|
145
|
+
reader.releaseLock();
|
|
146
|
+
await multi.close();
|
|
147
|
+
});
|
|
148
|
+
(0, import_vitest.it)("should keep other inputs alive when one errors", async () => {
|
|
149
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
150
|
+
const reader = multi.stream.getReader();
|
|
151
|
+
const goodSource = new import_web.ReadableStream({
|
|
152
|
+
async start(controller) {
|
|
153
|
+
await (0, import_utils.delay)(60);
|
|
154
|
+
controller.enqueue("good");
|
|
155
|
+
controller.close();
|
|
156
|
+
}
|
|
157
|
+
});
|
|
158
|
+
const badSource = new import_web.ReadableStream({
|
|
159
|
+
async start(controller) {
|
|
160
|
+
controller.error(new Error("bad"));
|
|
161
|
+
}
|
|
162
|
+
});
|
|
163
|
+
multi.addInputStream(goodSource);
|
|
164
|
+
multi.addInputStream(badSource);
|
|
165
|
+
await (0, import_utils.delay)(10);
|
|
166
|
+
const result = await reader.read();
|
|
167
|
+
(0, import_vitest.expect)(result.value).toBe("good");
|
|
168
|
+
reader.releaseLock();
|
|
169
|
+
await multi.close();
|
|
170
|
+
});
|
|
171
|
+
(0, import_vitest.it)("should end the output stream with done:true when close is called", async () => {
|
|
172
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
173
|
+
const reader = multi.stream.getReader();
|
|
174
|
+
multi.addInputStream(streamFrom(["data"]));
|
|
175
|
+
const r1 = await reader.read();
|
|
176
|
+
(0, import_vitest.expect)(r1.value).toBe("data");
|
|
177
|
+
await multi.close();
|
|
178
|
+
const r2 = await reader.read();
|
|
179
|
+
(0, import_vitest.expect)(r2.done).toBe(true);
|
|
180
|
+
(0, import_vitest.expect)(r2.value).toBeUndefined();
|
|
181
|
+
reader.releaseLock();
|
|
182
|
+
});
|
|
183
|
+
(0, import_vitest.it)("should resolve pending reads as done when close is called", async () => {
|
|
184
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
185
|
+
const reader = multi.stream.getReader();
|
|
186
|
+
const readPromise = reader.read();
|
|
187
|
+
await (0, import_utils.delay)(10);
|
|
188
|
+
await multi.close();
|
|
189
|
+
const result = await readPromise;
|
|
190
|
+
(0, import_vitest.expect)(result.done).toBe(true);
|
|
191
|
+
(0, import_vitest.expect)(result.value).toBeUndefined();
|
|
192
|
+
reader.releaseLock();
|
|
193
|
+
});
|
|
194
|
+
(0, import_vitest.it)("should be idempotent for multiple close calls", async () => {
|
|
195
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
196
|
+
await multi.close();
|
|
197
|
+
await multi.close();
|
|
198
|
+
(0, import_vitest.expect)(multi.isClosed).toBe(true);
|
|
199
|
+
});
|
|
200
|
+
(0, import_vitest.it)("should throw when adding input after close", async () => {
|
|
201
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
202
|
+
await multi.close();
|
|
203
|
+
(0, import_vitest.expect)(() => multi.addInputStream(streamFrom(["x"]))).toThrow("MultiInputStream is closed");
|
|
204
|
+
});
|
|
205
|
+
(0, import_vitest.it)("should no-op when removing a non-existent input", async () => {
|
|
206
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
207
|
+
await multi.removeInputStream("does-not-exist");
|
|
208
|
+
await multi.close();
|
|
209
|
+
});
|
|
210
|
+
(0, import_vitest.it)("should release the source reader lock so the source can be reused", async () => {
|
|
211
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
212
|
+
const reader = multi.stream.getReader();
|
|
213
|
+
const source = new import_web.ReadableStream({
|
|
214
|
+
async start(controller) {
|
|
215
|
+
controller.enqueue("chunk-0");
|
|
216
|
+
await (0, import_utils.delay)(30);
|
|
217
|
+
controller.enqueue("chunk-1");
|
|
218
|
+
controller.close();
|
|
219
|
+
}
|
|
220
|
+
});
|
|
221
|
+
const id = multi.addInputStream(source);
|
|
222
|
+
const r1 = await reader.read();
|
|
223
|
+
(0, import_vitest.expect)(r1.value).toBe("chunk-0");
|
|
224
|
+
await multi.removeInputStream(id);
|
|
225
|
+
const sourceReader = source.getReader();
|
|
226
|
+
const sr = await sourceReader.read();
|
|
227
|
+
(0, import_vitest.expect)(sr.value).toBe("chunk-1");
|
|
228
|
+
sourceReader.releaseLock();
|
|
229
|
+
reader.releaseLock();
|
|
230
|
+
await multi.close();
|
|
231
|
+
});
|
|
232
|
+
(0, import_vitest.it)("should track inputCount correctly through add / remove / natural end", async () => {
|
|
233
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
234
|
+
(0, import_vitest.expect)(multi.inputCount).toBe(0);
|
|
235
|
+
const id1 = multi.addInputStream(streamFrom(["a"]));
|
|
236
|
+
const id2 = multi.addInputStream(streamFrom(["b"]));
|
|
237
|
+
(0, import_vitest.expect)(multi.inputCount).toBe(2);
|
|
238
|
+
await multi.removeInputStream(id1);
|
|
239
|
+
(0, import_vitest.expect)(multi.inputCount).toBeLessThanOrEqual(1);
|
|
240
|
+
await (0, import_utils.delay)(20);
|
|
241
|
+
(0, import_vitest.expect)(multi.inputCount).toBe(0);
|
|
242
|
+
await multi.removeInputStream(id2);
|
|
243
|
+
(0, import_vitest.expect)(multi.inputCount).toBe(0);
|
|
244
|
+
await multi.close();
|
|
245
|
+
});
|
|
246
|
+
(0, import_vitest.it)("should handle concurrent reads and slow writes", async () => {
|
|
247
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
248
|
+
const reader = multi.stream.getReader();
|
|
249
|
+
const chunks = ["a", "b", "c", "d", "e"];
|
|
250
|
+
let idx = 0;
|
|
251
|
+
const source = new import_web.ReadableStream({
|
|
252
|
+
start(controller) {
|
|
253
|
+
const writeNext = () => {
|
|
254
|
+
if (idx < chunks.length) {
|
|
255
|
+
controller.enqueue(chunks[idx++]);
|
|
256
|
+
setTimeout(writeNext, 5);
|
|
257
|
+
} else {
|
|
258
|
+
controller.close();
|
|
259
|
+
}
|
|
260
|
+
};
|
|
261
|
+
writeNext();
|
|
262
|
+
}
|
|
263
|
+
});
|
|
264
|
+
multi.addInputStream(source);
|
|
265
|
+
const results = [];
|
|
266
|
+
for (let i = 0; i < chunks.length; i++) {
|
|
267
|
+
const { value } = await reader.read();
|
|
268
|
+
results.push(value);
|
|
269
|
+
}
|
|
270
|
+
(0, import_vitest.expect)(results).toEqual(chunks);
|
|
271
|
+
reader.releaseLock();
|
|
272
|
+
await multi.close();
|
|
273
|
+
});
|
|
274
|
+
(0, import_vitest.it)("should handle backpressure with large data", async () => {
|
|
275
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
276
|
+
const largeChunks = Array.from({ length: 1e3 }, (_, i) => `chunk-${i}`);
|
|
277
|
+
multi.addInputStream(streamFrom(largeChunks));
|
|
278
|
+
const reader = multi.stream.getReader();
|
|
279
|
+
const results = [];
|
|
280
|
+
let result = await reader.read();
|
|
281
|
+
while (!result.done) {
|
|
282
|
+
results.push(result.value);
|
|
283
|
+
if (results.length === largeChunks.length) break;
|
|
284
|
+
result = await reader.read();
|
|
285
|
+
}
|
|
286
|
+
(0, import_vitest.expect)(results).toEqual(largeChunks);
|
|
287
|
+
reader.releaseLock();
|
|
288
|
+
await multi.close();
|
|
289
|
+
});
|
|
290
|
+
(0, import_vitest.it)("should support tee on the output stream", async () => {
|
|
291
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
292
|
+
const [s1, s2] = multi.stream.tee();
|
|
293
|
+
const r1 = s1.getReader();
|
|
294
|
+
const r2 = s2.getReader();
|
|
295
|
+
multi.addInputStream(streamFrom([10, 20]));
|
|
296
|
+
const [a1, a2] = await Promise.all([r1.read(), r2.read()]);
|
|
297
|
+
(0, import_vitest.expect)(a1.value).toBe(10);
|
|
298
|
+
(0, import_vitest.expect)(a2.value).toBe(10);
|
|
299
|
+
const [b1, b2] = await Promise.all([r1.read(), r2.read()]);
|
|
300
|
+
(0, import_vitest.expect)(b1.value).toBe(20);
|
|
301
|
+
(0, import_vitest.expect)(b2.value).toBe(20);
|
|
302
|
+
r1.releaseLock();
|
|
303
|
+
r2.releaseLock();
|
|
304
|
+
await multi.close();
|
|
305
|
+
});
|
|
306
|
+
(0, import_vitest.it)("should return unique IDs from addInputStream", () => {
|
|
307
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
308
|
+
const id1 = multi.addInputStream(streamFrom(["a"]));
|
|
309
|
+
const id2 = multi.addInputStream(streamFrom(["b"]));
|
|
310
|
+
const id3 = multi.addInputStream(streamFrom(["c"]));
|
|
311
|
+
(0, import_vitest.expect)(id1).not.toBe(id2);
|
|
312
|
+
(0, import_vitest.expect)(id2).not.toBe(id3);
|
|
313
|
+
(0, import_vitest.expect)(id1).not.toBe(id3);
|
|
314
|
+
});
|
|
315
|
+
(0, import_vitest.it)("should cleanly close while pumps are actively writing", async () => {
|
|
316
|
+
const multi = new import_multi_input_stream.MultiInputStream();
|
|
317
|
+
const reader = multi.stream.getReader();
|
|
318
|
+
const infiniteSource = new import_web.ReadableStream({
|
|
319
|
+
async start(controller) {
|
|
320
|
+
let i = 0;
|
|
321
|
+
while (true) {
|
|
322
|
+
try {
|
|
323
|
+
controller.enqueue(`tick-${i++}`);
|
|
324
|
+
} catch {
|
|
325
|
+
break;
|
|
326
|
+
}
|
|
327
|
+
await (0, import_utils.delay)(5);
|
|
328
|
+
}
|
|
329
|
+
}
|
|
330
|
+
});
|
|
331
|
+
multi.addInputStream(infiniteSource);
|
|
332
|
+
const r1 = await reader.read();
|
|
333
|
+
(0, import_vitest.expect)(r1.done).toBe(false);
|
|
334
|
+
await multi.close();
|
|
335
|
+
const r2 = await reader.read();
|
|
336
|
+
(0, import_vitest.expect)(r2.done).toBe(true);
|
|
337
|
+
reader.releaseLock();
|
|
338
|
+
});
|
|
339
|
+
});
|
|
340
|
+
//# sourceMappingURL=multi_input_stream.test.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/stream/multi_input_stream.test.ts"],"sourcesContent":["// SPDX-FileCopyrightText: 2025 LiveKit, Inc.\n//\n// SPDX-License-Identifier: Apache-2.0\nimport { ReadableStream } from 'node:stream/web';\nimport { describe, expect, it } from 'vitest';\nimport { delay } from '../utils.js';\nimport { MultiInputStream } from './multi_input_stream.js';\n\nfunction streamFrom<T>(values: T[]): ReadableStream<T> {\n return new ReadableStream<T>({\n start(controller) {\n for (const v of values) controller.enqueue(v);\n controller.close();\n },\n });\n}\n\ndescribe('MultiInputStream', () => {\n // ---------------------------------------------------------------------------\n // Basic functionality\n // ---------------------------------------------------------------------------\n\n it('should create a readable output stream', () => {\n const multi = new MultiInputStream<string>();\n expect(multi.stream).toBeInstanceOf(ReadableStream);\n expect(multi.inputCount).toBe(0);\n expect(multi.isClosed).toBe(false);\n });\n\n it('should read data from a single input stream', async () => {\n const multi = new MultiInputStream<string>();\n const reader = multi.stream.getReader();\n\n multi.addInputStream(streamFrom(['a', 'b', 'c']));\n\n const results: string[] = [];\n // Read three values then close manually (output stays open after input ends).\n for (let i = 0; i < 3; i++) {\n const { value } = await reader.read();\n results.push(value!);\n }\n\n expect(results).toEqual(['a', 'b', 'c']);\n reader.releaseLock();\n await multi.close();\n });\n\n it('should merge data from multiple input streams', async () => {\n const multi = new MultiInputStream<number>();\n const reader = multi.stream.getReader();\n\n multi.addInputStream(streamFrom([1, 2]));\n multi.addInputStream(streamFrom([3, 4]));\n\n const results: number[] = [];\n for (let i = 0; i < 4; i++) {\n const { value } = await reader.read();\n results.push(value!);\n }\n\n // Order is non-deterministic but all values must arrive.\n expect(results.sort()).toEqual([1, 2, 3, 4]);\n reader.releaseLock();\n await multi.close();\n });\n\n // ---------------------------------------------------------------------------\n // Dynamic add / remove\n // ---------------------------------------------------------------------------\n\n it('should allow adding inputs dynamically while reading', async () => {\n const multi = new MultiInputStream<string>();\n const reader = multi.stream.getReader();\n\n multi.addInputStream(streamFrom(['first']));\n\n const r1 = await reader.read();\n expect(r1.value).toBe('first');\n\n // Add a second input after reading from the first.\n multi.addInputStream(streamFrom(['second']));\n\n const r2 = await reader.read();\n expect(r2.value).toBe('second');\n\n reader.releaseLock();\n await multi.close();\n });\n\n it('should continue reading from remaining inputs after removing one', async () => {\n const multi = new MultiInputStream<string>();\n const reader = multi.stream.getReader();\n\n // A slow stream that emits over time.\n const slowSource = new ReadableStream<string>({\n async start(controller) {\n controller.enqueue('slow-1');\n await delay(50);\n controller.enqueue('slow-2');\n await delay(50);\n controller.enqueue('slow-3');\n controller.close();\n },\n });\n\n const slowId = multi.addInputStream(slowSource);\n\n // Read first value from slow source.\n const r1 = await reader.read();\n expect(r1.value).toBe('slow-1');\n\n // Remove the slow source and add a fast one.\n await multi.removeInputStream(slowId);\n\n multi.addInputStream(streamFrom(['fast-1', 'fast-2']));\n\n const r2 = await reader.read();\n expect(r2.value).toBe('fast-1');\n\n const r3 = await reader.read();\n expect(r3.value).toBe('fast-2');\n\n reader.releaseLock();\n await multi.close();\n });\n\n it('should handle swapping inputs (remove then add)', async () => {\n const multi = new MultiInputStream<string>();\n const reader = multi.stream.getReader();\n\n const id1 = multi.addInputStream(streamFrom(['from-A']));\n\n const r1 = await reader.read();\n expect(r1.value).toBe('from-A');\n\n await multi.removeInputStream(id1);\n\n const id2 = multi.addInputStream(streamFrom(['from-B']));\n\n const r2 = await reader.read();\n expect(r2.value).toBe('from-B');\n\n await multi.removeInputStream(id2);\n reader.releaseLock();\n await multi.close();\n });\n\n // ---------------------------------------------------------------------------\n // Reading before any input is added\n // ---------------------------------------------------------------------------\n\n it('should keep reader awaiting until an input is added', async () => {\n const multi = new MultiInputStream<string>();\n const reader = multi.stream.getReader();\n\n let readCompleted = false;\n const readPromise = reader.read().then((result) => {\n readCompleted = true;\n return result;\n });\n\n await delay(50);\n expect(readCompleted).toBe(false);\n\n // Now add an input to unblock the read.\n multi.addInputStream(streamFrom(['hello']));\n\n const result = await readPromise;\n expect(readCompleted).toBe(true);\n expect(result.value).toBe('hello');\n\n reader.releaseLock();\n await multi.close();\n });\n\n // ---------------------------------------------------------------------------\n // Empty input streams\n // ---------------------------------------------------------------------------\n\n it('should handle empty input streams without closing the output', async () => {\n const multi = new MultiInputStream<string>();\n const reader = multi.stream.getReader();\n\n // Add an empty stream — it should end immediately without affecting the output.\n multi.addInputStream(streamFrom([]));\n\n await delay(20);\n\n // The output should still be open. Adding a real input should work.\n multi.addInputStream(streamFrom(['data']));\n\n const result = await reader.read();\n expect(result.value).toBe('data');\n\n reader.releaseLock();\n await multi.close();\n });\n\n // ---------------------------------------------------------------------------\n // Error handling\n // ---------------------------------------------------------------------------\n\n it('should remove errored input without killing the output', async () => {\n const multi = new MultiInputStream<string>();\n const reader = multi.stream.getReader();\n\n // An input that errors after emitting one value.\n const errorSource = new ReadableStream<string>({\n async start(controller) {\n controller.enqueue('before-error');\n await delay(20);\n controller.error(new Error('boom'));\n },\n });\n\n multi.addInputStream(errorSource);\n\n const r1 = await reader.read();\n expect(r1.value).toBe('before-error');\n\n // Wait for the error to propagate and the input to be removed.\n await delay(50);\n\n expect(multi.inputCount).toBe(0);\n\n // The output is still alive — we can add another input.\n multi.addInputStream(streamFrom(['after-error']));\n\n const r2 = await reader.read();\n expect(r2.value).toBe('after-error');\n\n reader.releaseLock();\n await multi.close();\n });\n\n it('should keep other inputs alive when one errors', async () => {\n const multi = new MultiInputStream<string>();\n const reader = multi.stream.getReader();\n\n const goodSource = new ReadableStream<string>({\n async start(controller) {\n await delay(60);\n controller.enqueue('good');\n controller.close();\n },\n });\n\n const badSource = new ReadableStream<string>({\n async start(controller) {\n controller.error(new Error('bad'));\n },\n });\n\n multi.addInputStream(goodSource);\n multi.addInputStream(badSource);\n\n // Wait a bit for the bad source to error and be removed.\n await delay(10);\n\n // The good source should still be pumping.\n const result = await reader.read();\n expect(result.value).toBe('good');\n\n reader.releaseLock();\n await multi.close();\n });\n\n // ---------------------------------------------------------------------------\n // Close semantics\n // ---------------------------------------------------------------------------\n\n it('should end the output stream with done:true when close is called', async () => {\n const multi = new MultiInputStream<string>();\n const reader = multi.stream.getReader();\n\n multi.addInputStream(streamFrom(['data']));\n\n const r1 = await reader.read();\n expect(r1.value).toBe('data');\n\n await multi.close();\n\n const r2 = await reader.read();\n expect(r2.done).toBe(true);\n expect(r2.value).toBeUndefined();\n\n reader.releaseLock();\n });\n\n it('should resolve pending reads as done when close is called', async () => {\n const multi = new MultiInputStream<string>();\n const reader = multi.stream.getReader();\n\n // No inputs — read will be pending.\n const readPromise = reader.read();\n\n await delay(10);\n await multi.close();\n\n const result = await readPromise;\n expect(result.done).toBe(true);\n expect(result.value).toBeUndefined();\n\n reader.releaseLock();\n });\n\n it('should be idempotent for multiple close calls', async () => {\n const multi = new MultiInputStream<string>();\n\n await multi.close();\n await multi.close();\n\n expect(multi.isClosed).toBe(true);\n });\n\n it('should throw when adding input after close', async () => {\n const multi = new MultiInputStream<string>();\n await multi.close();\n\n expect(() => multi.addInputStream(streamFrom(['x']))).toThrow('MultiInputStream is closed');\n });\n\n // ---------------------------------------------------------------------------\n // removeInputStream edge cases\n // ---------------------------------------------------------------------------\n\n it('should no-op when removing a non-existent input', async () => {\n const multi = new MultiInputStream<string>();\n\n // Should not throw.\n await multi.removeInputStream('does-not-exist');\n\n await multi.close();\n });\n\n it('should release the source reader lock so the source can be reused', async () => {\n const multi = new MultiInputStream<string>();\n const reader = multi.stream.getReader();\n\n const source = new ReadableStream<string>({\n async start(controller) {\n controller.enqueue('chunk-0');\n await delay(30);\n controller.enqueue('chunk-1');\n controller.close();\n },\n });\n\n const id = multi.addInputStream(source);\n\n const r1 = await reader.read();\n expect(r1.value).toBe('chunk-0');\n\n await multi.removeInputStream(id);\n\n // The source's reader lock should be released — we can get a new reader.\n const sourceReader = source.getReader();\n const sr = await sourceReader.read();\n expect(sr.value).toBe('chunk-1');\n sourceReader.releaseLock();\n\n reader.releaseLock();\n await multi.close();\n });\n\n // ---------------------------------------------------------------------------\n // Input count tracking\n // ---------------------------------------------------------------------------\n\n it('should track inputCount correctly through add / remove / natural end', async () => {\n const multi = new MultiInputStream<string>();\n\n expect(multi.inputCount).toBe(0);\n\n const id1 = multi.addInputStream(streamFrom(['a']));\n const id2 = multi.addInputStream(streamFrom(['b']));\n\n expect(multi.inputCount).toBe(2);\n\n await multi.removeInputStream(id1);\n expect(multi.inputCount).toBeLessThanOrEqual(1);\n\n // Let the remaining stream finish.\n await delay(20);\n expect(multi.inputCount).toBe(0);\n\n await multi.removeInputStream(id2); // already gone, no-op\n expect(multi.inputCount).toBe(0);\n\n await multi.close();\n });\n\n // ---------------------------------------------------------------------------\n // Concurrent reads and writes\n // ---------------------------------------------------------------------------\n\n it('should handle concurrent reads and slow writes', async () => {\n const multi = new MultiInputStream<string>();\n const reader = multi.stream.getReader();\n\n const chunks = ['a', 'b', 'c', 'd', 'e'];\n let idx = 0;\n\n const source = new ReadableStream<string>({\n start(controller) {\n const writeNext = () => {\n if (idx < chunks.length) {\n controller.enqueue(chunks[idx++]);\n setTimeout(writeNext, 5);\n } else {\n controller.close();\n }\n };\n writeNext();\n },\n });\n\n multi.addInputStream(source);\n\n const results: string[] = [];\n for (let i = 0; i < chunks.length; i++) {\n const { value } = await reader.read();\n results.push(value!);\n }\n\n expect(results).toEqual(chunks);\n\n reader.releaseLock();\n await multi.close();\n });\n\n // ---------------------------------------------------------------------------\n // Backpressure\n // ---------------------------------------------------------------------------\n\n it('should handle backpressure with large data', async () => {\n const multi = new MultiInputStream<string>();\n\n const largeChunks = Array.from({ length: 1000 }, (_, i) => `chunk-${i}`);\n multi.addInputStream(streamFrom(largeChunks));\n\n const reader = multi.stream.getReader();\n const results: string[] = [];\n\n let result = await reader.read();\n while (!result.done) {\n results.push(result.value);\n // Check if we've collected all expected values before reading again,\n // to avoid hanging on the output which stays open after input ends.\n if (results.length === largeChunks.length) break;\n result = await reader.read();\n }\n\n expect(results).toEqual(largeChunks);\n\n reader.releaseLock();\n await multi.close();\n });\n\n // ---------------------------------------------------------------------------\n // Multiple tee / concurrent consumers\n // ---------------------------------------------------------------------------\n\n it('should support tee on the output stream', async () => {\n const multi = new MultiInputStream<number>();\n\n const [s1, s2] = multi.stream.tee();\n const r1 = s1.getReader();\n const r2 = s2.getReader();\n\n multi.addInputStream(streamFrom([10, 20]));\n\n const [a1, a2] = await Promise.all([r1.read(), r2.read()]);\n expect(a1.value).toBe(10);\n expect(a2.value).toBe(10);\n\n const [b1, b2] = await Promise.all([r1.read(), r2.read()]);\n expect(b1.value).toBe(20);\n expect(b2.value).toBe(20);\n\n r1.releaseLock();\n r2.releaseLock();\n await multi.close();\n });\n\n // ---------------------------------------------------------------------------\n // Return value of addInputStream\n // ---------------------------------------------------------------------------\n\n it('should return unique IDs from addInputStream', () => {\n const multi = new MultiInputStream<string>();\n\n const id1 = multi.addInputStream(streamFrom(['a']));\n const id2 = multi.addInputStream(streamFrom(['b']));\n const id3 = multi.addInputStream(streamFrom(['c']));\n\n expect(id1).not.toBe(id2);\n expect(id2).not.toBe(id3);\n expect(id1).not.toBe(id3);\n });\n\n // ---------------------------------------------------------------------------\n // close() while pumps are actively writing\n // ---------------------------------------------------------------------------\n\n it('should cleanly close while pumps are actively writing', async () => {\n const multi = new MultiInputStream<string>();\n const reader = multi.stream.getReader();\n\n // A source that never stops on its own.\n const infiniteSource = new ReadableStream<string>({\n async start(controller) {\n let i = 0;\n while (true) {\n try {\n controller.enqueue(`tick-${i++}`);\n } catch {\n // controller.enqueue throws after stream is canceled\n break;\n }\n await delay(5);\n }\n },\n });\n\n multi.addInputStream(infiniteSource);\n\n // Read a couple of values.\n const r1 = await reader.read();\n expect(r1.done).toBe(false);\n\n // Close while the infinite source is still pumping.\n await multi.close();\n\n const r2 = await reader.read();\n expect(r2.done).toBe(true);\n\n reader.releaseLock();\n });\n});\n"],"mappings":"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","names":["result"]}
|