@livekit/agents 1.0.34 → 1.0.36-dev.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (187) hide show
  1. package/dist/cli.cjs.map +1 -1
  2. package/dist/index.cjs +3 -1
  3. package/dist/index.cjs.map +1 -1
  4. package/dist/index.d.cts +1 -0
  5. package/dist/index.d.ts +1 -0
  6. package/dist/index.d.ts.map +1 -1
  7. package/dist/index.js +1 -0
  8. package/dist/index.js.map +1 -1
  9. package/dist/inference/api_protos.d.cts +4 -4
  10. package/dist/inference/api_protos.d.ts +4 -4
  11. package/dist/inference/interruption/AdaptiveInterruptionDetector.cjs +152 -0
  12. package/dist/inference/interruption/AdaptiveInterruptionDetector.cjs.map +1 -0
  13. package/dist/inference/interruption/AdaptiveInterruptionDetector.d.cts +50 -0
  14. package/dist/inference/interruption/AdaptiveInterruptionDetector.d.ts +50 -0
  15. package/dist/inference/interruption/AdaptiveInterruptionDetector.d.ts.map +1 -0
  16. package/dist/inference/interruption/AdaptiveInterruptionDetector.js +125 -0
  17. package/dist/inference/interruption/AdaptiveInterruptionDetector.js.map +1 -0
  18. package/dist/inference/interruption/InterruptionStream.cjs +310 -0
  19. package/dist/inference/interruption/InterruptionStream.cjs.map +1 -0
  20. package/dist/inference/interruption/InterruptionStream.d.cts +57 -0
  21. package/dist/inference/interruption/InterruptionStream.d.ts +57 -0
  22. package/dist/inference/interruption/InterruptionStream.d.ts.map +1 -0
  23. package/dist/inference/interruption/InterruptionStream.js +288 -0
  24. package/dist/inference/interruption/InterruptionStream.js.map +1 -0
  25. package/dist/inference/interruption/defaults.cjs +76 -0
  26. package/dist/inference/interruption/defaults.cjs.map +1 -0
  27. package/dist/inference/interruption/defaults.d.cts +14 -0
  28. package/dist/inference/interruption/defaults.d.ts +14 -0
  29. package/dist/inference/interruption/defaults.d.ts.map +1 -0
  30. package/dist/inference/interruption/defaults.js +42 -0
  31. package/dist/inference/interruption/defaults.js.map +1 -0
  32. package/dist/inference/interruption/errors.cjs +2 -0
  33. package/dist/inference/interruption/errors.cjs.map +1 -0
  34. package/dist/inference/interruption/errors.d.cts +2 -0
  35. package/dist/inference/interruption/errors.d.ts +2 -0
  36. package/dist/inference/interruption/errors.d.ts.map +1 -0
  37. package/dist/inference/interruption/errors.js +1 -0
  38. package/dist/inference/interruption/errors.js.map +1 -0
  39. package/dist/inference/interruption/http_transport.cjs +57 -0
  40. package/dist/inference/interruption/http_transport.cjs.map +1 -0
  41. package/dist/inference/interruption/http_transport.d.cts +23 -0
  42. package/dist/inference/interruption/http_transport.d.ts +23 -0
  43. package/dist/inference/interruption/http_transport.d.ts.map +1 -0
  44. package/dist/inference/interruption/http_transport.js +33 -0
  45. package/dist/inference/interruption/http_transport.js.map +1 -0
  46. package/dist/inference/interruption/index.cjs +34 -0
  47. package/dist/inference/interruption/index.cjs.map +1 -0
  48. package/dist/inference/interruption/index.d.cts +5 -0
  49. package/dist/inference/interruption/index.d.ts +5 -0
  50. package/dist/inference/interruption/index.d.ts.map +1 -0
  51. package/dist/inference/interruption/index.js +7 -0
  52. package/dist/inference/interruption/index.js.map +1 -0
  53. package/dist/inference/interruption/interruption.cjs +85 -0
  54. package/dist/inference/interruption/interruption.cjs.map +1 -0
  55. package/dist/inference/interruption/interruption.d.cts +48 -0
  56. package/dist/inference/interruption/interruption.d.ts +48 -0
  57. package/dist/inference/interruption/interruption.d.ts.map +1 -0
  58. package/dist/inference/interruption/interruption.js +59 -0
  59. package/dist/inference/interruption/interruption.js.map +1 -0
  60. package/dist/inference/llm.cjs +30 -3
  61. package/dist/inference/llm.cjs.map +1 -1
  62. package/dist/inference/llm.d.cts +3 -1
  63. package/dist/inference/llm.d.ts +3 -1
  64. package/dist/inference/llm.d.ts.map +1 -1
  65. package/dist/inference/llm.js +30 -3
  66. package/dist/inference/llm.js.map +1 -1
  67. package/dist/inference/utils.cjs +15 -2
  68. package/dist/inference/utils.cjs.map +1 -1
  69. package/dist/inference/utils.d.cts +1 -0
  70. package/dist/inference/utils.d.ts +1 -0
  71. package/dist/inference/utils.d.ts.map +1 -1
  72. package/dist/inference/utils.js +13 -1
  73. package/dist/inference/utils.js.map +1 -1
  74. package/dist/inference/utils.test.cjs +20 -0
  75. package/dist/inference/utils.test.cjs.map +1 -0
  76. package/dist/inference/utils.test.js +19 -0
  77. package/dist/inference/utils.test.js.map +1 -0
  78. package/dist/ipc/inference_proc_executor.cjs.map +1 -1
  79. package/dist/ipc/job_proc_executor.cjs.map +1 -1
  80. package/dist/ipc/job_proc_lazy_main.cjs +1 -1
  81. package/dist/ipc/job_proc_lazy_main.cjs.map +1 -1
  82. package/dist/ipc/job_proc_lazy_main.js +1 -1
  83. package/dist/ipc/job_proc_lazy_main.js.map +1 -1
  84. package/dist/llm/chat_context.cjs +20 -2
  85. package/dist/llm/chat_context.cjs.map +1 -1
  86. package/dist/llm/chat_context.d.cts +9 -0
  87. package/dist/llm/chat_context.d.ts +9 -0
  88. package/dist/llm/chat_context.d.ts.map +1 -1
  89. package/dist/llm/chat_context.js +20 -2
  90. package/dist/llm/chat_context.js.map +1 -1
  91. package/dist/llm/llm.cjs.map +1 -1
  92. package/dist/llm/llm.d.cts +1 -0
  93. package/dist/llm/llm.d.ts +1 -0
  94. package/dist/llm/llm.d.ts.map +1 -1
  95. package/dist/llm/llm.js.map +1 -1
  96. package/dist/llm/provider_format/openai.cjs +43 -20
  97. package/dist/llm/provider_format/openai.cjs.map +1 -1
  98. package/dist/llm/provider_format/openai.d.ts.map +1 -1
  99. package/dist/llm/provider_format/openai.js +43 -20
  100. package/dist/llm/provider_format/openai.js.map +1 -1
  101. package/dist/llm/provider_format/openai.test.cjs +35 -0
  102. package/dist/llm/provider_format/openai.test.cjs.map +1 -1
  103. package/dist/llm/provider_format/openai.test.js +35 -0
  104. package/dist/llm/provider_format/openai.test.js.map +1 -1
  105. package/dist/llm/provider_format/utils.cjs +1 -1
  106. package/dist/llm/provider_format/utils.cjs.map +1 -1
  107. package/dist/llm/provider_format/utils.d.ts.map +1 -1
  108. package/dist/llm/provider_format/utils.js +1 -1
  109. package/dist/llm/provider_format/utils.js.map +1 -1
  110. package/dist/stream/stream_channel.cjs +3 -0
  111. package/dist/stream/stream_channel.cjs.map +1 -1
  112. package/dist/stream/stream_channel.d.cts +3 -2
  113. package/dist/stream/stream_channel.d.ts +3 -2
  114. package/dist/stream/stream_channel.d.ts.map +1 -1
  115. package/dist/stream/stream_channel.js +3 -0
  116. package/dist/stream/stream_channel.js.map +1 -1
  117. package/dist/telemetry/trace_types.cjs +15 -0
  118. package/dist/telemetry/trace_types.cjs.map +1 -1
  119. package/dist/telemetry/trace_types.d.cts +5 -0
  120. package/dist/telemetry/trace_types.d.ts +5 -0
  121. package/dist/telemetry/trace_types.d.ts.map +1 -1
  122. package/dist/telemetry/trace_types.js +10 -0
  123. package/dist/telemetry/trace_types.js.map +1 -1
  124. package/dist/utils/ws_transport.cjs +51 -0
  125. package/dist/utils/ws_transport.cjs.map +1 -0
  126. package/dist/utils/ws_transport.d.cts +9 -0
  127. package/dist/utils/ws_transport.d.ts +9 -0
  128. package/dist/utils/ws_transport.d.ts.map +1 -0
  129. package/dist/utils/ws_transport.js +17 -0
  130. package/dist/utils/ws_transport.js.map +1 -0
  131. package/dist/utils/ws_transport.test.cjs +212 -0
  132. package/dist/utils/ws_transport.test.cjs.map +1 -0
  133. package/dist/utils/ws_transport.test.js +211 -0
  134. package/dist/utils/ws_transport.test.js.map +1 -0
  135. package/dist/voice/agent_activity.cjs +49 -0
  136. package/dist/voice/agent_activity.cjs.map +1 -1
  137. package/dist/voice/agent_activity.d.cts +14 -0
  138. package/dist/voice/agent_activity.d.ts +14 -0
  139. package/dist/voice/agent_activity.d.ts.map +1 -1
  140. package/dist/voice/agent_activity.js +49 -0
  141. package/dist/voice/agent_activity.js.map +1 -1
  142. package/dist/voice/agent_session.cjs +12 -1
  143. package/dist/voice/agent_session.cjs.map +1 -1
  144. package/dist/voice/agent_session.d.cts +3 -0
  145. package/dist/voice/agent_session.d.ts +3 -0
  146. package/dist/voice/agent_session.d.ts.map +1 -1
  147. package/dist/voice/agent_session.js +12 -1
  148. package/dist/voice/agent_session.js.map +1 -1
  149. package/dist/voice/audio_recognition.cjs +124 -2
  150. package/dist/voice/audio_recognition.cjs.map +1 -1
  151. package/dist/voice/audio_recognition.d.cts +32 -1
  152. package/dist/voice/audio_recognition.d.ts +32 -1
  153. package/dist/voice/audio_recognition.d.ts.map +1 -1
  154. package/dist/voice/audio_recognition.js +127 -2
  155. package/dist/voice/audio_recognition.js.map +1 -1
  156. package/dist/voice/background_audio.cjs.map +1 -1
  157. package/dist/voice/generation.cjs +2 -1
  158. package/dist/voice/generation.cjs.map +1 -1
  159. package/dist/voice/generation.d.ts.map +1 -1
  160. package/dist/voice/generation.js +2 -1
  161. package/dist/voice/generation.js.map +1 -1
  162. package/package.json +2 -1
  163. package/src/index.ts +2 -0
  164. package/src/inference/interruption/AdaptiveInterruptionDetector.ts +166 -0
  165. package/src/inference/interruption/InterruptionStream.ts +397 -0
  166. package/src/inference/interruption/defaults.ts +33 -0
  167. package/src/inference/interruption/errors.ts +0 -0
  168. package/src/inference/interruption/http_transport.ts +61 -0
  169. package/src/inference/interruption/index.ts +4 -0
  170. package/src/inference/interruption/interruption.ts +88 -0
  171. package/src/inference/llm.ts +42 -3
  172. package/src/inference/utils.test.ts +31 -0
  173. package/src/inference/utils.ts +15 -0
  174. package/src/ipc/job_proc_lazy_main.ts +1 -1
  175. package/src/llm/chat_context.ts +32 -2
  176. package/src/llm/llm.ts +1 -0
  177. package/src/llm/provider_format/openai.test.ts +40 -0
  178. package/src/llm/provider_format/openai.ts +46 -19
  179. package/src/llm/provider_format/utils.ts +5 -1
  180. package/src/stream/stream_channel.ts +6 -2
  181. package/src/telemetry/trace_types.ts +7 -0
  182. package/src/utils/ws_transport.test.ts +282 -0
  183. package/src/utils/ws_transport.ts +22 -0
  184. package/src/voice/agent_activity.ts +61 -0
  185. package/src/voice/agent_session.ts +22 -2
  186. package/src/voice/audio_recognition.ts +161 -1
  187. package/src/voice/generation.ts +1 -0
@@ -0,0 +1,9 @@
1
+ /// <reference types="node" resolution-mode="require"/>
2
+ /// <reference types="node" resolution-mode="require"/>
3
+ /// <reference types="node" resolution-mode="require"/>
4
+ export declare function webSocketStream(wsUrl: string): {
5
+ readable: import("stream/web").ReadableStream<any>;
6
+ writable: import("stream/web").WritableStream<any>;
7
+ close: (code?: number | undefined, data?: string | Buffer | undefined) => void;
8
+ };
9
+ //# sourceMappingURL=ws_transport.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ws_transport.d.ts","sourceRoot":"","sources":["../../src/utils/ws_transport.ts"],"names":[],"mappings":";;;AAGA,wBAAgB,eAAe,CAAC,KAAK,EAAE,MAAM;;;;EAkB5C"}
@@ -0,0 +1,17 @@
1
+ import { Readable, Writable } from "node:stream";
2
+ import WebSocket, { createWebSocketStream } from "ws";
3
+ function webSocketStream(wsUrl) {
4
+ const ws = new WebSocket(wsUrl);
5
+ const duplex = createWebSocketStream(ws);
6
+ duplex.on("error", console.error);
7
+ duplex.on("end", () => {
8
+ duplex.end();
9
+ });
10
+ const writable = Writable.toWeb(duplex);
11
+ const readable = Readable.toWeb(duplex);
12
+ return { readable, writable, close: ws.close };
13
+ }
14
+ export {
15
+ webSocketStream
16
+ };
17
+ //# sourceMappingURL=ws_transport.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/utils/ws_transport.ts"],"sourcesContent":["import { Readable, Writable } from 'node:stream';\nimport WebSocket, { createWebSocketStream } from 'ws';\n\nexport function webSocketStream(wsUrl: string) {\n const ws = new WebSocket(wsUrl);\n const duplex = createWebSocketStream(ws);\n duplex.on('error', console.error);\n\n // End the write side when the read side ends to properly close the stream.\n // This is needed because Readable.toWeb() waits for both sides of the duplex\n // to close before signaling done on the ReadableStream.\n duplex.on('end', () => {\n duplex.end();\n });\n\n // Convert the writable side\n const writable = Writable.toWeb(duplex);\n // Convert the readable side\n const readable = Readable.toWeb(duplex);\n\n return { readable, writable, close: ws.close };\n}\n"],"mappings":"AAAA,SAAS,UAAU,gBAAgB;AACnC,OAAO,aAAa,6BAA6B;AAE1C,SAAS,gBAAgB,OAAe;AAC7C,QAAM,KAAK,IAAI,UAAU,KAAK;AAC9B,QAAM,SAAS,sBAAsB,EAAE;AACvC,SAAO,GAAG,SAAS,QAAQ,KAAK;AAKhC,SAAO,GAAG,OAAO,MAAM;AACrB,WAAO,IAAI;AAAA,EACb,CAAC;AAGD,QAAM,WAAW,SAAS,MAAM,MAAM;AAEtC,QAAM,WAAW,SAAS,MAAM,MAAM;AAEtC,SAAO,EAAE,UAAU,UAAU,OAAO,GAAG,MAAM;AAC/C;","names":[]}
@@ -0,0 +1,212 @@
1
+ "use strict";
2
+ var import_vitest = require("vitest");
3
+ var import_ws = require("ws");
4
+ var import_ws_transport = require("./ws_transport.cjs");
5
+ (0, import_vitest.describe)("webSocketStream", () => {
6
+ (0, import_vitest.describe)("readable stream", () => {
7
+ (0, import_vitest.it)("receives messages from the WebSocket", async () => {
8
+ const wss = await new Promise((resolve) => {
9
+ const server = new import_ws.WebSocketServer({ port: 0 }, () => resolve(server));
10
+ });
11
+ const port = wss.address().port;
12
+ wss.on("connection", (serverWs) => {
13
+ serverWs.send("hello");
14
+ serverWs.send("world");
15
+ serverWs.close();
16
+ });
17
+ const { readable } = (0, import_ws_transport.webSocketStream)(`ws://localhost:${port}`);
18
+ const reader = readable.getReader();
19
+ const messages = [];
20
+ try {
21
+ while (true) {
22
+ const { done, value } = await reader.read();
23
+ if (done) break;
24
+ messages.push(Buffer.from(value).toString());
25
+ }
26
+ } finally {
27
+ reader.releaseLock();
28
+ }
29
+ (0, import_vitest.expect)(messages).toEqual(["hello", "world"]);
30
+ wss.close();
31
+ });
32
+ (0, import_vitest.it)("handles binary messages", async () => {
33
+ const wss = await new Promise((resolve) => {
34
+ const server = new import_ws.WebSocketServer({ port: 0 }, () => resolve(server));
35
+ });
36
+ const port = wss.address().port;
37
+ const binaryData = new Uint8Array([1, 2, 3, 4, 5]);
38
+ wss.on("connection", (serverWs) => {
39
+ serverWs.send(binaryData);
40
+ serverWs.close();
41
+ });
42
+ const { readable } = (0, import_ws_transport.webSocketStream)(`ws://localhost:${port}`);
43
+ const reader = readable.getReader();
44
+ const chunks = [];
45
+ try {
46
+ while (true) {
47
+ const { done, value } = await reader.read();
48
+ if (done) break;
49
+ chunks.push(new Uint8Array(value));
50
+ }
51
+ } finally {
52
+ reader.releaseLock();
53
+ }
54
+ (0, import_vitest.expect)(chunks).toHaveLength(1);
55
+ (0, import_vitest.expect)(Array.from(chunks[0])).toEqual([1, 2, 3, 4, 5]);
56
+ wss.close();
57
+ });
58
+ (0, import_vitest.it)("handles empty stream when connection closes immediately", async () => {
59
+ const wss = await new Promise((resolve) => {
60
+ const server = new import_ws.WebSocketServer({ port: 0 }, () => resolve(server));
61
+ });
62
+ const port = wss.address().port;
63
+ wss.on("connection", (serverWs) => {
64
+ serverWs.close();
65
+ });
66
+ const { readable } = (0, import_ws_transport.webSocketStream)(`ws://localhost:${port}`);
67
+ const reader = readable.getReader();
68
+ const chunks = [];
69
+ try {
70
+ while (true) {
71
+ const { done, value } = await reader.read();
72
+ if (done) break;
73
+ chunks.push(value);
74
+ }
75
+ } finally {
76
+ reader.releaseLock();
77
+ }
78
+ (0, import_vitest.expect)(chunks).toEqual([]);
79
+ wss.close();
80
+ });
81
+ });
82
+ (0, import_vitest.describe)("writable stream", () => {
83
+ (0, import_vitest.it)("sends messages through the WebSocket", async () => {
84
+ const wss = await new Promise((resolve) => {
85
+ const server = new import_ws.WebSocketServer({ port: 0 }, () => resolve(server));
86
+ });
87
+ const port = wss.address().port;
88
+ const ws = new import_ws.WebSocket(`ws://localhost:${port}`);
89
+ const connected = new Promise((resolve) => {
90
+ ws.on("open", resolve);
91
+ });
92
+ const messagesReceived = [];
93
+ const serverClosed = new Promise((resolve) => {
94
+ wss.on("connection", (serverWs) => {
95
+ serverWs.on("message", (data) => {
96
+ messagesReceived.push(data.toString());
97
+ });
98
+ serverWs.on("close", resolve);
99
+ });
100
+ });
101
+ await connected;
102
+ const { writable } = (0, import_ws_transport.webSocketStream)(`ws://localhost:${port}`);
103
+ const writer = writable.getWriter();
104
+ await writer.write(new TextEncoder().encode("hello"));
105
+ await writer.write(new TextEncoder().encode("world"));
106
+ await writer.close();
107
+ await serverClosed;
108
+ (0, import_vitest.expect)(messagesReceived).toEqual(["hello", "world"]);
109
+ wss.close();
110
+ });
111
+ (0, import_vitest.it)("sends binary data through the WebSocket", async () => {
112
+ const wss = await new Promise((resolve) => {
113
+ const server = new import_ws.WebSocketServer({ port: 0 }, () => resolve(server));
114
+ });
115
+ const port = wss.address().port;
116
+ const chunksReceived = [];
117
+ const serverClosed = new Promise((resolve) => {
118
+ wss.on("connection", (serverWs) => {
119
+ serverWs.on("message", (data) => {
120
+ chunksReceived.push(Buffer.from(data));
121
+ });
122
+ serverWs.on("close", resolve);
123
+ });
124
+ });
125
+ const { writable } = (0, import_ws_transport.webSocketStream)(`ws://localhost:${port}`);
126
+ const writer = writable.getWriter();
127
+ const binaryData = new Uint8Array([10, 20, 30, 40, 50]);
128
+ await writer.write(binaryData);
129
+ await writer.close();
130
+ await serverClosed;
131
+ (0, import_vitest.expect)(chunksReceived).toHaveLength(1);
132
+ (0, import_vitest.expect)(Array.from(chunksReceived[0])).toEqual([10, 20, 30, 40, 50]);
133
+ wss.close();
134
+ });
135
+ (0, import_vitest.it)("buffers writes if readyState is CONNECTING", async () => {
136
+ const wss = await new Promise((resolve) => {
137
+ const server = new import_ws.WebSocketServer({ port: 0 }, () => resolve(server));
138
+ });
139
+ const port = wss.address().port;
140
+ const { writable } = (0, import_ws_transport.webSocketStream)(`ws://localhost:${port}`);
141
+ const writer = writable.getWriter();
142
+ const messagesReceived = [];
143
+ const serverClosed = new Promise((resolve) => {
144
+ wss.on("connection", (serverWs) => {
145
+ serverWs.on("message", (data) => {
146
+ messagesReceived.push(data.toString());
147
+ });
148
+ serverWs.on("close", resolve);
149
+ });
150
+ });
151
+ await writer.write(new TextEncoder().encode("buffered message"));
152
+ await writer.close();
153
+ await serverClosed;
154
+ (0, import_vitest.expect)(messagesReceived).toEqual(["buffered message"]);
155
+ wss.close();
156
+ });
157
+ });
158
+ (0, import_vitest.describe)("bidirectional communication", () => {
159
+ (0, import_vitest.it)("supports echo pattern with readable and writable", async () => {
160
+ const wss = await new Promise((resolve) => {
161
+ const server = new import_ws.WebSocketServer({ port: 0 }, () => resolve(server));
162
+ });
163
+ const port = wss.address().port;
164
+ wss.on("connection", (serverWs) => {
165
+ serverWs.on("message", (data) => {
166
+ serverWs.send(data);
167
+ });
168
+ });
169
+ const { readable, writable } = (0, import_ws_transport.webSocketStream)(`ws://localhost:${port}`);
170
+ const writer = writable.getWriter();
171
+ const reader = readable.getReader();
172
+ await writer.write(new TextEncoder().encode("ping1"));
173
+ await writer.write(new TextEncoder().encode("ping2"));
174
+ const { value: response1 } = await reader.read();
175
+ const { value: response2 } = await reader.read();
176
+ (0, import_vitest.expect)(Buffer.from(response1).toString()).toBe("ping1");
177
+ (0, import_vitest.expect)(Buffer.from(response2).toString()).toBe("ping2");
178
+ reader.releaseLock();
179
+ await writer.close();
180
+ wss.close();
181
+ });
182
+ });
183
+ (0, import_vitest.describe)("error handling", () => {
184
+ (0, import_vitest.it)("readable stream ends when WebSocket closes unexpectedly", async () => {
185
+ const wss = await new Promise((resolve) => {
186
+ const server = new import_ws.WebSocketServer({ port: 0 }, () => resolve(server));
187
+ });
188
+ const port = wss.address().port;
189
+ wss.on("connection", (serverWs) => {
190
+ serverWs.send("before close");
191
+ serverWs.terminate();
192
+ });
193
+ const { readable } = (0, import_ws_transport.webSocketStream)(`ws://localhost:${port}`);
194
+ const reader = readable.getReader();
195
+ const chunks = [];
196
+ try {
197
+ while (true) {
198
+ const { done, value } = await reader.read();
199
+ if (done) break;
200
+ chunks.push(Buffer.from(value).toString());
201
+ }
202
+ } catch (error) {
203
+ console.error(error);
204
+ } finally {
205
+ reader.releaseLock();
206
+ }
207
+ (0, import_vitest.expect)(chunks).toContain("before close");
208
+ wss.close();
209
+ });
210
+ });
211
+ });
212
+ //# sourceMappingURL=ws_transport.test.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/utils/ws_transport.test.ts"],"sourcesContent":["// SPDX-FileCopyrightText: 2024 LiveKit, Inc.\n//\n// SPDX-License-Identifier: Apache-2.0\nimport { describe, expect, it } from 'vitest';\nimport { WebSocket, WebSocketServer } from 'ws';\nimport { webSocketStream } from './ws_transport.js';\n\ndescribe('webSocketStream', () => {\n describe('readable stream', () => {\n it('receives messages from the WebSocket', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n\n wss.on('connection', (serverWs) => {\n serverWs.send('hello');\n serverWs.send('world');\n serverWs.close();\n });\n\n const { readable } = webSocketStream(`ws://localhost:${port}`);\n const reader = readable.getReader();\n\n const messages: string[] = [];\n try {\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n messages.push(Buffer.from(value).toString());\n }\n } finally {\n reader.releaseLock();\n }\n\n expect(messages).toEqual(['hello', 'world']);\n\n wss.close();\n });\n\n it('handles binary messages', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n\n const binaryData = new Uint8Array([1, 2, 3, 4, 5]);\n\n wss.on('connection', (serverWs) => {\n serverWs.send(binaryData);\n serverWs.close();\n });\n\n const { readable } = webSocketStream(`ws://localhost:${port}`);\n const reader = readable.getReader();\n\n const chunks: Uint8Array[] = [];\n try {\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n chunks.push(new Uint8Array(value));\n }\n } finally {\n reader.releaseLock();\n }\n\n expect(chunks).toHaveLength(1);\n expect(Array.from(chunks[0]!)).toEqual([1, 2, 3, 4, 5]);\n\n wss.close();\n });\n\n it('handles empty stream when connection closes immediately', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n\n wss.on('connection', (serverWs) => {\n serverWs.close();\n });\n const { readable } = webSocketStream(`ws://localhost:${port}`);\n const reader = readable.getReader();\n\n const chunks: Uint8Array[] = [];\n try {\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n chunks.push(value);\n }\n } finally {\n reader.releaseLock();\n }\n\n expect(chunks).toEqual([]);\n\n wss.close();\n });\n });\n\n describe('writable stream', () => {\n it('sends messages through the WebSocket', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n const ws = new WebSocket(`ws://localhost:${port}`);\n\n const connected = new Promise<void>((resolve) => {\n ws.on('open', resolve);\n });\n\n const messagesReceived: string[] = [];\n const serverClosed = new Promise<void>((resolve) => {\n wss.on('connection', (serverWs) => {\n serverWs.on('message', (data) => {\n messagesReceived.push(data.toString());\n });\n serverWs.on('close', resolve);\n });\n });\n\n await connected;\n const { writable } = webSocketStream(`ws://localhost:${port}`);\n const writer = writable.getWriter();\n\n await writer.write(new TextEncoder().encode('hello'));\n await writer.write(new TextEncoder().encode('world'));\n await writer.close();\n\n await serverClosed;\n\n expect(messagesReceived).toEqual(['hello', 'world']);\n\n wss.close();\n });\n\n it('sends binary data through the WebSocket', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n\n const chunksReceived: Buffer[] = [];\n const serverClosed = new Promise<void>((resolve) => {\n wss.on('connection', (serverWs) => {\n serverWs.on('message', (data) => {\n chunksReceived.push(Buffer.from(data as Buffer));\n });\n serverWs.on('close', resolve);\n });\n });\n\n const { writable } = webSocketStream(`ws://localhost:${port}`);\n const writer = writable.getWriter();\n\n const binaryData = new Uint8Array([10, 20, 30, 40, 50]);\n await writer.write(binaryData);\n await writer.close();\n\n await serverClosed;\n\n expect(chunksReceived).toHaveLength(1);\n expect(Array.from(chunksReceived[0]!)).toEqual([10, 20, 30, 40, 50]);\n\n wss.close();\n });\n\n it('buffers writes if readyState is CONNECTING', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n\n const { writable } = webSocketStream(`ws://localhost:${port}`);\n const writer = writable.getWriter();\n\n const messagesReceived: string[] = [];\n const serverClosed = new Promise<void>((resolve) => {\n wss.on('connection', (serverWs) => {\n serverWs.on('message', (data) => {\n messagesReceived.push(data.toString());\n });\n serverWs.on('close', resolve);\n });\n });\n\n // These writes should be buffered\n await writer.write(new TextEncoder().encode('buffered message'));\n await writer.close();\n\n await serverClosed;\n\n expect(messagesReceived).toEqual(['buffered message']);\n\n wss.close();\n });\n });\n\n describe('bidirectional communication', () => {\n it('supports echo pattern with readable and writable', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n\n // Server echoes messages back\n wss.on('connection', (serverWs) => {\n serverWs.on('message', (data) => {\n serverWs.send(data);\n });\n });\n\n const { readable, writable } = webSocketStream(`ws://localhost:${port}`);\n const writer = writable.getWriter();\n const reader = readable.getReader();\n\n // Send messages\n await writer.write(new TextEncoder().encode('ping1'));\n await writer.write(new TextEncoder().encode('ping2'));\n\n // Read echoed responses\n const { value: response1 } = await reader.read();\n const { value: response2 } = await reader.read();\n\n expect(Buffer.from(response1!).toString()).toBe('ping1');\n expect(Buffer.from(response2!).toString()).toBe('ping2');\n\n reader.releaseLock();\n await writer.close();\n\n wss.close();\n });\n });\n\n describe('error handling', () => {\n it('readable stream ends when WebSocket closes unexpectedly', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n\n wss.on('connection', (serverWs) => {\n serverWs.send('before close');\n // Terminate connection abruptly\n serverWs.terminate();\n });\n\n const { readable } = webSocketStream(`ws://localhost:${port}`);\n const reader = readable.getReader();\n\n const chunks: string[] = [];\n try {\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n chunks.push(Buffer.from(value).toString());\n }\n } catch (error) {\n console.error(error);\n // Connection terminated, stream may error\n } finally {\n reader.releaseLock();\n }\n\n // Should have received the message sent before termination\n expect(chunks).toContain('before close');\n\n wss.close();\n });\n });\n});\n"],"mappings":"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","names":[]}
@@ -0,0 +1,211 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import { WebSocket, WebSocketServer } from "ws";
3
+ import { webSocketStream } from "./ws_transport.js";
4
+ describe("webSocketStream", () => {
5
+ describe("readable stream", () => {
6
+ it("receives messages from the WebSocket", async () => {
7
+ const wss = await new Promise((resolve) => {
8
+ const server = new WebSocketServer({ port: 0 }, () => resolve(server));
9
+ });
10
+ const port = wss.address().port;
11
+ wss.on("connection", (serverWs) => {
12
+ serverWs.send("hello");
13
+ serverWs.send("world");
14
+ serverWs.close();
15
+ });
16
+ const { readable } = webSocketStream(`ws://localhost:${port}`);
17
+ const reader = readable.getReader();
18
+ const messages = [];
19
+ try {
20
+ while (true) {
21
+ const { done, value } = await reader.read();
22
+ if (done) break;
23
+ messages.push(Buffer.from(value).toString());
24
+ }
25
+ } finally {
26
+ reader.releaseLock();
27
+ }
28
+ expect(messages).toEqual(["hello", "world"]);
29
+ wss.close();
30
+ });
31
+ it("handles binary messages", async () => {
32
+ const wss = await new Promise((resolve) => {
33
+ const server = new WebSocketServer({ port: 0 }, () => resolve(server));
34
+ });
35
+ const port = wss.address().port;
36
+ const binaryData = new Uint8Array([1, 2, 3, 4, 5]);
37
+ wss.on("connection", (serverWs) => {
38
+ serverWs.send(binaryData);
39
+ serverWs.close();
40
+ });
41
+ const { readable } = webSocketStream(`ws://localhost:${port}`);
42
+ const reader = readable.getReader();
43
+ const chunks = [];
44
+ try {
45
+ while (true) {
46
+ const { done, value } = await reader.read();
47
+ if (done) break;
48
+ chunks.push(new Uint8Array(value));
49
+ }
50
+ } finally {
51
+ reader.releaseLock();
52
+ }
53
+ expect(chunks).toHaveLength(1);
54
+ expect(Array.from(chunks[0])).toEqual([1, 2, 3, 4, 5]);
55
+ wss.close();
56
+ });
57
+ it("handles empty stream when connection closes immediately", async () => {
58
+ const wss = await new Promise((resolve) => {
59
+ const server = new WebSocketServer({ port: 0 }, () => resolve(server));
60
+ });
61
+ const port = wss.address().port;
62
+ wss.on("connection", (serverWs) => {
63
+ serverWs.close();
64
+ });
65
+ const { readable } = webSocketStream(`ws://localhost:${port}`);
66
+ const reader = readable.getReader();
67
+ const chunks = [];
68
+ try {
69
+ while (true) {
70
+ const { done, value } = await reader.read();
71
+ if (done) break;
72
+ chunks.push(value);
73
+ }
74
+ } finally {
75
+ reader.releaseLock();
76
+ }
77
+ expect(chunks).toEqual([]);
78
+ wss.close();
79
+ });
80
+ });
81
+ describe("writable stream", () => {
82
+ it("sends messages through the WebSocket", async () => {
83
+ const wss = await new Promise((resolve) => {
84
+ const server = new WebSocketServer({ port: 0 }, () => resolve(server));
85
+ });
86
+ const port = wss.address().port;
87
+ const ws = new WebSocket(`ws://localhost:${port}`);
88
+ const connected = new Promise((resolve) => {
89
+ ws.on("open", resolve);
90
+ });
91
+ const messagesReceived = [];
92
+ const serverClosed = new Promise((resolve) => {
93
+ wss.on("connection", (serverWs) => {
94
+ serverWs.on("message", (data) => {
95
+ messagesReceived.push(data.toString());
96
+ });
97
+ serverWs.on("close", resolve);
98
+ });
99
+ });
100
+ await connected;
101
+ const { writable } = webSocketStream(`ws://localhost:${port}`);
102
+ const writer = writable.getWriter();
103
+ await writer.write(new TextEncoder().encode("hello"));
104
+ await writer.write(new TextEncoder().encode("world"));
105
+ await writer.close();
106
+ await serverClosed;
107
+ expect(messagesReceived).toEqual(["hello", "world"]);
108
+ wss.close();
109
+ });
110
+ it("sends binary data through the WebSocket", async () => {
111
+ const wss = await new Promise((resolve) => {
112
+ const server = new WebSocketServer({ port: 0 }, () => resolve(server));
113
+ });
114
+ const port = wss.address().port;
115
+ const chunksReceived = [];
116
+ const serverClosed = new Promise((resolve) => {
117
+ wss.on("connection", (serverWs) => {
118
+ serverWs.on("message", (data) => {
119
+ chunksReceived.push(Buffer.from(data));
120
+ });
121
+ serverWs.on("close", resolve);
122
+ });
123
+ });
124
+ const { writable } = webSocketStream(`ws://localhost:${port}`);
125
+ const writer = writable.getWriter();
126
+ const binaryData = new Uint8Array([10, 20, 30, 40, 50]);
127
+ await writer.write(binaryData);
128
+ await writer.close();
129
+ await serverClosed;
130
+ expect(chunksReceived).toHaveLength(1);
131
+ expect(Array.from(chunksReceived[0])).toEqual([10, 20, 30, 40, 50]);
132
+ wss.close();
133
+ });
134
+ it("buffers writes if readyState is CONNECTING", async () => {
135
+ const wss = await new Promise((resolve) => {
136
+ const server = new WebSocketServer({ port: 0 }, () => resolve(server));
137
+ });
138
+ const port = wss.address().port;
139
+ const { writable } = webSocketStream(`ws://localhost:${port}`);
140
+ const writer = writable.getWriter();
141
+ const messagesReceived = [];
142
+ const serverClosed = new Promise((resolve) => {
143
+ wss.on("connection", (serverWs) => {
144
+ serverWs.on("message", (data) => {
145
+ messagesReceived.push(data.toString());
146
+ });
147
+ serverWs.on("close", resolve);
148
+ });
149
+ });
150
+ await writer.write(new TextEncoder().encode("buffered message"));
151
+ await writer.close();
152
+ await serverClosed;
153
+ expect(messagesReceived).toEqual(["buffered message"]);
154
+ wss.close();
155
+ });
156
+ });
157
+ describe("bidirectional communication", () => {
158
+ it("supports echo pattern with readable and writable", async () => {
159
+ const wss = await new Promise((resolve) => {
160
+ const server = new WebSocketServer({ port: 0 }, () => resolve(server));
161
+ });
162
+ const port = wss.address().port;
163
+ wss.on("connection", (serverWs) => {
164
+ serverWs.on("message", (data) => {
165
+ serverWs.send(data);
166
+ });
167
+ });
168
+ const { readable, writable } = webSocketStream(`ws://localhost:${port}`);
169
+ const writer = writable.getWriter();
170
+ const reader = readable.getReader();
171
+ await writer.write(new TextEncoder().encode("ping1"));
172
+ await writer.write(new TextEncoder().encode("ping2"));
173
+ const { value: response1 } = await reader.read();
174
+ const { value: response2 } = await reader.read();
175
+ expect(Buffer.from(response1).toString()).toBe("ping1");
176
+ expect(Buffer.from(response2).toString()).toBe("ping2");
177
+ reader.releaseLock();
178
+ await writer.close();
179
+ wss.close();
180
+ });
181
+ });
182
+ describe("error handling", () => {
183
+ it("readable stream ends when WebSocket closes unexpectedly", async () => {
184
+ const wss = await new Promise((resolve) => {
185
+ const server = new WebSocketServer({ port: 0 }, () => resolve(server));
186
+ });
187
+ const port = wss.address().port;
188
+ wss.on("connection", (serverWs) => {
189
+ serverWs.send("before close");
190
+ serverWs.terminate();
191
+ });
192
+ const { readable } = webSocketStream(`ws://localhost:${port}`);
193
+ const reader = readable.getReader();
194
+ const chunks = [];
195
+ try {
196
+ while (true) {
197
+ const { done, value } = await reader.read();
198
+ if (done) break;
199
+ chunks.push(Buffer.from(value).toString());
200
+ }
201
+ } catch (error) {
202
+ console.error(error);
203
+ } finally {
204
+ reader.releaseLock();
205
+ }
206
+ expect(chunks).toContain("before close");
207
+ wss.close();
208
+ });
209
+ });
210
+ });
211
+ //# sourceMappingURL=ws_transport.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/utils/ws_transport.test.ts"],"sourcesContent":["// SPDX-FileCopyrightText: 2024 LiveKit, Inc.\n//\n// SPDX-License-Identifier: Apache-2.0\nimport { describe, expect, it } from 'vitest';\nimport { WebSocket, WebSocketServer } from 'ws';\nimport { webSocketStream } from './ws_transport.js';\n\ndescribe('webSocketStream', () => {\n describe('readable stream', () => {\n it('receives messages from the WebSocket', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n\n wss.on('connection', (serverWs) => {\n serverWs.send('hello');\n serverWs.send('world');\n serverWs.close();\n });\n\n const { readable } = webSocketStream(`ws://localhost:${port}`);\n const reader = readable.getReader();\n\n const messages: string[] = [];\n try {\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n messages.push(Buffer.from(value).toString());\n }\n } finally {\n reader.releaseLock();\n }\n\n expect(messages).toEqual(['hello', 'world']);\n\n wss.close();\n });\n\n it('handles binary messages', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n\n const binaryData = new Uint8Array([1, 2, 3, 4, 5]);\n\n wss.on('connection', (serverWs) => {\n serverWs.send(binaryData);\n serverWs.close();\n });\n\n const { readable } = webSocketStream(`ws://localhost:${port}`);\n const reader = readable.getReader();\n\n const chunks: Uint8Array[] = [];\n try {\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n chunks.push(new Uint8Array(value));\n }\n } finally {\n reader.releaseLock();\n }\n\n expect(chunks).toHaveLength(1);\n expect(Array.from(chunks[0]!)).toEqual([1, 2, 3, 4, 5]);\n\n wss.close();\n });\n\n it('handles empty stream when connection closes immediately', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n\n wss.on('connection', (serverWs) => {\n serverWs.close();\n });\n const { readable } = webSocketStream(`ws://localhost:${port}`);\n const reader = readable.getReader();\n\n const chunks: Uint8Array[] = [];\n try {\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n chunks.push(value);\n }\n } finally {\n reader.releaseLock();\n }\n\n expect(chunks).toEqual([]);\n\n wss.close();\n });\n });\n\n describe('writable stream', () => {\n it('sends messages through the WebSocket', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n const ws = new WebSocket(`ws://localhost:${port}`);\n\n const connected = new Promise<void>((resolve) => {\n ws.on('open', resolve);\n });\n\n const messagesReceived: string[] = [];\n const serverClosed = new Promise<void>((resolve) => {\n wss.on('connection', (serverWs) => {\n serverWs.on('message', (data) => {\n messagesReceived.push(data.toString());\n });\n serverWs.on('close', resolve);\n });\n });\n\n await connected;\n const { writable } = webSocketStream(`ws://localhost:${port}`);\n const writer = writable.getWriter();\n\n await writer.write(new TextEncoder().encode('hello'));\n await writer.write(new TextEncoder().encode('world'));\n await writer.close();\n\n await serverClosed;\n\n expect(messagesReceived).toEqual(['hello', 'world']);\n\n wss.close();\n });\n\n it('sends binary data through the WebSocket', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n\n const chunksReceived: Buffer[] = [];\n const serverClosed = new Promise<void>((resolve) => {\n wss.on('connection', (serverWs) => {\n serverWs.on('message', (data) => {\n chunksReceived.push(Buffer.from(data as Buffer));\n });\n serverWs.on('close', resolve);\n });\n });\n\n const { writable } = webSocketStream(`ws://localhost:${port}`);\n const writer = writable.getWriter();\n\n const binaryData = new Uint8Array([10, 20, 30, 40, 50]);\n await writer.write(binaryData);\n await writer.close();\n\n await serverClosed;\n\n expect(chunksReceived).toHaveLength(1);\n expect(Array.from(chunksReceived[0]!)).toEqual([10, 20, 30, 40, 50]);\n\n wss.close();\n });\n\n it('buffers writes if readyState is CONNECTING', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n\n const { writable } = webSocketStream(`ws://localhost:${port}`);\n const writer = writable.getWriter();\n\n const messagesReceived: string[] = [];\n const serverClosed = new Promise<void>((resolve) => {\n wss.on('connection', (serverWs) => {\n serverWs.on('message', (data) => {\n messagesReceived.push(data.toString());\n });\n serverWs.on('close', resolve);\n });\n });\n\n // These writes should be buffered\n await writer.write(new TextEncoder().encode('buffered message'));\n await writer.close();\n\n await serverClosed;\n\n expect(messagesReceived).toEqual(['buffered message']);\n\n wss.close();\n });\n });\n\n describe('bidirectional communication', () => {\n it('supports echo pattern with readable and writable', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n\n // Server echoes messages back\n wss.on('connection', (serverWs) => {\n serverWs.on('message', (data) => {\n serverWs.send(data);\n });\n });\n\n const { readable, writable } = webSocketStream(`ws://localhost:${port}`);\n const writer = writable.getWriter();\n const reader = readable.getReader();\n\n // Send messages\n await writer.write(new TextEncoder().encode('ping1'));\n await writer.write(new TextEncoder().encode('ping2'));\n\n // Read echoed responses\n const { value: response1 } = await reader.read();\n const { value: response2 } = await reader.read();\n\n expect(Buffer.from(response1!).toString()).toBe('ping1');\n expect(Buffer.from(response2!).toString()).toBe('ping2');\n\n reader.releaseLock();\n await writer.close();\n\n wss.close();\n });\n });\n\n describe('error handling', () => {\n it('readable stream ends when WebSocket closes unexpectedly', async () => {\n const wss = await new Promise<WebSocketServer>((resolve) => {\n const server: WebSocketServer = new WebSocketServer({ port: 0 }, () => resolve(server));\n });\n\n const port = (wss.address() as { port: number }).port;\n\n wss.on('connection', (serverWs) => {\n serverWs.send('before close');\n // Terminate connection abruptly\n serverWs.terminate();\n });\n\n const { readable } = webSocketStream(`ws://localhost:${port}`);\n const reader = readable.getReader();\n\n const chunks: string[] = [];\n try {\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n chunks.push(Buffer.from(value).toString());\n }\n } catch (error) {\n console.error(error);\n // Connection terminated, stream may error\n } finally {\n reader.releaseLock();\n }\n\n // Should have received the message sent before termination\n expect(chunks).toContain('before close');\n\n wss.close();\n });\n });\n});\n"],"mappings":"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","names":[]}
@@ -36,6 +36,7 @@ var import_telemetry = require("../telemetry/index.cjs");
36
36
  var import_word = require("../tokenize/basic/word.cjs");
37
37
  var import_tts = require("../tts/tts.cjs");
38
38
  var import_utils = require("../utils.cjs");
39
+ var import_interruption = require("../inference/interruption/interruption.cjs");
39
40
  var import_vad = require("../vad.cjs");
40
41
  var import_agent = require("./agent.cjs");
41
42
  var import_agent_session = require("./agent_session.cjs");
@@ -69,6 +70,24 @@ class AgentActivity {
69
70
  /** @internal */
70
71
  _mainTask;
71
72
  _userTurnCompletedTask;
73
+ /**
74
+ * Notify that agent started speaking.
75
+ * This enables interruption detection in AudioRecognition.
76
+ * @internal
77
+ */
78
+ notifyAgentSpeechStarted() {
79
+ var _a;
80
+ (_a = this.audioRecognition) == null ? void 0 : _a.onStartOfAgentSpeech();
81
+ }
82
+ /**
83
+ * Notify that agent stopped speaking.
84
+ * This disables interruption detection in AudioRecognition.
85
+ * @internal
86
+ */
87
+ notifyAgentSpeechEnded() {
88
+ var _a;
89
+ (_a = this.audioRecognition) == null ? void 0 : _a.onEndOfAgentSpeech();
90
+ }
72
91
  constructor(agent, agentSession) {
73
92
  this.agent = agent;
74
93
  this.agentSession = agentSession;
@@ -201,6 +220,7 @@ class AgentActivity {
201
220
  // Disable stt node if stt is not provided
202
221
  stt: this.stt ? (...args) => this.agent.sttNode(...args) : void 0,
203
222
  vad: this.vad,
223
+ interruptionDetector: this.agentSession.interruptionDetector,
204
224
  turnDetector: typeof this.turnDetection === "string" ? void 0 : this.turnDetection,
205
225
  turnDetectionMode: this.turnDetectionMode,
206
226
  minEndpointingDelay: this.agentSession.options.minEndpointingDelay,
@@ -494,6 +514,35 @@ class AgentActivity {
494
514
  this._currentSpeech.interrupt();
495
515
  }
496
516
  }
517
+ onInterruption(ev) {
518
+ var _a, _b;
519
+ if (ev.type !== import_interruption.InterruptionEventType.INTERRUPTION) {
520
+ return;
521
+ }
522
+ this.logger.info(
523
+ {
524
+ probability: ev.probability,
525
+ detectionDelay: ev.detectionDelay,
526
+ totalDuration: ev.totalDuration
527
+ },
528
+ "adaptive interruption detected"
529
+ );
530
+ if (this.turnDetection === "manual" || this.turnDetection === "realtime_llm") {
531
+ return;
532
+ }
533
+ if (this.llm instanceof import_llm.RealtimeModel && this.llm.capabilities.turnDetection) {
534
+ return;
535
+ }
536
+ (_a = this.realtimeSession) == null ? void 0 : _a.startUserActivity();
537
+ if (this._currentSpeech && !this._currentSpeech.interrupted && this._currentSpeech.allowInterruptions) {
538
+ this.logger.info(
539
+ { "speech id": this._currentSpeech.id },
540
+ "speech interrupted by adaptive interruption detector"
541
+ );
542
+ (_b = this.realtimeSession) == null ? void 0 : _b.interrupt();
543
+ this._currentSpeech.interrupt();
544
+ }
545
+ }
497
546
  onInterimTranscript(ev) {
498
547
  if (this.llm instanceof import_llm.RealtimeModel && this.llm.capabilities.userTranscription) {
499
548
  return;