@livekit/agents 1.0.47 → 1.1.0-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 (444) hide show
  1. package/dist/beta/index.cjs +29 -0
  2. package/dist/beta/index.cjs.map +1 -0
  3. package/dist/beta/index.d.cts +2 -0
  4. package/dist/beta/index.d.ts +2 -0
  5. package/dist/beta/index.d.ts.map +1 -0
  6. package/dist/beta/index.js +7 -0
  7. package/dist/beta/index.js.map +1 -0
  8. package/dist/beta/workflows/index.cjs +29 -0
  9. package/dist/beta/workflows/index.cjs.map +1 -0
  10. package/dist/beta/workflows/index.d.cts +2 -0
  11. package/dist/beta/workflows/index.d.ts +2 -0
  12. package/dist/beta/workflows/index.d.ts.map +1 -0
  13. package/dist/beta/workflows/index.js +7 -0
  14. package/dist/beta/workflows/index.js.map +1 -0
  15. package/dist/beta/workflows/task_group.cjs +162 -0
  16. package/dist/beta/workflows/task_group.cjs.map +1 -0
  17. package/dist/beta/workflows/task_group.d.cts +32 -0
  18. package/dist/beta/workflows/task_group.d.ts +32 -0
  19. package/dist/beta/workflows/task_group.d.ts.map +1 -0
  20. package/dist/beta/workflows/task_group.js +138 -0
  21. package/dist/beta/workflows/task_group.js.map +1 -0
  22. package/dist/constants.cjs +27 -0
  23. package/dist/constants.cjs.map +1 -1
  24. package/dist/constants.d.cts +9 -0
  25. package/dist/constants.d.ts +9 -0
  26. package/dist/constants.d.ts.map +1 -1
  27. package/dist/constants.js +18 -0
  28. package/dist/constants.js.map +1 -1
  29. package/dist/index.cjs +3 -0
  30. package/dist/index.cjs.map +1 -1
  31. package/dist/index.d.cts +2 -1
  32. package/dist/index.d.ts +2 -1
  33. package/dist/index.d.ts.map +1 -1
  34. package/dist/index.js +2 -0
  35. package/dist/index.js.map +1 -1
  36. package/dist/inference/api_protos.d.cts +12 -12
  37. package/dist/inference/api_protos.d.ts +12 -12
  38. package/dist/inference/interruption/defaults.cjs +81 -0
  39. package/dist/inference/interruption/defaults.cjs.map +1 -0
  40. package/dist/inference/interruption/defaults.d.cts +19 -0
  41. package/dist/inference/interruption/defaults.d.ts +19 -0
  42. package/dist/inference/interruption/defaults.d.ts.map +1 -0
  43. package/dist/inference/interruption/defaults.js +46 -0
  44. package/dist/inference/interruption/defaults.js.map +1 -0
  45. package/dist/inference/interruption/errors.cjs +44 -0
  46. package/dist/inference/interruption/errors.cjs.map +1 -0
  47. package/dist/inference/interruption/errors.d.cts +12 -0
  48. package/dist/inference/interruption/errors.d.ts +12 -0
  49. package/dist/inference/interruption/errors.d.ts.map +1 -0
  50. package/dist/inference/interruption/errors.js +20 -0
  51. package/dist/inference/interruption/errors.js.map +1 -0
  52. package/dist/inference/interruption/http_transport.cjs +147 -0
  53. package/dist/inference/interruption/http_transport.cjs.map +1 -0
  54. package/dist/inference/interruption/http_transport.d.cts +63 -0
  55. package/dist/inference/interruption/http_transport.d.ts +63 -0
  56. package/dist/inference/interruption/http_transport.d.ts.map +1 -0
  57. package/dist/inference/interruption/http_transport.js +121 -0
  58. package/dist/inference/interruption/http_transport.js.map +1 -0
  59. package/dist/inference/interruption/interruption_cache_entry.cjs +58 -0
  60. package/dist/inference/interruption/interruption_cache_entry.cjs.map +1 -0
  61. package/dist/inference/interruption/interruption_cache_entry.d.cts +30 -0
  62. package/dist/inference/interruption/interruption_cache_entry.d.ts +30 -0
  63. package/dist/inference/interruption/interruption_cache_entry.d.ts.map +1 -0
  64. package/dist/inference/interruption/interruption_cache_entry.js +34 -0
  65. package/dist/inference/interruption/interruption_cache_entry.js.map +1 -0
  66. package/dist/inference/interruption/interruption_detector.cjs +181 -0
  67. package/dist/inference/interruption/interruption_detector.cjs.map +1 -0
  68. package/dist/inference/interruption/interruption_detector.d.cts +59 -0
  69. package/dist/inference/interruption/interruption_detector.d.ts +59 -0
  70. package/dist/inference/interruption/interruption_detector.d.ts.map +1 -0
  71. package/dist/inference/interruption/interruption_detector.js +147 -0
  72. package/dist/inference/interruption/interruption_detector.js.map +1 -0
  73. package/dist/inference/interruption/interruption_stream.cjs +368 -0
  74. package/dist/inference/interruption/interruption_stream.cjs.map +1 -0
  75. package/dist/inference/interruption/interruption_stream.d.cts +46 -0
  76. package/dist/inference/interruption/interruption_stream.d.ts +46 -0
  77. package/dist/inference/interruption/interruption_stream.d.ts.map +1 -0
  78. package/dist/inference/interruption/interruption_stream.js +344 -0
  79. package/dist/inference/interruption/interruption_stream.js.map +1 -0
  80. package/dist/inference/interruption/types.cjs +17 -0
  81. package/dist/inference/interruption/types.cjs.map +1 -0
  82. package/dist/inference/interruption/types.d.cts +66 -0
  83. package/dist/inference/interruption/types.d.ts +66 -0
  84. package/dist/inference/interruption/types.d.ts.map +1 -0
  85. package/dist/inference/interruption/types.js +1 -0
  86. package/dist/inference/interruption/types.js.map +1 -0
  87. package/dist/inference/interruption/utils.cjs +130 -0
  88. package/dist/inference/interruption/utils.cjs.map +1 -0
  89. package/dist/inference/interruption/utils.d.cts +41 -0
  90. package/dist/inference/interruption/utils.d.ts +41 -0
  91. package/dist/inference/interruption/utils.d.ts.map +1 -0
  92. package/dist/inference/interruption/utils.js +105 -0
  93. package/dist/inference/interruption/utils.js.map +1 -0
  94. package/dist/inference/interruption/utils.test.cjs +105 -0
  95. package/dist/inference/interruption/utils.test.cjs.map +1 -0
  96. package/dist/inference/interruption/utils.test.js +104 -0
  97. package/dist/inference/interruption/utils.test.js.map +1 -0
  98. package/dist/inference/interruption/ws_transport.cjs +329 -0
  99. package/dist/inference/interruption/ws_transport.cjs.map +1 -0
  100. package/dist/inference/interruption/ws_transport.d.cts +33 -0
  101. package/dist/inference/interruption/ws_transport.d.ts +33 -0
  102. package/dist/inference/interruption/ws_transport.d.ts.map +1 -0
  103. package/dist/inference/interruption/ws_transport.js +295 -0
  104. package/dist/inference/interruption/ws_transport.js.map +1 -0
  105. package/dist/inference/llm.cjs +14 -10
  106. package/dist/inference/llm.cjs.map +1 -1
  107. package/dist/inference/llm.d.cts +2 -1
  108. package/dist/inference/llm.d.ts +2 -1
  109. package/dist/inference/llm.d.ts.map +1 -1
  110. package/dist/inference/llm.js +8 -10
  111. package/dist/inference/llm.js.map +1 -1
  112. package/dist/inference/stt.cjs +7 -2
  113. package/dist/inference/stt.cjs.map +1 -1
  114. package/dist/inference/stt.d.cts +2 -0
  115. package/dist/inference/stt.d.ts +2 -0
  116. package/dist/inference/stt.d.ts.map +1 -1
  117. package/dist/inference/stt.js +8 -3
  118. package/dist/inference/stt.js.map +1 -1
  119. package/dist/inference/tts.cjs +7 -2
  120. package/dist/inference/tts.cjs.map +1 -1
  121. package/dist/inference/tts.d.cts +2 -0
  122. package/dist/inference/tts.d.ts +2 -0
  123. package/dist/inference/tts.d.ts.map +1 -1
  124. package/dist/inference/tts.js +8 -3
  125. package/dist/inference/tts.js.map +1 -1
  126. package/dist/inference/utils.cjs +26 -7
  127. package/dist/inference/utils.cjs.map +1 -1
  128. package/dist/inference/utils.d.cts +13 -0
  129. package/dist/inference/utils.d.ts +13 -0
  130. package/dist/inference/utils.d.ts.map +1 -1
  131. package/dist/inference/utils.js +18 -2
  132. package/dist/inference/utils.js.map +1 -1
  133. package/dist/llm/chat_context.cjs +108 -2
  134. package/dist/llm/chat_context.cjs.map +1 -1
  135. package/dist/llm/chat_context.d.cts +28 -1
  136. package/dist/llm/chat_context.d.ts +28 -1
  137. package/dist/llm/chat_context.d.ts.map +1 -1
  138. package/dist/llm/chat_context.js +108 -2
  139. package/dist/llm/chat_context.js.map +1 -1
  140. package/dist/llm/chat_context.test.cjs +43 -0
  141. package/dist/llm/chat_context.test.cjs.map +1 -1
  142. package/dist/llm/chat_context.test.js +43 -0
  143. package/dist/llm/chat_context.test.js.map +1 -1
  144. package/dist/llm/index.cjs +2 -0
  145. package/dist/llm/index.cjs.map +1 -1
  146. package/dist/llm/index.d.cts +2 -2
  147. package/dist/llm/index.d.ts +2 -2
  148. package/dist/llm/index.d.ts.map +1 -1
  149. package/dist/llm/index.js +3 -1
  150. package/dist/llm/index.js.map +1 -1
  151. package/dist/llm/llm.cjs +16 -1
  152. package/dist/llm/llm.cjs.map +1 -1
  153. package/dist/llm/llm.d.cts +9 -0
  154. package/dist/llm/llm.d.ts +9 -0
  155. package/dist/llm/llm.d.ts.map +1 -1
  156. package/dist/llm/llm.js +16 -1
  157. package/dist/llm/llm.js.map +1 -1
  158. package/dist/llm/provider_format/index.d.cts +1 -1
  159. package/dist/llm/provider_format/index.d.ts +1 -1
  160. package/dist/llm/realtime.cjs +3 -0
  161. package/dist/llm/realtime.cjs.map +1 -1
  162. package/dist/llm/realtime.d.cts +1 -0
  163. package/dist/llm/realtime.d.ts +1 -0
  164. package/dist/llm/realtime.d.ts.map +1 -1
  165. package/dist/llm/realtime.js +3 -0
  166. package/dist/llm/realtime.js.map +1 -1
  167. package/dist/llm/tool_context.cjs +7 -0
  168. package/dist/llm/tool_context.cjs.map +1 -1
  169. package/dist/llm/tool_context.d.cts +10 -2
  170. package/dist/llm/tool_context.d.ts +10 -2
  171. package/dist/llm/tool_context.d.ts.map +1 -1
  172. package/dist/llm/tool_context.js +6 -0
  173. package/dist/llm/tool_context.js.map +1 -1
  174. package/dist/metrics/base.cjs.map +1 -1
  175. package/dist/metrics/base.d.cts +45 -1
  176. package/dist/metrics/base.d.ts +45 -1
  177. package/dist/metrics/base.d.ts.map +1 -1
  178. package/dist/metrics/index.cjs +5 -0
  179. package/dist/metrics/index.cjs.map +1 -1
  180. package/dist/metrics/index.d.cts +2 -1
  181. package/dist/metrics/index.d.ts +2 -1
  182. package/dist/metrics/index.d.ts.map +1 -1
  183. package/dist/metrics/index.js +6 -0
  184. package/dist/metrics/index.js.map +1 -1
  185. package/dist/metrics/model_usage.cjs +189 -0
  186. package/dist/metrics/model_usage.cjs.map +1 -0
  187. package/dist/metrics/model_usage.d.cts +92 -0
  188. package/dist/metrics/model_usage.d.ts +92 -0
  189. package/dist/metrics/model_usage.d.ts.map +1 -0
  190. package/dist/metrics/model_usage.js +164 -0
  191. package/dist/metrics/model_usage.js.map +1 -0
  192. package/dist/metrics/model_usage.test.cjs +474 -0
  193. package/dist/metrics/model_usage.test.cjs.map +1 -0
  194. package/dist/metrics/model_usage.test.js +476 -0
  195. package/dist/metrics/model_usage.test.js.map +1 -0
  196. package/dist/metrics/usage_collector.cjs +3 -0
  197. package/dist/metrics/usage_collector.cjs.map +1 -1
  198. package/dist/metrics/usage_collector.d.cts +9 -0
  199. package/dist/metrics/usage_collector.d.ts +9 -0
  200. package/dist/metrics/usage_collector.d.ts.map +1 -1
  201. package/dist/metrics/usage_collector.js +3 -0
  202. package/dist/metrics/usage_collector.js.map +1 -1
  203. package/dist/metrics/utils.cjs +9 -0
  204. package/dist/metrics/utils.cjs.map +1 -1
  205. package/dist/metrics/utils.d.ts.map +1 -1
  206. package/dist/metrics/utils.js +9 -0
  207. package/dist/metrics/utils.js.map +1 -1
  208. package/dist/stream/multi_input_stream.test.cjs +4 -0
  209. package/dist/stream/multi_input_stream.test.cjs.map +1 -1
  210. package/dist/stream/multi_input_stream.test.js +5 -1
  211. package/dist/stream/multi_input_stream.test.js.map +1 -1
  212. package/dist/stream/stream_channel.cjs +31 -0
  213. package/dist/stream/stream_channel.cjs.map +1 -1
  214. package/dist/stream/stream_channel.d.cts +4 -2
  215. package/dist/stream/stream_channel.d.ts +4 -2
  216. package/dist/stream/stream_channel.d.ts.map +1 -1
  217. package/dist/stream/stream_channel.js +31 -0
  218. package/dist/stream/stream_channel.js.map +1 -1
  219. package/dist/stt/stt.cjs +34 -2
  220. package/dist/stt/stt.cjs.map +1 -1
  221. package/dist/stt/stt.d.cts +22 -0
  222. package/dist/stt/stt.d.ts +22 -0
  223. package/dist/stt/stt.d.ts.map +1 -1
  224. package/dist/stt/stt.js +34 -2
  225. package/dist/stt/stt.js.map +1 -1
  226. package/dist/telemetry/otel_http_exporter.cjs +24 -5
  227. package/dist/telemetry/otel_http_exporter.cjs.map +1 -1
  228. package/dist/telemetry/otel_http_exporter.d.cts +1 -0
  229. package/dist/telemetry/otel_http_exporter.d.ts +1 -0
  230. package/dist/telemetry/otel_http_exporter.d.ts.map +1 -1
  231. package/dist/telemetry/otel_http_exporter.js +24 -5
  232. package/dist/telemetry/otel_http_exporter.js.map +1 -1
  233. package/dist/telemetry/trace_types.cjs +5 -5
  234. package/dist/telemetry/trace_types.cjs.map +1 -1
  235. package/dist/telemetry/trace_types.d.cts +9 -5
  236. package/dist/telemetry/trace_types.d.ts +9 -5
  237. package/dist/telemetry/trace_types.d.ts.map +1 -1
  238. package/dist/telemetry/trace_types.js +5 -5
  239. package/dist/telemetry/trace_types.js.map +1 -1
  240. package/dist/telemetry/traces.cjs +47 -8
  241. package/dist/telemetry/traces.cjs.map +1 -1
  242. package/dist/telemetry/traces.d.ts.map +1 -1
  243. package/dist/telemetry/traces.js +47 -8
  244. package/dist/telemetry/traces.js.map +1 -1
  245. package/dist/tts/tts.cjs +64 -2
  246. package/dist/tts/tts.cjs.map +1 -1
  247. package/dist/tts/tts.d.cts +34 -0
  248. package/dist/tts/tts.d.ts +34 -0
  249. package/dist/tts/tts.d.ts.map +1 -1
  250. package/dist/tts/tts.js +64 -2
  251. package/dist/tts/tts.js.map +1 -1
  252. package/dist/utils.cjs +1 -0
  253. package/dist/utils.cjs.map +1 -1
  254. package/dist/utils.d.ts.map +1 -1
  255. package/dist/utils.js +1 -0
  256. package/dist/utils.js.map +1 -1
  257. package/dist/version.cjs +1 -1
  258. package/dist/version.js +1 -1
  259. package/dist/voice/agent.cjs +34 -4
  260. package/dist/voice/agent.cjs.map +1 -1
  261. package/dist/voice/agent.d.cts +11 -2
  262. package/dist/voice/agent.d.ts +11 -2
  263. package/dist/voice/agent.d.ts.map +1 -1
  264. package/dist/voice/agent.js +34 -4
  265. package/dist/voice/agent.js.map +1 -1
  266. package/dist/voice/agent_activity.cjs +292 -44
  267. package/dist/voice/agent_activity.cjs.map +1 -1
  268. package/dist/voice/agent_activity.d.cts +27 -6
  269. package/dist/voice/agent_activity.d.ts +27 -6
  270. package/dist/voice/agent_activity.d.ts.map +1 -1
  271. package/dist/voice/agent_activity.js +293 -45
  272. package/dist/voice/agent_activity.js.map +1 -1
  273. package/dist/voice/agent_session.cjs +105 -48
  274. package/dist/voice/agent_session.cjs.map +1 -1
  275. package/dist/voice/agent_session.d.cts +90 -20
  276. package/dist/voice/agent_session.d.ts +90 -20
  277. package/dist/voice/agent_session.d.ts.map +1 -1
  278. package/dist/voice/agent_session.js +105 -46
  279. package/dist/voice/agent_session.js.map +1 -1
  280. package/dist/voice/audio_recognition.cjs +287 -6
  281. package/dist/voice/audio_recognition.cjs.map +1 -1
  282. package/dist/voice/audio_recognition.d.cts +42 -3
  283. package/dist/voice/audio_recognition.d.ts +42 -3
  284. package/dist/voice/audio_recognition.d.ts.map +1 -1
  285. package/dist/voice/audio_recognition.js +289 -7
  286. package/dist/voice/audio_recognition.js.map +1 -1
  287. package/dist/voice/client_events.cjs +554 -0
  288. package/dist/voice/client_events.cjs.map +1 -0
  289. package/dist/voice/client_events.d.cts +195 -0
  290. package/dist/voice/client_events.d.ts +195 -0
  291. package/dist/voice/client_events.d.ts.map +1 -0
  292. package/dist/voice/client_events.js +548 -0
  293. package/dist/voice/client_events.js.map +1 -0
  294. package/dist/voice/events.cjs +1 -0
  295. package/dist/voice/events.cjs.map +1 -1
  296. package/dist/voice/events.d.cts +8 -5
  297. package/dist/voice/events.d.ts +8 -5
  298. package/dist/voice/events.d.ts.map +1 -1
  299. package/dist/voice/events.js +1 -0
  300. package/dist/voice/events.js.map +1 -1
  301. package/dist/voice/generation.cjs +43 -8
  302. package/dist/voice/generation.cjs.map +1 -1
  303. package/dist/voice/generation.d.cts +3 -3
  304. package/dist/voice/generation.d.ts +3 -3
  305. package/dist/voice/generation.d.ts.map +1 -1
  306. package/dist/voice/generation.js +43 -8
  307. package/dist/voice/generation.js.map +1 -1
  308. package/dist/voice/index.cjs +1 -0
  309. package/dist/voice/index.cjs.map +1 -1
  310. package/dist/voice/index.d.cts +1 -0
  311. package/dist/voice/index.d.ts +1 -0
  312. package/dist/voice/index.d.ts.map +1 -1
  313. package/dist/voice/index.js +1 -0
  314. package/dist/voice/index.js.map +1 -1
  315. package/dist/voice/report.cjs +20 -8
  316. package/dist/voice/report.cjs.map +1 -1
  317. package/dist/voice/report.d.cts +5 -0
  318. package/dist/voice/report.d.ts +5 -0
  319. package/dist/voice/report.d.ts.map +1 -1
  320. package/dist/voice/report.js +20 -8
  321. package/dist/voice/report.js.map +1 -1
  322. package/dist/voice/report.test.cjs +106 -0
  323. package/dist/voice/report.test.cjs.map +1 -0
  324. package/dist/voice/report.test.js +105 -0
  325. package/dist/voice/report.test.js.map +1 -0
  326. package/dist/voice/room_io/room_io.cjs +16 -41
  327. package/dist/voice/room_io/room_io.cjs.map +1 -1
  328. package/dist/voice/room_io/room_io.d.cts +4 -9
  329. package/dist/voice/room_io/room_io.d.ts +4 -9
  330. package/dist/voice/room_io/room_io.d.ts.map +1 -1
  331. package/dist/voice/room_io/room_io.js +17 -43
  332. package/dist/voice/room_io/room_io.js.map +1 -1
  333. package/dist/voice/testing/fake_llm.cjs +127 -0
  334. package/dist/voice/testing/fake_llm.cjs.map +1 -0
  335. package/dist/voice/testing/fake_llm.d.cts +30 -0
  336. package/dist/voice/testing/fake_llm.d.ts +30 -0
  337. package/dist/voice/testing/fake_llm.d.ts.map +1 -0
  338. package/dist/voice/testing/fake_llm.js +103 -0
  339. package/dist/voice/testing/fake_llm.js.map +1 -0
  340. package/dist/voice/testing/index.cjs +3 -0
  341. package/dist/voice/testing/index.cjs.map +1 -1
  342. package/dist/voice/testing/index.d.cts +1 -0
  343. package/dist/voice/testing/index.d.ts +1 -0
  344. package/dist/voice/testing/index.d.ts.map +1 -1
  345. package/dist/voice/testing/index.js +2 -0
  346. package/dist/voice/testing/index.js.map +1 -1
  347. package/dist/voice/turn_config/endpointing.cjs +33 -0
  348. package/dist/voice/turn_config/endpointing.cjs.map +1 -0
  349. package/dist/voice/turn_config/endpointing.d.cts +30 -0
  350. package/dist/voice/turn_config/endpointing.d.ts +30 -0
  351. package/dist/voice/turn_config/endpointing.d.ts.map +1 -0
  352. package/dist/voice/turn_config/endpointing.js +9 -0
  353. package/dist/voice/turn_config/endpointing.js.map +1 -0
  354. package/dist/voice/turn_config/interruption.cjs +37 -0
  355. package/dist/voice/turn_config/interruption.cjs.map +1 -0
  356. package/dist/voice/turn_config/interruption.d.cts +53 -0
  357. package/dist/voice/turn_config/interruption.d.ts +53 -0
  358. package/dist/voice/turn_config/interruption.d.ts.map +1 -0
  359. package/dist/voice/turn_config/interruption.js +13 -0
  360. package/dist/voice/turn_config/interruption.js.map +1 -0
  361. package/dist/voice/turn_config/turn_handling.cjs +35 -0
  362. package/dist/voice/turn_config/turn_handling.cjs.map +1 -0
  363. package/dist/voice/turn_config/turn_handling.d.cts +36 -0
  364. package/dist/voice/turn_config/turn_handling.d.ts +36 -0
  365. package/dist/voice/turn_config/turn_handling.d.ts.map +1 -0
  366. package/dist/voice/turn_config/turn_handling.js +11 -0
  367. package/dist/voice/turn_config/turn_handling.js.map +1 -0
  368. package/dist/voice/turn_config/utils.cjs +97 -0
  369. package/dist/voice/turn_config/utils.cjs.map +1 -0
  370. package/dist/voice/turn_config/utils.d.cts +25 -0
  371. package/dist/voice/turn_config/utils.d.ts +25 -0
  372. package/dist/voice/turn_config/utils.d.ts.map +1 -0
  373. package/dist/voice/turn_config/utils.js +73 -0
  374. package/dist/voice/turn_config/utils.js.map +1 -0
  375. package/dist/voice/turn_config/utils.test.cjs +86 -0
  376. package/dist/voice/turn_config/utils.test.cjs.map +1 -0
  377. package/dist/voice/turn_config/utils.test.js +85 -0
  378. package/dist/voice/turn_config/utils.test.js.map +1 -0
  379. package/dist/voice/wire_format.cjs +798 -0
  380. package/dist/voice/wire_format.cjs.map +1 -0
  381. package/dist/voice/wire_format.d.cts +5503 -0
  382. package/dist/voice/wire_format.d.ts +5503 -0
  383. package/dist/voice/wire_format.d.ts.map +1 -0
  384. package/dist/voice/wire_format.js +728 -0
  385. package/dist/voice/wire_format.js.map +1 -0
  386. package/package.json +2 -1
  387. package/src/beta/index.ts +9 -0
  388. package/src/beta/workflows/index.ts +9 -0
  389. package/src/beta/workflows/task_group.ts +194 -0
  390. package/src/constants.ts +13 -0
  391. package/src/index.ts +2 -1
  392. package/src/inference/interruption/defaults.ts +51 -0
  393. package/src/inference/interruption/errors.ts +25 -0
  394. package/src/inference/interruption/http_transport.ts +187 -0
  395. package/src/inference/interruption/interruption_cache_entry.ts +50 -0
  396. package/src/inference/interruption/interruption_detector.ts +188 -0
  397. package/src/inference/interruption/interruption_stream.ts +467 -0
  398. package/src/inference/interruption/types.ts +84 -0
  399. package/src/inference/interruption/utils.test.ts +132 -0
  400. package/src/inference/interruption/utils.ts +137 -0
  401. package/src/inference/interruption/ws_transport.ts +402 -0
  402. package/src/inference/llm.ts +9 -12
  403. package/src/inference/stt.ts +10 -3
  404. package/src/inference/tts.ts +10 -3
  405. package/src/inference/utils.ts +29 -1
  406. package/src/llm/chat_context.test.ts +48 -0
  407. package/src/llm/chat_context.ts +161 -0
  408. package/src/llm/index.ts +2 -0
  409. package/src/llm/llm.ts +16 -0
  410. package/src/llm/realtime.ts +4 -0
  411. package/src/llm/tool_context.ts +14 -0
  412. package/src/metrics/base.ts +48 -1
  413. package/src/metrics/index.ts +11 -0
  414. package/src/metrics/model_usage.test.ts +545 -0
  415. package/src/metrics/model_usage.ts +262 -0
  416. package/src/metrics/usage_collector.ts +11 -0
  417. package/src/metrics/utils.ts +11 -0
  418. package/src/stream/multi_input_stream.test.ts +6 -1
  419. package/src/stream/stream_channel.ts +34 -2
  420. package/src/stt/stt.ts +38 -0
  421. package/src/telemetry/otel_http_exporter.ts +28 -5
  422. package/src/telemetry/trace_types.ts +11 -8
  423. package/src/telemetry/traces.ts +111 -54
  424. package/src/tts/tts.ts +69 -1
  425. package/src/utils.ts +5 -0
  426. package/src/voice/agent.ts +41 -3
  427. package/src/voice/agent_activity.ts +371 -34
  428. package/src/voice/agent_session.ts +207 -59
  429. package/src/voice/audio_recognition.ts +385 -9
  430. package/src/voice/client_events.ts +838 -0
  431. package/src/voice/events.ts +14 -4
  432. package/src/voice/generation.ts +52 -9
  433. package/src/voice/index.ts +1 -0
  434. package/src/voice/report.test.ts +117 -0
  435. package/src/voice/report.ts +29 -6
  436. package/src/voice/room_io/room_io.ts +21 -64
  437. package/src/voice/testing/fake_llm.ts +138 -0
  438. package/src/voice/testing/index.ts +2 -0
  439. package/src/voice/turn_config/endpointing.ts +33 -0
  440. package/src/voice/turn_config/interruption.ts +56 -0
  441. package/src/voice/turn_config/turn_handling.ts +45 -0
  442. package/src/voice/turn_config/utils.test.ts +100 -0
  443. package/src/voice/turn_config/utils.ts +103 -0
  444. package/src/voice/wire_format.ts +827 -0
@@ -0,0 +1,368 @@
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 interruption_stream_exports = {};
20
+ __export(interruption_stream_exports, {
21
+ InterruptionStreamBase: () => InterruptionStreamBase,
22
+ InterruptionStreamSentinel: () => InterruptionStreamSentinel
23
+ });
24
+ module.exports = __toCommonJS(interruption_stream_exports);
25
+ var import_rtc_node = require("@livekit/rtc-node");
26
+ var import_web = require("stream/web");
27
+ var import_log = require("../../log.cjs");
28
+ var import_stream_channel = require("../../stream/stream_channel.cjs");
29
+ var import_telemetry = require("../../telemetry/index.cjs");
30
+ var import_defaults = require("./defaults.cjs");
31
+ var import_http_transport = require("./http_transport.cjs");
32
+ var import_interruption_cache_entry = require("./interruption_cache_entry.cjs");
33
+ var import_types = require("./types.cjs");
34
+ var import_utils = require("./utils.cjs");
35
+ var import_ws_transport = require("./ws_transport.cjs");
36
+ class InterruptionStreamSentinel {
37
+ static agentSpeechStarted() {
38
+ return { type: "agent-speech-started" };
39
+ }
40
+ static agentSpeechEnded() {
41
+ return { type: "agent-speech-ended" };
42
+ }
43
+ static overlapSpeechStarted(speechDuration, startedAt, userSpeakingSpan) {
44
+ return { type: "overlap-speech-started", speechDuration, startedAt, userSpeakingSpan };
45
+ }
46
+ static overlapSpeechEnded(endedAt) {
47
+ return { type: "overlap-speech-ended", endedAt };
48
+ }
49
+ static flush() {
50
+ return { type: "flush" };
51
+ }
52
+ }
53
+ function updateUserSpeakingSpan(span, entry) {
54
+ span.setAttribute(
55
+ import_telemetry.traceTypes.ATTR_IS_INTERRUPTION,
56
+ (entry.isInterruption ?? false).toString().toLowerCase()
57
+ );
58
+ span.setAttribute(import_telemetry.traceTypes.ATTR_INTERRUPTION_PROBABILITY, entry.probability);
59
+ span.setAttribute(import_telemetry.traceTypes.ATTR_INTERRUPTION_TOTAL_DURATION, entry.totalDurationInS);
60
+ span.setAttribute(import_telemetry.traceTypes.ATTR_INTERRUPTION_PREDICTION_DURATION, entry.predictionDurationInS);
61
+ span.setAttribute(import_telemetry.traceTypes.ATTR_INTERRUPTION_DETECTION_DELAY, entry.detectionDelayInS);
62
+ }
63
+ class InterruptionStreamBase {
64
+ inputStream;
65
+ eventStream;
66
+ resampler;
67
+ numRequests = 0;
68
+ userSpeakingSpan;
69
+ overlapSpeechStartedAt;
70
+ options;
71
+ apiOptions;
72
+ model;
73
+ logger = (0, import_log.log)();
74
+ // Store reconnect function for WebSocket transport
75
+ wsReconnect;
76
+ // Mutable transport options that can be updated via updateOptions()
77
+ transportOptions;
78
+ constructor(model, apiOptions) {
79
+ this.inputStream = (0, import_stream_channel.createStreamChannel)();
80
+ this.model = model;
81
+ this.options = { ...model.options };
82
+ this.apiOptions = { ...import_defaults.apiConnectDefaults, ...apiOptions };
83
+ this.transportOptions = {
84
+ baseUrl: this.options.baseUrl,
85
+ apiKey: this.options.apiKey,
86
+ apiSecret: this.options.apiSecret,
87
+ sampleRate: this.options.sampleRate,
88
+ threshold: this.options.threshold,
89
+ minFrames: this.options.minFrames,
90
+ timeout: this.options.inferenceTimeout,
91
+ maxRetries: this.apiOptions.maxRetries
92
+ };
93
+ this.eventStream = this.setupTransform();
94
+ }
95
+ /**
96
+ * Update stream options. For WebSocket transport, this triggers a reconnection.
97
+ */
98
+ async updateOptions(options) {
99
+ if (options.threshold !== void 0) {
100
+ this.options.threshold = options.threshold;
101
+ this.transportOptions.threshold = options.threshold;
102
+ }
103
+ if (options.minInterruptionDurationInS !== void 0) {
104
+ this.options.minInterruptionDurationInS = options.minInterruptionDurationInS;
105
+ this.options.minFrames = Math.ceil(options.minInterruptionDurationInS * import_defaults.FRAMES_PER_SECOND);
106
+ this.transportOptions.minFrames = this.options.minFrames;
107
+ }
108
+ if (this.options.useProxy && this.wsReconnect) {
109
+ await this.wsReconnect();
110
+ }
111
+ }
112
+ setupTransform() {
113
+ let agentSpeechStarted = false;
114
+ let startIdx = 0;
115
+ let accumulatedSamples = 0;
116
+ let overlapSpeechStarted = false;
117
+ let overlapCount = 0;
118
+ const cache = new import_utils.BoundedCache(10);
119
+ const inferenceS16Data = new Int16Array(
120
+ Math.ceil(this.options.maxAudioDurationInS * this.options.sampleRate)
121
+ ).fill(0);
122
+ const getState = () => ({
123
+ overlapSpeechStarted,
124
+ overlapSpeechStartedAt: this.overlapSpeechStartedAt,
125
+ cache,
126
+ overlapCount
127
+ });
128
+ const setState = (partial) => {
129
+ if (partial.overlapSpeechStarted !== void 0) {
130
+ overlapSpeechStarted = partial.overlapSpeechStarted;
131
+ }
132
+ };
133
+ const handleSpanUpdate = (entry) => {
134
+ if (this.userSpeakingSpan) {
135
+ updateUserSpeakingSpan(this.userSpeakingSpan, entry);
136
+ this.userSpeakingSpan = void 0;
137
+ }
138
+ };
139
+ const onRequestSent = () => {
140
+ this.numRequests++;
141
+ };
142
+ const getAndResetNumRequests = () => {
143
+ const n = this.numRequests;
144
+ this.numRequests = 0;
145
+ return n;
146
+ };
147
+ const audioTransformer = new import_web.TransformStream(
148
+ {
149
+ transform: (chunk, controller) => {
150
+ if (chunk instanceof import_rtc_node.AudioFrame) {
151
+ if (!agentSpeechStarted) {
152
+ return;
153
+ }
154
+ if (this.options.sampleRate !== chunk.sampleRate) {
155
+ controller.error("the sample rate of the input frames must be consistent");
156
+ this.logger.error("the sample rate of the input frames must be consistent");
157
+ return;
158
+ }
159
+ const result = writeToInferenceS16Data(
160
+ chunk,
161
+ startIdx,
162
+ inferenceS16Data,
163
+ this.options.maxAudioDurationInS
164
+ );
165
+ startIdx = result.startIdx;
166
+ accumulatedSamples += result.samplesWritten;
167
+ if (accumulatedSamples >= Math.floor(this.options.detectionIntervalInS * this.options.sampleRate) && overlapSpeechStarted) {
168
+ const audioSlice = inferenceS16Data.slice(0, startIdx);
169
+ accumulatedSamples = 0;
170
+ controller.enqueue(audioSlice);
171
+ }
172
+ } else if (chunk.type === "agent-speech-started") {
173
+ this.logger.debug("agent speech started");
174
+ agentSpeechStarted = true;
175
+ overlapSpeechStarted = false;
176
+ this.overlapSpeechStartedAt = void 0;
177
+ accumulatedSamples = 0;
178
+ overlapCount = 0;
179
+ startIdx = 0;
180
+ this.numRequests = 0;
181
+ cache.clear();
182
+ } else if (chunk.type === "agent-speech-ended") {
183
+ this.logger.debug("agent speech ended");
184
+ agentSpeechStarted = false;
185
+ overlapSpeechStarted = false;
186
+ this.overlapSpeechStartedAt = void 0;
187
+ accumulatedSamples = 0;
188
+ overlapCount = 0;
189
+ startIdx = 0;
190
+ this.numRequests = 0;
191
+ cache.clear();
192
+ } else if (chunk.type === "overlap-speech-started" && agentSpeechStarted) {
193
+ this.overlapSpeechStartedAt = chunk.startedAt;
194
+ this.userSpeakingSpan = chunk.userSpeakingSpan;
195
+ this.logger.debug("overlap speech started, starting interruption inference");
196
+ overlapSpeechStarted = true;
197
+ accumulatedSamples = 0;
198
+ overlapCount += 1;
199
+ if (overlapCount <= 1) {
200
+ const keepSize = Math.round(chunk.speechDuration / 1e3 * this.options.sampleRate) + Math.round(this.options.audioPrefixDurationInS * this.options.sampleRate);
201
+ const shiftCount = Math.max(0, startIdx - keepSize);
202
+ inferenceS16Data.copyWithin(0, shiftCount, startIdx);
203
+ startIdx -= shiftCount;
204
+ }
205
+ cache.clear();
206
+ } else if (chunk.type === "overlap-speech-ended") {
207
+ this.logger.debug("overlap speech ended");
208
+ if (overlapSpeechStarted) {
209
+ this.userSpeakingSpan = void 0;
210
+ let latestEntry = cache.pop(
211
+ (entry) => entry.totalDurationInS !== void 0 && entry.totalDurationInS > 0
212
+ );
213
+ if (!latestEntry) {
214
+ this.logger.debug("no request made for overlap speech");
215
+ latestEntry = import_interruption_cache_entry.InterruptionCacheEntry.default();
216
+ }
217
+ const e = latestEntry ?? import_interruption_cache_entry.InterruptionCacheEntry.default();
218
+ const event = {
219
+ type: "user_overlapping_speech",
220
+ timestamp: chunk.endedAt,
221
+ isInterruption: false,
222
+ overlapStartedAt: this.overlapSpeechStartedAt,
223
+ speechInput: e.speechInput,
224
+ probabilities: e.probabilities,
225
+ totalDurationInS: e.totalDurationInS,
226
+ detectionDelayInS: e.detectionDelayInS,
227
+ predictionDurationInS: e.predictionDurationInS,
228
+ probability: e.probability,
229
+ numRequests: getAndResetNumRequests()
230
+ };
231
+ controller.enqueue(event);
232
+ overlapSpeechStarted = false;
233
+ accumulatedSamples = 0;
234
+ }
235
+ this.overlapSpeechStartedAt = void 0;
236
+ } else if (chunk.type === "flush") {
237
+ }
238
+ }
239
+ },
240
+ { highWaterMark: 32 },
241
+ { highWaterMark: 32 }
242
+ );
243
+ const transportOptions = this.transportOptions;
244
+ let transport;
245
+ if (this.options.useProxy) {
246
+ const wsResult = (0, import_ws_transport.createWsTransport)(
247
+ transportOptions,
248
+ getState,
249
+ setState,
250
+ handleSpanUpdate,
251
+ onRequestSent,
252
+ getAndResetNumRequests
253
+ );
254
+ transport = wsResult.transport;
255
+ this.wsReconnect = wsResult.reconnect;
256
+ } else {
257
+ transport = (0, import_http_transport.createHttpTransport)(
258
+ transportOptions,
259
+ getState,
260
+ setState,
261
+ handleSpanUpdate,
262
+ getAndResetNumRequests
263
+ );
264
+ }
265
+ const eventEmitter = new import_web.TransformStream({
266
+ transform: (chunk, controller) => {
267
+ this.model.emit("user_overlapping_speech", chunk);
268
+ const metrics = {
269
+ type: "interruption_metrics",
270
+ timestamp: chunk.timestamp,
271
+ totalDuration: chunk.totalDurationInS * 1e3,
272
+ predictionDuration: chunk.predictionDurationInS * 1e3,
273
+ detectionDelay: chunk.detectionDelayInS * 1e3,
274
+ numInterruptions: chunk.isInterruption ? 1 : 0,
275
+ numBackchannels: chunk.isInterruption ? 0 : 1,
276
+ numRequests: chunk.numRequests,
277
+ metadata: {
278
+ modelProvider: this.model.provider,
279
+ modelName: this.model.model
280
+ }
281
+ };
282
+ this.model.emit("metrics_collected", metrics);
283
+ controller.enqueue(chunk);
284
+ }
285
+ });
286
+ return this.inputStream.stream().pipeThrough(audioTransformer).pipeThrough(transport).pipeThrough(eventEmitter);
287
+ }
288
+ ensureInputNotEnded() {
289
+ if (this.inputStream.closed) {
290
+ throw new Error("input stream is closed");
291
+ }
292
+ }
293
+ ensureStreamsNotEnded() {
294
+ this.ensureInputNotEnded();
295
+ }
296
+ getResamplerFor(inputSampleRate) {
297
+ if (!this.resampler) {
298
+ this.resampler = new import_rtc_node.AudioResampler(inputSampleRate, this.options.sampleRate);
299
+ }
300
+ return this.resampler;
301
+ }
302
+ stream() {
303
+ return this.eventStream;
304
+ }
305
+ async pushFrame(frame) {
306
+ this.ensureStreamsNotEnded();
307
+ if (!(frame instanceof import_rtc_node.AudioFrame)) {
308
+ return this.inputStream.write(frame);
309
+ } else if (this.options.sampleRate !== frame.sampleRate) {
310
+ const resampler = this.getResamplerFor(frame.sampleRate);
311
+ if (resampler.inputRate !== frame.sampleRate) {
312
+ throw new Error("the sample rate of the input frames must be consistent");
313
+ }
314
+ for (const resampledFrame of resampler.push(frame)) {
315
+ await this.inputStream.write(resampledFrame);
316
+ }
317
+ } else {
318
+ await this.inputStream.write(frame);
319
+ }
320
+ }
321
+ async flush() {
322
+ this.ensureStreamsNotEnded();
323
+ await this.inputStream.write(InterruptionStreamSentinel.flush());
324
+ }
325
+ async endInput() {
326
+ await this.flush();
327
+ await this.inputStream.close();
328
+ }
329
+ async close() {
330
+ if (!this.inputStream.closed) await this.inputStream.close();
331
+ this.model.removeStream(this);
332
+ }
333
+ }
334
+ function writeToInferenceS16Data(frame, startIdx, outData, maxAudioDuration) {
335
+ const maxWindowSize = Math.floor(maxAudioDuration * frame.sampleRate);
336
+ if (frame.samplesPerChannel > outData.length) {
337
+ throw new Error("frame samples are greater than the max window size");
338
+ }
339
+ const shift = startIdx + frame.samplesPerChannel - maxWindowSize;
340
+ if (shift > 0) {
341
+ outData.copyWithin(0, shift, startIdx);
342
+ startIdx -= shift;
343
+ }
344
+ const frameData = new Int16Array(
345
+ frame.data.buffer,
346
+ frame.data.byteOffset,
347
+ frame.samplesPerChannel * frame.channels
348
+ );
349
+ if (frame.channels > 1) {
350
+ for (let i = 0; i < frame.samplesPerChannel; i++) {
351
+ let sum = 0;
352
+ for (let ch = 0; ch < frame.channels; ch++) {
353
+ sum += frameData[i * frame.channels + ch] ?? 0;
354
+ }
355
+ outData[startIdx + i] = Math.floor(sum / frame.channels);
356
+ }
357
+ } else {
358
+ outData.set(frameData, startIdx);
359
+ }
360
+ startIdx += frame.samplesPerChannel;
361
+ return { startIdx, samplesWritten: frame.samplesPerChannel };
362
+ }
363
+ // Annotate the CommonJS export names for ESM import in node:
364
+ 0 && (module.exports = {
365
+ InterruptionStreamBase,
366
+ InterruptionStreamSentinel
367
+ });
368
+ //# sourceMappingURL=interruption_stream.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../src/inference/interruption/interruption_stream.ts"],"sourcesContent":["// SPDX-FileCopyrightText: 2026 LiveKit, Inc.\n//\n// SPDX-License-Identifier: Apache-2.0\nimport { AudioFrame, AudioResampler } from '@livekit/rtc-node';\nimport type { Span } from '@opentelemetry/api';\nimport { type ReadableStream, TransformStream } from 'stream/web';\nimport { log } from '../../log.js';\nimport type { InterruptionMetrics } from '../../metrics/base.js';\nimport { type StreamChannel, createStreamChannel } from '../../stream/stream_channel.js';\nimport { traceTypes } from '../../telemetry/index.js';\nimport { FRAMES_PER_SECOND, apiConnectDefaults } from './defaults.js';\nimport type { InterruptionDetectionError } from './errors.js';\nimport { createHttpTransport } from './http_transport.js';\nimport { InterruptionCacheEntry } from './interruption_cache_entry.js';\nimport type { AdaptiveInterruptionDetector } from './interruption_detector.js';\nimport {\n type AgentSpeechEnded,\n type AgentSpeechStarted,\n type ApiConnectOptions,\n type Flush,\n type InterruptionOptions,\n type InterruptionSentinel,\n type OverlapSpeechEnded,\n type OverlapSpeechStarted,\n type OverlappingSpeechEvent,\n} from './types.js';\nimport { BoundedCache } from './utils.js';\nimport { createWsTransport } from './ws_transport.js';\n\n// Re-export sentinel types for backwards compatibility\nexport type {\n AgentSpeechEnded,\n AgentSpeechStarted,\n ApiConnectOptions,\n Flush,\n InterruptionSentinel,\n OverlapSpeechEnded,\n OverlapSpeechStarted,\n};\n\nexport class InterruptionStreamSentinel {\n static agentSpeechStarted(): AgentSpeechStarted {\n return { type: 'agent-speech-started' };\n }\n\n static agentSpeechEnded(): AgentSpeechEnded {\n return { type: 'agent-speech-ended' };\n }\n\n static overlapSpeechStarted(\n speechDuration: number,\n startedAt: number,\n userSpeakingSpan?: Span,\n ): OverlapSpeechStarted {\n return { type: 'overlap-speech-started', speechDuration, startedAt, userSpeakingSpan };\n }\n\n static overlapSpeechEnded(endedAt: number): OverlapSpeechEnded {\n return { type: 'overlap-speech-ended', endedAt };\n }\n\n static flush(): Flush {\n return { type: 'flush' };\n }\n}\n\nfunction updateUserSpeakingSpan(span: Span, entry: InterruptionCacheEntry) {\n span.setAttribute(\n traceTypes.ATTR_IS_INTERRUPTION,\n (entry.isInterruption ?? false).toString().toLowerCase(),\n );\n span.setAttribute(traceTypes.ATTR_INTERRUPTION_PROBABILITY, entry.probability);\n span.setAttribute(traceTypes.ATTR_INTERRUPTION_TOTAL_DURATION, entry.totalDurationInS);\n span.setAttribute(traceTypes.ATTR_INTERRUPTION_PREDICTION_DURATION, entry.predictionDurationInS);\n span.setAttribute(traceTypes.ATTR_INTERRUPTION_DETECTION_DELAY, entry.detectionDelayInS);\n}\n\nexport class InterruptionStreamBase {\n private inputStream: StreamChannel<InterruptionSentinel | AudioFrame, InterruptionDetectionError>;\n\n private eventStream: ReadableStream<OverlappingSpeechEvent>;\n\n private resampler?: AudioResampler;\n\n private numRequests = 0;\n\n private userSpeakingSpan: Span | undefined;\n\n private overlapSpeechStartedAt: number | undefined;\n\n private options: InterruptionOptions;\n\n private apiOptions: ApiConnectOptions;\n\n private model: AdaptiveInterruptionDetector;\n\n private logger = log();\n\n // Store reconnect function for WebSocket transport\n private wsReconnect?: () => Promise<void>;\n\n // Mutable transport options that can be updated via updateOptions()\n private transportOptions: {\n baseUrl: string;\n apiKey: string;\n apiSecret: string;\n sampleRate: number;\n threshold: number;\n minFrames: number;\n timeout: number;\n maxRetries: number;\n };\n\n constructor(model: AdaptiveInterruptionDetector, apiOptions: Partial<ApiConnectOptions>) {\n this.inputStream = createStreamChannel<\n InterruptionSentinel | AudioFrame,\n InterruptionDetectionError\n >();\n\n this.model = model;\n this.options = { ...model.options };\n this.apiOptions = { ...apiConnectDefaults, ...apiOptions };\n\n // Initialize mutable transport options\n this.transportOptions = {\n baseUrl: this.options.baseUrl,\n apiKey: this.options.apiKey,\n apiSecret: this.options.apiSecret,\n sampleRate: this.options.sampleRate,\n threshold: this.options.threshold,\n minFrames: this.options.minFrames,\n timeout: this.options.inferenceTimeout,\n maxRetries: this.apiOptions.maxRetries,\n };\n\n this.eventStream = this.setupTransform();\n }\n\n /**\n * Update stream options. For WebSocket transport, this triggers a reconnection.\n */\n async updateOptions(options: {\n threshold?: number;\n minInterruptionDurationInS?: number;\n }): Promise<void> {\n if (options.threshold !== undefined) {\n this.options.threshold = options.threshold;\n this.transportOptions.threshold = options.threshold;\n }\n if (options.minInterruptionDurationInS !== undefined) {\n this.options.minInterruptionDurationInS = options.minInterruptionDurationInS;\n this.options.minFrames = Math.ceil(options.minInterruptionDurationInS * FRAMES_PER_SECOND);\n this.transportOptions.minFrames = this.options.minFrames;\n }\n // Trigger WebSocket reconnection if using proxy (WebSocket transport)\n if (this.options.useProxy && this.wsReconnect) {\n await this.wsReconnect();\n }\n }\n\n private setupTransform(): ReadableStream<OverlappingSpeechEvent> {\n let agentSpeechStarted = false;\n let startIdx = 0;\n let accumulatedSamples = 0;\n let overlapSpeechStarted = false;\n let overlapCount = 0;\n const cache = new BoundedCache<number, InterruptionCacheEntry>(10);\n const inferenceS16Data = new Int16Array(\n Math.ceil(this.options.maxAudioDurationInS * this.options.sampleRate),\n ).fill(0);\n\n // State accessors for transport\n const getState = () => ({\n overlapSpeechStarted,\n overlapSpeechStartedAt: this.overlapSpeechStartedAt,\n cache,\n overlapCount,\n });\n const setState = (partial: { overlapSpeechStarted?: boolean }) => {\n if (partial.overlapSpeechStarted !== undefined) {\n overlapSpeechStarted = partial.overlapSpeechStarted;\n }\n };\n const handleSpanUpdate = (entry: InterruptionCacheEntry) => {\n if (this.userSpeakingSpan) {\n updateUserSpeakingSpan(this.userSpeakingSpan, entry);\n this.userSpeakingSpan = undefined;\n }\n };\n\n const onRequestSent = () => {\n this.numRequests++;\n };\n\n const getAndResetNumRequests = (): number => {\n const n = this.numRequests;\n this.numRequests = 0;\n return n;\n };\n\n // First transform: process input frames/sentinels and output audio slices or events\n const audioTransformer = new TransformStream<\n InterruptionSentinel | AudioFrame,\n Int16Array | OverlappingSpeechEvent\n >(\n {\n transform: (chunk, controller) => {\n if (chunk instanceof AudioFrame) {\n if (!agentSpeechStarted) {\n return;\n }\n if (this.options.sampleRate !== chunk.sampleRate) {\n controller.error('the sample rate of the input frames must be consistent');\n this.logger.error('the sample rate of the input frames must be consistent');\n return;\n }\n const result = writeToInferenceS16Data(\n chunk,\n startIdx,\n inferenceS16Data,\n this.options.maxAudioDurationInS,\n );\n startIdx = result.startIdx;\n accumulatedSamples += result.samplesWritten;\n\n if (\n accumulatedSamples >=\n Math.floor(this.options.detectionIntervalInS * this.options.sampleRate) &&\n overlapSpeechStarted\n ) {\n const audioSlice = inferenceS16Data.slice(0, startIdx);\n accumulatedSamples = 0;\n controller.enqueue(audioSlice);\n }\n } else if (chunk.type === 'agent-speech-started') {\n this.logger.debug('agent speech started');\n agentSpeechStarted = true;\n overlapSpeechStarted = false;\n this.overlapSpeechStartedAt = undefined;\n accumulatedSamples = 0;\n overlapCount = 0;\n startIdx = 0;\n this.numRequests = 0;\n cache.clear();\n } else if (chunk.type === 'agent-speech-ended') {\n this.logger.debug('agent speech ended');\n agentSpeechStarted = false;\n overlapSpeechStarted = false;\n this.overlapSpeechStartedAt = undefined;\n accumulatedSamples = 0;\n overlapCount = 0;\n startIdx = 0;\n this.numRequests = 0;\n cache.clear();\n } else if (chunk.type === 'overlap-speech-started' && agentSpeechStarted) {\n this.overlapSpeechStartedAt = chunk.startedAt;\n this.userSpeakingSpan = chunk.userSpeakingSpan;\n this.logger.debug('overlap speech started, starting interruption inference');\n overlapSpeechStarted = true;\n accumulatedSamples = 0;\n overlapCount += 1;\n if (overlapCount <= 1) {\n const keepSize =\n Math.round((chunk.speechDuration / 1000) * this.options.sampleRate) +\n Math.round(this.options.audioPrefixDurationInS * this.options.sampleRate);\n const shiftCount = Math.max(0, startIdx - keepSize);\n inferenceS16Data.copyWithin(0, shiftCount, startIdx);\n startIdx -= shiftCount;\n }\n cache.clear();\n } else if (chunk.type === 'overlap-speech-ended') {\n this.logger.debug('overlap speech ended');\n if (overlapSpeechStarted) {\n this.userSpeakingSpan = undefined;\n let latestEntry = cache.pop(\n (entry) => entry.totalDurationInS !== undefined && entry.totalDurationInS > 0,\n );\n if (!latestEntry) {\n this.logger.debug('no request made for overlap speech');\n latestEntry = InterruptionCacheEntry.default();\n }\n const e = latestEntry ?? InterruptionCacheEntry.default();\n const event: OverlappingSpeechEvent = {\n type: 'user_overlapping_speech',\n timestamp: chunk.endedAt,\n isInterruption: false,\n overlapStartedAt: this.overlapSpeechStartedAt,\n speechInput: e.speechInput,\n probabilities: e.probabilities,\n totalDurationInS: e.totalDurationInS,\n detectionDelayInS: e.detectionDelayInS,\n predictionDurationInS: e.predictionDurationInS,\n probability: e.probability,\n numRequests: getAndResetNumRequests(),\n };\n controller.enqueue(event);\n overlapSpeechStarted = false;\n accumulatedSamples = 0;\n }\n this.overlapSpeechStartedAt = undefined;\n } else if (chunk.type === 'flush') {\n // no-op\n }\n },\n },\n { highWaterMark: 32 },\n { highWaterMark: 32 },\n );\n\n // Second transform: transport layer (HTTP or WebSocket based on useProxy)\n const transportOptions = this.transportOptions;\n\n let transport: TransformStream<Int16Array | OverlappingSpeechEvent, OverlappingSpeechEvent>;\n if (this.options.useProxy) {\n const wsResult = createWsTransport(\n transportOptions,\n getState,\n setState,\n handleSpanUpdate,\n onRequestSent,\n getAndResetNumRequests,\n );\n transport = wsResult.transport;\n this.wsReconnect = wsResult.reconnect;\n } else {\n transport = createHttpTransport(\n transportOptions,\n getState,\n setState,\n handleSpanUpdate,\n getAndResetNumRequests,\n );\n }\n\n const eventEmitter = new TransformStream<OverlappingSpeechEvent, OverlappingSpeechEvent>({\n transform: (chunk, controller) => {\n this.model.emit('user_overlapping_speech', chunk);\n\n const metrics: InterruptionMetrics = {\n type: 'interruption_metrics',\n timestamp: chunk.timestamp,\n totalDuration: chunk.totalDurationInS * 1000,\n predictionDuration: chunk.predictionDurationInS * 1000,\n detectionDelay: chunk.detectionDelayInS * 1000,\n numInterruptions: chunk.isInterruption ? 1 : 0,\n numBackchannels: chunk.isInterruption ? 0 : 1,\n numRequests: chunk.numRequests,\n metadata: {\n modelProvider: this.model.provider,\n modelName: this.model.model,\n },\n };\n this.model.emit('metrics_collected', metrics);\n\n controller.enqueue(chunk);\n },\n });\n\n // Pipeline: input -> audioTransformer -> transport -> eventEmitter -> eventStream\n return this.inputStream\n .stream()\n .pipeThrough(audioTransformer)\n .pipeThrough(transport)\n .pipeThrough(eventEmitter);\n }\n\n private ensureInputNotEnded() {\n if (this.inputStream.closed) {\n throw new Error('input stream is closed');\n }\n }\n\n private ensureStreamsNotEnded() {\n this.ensureInputNotEnded();\n }\n\n private getResamplerFor(inputSampleRate: number): AudioResampler {\n if (!this.resampler) {\n this.resampler = new AudioResampler(inputSampleRate, this.options.sampleRate);\n }\n return this.resampler;\n }\n\n stream(): ReadableStream<OverlappingSpeechEvent> {\n return this.eventStream;\n }\n\n async pushFrame(frame: InterruptionSentinel | AudioFrame): Promise<void> {\n this.ensureStreamsNotEnded();\n if (!(frame instanceof AudioFrame)) {\n return this.inputStream.write(frame);\n } else if (this.options.sampleRate !== frame.sampleRate) {\n const resampler = this.getResamplerFor(frame.sampleRate);\n if (resampler.inputRate !== frame.sampleRate) {\n throw new Error('the sample rate of the input frames must be consistent');\n }\n for (const resampledFrame of resampler.push(frame)) {\n await this.inputStream.write(resampledFrame);\n }\n } else {\n await this.inputStream.write(frame);\n }\n }\n\n async flush(): Promise<void> {\n this.ensureStreamsNotEnded();\n await this.inputStream.write(InterruptionStreamSentinel.flush());\n }\n\n async endInput(): Promise<void> {\n await this.flush();\n await this.inputStream.close();\n }\n\n async close(): Promise<void> {\n if (!this.inputStream.closed) await this.inputStream.close();\n this.model.removeStream(this);\n }\n}\n\n/**\n * Write the audio frame to the output data array and return the new start index\n * and the number of samples written.\n */\nfunction writeToInferenceS16Data(\n frame: AudioFrame,\n startIdx: number,\n outData: Int16Array,\n maxAudioDuration: number,\n): { startIdx: number; samplesWritten: number } {\n const maxWindowSize = Math.floor(maxAudioDuration * frame.sampleRate);\n\n if (frame.samplesPerChannel > outData.length) {\n throw new Error('frame samples are greater than the max window size');\n }\n\n // Shift the data to the left if the window would overflow\n const shift = startIdx + frame.samplesPerChannel - maxWindowSize;\n if (shift > 0) {\n outData.copyWithin(0, shift, startIdx);\n startIdx -= shift;\n }\n\n // Get the frame data as Int16Array\n const frameData = new Int16Array(\n frame.data.buffer,\n frame.data.byteOffset,\n frame.samplesPerChannel * frame.channels,\n );\n\n if (frame.channels > 1) {\n // Mix down multiple channels to mono by averaging\n for (let i = 0; i < frame.samplesPerChannel; i++) {\n let sum = 0;\n for (let ch = 0; ch < frame.channels; ch++) {\n sum += frameData[i * frame.channels + ch] ?? 0;\n }\n outData[startIdx + i] = Math.floor(sum / frame.channels);\n }\n } else {\n // Single channel - copy directly\n outData.set(frameData, startIdx);\n }\n\n startIdx += frame.samplesPerChannel;\n return { startIdx, samplesWritten: frame.samplesPerChannel };\n}\n"],"mappings":"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","names":[]}
@@ -0,0 +1,46 @@
1
+ /// <reference types="node" resolution-mode="require"/>
2
+ import { AudioFrame } from '@livekit/rtc-node';
3
+ import type { Span } from '@opentelemetry/api';
4
+ import { type ReadableStream } from 'stream/web';
5
+ import type { AdaptiveInterruptionDetector } from './interruption_detector.js';
6
+ import { type AgentSpeechEnded, type AgentSpeechStarted, type ApiConnectOptions, type Flush, type InterruptionSentinel, type OverlapSpeechEnded, type OverlapSpeechStarted, type OverlappingSpeechEvent } from './types.js';
7
+ export type { AgentSpeechEnded, AgentSpeechStarted, ApiConnectOptions, Flush, InterruptionSentinel, OverlapSpeechEnded, OverlapSpeechStarted, };
8
+ export declare class InterruptionStreamSentinel {
9
+ static agentSpeechStarted(): AgentSpeechStarted;
10
+ static agentSpeechEnded(): AgentSpeechEnded;
11
+ static overlapSpeechStarted(speechDuration: number, startedAt: number, userSpeakingSpan?: Span): OverlapSpeechStarted;
12
+ static overlapSpeechEnded(endedAt: number): OverlapSpeechEnded;
13
+ static flush(): Flush;
14
+ }
15
+ export declare class InterruptionStreamBase {
16
+ private inputStream;
17
+ private eventStream;
18
+ private resampler?;
19
+ private numRequests;
20
+ private userSpeakingSpan;
21
+ private overlapSpeechStartedAt;
22
+ private options;
23
+ private apiOptions;
24
+ private model;
25
+ private logger;
26
+ private wsReconnect?;
27
+ private transportOptions;
28
+ constructor(model: AdaptiveInterruptionDetector, apiOptions: Partial<ApiConnectOptions>);
29
+ /**
30
+ * Update stream options. For WebSocket transport, this triggers a reconnection.
31
+ */
32
+ updateOptions(options: {
33
+ threshold?: number;
34
+ minInterruptionDurationInS?: number;
35
+ }): Promise<void>;
36
+ private setupTransform;
37
+ private ensureInputNotEnded;
38
+ private ensureStreamsNotEnded;
39
+ private getResamplerFor;
40
+ stream(): ReadableStream<OverlappingSpeechEvent>;
41
+ pushFrame(frame: InterruptionSentinel | AudioFrame): Promise<void>;
42
+ flush(): Promise<void>;
43
+ endInput(): Promise<void>;
44
+ close(): Promise<void>;
45
+ }
46
+ //# sourceMappingURL=interruption_stream.d.ts.map
@@ -0,0 +1,46 @@
1
+ /// <reference types="node" resolution-mode="require"/>
2
+ import { AudioFrame } from '@livekit/rtc-node';
3
+ import type { Span } from '@opentelemetry/api';
4
+ import { type ReadableStream } from 'stream/web';
5
+ import type { AdaptiveInterruptionDetector } from './interruption_detector.js';
6
+ import { type AgentSpeechEnded, type AgentSpeechStarted, type ApiConnectOptions, type Flush, type InterruptionSentinel, type OverlapSpeechEnded, type OverlapSpeechStarted, type OverlappingSpeechEvent } from './types.js';
7
+ export type { AgentSpeechEnded, AgentSpeechStarted, ApiConnectOptions, Flush, InterruptionSentinel, OverlapSpeechEnded, OverlapSpeechStarted, };
8
+ export declare class InterruptionStreamSentinel {
9
+ static agentSpeechStarted(): AgentSpeechStarted;
10
+ static agentSpeechEnded(): AgentSpeechEnded;
11
+ static overlapSpeechStarted(speechDuration: number, startedAt: number, userSpeakingSpan?: Span): OverlapSpeechStarted;
12
+ static overlapSpeechEnded(endedAt: number): OverlapSpeechEnded;
13
+ static flush(): Flush;
14
+ }
15
+ export declare class InterruptionStreamBase {
16
+ private inputStream;
17
+ private eventStream;
18
+ private resampler?;
19
+ private numRequests;
20
+ private userSpeakingSpan;
21
+ private overlapSpeechStartedAt;
22
+ private options;
23
+ private apiOptions;
24
+ private model;
25
+ private logger;
26
+ private wsReconnect?;
27
+ private transportOptions;
28
+ constructor(model: AdaptiveInterruptionDetector, apiOptions: Partial<ApiConnectOptions>);
29
+ /**
30
+ * Update stream options. For WebSocket transport, this triggers a reconnection.
31
+ */
32
+ updateOptions(options: {
33
+ threshold?: number;
34
+ minInterruptionDurationInS?: number;
35
+ }): Promise<void>;
36
+ private setupTransform;
37
+ private ensureInputNotEnded;
38
+ private ensureStreamsNotEnded;
39
+ private getResamplerFor;
40
+ stream(): ReadableStream<OverlappingSpeechEvent>;
41
+ pushFrame(frame: InterruptionSentinel | AudioFrame): Promise<void>;
42
+ flush(): Promise<void>;
43
+ endInput(): Promise<void>;
44
+ close(): Promise<void>;
45
+ }
46
+ //# sourceMappingURL=interruption_stream.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"interruption_stream.d.ts","sourceRoot":"","sources":["../../../src/inference/interruption/interruption_stream.ts"],"names":[],"mappings":";AAGA,OAAO,EAAE,UAAU,EAAkB,MAAM,mBAAmB,CAAC;AAC/D,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,oBAAoB,CAAC;AAC/C,OAAO,EAAE,KAAK,cAAc,EAAmB,MAAM,YAAY,CAAC;AASlE,OAAO,KAAK,EAAE,4BAA4B,EAAE,MAAM,4BAA4B,CAAC;AAC/E,OAAO,EACL,KAAK,gBAAgB,EACrB,KAAK,kBAAkB,EACvB,KAAK,iBAAiB,EACtB,KAAK,KAAK,EAEV,KAAK,oBAAoB,EACzB,KAAK,kBAAkB,EACvB,KAAK,oBAAoB,EACzB,KAAK,sBAAsB,EAC5B,MAAM,YAAY,CAAC;AAKpB,YAAY,EACV,gBAAgB,EAChB,kBAAkB,EAClB,iBAAiB,EACjB,KAAK,EACL,oBAAoB,EACpB,kBAAkB,EAClB,oBAAoB,GACrB,CAAC;AAEF,qBAAa,0BAA0B;IACrC,MAAM,CAAC,kBAAkB,IAAI,kBAAkB;IAI/C,MAAM,CAAC,gBAAgB,IAAI,gBAAgB;IAI3C,MAAM,CAAC,oBAAoB,CACzB,cAAc,EAAE,MAAM,EACtB,SAAS,EAAE,MAAM,EACjB,gBAAgB,CAAC,EAAE,IAAI,GACtB,oBAAoB;IAIvB,MAAM,CAAC,kBAAkB,CAAC,OAAO,EAAE,MAAM,GAAG,kBAAkB;IAI9D,MAAM,CAAC,KAAK,IAAI,KAAK;CAGtB;AAaD,qBAAa,sBAAsB;IACjC,OAAO,CAAC,WAAW,CAA+E;IAElG,OAAO,CAAC,WAAW,CAAyC;IAE5D,OAAO,CAAC,SAAS,CAAC,CAAiB;IAEnC,OAAO,CAAC,WAAW,CAAK;IAExB,OAAO,CAAC,gBAAgB,CAAmB;IAE3C,OAAO,CAAC,sBAAsB,CAAqB;IAEnD,OAAO,CAAC,OAAO,CAAsB;IAErC,OAAO,CAAC,UAAU,CAAoB;IAEtC,OAAO,CAAC,KAAK,CAA+B;IAE5C,OAAO,CAAC,MAAM,CAAS;IAGvB,OAAO,CAAC,WAAW,CAAC,CAAsB;IAG1C,OAAO,CAAC,gBAAgB,CAStB;gBAEU,KAAK,EAAE,4BAA4B,EAAE,UAAU,EAAE,OAAO,CAAC,iBAAiB,CAAC;IAyBvF;;OAEG;IACG,aAAa,CAAC,OAAO,EAAE;QAC3B,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,0BAA0B,CAAC,EAAE,MAAM,CAAC;KACrC,GAAG,OAAO,CAAC,IAAI,CAAC;IAgBjB,OAAO,CAAC,cAAc;IA8MtB,OAAO,CAAC,mBAAmB;IAM3B,OAAO,CAAC,qBAAqB;IAI7B,OAAO,CAAC,eAAe;IAOvB,MAAM,IAAI,cAAc,CAAC,sBAAsB,CAAC;IAI1C,SAAS,CAAC,KAAK,EAAE,oBAAoB,GAAG,UAAU,GAAG,OAAO,CAAC,IAAI,CAAC;IAiBlE,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAKtB,QAAQ,IAAI,OAAO,CAAC,IAAI,CAAC;IAKzB,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;CAI7B"}