airaknit 1.1.2-rc.9

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 (440) hide show
  1. package/LICENSE +84 -0
  2. package/README.md +202 -0
  3. package/bin/airaknit +9 -0
  4. package/bin/airaknit-project +14 -0
  5. package/bin/kanna +9 -0
  6. package/dist/client/assets/CompactSummaryMessage-Yw0BDWEJ.js +1 -0
  7. package/dist/client/assets/ExitPlanModeMessage-DIdkQ4uF.js +1 -0
  8. package/dist/client/assets/LocalFilePreviewDialog-DQx2eiCc.js +3 -0
  9. package/dist/client/assets/LocalProjectsSection-C4xlWkgS.js +1 -0
  10. package/dist/client/assets/TextMessage-B5G39DEJ.js +1 -0
  11. package/dist/client/assets/UserMessage-CIkWk-0L.js +1 -0
  12. package/dist/client/assets/_basePickBy-CVrAFfnZ.js +1 -0
  13. package/dist/client/assets/_baseUniq-JL-aaF4P.js +1 -0
  14. package/dist/client/assets/arc-B07zg7ol.js +1 -0
  15. package/dist/client/assets/architecture-YZFGNWBL-PSLVJL3p.js +1 -0
  16. package/dist/client/assets/architectureDiagram-Q4EWVU46-DfWIF1G_.js +36 -0
  17. package/dist/client/assets/array-BGFCBI0e.js +1 -0
  18. package/dist/client/assets/blockDiagram-DXYQGD6D-CzwIeo_B.js +132 -0
  19. package/dist/client/assets/bundle-mjs-BDE2gWbQ.js +1 -0
  20. package/dist/client/assets/button-DO50qOGv.js +1 -0
  21. package/dist/client/assets/c4Diagram-AHTNJAMY-CR8DCQRE.js +10 -0
  22. package/dist/client/assets/channel-Dj-UUfaF.js +1 -0
  23. package/dist/client/assets/chunk-2KRD3SAO-dznP-cn8.js +1 -0
  24. package/dist/client/assets/chunk-336JU56O-Dqss5Vu6.js +2 -0
  25. package/dist/client/assets/chunk-426QAEUC-CEKis_0_.js +1 -0
  26. package/dist/client/assets/chunk-4BX2VUAB-BYOv0Gm1.js +1 -0
  27. package/dist/client/assets/chunk-4TB4RGXK-BxzubH5S.js +206 -0
  28. package/dist/client/assets/chunk-55IACEB6-BSlTj03a.js +1 -0
  29. package/dist/client/assets/chunk-5FUZZQ4R-9Au93Bi1.js +62 -0
  30. package/dist/client/assets/chunk-5PVQY5BW-BhksHFEZ.js +2 -0
  31. package/dist/client/assets/chunk-67CJDMHE-BFwhz-8t.js +1 -0
  32. package/dist/client/assets/chunk-7N4EOEYR-BDUPds87.js +1 -0
  33. package/dist/client/assets/chunk-AA7GKIK3-CEWTdyXO.js +1 -0
  34. package/dist/client/assets/chunk-BO2N2NFS-D0LvxnhU.js +103 -0
  35. package/dist/client/assets/chunk-BSJP7CBP-BNJnK6sq.js +1 -0
  36. package/dist/client/assets/chunk-Bj-mKKzh.js +1 -0
  37. package/dist/client/assets/chunk-CIAEETIT-CYhfoCeN.js +1 -0
  38. package/dist/client/assets/chunk-EDXVE4YY-C5ovJLc0.js +1 -0
  39. package/dist/client/assets/chunk-ENJZ2VHE-HOhYaeGr.js +10 -0
  40. package/dist/client/assets/chunk-FMBD7UC4-BLCiKcAQ.js +15 -0
  41. package/dist/client/assets/chunk-FOC6F5B3-B6GtY2ek.js +1 -0
  42. package/dist/client/assets/chunk-ICPOFSXX-DPoIZoC5.js +122 -0
  43. package/dist/client/assets/chunk-K5T4RW27-BsKN63rv.js +94 -0
  44. package/dist/client/assets/chunk-KGLVRYIC-BUGn9uuY.js +1 -0
  45. package/dist/client/assets/chunk-LIHQZDEY-DhaZyo03.js +1 -0
  46. package/dist/client/assets/chunk-ORNJ4GCN-DlpeeJyi.js +1 -0
  47. package/dist/client/assets/chunk-OYMX7WX6-Dc9q7aYA.js +231 -0
  48. package/dist/client/assets/chunk-QZHKN3VN-BEdrPoSb.js +1 -0
  49. package/dist/client/assets/chunk-U2HBQHQK-CIB3Bjjd.js +70 -0
  50. package/dist/client/assets/chunk-X2U36JSP-CtB-o8Yp.js +1 -0
  51. package/dist/client/assets/chunk-XPW4576I-C6iHhX_8.js +32 -0
  52. package/dist/client/assets/chunk-YZCP3GAM-CTmKr6ZH.js +1 -0
  53. package/dist/client/assets/chunk-ZZ45TVLE-BgU8A2RF.js +1 -0
  54. package/dist/client/assets/classDiagram-6PBFFD2Q-Bqk5e679.js +1 -0
  55. package/dist/client/assets/classDiagram-v2-HSJHXN6E-6pSaZOkC.js +1 -0
  56. package/dist/client/assets/client-BrKWI4CM.js +1 -0
  57. package/dist/client/assets/client-CGgNRU9w.js +1 -0
  58. package/dist/client/assets/client-DMSLRzg9.js +6 -0
  59. package/dist/client/assets/clone-DWcL7whJ.js +1 -0
  60. package/dist/client/assets/cose-bilkent-S5V4N54A-CrV5wsV_.js +1 -0
  61. package/dist/client/assets/cytoscape.esm--aLzKuep.js +321 -0
  62. package/dist/client/assets/dagre-CuRxWcrj.js +1 -0
  63. package/dist/client/assets/dagre-KV5264BT-BIDiVnkA.js +4 -0
  64. package/dist/client/assets/defaultLocale-CRZydyG6.js +1 -0
  65. package/dist/client/assets/diagram-5BDNPKRD-i1kjKRCB.js +10 -0
  66. package/dist/client/assets/diagram-G4DWMVQ6-9ZSLuhbl.js +24 -0
  67. package/dist/client/assets/diagram-MMDJMWI5-B4_CUjgv.js +43 -0
  68. package/dist/client/assets/diagram-TYMM5635-Ct5eTGS8.js +24 -0
  69. package/dist/client/assets/dist-CuB4kiSK.js +1 -0
  70. package/dist/client/assets/erDiagram-SMLLAGMA-Cy38ercc.js +85 -0
  71. package/dist/client/assets/flowDiagram-DWJPFMVM-CZKuYl0V.js +162 -0
  72. package/dist/client/assets/ganttDiagram-T4ZO3ILL-DLPjCh7a.js +292 -0
  73. package/dist/client/assets/gitGraph-7Q5UKJZL-DqbrtEp9.js +1 -0
  74. package/dist/client/assets/gitGraphDiagram-UUTBAWPF-BoRBkDhQ.js +106 -0
  75. package/dist/client/assets/graphlib-BcQ6qlQh.js +1 -0
  76. package/dist/client/assets/highlighted-body-OFNGDK62-BEpBVDTX.js +1 -0
  77. package/dist/client/assets/index-CetCiuqP.js +105 -0
  78. package/dist/client/assets/index-N29Mip7A.css +1 -0
  79. package/dist/client/assets/info-OMHHGYJF-D98DRBJX.js +1 -0
  80. package/dist/client/assets/infoDiagram-42DDH7IO-BAcdTWbt.js +2 -0
  81. package/dist/client/assets/init-B8gtcn7T.js +1 -0
  82. package/dist/client/assets/isArrayLikeObject-D8SJFmkN.js +1 -0
  83. package/dist/client/assets/isEmpty-BF3YX5Jk.js +1 -0
  84. package/dist/client/assets/ishikawaDiagram-UXIWVN3A-Ynu2VKdC.js +70 -0
  85. package/dist/client/assets/journeyDiagram-VCZTEJTY-BjfhQaN3.js +139 -0
  86. package/dist/client/assets/jsx-runtime-CyI9ICYU.js +1 -0
  87. package/dist/client/assets/kanban-definition-6JOO6SKY-JLXH9zUJ.js +89 -0
  88. package/dist/client/assets/katex-B94qP8b6.js +265 -0
  89. package/dist/client/assets/lib--QVjyxmL.js +29 -0
  90. package/dist/client/assets/lib-B6rgJiZ9.js +1 -0
  91. package/dist/client/assets/line-DCrYfLBn.js +1 -0
  92. package/dist/client/assets/linear-_4upLmeo.js +1 -0
  93. package/dist/client/assets/mermaid-GHXKKRXX-rwJHYUmW.js +1 -0
  94. package/dist/client/assets/mermaid-parser.core-KZinfW8o.js +4 -0
  95. package/dist/client/assets/mermaid.core-QqY9gSNe.js +11 -0
  96. package/dist/client/assets/mindmap-definition-QFDTVHPH-TWgHDAzp.js +96 -0
  97. package/dist/client/assets/ordinal-CCj7PWgZ.js +1 -0
  98. package/dist/client/assets/packet-4T2RLAQJ-DEvfkn3F.js +1 -0
  99. package/dist/client/assets/path-DZF-JdEe.js +1 -0
  100. package/dist/client/assets/pie-ZZUOXDRM-72e6WVjb.js +1 -0
  101. package/dist/client/assets/pieDiagram-DEJITSTG-Cl8PCsoj.js +30 -0
  102. package/dist/client/assets/preload-helper-rov5CBGT.js +1 -0
  103. package/dist/client/assets/pty-client-DZ27IS00.js +1 -0
  104. package/dist/client/assets/ptyInstancesStore-D9ag7SYd.js +1 -0
  105. package/dist/client/assets/quadrantDiagram-34T5L4WZ-CHyVGp9E.js +7 -0
  106. package/dist/client/assets/radar-PYXPWWZC-Cp7xd_EY.js +1 -0
  107. package/dist/client/assets/react-CClhXMB2.js +1 -0
  108. package/dist/client/assets/react-Dd6D81m0.js +1 -0
  109. package/dist/client/assets/react-dom--G6_6fQ_.js +1 -0
  110. package/dist/client/assets/requirementDiagram-MS252O5E-DaanG2iM.js +84 -0
  111. package/dist/client/assets/rough.esm-BsmKo2S5.js +1 -0
  112. package/dist/client/assets/sankeyDiagram-XADWPNL6-B_fhLY36.js +10 -0
  113. package/dist/client/assets/sequenceDiagram-FGHM5R23-C5FNrveI.js +157 -0
  114. package/dist/client/assets/src-DeTlMJAU.js +1 -0
  115. package/dist/client/assets/stateDiagram-FHFEXIEX-nTTcdjjQ.js +1 -0
  116. package/dist/client/assets/stateDiagram-v2-QKLJ7IA2-Dw0632j_.js +1 -0
  117. package/dist/client/assets/timeline-definition-GMOUNBTQ-DkQV1yP8.js +120 -0
  118. package/dist/client/assets/treeView-SZITEDCU-ZLIgC7_K.js +1 -0
  119. package/dist/client/assets/treemap-W4RFUUIX-BqaMbB6N.js +1 -0
  120. package/dist/client/assets/uiIdentityOverlay-Ba7GNj7m.js +1 -0
  121. package/dist/client/assets/vennDiagram-DHZGUBPP-DbZ2xgs6.js +34 -0
  122. package/dist/client/assets/wardley-RL74JXVD-DXQS8zf4.js +1 -0
  123. package/dist/client/assets/wardleyDiagram-NUSXRM2D-BzCJ6MAu.js +20 -0
  124. package/dist/client/assets/xychartDiagram-5P7HB3ND-BSlFecop.js +7 -0
  125. package/dist/client/favicon.png +0 -0
  126. package/dist/client/favicon.svg +15 -0
  127. package/dist/client/fonts/body-medium.woff2 +0 -0
  128. package/dist/client/fonts/body-regular-italic.woff2 +0 -0
  129. package/dist/client/fonts/body-regular.woff2 +0 -0
  130. package/dist/client/fonts/body-semibold.woff2 +0 -0
  131. package/dist/client/index.html +31 -0
  132. package/dist/client/manifest.webmanifest +12 -0
  133. package/dist/client/sw.js +32 -0
  134. package/package.json +122 -0
  135. package/src/nats/auth-callout/callout-config.test.ts +93 -0
  136. package/src/nats/auth-callout/callout-config.ts +109 -0
  137. package/src/nats/auth-callout/callout.integration.test.ts +332 -0
  138. package/src/nats/auth-callout/keys.ts +103 -0
  139. package/src/nats/auth-callout/responder.ts +241 -0
  140. package/src/nats/auth-callout/scope-policy.test.ts +159 -0
  141. package/src/nats/auth-callout/scope-policy.ts +210 -0
  142. package/src/nats/auth-callout/token.test.ts +163 -0
  143. package/src/nats/auth-callout/token.ts +157 -0
  144. package/src/nats/nats-daemon-callout.ts +194 -0
  145. package/src/nats/nats-daemon.test.ts +77 -0
  146. package/src/nats/nats-daemon.ts +50 -0
  147. package/src/nats/nats-token.test.ts +61 -0
  148. package/src/nats/nats-token.ts +59 -0
  149. package/src/runner/coordination-mcp-integration.test.ts +134 -0
  150. package/src/runner/nats-coordination-client.test.ts +49 -0
  151. package/src/runner/nats-coordination-client.ts +94 -0
  152. package/src/runner/runner-agent.test.ts +469 -0
  153. package/src/runner/runner-agent.ts +453 -0
  154. package/src/runner/runner-credential.test.ts +93 -0
  155. package/src/runner/runner-credential.ts +82 -0
  156. package/src/runner/runner-nats.test.ts +495 -0
  157. package/src/runner/runner-nats.ts +323 -0
  158. package/src/runner/runner-pair.test.ts +107 -0
  159. package/src/runner/runner-pair.ts +81 -0
  160. package/src/runner/runner.test.ts +135 -0
  161. package/src/runner/runner.ts +212 -0
  162. package/src/runner/turn-factories.test.ts +97 -0
  163. package/src/runner/turn-factories.ts +475 -0
  164. package/src/server/agent-config-journey.test.ts +106 -0
  165. package/src/server/agent.ts +8 -0
  166. package/src/server/auto-continue/auth-error-detector.ts +66 -0
  167. package/src/server/auto-continue/limit-detector.ts +194 -0
  168. package/src/server/bm25.test.ts +92 -0
  169. package/src/server/bm25.ts +101 -0
  170. package/src/server/chat-events-jetstream.test.ts +135 -0
  171. package/src/server/claude-harness.ts +360 -0
  172. package/src/server/claude-pty/agent-normalizers.ts +309 -0
  173. package/src/server/claude-pty/auth.test.ts +38 -0
  174. package/src/server/claude-pty/auth.ts +32 -0
  175. package/src/server/claude-pty/claude-session-registry.adapter.ts +81 -0
  176. package/src/server/claude-pty/claude-session-registry.test.ts +149 -0
  177. package/src/server/claude-pty/driver.test.ts +902 -0
  178. package/src/server/claude-pty/driver.ts +807 -0
  179. package/src/server/claude-pty/jsonl-path.adapter.ts +57 -0
  180. package/src/server/claude-pty/jsonl-path.test.ts +114 -0
  181. package/src/server/claude-pty/jsonl-to-event.test.ts +241 -0
  182. package/src/server/claude-pty/jsonl-to-event.ts +174 -0
  183. package/src/server/claude-pty/output-ring.test.ts +35 -0
  184. package/src/server/claude-pty/output-ring.ts +25 -0
  185. package/src/server/claude-pty/parity-matrix.test.ts +227 -0
  186. package/src/server/claude-pty/pid-registry.adapter.ts +135 -0
  187. package/src/server/claude-pty/pid-registry.test.ts +122 -0
  188. package/src/server/claude-pty/preflight/binary-fingerprint.adapter.ts +20 -0
  189. package/src/server/claude-pty/preflight/binary-fingerprint.test.ts +32 -0
  190. package/src/server/claude-pty/pty-instance-registry.test.ts +177 -0
  191. package/src/server/claude-pty/pty-instance-registry.ts +166 -0
  192. package/src/server/claude-pty/pty-memory-sampler.adapter.test.ts +103 -0
  193. package/src/server/claude-pty/pty-memory-sampler.adapter.ts +85 -0
  194. package/src/server/claude-pty/pty-process.adapter.ts +66 -0
  195. package/src/server/claude-pty/pty-process.test.ts +49 -0
  196. package/src/server/claude-pty/resolve-binary.adapter.ts +106 -0
  197. package/src/server/claude-pty/resolve-binary.test.ts +118 -0
  198. package/src/server/claude-pty/runtime-dir.adapter.ts +19 -0
  199. package/src/server/claude-pty/settings-writer.adapter.ts +27 -0
  200. package/src/server/claude-pty/settings-writer.test.ts +22 -0
  201. package/src/server/claude-pty/smoke-test-io.adapter.ts +28 -0
  202. package/src/server/claude-pty/smoke-test.test.ts +191 -0
  203. package/src/server/claude-pty/smoke-test.ts +185 -0
  204. package/src/server/claude-pty/subagent-orchestrator.ts +887 -0
  205. package/src/server/claude-pty/tool-callback.ts +274 -0
  206. package/src/server/claude-pty/tui-control.test.ts +272 -0
  207. package/src/server/claude-pty/tui-control.ts +182 -0
  208. package/src/server/claude-pty/tui-source.adapter.test.ts +360 -0
  209. package/src/server/claude-pty/tui-source.adapter.ts +343 -0
  210. package/src/server/claude-pty/tunnel-gateway.ts +12 -0
  211. package/src/server/claude-pty-mcp/canonical-args.ts +15 -0
  212. package/src/server/claude-pty-mcp/fs-stat.adapter.ts +8 -0
  213. package/src/server/claude-pty-mcp/history-primer.ts +90 -0
  214. package/src/server/claude-pty-mcp/http-server.adapter.ts +33 -0
  215. package/src/server/claude-pty-mcp/mcp-http.ts +177 -0
  216. package/src/server/claude-pty-mcp/mcp.ts +412 -0
  217. package/src/server/claude-pty-mcp/mention-parser.ts +25 -0
  218. package/src/server/claude-pty-mcp/paths.ts +24 -0
  219. package/src/server/claude-pty-mcp/permission-gate.ts +243 -0
  220. package/src/server/claude-pty-mcp/terminal-pid-registry.adapter.ts +107 -0
  221. package/src/server/claude-pty-mcp/tools/ask-user-question.test.ts +119 -0
  222. package/src/server/claude-pty-mcp/tools/ask-user-question.ts +61 -0
  223. package/src/server/claude-pty-mcp/tools/bash.adapter.ts +76 -0
  224. package/src/server/claude-pty-mcp/tools/bash.test.ts +56 -0
  225. package/src/server/claude-pty-mcp/tools/delegate-subagent.test.ts +155 -0
  226. package/src/server/claude-pty-mcp/tools/delegate-subagent.ts +111 -0
  227. package/src/server/claude-pty-mcp/tools/edit.adapter.ts +95 -0
  228. package/src/server/claude-pty-mcp/tools/edit.test.ts +93 -0
  229. package/src/server/claude-pty-mcp/tools/exit-plan-mode.test.ts +61 -0
  230. package/src/server/claude-pty-mcp/tools/exit-plan-mode.ts +50 -0
  231. package/src/server/claude-pty-mcp/tools/glob.adapter.ts +86 -0
  232. package/src/server/claude-pty-mcp/tools/glob.test.ts +61 -0
  233. package/src/server/claude-pty-mcp/tools/grep.adapter.ts +126 -0
  234. package/src/server/claude-pty-mcp/tools/grep.test.ts +62 -0
  235. package/src/server/claude-pty-mcp/tools/read.adapter.ts +58 -0
  236. package/src/server/claude-pty-mcp/tools/read.test.ts +62 -0
  237. package/src/server/claude-pty-mcp/tools/tool-callback-shim.ts +42 -0
  238. package/src/server/claude-pty-mcp/tools/webfetch.test.ts +81 -0
  239. package/src/server/claude-pty-mcp/tools/webfetch.ts +82 -0
  240. package/src/server/claude-pty-mcp/tools/websearch.test.ts +40 -0
  241. package/src/server/claude-pty-mcp/tools/websearch.ts +42 -0
  242. package/src/server/claude-pty-mcp/tools/write.adapter.ts +60 -0
  243. package/src/server/claude-pty-mcp/tools/write.test.ts +52 -0
  244. package/src/server/claude-pty-mcp/uploads.adapter.ts +98 -0
  245. package/src/server/claude-pty-mcp/uploads.ts +38 -0
  246. package/src/server/claude-turn.test.ts +176 -0
  247. package/src/server/cli-runtime.test.ts +456 -0
  248. package/src/server/cli-runtime.ts +374 -0
  249. package/src/server/cli-supervisor.ts +81 -0
  250. package/src/server/cli.ts +78 -0
  251. package/src/server/client-log-forwarder.test.ts +74 -0
  252. package/src/server/client-log-forwarder.ts +75 -0
  253. package/src/server/codex-app-server-protocol.ts +449 -0
  254. package/src/server/codex-app-server.test.ts +2990 -0
  255. package/src/server/codex-app-server.ts +1713 -0
  256. package/src/server/coordination-integration.test.ts +63 -0
  257. package/src/server/coordination-mcp.test.ts +149 -0
  258. package/src/server/coordination-mcp.ts +197 -0
  259. package/src/server/delegation-coordinator.test.ts +675 -0
  260. package/src/server/delegation-coordinator.ts +454 -0
  261. package/src/server/discovery.test.ts +211 -0
  262. package/src/server/discovery.ts +301 -0
  263. package/src/server/event-store-agent-config.test.ts +124 -0
  264. package/src/server/event-store-coordination.test.ts +149 -0
  265. package/src/server/event-store-profile.test.ts +132 -0
  266. package/src/server/event-store-repo.test.ts +154 -0
  267. package/src/server/event-store-runner-team.test.ts +104 -0
  268. package/src/server/event-store.test.ts +342 -0
  269. package/src/server/event-store.ts +2208 -0
  270. package/src/server/events.ts +379 -0
  271. package/src/server/extension-router.test.ts +183 -0
  272. package/src/server/extension-router.ts +114 -0
  273. package/src/server/extensions/agents/server.test.ts +191 -0
  274. package/src/server/extensions/agents/server.ts +108 -0
  275. package/src/server/extensions/c3/server.test.ts +284 -0
  276. package/src/server/extensions/c3/server.ts +212 -0
  277. package/src/server/extensions/code/server.test.ts +200 -0
  278. package/src/server/extensions/code/server.ts +150 -0
  279. package/src/server/extensions.config.ts +10 -0
  280. package/src/server/external-open.ts +69 -0
  281. package/src/server/generate-fork-context.ts +58 -0
  282. package/src/server/generate-merge-context.test.ts +290 -0
  283. package/src/server/generate-merge-context.ts +141 -0
  284. package/src/server/generate-title.ts +36 -0
  285. package/src/server/git-clone-policy.test.ts +138 -0
  286. package/src/server/git-clone-policy.ts +27 -0
  287. package/src/server/harness-types.ts +1 -0
  288. package/src/server/journey-verification.test.ts +640 -0
  289. package/src/server/journey-verification.ts +195 -0
  290. package/src/server/machine-name.ts +22 -0
  291. package/src/server/nats-auth.test.ts +92 -0
  292. package/src/server/nats-auth.ts +6 -0
  293. package/src/server/nats-bind-guard.test.ts +71 -0
  294. package/src/server/nats-bind-guard.ts +42 -0
  295. package/src/server/nats-bridge.test.ts +141 -0
  296. package/src/server/nats-bridge.ts +111 -0
  297. package/src/server/nats-connector.test.ts +56 -0
  298. package/src/server/nats-connector.ts +71 -0
  299. package/src/server/nats-daemon-manager.test.ts +76 -0
  300. package/src/server/nats-daemon-manager.ts +174 -0
  301. package/src/server/nats-publisher.test.ts +356 -0
  302. package/src/server/nats-publisher.ts +271 -0
  303. package/src/server/nats-responders.test.ts +1018 -0
  304. package/src/server/nats-responders.ts +925 -0
  305. package/src/server/nats-streams.test.ts +112 -0
  306. package/src/server/nats-streams.ts +129 -0
  307. package/src/server/oauth-pool/oauth-responders.ts +185 -0
  308. package/src/server/oauth-pool/oauth-settings-store.ts +173 -0
  309. package/src/server/oauth-pool/oauth-token-pool.ts +303 -0
  310. package/src/server/orchestration.test.ts +1063 -0
  311. package/src/server/orchestration.ts +716 -0
  312. package/src/server/pairing-endpoints.test.ts +171 -0
  313. package/src/server/pairing-store.test.ts +154 -0
  314. package/src/server/pairing-store.ts +124 -0
  315. package/src/server/paths.ts +27 -0
  316. package/src/server/pr3-liveness.test.ts +252 -0
  317. package/src/server/process-utils.ts +10 -0
  318. package/src/server/project-cli.ts +180 -0
  319. package/src/server/provider-catalog.test.ts +177 -0
  320. package/src/server/provider-catalog.ts +146 -0
  321. package/src/server/pty-responders.ts +345 -0
  322. package/src/server/push-notifications.test.ts +234 -0
  323. package/src/server/push-notifications.ts +188 -0
  324. package/src/server/quick-response.test.ts +196 -0
  325. package/src/server/quick-response.ts +154 -0
  326. package/src/server/read-models-agent-config.test.ts +59 -0
  327. package/src/server/read-models-coordination.test.ts +69 -0
  328. package/src/server/read-models.test.ts +332 -0
  329. package/src/server/read-models.ts +258 -0
  330. package/src/server/repo-journey.test.ts +96 -0
  331. package/src/server/repo-manager.test.ts +118 -0
  332. package/src/server/repo-manager.ts +97 -0
  333. package/src/server/repo-status.test.ts +54 -0
  334. package/src/server/repo-status.ts +82 -0
  335. package/src/server/restart.test.ts +27 -0
  336. package/src/server/restart.ts +30 -0
  337. package/src/server/runner-incompatible-gate.test.ts +383 -0
  338. package/src/server/runner-manager.test.ts +72 -0
  339. package/src/server/runner-manager.ts +312 -0
  340. package/src/server/runner-pairing-urls.test.ts +59 -0
  341. package/src/server/runner-pairing-urls.ts +67 -0
  342. package/src/server/runner-proxy.test.ts +456 -0
  343. package/src/server/runner-proxy.ts +494 -0
  344. package/src/server/runner-router.test.ts +429 -0
  345. package/src/server/runner-router.ts +212 -0
  346. package/src/server/runner-routing.test.ts +584 -0
  347. package/src/server/runtime-registry.test.ts +436 -0
  348. package/src/server/runtime-registry.ts +307 -0
  349. package/src/server/sandbox-health.test.ts +127 -0
  350. package/src/server/sandbox-health.ts +94 -0
  351. package/src/server/sandbox-journey.test.ts +232 -0
  352. package/src/server/sandbox-manager.test.ts +146 -0
  353. package/src/server/sandbox-manager.ts +159 -0
  354. package/src/server/server.test.ts +287 -0
  355. package/src/server/server.ts +1108 -0
  356. package/src/server/session-discovery.test.ts +129 -0
  357. package/src/server/session-discovery.ts +475 -0
  358. package/src/server/session-index.test.ts +362 -0
  359. package/src/server/session-index.ts +119 -0
  360. package/src/server/session-seed.ts +288 -0
  361. package/src/server/share.test.ts +108 -0
  362. package/src/server/share.ts +113 -0
  363. package/src/server/skill-discovery.test.ts +201 -0
  364. package/src/server/skill-discovery.ts +77 -0
  365. package/src/server/storage/test-helpers.ts +67 -0
  366. package/src/server/terminal-manager.test.ts +309 -0
  367. package/src/server/terminal-manager.ts +354 -0
  368. package/src/server/transcript-consumer.test.ts +339 -0
  369. package/src/server/transcript-consumer.ts +162 -0
  370. package/src/server/transcript-search.test.ts +193 -0
  371. package/src/server/transcript-search.ts +83 -0
  372. package/src/server/transcript-utils.ts +52 -0
  373. package/src/server/update-manager.test.ts +107 -0
  374. package/src/server/update-manager.ts +230 -0
  375. package/src/server/workflow-engine.test.ts +251 -0
  376. package/src/server/workflow-engine.ts +169 -0
  377. package/src/server/workflow-mcp.ts +49 -0
  378. package/src/server/workflow-store.test.ts +155 -0
  379. package/src/server/workflow-store.ts +139 -0
  380. package/src/server/workspace-agent-integration.test.ts +167 -0
  381. package/src/server/workspace-agent-routes.test.ts +127 -0
  382. package/src/server/workspace-agent-routes.ts +89 -0
  383. package/src/server/workspace-agent.test.ts +103 -0
  384. package/src/server/workspace-agent.ts +102 -0
  385. package/src/server/workspace-cli.test.ts +79 -0
  386. package/src/server/workspace-config-manager.test.ts +109 -0
  387. package/src/server/workspace-config-manager.ts +83 -0
  388. package/src/server/workspace-directory-policy.test.ts +109 -0
  389. package/src/server/workspace-directory-policy.ts +56 -0
  390. package/src/shared/agent-config-types.ts +25 -0
  391. package/src/shared/branding.test.ts +42 -0
  392. package/src/shared/branding.ts +54 -0
  393. package/src/shared/compression.test.ts +85 -0
  394. package/src/shared/compression.ts +42 -0
  395. package/src/shared/coordination-store.test.ts +24 -0
  396. package/src/shared/coordination-store.ts +26 -0
  397. package/src/shared/dev-ports.test.ts +84 -0
  398. package/src/shared/dev-ports.ts +100 -0
  399. package/src/shared/extension-types.ts +45 -0
  400. package/src/shared/fork-presets.ts +54 -0
  401. package/src/shared/harness-types.test.ts +15 -0
  402. package/src/shared/harness-types.ts +21 -0
  403. package/src/shared/log-sink.test.ts +112 -0
  404. package/src/shared/log-sink.ts +185 -0
  405. package/src/shared/mention-pattern.ts +7 -0
  406. package/src/shared/merge-presets.ts +41 -0
  407. package/src/shared/nats-subjects.test.ts +61 -0
  408. package/src/shared/nats-subjects.ts +131 -0
  409. package/src/shared/permission-policy.ts +136 -0
  410. package/src/shared/ports.ts +3 -0
  411. package/src/shared/preset-types.ts +15 -0
  412. package/src/shared/profile-types.ts +49 -0
  413. package/src/shared/projectFileUrl.ts +36 -0
  414. package/src/shared/protocol.ts +153 -0
  415. package/src/shared/pty-instance.ts +43 -0
  416. package/src/shared/puggy/diagnostics/result.ts +18 -0
  417. package/src/shared/puggy/expressions/evaluate.ts +292 -0
  418. package/src/shared/puggy/index.test.ts +101 -0
  419. package/src/shared/puggy/index.ts +69 -0
  420. package/src/shared/puggy/parser/ast.ts +110 -0
  421. package/src/shared/puggy/parser/parser.ts +624 -0
  422. package/src/shared/puggy/renderer/html.ts +447 -0
  423. package/src/shared/runner-protocol.test.ts +277 -0
  424. package/src/shared/runner-protocol.ts +210 -0
  425. package/src/shared/runner-team-types.ts +73 -0
  426. package/src/shared/runtime-types.ts +48 -0
  427. package/src/shared/sandbox-types.ts +53 -0
  428. package/src/shared/tailwind-build.test.ts +12 -0
  429. package/src/shared/tinkaria-system-prompt.ts +101 -0
  430. package/src/shared/tools.test.ts +335 -0
  431. package/src/shared/tools.ts +397 -0
  432. package/src/shared/transcript-entries.ts +27 -0
  433. package/src/shared/transcript-render.test.ts +225 -0
  434. package/src/shared/transcript-render.ts +467 -0
  435. package/src/shared/types.ts +1031 -0
  436. package/src/shared/vite-config.test.ts +47 -0
  437. package/src/shared/web-context.test.ts +110 -0
  438. package/src/shared/web-context.ts +76 -0
  439. package/src/shared/workflow-types.ts +51 -0
  440. package/src/shared/workspace-types.ts +100 -0
@@ -0,0 +1,397 @@
1
+ export const KANNA_MCP_SERVER_NAME = "tinkaria-mcp"
2
+
3
+ import type {
4
+ AskUserQuestionItem,
5
+ AskUserQuestionAnswerMap,
6
+ AskUserQuestionToolResult,
7
+ ExitPlanModeToolResult,
8
+ HydratedToolCall,
9
+ ImageContentBlock,
10
+ NormalizedToolCall,
11
+ PresentContentInput,
12
+ PresentContentErrorToolResult,
13
+ PresentContentSchemaValidationError,
14
+ PresentContentValidationIssue,
15
+ PresentContentToolResult,
16
+ ReadFileImageResult,
17
+ ReadFileToolResult,
18
+ TodoItem,
19
+ } from "./types"
20
+
21
+ function asRecord(value: unknown): Record<string, unknown> | null {
22
+ if (!value || typeof value !== "object" || Array.isArray(value)) return null
23
+ return value as Record<string, unknown>
24
+ }
25
+
26
+ function isPresentContentKind(value: unknown): value is PresentContentInput["kind"] {
27
+ return value === "markdown" || value === "code" || value === "diagram"
28
+ }
29
+
30
+ function isPresentContentValidationIssue(value: unknown): value is PresentContentValidationIssue {
31
+ return Boolean(value)
32
+ && typeof value === "object"
33
+ && Array.isArray((value as { path?: unknown }).path)
34
+ && ((value as { path?: unknown[] }).path?.every((segment) => typeof segment === "string") ?? false)
35
+ && typeof (value as { code?: unknown }).code === "string"
36
+ && typeof (value as { message?: unknown }).message === "string"
37
+ }
38
+
39
+ function isPresentContentSchemaValidationError(value: unknown): value is PresentContentSchemaValidationError {
40
+ return Boolean(value)
41
+ && typeof value === "object"
42
+ && (value as { source?: unknown }).source === "schema_validation"
43
+ && (value as { schema?: unknown }).schema === "present_content"
44
+ && Array.isArray((value as { issues?: unknown }).issues)
45
+ && ((value as { issues?: unknown[] }).issues?.every(isPresentContentValidationIssue) ?? false)
46
+ }
47
+
48
+ export function normalizeToolCall(args: {
49
+ toolName: string
50
+ toolId: string
51
+ input: Record<string, unknown>
52
+ }): NormalizedToolCall {
53
+ const { toolName, toolId, input } = args
54
+
55
+ switch (toolName) {
56
+ case "AskUserQuestion":
57
+ return {
58
+ kind: "tool",
59
+ toolKind: "ask_user_question",
60
+ toolName,
61
+ toolId,
62
+ input: {
63
+ questions: Array.isArray(input.questions) ? (input.questions as AskUserQuestionItem[]) : [],
64
+ },
65
+ rawInput: input,
66
+ }
67
+ case "present_content":
68
+ return {
69
+ kind: "tool",
70
+ toolKind: "present_content",
71
+ toolName,
72
+ toolId,
73
+ input: {
74
+ title: typeof input.title === "string" ? input.title : "",
75
+ kind: isPresentContentKind(input.kind) ? input.kind : "markdown",
76
+ format: typeof input.format === "string" ? input.format : "text",
77
+ source: typeof input.source === "string" ? input.source : "",
78
+ summary: typeof input.summary === "string" ? input.summary : undefined,
79
+ collapsed: typeof input.collapsed === "boolean" ? input.collapsed : undefined,
80
+ },
81
+ rawInput: input,
82
+ }
83
+ case "ExitPlanMode":
84
+ return {
85
+ kind: "tool",
86
+ toolKind: "exit_plan_mode",
87
+ toolName,
88
+ toolId,
89
+ input: {
90
+ plan: typeof input.plan === "string" ? input.plan : undefined,
91
+ summary: typeof input.summary === "string" ? input.summary : undefined,
92
+ },
93
+ rawInput: input,
94
+ }
95
+ case "TodoWrite":
96
+ return {
97
+ kind: "tool",
98
+ toolKind: "todo_write",
99
+ toolName,
100
+ toolId,
101
+ input: {
102
+ todos: Array.isArray(input.todos) ? (input.todos as TodoItem[]) : [],
103
+ },
104
+ rawInput: input,
105
+ }
106
+ case "Skill":
107
+ return {
108
+ kind: "tool",
109
+ toolKind: "skill",
110
+ toolName,
111
+ toolId,
112
+ input: {
113
+ skill: typeof input.skill === "string" ? input.skill : "",
114
+ },
115
+ rawInput: input,
116
+ }
117
+ case "Glob":
118
+ return {
119
+ kind: "tool",
120
+ toolKind: "glob",
121
+ toolName,
122
+ toolId,
123
+ input: {
124
+ pattern: typeof input.pattern === "string" ? input.pattern : "",
125
+ },
126
+ rawInput: input,
127
+ }
128
+ case "Grep":
129
+ return {
130
+ kind: "tool",
131
+ toolKind: "grep",
132
+ toolName,
133
+ toolId,
134
+ input: {
135
+ pattern: typeof input.pattern === "string" ? input.pattern : "",
136
+ outputMode: typeof input.output_mode === "string" ? input.output_mode : undefined,
137
+ },
138
+ rawInput: input,
139
+ }
140
+ case "Bash":
141
+ return {
142
+ kind: "tool",
143
+ toolKind: "bash",
144
+ toolName,
145
+ toolId,
146
+ input: {
147
+ command: typeof input.command === "string" ? input.command : "",
148
+ description: typeof input.description === "string" ? input.description : undefined,
149
+ timeoutMs: typeof input.timeout === "number" ? input.timeout : undefined,
150
+ runInBackground: Boolean(input.run_in_background),
151
+ },
152
+ rawInput: input,
153
+ }
154
+ case "WebSearch":
155
+ return {
156
+ kind: "tool",
157
+ toolKind: "web_search",
158
+ toolName,
159
+ toolId,
160
+ input: {
161
+ query: typeof input.query === "string" ? input.query : "",
162
+ },
163
+ rawInput: input,
164
+ }
165
+ case "Read":
166
+ return {
167
+ kind: "tool",
168
+ toolKind: "read_file",
169
+ toolName,
170
+ toolId,
171
+ input: {
172
+ filePath: typeof input.file_path === "string" ? input.file_path : "",
173
+ },
174
+ rawInput: input,
175
+ }
176
+ case "Write":
177
+ return {
178
+ kind: "tool",
179
+ toolKind: "write_file",
180
+ toolName,
181
+ toolId,
182
+ input: {
183
+ filePath: typeof input.file_path === "string" ? input.file_path : "",
184
+ content: typeof input.content === "string" ? input.content : "",
185
+ },
186
+ rawInput: input,
187
+ }
188
+ case "Edit":
189
+ return {
190
+ kind: "tool",
191
+ toolKind: "edit_file",
192
+ toolName,
193
+ toolId,
194
+ input: {
195
+ filePath: typeof input.file_path === "string" ? input.file_path : "",
196
+ oldString: typeof input.old_string === "string" ? input.old_string : "",
197
+ newString: typeof input.new_string === "string" ? input.new_string : "",
198
+ },
199
+ rawInput: input,
200
+ }
201
+ }
202
+
203
+ const mcpMatch = toolName.match(/^mcp__(.+?)__(.+)$/)
204
+ if (mcpMatch) {
205
+ return {
206
+ kind: "tool",
207
+ toolKind: "mcp_generic",
208
+ toolName,
209
+ toolId,
210
+ input: {
211
+ server: mcpMatch[1],
212
+ tool: mcpMatch[2],
213
+ payload: input,
214
+ },
215
+ rawInput: input,
216
+ }
217
+ }
218
+
219
+ if (
220
+ toolName === "spawn_agent"
221
+ || toolName === "list_agents"
222
+ || toolName === "send_input"
223
+ || toolName === "wait_agent"
224
+ || toolName === "close_agent"
225
+ ) {
226
+ return {
227
+ kind: "tool",
228
+ toolKind: "mcp_generic",
229
+ toolName,
230
+ toolId,
231
+ input: {
232
+ server: "session-orchestration",
233
+ tool: toolName,
234
+ payload: input,
235
+ },
236
+ rawInput: input,
237
+ }
238
+ }
239
+
240
+ if (typeof input.subagent_type === "string") {
241
+ return {
242
+ kind: "tool",
243
+ toolKind: "subagent_task",
244
+ toolName,
245
+ toolId,
246
+ input: {
247
+ subagentType: input.subagent_type,
248
+ },
249
+ rawInput: input,
250
+ }
251
+ }
252
+
253
+ return {
254
+ kind: "tool",
255
+ toolKind: "unknown_tool",
256
+ toolName,
257
+ toolId,
258
+ input: {
259
+ payload: input,
260
+ },
261
+ rawInput: input,
262
+ }
263
+ }
264
+
265
+ const MAX_IMAGE_BASE64_LENGTH = 10 * 1024 * 1024 // ~7.5MB decoded
266
+
267
+ const ALLOWED_IMAGE_MEDIA_TYPES = new Set([
268
+ "image/png", "image/jpeg", "image/gif", "image/webp", "image/svg+xml", "image/bmp",
269
+ ])
270
+
271
+ function extractImageBlocks(parsed: unknown): ReadFileImageResult | null {
272
+ if (!Array.isArray(parsed)) return null
273
+
274
+ const images: ImageContentBlock[] = []
275
+ let text: string | undefined
276
+
277
+ for (const block of parsed) {
278
+ const record = asRecord(block)
279
+ if (!record) continue
280
+
281
+ if (record.type === "image") {
282
+ const source = asRecord(record.source)
283
+ if (
284
+ source &&
285
+ source.type === "base64" &&
286
+ typeof source.media_type === "string" &&
287
+ ALLOWED_IMAGE_MEDIA_TYPES.has(source.media_type) &&
288
+ typeof source.data === "string" &&
289
+ source.data.length <= MAX_IMAGE_BASE64_LENGTH
290
+ ) {
291
+ images.push({ mediaType: source.media_type, data: source.data })
292
+ }
293
+ }
294
+
295
+ if (record.type === "text" && typeof record.text === "string") {
296
+ text = text ? `${text}\n${record.text}` : record.text
297
+ }
298
+ }
299
+
300
+ if (images.length === 0) return null
301
+ return { images, text }
302
+ }
303
+
304
+ function parseJsonValue(value: unknown): unknown {
305
+ if (typeof value !== "string") return value
306
+ try {
307
+ return JSON.parse(value)
308
+ } catch (error: unknown) {
309
+ void error
310
+ return value
311
+ }
312
+ }
313
+
314
+ export function hydrateToolResult(tool: NormalizedToolCall, raw: unknown): HydratedToolCall["result"] {
315
+ const parsed = parseJsonValue(raw)
316
+
317
+ switch (tool.toolKind) {
318
+ case "ask_user_question": {
319
+ const record = asRecord(parsed)
320
+ const answers = asRecord(record?.answers) ?? (record ? record : {})
321
+ return {
322
+ answers: Object.fromEntries(
323
+ Object.entries(answers).map(([key, value]) => {
324
+ if (Array.isArray(value)) {
325
+ return [key, value.map((entry) => String(entry))]
326
+ }
327
+ if (value && typeof value === "object" && Array.isArray((value as { answers?: unknown }).answers)) {
328
+ return [key, (value as { answers: unknown[] }).answers.map((entry) => String(entry))]
329
+ }
330
+ if (value == null || value === "") {
331
+ return [key, []]
332
+ }
333
+ return [key, [String(value)]]
334
+ })
335
+ ) as AskUserQuestionAnswerMap,
336
+ ...(record?.discarded === true ? { discarded: true } : {}),
337
+ } satisfies AskUserQuestionToolResult
338
+ }
339
+ case "exit_plan_mode": {
340
+ const record = asRecord(parsed)
341
+ return {
342
+ confirmed: typeof record?.confirmed === "boolean" ? record.confirmed : undefined,
343
+ clearContext: typeof record?.clearContext === "boolean" ? record.clearContext : undefined,
344
+ message: typeof record?.message === "string" ? record.message : undefined,
345
+ ...(record?.discarded === true ? { discarded: true } : {}),
346
+ } satisfies ExitPlanModeToolResult
347
+ }
348
+ case "read_file": {
349
+ if (typeof parsed === "string") {
350
+ return parsed
351
+ }
352
+ const imageResult = extractImageBlocks(parsed)
353
+ if (imageResult) return imageResult
354
+ const record = asRecord(parsed)
355
+ return {
356
+ content: typeof record?.content === "string" ? record.content : JSON.stringify(parsed, null, 2),
357
+ } satisfies ReadFileToolResult
358
+ }
359
+ case "present_content": {
360
+ const record = asRecord(parsed)
361
+ if (isPresentContentSchemaValidationError(record?.error)) {
362
+ return {
363
+ error: record.error,
364
+ } satisfies PresentContentErrorToolResult
365
+ }
366
+ if (typeof record?.error === "string") {
367
+ return {
368
+ error: {
369
+ source: "schema_validation",
370
+ schema: "present_content",
371
+ issues: [
372
+ {
373
+ path: [],
374
+ code: "custom",
375
+ message: record.error,
376
+ },
377
+ ],
378
+ },
379
+ } satisfies PresentContentErrorToolResult
380
+ }
381
+ return {
382
+ accepted: true,
383
+ title: typeof record?.title === "string" ? record.title : tool.input.title,
384
+ kind: isPresentContentKind(record?.kind) ? record.kind : tool.input.kind,
385
+ format: typeof record?.format === "string" ? record.format : tool.input.format,
386
+ source: typeof record?.source === "string" ? record.source : tool.input.source,
387
+ summary: typeof record?.summary === "string" ? record.summary : tool.input.summary,
388
+ collapsed: typeof record?.collapsed === "boolean" ? record.collapsed : tool.input.collapsed,
389
+ } satisfies PresentContentToolResult
390
+ }
391
+ default: {
392
+ const imageResult = extractImageBlocks(parsed)
393
+ if (imageResult) return imageResult
394
+ return parsed
395
+ }
396
+ }
397
+ }
@@ -0,0 +1,27 @@
1
+ import type { NormalizedToolCall, TranscriptEntry } from "./types"
2
+
3
+ export function timestamped<T extends Omit<TranscriptEntry, "_id" | "createdAt">>(
4
+ entry: T,
5
+ createdAt = Date.now()
6
+ ): TranscriptEntry {
7
+ return {
8
+ _id: crypto.randomUUID(),
9
+ createdAt,
10
+ ...entry,
11
+ } as TranscriptEntry
12
+ }
13
+
14
+ export function discardedToolResult(
15
+ tool: NormalizedToolCall & { toolKind: "ask_user_question" | "exit_plan_mode" }
16
+ ) {
17
+ if (tool.toolKind === "ask_user_question") {
18
+ return {
19
+ discarded: true,
20
+ answers: {},
21
+ }
22
+ }
23
+
24
+ return {
25
+ discarded: true,
26
+ }
27
+ }
@@ -0,0 +1,225 @@
1
+ import { describe, expect, test } from "bun:test"
2
+ import type { NormalizedToolCall, TranscriptEntry } from "./types"
3
+ import {
4
+ foldTranscriptRenderUnits,
5
+ getTranscriptRenderUnitId,
6
+ } from "./transcript-render"
7
+
8
+ let sequence = 0
9
+
10
+ function entry<T extends Omit<TranscriptEntry, "_id" | "createdAt">>(partial: T): TranscriptEntry {
11
+ sequence += 1
12
+ return {
13
+ _id: `e${sequence}`,
14
+ createdAt: sequence,
15
+ ...partial,
16
+ } as TranscriptEntry
17
+ }
18
+
19
+ function text(value: string): TranscriptEntry {
20
+ return entry({ kind: "assistant_text", text: value })
21
+ }
22
+
23
+ function user(content = "Hello"): TranscriptEntry {
24
+ return entry({ kind: "user_prompt", content })
25
+ }
26
+
27
+ function result(value = "Done"): TranscriptEntry {
28
+ return entry({
29
+ kind: "result",
30
+ subtype: "success",
31
+ isError: false,
32
+ durationMs: 100,
33
+ result: value,
34
+ })
35
+ }
36
+
37
+ function status(value: string): TranscriptEntry {
38
+ return entry({ kind: "status", status: value })
39
+ }
40
+
41
+ function toolCall(tool: NormalizedToolCall): TranscriptEntry {
42
+ return entry({ kind: "tool_call", tool })
43
+ }
44
+
45
+ function toolResult(toolId: string, content: unknown, isError = false): TranscriptEntry {
46
+ return entry({ kind: "tool_result", toolId, content, isError })
47
+ }
48
+
49
+ function bash(toolId: string, command = "echo hi"): TranscriptEntry {
50
+ return toolCall({
51
+ kind: "tool",
52
+ toolKind: "bash",
53
+ toolName: "Bash",
54
+ toolId,
55
+ input: { command },
56
+ })
57
+ }
58
+
59
+ function erroredSkill(toolId: string): TranscriptEntry[] {
60
+ return [
61
+ toolCall({
62
+ kind: "tool",
63
+ toolKind: "skill",
64
+ toolName: "Skill",
65
+ toolId,
66
+ input: { skill: "missing" },
67
+ }),
68
+ toolResult(toolId, "Skill not found: missing", true),
69
+ ]
70
+ }
71
+
72
+ function askUser(toolId: string): TranscriptEntry {
73
+ return toolCall({
74
+ kind: "tool",
75
+ toolKind: "ask_user_question",
76
+ toolName: "AskUserQuestion",
77
+ toolId,
78
+ input: { questions: [{ question: "Pick one" }] },
79
+ })
80
+ }
81
+
82
+ function todoWrite(toolId: string, content: string): TranscriptEntry {
83
+ return toolCall({
84
+ kind: "tool",
85
+ toolKind: "todo_write",
86
+ toolName: "TodoWrite",
87
+ toolId,
88
+ input: { todos: [{ content, status: "pending", activeForm: content }] },
89
+ })
90
+ }
91
+
92
+ function presentContent(toolId: string): TranscriptEntry {
93
+ return toolCall({
94
+ kind: "tool",
95
+ toolKind: "present_content",
96
+ toolName: "present_content",
97
+ toolId,
98
+ input: {
99
+ title: "Artifact",
100
+ kind: "markdown",
101
+ format: "markdown",
102
+ source: "Hello",
103
+ },
104
+ })
105
+ }
106
+
107
+ function unknownTool(toolId: string): TranscriptEntry {
108
+ return toolCall({
109
+ kind: "tool",
110
+ toolKind: "unknown_tool",
111
+ toolName: "FutureTool",
112
+ toolId,
113
+ input: { payload: { hello: "world" } },
114
+ })
115
+ }
116
+
117
+ describe("getTranscriptRenderUnitId", () => {
118
+ test("uses deterministic prefixed ids", () => {
119
+ expect(getTranscriptRenderUnitId("assistant_response", ["e1"])).toBe("assistant_response:e1")
120
+ expect(getTranscriptRenderUnitId("wip_block", ["e1", "e2", "e3"])).toBe("wip:e1:e3")
121
+ expect(getTranscriptRenderUnitId("tool_group", ["e2", "e3"])).toBe("tools:e2:e3")
122
+ expect(getTranscriptRenderUnitId("artifact", ["e4"])).toBe("artifact:e4")
123
+ expect(getTranscriptRenderUnitId("unknown", ["e5"])).toBe("unknown:e5")
124
+ })
125
+ })
126
+
127
+ describe("foldTranscriptRenderUnits", () => {
128
+ test("returns an empty render window for an empty transcript", () => {
129
+ expect(foldTranscriptRenderUnits([])).toEqual([])
130
+ })
131
+
132
+ test("folds narration plus work tool into WIP and final text into response", () => {
133
+ sequence = 0
134
+ const units = foldTranscriptRenderUnits([text("Checking"), bash("t1"), text("Fixed")])
135
+ expect(units.map((unit) => unit.kind)).toEqual(["wip_block", "assistant_response"])
136
+ expect(units[0]?.id).toBe("wip:e1:e2")
137
+ expect(units[1]?.id).toBe("assistant_response:e3")
138
+ })
139
+
140
+ test("ejects rationale text before interactive question", () => {
141
+ sequence = 0
142
+ const units = foldTranscriptRenderUnits([
143
+ text("Checking"),
144
+ bash("t1"),
145
+ text("Pick one:"),
146
+ askUser("q1"),
147
+ ], { isLoading: true })
148
+ expect(units.map((unit) => unit.kind)).toEqual(["wip_block", "assistant_response", "standalone_tool"])
149
+ expect(units[1]?.sourceEntryIds).toEqual(["e3"])
150
+ })
151
+
152
+ test("keeps only the latest TodoWrite projection visible", () => {
153
+ sequence = 0
154
+ const units = foldTranscriptRenderUnits([
155
+ todoWrite("todo-1", "old"),
156
+ text("Working"),
157
+ todoWrite("todo-2", "new"),
158
+ ], { isLoading: true })
159
+ expect(units.map((unit) => unit.kind)).toEqual(["wip_block", "standalone_tool"])
160
+ expect(units[1]?.sourceEntryIds).toEqual(["e3"])
161
+ })
162
+
163
+ test("renders present_content as an artifact unit, not a tool", () => {
164
+ sequence = 0
165
+ const units = foldTranscriptRenderUnits([presentContent("p1")])
166
+ expect(units.map((unit) => unit.kind)).toEqual(["artifact"])
167
+ expect(units[0]?.id).toBe("artifact:e1")
168
+ })
169
+
170
+ test("keeps errored tools standalone and visible", () => {
171
+ sequence = 0
172
+ const units = foldTranscriptRenderUnits([text("Trying skill"), ...erroredSkill("s1"), text("Failed")])
173
+ expect(units.map((unit) => unit.kind)).toEqual(["assistant_response", "standalone_tool", "assistant_response"])
174
+ expect(units[1]?.sourceEntryIds).toEqual(["e2", "e3"])
175
+ })
176
+
177
+ test("deduplicates system_init and account_info by first visible entry", () => {
178
+ sequence = 0
179
+ const units = foldTranscriptRenderUnits([
180
+ entry({ kind: "system_init", provider: "claude", model: "sonnet", tools: [], agents: [], slashCommands: [], mcpServers: [] }),
181
+ entry({ kind: "system_init", provider: "claude", model: "sonnet", tools: [], agents: [], slashCommands: [], mcpServers: [] }),
182
+ entry({ kind: "account_info", accountInfo: { email: "a@example.com" } }),
183
+ entry({ kind: "account_info", accountInfo: { email: "b@example.com" } }),
184
+ ])
185
+ expect(units.map((unit) => unit.kind)).toEqual(["system_init", "account_info"])
186
+ expect(units.map((unit) => unit.sourceEntryIds[0])).toEqual(["e1", "e3"])
187
+ })
188
+
189
+ test("renders only latest status and hides result next to context_cleared", () => {
190
+ sequence = 0
191
+ const units = foldTranscriptRenderUnits([
192
+ status("running"),
193
+ status("waiting_for_user"),
194
+ result("Session compacted"),
195
+ entry({ kind: "context_cleared" }),
196
+ ], { isLoading: true })
197
+ expect(units.map((unit) => unit.kind)).toEqual(["context_cleared", "status"])
198
+ expect(units[1]?.sourceEntryIds).toEqual(["e2"])
199
+ })
200
+
201
+ test("unknown tools render standalone instead of hiding", () => {
202
+ sequence = 0
203
+ const units = foldTranscriptRenderUnits([unknownTool("u1")])
204
+ expect(units.map((unit) => unit.kind)).toEqual(["standalone_tool"])
205
+ })
206
+
207
+ test("does not crash on pending tools or interrupted streams", () => {
208
+ sequence = 0
209
+ const units = foldTranscriptRenderUnits([text("Working"), bash("t1"), entry({ kind: "interrupted" })], { isLoading: true })
210
+ expect(units.map((unit) => unit.kind)).toEqual(["wip_block", "interrupted"])
211
+ })
212
+
213
+ test("does not group across consecutive user prompts", () => {
214
+ sequence = 0
215
+ const units = foldTranscriptRenderUnits([user("one"), user("two"), text("Answer")])
216
+ expect(units.map((unit) => unit.kind)).toEqual(["user_prompt", "user_prompt", "assistant_response"])
217
+ })
218
+
219
+ test("groups consecutive work tools outside assistant WIP context as a tool group", () => {
220
+ sequence = 0
221
+ const units = foldTranscriptRenderUnits([bash("t1"), bash("t2")])
222
+ expect(units.map((unit) => unit.kind)).toEqual(["tool_group"])
223
+ expect(units[0]?.id).toBe("tools:e1:e2")
224
+ })
225
+ })