@dxos/plugin-assistant 0.7.5-main.5ae2ba8

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 (305) hide show
  1. package/LICENSE +8 -0
  2. package/README.md +15 -0
  3. package/dist/lib/browser/AssistantDialog-TX6YYBUG.mjs +116 -0
  4. package/dist/lib/browser/AssistantDialog-TX6YYBUG.mjs.map +7 -0
  5. package/dist/lib/browser/ChatContainer-AT3OAUT3.mjs +33 -0
  6. package/dist/lib/browser/ChatContainer-AT3OAUT3.mjs.map +7 -0
  7. package/dist/lib/browser/TemplateContainer-B7MQNUPY.mjs +23 -0
  8. package/dist/lib/browser/TemplateContainer-B7MQNUPY.mjs.map +7 -0
  9. package/dist/lib/browser/ai-client-RTCGRKZE.mjs +22 -0
  10. package/dist/lib/browser/ai-client-RTCGRKZE.mjs.map +7 -0
  11. package/dist/lib/browser/app-graph-builder-AXAIFOGV.mjs +110 -0
  12. package/dist/lib/browser/app-graph-builder-AXAIFOGV.mjs.map +7 -0
  13. package/dist/lib/browser/chunk-EUMPBC4T.mjs +81 -0
  14. package/dist/lib/browser/chunk-EUMPBC4T.mjs.map +7 -0
  15. package/dist/lib/browser/chunk-FRIKXDDQ.mjs +162 -0
  16. package/dist/lib/browser/chunk-FRIKXDDQ.mjs.map +7 -0
  17. package/dist/lib/browser/chunk-G7B54APW.mjs +106 -0
  18. package/dist/lib/browser/chunk-G7B54APW.mjs.map +7 -0
  19. package/dist/lib/browser/chunk-NFVIZS3B.mjs +1804 -0
  20. package/dist/lib/browser/chunk-NFVIZS3B.mjs.map +7 -0
  21. package/dist/lib/browser/chunk-NV7SVHMV.mjs +20 -0
  22. package/dist/lib/browser/chunk-NV7SVHMV.mjs.map +7 -0
  23. package/dist/lib/browser/chunk-VZ4W6SHE.mjs +15 -0
  24. package/dist/lib/browser/chunk-VZ4W6SHE.mjs.map +7 -0
  25. package/dist/lib/browser/index.mjs +218 -0
  26. package/dist/lib/browser/index.mjs.map +7 -0
  27. package/dist/lib/browser/intent-resolver-QRVRZL6K.mjs +44 -0
  28. package/dist/lib/browser/intent-resolver-QRVRZL6K.mjs.map +7 -0
  29. package/dist/lib/browser/meta.json +1 -0
  30. package/dist/lib/browser/react-surface-JLXNWOI6.mjs +69 -0
  31. package/dist/lib/browser/react-surface-JLXNWOI6.mjs.map +7 -0
  32. package/dist/lib/browser/settings-JTT62IHD.mjs +22 -0
  33. package/dist/lib/browser/settings-JTT62IHD.mjs.map +7 -0
  34. package/dist/lib/browser/types/index.mjs +24 -0
  35. package/dist/lib/browser/types/index.mjs.map +7 -0
  36. package/dist/lib/node/AssistantDialog-U4GBPZD6.cjs +140 -0
  37. package/dist/lib/node/AssistantDialog-U4GBPZD6.cjs.map +7 -0
  38. package/dist/lib/node/ChatContainer-CVHXNHGA.cjs +61 -0
  39. package/dist/lib/node/ChatContainer-CVHXNHGA.cjs.map +7 -0
  40. package/dist/lib/node/TemplateContainer-R4BZZP3E.cjs +53 -0
  41. package/dist/lib/node/TemplateContainer-R4BZZP3E.cjs.map +7 -0
  42. package/dist/lib/node/ai-client-YANJEPO3.cjs +38 -0
  43. package/dist/lib/node/ai-client-YANJEPO3.cjs.map +7 -0
  44. package/dist/lib/node/app-graph-builder-D7SHQTZS.cjs +122 -0
  45. package/dist/lib/node/app-graph-builder-D7SHQTZS.cjs.map +7 -0
  46. package/dist/lib/node/chunk-37GI4NYH.cjs +183 -0
  47. package/dist/lib/node/chunk-37GI4NYH.cjs.map +7 -0
  48. package/dist/lib/node/chunk-GNPXCHFT.cjs +44 -0
  49. package/dist/lib/node/chunk-GNPXCHFT.cjs.map +7 -0
  50. package/dist/lib/node/chunk-IXJCGW7U.cjs +130 -0
  51. package/dist/lib/node/chunk-IXJCGW7U.cjs.map +7 -0
  52. package/dist/lib/node/chunk-NV4TQQSU.cjs +111 -0
  53. package/dist/lib/node/chunk-NV4TQQSU.cjs.map +7 -0
  54. package/dist/lib/node/chunk-XUTDR7HI.cjs +1781 -0
  55. package/dist/lib/node/chunk-XUTDR7HI.cjs.map +7 -0
  56. package/dist/lib/node/chunk-ZGH6F5YA.cjs +34 -0
  57. package/dist/lib/node/chunk-ZGH6F5YA.cjs.map +7 -0
  58. package/dist/lib/node/index.cjs +226 -0
  59. package/dist/lib/node/index.cjs.map +7 -0
  60. package/dist/lib/node/intent-resolver-YMMAFVOB.cjs +58 -0
  61. package/dist/lib/node/intent-resolver-YMMAFVOB.cjs.map +7 -0
  62. package/dist/lib/node/meta.json +1 -0
  63. package/dist/lib/node/react-surface-BSUZQ3HZ.cjs +85 -0
  64. package/dist/lib/node/react-surface-BSUZQ3HZ.cjs.map +7 -0
  65. package/dist/lib/node/settings-4YEO7KXF.cjs +36 -0
  66. package/dist/lib/node/settings-4YEO7KXF.cjs.map +7 -0
  67. package/dist/lib/node/types/index.cjs +46 -0
  68. package/dist/lib/node/types/index.cjs.map +7 -0
  69. package/dist/lib/node-esm/AssistantDialog-5AT5JAZL.mjs +117 -0
  70. package/dist/lib/node-esm/AssistantDialog-5AT5JAZL.mjs.map +7 -0
  71. package/dist/lib/node-esm/ChatContainer-VR766C4M.mjs +34 -0
  72. package/dist/lib/node-esm/ChatContainer-VR766C4M.mjs.map +7 -0
  73. package/dist/lib/node-esm/TemplateContainer-WSHTZBB5.mjs +24 -0
  74. package/dist/lib/node-esm/TemplateContainer-WSHTZBB5.mjs.map +7 -0
  75. package/dist/lib/node-esm/ai-client-66IBZVCX.mjs +23 -0
  76. package/dist/lib/node-esm/ai-client-66IBZVCX.mjs.map +7 -0
  77. package/dist/lib/node-esm/app-graph-builder-H2GC2AZA.mjs +111 -0
  78. package/dist/lib/node-esm/app-graph-builder-H2GC2AZA.mjs.map +7 -0
  79. package/dist/lib/node-esm/chunk-77ARTFBA.mjs +16 -0
  80. package/dist/lib/node-esm/chunk-77ARTFBA.mjs.map +7 -0
  81. package/dist/lib/node-esm/chunk-7JENJTLB.mjs +1805 -0
  82. package/dist/lib/node-esm/chunk-7JENJTLB.mjs.map +7 -0
  83. package/dist/lib/node-esm/chunk-7SV6X6XU.mjs +22 -0
  84. package/dist/lib/node-esm/chunk-7SV6X6XU.mjs.map +7 -0
  85. package/dist/lib/node-esm/chunk-AMQMVQJO.mjs +107 -0
  86. package/dist/lib/node-esm/chunk-AMQMVQJO.mjs.map +7 -0
  87. package/dist/lib/node-esm/chunk-CJ4Y3QW5.mjs +163 -0
  88. package/dist/lib/node-esm/chunk-CJ4Y3QW5.mjs.map +7 -0
  89. package/dist/lib/node-esm/chunk-LBQGJE5T.mjs +82 -0
  90. package/dist/lib/node-esm/chunk-LBQGJE5T.mjs.map +7 -0
  91. package/dist/lib/node-esm/index.mjs +219 -0
  92. package/dist/lib/node-esm/index.mjs.map +7 -0
  93. package/dist/lib/node-esm/intent-resolver-MR7BOKEW.mjs +45 -0
  94. package/dist/lib/node-esm/intent-resolver-MR7BOKEW.mjs.map +7 -0
  95. package/dist/lib/node-esm/meta.json +1 -0
  96. package/dist/lib/node-esm/react-surface-IGVYAOGL.mjs +70 -0
  97. package/dist/lib/node-esm/react-surface-IGVYAOGL.mjs.map +7 -0
  98. package/dist/lib/node-esm/settings-S7P5RWQI.mjs +23 -0
  99. package/dist/lib/node-esm/settings-S7P5RWQI.mjs.map +7 -0
  100. package/dist/lib/node-esm/types/index.mjs +25 -0
  101. package/dist/lib/node-esm/types/index.mjs.map +7 -0
  102. package/dist/types/src/AssistantPlugin.d.ts +2 -0
  103. package/dist/types/src/AssistantPlugin.d.ts.map +1 -0
  104. package/dist/types/src/capabilities/ai-client.d.ts +5 -0
  105. package/dist/types/src/capabilities/ai-client.d.ts.map +1 -0
  106. package/dist/types/src/capabilities/app-graph-builder.d.ts +181 -0
  107. package/dist/types/src/capabilities/app-graph-builder.d.ts.map +1 -0
  108. package/dist/types/src/capabilities/capabilities.d.ts +5 -0
  109. package/dist/types/src/capabilities/capabilities.d.ts.map +1 -0
  110. package/dist/types/src/capabilities/index.d.ts +183 -0
  111. package/dist/types/src/capabilities/index.d.ts.map +1 -0
  112. package/dist/types/src/capabilities/intent-resolver.d.ts +4 -0
  113. package/dist/types/src/capabilities/intent-resolver.d.ts.map +1 -0
  114. package/dist/types/src/capabilities/react-surface.d.ts +4 -0
  115. package/dist/types/src/capabilities/react-surface.d.ts.map +1 -0
  116. package/dist/types/src/capabilities/settings.d.ts +4 -0
  117. package/dist/types/src/capabilities/settings.d.ts.map +1 -0
  118. package/dist/types/src/components/AmbientDialog/AmbientDialog.d.ts +7 -0
  119. package/dist/types/src/components/AmbientDialog/AmbientDialog.d.ts.map +1 -0
  120. package/dist/types/src/components/AmbientDialog/AmbientDialog.stories.d.ts +8 -0
  121. package/dist/types/src/components/AmbientDialog/AmbientDialog.stories.d.ts.map +1 -0
  122. package/dist/types/src/components/AmbientDialog/index.d.ts +2 -0
  123. package/dist/types/src/components/AmbientDialog/index.d.ts.map +1 -0
  124. package/dist/types/src/components/AssistantDialog.d.ts +7 -0
  125. package/dist/types/src/components/AssistantDialog.d.ts.map +1 -0
  126. package/dist/types/src/components/AssistantSettings/AssistantSettings.d.ts +5 -0
  127. package/dist/types/src/components/AssistantSettings/AssistantSettings.d.ts.map +1 -0
  128. package/dist/types/src/components/AssistantSettings/index.d.ts +2 -0
  129. package/dist/types/src/components/AssistantSettings/index.d.ts.map +1 -0
  130. package/dist/types/src/components/ChatContainer.d.ts +7 -0
  131. package/dist/types/src/components/ChatContainer.d.ts.map +1 -0
  132. package/dist/types/src/components/Prompt/Prompt.d.ts +17 -0
  133. package/dist/types/src/components/Prompt/Prompt.d.ts.map +1 -0
  134. package/dist/types/src/components/Prompt/Prompt.stories.d.ts +9 -0
  135. package/dist/types/src/components/Prompt/Prompt.stories.d.ts.map +1 -0
  136. package/dist/types/src/components/Prompt/PromptBar.d.ts +10 -0
  137. package/dist/types/src/components/Prompt/PromptBar.d.ts.map +1 -0
  138. package/dist/types/src/components/Prompt/autocomplete.d.ts +21 -0
  139. package/dist/types/src/components/Prompt/autocomplete.d.ts.map +1 -0
  140. package/dist/types/src/components/Prompt/index.d.ts +3 -0
  141. package/dist/types/src/components/Prompt/index.d.ts.map +1 -0
  142. package/dist/types/src/components/ServiceRegistry/ServiceRegistry.d.ts +5 -0
  143. package/dist/types/src/components/ServiceRegistry/ServiceRegistry.d.ts.map +1 -0
  144. package/dist/types/src/components/ServiceRegistry/ServiceRegistry.stories.d.ts +8 -0
  145. package/dist/types/src/components/ServiceRegistry/ServiceRegistry.stories.d.ts.map +1 -0
  146. package/dist/types/src/components/ServiceRegistry/index.d.ts +2 -0
  147. package/dist/types/src/components/ServiceRegistry/index.d.ts.map +1 -0
  148. package/dist/types/src/components/TemplateContainer.d.ts +7 -0
  149. package/dist/types/src/components/TemplateContainer.d.ts.map +1 -0
  150. package/dist/types/src/components/TemplateEditor/TemplateEditor.d.ts +12 -0
  151. package/dist/types/src/components/TemplateEditor/TemplateEditor.d.ts.map +1 -0
  152. package/dist/types/src/components/TemplateEditor/TemplateEditor.stories.d.ts +12 -0
  153. package/dist/types/src/components/TemplateEditor/TemplateEditor.stories.d.ts.map +1 -0
  154. package/dist/types/src/components/TemplateEditor/TemplateForm.d.ts +10 -0
  155. package/dist/types/src/components/TemplateEditor/TemplateForm.d.ts.map +1 -0
  156. package/dist/types/src/components/TemplateEditor/TemplateForm.stories.d.ts +8 -0
  157. package/dist/types/src/components/TemplateEditor/TemplateForm.stories.d.ts.map +1 -0
  158. package/dist/types/src/components/TemplateEditor/index.d.ts +2 -0
  159. package/dist/types/src/components/TemplateEditor/index.d.ts.map +1 -0
  160. package/dist/types/src/components/TemplateEditor/types.d.ts +18 -0
  161. package/dist/types/src/components/TemplateEditor/types.d.ts.map +1 -0
  162. package/dist/types/src/components/Thread/Thread.d.ts +17 -0
  163. package/dist/types/src/components/Thread/Thread.d.ts.map +1 -0
  164. package/dist/types/src/components/Thread/Thread.stories.d.ts +11 -0
  165. package/dist/types/src/components/Thread/Thread.stories.d.ts.map +1 -0
  166. package/dist/types/src/components/Thread/ThreadContainer.d.ts +10 -0
  167. package/dist/types/src/components/Thread/ThreadContainer.d.ts.map +1 -0
  168. package/dist/types/src/components/Thread/ThreadContainer.stories.d.ts +15 -0
  169. package/dist/types/src/components/Thread/ThreadContainer.stories.d.ts.map +1 -0
  170. package/dist/types/src/components/Thread/ThreadMessage.d.ts +14 -0
  171. package/dist/types/src/components/Thread/ThreadMessage.d.ts.map +1 -0
  172. package/dist/types/src/components/Thread/ToolInvocations.d.ts +13 -0
  173. package/dist/types/src/components/Thread/ToolInvocations.d.ts.map +1 -0
  174. package/dist/types/src/components/Thread/index.d.ts +3 -0
  175. package/dist/types/src/components/Thread/index.d.ts.map +1 -0
  176. package/dist/types/src/components/Thread/reducer.d.ts +12 -0
  177. package/dist/types/src/components/Thread/reducer.d.ts.map +1 -0
  178. package/dist/types/src/components/Toolbox/Toolbox.d.ts +19 -0
  179. package/dist/types/src/components/Toolbox/Toolbox.d.ts.map +1 -0
  180. package/dist/types/src/components/Toolbox/Toolbox.stories.d.ts +8 -0
  181. package/dist/types/src/components/Toolbox/Toolbox.stories.d.ts.map +1 -0
  182. package/dist/types/src/components/Toolbox/index.d.ts +2 -0
  183. package/dist/types/src/components/Toolbox/index.d.ts.map +1 -0
  184. package/dist/types/src/components/index.d.ts +17 -0
  185. package/dist/types/src/components/index.d.ts.map +1 -0
  186. package/dist/types/src/hooks/email.d.ts +4 -0
  187. package/dist/types/src/hooks/email.d.ts.map +1 -0
  188. package/dist/types/src/hooks/index.d.ts +7 -0
  189. package/dist/types/src/hooks/index.d.ts.map +1 -0
  190. package/dist/types/src/hooks/invocation-handler.d.ts +5 -0
  191. package/dist/types/src/hooks/invocation-handler.d.ts.map +1 -0
  192. package/dist/types/src/hooks/processor.d.ts +76 -0
  193. package/dist/types/src/hooks/processor.d.ts.map +1 -0
  194. package/dist/types/src/hooks/processor.test.d.ts +2 -0
  195. package/dist/types/src/hooks/processor.test.d.ts.map +1 -0
  196. package/dist/types/src/hooks/useChatProcessor.d.ts +8 -0
  197. package/dist/types/src/hooks/useChatProcessor.d.ts.map +1 -0
  198. package/dist/types/src/hooks/useLocalTriggerManager.d.ts +3 -0
  199. package/dist/types/src/hooks/useLocalTriggerManager.d.ts.map +1 -0
  200. package/dist/types/src/hooks/useMessageQueue.d.ts +41 -0
  201. package/dist/types/src/hooks/useMessageQueue.d.ts.map +1 -0
  202. package/dist/types/src/hooks/useServices.d.ts +7 -0
  203. package/dist/types/src/hooks/useServices.d.ts.map +1 -0
  204. package/dist/types/src/hooks/useTextInputEvents.d.ts +13 -0
  205. package/dist/types/src/hooks/useTextInputEvents.d.ts.map +1 -0
  206. package/dist/types/src/index.d.ts +5 -0
  207. package/dist/types/src/index.d.ts.map +1 -0
  208. package/dist/types/src/meta.d.ts +11 -0
  209. package/dist/types/src/meta.d.ts.map +1 -0
  210. package/dist/types/src/testing/index.d.ts +3 -0
  211. package/dist/types/src/testing/index.d.ts.map +1 -0
  212. package/dist/types/src/testing/test-functions.d.ts +3 -0
  213. package/dist/types/src/testing/test-functions.d.ts.map +1 -0
  214. package/dist/types/src/testing/test-services.d.ts +5 -0
  215. package/dist/types/src/testing/test-services.d.ts.map +1 -0
  216. package/dist/types/src/tools/function.d.ts +5 -0
  217. package/dist/types/src/tools/function.d.ts.map +1 -0
  218. package/dist/types/src/tools/index.d.ts +3 -0
  219. package/dist/types/src/tools/index.d.ts.map +1 -0
  220. package/dist/types/src/tools/openapi.d.ts +10 -0
  221. package/dist/types/src/tools/openapi.d.ts.map +1 -0
  222. package/dist/types/src/tools/openapi.test.d.ts +2 -0
  223. package/dist/types/src/tools/openapi.test.d.ts.map +1 -0
  224. package/dist/types/src/translations.d.ts +82 -0
  225. package/dist/types/src/translations.d.ts.map +1 -0
  226. package/dist/types/src/types/chat.d.ts +12 -0
  227. package/dist/types/src/types/chat.d.ts.map +1 -0
  228. package/dist/types/src/types/index.d.ts +5 -0
  229. package/dist/types/src/types/index.d.ts.map +1 -0
  230. package/dist/types/src/types/service.d.ts +160 -0
  231. package/dist/types/src/types/service.d.ts.map +1 -0
  232. package/dist/types/src/types/template.d.ts +40 -0
  233. package/dist/types/src/types/template.d.ts.map +1 -0
  234. package/dist/types/src/types/types.d.ts +37 -0
  235. package/dist/types/src/types/types.d.ts.map +1 -0
  236. package/dist/types/tsconfig.tsbuildinfo +1 -0
  237. package/package.json +120 -0
  238. package/src/AssistantPlugin.tsx +105 -0
  239. package/src/capabilities/ai-client.ts +19 -0
  240. package/src/capabilities/app-graph-builder.ts +100 -0
  241. package/src/capabilities/capabilities.ts +12 -0
  242. package/src/capabilities/index.ts +13 -0
  243. package/src/capabilities/intent-resolver.ts +38 -0
  244. package/src/capabilities/react-surface.tsx +49 -0
  245. package/src/capabilities/settings.ts +19 -0
  246. package/src/components/AmbientDialog/AmbientDialog.stories.tsx +81 -0
  247. package/src/components/AmbientDialog/AmbientDialog.tsx +103 -0
  248. package/src/components/AmbientDialog/index.ts +5 -0
  249. package/src/components/AssistantDialog.tsx +31 -0
  250. package/src/components/AssistantSettings/AssistantSettings.tsx +49 -0
  251. package/src/components/AssistantSettings/index.ts +5 -0
  252. package/src/components/ChatContainer.tsx +27 -0
  253. package/src/components/Prompt/Prompt.stories.tsx +79 -0
  254. package/src/components/Prompt/Prompt.tsx +100 -0
  255. package/src/components/Prompt/PromptBar.tsx +100 -0
  256. package/src/components/Prompt/autocomplete.ts +212 -0
  257. package/src/components/Prompt/index.ts +6 -0
  258. package/src/components/ServiceRegistry/ServiceRegistry.stories.tsx +49 -0
  259. package/src/components/ServiceRegistry/ServiceRegistry.tsx +76 -0
  260. package/src/components/ServiceRegistry/index.ts +5 -0
  261. package/src/components/TemplateContainer.tsx +21 -0
  262. package/src/components/TemplateEditor/TemplateEditor.stories.tsx +83 -0
  263. package/src/components/TemplateEditor/TemplateEditor.tsx +76 -0
  264. package/src/components/TemplateEditor/TemplateForm.stories.tsx +65 -0
  265. package/src/components/TemplateEditor/TemplateForm.tsx +184 -0
  266. package/src/components/TemplateEditor/index.ts +5 -0
  267. package/src/components/TemplateEditor/types.tsx +28 -0
  268. package/src/components/Thread/Thread.stories.tsx +212 -0
  269. package/src/components/Thread/Thread.tsx +97 -0
  270. package/src/components/Thread/ThreadContainer.stories.tsx +268 -0
  271. package/src/components/Thread/ThreadContainer.tsx +79 -0
  272. package/src/components/Thread/ThreadMessage.tsx +188 -0
  273. package/src/components/Thread/ToolInvocations.tsx +104 -0
  274. package/src/components/Thread/index.ts +6 -0
  275. package/src/components/Thread/reducer.ts +52 -0
  276. package/src/components/Toolbox/Toolbox.stories.tsx +69 -0
  277. package/src/components/Toolbox/Toolbox.tsx +115 -0
  278. package/src/components/Toolbox/index.ts +5 -0
  279. package/src/components/index.ts +15 -0
  280. package/src/hooks/email.ts +49 -0
  281. package/src/hooks/index.ts +11 -0
  282. package/src/hooks/invocation-handler.ts +109 -0
  283. package/src/hooks/processor.test.ts +15 -0
  284. package/src/hooks/processor.ts +228 -0
  285. package/src/hooks/useChatProcessor.tsx +80 -0
  286. package/src/hooks/useLocalTriggerManager.ts +82 -0
  287. package/src/hooks/useMessageQueue.ts +23 -0
  288. package/src/hooks/useServices.ts +28 -0
  289. package/src/hooks/useTextInputEvents.ts +49 -0
  290. package/src/index.ts +9 -0
  291. package/src/meta.ts +18 -0
  292. package/src/testing/index.ts +6 -0
  293. package/src/testing/test-functions.ts +11 -0
  294. package/src/testing/test-services.ts +131 -0
  295. package/src/tools/function.ts +47 -0
  296. package/src/tools/index.ts +6 -0
  297. package/src/tools/openapi.test.ts +224 -0
  298. package/src/tools/openapi.ts +331 -0
  299. package/src/translations.ts +49 -0
  300. package/src/types/chat.ts +11 -0
  301. package/src/types/index.ts +8 -0
  302. package/src/types/service.ts +102 -0
  303. package/src/types/template.ts +34 -0
  304. package/src/types/types.ts +41 -0
  305. package/src/typings.d.ts +9 -0
@@ -0,0 +1,268 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import '@dxos-theme';
6
+
7
+ import { type StoryObj, type Meta } from '@storybook/react';
8
+ import React, { useCallback, useEffect, useMemo, useState } from 'react';
9
+
10
+ import {
11
+ Capabilities,
12
+ Events,
13
+ IntentPlugin,
14
+ SettingsPlugin,
15
+ Surface,
16
+ useCapabilities,
17
+ useIntentDispatcher,
18
+ } from '@dxos/app-framework';
19
+ import { withPluginManager } from '@dxos/app-framework/testing';
20
+ import { Message, type Tool } from '@dxos/artifact';
21
+ import { genericTools, localServiceEndpoints, type IsObject } from '@dxos/artifact-testing';
22
+ import { AIServiceClientImpl } from '@dxos/assistant';
23
+ import { createStatic, ObjectId } from '@dxos/echo-schema';
24
+ import { EdgeHttpClient } from '@dxos/edge-client';
25
+ import { invariant } from '@dxos/invariant';
26
+ import { DXN, QueueSubspaceTags, SpaceId } from '@dxos/keys';
27
+ import { ChessPlugin } from '@dxos/plugin-chess';
28
+ import { ChessType } from '@dxos/plugin-chess/types';
29
+ import { ClientPlugin } from '@dxos/plugin-client';
30
+ import { InboxPlugin } from '@dxos/plugin-inbox';
31
+ import { MapPlugin } from '@dxos/plugin-map';
32
+ import { SpacePlugin } from '@dxos/plugin-space';
33
+ import { TablePlugin } from '@dxos/plugin-table';
34
+ import { useSpace } from '@dxos/react-client/echo';
35
+ import { withClientProvider } from '@dxos/react-client/testing';
36
+ import { useQueue } from '@dxos/react-edge-client';
37
+ import { IconButton, Input, Toolbar } from '@dxos/react-ui';
38
+ import { mx } from '@dxos/react-ui-theme';
39
+ import { withLayout, withSignals, withTheme } from '@dxos/storybook-utils';
40
+
41
+ import { Thread, type ThreadProps } from './Thread';
42
+ import { ChatProcessor } from '../../hooks';
43
+ import { createProcessorOptions } from '../../testing';
44
+ import translations from '../../translations';
45
+
46
+ const endpoints = localServiceEndpoints;
47
+
48
+ type RenderProps = {
49
+ items?: IsObject[];
50
+ prompts?: string[];
51
+ } & Pick<ThreadProps, 'debug'>;
52
+
53
+ // TODO(burdon): Use ChatContainer.
54
+ const Render = ({ items: _items, prompts = [], ...props }: RenderProps) => {
55
+ const space = useSpace();
56
+ const artifactDefinitions = useCapabilities(Capabilities.ArtifactDefinition);
57
+ const tools = useMemo<Tool[]>(
58
+ () => [...genericTools, ...artifactDefinitions.flatMap((definition) => definition.tools)],
59
+ [genericTools, artifactDefinitions],
60
+ );
61
+
62
+ const [aiClient] = useState(() => new AIServiceClientImpl({ endpoint: endpoints.ai }));
63
+ const [edgeClient] = useState(() => new EdgeHttpClient(endpoints.edge));
64
+ const { dispatchPromise: dispatch } = useIntentDispatcher();
65
+
66
+ // TODO(burdon): Replace with useChatProcessor.
67
+ // const processor = useChatProcessor(space);
68
+ const processor = useMemo<ChatProcessor | undefined>(() => {
69
+ if (!space) {
70
+ return;
71
+ }
72
+
73
+ return new ChatProcessor(
74
+ aiClient,
75
+ tools,
76
+ {
77
+ space,
78
+ dispatch,
79
+ },
80
+ createProcessorOptions(artifactDefinitions.map((definition) => definition.instructions)),
81
+ );
82
+ }, [aiClient, tools, space, dispatch, artifactDefinitions]);
83
+
84
+ // Queue.
85
+ const [queueDxn, setQueueDxn] = useState(() => randomQueueDxn());
86
+ const queue = useQueue<Message>(edgeClient, DXN.tryParse(queueDxn));
87
+
88
+ useEffect(() => {
89
+ if (queue?.items.length === 0 && !queue.isLoading && prompts.length > 0) {
90
+ queue.append([
91
+ createStatic(Message, {
92
+ role: 'assistant',
93
+ content: prompts.map(
94
+ (prompt) =>
95
+ ({
96
+ type: 'json',
97
+ disposition: 'suggest',
98
+ json: JSON.stringify({ text: prompt }),
99
+ }) as const,
100
+ ),
101
+ }),
102
+ ]);
103
+ }
104
+ }, [queueDxn, prompts, queue?.items.length, queue?.isLoading]);
105
+
106
+ // State.
107
+ const artifactItems: any[] = []; // TODO(burdon): Query from space.
108
+ const messages = [...(queue?.items ?? []), ...(processor?.messages.value ?? [])];
109
+
110
+ const handleSubmit = processor
111
+ ? (message: string) => {
112
+ requestAnimationFrame(async () => {
113
+ invariant(processor);
114
+ if (processor.streaming.value) {
115
+ await processor.cancel();
116
+ }
117
+
118
+ invariant(queue);
119
+ await processor.request(message, {
120
+ history: queue.items,
121
+ onComplete: (messages) => {
122
+ queue.append(messages);
123
+ },
124
+ });
125
+ });
126
+
127
+ return true;
128
+ }
129
+ : undefined;
130
+
131
+ const handlePrompt = useCallback(
132
+ (text: string) => {
133
+ void handleSubmit?.(text);
134
+ },
135
+ [handleSubmit],
136
+ );
137
+
138
+ const handleDelete = useCallback(
139
+ (id: string) => {
140
+ invariant(ObjectId.isValid(id), 'Invalid message id');
141
+ void queue?.delete([id]);
142
+ },
143
+ [queue],
144
+ );
145
+
146
+ return (
147
+ <div className='grid grid-cols-2 w-full h-full divide-x divide-separator overflow-hidden'>
148
+ {/* Thread */}
149
+ <div className='flex flex-col gap-4 overflow-hidden'>
150
+ <Toolbar.Root classNames='p-2'>
151
+ <Input.Root>
152
+ <Input.TextInput
153
+ spellCheck={false}
154
+ placeholder='Queue DXN'
155
+ value={queueDxn}
156
+ onChange={(ev) => setQueueDxn(ev.target.value)}
157
+ />
158
+ <IconButton
159
+ iconOnly
160
+ label='Copy DXN'
161
+ icon='ph--copy--regular'
162
+ onClick={() => navigator.clipboard.writeText(queueDxn)}
163
+ />
164
+ <IconButton
165
+ iconOnly
166
+ label='Clear history'
167
+ icon='ph--trash--regular'
168
+ onClick={() => setQueueDxn(randomQueueDxn())}
169
+ />
170
+ <IconButton iconOnly label='Stop' icon='ph--stop--regular' onClick={() => processor?.cancel()} />
171
+ </Input.Root>
172
+ </Toolbar.Root>
173
+
174
+ {/* TODO(burdon): Replace with ThreadContainer. */}
175
+ <Thread
176
+ messages={messages}
177
+ processing={processor?.streaming.value}
178
+ error={processor?.error.value}
179
+ tools={processor?.tools}
180
+ onSubmit={processor ? handleSubmit : undefined}
181
+ onPrompt={processor ? handlePrompt : undefined}
182
+ onDelete={processor ? handleDelete : undefined}
183
+ {...props}
184
+ />
185
+ </div>
186
+
187
+ {/* Artifacts Deck */}
188
+ <div className='overflow-hidden grid grid-rows-[2fr_1fr] divide-y divide-separator'>
189
+ {artifactItems.length > 0 && (
190
+ <div className={mx('flex grow overflow-hidden', artifactItems.length === 1 && 'row-span-2')}>
191
+ <Surface role='canvas-node' limit={1} data={artifactItems[0]} />
192
+ </div>
193
+ )}
194
+
195
+ {artifactItems.length > 1 && (
196
+ <div className='flex shrink-0 overflow-hidden divide-x divide-separator'>
197
+ <div className='flex flex-1 h-full'>
198
+ {artifactItems.slice(1, 3).map((item, idx) => (
199
+ <Surface key={idx} role='canvas-node' limit={1} data={item} />
200
+ ))}
201
+ </div>
202
+ </div>
203
+ )}
204
+ </div>
205
+ </div>
206
+ );
207
+ };
208
+
209
+ const randomQueueDxn = () =>
210
+ new DXN(DXN.kind.QUEUE, [QueueSubspaceTags.DATA, SpaceId.random(), ObjectId.random()]).toString();
211
+
212
+ const meta: Meta<typeof Render> = {
213
+ title: 'plugins/plugin-automation/ThreadContainer',
214
+ render: Render,
215
+ decorators: [
216
+ withSignals,
217
+ withClientProvider({
218
+ createIdentity: true,
219
+ createSpace: true,
220
+ }),
221
+ withPluginManager({
222
+ plugins: [
223
+ ClientPlugin({
224
+ onClientInitialized: async (_, client) => {
225
+ await client.halo.createIdentity();
226
+ },
227
+ }),
228
+ SpacePlugin({ observability: false }),
229
+ SettingsPlugin(),
230
+ IntentPlugin(),
231
+
232
+ // Artifacts.
233
+ ChessPlugin(),
234
+ InboxPlugin(),
235
+ MapPlugin(),
236
+ TablePlugin(),
237
+ ],
238
+ fireEvents: [Events.SetupArtifactDefinition],
239
+ }),
240
+ withTheme,
241
+ withLayout({ fullscreen: true, tooltips: true }),
242
+ ],
243
+ parameters: {
244
+ translations,
245
+ },
246
+ };
247
+
248
+ export default meta;
249
+
250
+ type Story = StoryObj<typeof Render>;
251
+
252
+ export const Default: Story = {
253
+ args: {
254
+ debug: true,
255
+ prompts: ['What tools do you have access to?', 'Show me a chess puzzle'],
256
+ },
257
+ };
258
+
259
+ export const WithInitialItems: Story = {
260
+ args: {
261
+ debug: true,
262
+ items: [
263
+ createStatic(ChessType, {
264
+ fen: 'rnbqkbnr/pppppppp/8/8/8/8/PPPPPPPP/RNBQKBNR w KQkq - 0 1',
265
+ }),
266
+ ],
267
+ },
268
+ };
@@ -0,0 +1,79 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import React, { type FC, useCallback } from 'react';
6
+
7
+ import { invariant } from '@dxos/invariant';
8
+ import { log } from '@dxos/log';
9
+ import { getSpace } from '@dxos/react-client/echo';
10
+ import { type ThemedClassName } from '@dxos/react-ui';
11
+
12
+ import { Thread, type ThreadProps } from './Thread';
13
+ import { useChatProcessor, useMessageQueue } from '../../hooks';
14
+ import { type AIChatType, type AssistantSettingsProps } from '../../types';
15
+
16
+ export type ThreadContainerProps = {
17
+ chat?: AIChatType;
18
+ settings?: AssistantSettingsProps;
19
+ } & Pick<ThreadProps, 'debug' | 'transcription' | 'onOpenChange'>;
20
+
21
+ // TODO(burdon): Since this only wraps Thread, just separate out hook?
22
+ export const ThreadContainer: FC<ThemedClassName<ThreadContainerProps>> = ({
23
+ classNames,
24
+ chat,
25
+ settings,
26
+ onOpenChange,
27
+ ...props
28
+ }) => {
29
+ // Push up capabilities hooks out of components.
30
+ const space = getSpace(chat);
31
+ const processor = useChatProcessor(space, settings);
32
+ const messageQueue = useMessageQueue(chat);
33
+ const messages = [...(messageQueue?.items ?? []), ...processor.messages.value];
34
+
35
+ const handleSubmit = useCallback(
36
+ (text: string) => {
37
+ // Don't accept input if still processing.
38
+ if (processor.streaming.value) {
39
+ log.warn('ignoring submit; still processing.');
40
+ return false;
41
+ }
42
+
43
+ onOpenChange?.(true);
44
+
45
+ invariant(messageQueue);
46
+ void processor.request(text, {
47
+ history: messageQueue.items,
48
+ onComplete: (messages) => {
49
+ messageQueue.append(messages);
50
+ },
51
+ });
52
+
53
+ return true;
54
+ },
55
+ [processor, messageQueue, onOpenChange],
56
+ );
57
+
58
+ const handleCancel = useCallback(() => {
59
+ if (processor.streaming.value) {
60
+ void processor.cancel();
61
+ }
62
+ }, [processor]);
63
+
64
+ return (
65
+ <Thread
66
+ classNames={classNames}
67
+ space={space}
68
+ messages={messages}
69
+ processing={processor.streaming.value}
70
+ error={processor.error.value}
71
+ tools={processor.tools}
72
+ onSubmit={handleSubmit}
73
+ onCancel={handleCancel}
74
+ onPrompt={handleSubmit}
75
+ onOpenChange={onOpenChange}
76
+ {...props}
77
+ />
78
+ );
79
+ };
@@ -0,0 +1,188 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import React, { type PropsWithChildren, type FC } from 'react';
6
+
7
+ import { type MessageContentBlock, type Message, type ToolType } from '@dxos/artifact';
8
+ import { invariant } from '@dxos/invariant';
9
+ import { type Space } from '@dxos/react-client/echo';
10
+ import { Button, ButtonGroup, Icon, IconButton, type ThemedClassName } from '@dxos/react-ui';
11
+ import {
12
+ MarkdownViewer,
13
+ ToggleContainer as NativeToggleContainer,
14
+ type ToggleContainerProps,
15
+ } from '@dxos/react-ui-components';
16
+ import { Json } from '@dxos/react-ui-syntax-highlighter';
17
+ import { mx } from '@dxos/react-ui-theme';
18
+ import { safeParseJson } from '@dxos/util';
19
+
20
+ import { ToolBlock, isToolMessage } from './ToolInvocations';
21
+ import { ToolboxContainer } from '../Toolbox';
22
+
23
+ const panelClassNames = 'flex flex-col w-full px-2 bg-groupSurface rounded-md';
24
+ const userClassNames = 'bg-[--user-fill]';
25
+
26
+ const ToggleContainer = (props: ToggleContainerProps) => {
27
+ return <NativeToggleContainer {...props} classNames={mx(panelClassNames, props.classNames)} />;
28
+ };
29
+
30
+ const MessageContainer = ({ children, classNames, user }: ThemedClassName<PropsWithChildren<{ user?: boolean }>>) => {
31
+ if (!children) {
32
+ return null;
33
+ }
34
+
35
+ return (
36
+ <div role='list-item' className={mx('flex w-full', user && 'justify-end', classNames)}>
37
+ <div className={mx(user ? ['px-2 py-1 rounded-md', userClassNames] : 'w-full')}>{children}</div>
38
+ </div>
39
+ );
40
+ };
41
+
42
+ export type ThreadMessageProps = ThemedClassName<{
43
+ space?: Space;
44
+ message: Message;
45
+ debug?: boolean;
46
+ tools?: ToolType[];
47
+ onPrompt?: (text: string) => void;
48
+ onDelete?: (id: string) => void;
49
+ }>;
50
+
51
+ export const ThreadMessage: FC<ThreadMessageProps> = ({ classNames, space, message, tools, onPrompt }) => {
52
+ const { role, content = [] } = message;
53
+
54
+ // TODO(burdon): Restructure types to make check unnecessary.
55
+ if (isToolMessage(message)) {
56
+ return (
57
+ <MessageContainer classNames={classNames}>
58
+ <ToolBlock space={space} classNames={panelClassNames} message={message} tools={tools} />
59
+ </MessageContainer>
60
+ );
61
+ }
62
+
63
+ return content.map((block, idx) => {
64
+ // TODO(burdon): Filter empty messages.
65
+ if (block.type === 'text' && block.text.replaceAll(/\s+/g, '').length === 0) {
66
+ return null;
67
+ }
68
+
69
+ const Component = components[block.type] ?? components.default;
70
+
71
+ return (
72
+ <MessageContainer key={idx} classNames={classNames} user={block.type === 'text' && role === 'user'}>
73
+ <Component space={space} block={block} onPrompt={onPrompt} />
74
+ </MessageContainer>
75
+ );
76
+ });
77
+ };
78
+
79
+ type BlockComponent = FC<{ space?: Space; block: MessageContentBlock; onPrompt?: (text: string) => void }>;
80
+
81
+ const components: Record<string, BlockComponent> = {
82
+ //
83
+ // Text
84
+ //
85
+ ['text' as const]: ({ block }) => {
86
+ invariant(block.type === 'text');
87
+ // const [open, setOpen] = useState(block.disposition === 'cot' && block.pending);
88
+ const title = block.disposition ? titles[block.disposition] : undefined;
89
+ if (!title) {
90
+ return <MarkdownViewer content={block.text} />;
91
+ }
92
+
93
+ // TOOD(burdon): Store last time user opened/closed COT.
94
+ // Autoclose when streaming ends.
95
+ // useEffect(() => {
96
+ // if (block.disposition === 'cot' && !block.pending) {
97
+ // setOpen(false);
98
+ // }
99
+ // }, [block.disposition, block.pending]);
100
+
101
+ return (
102
+ <ToggleContainer
103
+ // open={open}
104
+ defaultOpen={block.disposition === 'cot' && block.pending}
105
+ title={title}
106
+ icon={
107
+ block.pending ? (
108
+ <Icon icon={'ph--circle-notch--regular'} classNames='text-subdued ml-2 animate-spin' size={4} />
109
+ ) : undefined
110
+ }
111
+ >
112
+ <MarkdownViewer
113
+ content={block.text}
114
+ classNames={['pbe-2', block.disposition === 'cot' && 'text-sm text-subdued']}
115
+ />
116
+ </ToggleContainer>
117
+ );
118
+ },
119
+
120
+ //
121
+ // JSON
122
+ //
123
+ ['json' as const]: ({ space, block, onPrompt }) => {
124
+ invariant(block.type === 'json');
125
+
126
+ switch (block.disposition) {
127
+ case 'tool_list': {
128
+ return (
129
+ <ToggleContainer title={titles[block.disposition]} defaultOpen={true}>
130
+ <ToolboxContainer space={space} classNames='pbe-2' />
131
+ </ToggleContainer>
132
+ );
133
+ }
134
+
135
+ case 'suggest': {
136
+ const { text = '' }: { text: string } = safeParseJson(block.json ?? '{}') ?? ({} as any);
137
+ return <IconButton icon='ph--lightning--regular' label={text} onClick={() => onPrompt?.(text)} />;
138
+ }
139
+
140
+ case 'select': {
141
+ const { options = [] }: { options: string[] } = safeParseJson(block.json ?? '{}') ?? ({} as any);
142
+ return (
143
+ <ButtonGroup>
144
+ {options.map((option) => (
145
+ <Button key={option} onClick={() => onPrompt?.(option)}>
146
+ {option}
147
+ </Button>
148
+ ))}
149
+ </ButtonGroup>
150
+ );
151
+ }
152
+
153
+ default: {
154
+ const title = block.disposition ? titles[block.disposition] : undefined;
155
+ return (
156
+ <ToggleContainer title={title ?? 'JSON'}>
157
+ <Json data={safeParseJson(block.json ?? block)} classNames='!p-1 text-xs' />
158
+ </ToggleContainer>
159
+ );
160
+ }
161
+ }
162
+ },
163
+
164
+ //
165
+ // Default
166
+ //
167
+ default: ({ block }) => {
168
+ let title = titles[block.type];
169
+ if (block.type === 'tool_use') {
170
+ title = `Tool [${block.name}]`; // TODO(burdon): Get label from tool.
171
+ }
172
+
173
+ return (
174
+ <ToggleContainer title={title ?? 'JSON'}>
175
+ <Json data={block} classNames='!p-1 text-xs' />
176
+ </ToggleContainer>
177
+ );
178
+ },
179
+ };
180
+
181
+ // TODO(burdon): Translations.
182
+ const titles: Record<string, string> = {
183
+ ['cot' as const]: 'Chain of thought',
184
+ ['artifact' as const]: 'Artifact',
185
+ ['tool_use' as const]: 'Tool request',
186
+ ['tool_result' as const]: 'Tool result',
187
+ ['tool_list' as const]: 'Tools',
188
+ };
@@ -0,0 +1,104 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import React, { type FC, useEffect, useMemo, useRef, useState } from 'react';
6
+
7
+ import { type Message, type ToolType } from '@dxos/artifact';
8
+ import { log } from '@dxos/log';
9
+ import { type ThemedClassName } from '@dxos/react-ui';
10
+ import { NumericTabs, StatusRoll, ToggleContainer } from '@dxos/react-ui-components';
11
+ import { Json } from '@dxos/react-ui-syntax-highlighter';
12
+ import { isNonNullable, isNotFalsy } from '@dxos/util';
13
+
14
+ import { type ThreadMessageProps } from './ThreadMessage';
15
+
16
+ export const isToolMessage = (message: Message) => {
17
+ return message.content.some((block) => block.type === 'tool_use' || block.type === 'tool_result');
18
+ };
19
+
20
+ const getToolName = (tool: ToolType) => {
21
+ return tool.namespace && tool.function ? `${tool.namespace}:${tool.function}` : tool.name.split('_').pop();
22
+ };
23
+
24
+ const getToolCaption = (tool: ToolType | undefined) => {
25
+ if (!tool) {
26
+ return 'Calling tool...';
27
+ }
28
+
29
+ return tool.caption ?? `Calling ${getToolName(tool)}...`;
30
+ };
31
+
32
+ export const ToolBlock: FC<ThemedClassName<ThreadMessageProps>> = ({ classNames, message, tools }) => {
33
+ const { content = [] } = message;
34
+
35
+ let request: { tool: ToolType | undefined; block: any } | undefined;
36
+ const blocks = content.filter((block) => block.type === 'tool_use' || block.type === 'tool_result');
37
+ const items = blocks
38
+ .map((block) => {
39
+ switch (block.type) {
40
+ case 'tool_use': {
41
+ // TODO(burdon): Skip these updates?
42
+ if (block.pending && request?.block.id === block.id) {
43
+ return null;
44
+ }
45
+
46
+ request = { tool: tools?.find((tool) => tool.name === block.name), block };
47
+ return { title: getToolCaption(request.tool), block };
48
+ }
49
+
50
+ case 'tool_result': {
51
+ if (!request) {
52
+ log.warn('unexpected message', { block });
53
+ return { title: 'Error', block };
54
+ }
55
+
56
+ return { title: `${getToolCaption(request.tool)} (Success)`, block };
57
+ }
58
+
59
+ default: {
60
+ request = undefined;
61
+ return { title: 'Error', block };
62
+ }
63
+ }
64
+ })
65
+ .filter(isNonNullable);
66
+
67
+ return <ToolContainer classNames={classNames} items={items} />;
68
+ };
69
+
70
+ export const ToolContainer: FC<ThemedClassName<{ items: { title: string; block: any }[] }>> = ({
71
+ classNames,
72
+ items,
73
+ }) => {
74
+ const tabsRef = useRef<HTMLDivElement>(null);
75
+ const [selected, setSelected] = useState(0);
76
+ const [open, setOpen] = useState(false);
77
+ useEffect(() => {
78
+ if (open) {
79
+ tabsRef.current?.focus();
80
+ }
81
+ }, [open]);
82
+
83
+ const handleSelect = (index: number) => {
84
+ if (index === selected) {
85
+ setOpen(false);
86
+ } else {
87
+ setSelected(index);
88
+ }
89
+ };
90
+
91
+ const title = useMemo(() => {
92
+ const lines = items.map((item) => item.title).filter(isNotFalsy);
93
+ return <StatusRoll key='status-roll' lines={lines} duration={1_000} autoAdvance />;
94
+ }, [items]);
95
+
96
+ return (
97
+ <ToggleContainer classNames={['flex flex-col', classNames]} title={title} open={open} onChangeOpen={setOpen}>
98
+ <div className='grid grid-cols-[32px_1fr]'>
99
+ <NumericTabs ref={tabsRef} length={items.length} selected={selected} onSelect={handleSelect} />
100
+ <Json data={items[selected].block} classNames='!p-1 text-xs' />
101
+ </div>
102
+ </ToggleContainer>
103
+ );
104
+ };
@@ -0,0 +1,6 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ export * from './Thread';
6
+ export * from './ThreadContainer';
@@ -0,0 +1,52 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { type Message } from '@dxos/artifact';
6
+
7
+ // TODO(burdon): Move to util?
8
+ type Reducer<R, I> = (acc: R, value: I) => R;
9
+
10
+ /**
11
+ * Reducer that collapses related message blocks into single messages.
12
+ * For example, combines tool request/response pairs into a single message.
13
+ */
14
+ export const messageReducer: Reducer<{ messages: Message[]; current?: Message }, Message> = (
15
+ { current, messages },
16
+ message,
17
+ ) => {
18
+ let i = 0;
19
+ for (const block of message.content) {
20
+ switch (block.type) {
21
+ case 'tool_use':
22
+ case 'tool_result': {
23
+ if (current) {
24
+ current.content.push(block);
25
+ } else {
26
+ current = {
27
+ id: [message.id, i].join('_'),
28
+ role: message.role,
29
+ content: [block],
30
+ };
31
+ messages.push(current);
32
+ }
33
+ break;
34
+ }
35
+
36
+ case 'text':
37
+ default: {
38
+ current = undefined;
39
+ messages.push({
40
+ id: [message.id, i].join('_'),
41
+ role: message.role,
42
+ content: [block],
43
+ });
44
+ break;
45
+ }
46
+ }
47
+
48
+ i++;
49
+ }
50
+
51
+ return { current, messages };
52
+ };