@livekit/agents 0.7.9 → 1.0.0-next.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 (627) hide show
  1. package/dist/_exceptions.cjs +109 -0
  2. package/dist/_exceptions.cjs.map +1 -0
  3. package/dist/_exceptions.d.cts +64 -0
  4. package/dist/_exceptions.d.ts +64 -0
  5. package/dist/_exceptions.d.ts.map +1 -0
  6. package/dist/_exceptions.js +80 -0
  7. package/dist/_exceptions.js.map +1 -0
  8. package/dist/audio.cjs +10 -3
  9. package/dist/audio.cjs.map +1 -1
  10. package/dist/audio.d.cts +2 -0
  11. package/dist/audio.d.ts +2 -0
  12. package/dist/audio.d.ts.map +1 -1
  13. package/dist/audio.js +8 -2
  14. package/dist/audio.js.map +1 -1
  15. package/dist/cli.cjs +25 -0
  16. package/dist/cli.cjs.map +1 -1
  17. package/dist/cli.d.ts.map +1 -1
  18. package/dist/cli.js +25 -0
  19. package/dist/cli.js.map +1 -1
  20. package/dist/constants.cjs +6 -3
  21. package/dist/constants.cjs.map +1 -1
  22. package/dist/constants.d.cts +2 -1
  23. package/dist/constants.d.ts +2 -1
  24. package/dist/constants.d.ts.map +1 -1
  25. package/dist/constants.js +4 -2
  26. package/dist/constants.js.map +1 -1
  27. package/dist/http_server.cjs.map +1 -1
  28. package/dist/http_server.d.cts +1 -0
  29. package/dist/http_server.d.ts +1 -0
  30. package/dist/http_server.d.ts.map +1 -1
  31. package/dist/http_server.js.map +1 -1
  32. package/dist/index.cjs +27 -20
  33. package/dist/index.cjs.map +1 -1
  34. package/dist/index.d.cts +13 -10
  35. package/dist/index.d.ts +13 -10
  36. package/dist/index.d.ts.map +1 -1
  37. package/dist/index.js +15 -11
  38. package/dist/index.js.map +1 -1
  39. package/dist/inference_runner.cjs +0 -1
  40. package/dist/inference_runner.cjs.map +1 -1
  41. package/dist/inference_runner.d.cts +2 -3
  42. package/dist/inference_runner.d.ts +2 -3
  43. package/dist/inference_runner.d.ts.map +1 -1
  44. package/dist/inference_runner.js +0 -1
  45. package/dist/inference_runner.js.map +1 -1
  46. package/dist/ipc/inference_proc_executor.cjs +2 -2
  47. package/dist/ipc/inference_proc_executor.cjs.map +1 -1
  48. package/dist/ipc/inference_proc_executor.js +2 -2
  49. package/dist/ipc/inference_proc_executor.js.map +1 -1
  50. package/dist/ipc/job_executor.cjs.map +1 -1
  51. package/dist/ipc/job_executor.js.map +1 -1
  52. package/dist/ipc/job_proc_executor.cjs +1 -0
  53. package/dist/ipc/job_proc_executor.cjs.map +1 -1
  54. package/dist/ipc/job_proc_executor.js +1 -0
  55. package/dist/ipc/job_proc_executor.js.map +1 -1
  56. package/dist/ipc/job_proc_lazy_main.cjs +1 -1
  57. package/dist/ipc/job_proc_lazy_main.cjs.map +1 -1
  58. package/dist/ipc/job_proc_lazy_main.js +1 -1
  59. package/dist/ipc/job_proc_lazy_main.js.map +1 -1
  60. package/dist/ipc/supervised_proc.d.cts +1 -1
  61. package/dist/ipc/supervised_proc.d.ts +1 -1
  62. package/dist/ipc/supervised_proc.d.ts.map +1 -1
  63. package/dist/job.cjs +14 -2
  64. package/dist/job.cjs.map +1 -1
  65. package/dist/job.d.cts +8 -0
  66. package/dist/job.d.ts +8 -0
  67. package/dist/job.d.ts.map +1 -1
  68. package/dist/job.js +12 -1
  69. package/dist/job.js.map +1 -1
  70. package/dist/llm/chat_context.cjs +332 -82
  71. package/dist/llm/chat_context.cjs.map +1 -1
  72. package/dist/llm/chat_context.d.cts +152 -48
  73. package/dist/llm/chat_context.d.ts +152 -48
  74. package/dist/llm/chat_context.d.ts.map +1 -1
  75. package/dist/llm/chat_context.js +327 -81
  76. package/dist/llm/chat_context.js.map +1 -1
  77. package/dist/llm/chat_context.test.cjs +380 -0
  78. package/dist/llm/chat_context.test.cjs.map +1 -0
  79. package/dist/llm/chat_context.test.js +385 -0
  80. package/dist/llm/chat_context.test.js.map +1 -0
  81. package/dist/llm/index.cjs +37 -8
  82. package/dist/llm/index.cjs.map +1 -1
  83. package/dist/llm/index.d.cts +7 -3
  84. package/dist/llm/index.d.ts +7 -3
  85. package/dist/llm/index.d.ts.map +1 -1
  86. package/dist/llm/index.js +39 -9
  87. package/dist/llm/index.js.map +1 -1
  88. package/dist/llm/llm.cjs +98 -33
  89. package/dist/llm/llm.cjs.map +1 -1
  90. package/dist/llm/llm.d.cts +50 -24
  91. package/dist/llm/llm.d.ts +50 -24
  92. package/dist/llm/llm.d.ts.map +1 -1
  93. package/dist/llm/llm.js +99 -33
  94. package/dist/llm/llm.js.map +1 -1
  95. package/dist/llm/provider_format/google.cjs +128 -0
  96. package/dist/llm/provider_format/google.cjs.map +1 -0
  97. package/dist/llm/provider_format/google.d.cts +6 -0
  98. package/dist/llm/provider_format/google.d.ts +6 -0
  99. package/dist/llm/provider_format/google.d.ts.map +1 -0
  100. package/dist/llm/provider_format/google.js +104 -0
  101. package/dist/llm/provider_format/google.js.map +1 -0
  102. package/dist/llm/provider_format/google.test.cjs +676 -0
  103. package/dist/llm/provider_format/google.test.cjs.map +1 -0
  104. package/dist/llm/provider_format/google.test.js +675 -0
  105. package/dist/llm/provider_format/google.test.js.map +1 -0
  106. package/dist/llm/provider_format/index.cjs +40 -0
  107. package/dist/llm/provider_format/index.cjs.map +1 -0
  108. package/dist/llm/provider_format/index.d.cts +4 -0
  109. package/dist/llm/provider_format/index.d.ts +4 -0
  110. package/dist/llm/provider_format/index.d.ts.map +1 -0
  111. package/dist/llm/provider_format/index.js +16 -0
  112. package/dist/llm/provider_format/index.js.map +1 -0
  113. package/dist/llm/provider_format/openai.cjs +116 -0
  114. package/dist/llm/provider_format/openai.cjs.map +1 -0
  115. package/dist/llm/provider_format/openai.d.cts +3 -0
  116. package/dist/llm/provider_format/openai.d.ts +3 -0
  117. package/dist/llm/provider_format/openai.d.ts.map +1 -0
  118. package/dist/llm/provider_format/openai.js +92 -0
  119. package/dist/llm/provider_format/openai.js.map +1 -0
  120. package/dist/llm/provider_format/openai.test.cjs +490 -0
  121. package/dist/llm/provider_format/openai.test.cjs.map +1 -0
  122. package/dist/llm/provider_format/openai.test.js +489 -0
  123. package/dist/llm/provider_format/openai.test.js.map +1 -0
  124. package/dist/llm/provider_format/utils.cjs +146 -0
  125. package/dist/llm/provider_format/utils.cjs.map +1 -0
  126. package/dist/llm/provider_format/utils.d.cts +38 -0
  127. package/dist/llm/provider_format/utils.d.ts +38 -0
  128. package/dist/llm/provider_format/utils.d.ts.map +1 -0
  129. package/dist/llm/provider_format/utils.js +122 -0
  130. package/dist/llm/provider_format/utils.js.map +1 -0
  131. package/dist/llm/realtime.cjs +77 -0
  132. package/dist/llm/realtime.cjs.map +1 -0
  133. package/dist/llm/realtime.d.cts +98 -0
  134. package/dist/llm/realtime.d.ts +98 -0
  135. package/dist/llm/realtime.d.ts.map +1 -0
  136. package/dist/llm/realtime.js +52 -0
  137. package/dist/llm/realtime.js.map +1 -0
  138. package/dist/llm/remote_chat_context.cjs +112 -0
  139. package/dist/llm/remote_chat_context.cjs.map +1 -0
  140. package/dist/llm/remote_chat_context.d.cts +23 -0
  141. package/dist/llm/remote_chat_context.d.ts +23 -0
  142. package/dist/llm/remote_chat_context.d.ts.map +1 -0
  143. package/dist/llm/remote_chat_context.js +88 -0
  144. package/dist/llm/remote_chat_context.js.map +1 -0
  145. package/dist/llm/remote_chat_context.test.cjs +225 -0
  146. package/dist/llm/remote_chat_context.test.cjs.map +1 -0
  147. package/dist/llm/remote_chat_context.test.js +224 -0
  148. package/dist/llm/remote_chat_context.test.js.map +1 -0
  149. package/dist/llm/tool_context.cjs +111 -0
  150. package/dist/llm/tool_context.cjs.map +1 -0
  151. package/dist/llm/tool_context.d.cts +125 -0
  152. package/dist/llm/tool_context.d.ts +125 -0
  153. package/dist/llm/tool_context.d.ts.map +1 -0
  154. package/dist/llm/tool_context.js +80 -0
  155. package/dist/llm/tool_context.js.map +1 -0
  156. package/dist/llm/tool_context.test.cjs +162 -0
  157. package/dist/llm/tool_context.test.cjs.map +1 -0
  158. package/dist/llm/tool_context.test.js +161 -0
  159. package/dist/llm/tool_context.test.js.map +1 -0
  160. package/dist/llm/tool_context.type.test.cjs +92 -0
  161. package/dist/llm/tool_context.type.test.cjs.map +1 -0
  162. package/dist/llm/tool_context.type.test.js +91 -0
  163. package/dist/llm/tool_context.type.test.js.map +1 -0
  164. package/dist/llm/utils.cjs +260 -0
  165. package/dist/llm/utils.cjs.map +1 -0
  166. package/dist/llm/utils.d.cts +42 -0
  167. package/dist/llm/utils.d.ts +42 -0
  168. package/dist/llm/utils.d.ts.map +1 -0
  169. package/dist/llm/utils.js +223 -0
  170. package/dist/llm/utils.js.map +1 -0
  171. package/dist/llm/utils.test.cjs +513 -0
  172. package/dist/llm/utils.test.cjs.map +1 -0
  173. package/dist/llm/utils.test.js +490 -0
  174. package/dist/llm/utils.test.js.map +1 -0
  175. package/dist/metrics/base.cjs +0 -27
  176. package/dist/metrics/base.cjs.map +1 -1
  177. package/dist/metrics/base.d.cts +105 -63
  178. package/dist/metrics/base.d.ts +105 -63
  179. package/dist/metrics/base.d.ts.map +1 -1
  180. package/dist/metrics/base.js +0 -19
  181. package/dist/metrics/base.js.map +1 -1
  182. package/dist/metrics/index.cjs +0 -3
  183. package/dist/metrics/index.cjs.map +1 -1
  184. package/dist/metrics/index.d.cts +2 -3
  185. package/dist/metrics/index.d.ts +2 -3
  186. package/dist/metrics/index.d.ts.map +1 -1
  187. package/dist/metrics/index.js +0 -2
  188. package/dist/metrics/index.js.map +1 -1
  189. package/dist/metrics/usage_collector.cjs +17 -12
  190. package/dist/metrics/usage_collector.cjs.map +1 -1
  191. package/dist/metrics/usage_collector.d.cts +3 -2
  192. package/dist/metrics/usage_collector.d.ts +3 -2
  193. package/dist/metrics/usage_collector.d.ts.map +1 -1
  194. package/dist/metrics/usage_collector.js +17 -12
  195. package/dist/metrics/usage_collector.js.map +1 -1
  196. package/dist/metrics/utils.cjs +22 -59
  197. package/dist/metrics/utils.cjs.map +1 -1
  198. package/dist/metrics/utils.d.cts +1 -8
  199. package/dist/metrics/utils.d.ts +1 -8
  200. package/dist/metrics/utils.d.ts.map +1 -1
  201. package/dist/metrics/utils.js +22 -52
  202. package/dist/metrics/utils.js.map +1 -1
  203. package/dist/multimodal/index.cjs +0 -2
  204. package/dist/multimodal/index.cjs.map +1 -1
  205. package/dist/multimodal/index.d.cts +0 -1
  206. package/dist/multimodal/index.d.ts +0 -1
  207. package/dist/multimodal/index.d.ts.map +1 -1
  208. package/dist/multimodal/index.js +0 -1
  209. package/dist/multimodal/index.js.map +1 -1
  210. package/dist/plugin.cjs +24 -8
  211. package/dist/plugin.cjs.map +1 -1
  212. package/dist/plugin.d.cts +18 -4
  213. package/dist/plugin.d.ts +18 -4
  214. package/dist/plugin.d.ts.map +1 -1
  215. package/dist/plugin.js +22 -7
  216. package/dist/plugin.js.map +1 -1
  217. package/dist/stream/deferred_stream.cjs +98 -0
  218. package/dist/stream/deferred_stream.cjs.map +1 -0
  219. package/dist/stream/deferred_stream.d.cts +27 -0
  220. package/dist/stream/deferred_stream.d.ts +27 -0
  221. package/dist/stream/deferred_stream.d.ts.map +1 -0
  222. package/dist/stream/deferred_stream.js +73 -0
  223. package/dist/stream/deferred_stream.js.map +1 -0
  224. package/dist/stream/deferred_stream.test.cjs +527 -0
  225. package/dist/stream/deferred_stream.test.cjs.map +1 -0
  226. package/dist/stream/deferred_stream.test.js +526 -0
  227. package/dist/stream/deferred_stream.test.js.map +1 -0
  228. package/dist/stream/identity_transform.cjs +42 -0
  229. package/dist/stream/identity_transform.cjs.map +1 -0
  230. package/dist/stream/identity_transform.d.cts +6 -0
  231. package/dist/stream/identity_transform.d.ts +6 -0
  232. package/dist/stream/identity_transform.d.ts.map +1 -0
  233. package/dist/stream/identity_transform.js +18 -0
  234. package/dist/stream/identity_transform.js.map +1 -0
  235. package/dist/stream/identity_transform.test.cjs +125 -0
  236. package/dist/stream/identity_transform.test.cjs.map +1 -0
  237. package/dist/stream/identity_transform.test.js +124 -0
  238. package/dist/stream/identity_transform.test.js.map +1 -0
  239. package/dist/stream/index.cjs +38 -0
  240. package/dist/stream/index.cjs.map +1 -0
  241. package/dist/stream/index.d.cts +5 -0
  242. package/dist/stream/index.d.ts +5 -0
  243. package/dist/stream/index.d.ts.map +1 -0
  244. package/dist/stream/index.js +11 -0
  245. package/dist/stream/index.js.map +1 -0
  246. package/dist/stream/merge_readable_streams.cjs +59 -0
  247. package/dist/stream/merge_readable_streams.cjs.map +1 -0
  248. package/dist/stream/merge_readable_streams.d.cts +4 -0
  249. package/dist/stream/merge_readable_streams.d.ts +4 -0
  250. package/dist/stream/merge_readable_streams.d.ts.map +1 -0
  251. package/dist/stream/merge_readable_streams.js +35 -0
  252. package/dist/stream/merge_readable_streams.js.map +1 -0
  253. package/dist/stream/stream_channel.cjs +47 -0
  254. package/dist/stream/stream_channel.cjs.map +1 -0
  255. package/dist/stream/stream_channel.d.cts +9 -0
  256. package/dist/stream/stream_channel.d.ts +9 -0
  257. package/dist/stream/stream_channel.d.ts.map +1 -0
  258. package/dist/stream/stream_channel.js +23 -0
  259. package/dist/stream/stream_channel.js.map +1 -0
  260. package/dist/stream/stream_channel.test.cjs +97 -0
  261. package/dist/stream/stream_channel.test.cjs.map +1 -0
  262. package/dist/stream/stream_channel.test.js +96 -0
  263. package/dist/stream/stream_channel.test.js.map +1 -0
  264. package/dist/stt/stream_adapter.cjs +3 -4
  265. package/dist/stt/stream_adapter.cjs.map +1 -1
  266. package/dist/stt/stream_adapter.d.cts +1 -0
  267. package/dist/stt/stream_adapter.d.ts +1 -0
  268. package/dist/stt/stream_adapter.d.ts.map +1 -1
  269. package/dist/stt/stream_adapter.js +3 -4
  270. package/dist/stt/stream_adapter.js.map +1 -1
  271. package/dist/stt/stt.cjs +101 -10
  272. package/dist/stt/stt.cjs.map +1 -1
  273. package/dist/stt/stt.d.cts +26 -5
  274. package/dist/stt/stt.d.ts +26 -5
  275. package/dist/stt/stt.d.ts.map +1 -1
  276. package/dist/stt/stt.js +102 -11
  277. package/dist/stt/stt.js.map +1 -1
  278. package/dist/tokenize/basic/basic.cjs +10 -5
  279. package/dist/tokenize/basic/basic.cjs.map +1 -1
  280. package/dist/tokenize/basic/basic.d.cts +7 -1
  281. package/dist/tokenize/basic/basic.d.ts +7 -1
  282. package/dist/tokenize/basic/basic.d.ts.map +1 -1
  283. package/dist/tokenize/basic/basic.js +10 -5
  284. package/dist/tokenize/basic/basic.js.map +1 -1
  285. package/dist/tokenize/basic/sentence.cjs +14 -6
  286. package/dist/tokenize/basic/sentence.cjs.map +1 -1
  287. package/dist/tokenize/basic/sentence.d.cts +1 -1
  288. package/dist/tokenize/basic/sentence.d.ts +1 -1
  289. package/dist/tokenize/basic/sentence.d.ts.map +1 -1
  290. package/dist/tokenize/basic/sentence.js +14 -6
  291. package/dist/tokenize/basic/sentence.js.map +1 -1
  292. package/dist/tokenize/token_stream.cjs +5 -3
  293. package/dist/tokenize/token_stream.cjs.map +1 -1
  294. package/dist/tokenize/token_stream.d.cts +1 -0
  295. package/dist/tokenize/token_stream.d.ts +1 -0
  296. package/dist/tokenize/token_stream.d.ts.map +1 -1
  297. package/dist/tokenize/token_stream.js +6 -4
  298. package/dist/tokenize/token_stream.js.map +1 -1
  299. package/dist/transcription.cjs +1 -2
  300. package/dist/transcription.cjs.map +1 -1
  301. package/dist/transcription.d.ts.map +1 -1
  302. package/dist/transcription.js +2 -3
  303. package/dist/transcription.js.map +1 -1
  304. package/dist/tts/index.cjs +2 -4
  305. package/dist/tts/index.cjs.map +1 -1
  306. package/dist/tts/index.d.cts +1 -1
  307. package/dist/tts/index.d.ts +1 -1
  308. package/dist/tts/index.d.ts.map +1 -1
  309. package/dist/tts/index.js +1 -3
  310. package/dist/tts/index.js.map +1 -1
  311. package/dist/tts/stream_adapter.cjs +26 -13
  312. package/dist/tts/stream_adapter.cjs.map +1 -1
  313. package/dist/tts/stream_adapter.d.cts +1 -1
  314. package/dist/tts/stream_adapter.d.ts +1 -1
  315. package/dist/tts/stream_adapter.d.ts.map +1 -1
  316. package/dist/tts/stream_adapter.js +27 -14
  317. package/dist/tts/stream_adapter.js.map +1 -1
  318. package/dist/tts/tts.cjs +157 -25
  319. package/dist/tts/tts.cjs.map +1 -1
  320. package/dist/tts/tts.d.cts +29 -5
  321. package/dist/tts/tts.d.ts +29 -5
  322. package/dist/tts/tts.d.ts.map +1 -1
  323. package/dist/tts/tts.js +157 -24
  324. package/dist/tts/tts.js.map +1 -1
  325. package/dist/types.cjs +60 -0
  326. package/dist/types.cjs.map +1 -0
  327. package/dist/types.d.cts +13 -0
  328. package/dist/types.d.ts +13 -0
  329. package/dist/types.d.ts.map +1 -0
  330. package/dist/types.js +35 -0
  331. package/dist/types.js.map +1 -0
  332. package/dist/utils.cjs +281 -27
  333. package/dist/utils.cjs.map +1 -1
  334. package/dist/utils.d.cts +134 -9
  335. package/dist/utils.d.ts +134 -9
  336. package/dist/utils.d.ts.map +1 -1
  337. package/dist/utils.js +265 -26
  338. package/dist/utils.js.map +1 -1
  339. package/dist/utils.test.cjs +492 -0
  340. package/dist/utils.test.cjs.map +1 -0
  341. package/dist/utils.test.js +498 -0
  342. package/dist/utils.test.js.map +1 -0
  343. package/dist/vad.cjs +76 -20
  344. package/dist/vad.cjs.map +1 -1
  345. package/dist/vad.d.cts +25 -5
  346. package/dist/vad.d.ts +25 -5
  347. package/dist/vad.d.ts.map +1 -1
  348. package/dist/vad.js +76 -20
  349. package/dist/vad.js.map +1 -1
  350. package/dist/voice/agent.cjs +245 -0
  351. package/dist/voice/agent.cjs.map +1 -0
  352. package/dist/voice/agent.d.cts +78 -0
  353. package/dist/voice/agent.d.ts +78 -0
  354. package/dist/voice/agent.d.ts.map +1 -0
  355. package/dist/voice/agent.js +220 -0
  356. package/dist/voice/agent.js.map +1 -0
  357. package/dist/voice/agent.test.cjs +61 -0
  358. package/dist/voice/agent.test.cjs.map +1 -0
  359. package/dist/voice/agent.test.js +60 -0
  360. package/dist/voice/agent.test.js.map +1 -0
  361. package/dist/voice/agent_activity.cjs +1453 -0
  362. package/dist/voice/agent_activity.cjs.map +1 -0
  363. package/dist/voice/agent_activity.d.cts +94 -0
  364. package/dist/voice/agent_activity.d.ts +94 -0
  365. package/dist/voice/agent_activity.d.ts.map +1 -0
  366. package/dist/voice/agent_activity.js +1449 -0
  367. package/dist/voice/agent_activity.js.map +1 -0
  368. package/dist/voice/agent_session.cjs +312 -0
  369. package/dist/voice/agent_session.cjs.map +1 -0
  370. package/dist/voice/agent_session.d.cts +121 -0
  371. package/dist/voice/agent_session.d.ts +121 -0
  372. package/dist/voice/agent_session.d.ts.map +1 -0
  373. package/dist/voice/agent_session.js +295 -0
  374. package/dist/voice/agent_session.js.map +1 -0
  375. package/dist/voice/audio_recognition.cjs +375 -0
  376. package/dist/voice/audio_recognition.cjs.map +1 -0
  377. package/dist/voice/audio_recognition.d.cts +80 -0
  378. package/dist/voice/audio_recognition.d.ts +80 -0
  379. package/dist/voice/audio_recognition.d.ts.map +1 -0
  380. package/dist/voice/audio_recognition.js +351 -0
  381. package/dist/voice/audio_recognition.js.map +1 -0
  382. package/dist/voice/events.cjs +145 -0
  383. package/dist/voice/events.cjs.map +1 -0
  384. package/dist/voice/events.d.cts +124 -0
  385. package/dist/voice/events.d.ts +124 -0
  386. package/dist/voice/events.d.ts.map +1 -0
  387. package/dist/voice/events.js +110 -0
  388. package/dist/voice/events.js.map +1 -0
  389. package/dist/voice/generation.cjs +700 -0
  390. package/dist/voice/generation.cjs.map +1 -0
  391. package/dist/voice/generation.d.cts +115 -0
  392. package/dist/voice/generation.d.ts +115 -0
  393. package/dist/voice/generation.d.ts.map +1 -0
  394. package/dist/voice/generation.js +672 -0
  395. package/dist/voice/generation.js.map +1 -0
  396. package/dist/voice/index.cjs +40 -0
  397. package/dist/voice/index.cjs.map +1 -0
  398. package/dist/voice/index.d.cts +5 -0
  399. package/dist/voice/index.d.ts +5 -0
  400. package/dist/voice/index.d.ts.map +1 -0
  401. package/dist/voice/index.js +11 -0
  402. package/dist/voice/index.js.map +1 -0
  403. package/dist/voice/io.cjs +245 -0
  404. package/dist/voice/io.cjs.map +1 -0
  405. package/dist/voice/io.d.cts +101 -0
  406. package/dist/voice/io.d.ts +101 -0
  407. package/dist/voice/io.d.ts.map +1 -0
  408. package/dist/voice/io.js +217 -0
  409. package/dist/voice/io.js.map +1 -0
  410. package/dist/voice/room_io/_input.cjs +121 -0
  411. package/dist/voice/room_io/_input.cjs.map +1 -0
  412. package/dist/voice/room_io/_input.d.cts +24 -0
  413. package/dist/voice/room_io/_input.d.ts +24 -0
  414. package/dist/voice/room_io/_input.d.ts.map +1 -0
  415. package/dist/voice/room_io/_input.js +102 -0
  416. package/dist/voice/room_io/_input.js.map +1 -0
  417. package/dist/voice/room_io/_output.cjs +358 -0
  418. package/dist/voice/room_io/_output.cjs.map +1 -0
  419. package/dist/voice/room_io/_output.d.cts +75 -0
  420. package/dist/voice/room_io/_output.d.ts +75 -0
  421. package/dist/voice/room_io/_output.d.ts.map +1 -0
  422. package/dist/voice/room_io/_output.js +342 -0
  423. package/dist/voice/room_io/_output.js.map +1 -0
  424. package/dist/voice/room_io/index.cjs +25 -0
  425. package/dist/voice/room_io/index.cjs.map +1 -0
  426. package/dist/voice/room_io/index.d.cts +3 -0
  427. package/dist/voice/room_io/index.d.ts +3 -0
  428. package/dist/voice/room_io/index.d.ts.map +1 -0
  429. package/dist/voice/room_io/index.js +3 -0
  430. package/dist/voice/room_io/index.js.map +1 -0
  431. package/dist/voice/room_io/room_io.cjs +370 -0
  432. package/dist/voice/room_io/room_io.cjs.map +1 -0
  433. package/dist/voice/room_io/room_io.d.cts +73 -0
  434. package/dist/voice/room_io/room_io.d.ts +73 -0
  435. package/dist/voice/room_io/room_io.d.ts.map +1 -0
  436. package/dist/voice/room_io/room_io.js +361 -0
  437. package/dist/voice/room_io/room_io.js.map +1 -0
  438. package/dist/{pipeline/index.cjs → voice/run_context.cjs} +16 -11
  439. package/dist/voice/run_context.cjs.map +1 -0
  440. package/dist/voice/run_context.d.cts +12 -0
  441. package/dist/voice/run_context.d.ts +12 -0
  442. package/dist/voice/run_context.d.ts.map +1 -0
  443. package/dist/voice/run_context.js +14 -0
  444. package/dist/voice/run_context.js.map +1 -0
  445. package/dist/voice/speech_handle.cjs +105 -0
  446. package/dist/voice/speech_handle.cjs.map +1 -0
  447. package/dist/voice/speech_handle.d.cts +46 -0
  448. package/dist/voice/speech_handle.d.ts +46 -0
  449. package/dist/voice/speech_handle.d.ts.map +1 -0
  450. package/dist/voice/speech_handle.js +81 -0
  451. package/dist/voice/speech_handle.js.map +1 -0
  452. package/dist/voice/transcription/_utils.cjs +45 -0
  453. package/dist/voice/transcription/_utils.cjs.map +1 -0
  454. package/dist/voice/transcription/_utils.d.cts +3 -0
  455. package/dist/voice/transcription/_utils.d.ts +3 -0
  456. package/dist/voice/transcription/_utils.d.ts.map +1 -0
  457. package/dist/voice/transcription/_utils.js +21 -0
  458. package/dist/voice/transcription/_utils.js.map +1 -0
  459. package/dist/voice/transcription/index.cjs +23 -0
  460. package/dist/voice/transcription/index.cjs.map +1 -0
  461. package/dist/voice/transcription/index.d.cts +2 -0
  462. package/dist/voice/transcription/index.d.ts +2 -0
  463. package/dist/voice/transcription/index.d.ts.map +1 -0
  464. package/dist/voice/transcription/index.js +2 -0
  465. package/dist/voice/transcription/index.js.map +1 -0
  466. package/dist/voice/transcription/synchronizer.cjs +380 -0
  467. package/dist/voice/transcription/synchronizer.cjs.map +1 -0
  468. package/dist/voice/transcription/synchronizer.d.cts +86 -0
  469. package/dist/voice/transcription/synchronizer.d.ts +86 -0
  470. package/dist/voice/transcription/synchronizer.d.ts.map +1 -0
  471. package/dist/voice/transcription/synchronizer.js +355 -0
  472. package/dist/voice/transcription/synchronizer.js.map +1 -0
  473. package/dist/worker.cjs +22 -4
  474. package/dist/worker.cjs.map +1 -1
  475. package/dist/worker.d.cts +1 -1
  476. package/dist/worker.d.ts +1 -1
  477. package/dist/worker.d.ts.map +1 -1
  478. package/dist/worker.js +22 -4
  479. package/dist/worker.js.map +1 -1
  480. package/package.json +9 -2
  481. package/src/_exceptions.ts +137 -0
  482. package/src/audio.ts +12 -1
  483. package/src/cli.ts +37 -0
  484. package/src/constants.ts +2 -1
  485. package/src/http_server.ts +1 -0
  486. package/src/index.ts +13 -10
  487. package/src/inference_runner.ts +2 -3
  488. package/src/ipc/inference_proc_executor.ts +2 -2
  489. package/src/ipc/job_executor.ts +1 -1
  490. package/src/ipc/job_proc_executor.ts +1 -1
  491. package/src/ipc/job_proc_lazy_main.ts +1 -1
  492. package/src/job.ts +18 -0
  493. package/src/llm/__snapshots__/chat_context.test.ts.snap +527 -0
  494. package/src/llm/__snapshots__/tool_context.test.ts.snap +177 -0
  495. package/src/llm/__snapshots__/utils.test.ts.snap +65 -0
  496. package/src/llm/chat_context.test.ts +450 -0
  497. package/src/llm/chat_context.ts +501 -103
  498. package/src/llm/index.ts +53 -18
  499. package/src/llm/llm.ts +149 -50
  500. package/src/llm/provider_format/google.test.ts +772 -0
  501. package/src/llm/provider_format/google.ts +130 -0
  502. package/src/llm/provider_format/index.ts +23 -0
  503. package/src/llm/provider_format/openai.test.ts +581 -0
  504. package/src/llm/provider_format/openai.ts +118 -0
  505. package/src/llm/provider_format/utils.ts +183 -0
  506. package/src/llm/realtime.ts +151 -0
  507. package/src/llm/remote_chat_context.test.ts +290 -0
  508. package/src/llm/remote_chat_context.ts +114 -0
  509. package/src/llm/tool_context.test.ts +198 -0
  510. package/src/llm/tool_context.ts +259 -0
  511. package/src/llm/tool_context.type.test.ts +115 -0
  512. package/src/llm/utils.test.ts +670 -0
  513. package/src/llm/utils.ts +324 -0
  514. package/src/metrics/base.ts +110 -78
  515. package/src/metrics/index.ts +3 -9
  516. package/src/metrics/usage_collector.ts +19 -13
  517. package/src/metrics/utils.ts +24 -69
  518. package/src/multimodal/index.ts +0 -1
  519. package/src/plugin.ts +26 -8
  520. package/src/stream/deferred_stream.test.ts +755 -0
  521. package/src/stream/deferred_stream.ts +110 -0
  522. package/src/stream/identity_transform.test.ts +179 -0
  523. package/src/stream/identity_transform.ts +18 -0
  524. package/src/stream/index.ts +7 -0
  525. package/src/stream/merge_readable_streams.ts +40 -0
  526. package/src/stream/stream_channel.test.ts +129 -0
  527. package/src/stream/stream_channel.ts +32 -0
  528. package/src/stt/stream_adapter.ts +3 -5
  529. package/src/stt/stt.ts +135 -17
  530. package/src/tokenize/basic/basic.ts +13 -5
  531. package/src/tokenize/basic/sentence.ts +20 -6
  532. package/src/tokenize/token_stream.ts +7 -4
  533. package/src/transcription.ts +2 -3
  534. package/src/tts/index.ts +0 -1
  535. package/src/tts/stream_adapter.ts +42 -16
  536. package/src/tts/tts.ts +203 -21
  537. package/src/types.ts +42 -0
  538. package/src/utils.test.ts +658 -0
  539. package/src/utils.ts +375 -44
  540. package/src/vad.ts +90 -22
  541. package/src/voice/agent.test.ts +80 -0
  542. package/src/voice/agent.ts +332 -0
  543. package/src/voice/agent_activity.ts +1913 -0
  544. package/src/voice/agent_session.ts +460 -0
  545. package/src/voice/audio_recognition.ts +474 -0
  546. package/src/voice/events.ts +252 -0
  547. package/src/voice/generation.ts +881 -0
  548. package/src/voice/index.ts +7 -0
  549. package/src/voice/io.ts +304 -0
  550. package/src/voice/room_io/_input.ts +144 -0
  551. package/src/voice/room_io/_output.ts +436 -0
  552. package/src/voice/room_io/index.ts +5 -0
  553. package/src/voice/room_io/room_io.ts +495 -0
  554. package/src/voice/run_context.ts +20 -0
  555. package/src/voice/speech_handle.ts +104 -0
  556. package/src/voice/transcription/_utils.ts +25 -0
  557. package/src/voice/transcription/index.ts +4 -0
  558. package/src/voice/transcription/synchronizer.ts +478 -0
  559. package/src/worker.ts +22 -2
  560. package/dist/llm/function_context.cjs +0 -103
  561. package/dist/llm/function_context.cjs.map +0 -1
  562. package/dist/llm/function_context.d.cts +0 -47
  563. package/dist/llm/function_context.d.ts +0 -47
  564. package/dist/llm/function_context.d.ts.map +0 -1
  565. package/dist/llm/function_context.js +0 -78
  566. package/dist/llm/function_context.js.map +0 -1
  567. package/dist/llm/function_context.test.cjs +0 -218
  568. package/dist/llm/function_context.test.cjs.map +0 -1
  569. package/dist/llm/function_context.test.js +0 -217
  570. package/dist/llm/function_context.test.js.map +0 -1
  571. package/dist/multimodal/multimodal_agent.cjs +0 -486
  572. package/dist/multimodal/multimodal_agent.cjs.map +0 -1
  573. package/dist/multimodal/multimodal_agent.d.cts +0 -48
  574. package/dist/multimodal/multimodal_agent.d.ts +0 -48
  575. package/dist/multimodal/multimodal_agent.d.ts.map +0 -1
  576. package/dist/multimodal/multimodal_agent.js +0 -461
  577. package/dist/multimodal/multimodal_agent.js.map +0 -1
  578. package/dist/pipeline/agent_output.cjs +0 -197
  579. package/dist/pipeline/agent_output.cjs.map +0 -1
  580. package/dist/pipeline/agent_output.d.cts +0 -33
  581. package/dist/pipeline/agent_output.d.ts +0 -33
  582. package/dist/pipeline/agent_output.d.ts.map +0 -1
  583. package/dist/pipeline/agent_output.js +0 -172
  584. package/dist/pipeline/agent_output.js.map +0 -1
  585. package/dist/pipeline/agent_playout.cjs +0 -175
  586. package/dist/pipeline/agent_playout.cjs.map +0 -1
  587. package/dist/pipeline/agent_playout.d.cts +0 -40
  588. package/dist/pipeline/agent_playout.d.ts +0 -40
  589. package/dist/pipeline/agent_playout.d.ts.map +0 -1
  590. package/dist/pipeline/agent_playout.js +0 -139
  591. package/dist/pipeline/agent_playout.js.map +0 -1
  592. package/dist/pipeline/human_input.cjs +0 -171
  593. package/dist/pipeline/human_input.cjs.map +0 -1
  594. package/dist/pipeline/human_input.d.cts +0 -30
  595. package/dist/pipeline/human_input.d.ts +0 -30
  596. package/dist/pipeline/human_input.d.ts.map +0 -1
  597. package/dist/pipeline/human_input.js +0 -146
  598. package/dist/pipeline/human_input.js.map +0 -1
  599. package/dist/pipeline/index.cjs.map +0 -1
  600. package/dist/pipeline/index.d.cts +0 -2
  601. package/dist/pipeline/index.d.ts +0 -2
  602. package/dist/pipeline/index.d.ts.map +0 -1
  603. package/dist/pipeline/index.js +0 -11
  604. package/dist/pipeline/index.js.map +0 -1
  605. package/dist/pipeline/pipeline_agent.cjs +0 -859
  606. package/dist/pipeline/pipeline_agent.cjs.map +0 -1
  607. package/dist/pipeline/pipeline_agent.d.cts +0 -150
  608. package/dist/pipeline/pipeline_agent.d.ts +0 -150
  609. package/dist/pipeline/pipeline_agent.d.ts.map +0 -1
  610. package/dist/pipeline/pipeline_agent.js +0 -837
  611. package/dist/pipeline/pipeline_agent.js.map +0 -1
  612. package/dist/pipeline/speech_handle.cjs +0 -176
  613. package/dist/pipeline/speech_handle.cjs.map +0 -1
  614. package/dist/pipeline/speech_handle.d.cts +0 -37
  615. package/dist/pipeline/speech_handle.d.ts +0 -37
  616. package/dist/pipeline/speech_handle.d.ts.map +0 -1
  617. package/dist/pipeline/speech_handle.js +0 -152
  618. package/dist/pipeline/speech_handle.js.map +0 -1
  619. package/src/llm/function_context.test.ts +0 -248
  620. package/src/llm/function_context.ts +0 -142
  621. package/src/multimodal/multimodal_agent.ts +0 -592
  622. package/src/pipeline/agent_output.ts +0 -219
  623. package/src/pipeline/agent_playout.ts +0 -192
  624. package/src/pipeline/human_input.ts +0 -188
  625. package/src/pipeline/index.ts +0 -15
  626. package/src/pipeline/pipeline_agent.ts +0 -1197
  627. package/src/pipeline/speech_handle.ts +0 -201
@@ -0,0 +1,581 @@
1
+ // SPDX-FileCopyrightText: 2025 LiveKit, Inc.
2
+ //
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ import { VideoBufferType, VideoFrame } from '@livekit/rtc-node';
5
+ import { beforeEach, describe, expect, it, vi } from 'vitest';
6
+ import { initializeLogger } from '../../log.js';
7
+ import { ChatContext, FunctionCall, FunctionCallOutput } from '../chat_context.js';
8
+ import { serializeImage } from '../utils.js';
9
+ import { toChatCtx } from './openai.js';
10
+
11
+ // Mock the serializeImage function
12
+ vi.mock('../utils.js', () => ({
13
+ serializeImage: vi.fn(),
14
+ }));
15
+
16
+ describe('toChatCtx', () => {
17
+ const serializeImageMock = vi.mocked(serializeImage);
18
+
19
+ // initialize logger at start of test
20
+ initializeLogger({ level: 'silent', pretty: false });
21
+
22
+ beforeEach(async () => {
23
+ vi.clearAllMocks();
24
+ });
25
+
26
+ it('should convert simple text messages', async () => {
27
+ const ctx = ChatContext.empty();
28
+ ctx.addMessage({ role: 'user', content: 'Hello' });
29
+ ctx.addMessage({ role: 'assistant', content: 'Hi there!' });
30
+
31
+ const result = await toChatCtx(ctx);
32
+
33
+ // Messages should be in the result, order may vary due to ID-based sorting
34
+ expect(result).toHaveLength(2);
35
+ expect(result[0]).toEqual({ role: 'user', content: 'Hello' });
36
+ expect(result[1]).toEqual({ role: 'assistant', content: 'Hi there!' });
37
+ });
38
+
39
+ it('should handle system messages', async () => {
40
+ const ctx = ChatContext.empty();
41
+ ctx.addMessage({ role: 'system', content: 'You are a helpful assistant' });
42
+ ctx.addMessage({ role: 'user', content: 'Hello' });
43
+
44
+ const result = await toChatCtx(ctx);
45
+
46
+ // Messages should be in the result, order may vary due to ID-based sorting
47
+ expect(result).toHaveLength(2);
48
+ expect(result[0]).toEqual({ role: 'system', content: 'You are a helpful assistant' });
49
+ expect(result[1]).toEqual({ role: 'user', content: 'Hello' });
50
+ });
51
+
52
+ it('should handle multi-line text content', async () => {
53
+ const ctx = ChatContext.empty();
54
+ ctx.addMessage({ role: 'user', content: ['Line 1', 'Line 2', 'Line 3'] });
55
+
56
+ const result = await toChatCtx(ctx);
57
+
58
+ expect(result).toHaveLength(1);
59
+ expect(result[0]).toEqual({ role: 'user', content: 'Line 1\nLine 2\nLine 3' });
60
+ });
61
+
62
+ it('should handle messages with external URL images', async () => {
63
+ serializeImageMock.mockResolvedValue({
64
+ inferenceDetail: 'high',
65
+ externalUrl: 'https://example.com/image.jpg',
66
+ });
67
+
68
+ const ctx = ChatContext.empty();
69
+ ctx.addMessage({
70
+ role: 'user',
71
+ content: [
72
+ 'Check out this image:',
73
+ {
74
+ id: 'img1',
75
+ type: 'image_content',
76
+ image: 'https://example.com/image.jpg',
77
+ inferenceDetail: 'high',
78
+ _cache: {},
79
+ },
80
+ ],
81
+ });
82
+
83
+ const result = await toChatCtx(ctx);
84
+
85
+ expect(result).toEqual([
86
+ {
87
+ role: 'user',
88
+ content: [
89
+ {
90
+ type: 'image_url',
91
+ image_url: {
92
+ url: 'https://example.com/image.jpg',
93
+ detail: 'high',
94
+ },
95
+ },
96
+ { type: 'text', text: 'Check out this image:' },
97
+ ],
98
+ },
99
+ ]);
100
+ });
101
+
102
+ it('should handle messages with base64 images', async () => {
103
+ serializeImageMock.mockResolvedValue({
104
+ inferenceDetail: 'auto',
105
+ mimeType: 'image/png',
106
+ base64Data: 'iVBORw0KGgoAAAANSUhEUgAAAAEAAAAB',
107
+ });
108
+
109
+ const ctx = ChatContext.empty();
110
+ ctx.addMessage({
111
+ role: 'assistant',
112
+ content: [
113
+ {
114
+ id: 'img1',
115
+ type: 'image_content',
116
+ image: 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAAB',
117
+ inferenceDetail: 'auto',
118
+ _cache: {},
119
+ },
120
+ 'Here is the image you requested',
121
+ ],
122
+ });
123
+
124
+ const result = await toChatCtx(ctx);
125
+
126
+ expect(result).toEqual([
127
+ {
128
+ role: 'assistant',
129
+ content: [
130
+ {
131
+ type: 'image_url',
132
+ image_url: {
133
+ url: 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAAB',
134
+ detail: 'auto',
135
+ },
136
+ },
137
+ { type: 'text', text: 'Here is the image you requested' },
138
+ ],
139
+ },
140
+ ]);
141
+ });
142
+
143
+ it('should handle VideoFrame images', async () => {
144
+ serializeImageMock.mockResolvedValue({
145
+ inferenceDetail: 'low',
146
+ mimeType: 'image/jpeg',
147
+ base64Data: '/9j/4AAQSkZJRg==',
148
+ });
149
+
150
+ const frameData = new Uint8Array(4 * 4 * 4); // 4x4 RGBA
151
+ const videoFrame = new VideoFrame(frameData, 4, 4, VideoBufferType.RGBA);
152
+
153
+ const ctx = ChatContext.empty();
154
+ ctx.addMessage({
155
+ role: 'user',
156
+ content: [
157
+ {
158
+ id: 'frame1',
159
+ type: 'image_content',
160
+ image: videoFrame,
161
+ inferenceDetail: 'low',
162
+ _cache: {},
163
+ },
164
+ ],
165
+ });
166
+
167
+ const result = await toChatCtx(ctx);
168
+
169
+ expect(result).toEqual([
170
+ {
171
+ role: 'user',
172
+ content: [
173
+ {
174
+ type: 'image_url',
175
+ image_url: {
176
+ url: 'data:image/jpeg;base64,/9j/4AAQSkZJRg==',
177
+ detail: 'low',
178
+ },
179
+ },
180
+ ],
181
+ },
182
+ ]);
183
+ });
184
+
185
+ it('should cache serialized images', async () => {
186
+ serializeImageMock.mockResolvedValue({
187
+ inferenceDetail: 'high',
188
+ mimeType: 'image/png',
189
+ base64Data: 'cached-data',
190
+ });
191
+
192
+ const imageContent = {
193
+ id: 'img1',
194
+ type: 'image_content' as const,
195
+ image: 'https://example.com/image.jpg',
196
+ inferenceDetail: 'high' as const,
197
+ _cache: {},
198
+ };
199
+
200
+ const ctx = ChatContext.empty();
201
+ ctx.addMessage({ role: 'user', content: [imageContent] });
202
+
203
+ // Call twice to test caching
204
+ await toChatCtx(ctx);
205
+ await toChatCtx(ctx);
206
+
207
+ // serializeImage should only be called once due to caching
208
+ expect(serializeImageMock).toHaveBeenCalledTimes(1);
209
+ expect(imageContent._cache).toHaveProperty('serialized_image');
210
+ });
211
+
212
+ it('should handle tool calls and outputs', async () => {
213
+ const ctx = ChatContext.empty();
214
+
215
+ // Add an assistant message with tool calls
216
+ const msg = ctx.addMessage({ role: 'assistant', content: 'Let me help you with that.' });
217
+ const toolCall = FunctionCall.create({
218
+ id: msg.id + '/tool_1',
219
+ callId: 'call_123',
220
+ name: 'get_weather',
221
+ args: '{"location": "San Francisco"}',
222
+ });
223
+ const toolOutput = FunctionCallOutput.create({
224
+ callId: 'call_123',
225
+ output: '{"temperature": 72, "condition": "sunny"}',
226
+ isError: false,
227
+ });
228
+
229
+ ctx.insert([toolCall, toolOutput]);
230
+
231
+ const result = await toChatCtx(ctx);
232
+
233
+ expect(result).toEqual([
234
+ {
235
+ role: 'assistant',
236
+ content: 'Let me help you with that.',
237
+ tool_calls: [
238
+ {
239
+ type: 'function',
240
+ id: 'call_123',
241
+ function: {
242
+ name: 'get_weather',
243
+ arguments: '{"location": "San Francisco"}',
244
+ },
245
+ },
246
+ ],
247
+ },
248
+ {
249
+ role: 'tool',
250
+ tool_call_id: 'call_123',
251
+ content: '{"temperature": 72, "condition": "sunny"}',
252
+ },
253
+ ]);
254
+ });
255
+
256
+ it('should handle multiple tool calls in one message', async () => {
257
+ const ctx = ChatContext.empty();
258
+
259
+ const msg = ctx.addMessage({ role: 'assistant', content: "I'll check both locations." });
260
+ const toolCall1 = new FunctionCall({
261
+ id: msg.id + '/tool_1',
262
+ callId: 'call_1',
263
+ name: 'get_weather',
264
+ args: '{"location": "NYC"}',
265
+ });
266
+ const toolCall2 = new FunctionCall({
267
+ id: msg.id + '/tool_2',
268
+ callId: 'call_2',
269
+ name: 'get_weather',
270
+ args: '{"location": "LA"}',
271
+ });
272
+ const toolOutput1 = new FunctionCallOutput({
273
+ callId: 'call_1',
274
+ output: '{"temperature": 65}',
275
+ isError: false,
276
+ });
277
+ const toolOutput2 = new FunctionCallOutput({
278
+ callId: 'call_2',
279
+ output: '{"temperature": 78}',
280
+ isError: false,
281
+ });
282
+
283
+ ctx.insert([toolCall1, toolCall2, toolOutput1, toolOutput2]);
284
+
285
+ const result = await toChatCtx(ctx);
286
+
287
+ expect(result).toEqual([
288
+ {
289
+ role: 'assistant',
290
+ content: "I'll check both locations.",
291
+ tool_calls: [
292
+ {
293
+ type: 'function',
294
+ id: 'call_1',
295
+ function: { name: 'get_weather', arguments: '{"location": "NYC"}' },
296
+ },
297
+ {
298
+ type: 'function',
299
+ id: 'call_2',
300
+ function: { name: 'get_weather', arguments: '{"location": "LA"}' },
301
+ },
302
+ ],
303
+ },
304
+ {
305
+ role: 'tool',
306
+ tool_call_id: 'call_1',
307
+ content: '{"temperature": 65}',
308
+ },
309
+ {
310
+ role: 'tool',
311
+ tool_call_id: 'call_2',
312
+ content: '{"temperature": 78}',
313
+ },
314
+ ]);
315
+ });
316
+
317
+ it('should handle tool calls without accompanying message', async () => {
318
+ const ctx = ChatContext.empty();
319
+
320
+ const toolCall = new FunctionCall({
321
+ id: 'func_123',
322
+ callId: 'call_456',
323
+ name: 'calculate',
324
+ args: '{"a": 5, "b": 3}',
325
+ });
326
+ const toolOutput = new FunctionCallOutput({
327
+ callId: 'call_456',
328
+ output: '{"result": 8}',
329
+ isError: false,
330
+ });
331
+
332
+ ctx.insert([toolCall, toolOutput]);
333
+
334
+ const result = await toChatCtx(ctx);
335
+
336
+ expect(result).toEqual([
337
+ {
338
+ role: 'assistant',
339
+ tool_calls: [
340
+ {
341
+ type: 'function',
342
+ id: 'call_456',
343
+ function: { name: 'calculate', arguments: '{"a": 5, "b": 3}' },
344
+ },
345
+ ],
346
+ },
347
+ {
348
+ role: 'tool',
349
+ tool_call_id: 'call_456',
350
+ content: '{"result": 8}',
351
+ },
352
+ ]);
353
+ });
354
+
355
+ it('should skip empty groups', async () => {
356
+ const ctx = ChatContext.empty();
357
+ ctx.addMessage({ role: 'user', content: 'Hello', createdAt: 1000 });
358
+
359
+ // Create an isolated tool output without corresponding call (will be filtered)
360
+ const orphanOutput = new FunctionCallOutput({
361
+ callId: 'orphan_call',
362
+ output: 'This should be ignored',
363
+ isError: false,
364
+ createdAt: 2000,
365
+ });
366
+ ctx.insert(orphanOutput);
367
+
368
+ ctx.addMessage({ role: 'assistant', content: 'Hi!', createdAt: 3000 });
369
+
370
+ const result = await toChatCtx(ctx);
371
+
372
+ // Messages should be in the result, orphan output should be filtered out
373
+ expect(result).toHaveLength(2);
374
+ expect(result).toContainEqual({ role: 'user', content: 'Hello' });
375
+ expect(result).toContainEqual({ role: 'assistant', content: 'Hi!' });
376
+ });
377
+
378
+ it('should handle mixed content with text and multiple images', async () => {
379
+ serializeImageMock
380
+ .mockResolvedValueOnce({
381
+ inferenceDetail: 'high',
382
+ externalUrl: 'https://example.com/image1.jpg',
383
+ })
384
+ .mockResolvedValueOnce({
385
+ inferenceDetail: 'low',
386
+ mimeType: 'image/png',
387
+ base64Data: 'base64data',
388
+ });
389
+
390
+ const ctx = ChatContext.empty();
391
+ ctx.addMessage({
392
+ role: 'user',
393
+ content: [
394
+ 'Here are two images:',
395
+ {
396
+ id: 'img1',
397
+ type: 'image_content',
398
+ image: 'https://example.com/image1.jpg',
399
+ inferenceDetail: 'high',
400
+ _cache: {},
401
+ },
402
+ 'And the second one:',
403
+ {
404
+ id: 'img2',
405
+ type: 'image_content',
406
+ image: 'data:image/png;base64,base64data',
407
+ inferenceDetail: 'low',
408
+ _cache: {},
409
+ },
410
+ 'What do you think?',
411
+ ],
412
+ });
413
+
414
+ const result = await toChatCtx(ctx);
415
+
416
+ expect(result).toEqual([
417
+ {
418
+ role: 'user',
419
+ content: [
420
+ {
421
+ type: 'image_url',
422
+ image_url: {
423
+ url: 'https://example.com/image1.jpg',
424
+ detail: 'high',
425
+ },
426
+ },
427
+ {
428
+ type: 'image_url',
429
+ image_url: {
430
+ url: 'data:image/png;base64,base64data',
431
+ detail: 'low',
432
+ },
433
+ },
434
+ {
435
+ type: 'text',
436
+ text: 'Here are two images:\nAnd the second one:\nWhat do you think?',
437
+ },
438
+ ],
439
+ },
440
+ ]);
441
+ });
442
+
443
+ it('should handle content with only images and no text', async () => {
444
+ serializeImageMock.mockResolvedValue({
445
+ inferenceDetail: 'auto',
446
+ externalUrl: 'https://example.com/image.jpg',
447
+ });
448
+
449
+ const ctx = ChatContext.empty();
450
+ ctx.addMessage({
451
+ role: 'user',
452
+ content: [
453
+ {
454
+ id: 'img1',
455
+ type: 'image_content',
456
+ image: 'https://example.com/image.jpg',
457
+ inferenceDetail: 'auto',
458
+ _cache: {},
459
+ },
460
+ ],
461
+ });
462
+
463
+ const result = await toChatCtx(ctx);
464
+
465
+ expect(result).toEqual([
466
+ {
467
+ role: 'user',
468
+ content: [
469
+ {
470
+ type: 'image_url',
471
+ image_url: {
472
+ url: 'https://example.com/image.jpg',
473
+ detail: 'auto',
474
+ },
475
+ },
476
+ ],
477
+ },
478
+ ]);
479
+ });
480
+
481
+ it('should throw error for unsupported content type', async () => {
482
+ const ctx = ChatContext.empty();
483
+ ctx.addMessage({
484
+ role: 'user',
485
+ content: [
486
+ {
487
+ type: 'audio_content',
488
+ frame: [],
489
+ },
490
+ ],
491
+ });
492
+
493
+ await expect(toChatCtx(ctx)).rejects.toThrow('Unsupported content type: audio_content');
494
+ });
495
+
496
+ it('should throw error when serialized image has no data', async () => {
497
+ serializeImageMock.mockResolvedValue({
498
+ inferenceDetail: 'high',
499
+ // No base64Data or externalUrl
500
+ });
501
+
502
+ const ctx = ChatContext.empty();
503
+ ctx.addMessage({
504
+ role: 'user',
505
+ content: [
506
+ {
507
+ id: 'img1',
508
+ type: 'image_content',
509
+ image: 'invalid-image',
510
+ inferenceDetail: 'high',
511
+ _cache: {},
512
+ },
513
+ ],
514
+ });
515
+
516
+ await expect(toChatCtx(ctx)).rejects.toThrow('Serialized image has no data bytes');
517
+ });
518
+
519
+ it('should filter out standalone function calls without outputs', async () => {
520
+ const ctx = ChatContext.empty();
521
+
522
+ // Add standalone function call without output
523
+ const funcCall = new FunctionCall({
524
+ id: 'func_standalone',
525
+ callId: 'call_999',
526
+ name: 'standalone_function',
527
+ args: '{}',
528
+ });
529
+
530
+ ctx.insert(funcCall);
531
+
532
+ const result = await toChatCtx(ctx);
533
+
534
+ // Standalone function calls without outputs are filtered out by groupToolCalls
535
+ expect(result).toEqual([]);
536
+ });
537
+
538
+ it('should handle function call output correctly', async () => {
539
+ const ctx = ChatContext.empty();
540
+
541
+ // First add a function call
542
+ const funcCall = new FunctionCall({
543
+ id: 'func_1',
544
+ callId: 'call_output_test',
545
+ name: 'test_function',
546
+ args: '{}',
547
+ });
548
+
549
+ // Then add its output
550
+ const funcOutput = new FunctionCallOutput({
551
+ callId: 'call_output_test',
552
+ output: 'Function executed successfully',
553
+ isError: false,
554
+ });
555
+
556
+ ctx.insert([funcCall, funcOutput]);
557
+
558
+ const result = await toChatCtx(ctx);
559
+
560
+ expect(result).toEqual([
561
+ {
562
+ role: 'assistant',
563
+ tool_calls: [
564
+ {
565
+ id: 'call_output_test',
566
+ type: 'function',
567
+ function: {
568
+ name: 'test_function',
569
+ arguments: '{}',
570
+ },
571
+ },
572
+ ],
573
+ },
574
+ {
575
+ role: 'tool',
576
+ tool_call_id: 'call_output_test',
577
+ content: 'Function executed successfully',
578
+ },
579
+ ]);
580
+ });
581
+ });
@@ -0,0 +1,118 @@
1
+ // SPDX-FileCopyrightText: 2025 LiveKit, Inc.
2
+ //
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ import type { ChatContext, ChatItem, ImageContent } from '../chat_context.js';
5
+ import { type SerializedImage, serializeImage } from '../utils.js';
6
+ import { groupToolCalls } from './utils.js';
7
+
8
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
9
+ export async function toChatCtx(chatCtx: ChatContext, injectDummyUserMessage: boolean = true) {
10
+ const itemGroups = groupToolCalls(chatCtx);
11
+ const messages: Record<string, any>[] = []; // eslint-disable-line @typescript-eslint/no-explicit-any
12
+
13
+ for (const group of itemGroups) {
14
+ if (group.isEmpty) continue;
15
+
16
+ const message: Record<string, any> = group.message // eslint-disable-line @typescript-eslint/no-explicit-any
17
+ ? await toChatItem(group.message)
18
+ : { role: 'assistant' };
19
+
20
+ const toolCalls = group.toolCalls.map((toolCall) => ({
21
+ type: 'function',
22
+ id: toolCall.callId,
23
+ function: { name: toolCall.name, arguments: toolCall.args },
24
+ }));
25
+
26
+ if (toolCalls.length > 0) {
27
+ message['tool_calls'] = toolCalls;
28
+ }
29
+
30
+ messages.push(message);
31
+
32
+ for (const toolOutput of group.toolOutputs) {
33
+ messages.push(await toChatItem(toolOutput));
34
+ }
35
+ }
36
+
37
+ return messages;
38
+ }
39
+
40
+ async function toChatItem(item: ChatItem) {
41
+ if (item.type === 'message') {
42
+ const listContent: Record<string, any>[] = []; // eslint-disable-line @typescript-eslint/no-explicit-any
43
+ let textContent = '';
44
+
45
+ for (const content of item.content) {
46
+ if (typeof content === 'string') {
47
+ if (textContent) textContent += '\n';
48
+ textContent += content;
49
+ } else if (content.type === 'image_content') {
50
+ listContent.push(await toImageContent(content));
51
+ } else {
52
+ throw new Error(`Unsupported content type: ${content.type}`);
53
+ }
54
+ }
55
+
56
+ const content =
57
+ listContent.length == 0
58
+ ? textContent
59
+ : textContent.length == 0
60
+ ? listContent
61
+ : [...listContent, { type: 'text', text: textContent }];
62
+
63
+ return { role: item.role, content };
64
+ } else if (item.type === 'function_call') {
65
+ return {
66
+ role: 'assistant',
67
+ tool_calls: [
68
+ {
69
+ id: item.callId,
70
+ type: 'function',
71
+ function: { name: item.name, arguments: item.args },
72
+ },
73
+ ],
74
+ };
75
+ } else if (item.type === 'function_call_output') {
76
+ return {
77
+ role: 'tool',
78
+ tool_call_id: item.callId,
79
+ content: item.output,
80
+ };
81
+ } else {
82
+ throw new Error(`Unsupported item type: ${item['type']}`);
83
+ }
84
+ }
85
+
86
+ async function toImageContent(content: ImageContent) {
87
+ const cacheKey = 'serialized_image'; // TODO: use hash of encoding options if available
88
+ let serialized: SerializedImage;
89
+
90
+ if (content._cache[cacheKey] === undefined) {
91
+ serialized = await serializeImage(content);
92
+ content._cache[cacheKey] = serialized;
93
+ }
94
+ serialized = content._cache[cacheKey];
95
+
96
+ // Convert SerializedImage to OpenAI format
97
+ if (serialized.externalUrl) {
98
+ return {
99
+ type: 'image_url',
100
+ image_url: {
101
+ url: serialized.externalUrl,
102
+ detail: serialized.inferenceDetail,
103
+ },
104
+ };
105
+ }
106
+
107
+ if (serialized.base64Data === undefined) {
108
+ throw new Error('Serialized image has no data bytes');
109
+ }
110
+
111
+ return {
112
+ type: 'image_url',
113
+ image_url: {
114
+ url: `data:${serialized.mimeType};base64,${serialized.base64Data}`,
115
+ detail: serialized.inferenceDetail,
116
+ },
117
+ };
118
+ }