orynacode-ai 1.16.2

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 (938) hide show
  1. package/AGENTS.md +131 -0
  2. package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
  3. package/Dockerfile +18 -0
  4. package/README.md +15 -0
  5. package/bin/orynacode +199 -0
  6. package/bunfig.toml +7 -0
  7. package/git +0 -0
  8. package/migration/20260511173437_session-metadata/migration.sql +1 -0
  9. package/migration/20260511173437_session-metadata/snapshot.json +1500 -0
  10. package/package.json +155 -0
  11. package/parsers-config.ts +386 -0
  12. package/script/bench-search.ts +115 -0
  13. package/script/bench-test-suite.ts +52 -0
  14. package/script/build.ts +244 -0
  15. package/script/generate.ts +14 -0
  16. package/script/httpapi-exercise.ts +1 -0
  17. package/script/postinstall.mjs +189 -0
  18. package/script/profile-test-files.ts +42 -0
  19. package/script/publish.ts +213 -0
  20. package/script/run-workspace-server +106 -0
  21. package/script/schema.ts +77 -0
  22. package/script/time.ts +6 -0
  23. package/script/trace-imports.ts +153 -0
  24. package/specs/effect/error-boundaries-plan.md +235 -0
  25. package/specs/effect/errors.md +207 -0
  26. package/specs/effect/facades.md +218 -0
  27. package/specs/effect/guide.md +247 -0
  28. package/specs/effect/instance-context.md +13 -0
  29. package/specs/effect/loose-ends.md +30 -0
  30. package/specs/effect/migration.md +62 -0
  31. package/specs/effect/routes.md +61 -0
  32. package/specs/effect/schema.md +88 -0
  33. package/specs/effect/server-package.md +58 -0
  34. package/specs/effect/todo.md +241 -0
  35. package/specs/effect/tools.md +88 -0
  36. package/specs/openapi-translation-cleanup.md +204 -0
  37. package/specs/tui-plugins.md +544 -0
  38. package/specs/v2/api.ts +67 -0
  39. package/specs/v2/message-shape.md +136 -0
  40. package/specs/v2/notifications.md +13 -0
  41. package/specs/v2/tui-command-shim.md +67 -0
  42. package/src/account/account.ts +459 -0
  43. package/src/account/repo.ts +170 -0
  44. package/src/account/schema.ts +99 -0
  45. package/src/account/url.ts +8 -0
  46. package/src/acp/agent.ts +95 -0
  47. package/src/acp/config-option.ts +203 -0
  48. package/src/acp/content.ts +250 -0
  49. package/src/acp/directory.ts +210 -0
  50. package/src/acp/error.ts +90 -0
  51. package/src/acp/event.ts +344 -0
  52. package/src/acp/permission.ts +145 -0
  53. package/src/acp/profile.ts +42 -0
  54. package/src/acp/service.ts +1062 -0
  55. package/src/acp/session.ts +231 -0
  56. package/src/acp/tool.ts +317 -0
  57. package/src/acp/usage.ts +239 -0
  58. package/src/agent/agent.ts +433 -0
  59. package/src/agent/generate.txt +75 -0
  60. package/src/agent/prompt/compaction.txt +9 -0
  61. package/src/agent/prompt/explore.txt +18 -0
  62. package/src/agent/prompt/summary.txt +11 -0
  63. package/src/agent/prompt/title.txt +44 -0
  64. package/src/agent/subagent-permissions.ts +35 -0
  65. package/src/audio.d.ts +14 -0
  66. package/src/auth/index.ts +96 -0
  67. package/src/background/job.ts +36 -0
  68. package/src/bus/global.ts +22 -0
  69. package/src/cli/bootstrap.ts +11 -0
  70. package/src/cli/cmd/account.ts +264 -0
  71. package/src/cli/cmd/acp.ts +76 -0
  72. package/src/cli/cmd/agent.ts +259 -0
  73. package/src/cli/cmd/cmd.ts +7 -0
  74. package/src/cli/cmd/db.ts +62 -0
  75. package/src/cli/cmd/debug/agent.handler.ts +193 -0
  76. package/src/cli/cmd/debug/agent.ts +27 -0
  77. package/src/cli/cmd/debug/config.ts +14 -0
  78. package/src/cli/cmd/debug/file.ts +87 -0
  79. package/src/cli/cmd/debug/index.ts +87 -0
  80. package/src/cli/cmd/debug/lsp.ts +51 -0
  81. package/src/cli/cmd/debug/ripgrep.ts +99 -0
  82. package/src/cli/cmd/debug/scrap.ts +18 -0
  83. package/src/cli/cmd/debug/skill.ts +15 -0
  84. package/src/cli/cmd/debug/snapshot.ts +50 -0
  85. package/src/cli/cmd/debug/startup.ts +11 -0
  86. package/src/cli/cmd/debug/v2.ts +46 -0
  87. package/src/cli/cmd/export.ts +292 -0
  88. package/src/cli/cmd/generate.ts +54 -0
  89. package/src/cli/cmd/github.handler.ts +1593 -0
  90. package/src/cli/cmd/github.shared.ts +30 -0
  91. package/src/cli/cmd/github.ts +42 -0
  92. package/src/cli/cmd/import.ts +224 -0
  93. package/src/cli/cmd/mcp.ts +846 -0
  94. package/src/cli/cmd/models.ts +66 -0
  95. package/src/cli/cmd/plug.ts +230 -0
  96. package/src/cli/cmd/pr.ts +115 -0
  97. package/src/cli/cmd/prompt-display.ts +48 -0
  98. package/src/cli/cmd/providers.ts +506 -0
  99. package/src/cli/cmd/run/demo.ts +1274 -0
  100. package/src/cli/cmd/run/entry.body.ts +194 -0
  101. package/src/cli/cmd/run/footer.command.tsx +899 -0
  102. package/src/cli/cmd/run/footer.menu.tsx +306 -0
  103. package/src/cli/cmd/run/footer.permission.tsx +475 -0
  104. package/src/cli/cmd/run/footer.prompt.tsx +1207 -0
  105. package/src/cli/cmd/run/footer.question.tsx +579 -0
  106. package/src/cli/cmd/run/footer.subagent.tsx +171 -0
  107. package/src/cli/cmd/run/footer.ts +1092 -0
  108. package/src/cli/cmd/run/footer.view.tsx +935 -0
  109. package/src/cli/cmd/run/otel.ts +117 -0
  110. package/src/cli/cmd/run/permission.shared.ts +256 -0
  111. package/src/cli/cmd/run/prompt.shared.ts +147 -0
  112. package/src/cli/cmd/run/question.shared.ts +340 -0
  113. package/src/cli/cmd/run/runtime.boot.ts +210 -0
  114. package/src/cli/cmd/run/runtime.lifecycle.ts +369 -0
  115. package/src/cli/cmd/run/runtime.queue.ts +347 -0
  116. package/src/cli/cmd/run/runtime.shared.ts +17 -0
  117. package/src/cli/cmd/run/runtime.stdin.ts +37 -0
  118. package/src/cli/cmd/run/runtime.ts +879 -0
  119. package/src/cli/cmd/run/scrollback.shared.ts +92 -0
  120. package/src/cli/cmd/run/scrollback.surface.ts +435 -0
  121. package/src/cli/cmd/run/scrollback.writer.tsx +335 -0
  122. package/src/cli/cmd/run/session-data.ts +1113 -0
  123. package/src/cli/cmd/run/session-replay.ts +301 -0
  124. package/src/cli/cmd/run/session.shared.ts +196 -0
  125. package/src/cli/cmd/run/splash.ts +310 -0
  126. package/src/cli/cmd/run/stream.transport.ts +1465 -0
  127. package/src/cli/cmd/run/stream.ts +175 -0
  128. package/src/cli/cmd/run/subagent-data.ts +844 -0
  129. package/src/cli/cmd/run/theme.ts +603 -0
  130. package/src/cli/cmd/run/tool.ts +1489 -0
  131. package/src/cli/cmd/run/trace.ts +94 -0
  132. package/src/cli/cmd/run/types.ts +342 -0
  133. package/src/cli/cmd/run/variant.shared.ts +215 -0
  134. package/src/cli/cmd/run.ts +879 -0
  135. package/src/cli/cmd/serve.ts +24 -0
  136. package/src/cli/cmd/session.ts +147 -0
  137. package/src/cli/cmd/stats.ts +393 -0
  138. package/src/cli/cmd/tui/app.tsx +1113 -0
  139. package/src/cli/cmd/tui/attach.ts +103 -0
  140. package/src/cli/cmd/tui/attention.ts +262 -0
  141. package/src/cli/cmd/tui/component/bg-pulse-render.ts +436 -0
  142. package/src/cli/cmd/tui/component/bg-pulse.tsx +99 -0
  143. package/src/cli/cmd/tui/component/border.tsx +21 -0
  144. package/src/cli/cmd/tui/component/command-palette.tsx +79 -0
  145. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  146. package/src/cli/cmd/tui/component/dialog-console-org.tsx +103 -0
  147. package/src/cli/cmd/tui/component/dialog-mcp.tsx +85 -0
  148. package/src/cli/cmd/tui/component/dialog-model.tsx +185 -0
  149. package/src/cli/cmd/tui/component/dialog-move-session.tsx +240 -0
  150. package/src/cli/cmd/tui/component/dialog-provider.tsx +687 -0
  151. package/src/cli/cmd/tui/component/dialog-retry-action.tsx +160 -0
  152. package/src/cli/cmd/tui/component/dialog-session-delete-failed.tsx +99 -0
  153. package/src/cli/cmd/tui/component/dialog-session-list.tsx +318 -0
  154. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  155. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  156. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  157. package/src/cli/cmd/tui/component/dialog-status.tsx +168 -0
  158. package/src/cli/cmd/tui/component/dialog-tag.tsx +47 -0
  159. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  160. package/src/cli/cmd/tui/component/dialog-variant.tsx +39 -0
  161. package/src/cli/cmd/tui/component/dialog-workspace-create.tsx +313 -0
  162. package/src/cli/cmd/tui/component/dialog-workspace-file-changes.tsx +144 -0
  163. package/src/cli/cmd/tui/component/dialog-workspace-list.tsx +112 -0
  164. package/src/cli/cmd/tui/component/dialog-workspace-unavailable.tsx +69 -0
  165. package/src/cli/cmd/tui/component/error-component.tsx +81 -0
  166. package/src/cli/cmd/tui/component/logo.tsx +885 -0
  167. package/src/cli/cmd/tui/component/plugin-route-missing.tsx +14 -0
  168. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +799 -0
  169. package/src/cli/cmd/tui/component/prompt/cwd.ts +0 -0
  170. package/src/cli/cmd/tui/component/prompt/frecency.tsx +90 -0
  171. package/src/cli/cmd/tui/component/prompt/history.tsx +117 -0
  172. package/src/cli/cmd/tui/component/prompt/index.tsx +1725 -0
  173. package/src/cli/cmd/tui/component/prompt/move.tsx +192 -0
  174. package/src/cli/cmd/tui/component/prompt/part.ts +31 -0
  175. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  176. package/src/cli/cmd/tui/component/prompt/traits.ts +35 -0
  177. package/src/cli/cmd/tui/component/prompt/workspace.tsx +137 -0
  178. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  179. package/src/cli/cmd/tui/component/startup-loading.tsx +63 -0
  180. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  181. package/src/cli/cmd/tui/component/use-connected.tsx +9 -0
  182. package/src/cli/cmd/tui/component/workspace-label.tsx +19 -0
  183. package/src/cli/cmd/tui/config/cwd.ts +5 -0
  184. package/src/cli/cmd/tui/config/keybind.ts +467 -0
  185. package/src/cli/cmd/tui/config/tui-migrate.ts +154 -0
  186. package/src/cli/cmd/tui/config/tui-schema.ts +88 -0
  187. package/src/cli/cmd/tui/config/tui.ts +308 -0
  188. package/src/cli/cmd/tui/context/agent.tsx +11 -0
  189. package/src/cli/cmd/tui/context/aggregate-failures.ts +51 -0
  190. package/src/cli/cmd/tui/context/args.tsx +15 -0
  191. package/src/cli/cmd/tui/context/directory.ts +15 -0
  192. package/src/cli/cmd/tui/context/editor-zed.ts +287 -0
  193. package/src/cli/cmd/tui/context/editor.ts +469 -0
  194. package/src/cli/cmd/tui/context/event.ts +38 -0
  195. package/src/cli/cmd/tui/context/exit.tsx +42 -0
  196. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  197. package/src/cli/cmd/tui/context/kv.tsx +76 -0
  198. package/src/cli/cmd/tui/context/local.tsx +510 -0
  199. package/src/cli/cmd/tui/context/path-format.tsx +39 -0
  200. package/src/cli/cmd/tui/context/project.tsx +111 -0
  201. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  202. package/src/cli/cmd/tui/context/route.tsx +52 -0
  203. package/src/cli/cmd/tui/context/sdk.tsx +142 -0
  204. package/src/cli/cmd/tui/context/sync-v2.tsx +447 -0
  205. package/src/cli/cmd/tui/context/sync.tsx +628 -0
  206. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  207. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  208. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  209. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +230 -0
  210. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +230 -0
  211. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  212. package/src/cli/cmd/tui/context/theme/cobalt2.json +225 -0
  213. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  214. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  215. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  216. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  217. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  218. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  219. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  220. package/src/cli/cmd/tui/context/theme/lucent-orng.json +234 -0
  221. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  222. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  223. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  224. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  225. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  226. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  227. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  228. package/src/cli/cmd/tui/context/theme/opencode.json +245 -0
  229. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  230. package/src/cli/cmd/tui/context/theme/oryna.json +95 -0
  231. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  232. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  233. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  234. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  235. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  236. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  237. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  238. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  239. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  240. package/src/cli/cmd/tui/context/theme.tsx +1341 -0
  241. package/src/cli/cmd/tui/context/thinking.ts +67 -0
  242. package/src/cli/cmd/tui/context/tui-config.tsx +9 -0
  243. package/src/cli/cmd/tui/event.ts +53 -0
  244. package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +98 -0
  245. package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +288 -0
  246. package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +59 -0
  247. package/src/cli/cmd/tui/feature-plugins/session/dialog.tsx +356 -0
  248. package/src/cli/cmd/tui/feature-plugins/session/index.tsx +32 -0
  249. package/src/cli/cmd/tui/feature-plugins/session/preview-pane.tsx +288 -0
  250. package/src/cli/cmd/tui/feature-plugins/session/util.tsx +54 -0
  251. package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +65 -0
  252. package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +70 -0
  253. package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +96 -0
  254. package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +65 -0
  255. package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +97 -0
  256. package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +49 -0
  257. package/src/cli/cmd/tui/feature-plugins/system/diff-viewer-file-tree-utils.ts +232 -0
  258. package/src/cli/cmd/tui/feature-plugins/system/diff-viewer-file-tree.tsx +162 -0
  259. package/src/cli/cmd/tui/feature-plugins/system/diff-viewer-ui.tsx +103 -0
  260. package/src/cli/cmd/tui/feature-plugins/system/diff-viewer.tsx +1058 -0
  261. package/src/cli/cmd/tui/feature-plugins/system/notifications.ts +94 -0
  262. package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +269 -0
  263. package/src/cli/cmd/tui/feature-plugins/system/session-v2.tsx +1184 -0
  264. package/src/cli/cmd/tui/feature-plugins/system/which-key.tsx +608 -0
  265. package/src/cli/cmd/tui/keymap.tsx +283 -0
  266. package/src/cli/cmd/tui/layer.ts +6 -0
  267. package/src/cli/cmd/tui/plugin/api.tsx +390 -0
  268. package/src/cli/cmd/tui/plugin/command-shim.ts +109 -0
  269. package/src/cli/cmd/tui/plugin/internal.ts +42 -0
  270. package/src/cli/cmd/tui/plugin/runtime.ts +1131 -0
  271. package/src/cli/cmd/tui/plugin/slots.tsx +60 -0
  272. package/src/cli/cmd/tui/routes/home/session-destination.tsx +39 -0
  273. package/src/cli/cmd/tui/routes/home.tsx +149 -0
  274. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +76 -0
  275. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +108 -0
  276. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  277. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  278. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  279. package/src/cli/cmd/tui/routes/session/index.tsx +2629 -0
  280. package/src/cli/cmd/tui/routes/session/permission.tsx +729 -0
  281. package/src/cli/cmd/tui/routes/session/question.tsx +514 -0
  282. package/src/cli/cmd/tui/routes/session/sidebar.tsx +102 -0
  283. package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +132 -0
  284. package/src/cli/cmd/tui/thread.ts +264 -0
  285. package/src/cli/cmd/tui/ui/dialog-alert.tsx +66 -0
  286. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +108 -0
  287. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +217 -0
  288. package/src/cli/cmd/tui/ui/dialog-help.tsx +40 -0
  289. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +126 -0
  290. package/src/cli/cmd/tui/ui/dialog-select.tsx +712 -0
  291. package/src/cli/cmd/tui/ui/dialog.tsx +218 -0
  292. package/src/cli/cmd/tui/ui/link.tsx +34 -0
  293. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  294. package/src/cli/cmd/tui/ui/toast.tsx +102 -0
  295. package/src/cli/cmd/tui/util/audio.ts +58 -0
  296. package/src/cli/cmd/tui/util/clipboard.ts +181 -0
  297. package/src/cli/cmd/tui/util/collapse-tool-output.ts +19 -0
  298. package/src/cli/cmd/tui/util/editor.ts +43 -0
  299. package/src/cli/cmd/tui/util/layout.ts +25 -0
  300. package/src/cli/cmd/tui/util/model.ts +23 -0
  301. package/src/cli/cmd/tui/util/provider-origin.ts +7 -0
  302. package/src/cli/cmd/tui/util/revert-diff.ts +18 -0
  303. package/src/cli/cmd/tui/util/scroll.ts +25 -0
  304. package/src/cli/cmd/tui/util/selection.ts +73 -0
  305. package/src/cli/cmd/tui/util/signal.ts +41 -0
  306. package/src/cli/cmd/tui/util/transcript.ts +112 -0
  307. package/src/cli/cmd/tui/validate-session.ts +29 -0
  308. package/src/cli/cmd/tui/win32.ts +130 -0
  309. package/src/cli/cmd/tui/worker.ts +99 -0
  310. package/src/cli/cmd/uninstall.ts +353 -0
  311. package/src/cli/cmd/upgrade.ts +74 -0
  312. package/src/cli/cmd/web.ts +84 -0
  313. package/src/cli/effect/prompt.ts +37 -0
  314. package/src/cli/effect-cmd.ts +96 -0
  315. package/src/cli/error.ts +118 -0
  316. package/src/cli/heap.ts +59 -0
  317. package/src/cli/logo.ts +21 -0
  318. package/src/cli/network.ts +64 -0
  319. package/src/cli/ui.ts +132 -0
  320. package/src/cli/upgrade.ts +53 -0
  321. package/src/command/index.ts +181 -0
  322. package/src/command/template/initialize.txt +66 -0
  323. package/src/command/template/review.txt +101 -0
  324. package/src/config/agent.ts +68 -0
  325. package/src/config/command.ts +45 -0
  326. package/src/config/config.ts +679 -0
  327. package/src/config/entry-name.ts +19 -0
  328. package/src/config/managed.ts +77 -0
  329. package/src/config/markdown.ts +36 -0
  330. package/src/config/parse.ts +79 -0
  331. package/src/config/paths.ts +45 -0
  332. package/src/config/plugin.ts +79 -0
  333. package/src/config/reference.ts +48 -0
  334. package/src/config/variable.ts +91 -0
  335. package/src/control-plane/adapters/index.ts +41 -0
  336. package/src/control-plane/adapters/worktree.ts +96 -0
  337. package/src/control-plane/dev/README.md +19 -0
  338. package/src/control-plane/dev/debug-workspace-plugin.ts +73 -0
  339. package/src/control-plane/types.ts +59 -0
  340. package/src/control-plane/util.ts +39 -0
  341. package/src/control-plane/workspace-adapter-runtime.ts +51 -0
  342. package/src/control-plane/workspace-context.ts +26 -0
  343. package/src/control-plane/workspace.ts +1075 -0
  344. package/src/effect/app-runtime.ts +133 -0
  345. package/src/effect/bootstrap-runtime.ts +23 -0
  346. package/src/effect/bridge.ts +84 -0
  347. package/src/effect/config-service.ts +67 -0
  348. package/src/effect/instance-ref.ts +11 -0
  349. package/src/effect/instance-registry.ts +12 -0
  350. package/src/effect/instance-state.ts +72 -0
  351. package/src/effect/promise.ts +17 -0
  352. package/src/effect/run-service.ts +47 -0
  353. package/src/effect/runner.ts +217 -0
  354. package/src/effect/runtime-flags.ts +76 -0
  355. package/src/env/index.ts +40 -0
  356. package/src/event-v2-bridge.ts +76 -0
  357. package/src/format/formatter.ts +404 -0
  358. package/src/format/index.ts +212 -0
  359. package/src/git/index.ts +347 -0
  360. package/src/id/id.ts +80 -0
  361. package/src/ide/index.ts +70 -0
  362. package/src/image/image.ts +177 -0
  363. package/src/index.ts +208 -0
  364. package/src/installation/index.ts +349 -0
  365. package/src/lsp/client.ts +686 -0
  366. package/src/lsp/diagnostic.ts +29 -0
  367. package/src/lsp/language.ts +121 -0
  368. package/src/lsp/launch.ts +21 -0
  369. package/src/lsp/lsp.ts +517 -0
  370. package/src/lsp/server.ts +2064 -0
  371. package/src/markdown.d.ts +4 -0
  372. package/src/mcp/auth.ts +171 -0
  373. package/src/mcp/index.ts +982 -0
  374. package/src/mcp/oauth-callback.ts +232 -0
  375. package/src/mcp/oauth-provider.ts +217 -0
  376. package/src/node.ts +5 -0
  377. package/src/oryna/agent.ts +112 -0
  378. package/src/oryna/reply-service.ts +8 -0
  379. package/src/patch/index.ts +689 -0
  380. package/src/permission/arity.ts +163 -0
  381. package/src/permission/evaluate.ts +1 -0
  382. package/src/permission/index.ts +230 -0
  383. package/src/plugin/azure.ts +26 -0
  384. package/src/plugin/cloudflare.ts +76 -0
  385. package/src/plugin/digitalocean.ts +391 -0
  386. package/src/plugin/github-copilot/copilot.ts +417 -0
  387. package/src/plugin/github-copilot/models.ts +246 -0
  388. package/src/plugin/index.ts +323 -0
  389. package/src/plugin/install.ts +439 -0
  390. package/src/plugin/loader.ts +237 -0
  391. package/src/plugin/meta.ts +188 -0
  392. package/src/plugin/openai/README.md +31 -0
  393. package/src/plugin/openai/codex.ts +647 -0
  394. package/src/plugin/openai/ws-pool.ts +290 -0
  395. package/src/plugin/openai/ws.ts +381 -0
  396. package/src/plugin/oryna.ts +349 -0
  397. package/src/plugin/shared.ts +323 -0
  398. package/src/plugin/xai.ts +742 -0
  399. package/src/project/bootstrap-service.ts +9 -0
  400. package/src/project/bootstrap.ts +80 -0
  401. package/src/project/instance-context.ts +24 -0
  402. package/src/project/instance-layer.ts +11 -0
  403. package/src/project/instance-runtime.ts +16 -0
  404. package/src/project/instance-store.ts +207 -0
  405. package/src/project/project.ts +520 -0
  406. package/src/project/vcs.ts +435 -0
  407. package/src/provider/auth.ts +230 -0
  408. package/src/provider/error.ts +188 -0
  409. package/src/provider/model-status.ts +8 -0
  410. package/src/provider/provider.ts +2009 -0
  411. package/src/provider/transform.ts +1363 -0
  412. package/src/pty-preparation.ts +30 -0
  413. package/src/question/index.ts +229 -0
  414. package/src/question/schema.ts +10 -0
  415. package/src/reference/reference.ts +239 -0
  416. package/src/reference/repository-cache.ts +320 -0
  417. package/src/server/auth.ts +48 -0
  418. package/src/server/cors.ts +34 -0
  419. package/src/server/event.ts +13 -0
  420. package/src/server/global-lifecycle.ts +37 -0
  421. package/src/server/init-projectors.ts +3 -0
  422. package/src/server/mdns.ts +60 -0
  423. package/src/server/projectors.ts +1 -0
  424. package/src/server/proxy-util.ts +48 -0
  425. package/src/server/routes/instance/httpapi/AGENTS.md +39 -0
  426. package/src/server/routes/instance/httpapi/api.ts +78 -0
  427. package/src/server/routes/instance/httpapi/errors.ts +193 -0
  428. package/src/server/routes/instance/httpapi/groups/config.ts +65 -0
  429. package/src/server/routes/instance/httpapi/groups/control-plane.ts +35 -0
  430. package/src/server/routes/instance/httpapi/groups/control.ts +76 -0
  431. package/src/server/routes/instance/httpapi/groups/event.ts +29 -0
  432. package/src/server/routes/instance/httpapi/groups/experimental.ts +260 -0
  433. package/src/server/routes/instance/httpapi/groups/file.ts +172 -0
  434. package/src/server/routes/instance/httpapi/groups/global.ts +138 -0
  435. package/src/server/routes/instance/httpapi/groups/instance.ts +206 -0
  436. package/src/server/routes/instance/httpapi/groups/mcp.ts +156 -0
  437. package/src/server/routes/instance/httpapi/groups/metadata.ts +18 -0
  438. package/src/server/routes/instance/httpapi/groups/permission.ts +61 -0
  439. package/src/server/routes/instance/httpapi/groups/project-copy.ts +88 -0
  440. package/src/server/routes/instance/httpapi/groups/project.ts +93 -0
  441. package/src/server/routes/instance/httpapi/groups/provider.ts +101 -0
  442. package/src/server/routes/instance/httpapi/groups/pty.ts +172 -0
  443. package/src/server/routes/instance/httpapi/groups/query.ts +12 -0
  444. package/src/server/routes/instance/httpapi/groups/question.ts +74 -0
  445. package/src/server/routes/instance/httpapi/groups/session.ts +462 -0
  446. package/src/server/routes/instance/httpapi/groups/sync.ts +113 -0
  447. package/src/server/routes/instance/httpapi/groups/tui.ts +208 -0
  448. package/src/server/routes/instance/httpapi/groups/workspace.ts +141 -0
  449. package/src/server/routes/instance/httpapi/handlers/config.ts +151 -0
  450. package/src/server/routes/instance/httpapi/handlers/control-plane.ts +37 -0
  451. package/src/server/routes/instance/httpapi/handlers/control.ts +40 -0
  452. package/src/server/routes/instance/httpapi/handlers/event.ts +102 -0
  453. package/src/server/routes/instance/httpapi/handlers/experimental.ts +187 -0
  454. package/src/server/routes/instance/httpapi/handlers/file.ts +103 -0
  455. package/src/server/routes/instance/httpapi/handlers/global.ts +157 -0
  456. package/src/server/routes/instance/httpapi/handlers/instance.ts +110 -0
  457. package/src/server/routes/instance/httpapi/handlers/mcp.ts +111 -0
  458. package/src/server/routes/instance/httpapi/handlers/permission.ts +41 -0
  459. package/src/server/routes/instance/httpapi/handlers/project-copy.ts +157 -0
  460. package/src/server/routes/instance/httpapi/handlers/project.ts +63 -0
  461. package/src/server/routes/instance/httpapi/handlers/provider.ts +160 -0
  462. package/src/server/routes/instance/httpapi/handlers/pty.ts +258 -0
  463. package/src/server/routes/instance/httpapi/handlers/question.ts +54 -0
  464. package/src/server/routes/instance/httpapi/handlers/session-errors.ts +21 -0
  465. package/src/server/routes/instance/httpapi/handlers/session.ts +442 -0
  466. package/src/server/routes/instance/httpapi/handlers/sync.ts +95 -0
  467. package/src/server/routes/instance/httpapi/handlers/tui.ts +131 -0
  468. package/src/server/routes/instance/httpapi/handlers/workspace.ts +102 -0
  469. package/src/server/routes/instance/httpapi/lifecycle.ts +57 -0
  470. package/src/server/routes/instance/httpapi/middleware/authorization.ts +147 -0
  471. package/src/server/routes/instance/httpapi/middleware/compression.ts +64 -0
  472. package/src/server/routes/instance/httpapi/middleware/cors-vary.ts +29 -0
  473. package/src/server/routes/instance/httpapi/middleware/error.ts +36 -0
  474. package/src/server/routes/instance/httpapi/middleware/fence.ts +25 -0
  475. package/src/server/routes/instance/httpapi/middleware/instance-context.ts +43 -0
  476. package/src/server/routes/instance/httpapi/middleware/proxy.ts +108 -0
  477. package/src/server/routes/instance/httpapi/middleware/schema-error.ts +42 -0
  478. package/src/server/routes/instance/httpapi/middleware/workspace-routing.ts +250 -0
  479. package/src/server/routes/instance/httpapi/public.ts +535 -0
  480. package/src/server/routes/instance/httpapi/server.ts +277 -0
  481. package/src/server/routes/instance/httpapi/websocket-tracker.ts +57 -0
  482. package/src/server/server.ts +218 -0
  483. package/src/server/shared/fence.ts +68 -0
  484. package/src/server/shared/pty-ticket.ts +15 -0
  485. package/src/server/shared/public-ui.ts +12 -0
  486. package/src/server/shared/tui-control.ts +28 -0
  487. package/src/server/shared/ui.ts +108 -0
  488. package/src/server/shared/workspace-routing.ts +38 -0
  489. package/src/session/compaction.ts +609 -0
  490. package/src/session/instruction.ts +237 -0
  491. package/src/session/llm/AGENTS.md +90 -0
  492. package/src/session/llm/ai-sdk.ts +288 -0
  493. package/src/session/llm/native-request.ts +196 -0
  494. package/src/session/llm/native-runtime.ts +195 -0
  495. package/src/session/llm/request.ts +215 -0
  496. package/src/session/llm.ts +402 -0
  497. package/src/session/message-error.ts +14 -0
  498. package/src/session/message-v2.ts +745 -0
  499. package/src/session/message.ts +148 -0
  500. package/src/session/overflow.ts +34 -0
  501. package/src/session/processor.ts +1063 -0
  502. package/src/session/prompt/anthropic.txt +105 -0
  503. package/src/session/prompt/beast.txt +147 -0
  504. package/src/session/prompt/build-switch.txt +5 -0
  505. package/src/session/prompt/codex.txt +79 -0
  506. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  507. package/src/session/prompt/default.txt +95 -0
  508. package/src/session/prompt/gemini.txt +155 -0
  509. package/src/session/prompt/gpt.txt +107 -0
  510. package/src/session/prompt/kimi.txt +95 -0
  511. package/src/session/prompt/max-steps.txt +16 -0
  512. package/src/session/prompt/plan-mode.txt +70 -0
  513. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  514. package/src/session/prompt/plan.txt +26 -0
  515. package/src/session/prompt/reference.ts +72 -0
  516. package/src/session/prompt/trinity.txt +97 -0
  517. package/src/session/prompt.ts +1755 -0
  518. package/src/session/reminders.ts +92 -0
  519. package/src/session/retry.ts +201 -0
  520. package/src/session/revert.ts +153 -0
  521. package/src/session/run-state.ts +153 -0
  522. package/src/session/schema.ts +26 -0
  523. package/src/session/session.ts +1116 -0
  524. package/src/session/status.ts +94 -0
  525. package/src/session/summary.ts +162 -0
  526. package/src/session/system.ts +84 -0
  527. package/src/session/todo.ts +87 -0
  528. package/src/session/tools.ts +211 -0
  529. package/src/share/session.ts +58 -0
  530. package/src/share/share-next.ts +379 -0
  531. package/src/shell/shell.ts +215 -0
  532. package/src/skill/discovery.ts +115 -0
  533. package/src/skill/index.ts +357 -0
  534. package/src/snapshot/index.ts +759 -0
  535. package/src/sql.d.ts +4 -0
  536. package/src/storage/schema.ts +5 -0
  537. package/src/storage/storage.ts +329 -0
  538. package/src/sync/README.md +179 -0
  539. package/src/sync/schema.ts +11 -0
  540. package/src/temporary.ts +33 -0
  541. package/src/tool/apply_patch.ts +313 -0
  542. package/src/tool/apply_patch.txt +33 -0
  543. package/src/tool/edit.ts +737 -0
  544. package/src/tool/edit.txt +10 -0
  545. package/src/tool/external-directory.ts +49 -0
  546. package/src/tool/glob.ts +84 -0
  547. package/src/tool/glob.txt +6 -0
  548. package/src/tool/grep.ts +140 -0
  549. package/src/tool/grep.txt +8 -0
  550. package/src/tool/invalid.ts +21 -0
  551. package/src/tool/json-schema.ts +164 -0
  552. package/src/tool/lsp.ts +113 -0
  553. package/src/tool/lsp.txt +24 -0
  554. package/src/tool/mcp-websearch.ts +96 -0
  555. package/src/tool/plan-enter.txt +14 -0
  556. package/src/tool/plan-exit.txt +13 -0
  557. package/src/tool/plan.ts +79 -0
  558. package/src/tool/question.ts +44 -0
  559. package/src/tool/question.txt +10 -0
  560. package/src/tool/read.ts +392 -0
  561. package/src/tool/read.txt +14 -0
  562. package/src/tool/registry.ts +475 -0
  563. package/src/tool/reply.ts +29 -0
  564. package/src/tool/schema.ts +14 -0
  565. package/src/tool/shell/id.ts +19 -0
  566. package/src/tool/shell/prompt.ts +307 -0
  567. package/src/tool/shell/shell.txt +21 -0
  568. package/src/tool/shell.ts +660 -0
  569. package/src/tool/skill.ts +72 -0
  570. package/src/tool/skill.txt +5 -0
  571. package/src/tool/task.ts +338 -0
  572. package/src/tool/task.txt +19 -0
  573. package/src/tool/todo.ts +57 -0
  574. package/src/tool/todowrite.txt +44 -0
  575. package/src/tool/tool.ts +183 -0
  576. package/src/tool/truncate.ts +160 -0
  577. package/src/tool/truncation-dir.ts +4 -0
  578. package/src/tool/webfetch.ts +192 -0
  579. package/src/tool/webfetch.txt +13 -0
  580. package/src/tool/websearch.ts +143 -0
  581. package/src/tool/websearch.txt +14 -0
  582. package/src/tool/write.ts +104 -0
  583. package/src/tool/write.txt +8 -0
  584. package/src/util/archive.ts +17 -0
  585. package/src/util/bom.ts +27 -0
  586. package/src/util/data-url.ts +9 -0
  587. package/src/util/defer.ts +10 -0
  588. package/src/util/effect-http-client.ts +11 -0
  589. package/src/util/error.ts +88 -0
  590. package/src/util/filesystem.ts +251 -0
  591. package/src/util/format.ts +20 -0
  592. package/src/util/iife.ts +3 -0
  593. package/src/util/lan-scan.ts +90 -0
  594. package/src/util/lazy.ts +20 -0
  595. package/src/util/local-context.ts +25 -0
  596. package/src/util/locale.ts +86 -0
  597. package/src/util/media.ts +26 -0
  598. package/src/util/process.ts +176 -0
  599. package/src/util/proxy-env.ts +72 -0
  600. package/src/util/queue.ts +32 -0
  601. package/src/util/record.ts +3 -0
  602. package/src/util/repository.ts +232 -0
  603. package/src/util/rpc.ts +66 -0
  604. package/src/util/signal.ts +12 -0
  605. package/src/util/timeout.ts +13 -0
  606. package/src/util/token.ts +1 -0
  607. package/src/util/wildcard.ts +59 -0
  608. package/src/worktree/index.ts +645 -0
  609. package/sst-env.d.ts +10 -0
  610. package/test/AGENTS.md +204 -0
  611. package/test/EFFECT_TEST_MIGRATION.md +169 -0
  612. package/test/account/repo.test.ts +353 -0
  613. package/test/account/service.test.ts +453 -0
  614. package/test/acp/config-option.test.ts +229 -0
  615. package/test/acp/content.test.ts +201 -0
  616. package/test/acp/directory.test.ts +186 -0
  617. package/test/acp/error.test.ts +67 -0
  618. package/test/acp/event.test.ts +711 -0
  619. package/test/acp/permission.test.ts +273 -0
  620. package/test/acp/service-session.test.ts +1174 -0
  621. package/test/acp/session.test.ts +200 -0
  622. package/test/acp/tool.test.ts +210 -0
  623. package/test/acp/usage.test.ts +315 -0
  624. package/test/agent/agent.test.ts +710 -0
  625. package/test/agent/plan-mode-subagent-bypass.test.ts +213 -0
  626. package/test/agent/plugin-agent-regression.test.ts +62 -0
  627. package/test/auth/auth.test.ts +77 -0
  628. package/test/background/job.test.ts +243 -0
  629. package/test/cli/account.test.ts +30 -0
  630. package/test/cli/acp/acp-test-client.ts +97 -0
  631. package/test/cli/acp/config-options.test.ts +103 -0
  632. package/test/cli/acp/helpers.ts +96 -0
  633. package/test/cli/acp/initialize-auth.test.ts +61 -0
  634. package/test/cli/acp/lifecycle.test.ts +118 -0
  635. package/test/cli/acp/prompt-content.test.ts +97 -0
  636. package/test/cli/acp/skills.test.ts +38 -0
  637. package/test/cli/cmd/tui/aggregate-failures.test.ts +93 -0
  638. package/test/cli/cmd/tui/attention.test.ts +484 -0
  639. package/test/cli/cmd/tui/dialog-workspace-create.test.ts +28 -0
  640. package/test/cli/cmd/tui/model-options.test.ts +30 -0
  641. package/test/cli/cmd/tui/notifications.test.ts +267 -0
  642. package/test/cli/cmd/tui/prompt-history.test.ts +44 -0
  643. package/test/cli/cmd/tui/prompt-part.test.ts +77 -0
  644. package/test/cli/cmd/tui/prompt-traits.test.ts +29 -0
  645. package/test/cli/cmd/tui/provider-options.test.ts +29 -0
  646. package/test/cli/cmd/tui/sync-fixture.tsx +64 -0
  647. package/test/cli/cmd/tui/sync-live-hydration.test.tsx +278 -0
  648. package/test/cli/cmd/tui/sync-undefined-messages.test.tsx +47 -0
  649. package/test/cli/cmd/tui/sync.test.tsx +70 -0
  650. package/test/cli/effect-cmd-instance-als.test.ts +39 -0
  651. package/test/cli/error.test.ts +95 -0
  652. package/test/cli/github-action.test.ts +199 -0
  653. package/test/cli/github-remote.test.ts +90 -0
  654. package/test/cli/help/__snapshots__/help-snapshots.test.ts.snap +631 -0
  655. package/test/cli/help/help-snapshots.test.ts +137 -0
  656. package/test/cli/import.test.ts +54 -0
  657. package/test/cli/mcp-add.test.ts +74 -0
  658. package/test/cli/plugin-auth-picker.test.ts +120 -0
  659. package/test/cli/run/entry.body.test.ts +536 -0
  660. package/test/cli/run/footer.menu.test.ts +43 -0
  661. package/test/cli/run/footer.view.test.tsx +927 -0
  662. package/test/cli/run/permission.shared.test.ts +144 -0
  663. package/test/cli/run/prompt.shared.test.ts +133 -0
  664. package/test/cli/run/question.shared.test.ts +115 -0
  665. package/test/cli/run/run-process.test.ts +84 -0
  666. package/test/cli/run/runtime.boot.test.ts +282 -0
  667. package/test/cli/run/runtime.queue.test.ts +465 -0
  668. package/test/cli/run/runtime.stdin.test.ts +71 -0
  669. package/test/cli/run/scrollback.surface.test.ts +1048 -0
  670. package/test/cli/run/session-data.test.ts +595 -0
  671. package/test/cli/run/session-replay.test.ts +456 -0
  672. package/test/cli/run/session.shared.test.ts +247 -0
  673. package/test/cli/run/stream.test.ts +56 -0
  674. package/test/cli/run/stream.transport.test.ts +2363 -0
  675. package/test/cli/run/subagent-data.test.ts +456 -0
  676. package/test/cli/run/theme.test.ts +152 -0
  677. package/test/cli/run/variant.shared.test.ts +217 -0
  678. package/test/cli/serve/serve-process.test.ts +61 -0
  679. package/test/cli/smokes/read-only.test.ts +115 -0
  680. package/test/cli/tui/__snapshots__/inline-tool-wrap-snapshot.test.tsx.snap +72 -0
  681. package/test/cli/tui/app-lifecycle.test.ts +261 -0
  682. package/test/cli/tui/dialog-prompt.test.tsx +146 -0
  683. package/test/cli/tui/diff-viewer-file-tree-utils.test.ts +323 -0
  684. package/test/cli/tui/diff-viewer-file-tree.test.tsx +197 -0
  685. package/test/cli/tui/diff-viewer.test.tsx +230 -0
  686. package/test/cli/tui/editor-context-zed.test.ts +384 -0
  687. package/test/cli/tui/editor-context.test.tsx +288 -0
  688. package/test/cli/tui/inline-tool-wrap-snapshot.test.tsx +232 -0
  689. package/test/cli/tui/keymap.test.tsx +136 -0
  690. package/test/cli/tui/plugin-add.test.ts +110 -0
  691. package/test/cli/tui/plugin-install.test.ts +87 -0
  692. package/test/cli/tui/plugin-lifecycle.test.ts +224 -0
  693. package/test/cli/tui/plugin-loader-entrypoint.test.ts +485 -0
  694. package/test/cli/tui/plugin-loader-pure.test.ts +72 -0
  695. package/test/cli/tui/plugin-loader.test.ts +1332 -0
  696. package/test/cli/tui/plugin-toggle.test.ts +264 -0
  697. package/test/cli/tui/prompt-submit-race.test.ts +98 -0
  698. package/test/cli/tui/revert-diff.test.ts +35 -0
  699. package/test/cli/tui/slot-replace.test.tsx +50 -0
  700. package/test/cli/tui/sync-v2.test.tsx +558 -0
  701. package/test/cli/tui/theme-store.test.ts +76 -0
  702. package/test/cli/tui/thinking.test.ts +36 -0
  703. package/test/cli/tui/thread.test.ts +28 -0
  704. package/test/cli/tui/transcript.test.ts +426 -0
  705. package/test/cli/tui/use-event.test.tsx +145 -0
  706. package/test/config/agent-color.test.ts +47 -0
  707. package/test/config/config.test.ts +1991 -0
  708. package/test/config/entry-name.test.ts +57 -0
  709. package/test/config/fixtures/empty-frontmatter.md +4 -0
  710. package/test/config/fixtures/frontmatter.md +28 -0
  711. package/test/config/fixtures/markdown-header.md +11 -0
  712. package/test/config/fixtures/no-frontmatter.md +1 -0
  713. package/test/config/fixtures/weird-model-id.md +13 -0
  714. package/test/config/lsp.test.ts +69 -0
  715. package/test/config/markdown.test.ts +228 -0
  716. package/test/config/plugin.test.ts +0 -0
  717. package/test/config/tui.test.ts +878 -0
  718. package/test/control-plane/adapters.test.ts +71 -0
  719. package/test/control-plane/workspace.test.ts +1704 -0
  720. package/test/effect/app-runtime-logger.test.ts +105 -0
  721. package/test/effect/config-service.test.ts +65 -0
  722. package/test/effect/instance-state.test.ts +391 -0
  723. package/test/effect/run-service.test.ts +89 -0
  724. package/test/effect/runner.test.ts +514 -0
  725. package/test/effect/runtime-flags.test.ts +373 -0
  726. package/test/fake/account.ts +9 -0
  727. package/test/fake/auth.ts +8 -0
  728. package/test/fake/npm.ts +8 -0
  729. package/test/fake/provider.ts +82 -0
  730. package/test/fake/skill.ts +8 -0
  731. package/test/filesystem/filesystem.test.ts +319 -0
  732. package/test/fixture/agent-plugin.constants.ts +6 -0
  733. package/test/fixture/agent-plugin.ts +12 -0
  734. package/test/fixture/config.ts +23 -0
  735. package/test/fixture/db.ts +11 -0
  736. package/test/fixture/fixture.test.ts +26 -0
  737. package/test/fixture/fixture.ts +224 -0
  738. package/test/fixture/flag.ts +20 -0
  739. package/test/fixture/flock-worker.ts +72 -0
  740. package/test/fixture/lsp/fake-lsp-server.js +249 -0
  741. package/test/fixture/plug-worker.ts +93 -0
  742. package/test/fixture/plugin-meta-worker.ts +19 -0
  743. package/test/fixture/plugin.ts +10 -0
  744. package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
  745. package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
  746. package/test/fixture/skills/cloudflare/SKILL.md +211 -0
  747. package/test/fixture/skills/index.json +6 -0
  748. package/test/fixture/tui-plugin.ts +355 -0
  749. package/test/fixture/tui-runtime.ts +64 -0
  750. package/test/fixture/tui-sdk.ts +82 -0
  751. package/test/fixture/workspace.ts +30 -0
  752. package/test/fixtures/recordings/session/native-anthropic-tool-loop.json +49 -0
  753. package/test/fixtures/recordings/session/native-openai-oauth-tool-loop.json +45 -0
  754. package/test/fixtures/recordings/session/native-zen-tool-loop.json +49 -0
  755. package/test/format/format.test.ts +228 -0
  756. package/test/git/git.test.ts +178 -0
  757. package/test/ide/ide.test.ts +82 -0
  758. package/test/image/fixtures/picture-5mb-base64.png +0 -0
  759. package/test/image/image.test.ts +123 -0
  760. package/test/installation/installation.test.ts +230 -0
  761. package/test/lib/cli-process.ts +459 -0
  762. package/test/lib/effect.ts +177 -0
  763. package/test/lib/filesystem.ts +10 -0
  764. package/test/lib/llm-server.ts +771 -0
  765. package/test/lib/snapshot.ts +73 -0
  766. package/test/lib/test-provider.ts +37 -0
  767. package/test/lib/websocket.ts +46 -0
  768. package/test/lsp/client.test.ts +493 -0
  769. package/test/lsp/index.test.ts +232 -0
  770. package/test/lsp/launch.test.ts +22 -0
  771. package/test/lsp/lifecycle.test.ts +160 -0
  772. package/test/mcp/auth.test.ts +78 -0
  773. package/test/mcp/headers.test.ts +126 -0
  774. package/test/mcp/lifecycle.test.ts +888 -0
  775. package/test/mcp/oauth-auto-connect.test.ts +236 -0
  776. package/test/mcp/oauth-browser.test.ts +228 -0
  777. package/test/mcp/oauth-callback.test.ts +34 -0
  778. package/test/mcp/oauth-provider.test.ts +61 -0
  779. package/test/patch/patch.test.ts +383 -0
  780. package/test/permission/arity.test.ts +33 -0
  781. package/test/permission/next.test.ts +1176 -0
  782. package/test/permission-task.test.ts +318 -0
  783. package/test/plugin/auth-override.test.ts +105 -0
  784. package/test/plugin/cloudflare.test.ts +68 -0
  785. package/test/plugin/codex.test.ts +247 -0
  786. package/test/plugin/github-copilot-models.test.ts +332 -0
  787. package/test/plugin/install-concurrency.test.ts +140 -0
  788. package/test/plugin/install.test.ts +570 -0
  789. package/test/plugin/loader-shared.test.ts +1303 -0
  790. package/test/plugin/meta.test.ts +137 -0
  791. package/test/plugin/openai-rollout.test.ts +17 -0
  792. package/test/plugin/openai-ws.test.ts +877 -0
  793. package/test/plugin/shared.test.ts +88 -0
  794. package/test/plugin/trigger.test.ts +120 -0
  795. package/test/plugin/workspace-adapter.test.ts +137 -0
  796. package/test/plugin/xai.test.ts +634 -0
  797. package/test/preload.ts +95 -0
  798. package/test/project/instance-bootstrap.test.ts +110 -0
  799. package/test/project/instance.test.ts +245 -0
  800. package/test/project/migrate-global.test.ts +170 -0
  801. package/test/project/project-directory.test.ts +169 -0
  802. package/test/project/project.test.ts +818 -0
  803. package/test/project/vcs.test.ts +336 -0
  804. package/test/project/worktree-remove.test.ts +126 -0
  805. package/test/project/worktree.test.ts +320 -0
  806. package/test/provider/amazon-bedrock.test.ts +360 -0
  807. package/test/provider/cf-ai-gateway-e2e.test.ts +132 -0
  808. package/test/provider/digitalocean.test.ts +123 -0
  809. package/test/provider/gitlab-duo.test.ts +412 -0
  810. package/test/provider/header-timeout.test.ts +233 -0
  811. package/test/provider/model-status.test.ts +61 -0
  812. package/test/provider/provider.test.ts +1793 -0
  813. package/test/provider/transform.test.ts +3937 -0
  814. package/test/pty/pty-shell.test.ts +102 -0
  815. package/test/question/question.test.ts +465 -0
  816. package/test/reference/reference.test.ts +310 -0
  817. package/test/server/AGENTS.md +15 -0
  818. package/test/server/auth.test.ts +59 -0
  819. package/test/server/global-bus.ts +31 -0
  820. package/test/server/global-session-list.test.ts +107 -0
  821. package/test/server/httpapi-authorization.test.ts +174 -0
  822. package/test/server/httpapi-compression.test.ts +154 -0
  823. package/test/server/httpapi-config.test.ts +113 -0
  824. package/test/server/httpapi-control-plane.test.ts +63 -0
  825. package/test/server/httpapi-cors-vary.test.ts +66 -0
  826. package/test/server/httpapi-cors.test.ts +122 -0
  827. package/test/server/httpapi-error-middleware.test.ts +96 -0
  828. package/test/server/httpapi-event.test.ts +97 -0
  829. package/test/server/httpapi-exercise/assertions.ts +64 -0
  830. package/test/server/httpapi-exercise/backend.ts +144 -0
  831. package/test/server/httpapi-exercise/dsl.ts +210 -0
  832. package/test/server/httpapi-exercise/environment.ts +40 -0
  833. package/test/server/httpapi-exercise/index.ts +1535 -0
  834. package/test/server/httpapi-exercise/report.ts +66 -0
  835. package/test/server/httpapi-exercise/routing.ts +96 -0
  836. package/test/server/httpapi-exercise/runner.ts +267 -0
  837. package/test/server/httpapi-exercise/runtime.ts +52 -0
  838. package/test/server/httpapi-exercise/types.ts +123 -0
  839. package/test/server/httpapi-experimental.test.ts +300 -0
  840. package/test/server/httpapi-file.test.ts +76 -0
  841. package/test/server/httpapi-global.test.ts +66 -0
  842. package/test/server/httpapi-instance-context.test.ts +347 -0
  843. package/test/server/httpapi-instance-route-auth.test.ts +84 -0
  844. package/test/server/httpapi-instance.test.ts +265 -0
  845. package/test/server/httpapi-layer.ts +33 -0
  846. package/test/server/httpapi-listen.test.ts +415 -0
  847. package/test/server/httpapi-mcp-oauth.test.ts +73 -0
  848. package/test/server/httpapi-mcp.test.ts +226 -0
  849. package/test/server/httpapi-mdns.test.ts +82 -0
  850. package/test/server/httpapi-promptasync-context.test.ts +222 -0
  851. package/test/server/httpapi-provider.test.ts +403 -0
  852. package/test/server/httpapi-pty.test.ts +275 -0
  853. package/test/server/httpapi-public-openapi.test.ts +297 -0
  854. package/test/server/httpapi-query-schema-drift.test.ts +330 -0
  855. package/test/server/httpapi-schema-error-body.test.ts +165 -0
  856. package/test/server/httpapi-sdk.test.ts +909 -0
  857. package/test/server/httpapi-session.test.ts +1013 -0
  858. package/test/server/httpapi-sync.test.ts +154 -0
  859. package/test/server/httpapi-ui.test.ts +456 -0
  860. package/test/server/httpapi-v2-location.test.ts +85 -0
  861. package/test/server/httpapi-workspace-routing.test.ts +554 -0
  862. package/test/server/httpapi-workspace.test.ts +515 -0
  863. package/test/server/negative-tokens-regression.test.ts +83 -0
  864. package/test/server/project-copy.test.ts +101 -0
  865. package/test/server/project-init-git.test.ts +117 -0
  866. package/test/server/proxy-util.test.ts +113 -0
  867. package/test/server/sdk-error-shape.test.ts +84 -0
  868. package/test/server/sdk-v1-smoke.test.ts +60 -0
  869. package/test/server/session-actions.test.ts +112 -0
  870. package/test/server/session-diff-missing-patch.test.ts +99 -0
  871. package/test/server/session-list.test.ts +314 -0
  872. package/test/server/session-messages.test.ts +182 -0
  873. package/test/server/session-select.test.ts +69 -0
  874. package/test/server/workspace-proxy.test.ts +181 -0
  875. package/test/server/workspace-routing.test.ts +94 -0
  876. package/test/server/worktree-endpoint-repro.test.ts +307 -0
  877. package/test/session/compaction.test.ts +1835 -0
  878. package/test/session/instruction.test.ts +256 -0
  879. package/test/session/llm-native-recorded.test.ts +433 -0
  880. package/test/session/llm-native.test.ts +760 -0
  881. package/test/session/llm.test.ts +1932 -0
  882. package/test/session/message-v2.test.ts +1661 -0
  883. package/test/session/messages-pagination.test.ts +1059 -0
  884. package/test/session/processor-effect.test.ts +1101 -0
  885. package/test/session/prompt.test.ts +2318 -0
  886. package/test/session/retry.test.ts +439 -0
  887. package/test/session/revert-compact.test.ts +642 -0
  888. package/test/session/schema-decoding.test.ts +313 -0
  889. package/test/session/session-schema.test.ts +78 -0
  890. package/test/session/session.test.ts +251 -0
  891. package/test/session/snapshot-tool-race.test.ts +280 -0
  892. package/test/session/structured-output-integration.test.ts +235 -0
  893. package/test/session/structured-output.test.ts +387 -0
  894. package/test/session/system.test.ts +84 -0
  895. package/test/share/share-next.test.ts +344 -0
  896. package/test/shell/shell.test.ts +99 -0
  897. package/test/skill/discovery.test.ts +139 -0
  898. package/test/skill/skill.test.ts +571 -0
  899. package/test/snapshot/snapshot.test.ts +1121 -0
  900. package/test/storage/storage.test.ts +296 -0
  901. package/test/storage/workspace-time-migration.test.ts +50 -0
  902. package/test/tool/__snapshots__/parameters.test.ts.snap +484 -0
  903. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  904. package/test/tool/apply_patch.test.ts +533 -0
  905. package/test/tool/edit.test.ts +578 -0
  906. package/test/tool/external-directory.test.ts +155 -0
  907. package/test/tool/fixtures/large-image.png +0 -0
  908. package/test/tool/fixtures/models-api.json +117299 -0
  909. package/test/tool/glob.test.ts +188 -0
  910. package/test/tool/grep.test.ts +266 -0
  911. package/test/tool/lsp.test.ts +181 -0
  912. package/test/tool/parameters.test.ts +293 -0
  913. package/test/tool/question.test.ts +138 -0
  914. package/test/tool/read.test.ts +654 -0
  915. package/test/tool/registry.test.ts +539 -0
  916. package/test/tool/shell.test.ts +1238 -0
  917. package/test/tool/skill.test.ts +132 -0
  918. package/test/tool/task.test.ts +901 -0
  919. package/test/tool/tool-define.test.ts +153 -0
  920. package/test/tool/truncation.test.ts +266 -0
  921. package/test/tool/webfetch.test.ts +113 -0
  922. package/test/tool/websearch.test.ts +99 -0
  923. package/test/tool/write.test.ts +276 -0
  924. package/test/util/data-url.test.ts +14 -0
  925. package/test/util/error.test.ts +64 -0
  926. package/test/util/filesystem.test.ts +656 -0
  927. package/test/util/format.test.ts +59 -0
  928. package/test/util/glob.test.ts +164 -0
  929. package/test/util/iife.test.ts +36 -0
  930. package/test/util/lazy.test.ts +50 -0
  931. package/test/util/log.test.ts +77 -0
  932. package/test/util/module.test.ts +59 -0
  933. package/test/util/process.test.ts +128 -0
  934. package/test/util/repository.test.ts +93 -0
  935. package/test/util/timeout.test.ts +21 -0
  936. package/test/util/wildcard.test.ts +90 -0
  937. package/test/v2/session-message-updater.test.ts +270 -0
  938. package/tsconfig.json +17 -0
@@ -0,0 +1,2363 @@
1
+ import { afterEach, describe, expect, mock, spyOn, test } from "bun:test"
2
+ import { OpencodeClient, type GlobalEvent } from "@opencode-ai/sdk/v2"
3
+ import { createSessionTransport } from "@/cli/cmd/run/stream.transport"
4
+ import type { FooterApi, FooterEvent, LocalReplayRow, RunFilePart, StreamCommit } from "@/cli/cmd/run/types"
5
+
6
+ type EventStream = Awaited<ReturnType<OpencodeClient["event"]["subscribe"]>>["stream"]
7
+ type GlobalEventStream = Awaited<ReturnType<OpencodeClient["global"]["event"]>>["stream"]
8
+ type SdkEvent = EventStream extends AsyncGenerator<infer T, unknown, unknown> ? T : never
9
+ type SessionMessage = NonNullable<Awaited<ReturnType<OpencodeClient["session"]["messages"]>>["data"]>[number]
10
+ type SessionChild = NonNullable<Awaited<ReturnType<OpencodeClient["session"]["children"]>>["data"]>[number]
11
+ type SessionToolPart = Extract<SessionMessage["parts"][number], { type: "tool" }>
12
+ type SessionStatusMap = NonNullable<Awaited<ReturnType<OpencodeClient["session"]["status"]>>["data"]>
13
+ type TextPart = Extract<SessionMessage["parts"][number], { type: "text" }>
14
+ type ReasoningPart = Extract<SessionMessage["parts"][number], { type: "reasoning" }>
15
+
16
+ afterEach(() => {
17
+ mock.restore()
18
+ })
19
+
20
+ function defer<T = void>() {
21
+ let resolve!: (value: T | PromiseLike<T>) => void
22
+ let reject!: (error?: unknown) => void
23
+ const promise = new Promise<T>((next, fail) => {
24
+ resolve = next
25
+ reject = fail
26
+ })
27
+
28
+ return { promise, resolve, reject }
29
+ }
30
+
31
+ async function waitFor<T>(check: () => T | undefined, timeout = 1_000): Promise<T> {
32
+ const end = Date.now() + timeout
33
+ while (Date.now() < end) {
34
+ const value = check()
35
+ if (value !== undefined) {
36
+ return value
37
+ }
38
+
39
+ await Bun.sleep(10)
40
+ }
41
+
42
+ throw new Error("timed out waiting for value")
43
+ }
44
+
45
+ function busy(sessionID = "session-1") {
46
+ return {
47
+ id: `evt-${sessionID}-busy`,
48
+ type: "session.status",
49
+ properties: {
50
+ sessionID,
51
+ status: {
52
+ type: "busy",
53
+ },
54
+ },
55
+ } satisfies SdkEvent
56
+ }
57
+
58
+ function idle(sessionID = "session-1") {
59
+ return {
60
+ id: `evt-${sessionID}-idle`,
61
+ type: "session.status",
62
+ properties: {
63
+ sessionID,
64
+ status: {
65
+ type: "idle",
66
+ },
67
+ },
68
+ } satisfies SdkEvent
69
+ }
70
+
71
+ function retry(sessionID: string, attempt: number, message: string) {
72
+ return {
73
+ id: `evt-${sessionID}-retry-${attempt}`,
74
+ type: "session.status",
75
+ properties: {
76
+ sessionID,
77
+ status: {
78
+ type: "retry",
79
+ attempt,
80
+ message,
81
+ next: 1,
82
+ },
83
+ },
84
+ } satisfies SdkEvent
85
+ }
86
+
87
+ function assistant(id: string) {
88
+ return {
89
+ id: `evt-${id}`,
90
+ type: "message.updated",
91
+ properties: {
92
+ sessionID: "session-1",
93
+ info: assistantMessage({
94
+ sessionID: "session-1",
95
+ id,
96
+ parts: [],
97
+ }).info,
98
+ },
99
+ } satisfies SdkEvent
100
+ }
101
+
102
+ const StreamClosed = undefined as never
103
+
104
+ function feed<T, R = never>(returnValue: R = StreamClosed) {
105
+ const list: T[] = []
106
+ let done = false
107
+ let wake: (() => void) | undefined
108
+
109
+ const wrapped = (async function* (): AsyncGenerator<T, R, unknown> {
110
+ while (!done || list.length > 0) {
111
+ if (list.length === 0) {
112
+ await new Promise<void>((resolve) => {
113
+ wake = resolve
114
+ })
115
+ continue
116
+ }
117
+
118
+ const next = list.shift()
119
+ if (!next) {
120
+ continue
121
+ }
122
+
123
+ yield next
124
+ }
125
+ return returnValue as R
126
+ })()
127
+
128
+ return {
129
+ stream: wrapped,
130
+ push(value: T) {
131
+ list.push(value)
132
+ wake?.()
133
+ wake = undefined
134
+ },
135
+ close() {
136
+ done = true
137
+ wake?.()
138
+ wake = undefined
139
+ },
140
+ }
141
+ }
142
+
143
+ function eventFeed() {
144
+ return feed<SdkEvent>()
145
+ }
146
+
147
+ function globalFeed() {
148
+ return feed<GlobalEvent>()
149
+ }
150
+
151
+ function emptyStream(): EventStream {
152
+ return (async function* (): AsyncGenerator<SdkEvent> {})()
153
+ }
154
+
155
+ function ok<T>(data: T) {
156
+ return Promise.resolve({
157
+ data,
158
+ error: undefined,
159
+ request: new Request("https://opencode.test"),
160
+ response: new Response(),
161
+ })
162
+ }
163
+
164
+ function sse(stream: EventStream) {
165
+ return Promise.resolve({ stream })
166
+ }
167
+
168
+ function globalSse(stream: GlobalEventStream) {
169
+ return Promise.resolve({ stream })
170
+ }
171
+
172
+ function wrapGlobalStream(stream: EventStream): GlobalEventStream {
173
+ return (async function* (): GlobalEventStream {
174
+ for await (const event of stream) {
175
+ yield globalEvent(event)
176
+ }
177
+ return StreamClosed
178
+ })()
179
+ }
180
+
181
+ function statusMap(busy: boolean): SessionStatusMap {
182
+ if (busy) {
183
+ return { "session-1": { type: "busy" } }
184
+ }
185
+
186
+ return {}
187
+ }
188
+
189
+ function assistantMessage(input: { sessionID: string; id: string; parts: SessionMessage["parts"] }): SessionMessage {
190
+ return {
191
+ info: {
192
+ id: input.id,
193
+ sessionID: input.sessionID,
194
+ role: "assistant",
195
+ time: {
196
+ created: 1,
197
+ },
198
+ parentID: "msg-user-1",
199
+ modelID: "gpt-5",
200
+ providerID: "openai",
201
+ mode: "chat",
202
+ agent: "build",
203
+ path: {
204
+ cwd: "/tmp",
205
+ root: "/tmp",
206
+ },
207
+ cost: 0,
208
+ tokens: {
209
+ input: 1,
210
+ output: 1,
211
+ reasoning: 0,
212
+ cache: {
213
+ read: 0,
214
+ write: 0,
215
+ },
216
+ },
217
+ },
218
+ parts: input.parts,
219
+ }
220
+ }
221
+
222
+ function runningTool(input: {
223
+ sessionID: string
224
+ messageID: string
225
+ id: string
226
+ callID: string
227
+ tool: string
228
+ body: Record<string, unknown>
229
+ metadata?: Record<string, unknown>
230
+ }): SessionToolPart {
231
+ return {
232
+ id: input.id,
233
+ sessionID: input.sessionID,
234
+ messageID: input.messageID,
235
+ type: "tool",
236
+ callID: input.callID,
237
+ tool: input.tool,
238
+ state: {
239
+ status: "running",
240
+ input: input.body,
241
+ ...(input.metadata ? { metadata: input.metadata } : {}),
242
+ time: {
243
+ start: 1,
244
+ },
245
+ },
246
+ }
247
+ }
248
+
249
+ function completedTool(input: {
250
+ sessionID: string
251
+ messageID: string
252
+ id: string
253
+ callID: string
254
+ tool: string
255
+ body: Record<string, unknown>
256
+ output?: string
257
+ metadata?: Record<string, unknown>
258
+ }): SessionToolPart {
259
+ return {
260
+ id: input.id,
261
+ sessionID: input.sessionID,
262
+ messageID: input.messageID,
263
+ type: "tool",
264
+ callID: input.callID,
265
+ tool: input.tool,
266
+ state: {
267
+ status: "completed",
268
+ input: input.body,
269
+ output: input.output ?? "",
270
+ title: input.tool,
271
+ metadata: input.metadata ?? {},
272
+ time: {
273
+ start: 1,
274
+ end: 2,
275
+ },
276
+ },
277
+ }
278
+ }
279
+
280
+ function textPart(id: string, messageID: string, text: string, sessionID = "session-1"): TextPart {
281
+ return {
282
+ id,
283
+ sessionID,
284
+ messageID,
285
+ type: "text",
286
+ text,
287
+ }
288
+ }
289
+
290
+ function textUpdated(part: TextPart): SdkEvent {
291
+ return {
292
+ id: `evt-${part.id}-updated`,
293
+ type: "message.part.updated",
294
+ properties: {
295
+ sessionID: part.sessionID,
296
+ part,
297
+ time: 1,
298
+ },
299
+ }
300
+ }
301
+
302
+ function reasoningPart(id: string, messageID: string, text: string): ReasoningPart {
303
+ return {
304
+ id,
305
+ sessionID: "session-1",
306
+ messageID,
307
+ type: "reasoning",
308
+ text,
309
+ time: { start: 1 },
310
+ }
311
+ }
312
+
313
+ function reasoningUpdated(part: ReasoningPart): SdkEvent {
314
+ return {
315
+ id: `evt-${part.id}-updated`,
316
+ type: "message.part.updated",
317
+ properties: {
318
+ sessionID: part.sessionID,
319
+ part,
320
+ time: 1,
321
+ },
322
+ }
323
+ }
324
+
325
+ function toolUpdated(part: SessionToolPart): SdkEvent {
326
+ return {
327
+ id: `evt-${part.id}-updated`,
328
+ type: "message.part.updated",
329
+ properties: {
330
+ sessionID: part.sessionID,
331
+ part,
332
+ time: 1,
333
+ },
334
+ }
335
+ }
336
+
337
+ function textDelta(messageID: string, partID: string, delta: string, sessionID = "session-1"): SdkEvent {
338
+ return {
339
+ id: `evt-${partID}-delta`,
340
+ type: "message.part.delta",
341
+ properties: {
342
+ sessionID,
343
+ messageID,
344
+ partID,
345
+ field: "text",
346
+ delta,
347
+ },
348
+ }
349
+ }
350
+
351
+ function child(id: string): SessionChild {
352
+ return {
353
+ id,
354
+ slug: id,
355
+ projectID: "project-1",
356
+ directory: "/tmp",
357
+ title: id,
358
+ version: "1",
359
+ time: {
360
+ created: 1,
361
+ updated: 1,
362
+ },
363
+ }
364
+ }
365
+
366
+ function globalEvent(payload: GlobalEvent["payload"]): GlobalEvent {
367
+ return {
368
+ directory: "/tmp",
369
+ project: "project-1",
370
+ payload,
371
+ }
372
+ }
373
+
374
+ function footer(fn?: (commit: StreamCommit) => void) {
375
+ const commits: StreamCommit[] = []
376
+ const events: FooterEvent[] = []
377
+ let closed = false
378
+ let idleCalls = 0
379
+
380
+ const api: FooterApi = {
381
+ get isClosed() {
382
+ return closed
383
+ },
384
+ onPrompt: () => () => {},
385
+ onQueuedRemove: () => () => {},
386
+ onClose: () => () => {},
387
+ event(next) {
388
+ events.push(next)
389
+ },
390
+ append(next) {
391
+ commits.push(next)
392
+ fn?.(next)
393
+ },
394
+ idle() {
395
+ idleCalls += 1
396
+ return Promise.resolve()
397
+ },
398
+ close() {
399
+ closed = true
400
+ },
401
+ destroy() {
402
+ closed = true
403
+ },
404
+ }
405
+
406
+ return {
407
+ api,
408
+ commits,
409
+ events,
410
+ get idleCalls() {
411
+ return idleCalls
412
+ },
413
+ }
414
+ }
415
+
416
+ function sdk(
417
+ input: {
418
+ stream?: EventStream
419
+ globalStream?: GlobalEventStream
420
+ subscribe?: OpencodeClient["event"]["subscribe"]
421
+ globalEvent?: OpencodeClient["global"]["event"]
422
+ promptAsync?: OpencodeClient["session"]["promptAsync"]
423
+ status?: OpencodeClient["session"]["status"]
424
+ messages?: OpencodeClient["session"]["messages"]
425
+ children?: OpencodeClient["session"]["children"]
426
+ permissions?: OpencodeClient["permission"]["list"]
427
+ questions?: OpencodeClient["question"]["list"]
428
+ } = {},
429
+ ) {
430
+ const client = new OpencodeClient()
431
+
432
+ const subscribe: OpencodeClient["event"]["subscribe"] = input.subscribe ?? (() => sse(input.stream ?? emptyStream()))
433
+ const globalEvent: OpencodeClient["global"]["event"] =
434
+ input.globalEvent ?? (() => globalSse(input.globalStream ?? wrapGlobalStream(input.stream ?? emptyStream())))
435
+ const promptAsync: OpencodeClient["session"]["promptAsync"] = input.promptAsync ?? (() => ok(undefined))
436
+ const status: OpencodeClient["session"]["status"] = input.status ?? (() => ok({}))
437
+ const messages: OpencodeClient["session"]["messages"] = input.messages ?? (() => ok([]))
438
+ const children: OpencodeClient["session"]["children"] = input.children ?? (() => ok([]))
439
+ const permissions: OpencodeClient["permission"]["list"] = input.permissions ?? (() => ok([]))
440
+ const questions: OpencodeClient["question"]["list"] = input.questions ?? (() => ok([]))
441
+
442
+ spyOn(client.event, "subscribe").mockImplementation(subscribe)
443
+ spyOn(client.global, "event").mockImplementation(globalEvent)
444
+ spyOn(client.session, "promptAsync").mockImplementation(promptAsync)
445
+ spyOn(client.session, "status").mockImplementation(status)
446
+ spyOn(client.session, "messages").mockImplementation(messages)
447
+ spyOn(client.session, "children").mockImplementation(children)
448
+ spyOn(client.permission, "list").mockImplementation(permissions)
449
+ spyOn(client.question, "list").mockImplementation(questions)
450
+
451
+ return client
452
+ }
453
+
454
+ describe("run stream transport", () => {
455
+ test("does not replay persisted main-session history during bootstrap by default", async () => {
456
+ const src = eventFeed()
457
+ const ui = footer()
458
+ const transport = await createSessionTransport({
459
+ sdk: sdk({
460
+ stream: src.stream,
461
+ messages: async ({ sessionID }) =>
462
+ sessionID === "session-1"
463
+ ? ok([
464
+ assistantMessage({
465
+ sessionID: "session-1",
466
+ id: "msg-1",
467
+ parts: [
468
+ {
469
+ ...textPart("text-1", "msg-1", "Hello."),
470
+ time: {
471
+ start: 1,
472
+ end: 2,
473
+ },
474
+ },
475
+ ],
476
+ }),
477
+ ])
478
+ : ok([]),
479
+ }),
480
+ sessionID: "session-1",
481
+ thinking: true,
482
+ limits: () => ({}),
483
+ footer: ui.api,
484
+ })
485
+
486
+ try {
487
+ expect(ui.commits).toEqual([])
488
+ expect(ui.idleCalls).toBe(0)
489
+ } finally {
490
+ src.close()
491
+ await transport.close()
492
+ }
493
+ })
494
+
495
+ test("replays persisted main-session history during bootstrap when enabled", async () => {
496
+ const src = eventFeed()
497
+ const ui = footer()
498
+ const transport = await createSessionTransport({
499
+ sdk: sdk({
500
+ stream: src.stream,
501
+ messages: async ({ sessionID }) =>
502
+ sessionID === "session-1"
503
+ ? ok([
504
+ assistantMessage({
505
+ sessionID: "session-1",
506
+ id: "msg-1",
507
+ parts: [
508
+ {
509
+ ...textPart("text-1", "msg-1", "Hello."),
510
+ time: {
511
+ start: 1,
512
+ end: 2,
513
+ },
514
+ },
515
+ ],
516
+ }),
517
+ ])
518
+ : ok([]),
519
+ }),
520
+ sessionID: "session-1",
521
+ thinking: true,
522
+ replay: true,
523
+ limits: () => ({}),
524
+ footer: ui.api,
525
+ })
526
+
527
+ try {
528
+ await waitFor(() => ui.commits.find((item) => item.kind === "assistant" && item.text === "Hello."))
529
+ expect(ui.idleCalls).toBeGreaterThan(0)
530
+ } finally {
531
+ src.close()
532
+ await transport.close()
533
+ }
534
+ })
535
+
536
+ test("caps replayed bootstrap history to the configured number of messages", async () => {
537
+ const src = eventFeed()
538
+ const ui = footer()
539
+ const transport = await createSessionTransport({
540
+ sdk: sdk({
541
+ stream: src.stream,
542
+ messages: async ({ sessionID }) =>
543
+ ok(
544
+ sessionID === "session-1"
545
+ ? [
546
+ assistantMessage({
547
+ sessionID: "session-1",
548
+ id: "msg-1",
549
+ parts: [
550
+ {
551
+ ...textPart("text-1", "msg-1", "Hello."),
552
+ time: {
553
+ start: 1,
554
+ end: 2,
555
+ },
556
+ },
557
+ ],
558
+ }),
559
+ assistantMessage({
560
+ sessionID: "session-1",
561
+ id: "msg-2",
562
+ parts: [
563
+ {
564
+ ...textPart("text-2", "msg-2", "World."),
565
+ time: {
566
+ start: 3,
567
+ end: 4,
568
+ },
569
+ },
570
+ ],
571
+ }),
572
+ ]
573
+ : [],
574
+ ),
575
+ }),
576
+ sessionID: "session-1",
577
+ thinking: true,
578
+ replay: true,
579
+ replayLimit: 1,
580
+ limits: () => ({}),
581
+ footer: ui.api,
582
+ })
583
+
584
+ try {
585
+ await waitFor(() => (ui.commits.length > 0 ? ui.commits : undefined))
586
+ expect(ui.commits.filter((item) => item.kind === "assistant")).toEqual([
587
+ expect.objectContaining({
588
+ text: "World.",
589
+ }),
590
+ ])
591
+ } finally {
592
+ src.close()
593
+ await transport.close()
594
+ }
595
+ })
596
+
597
+ test("skips buffered pre-bootstrap deltas already covered by replay history", async () => {
598
+ const src = eventFeed()
599
+ const ui = footer()
600
+ const gate = defer<void>()
601
+ let transport: Awaited<ReturnType<typeof createSessionTransport>> | undefined
602
+ const task = createSessionTransport({
603
+ sdk: sdk({
604
+ stream: src.stream,
605
+ messages: async ({ sessionID }) => {
606
+ if (sessionID !== "session-1") {
607
+ return ok([])
608
+ }
609
+
610
+ await gate.promise
611
+ return ok([
612
+ assistantMessage({
613
+ sessionID: "session-1",
614
+ id: "msg-1",
615
+ parts: [textPart("text-1", "msg-1", "Hello")],
616
+ }),
617
+ ])
618
+ },
619
+ }),
620
+ sessionID: "session-1",
621
+ thinking: true,
622
+ replay: true,
623
+ limits: () => ({}),
624
+ footer: ui.api,
625
+ })
626
+
627
+ try {
628
+ await Promise.resolve()
629
+ src.push(textDelta("msg-1", "text-1", "lo"))
630
+ gate.resolve()
631
+ transport = await task
632
+
633
+ await waitFor(() => (ui.commits.length > 0 ? ui.commits : undefined))
634
+ await Bun.sleep(20)
635
+ expect(ui.commits.filter((item) => item.kind === "assistant")).toEqual([
636
+ expect.objectContaining({
637
+ text: "Hello",
638
+ }),
639
+ ])
640
+ } finally {
641
+ src.close()
642
+ await transport?.close()
643
+ }
644
+ })
645
+
646
+ test("applies buffered pre-bootstrap deltas not yet persisted", async () => {
647
+ const src = eventFeed()
648
+ const ui = footer()
649
+ const gate = defer<void>()
650
+ let transport: Awaited<ReturnType<typeof createSessionTransport>> | undefined
651
+ const task = createSessionTransport({
652
+ sdk: sdk({
653
+ stream: src.stream,
654
+ messages: async ({ sessionID }) => {
655
+ if (sessionID !== "session-1") {
656
+ return ok([])
657
+ }
658
+
659
+ await gate.promise
660
+ return ok([
661
+ assistantMessage({
662
+ sessionID: "session-1",
663
+ id: "msg-1",
664
+ parts: [textPart("text-1", "msg-1", "")],
665
+ }),
666
+ ])
667
+ },
668
+ }),
669
+ sessionID: "session-1",
670
+ thinking: true,
671
+ replay: true,
672
+ limits: () => ({}),
673
+ footer: ui.api,
674
+ })
675
+
676
+ try {
677
+ await Promise.resolve()
678
+ src.push(textDelta("msg-1", "text-1", "Hello"))
679
+ gate.resolve()
680
+ transport = await task
681
+
682
+ await waitFor(() => (ui.commits.length > 0 ? ui.commits : undefined))
683
+ await Bun.sleep(20)
684
+ expect(ui.commits.filter((item) => item.kind === "assistant")).toEqual([
685
+ expect.objectContaining({
686
+ text: "Hello",
687
+ }),
688
+ ])
689
+ } finally {
690
+ src.close()
691
+ await transport?.close()
692
+ }
693
+ })
694
+
695
+ test("preserves running footer state for resumed active sessions", async () => {
696
+ const src = eventFeed()
697
+ const ui = footer()
698
+ const transport = await createSessionTransport({
699
+ sdk: sdk({
700
+ stream: src.stream,
701
+ messages: async ({ sessionID }) =>
702
+ sessionID === "session-1"
703
+ ? ok([
704
+ assistantMessage({
705
+ sessionID: "session-1",
706
+ id: "msg-1",
707
+ parts: [
708
+ runningTool({
709
+ sessionID: "session-1",
710
+ messageID: "msg-1",
711
+ id: "bash-1",
712
+ callID: "call-1",
713
+ tool: "bash",
714
+ body: {
715
+ command: "pwd",
716
+ },
717
+ }),
718
+ ],
719
+ }),
720
+ ])
721
+ : ok([]),
722
+ }),
723
+ sessionID: "session-1",
724
+ thinking: true,
725
+ replay: true,
726
+ limits: () => ({}),
727
+ footer: ui.api,
728
+ })
729
+
730
+ try {
731
+ const patch = await waitFor(() => {
732
+ const item = ui.events.findLast((event) => event.type === "stream.patch")
733
+ return item?.type === "stream.patch" ? item.patch : undefined
734
+ })
735
+
736
+ expect(patch).toEqual(
737
+ expect.objectContaining({
738
+ phase: "running",
739
+ status: "running bash",
740
+ }),
741
+ )
742
+ } finally {
743
+ src.close()
744
+ await transport.close()
745
+ }
746
+ })
747
+
748
+ test("rebuilds session output on resize and continues live deltas from replayed state", async () => {
749
+ const src = eventFeed()
750
+ const ui = footer()
751
+ let calls = 0
752
+ const transport = await createSessionTransport({
753
+ sdk: sdk({
754
+ stream: src.stream,
755
+ messages: async () => {
756
+ calls += 1
757
+ if (calls === 1) {
758
+ return ok([])
759
+ }
760
+
761
+ return ok([
762
+ assistantMessage({
763
+ sessionID: "session-1",
764
+ id: "msg-1",
765
+ parts: [textPart("text-1", "msg-1", "Hello")],
766
+ }),
767
+ ])
768
+ },
769
+ }),
770
+ sessionID: "session-1",
771
+ thinking: true,
772
+ replay: true,
773
+ limits: () => ({}),
774
+ footer: ui.api,
775
+ })
776
+ const localRows: LocalReplayRow[] = [
777
+ { commit: { kind: "user", text: "pending prompt", phase: "start", source: "system", messageID: "msg-pending" } },
778
+ ]
779
+ const reset = mock(() => {
780
+ localRows.push({
781
+ commit: {
782
+ kind: "user",
783
+ text: "sent during reset",
784
+ phase: "start",
785
+ source: "system",
786
+ messageID: "msg-during-reset",
787
+ },
788
+ })
789
+ return Promise.resolve()
790
+ })
791
+
792
+ try {
793
+ expect(
794
+ await transport.replayOnResize({
795
+ localRows: () => localRows,
796
+ reset,
797
+ }),
798
+ ).toBe(true)
799
+ expect(reset).toHaveBeenCalledTimes(1)
800
+ expect(ui.commits).toEqual(
801
+ expect.arrayContaining([
802
+ expect.objectContaining({ kind: "assistant", text: "Hello" }),
803
+ expect.objectContaining({ kind: "user", text: "sent during reset", messageID: "msg-during-reset" }),
804
+ ]),
805
+ )
806
+
807
+ src.push(textUpdated(textPart("text-1", "msg-1", "Hello world")))
808
+ await waitFor(() => ui.commits.find((commit) => commit.kind === "assistant" && commit.text === " world"))
809
+ expect(ui.commits.filter((commit) => commit.kind === "assistant").map((commit) => commit.text)).toEqual([
810
+ "Hello",
811
+ " world",
812
+ ])
813
+ } finally {
814
+ src.close()
815
+ await transport.close()
816
+ }
817
+ })
818
+
819
+ test("coalesces active resize requests into one trailing replay", async () => {
820
+ const src = eventFeed()
821
+ const ui = footer()
822
+ const firstReset = defer()
823
+ const resetA = mock(() => firstReset.promise)
824
+ const resetB = mock(() => Promise.resolve())
825
+ const resetC = mock(() => Promise.resolve())
826
+ const transport = await createSessionTransport({
827
+ sdk: sdk({ stream: src.stream }),
828
+ sessionID: "session-1",
829
+ thinking: true,
830
+ replay: true,
831
+ limits: () => ({}),
832
+ footer: ui.api,
833
+ })
834
+
835
+ try {
836
+ const active = transport.replayOnResize({ localRows: () => [], reset: resetA })
837
+ await waitFor(() => (resetA.mock.calls.length === 1 ? true : undefined))
838
+
839
+ expect(await transport.replayOnResize({ localRows: () => [], reset: resetB })).toBe(false)
840
+ expect(await transport.replayOnResize({ localRows: () => [], reset: resetC })).toBe(false)
841
+ expect(resetB).not.toHaveBeenCalled()
842
+
843
+ firstReset.resolve()
844
+ expect(await active).toBe(true)
845
+ expect(resetA).toHaveBeenCalledTimes(1)
846
+ expect(resetB).not.toHaveBeenCalled()
847
+ expect(resetC).toHaveBeenCalledTimes(1)
848
+ } finally {
849
+ src.close()
850
+ await transport.close()
851
+ }
852
+ })
853
+
854
+ test("keeps coalescing resize requests while buffered events drain", async () => {
855
+ const src = eventFeed()
856
+ const ui = footer()
857
+ const firstReset = defer()
858
+ const statusGate = defer()
859
+ const statusStarted = defer()
860
+ let blockStatus = false
861
+ const trace = mock((_type: string, _data?: unknown) => {})
862
+ const resetA = mock(() => firstReset.promise)
863
+ const resetB = mock(() => Promise.resolve())
864
+ const resetC = mock(() => Promise.resolve())
865
+ const transport = await createSessionTransport({
866
+ sdk: sdk({
867
+ stream: src.stream,
868
+ status: async () => {
869
+ if (blockStatus) {
870
+ statusStarted.resolve()
871
+ await statusGate.promise
872
+ }
873
+ return ok(statusMap(true))
874
+ },
875
+ }),
876
+ sessionID: "session-1",
877
+ thinking: true,
878
+ replay: true,
879
+ limits: () => ({}),
880
+ footer: ui.api,
881
+ trace: { write: trace },
882
+ })
883
+ const turn = transport.runPromptTurn({
884
+ agent: undefined,
885
+ model: undefined,
886
+ variant: undefined,
887
+ prompt: { text: "active", parts: [] },
888
+ files: [],
889
+ includeFiles: false,
890
+ })
891
+
892
+ try {
893
+ await waitFor(() => ui.events.find((event) => event.type === "turn.wait"))
894
+ const active = transport.replayOnResize({ localRows: () => [], reset: resetA })
895
+ await waitFor(() => (resetA.mock.calls.length === 1 ? true : undefined))
896
+ blockStatus = true
897
+ src.push(busy())
898
+ src.push(idle())
899
+ await waitFor(() => (trace.mock.calls.filter((call) => call[0] === "recv.event").length >= 2 ? true : undefined))
900
+
901
+ expect(await transport.replayOnResize({ localRows: () => [], reset: resetB })).toBe(false)
902
+ firstReset.resolve()
903
+ await Promise.race([
904
+ statusStarted.promise,
905
+ Bun.sleep(1_000).then(() => {
906
+ throw new Error("timed out waiting for buffered status drain")
907
+ }),
908
+ ])
909
+
910
+ expect(await transport.replayOnResize({ localRows: () => [], reset: resetC })).toBe(false)
911
+ expect(resetC).not.toHaveBeenCalled()
912
+ blockStatus = false
913
+ statusGate.resolve()
914
+
915
+ expect(
916
+ await Promise.race([
917
+ active,
918
+ Bun.sleep(1_000).then(() => {
919
+ throw new Error("timed out waiting for trailing resize replay")
920
+ }),
921
+ ]),
922
+ ).toBe(true)
923
+ expect(resetB).not.toHaveBeenCalled()
924
+ expect(resetC).toHaveBeenCalledTimes(1)
925
+ } finally {
926
+ src.close()
927
+ await transport.close()
928
+ await turn
929
+ }
930
+ })
931
+
932
+ test("preserves assistant deltas not yet persisted when replaying during a live stream", async () => {
933
+ const src = eventFeed()
934
+ const ui = footer()
935
+ let calls = 0
936
+ const transport = await createSessionTransport({
937
+ sdk: sdk({
938
+ stream: src.stream,
939
+ messages: async () => {
940
+ calls += 1
941
+ if (calls === 1) {
942
+ return ok([])
943
+ }
944
+
945
+ return ok([
946
+ assistantMessage({
947
+ sessionID: "session-1",
948
+ id: "msg-live",
949
+ parts: [textPart("text-live", "msg-live", "")],
950
+ }),
951
+ ])
952
+ },
953
+ }),
954
+ sessionID: "session-1",
955
+ thinking: true,
956
+ replay: true,
957
+ limits: () => ({}),
958
+ footer: ui.api,
959
+ })
960
+
961
+ try {
962
+ src.push(assistant("msg-live"))
963
+ src.push(textUpdated(textPart("text-live", "msg-live", "")))
964
+ src.push(textDelta("msg-live", "text-live", "Hello"))
965
+ await waitFor(() => ui.commits.find((commit) => commit.kind === "assistant" && commit.text === "Hello"))
966
+ ui.commits.length = 0
967
+
968
+ expect(await transport.replayOnResize({ localRows: () => [], reset: () => Promise.resolve() })).toBe(true)
969
+ src.push(textDelta("msg-live", "text-live", "Hello"))
970
+ src.push(
971
+ textUpdated({
972
+ ...textPart("text-live", "msg-live", "HelloHello"),
973
+ time: { start: 1, end: 2 },
974
+ }),
975
+ )
976
+
977
+ await waitFor(() =>
978
+ ui.commits.filter((commit) => commit.kind === "assistant" && commit.text === "Hello").length === 2
979
+ ? true
980
+ : undefined,
981
+ )
982
+ expect(
983
+ ui.commits.filter((commit) => commit.kind === "assistant" && commit.text).map((commit) => commit.text),
984
+ ).toEqual(["Hello", "Hello"])
985
+ } finally {
986
+ src.close()
987
+ await transport.close()
988
+ }
989
+ })
990
+
991
+ test("preserves the display prefix for active reasoning restored during replay", async () => {
992
+ const src = eventFeed()
993
+ const ui = footer()
994
+ let calls = 0
995
+ const transport = await createSessionTransport({
996
+ sdk: sdk({
997
+ stream: src.stream,
998
+ messages: async () => {
999
+ calls += 1
1000
+ if (calls === 1) {
1001
+ return ok([])
1002
+ }
1003
+
1004
+ return ok([
1005
+ assistantMessage({
1006
+ sessionID: "session-1",
1007
+ id: "msg-thinking",
1008
+ parts: [reasoningPart("thinking-1", "msg-thinking", "")],
1009
+ }),
1010
+ ])
1011
+ },
1012
+ }),
1013
+ sessionID: "session-1",
1014
+ thinking: true,
1015
+ replay: true,
1016
+ limits: () => ({}),
1017
+ footer: ui.api,
1018
+ })
1019
+
1020
+ try {
1021
+ src.push(assistant("msg-thinking"))
1022
+ src.push(reasoningUpdated(reasoningPart("thinking-1", "msg-thinking", "")))
1023
+ src.push(textDelta("msg-thinking", "thinking-1", "plan"))
1024
+ await waitFor(() => ui.commits.find((commit) => commit.kind === "reasoning" && commit.text === "Thinking: plan"))
1025
+ ui.commits.length = 0
1026
+
1027
+ expect(await transport.replayOnResize({ localRows: () => [], reset: () => Promise.resolve() })).toBe(true)
1028
+ expect(ui.commits.filter((commit) => commit.kind === "reasoning").map((commit) => commit.text)).toEqual([
1029
+ "Thinking: plan",
1030
+ ])
1031
+ } finally {
1032
+ src.close()
1033
+ await transport.close()
1034
+ }
1035
+ })
1036
+
1037
+ test("does not overlay stale active text when persistence completes during replay", async () => {
1038
+ const src = eventFeed()
1039
+ const ui = footer()
1040
+ let calls = 0
1041
+ const transport = await createSessionTransport({
1042
+ sdk: sdk({
1043
+ stream: src.stream,
1044
+ messages: async () => {
1045
+ calls += 1
1046
+ if (calls === 1) {
1047
+ return ok([])
1048
+ }
1049
+
1050
+ return ok([
1051
+ assistantMessage({
1052
+ sessionID: "session-1",
1053
+ id: "msg-finished",
1054
+ parts: [
1055
+ {
1056
+ ...textPart("text-finished", "msg-finished", "Hello"),
1057
+ time: { start: 1, end: 2 },
1058
+ },
1059
+ ],
1060
+ }),
1061
+ ])
1062
+ },
1063
+ }),
1064
+ sessionID: "session-1",
1065
+ thinking: true,
1066
+ replay: true,
1067
+ limits: () => ({}),
1068
+ footer: ui.api,
1069
+ })
1070
+
1071
+ try {
1072
+ src.push(assistant("msg-finished"))
1073
+ src.push(textUpdated(textPart("text-finished", "msg-finished", "")))
1074
+ src.push(textDelta("msg-finished", "text-finished", "Hello"))
1075
+ await waitFor(() => ui.commits.find((commit) => commit.kind === "assistant" && commit.text === "Hello"))
1076
+ ui.commits.length = 0
1077
+
1078
+ expect(await transport.replayOnResize({ localRows: () => [], reset: () => Promise.resolve() })).toBe(true)
1079
+ expect(
1080
+ ui.commits.filter((commit) => commit.kind === "assistant" && commit.text).map((commit) => commit.text),
1081
+ ).toEqual(["Hello"])
1082
+ } finally {
1083
+ src.close()
1084
+ await transport.close()
1085
+ }
1086
+ })
1087
+
1088
+ test("does not clear the terminal when resize replay snapshot fetch fails", async () => {
1089
+ const src = eventFeed()
1090
+ const ui = footer()
1091
+ let calls = 0
1092
+ const transport = await createSessionTransport({
1093
+ sdk: sdk({
1094
+ stream: src.stream,
1095
+ messages: async () => {
1096
+ calls += 1
1097
+ if (calls === 1) {
1098
+ return ok([])
1099
+ }
1100
+
1101
+ throw new Error("snapshot failed")
1102
+ },
1103
+ }),
1104
+ sessionID: "session-1",
1105
+ thinking: true,
1106
+ replay: true,
1107
+ limits: () => ({}),
1108
+ footer: ui.api,
1109
+ })
1110
+ const reset = mock(() => Promise.resolve())
1111
+
1112
+ try {
1113
+ expect(await transport.replayOnResize({ localRows: () => [], reset })).toBe(false)
1114
+ expect(reset).not.toHaveBeenCalled()
1115
+ expect(ui.commits).toEqual([])
1116
+ } finally {
1117
+ src.close()
1118
+ await transport.close()
1119
+ }
1120
+ })
1121
+
1122
+ test("disables resize replay for the session after terminal reset fails", async () => {
1123
+ const src = eventFeed()
1124
+ const ui = footer()
1125
+ const transport = await createSessionTransport({
1126
+ sdk: sdk({ stream: src.stream }),
1127
+ sessionID: "session-1",
1128
+ thinking: true,
1129
+ replay: true,
1130
+ limits: () => ({}),
1131
+ footer: ui.api,
1132
+ })
1133
+ const reset = mock(() => Promise.reject(new Error("clear failed")))
1134
+
1135
+ try {
1136
+ expect(await transport.replayOnResize({ localRows: () => [], reset })).toBe(false)
1137
+ expect(await transport.replayOnResize({ localRows: () => [], reset })).toBe(false)
1138
+ expect(reset).toHaveBeenCalledTimes(1)
1139
+ expect(ui.commits).toContainEqual({
1140
+ kind: "error",
1141
+ text: "resize replay failed; disabled for this session",
1142
+ phase: "start",
1143
+ source: "system",
1144
+ })
1145
+ } finally {
1146
+ src.close()
1147
+ await transport.close()
1148
+ }
1149
+ })
1150
+
1151
+ test("disables resize replay when rebuilding scrollback fails after terminal reset", async () => {
1152
+ const src = eventFeed()
1153
+ const ui = footer()
1154
+ let cleared = false
1155
+ const idle = ui.api.idle
1156
+ ui.api.idle = () => (cleared ? Promise.reject(new Error("render failed")) : idle())
1157
+ const transport = await createSessionTransport({
1158
+ sdk: sdk({ stream: src.stream }),
1159
+ sessionID: "session-1",
1160
+ thinking: true,
1161
+ replay: true,
1162
+ limits: () => ({}),
1163
+ footer: ui.api,
1164
+ })
1165
+ const reset = mock(() => {
1166
+ cleared = true
1167
+ return Promise.resolve()
1168
+ })
1169
+
1170
+ try {
1171
+ expect(await transport.replayOnResize({ localRows: () => [], reset })).toBe(false)
1172
+ expect(await transport.replayOnResize({ localRows: () => [], reset })).toBe(false)
1173
+ expect(reset).toHaveBeenCalledTimes(1)
1174
+ expect(ui.commits).toContainEqual({
1175
+ kind: "error",
1176
+ text: "resize replay failed; disabled for this session",
1177
+ phase: "start",
1178
+ source: "system",
1179
+ })
1180
+ } finally {
1181
+ src.close()
1182
+ await transport.close()
1183
+ }
1184
+ })
1185
+
1186
+ test("drops completed historical subagent tabs during bootstrap", async () => {
1187
+ const src = eventFeed()
1188
+ const ui = footer()
1189
+ const transport = await createSessionTransport({
1190
+ sdk: sdk({
1191
+ stream: src.stream,
1192
+ messages: async ({ sessionID }) => {
1193
+ if (sessionID !== "session-1") {
1194
+ return ok([])
1195
+ }
1196
+
1197
+ return ok([
1198
+ assistantMessage({
1199
+ sessionID: "session-1",
1200
+ id: "msg-1",
1201
+ parts: [
1202
+ completedTool({
1203
+ sessionID: "session-1",
1204
+ messageID: "msg-1",
1205
+ id: "task-1",
1206
+ callID: "call-1",
1207
+ tool: "task",
1208
+ body: {
1209
+ description: "Explore run folder",
1210
+ subagent_type: "explore",
1211
+ },
1212
+ metadata: {
1213
+ sessionId: "child-1",
1214
+ },
1215
+ }),
1216
+ ],
1217
+ }),
1218
+ ])
1219
+ },
1220
+ children: async () => ok([child("child-1")]),
1221
+ }),
1222
+ sessionID: "session-1",
1223
+ thinking: true,
1224
+ limits: () => ({}),
1225
+ footer: ui.api,
1226
+ })
1227
+
1228
+ try {
1229
+ const state = await waitFor(() => {
1230
+ const item = ui.events.findLast((event) => event.type === "stream.subagent")
1231
+ return item?.type === "stream.subagent" ? item.state : undefined
1232
+ })
1233
+
1234
+ expect(state.tabs).toEqual([])
1235
+ expect(state.details).toEqual({})
1236
+ } finally {
1237
+ src.close()
1238
+ await transport.close()
1239
+ }
1240
+ })
1241
+
1242
+ test("bootstraps child tabs and resumed blocker input", async () => {
1243
+ const src = eventFeed()
1244
+ const ui = footer()
1245
+ const transport = await createSessionTransport({
1246
+ sdk: sdk({
1247
+ stream: src.stream,
1248
+ messages: async ({ sessionID }) => {
1249
+ if (sessionID === "session-1") {
1250
+ return ok([
1251
+ assistantMessage({
1252
+ sessionID: "session-1",
1253
+ id: "msg-1",
1254
+ parts: [
1255
+ runningTool({
1256
+ sessionID: "session-1",
1257
+ messageID: "msg-1",
1258
+ id: "task-1",
1259
+ callID: "call-1",
1260
+ tool: "task",
1261
+ body: {
1262
+ description: "Explore run folder",
1263
+ subagent_type: "explore",
1264
+ },
1265
+ metadata: {
1266
+ sessionId: "child-1",
1267
+ },
1268
+ }),
1269
+ ],
1270
+ }),
1271
+ ])
1272
+ }
1273
+
1274
+ return ok([
1275
+ assistantMessage({
1276
+ sessionID: "child-1",
1277
+ id: "msg-child-1",
1278
+ parts: [
1279
+ runningTool({
1280
+ sessionID: "child-1",
1281
+ messageID: "msg-child-1",
1282
+ id: "edit-1",
1283
+ callID: "call-edit-1",
1284
+ tool: "edit",
1285
+ body: {
1286
+ filePath: "src/run/subagent-data.ts",
1287
+ diff: "@@ -1 +1 @@",
1288
+ },
1289
+ }),
1290
+ ],
1291
+ }),
1292
+ ])
1293
+ },
1294
+ children: async () => ok([child("child-1")]),
1295
+ permissions: async () =>
1296
+ ok([
1297
+ {
1298
+ id: "perm-1",
1299
+ sessionID: "child-1",
1300
+ permission: "edit",
1301
+ patterns: ["src/run/subagent-data.ts"],
1302
+ metadata: {},
1303
+ always: [],
1304
+ tool: {
1305
+ messageID: "msg-child-1",
1306
+ callID: "call-edit-1",
1307
+ },
1308
+ },
1309
+ ]),
1310
+ }),
1311
+ sessionID: "session-1",
1312
+ thinking: true,
1313
+ limits: () => ({}),
1314
+ footer: ui.api,
1315
+ })
1316
+
1317
+ try {
1318
+ const boot = await waitFor(() => {
1319
+ const item = ui.events.findLast((event) => event.type === "stream.subagent")
1320
+ const state = item?.type === "stream.subagent" ? item.state : undefined
1321
+ return state?.tabs.some((tab) => tab.sessionID === "child-1") &&
1322
+ state.permissions.some((req) => req.id === "perm-1")
1323
+ ? state
1324
+ : undefined
1325
+ })
1326
+
1327
+ expect(boot.tabs).toEqual([
1328
+ expect.objectContaining({
1329
+ sessionID: "child-1",
1330
+ label: "Explore",
1331
+ description: "Pending permission",
1332
+ status: "running",
1333
+ }),
1334
+ ])
1335
+ expect(boot.permissions).toEqual([
1336
+ expect.objectContaining({
1337
+ id: "perm-1",
1338
+ sessionID: "child-1",
1339
+ metadata: {
1340
+ input: {
1341
+ filePath: "src/run/subagent-data.ts",
1342
+ diff: "@@ -1 +1 @@",
1343
+ },
1344
+ },
1345
+ }),
1346
+ ])
1347
+
1348
+ transport.selectSubagent("child-1")
1349
+
1350
+ const selected = await waitFor(() => {
1351
+ const item = ui.events.findLast((event) => event.type === "stream.subagent")
1352
+ const state = item?.type === "stream.subagent" ? item.state : undefined
1353
+ const detail = state?.details["child-1"]
1354
+ return detail?.commits.some(
1355
+ (commit) => commit.kind === "tool" && commit.tool === "edit" && commit.phase === "start",
1356
+ )
1357
+ ? state
1358
+ : undefined
1359
+ })
1360
+
1361
+ expect(selected.details).toEqual({
1362
+ "child-1": {
1363
+ sessionID: "child-1",
1364
+ commits: [
1365
+ expect.objectContaining({
1366
+ kind: "tool",
1367
+ tool: "edit",
1368
+ phase: "start",
1369
+ }),
1370
+ ],
1371
+ },
1372
+ })
1373
+
1374
+ expect(
1375
+ await waitFor(() => {
1376
+ const item = ui.events.findLast((event) => event.type === "stream.view")
1377
+ return item?.type === "stream.view" && item.view.type === "permission" && item.view.request.id === "perm-1"
1378
+ ? item
1379
+ : undefined
1380
+ }),
1381
+ ).toEqual({
1382
+ type: "stream.view",
1383
+ view: {
1384
+ type: "permission",
1385
+ request: expect.objectContaining({
1386
+ id: "perm-1",
1387
+ metadata: {
1388
+ input: {
1389
+ filePath: "src/run/subagent-data.ts",
1390
+ diff: "@@ -1 +1 @@",
1391
+ },
1392
+ },
1393
+ }),
1394
+ },
1395
+ })
1396
+ } finally {
1397
+ src.close()
1398
+ await transport.close()
1399
+ }
1400
+ })
1401
+
1402
+ test("bootstraps child session output before selection", async () => {
1403
+ const ui = footer()
1404
+ const transport = await createSessionTransport({
1405
+ sdk: sdk({
1406
+ messages: async ({ sessionID }) => {
1407
+ if (sessionID === "session-1") {
1408
+ return ok([
1409
+ assistantMessage({
1410
+ sessionID: "session-1",
1411
+ id: "msg-1",
1412
+ parts: [
1413
+ runningTool({
1414
+ sessionID: "session-1",
1415
+ messageID: "msg-1",
1416
+ id: "task-1",
1417
+ callID: "call-1",
1418
+ tool: "task",
1419
+ body: {
1420
+ description: "Explore run.ts",
1421
+ subagent_type: "explore",
1422
+ },
1423
+ metadata: {
1424
+ sessionId: "child-1",
1425
+ },
1426
+ }),
1427
+ ],
1428
+ }),
1429
+ ])
1430
+ }
1431
+
1432
+ return sessionID === "child-1"
1433
+ ? ok([
1434
+ assistantMessage({
1435
+ sessionID: "child-1",
1436
+ id: "msg-child-1",
1437
+ parts: [textPart("txt-child-1", "msg-child-1", "subagent summary", "child-1")],
1438
+ }),
1439
+ ])
1440
+ : ok([])
1441
+ },
1442
+ children: async () => ok([child("child-1")]),
1443
+ }),
1444
+ sessionID: "session-1",
1445
+ thinking: true,
1446
+ limits: () => ({}),
1447
+ footer: ui.api,
1448
+ })
1449
+
1450
+ try {
1451
+ await waitFor(() => {
1452
+ const item = ui.events.findLast((event) => event.type === "stream.subagent")
1453
+ return item?.type === "stream.subagent" && item.state.tabs.some((tab) => tab.sessionID === "child-1")
1454
+ ? item
1455
+ : undefined
1456
+ })
1457
+
1458
+ transport.selectSubagent("child-1")
1459
+
1460
+ expect(
1461
+ await waitFor(() => {
1462
+ const item = ui.events.findLast((event) => event.type === "stream.subagent")
1463
+ const detail = item?.type === "stream.subagent" ? item.state.details["child-1"] : undefined
1464
+ return detail?.commits.some((commit) => commit.kind === "assistant" && commit.text === "subagent summary")
1465
+ ? detail
1466
+ : undefined
1467
+ }),
1468
+ ).toEqual({
1469
+ sessionID: "child-1",
1470
+ commits: [
1471
+ expect.objectContaining({
1472
+ kind: "assistant",
1473
+ text: "subagent summary",
1474
+ }),
1475
+ ],
1476
+ })
1477
+ } finally {
1478
+ await transport.close()
1479
+ }
1480
+ })
1481
+
1482
+ test("does not block startup on child history bootstrap", async () => {
1483
+ const pending = defer<Awaited<ReturnType<typeof ok<SessionMessage[]>>>>()
1484
+ const ui = footer()
1485
+ let transport: Awaited<ReturnType<typeof createSessionTransport>> | undefined
1486
+
1487
+ const task = createSessionTransport({
1488
+ sdk: sdk({
1489
+ messages: async ({ sessionID }) => {
1490
+ if (sessionID === "session-1") {
1491
+ return ok([
1492
+ assistantMessage({
1493
+ sessionID: "session-1",
1494
+ id: "msg-1",
1495
+ parts: [
1496
+ runningTool({
1497
+ sessionID: "session-1",
1498
+ messageID: "msg-1",
1499
+ id: "task-1",
1500
+ callID: "call-1",
1501
+ tool: "task",
1502
+ body: {
1503
+ description: "Explore run.ts",
1504
+ subagent_type: "explore",
1505
+ },
1506
+ metadata: {
1507
+ sessionId: "child-1",
1508
+ },
1509
+ }),
1510
+ ],
1511
+ }),
1512
+ ])
1513
+ }
1514
+
1515
+ if (sessionID === "child-1") {
1516
+ return pending.promise
1517
+ }
1518
+
1519
+ return ok([])
1520
+ },
1521
+ children: async () => ok([child("child-1")]),
1522
+ }),
1523
+ sessionID: "session-1",
1524
+ thinking: true,
1525
+ limits: () => ({}),
1526
+ footer: ui.api,
1527
+ }).then((item) => {
1528
+ transport = item
1529
+ return item
1530
+ })
1531
+
1532
+ try {
1533
+ const state = await waitFor(() => {
1534
+ const item = ui.events.findLast((event) => event.type === "stream.subagent")
1535
+ return item?.type === "stream.subagent" && item.state.tabs.some((tab) => tab.sessionID === "child-1")
1536
+ ? item.state
1537
+ : undefined
1538
+ })
1539
+
1540
+ await waitFor(() => transport)
1541
+
1542
+ expect(state).toEqual({
1543
+ tabs: [expect.objectContaining({ sessionID: "child-1", status: "running" })],
1544
+ details: {},
1545
+ permissions: [],
1546
+ questions: [],
1547
+ })
1548
+ } finally {
1549
+ pending.resolve(ok([]))
1550
+ await task
1551
+ await transport?.close()
1552
+ }
1553
+ })
1554
+
1555
+ test("replays child events buffered during bootstrap once the tab is known", async () => {
1556
+ const global = globalFeed()
1557
+ const ui = footer()
1558
+ const gate = defer<void>()
1559
+ let transport: Awaited<ReturnType<typeof createSessionTransport>> | undefined
1560
+ const task = createSessionTransport({
1561
+ sdk: sdk({
1562
+ globalStream: global.stream,
1563
+ messages: async ({ sessionID }) => {
1564
+ if (sessionID !== "session-1") {
1565
+ return ok([])
1566
+ }
1567
+
1568
+ await gate.promise
1569
+ return ok([])
1570
+ },
1571
+ children: async () => ok([]),
1572
+ }),
1573
+ sessionID: "session-1",
1574
+ thinking: true,
1575
+ limits: () => ({}),
1576
+ footer: ui.api,
1577
+ })
1578
+
1579
+ try {
1580
+ await Promise.resolve()
1581
+ global.push(globalEvent(retry("child-1", 1, "retry child")))
1582
+ global.push(
1583
+ globalEvent({
1584
+ id: "evt-child-message",
1585
+ type: "message.updated",
1586
+ properties: {
1587
+ sessionID: "child-1",
1588
+ info: assistantMessage({
1589
+ sessionID: "child-1",
1590
+ id: "msg-child-1",
1591
+ parts: [],
1592
+ }).info,
1593
+ },
1594
+ }),
1595
+ )
1596
+ global.push(globalEvent(textUpdated(textPart("txt-child-1", "msg-child-1", "", "child-1"))))
1597
+ global.push(globalEvent(textDelta("msg-child-1", "txt-child-1", "Hello", "child-1")))
1598
+ global.push(
1599
+ globalEvent(
1600
+ toolUpdated(
1601
+ runningTool({
1602
+ sessionID: "session-1",
1603
+ messageID: "msg-1",
1604
+ id: "task-1",
1605
+ callID: "call-1",
1606
+ tool: "task",
1607
+ body: {
1608
+ description: "Explore run.ts",
1609
+ subagent_type: "explore",
1610
+ },
1611
+ metadata: {
1612
+ sessionId: "child-1",
1613
+ },
1614
+ }),
1615
+ ),
1616
+ ),
1617
+ )
1618
+ gate.resolve()
1619
+ transport = await task
1620
+
1621
+ await waitFor(() => {
1622
+ const item = ui.events.findLast((event) => event.type === "stream.subagent")
1623
+ return item?.type === "stream.subagent" && item.state.tabs.some((tab) => tab.sessionID === "child-1")
1624
+ ? item
1625
+ : undefined
1626
+ })
1627
+
1628
+ transport.selectSubagent("child-1")
1629
+
1630
+ const detail = await waitFor(() => {
1631
+ const item = ui.events.findLast((event) => event.type === "stream.subagent")
1632
+ const next = item?.type === "stream.subagent" ? item.state.details["child-1"] : undefined
1633
+ return next?.commits.some((commit) => commit.kind === "error" && commit.text === "retry child") &&
1634
+ next.commits.some((commit) => commit.kind === "assistant" && commit.text === "Hello")
1635
+ ? next
1636
+ : undefined
1637
+ })
1638
+
1639
+ expect(detail).toEqual({
1640
+ sessionID: "child-1",
1641
+ commits: expect.arrayContaining([
1642
+ expect.objectContaining({
1643
+ kind: "error",
1644
+ text: "retry child",
1645
+ }),
1646
+ expect.objectContaining({
1647
+ kind: "assistant",
1648
+ text: "Hello",
1649
+ }),
1650
+ ]),
1651
+ })
1652
+ } finally {
1653
+ global.close()
1654
+ await transport?.close()
1655
+ }
1656
+ })
1657
+
1658
+ test("streams selected subagent output from global events while it is running", async () => {
1659
+ const global = globalFeed()
1660
+ const ui = footer()
1661
+ const transport = await createSessionTransport({
1662
+ sdk: sdk({
1663
+ globalStream: global.stream,
1664
+ }),
1665
+ sessionID: "session-1",
1666
+ thinking: true,
1667
+ limits: () => ({}),
1668
+ footer: ui.api,
1669
+ })
1670
+
1671
+ try {
1672
+ global.push(globalEvent(assistant("msg-1")))
1673
+ global.push(
1674
+ globalEvent(
1675
+ toolUpdated(
1676
+ runningTool({
1677
+ sessionID: "session-1",
1678
+ messageID: "msg-1",
1679
+ id: "task-1",
1680
+ callID: "call-1",
1681
+ tool: "task",
1682
+ body: {
1683
+ description: "Explore run.ts",
1684
+ subagent_type: "explore",
1685
+ },
1686
+ metadata: {
1687
+ sessionId: "child-1",
1688
+ },
1689
+ }),
1690
+ ),
1691
+ ),
1692
+ )
1693
+
1694
+ await waitFor(() => {
1695
+ const item = ui.events.findLast((event) => event.type === "stream.subagent")
1696
+ return item?.type === "stream.subagent" && item.state.tabs.some((tab) => tab.sessionID === "child-1")
1697
+ ? item
1698
+ : undefined
1699
+ })
1700
+
1701
+ transport.selectSubagent("child-1")
1702
+
1703
+ global.push(
1704
+ globalEvent({
1705
+ id: "evt-child-message",
1706
+ type: "message.updated",
1707
+ properties: {
1708
+ sessionID: "child-1",
1709
+ info: assistantMessage({
1710
+ sessionID: "child-1",
1711
+ id: "msg-child-1",
1712
+ parts: [],
1713
+ }).info,
1714
+ },
1715
+ }),
1716
+ )
1717
+ global.push(globalEvent(textUpdated(textPart("txt-child-1", "msg-child-1", "hello", "child-1"))))
1718
+
1719
+ expect(
1720
+ await waitFor(() => {
1721
+ const item = ui.events.findLast((event) => event.type === "stream.subagent")
1722
+ const detail = item?.type === "stream.subagent" ? item.state.details["child-1"] : undefined
1723
+ return detail?.commits.some((commit) => commit.kind === "assistant" && commit.text === "hello")
1724
+ ? detail
1725
+ : undefined
1726
+ }),
1727
+ ).toEqual({
1728
+ sessionID: "child-1",
1729
+ commits: [
1730
+ expect.objectContaining({
1731
+ kind: "assistant",
1732
+ text: "hello",
1733
+ }),
1734
+ ],
1735
+ })
1736
+
1737
+ global.push(globalEvent(textUpdated(textPart("txt-child-1", "msg-child-1", "hello world", "child-1"))))
1738
+
1739
+ expect(
1740
+ await waitFor(() => {
1741
+ const item = ui.events.findLast((event) => event.type === "stream.subagent")
1742
+ const detail = item?.type === "stream.subagent" ? item.state.details["child-1"] : undefined
1743
+ return detail?.commits.some((commit) => commit.kind === "assistant" && commit.text === "hello world")
1744
+ ? detail
1745
+ : undefined
1746
+ }, 2_000),
1747
+ ).toEqual({
1748
+ sessionID: "child-1",
1749
+ commits: [
1750
+ expect.objectContaining({
1751
+ kind: "assistant",
1752
+ text: "hello world",
1753
+ }),
1754
+ ],
1755
+ })
1756
+ } finally {
1757
+ global.close()
1758
+ await transport.close()
1759
+ }
1760
+ })
1761
+
1762
+ test("recovers pending questions from question.list when question.asked is missed", async () => {
1763
+ const src = eventFeed()
1764
+ const ui = footer()
1765
+ let questionCalls = 0
1766
+ const request = {
1767
+ id: "question-1",
1768
+ sessionID: "session-1",
1769
+ questions: [
1770
+ {
1771
+ question: "Which area should I inspect first?",
1772
+ header: "Area",
1773
+ options: [{ label: "CLI", description: "Look at the direct run flow." }],
1774
+ multiple: false,
1775
+ },
1776
+ ],
1777
+ tool: {
1778
+ messageID: "msg-1",
1779
+ callID: "call-question-1",
1780
+ },
1781
+ }
1782
+ const transport = await createSessionTransport({
1783
+ sdk: sdk({
1784
+ stream: src.stream,
1785
+ questions: async () => {
1786
+ questionCalls += 1
1787
+ return ok(questionCalls > 1 ? [request] : [])
1788
+ },
1789
+ promptAsync: async () => {
1790
+ queueMicrotask(() => {
1791
+ src.push(busy())
1792
+ src.push(assistant("msg-1"))
1793
+ src.push(
1794
+ toolUpdated(
1795
+ runningTool({
1796
+ sessionID: "session-1",
1797
+ messageID: "msg-1",
1798
+ id: "question-tool-1",
1799
+ callID: "call-question-1",
1800
+ tool: "question",
1801
+ body: {
1802
+ questions: request.questions,
1803
+ },
1804
+ }),
1805
+ ),
1806
+ )
1807
+ })
1808
+ return ok(undefined)
1809
+ },
1810
+ }),
1811
+ sessionID: "session-1",
1812
+ thinking: true,
1813
+ limits: () => ({}),
1814
+ footer: ui.api,
1815
+ })
1816
+
1817
+ const ctrl = new AbortController()
1818
+
1819
+ try {
1820
+ const run = transport.runPromptTurn({
1821
+ agent: undefined,
1822
+ model: undefined,
1823
+ variant: undefined,
1824
+ prompt: { text: "hello", parts: [] },
1825
+ files: [],
1826
+ includeFiles: false,
1827
+ signal: ctrl.signal,
1828
+ })
1829
+
1830
+ const view = await waitFor(() => {
1831
+ const item = ui.events.findLast((event) => event.type === "stream.view")
1832
+ return item?.type === "stream.view" && item.view.type === "question" ? item.view : undefined
1833
+ })
1834
+
1835
+ expect(view).toEqual({
1836
+ type: "question",
1837
+ request,
1838
+ })
1839
+
1840
+ expect(ui.events).toContainEqual({
1841
+ type: "stream.patch",
1842
+ patch: {
1843
+ phase: "running",
1844
+ status: "awaiting answer",
1845
+ },
1846
+ })
1847
+
1848
+ src.push(
1849
+ toolUpdated(
1850
+ completedTool({
1851
+ sessionID: "session-1",
1852
+ messageID: "msg-1",
1853
+ id: "question-tool-1",
1854
+ callID: "call-question-1",
1855
+ tool: "question",
1856
+ body: {
1857
+ questions: request.questions,
1858
+ },
1859
+ output: "User has answered your questions.",
1860
+ metadata: {
1861
+ answers: [["CLI"]],
1862
+ },
1863
+ }),
1864
+ ),
1865
+ )
1866
+
1867
+ expect(
1868
+ await waitFor(() => {
1869
+ const item = ui.events.findLast((event) => event.type === "stream.view")
1870
+ return item?.type === "stream.view" && item.view.type === "prompt" ? item : undefined
1871
+ }),
1872
+ ).toEqual({
1873
+ type: "stream.view",
1874
+ view: { type: "prompt" },
1875
+ })
1876
+
1877
+ ctrl.abort()
1878
+ await run
1879
+ } finally {
1880
+ src.close()
1881
+ await transport.close()
1882
+ }
1883
+ })
1884
+
1885
+ test("does not resurrect questions if question.list resolves after tool completion", async () => {
1886
+ const src = eventFeed()
1887
+ const ui = footer()
1888
+ const started = defer()
1889
+ const request = {
1890
+ id: "question-race-1",
1891
+ sessionID: "session-1",
1892
+ questions: [
1893
+ {
1894
+ question: "Which area should I inspect first?",
1895
+ header: "Area",
1896
+ options: [{ label: "CLI", description: "Look at the direct run flow." }],
1897
+ multiple: false,
1898
+ },
1899
+ ],
1900
+ tool: {
1901
+ messageID: "msg-1",
1902
+ callID: "call-question-race-1",
1903
+ },
1904
+ }
1905
+ const pending = defer<Awaited<ReturnType<typeof ok<(typeof request)[]>>>>()
1906
+ let questionCalls = 0
1907
+ const transport = await createSessionTransport({
1908
+ sdk: sdk({
1909
+ stream: src.stream,
1910
+ questions: async () => {
1911
+ questionCalls += 1
1912
+ if (questionCalls === 1) {
1913
+ return ok([])
1914
+ }
1915
+
1916
+ if (questionCalls === 2) {
1917
+ started.resolve()
1918
+ return pending.promise
1919
+ }
1920
+
1921
+ return ok([])
1922
+ },
1923
+ promptAsync: async () => {
1924
+ queueMicrotask(() => {
1925
+ src.push(busy())
1926
+ src.push(assistant("msg-1"))
1927
+ src.push(
1928
+ toolUpdated(
1929
+ runningTool({
1930
+ sessionID: "session-1",
1931
+ messageID: "msg-1",
1932
+ id: "question-race-tool-1",
1933
+ callID: "call-question-race-1",
1934
+ tool: "question",
1935
+ body: {
1936
+ questions: request.questions,
1937
+ },
1938
+ }),
1939
+ ),
1940
+ )
1941
+ })
1942
+ return ok(undefined)
1943
+ },
1944
+ }),
1945
+ sessionID: "session-1",
1946
+ thinking: true,
1947
+ limits: () => ({}),
1948
+ footer: ui.api,
1949
+ })
1950
+
1951
+ const ctrl = new AbortController()
1952
+
1953
+ try {
1954
+ const run = transport.runPromptTurn({
1955
+ agent: undefined,
1956
+ model: undefined,
1957
+ variant: undefined,
1958
+ prompt: { text: "hello", parts: [] },
1959
+ files: [],
1960
+ includeFiles: false,
1961
+ signal: ctrl.signal,
1962
+ })
1963
+
1964
+ await started.promise
1965
+ src.push(
1966
+ toolUpdated(
1967
+ completedTool({
1968
+ sessionID: "session-1",
1969
+ messageID: "msg-1",
1970
+ id: "question-race-tool-1",
1971
+ callID: "call-question-race-1",
1972
+ tool: "question",
1973
+ body: {
1974
+ questions: request.questions,
1975
+ },
1976
+ output: "User has answered your questions.",
1977
+ metadata: {
1978
+ answers: [["CLI"]],
1979
+ },
1980
+ }),
1981
+ ),
1982
+ )
1983
+ await waitFor(() => {
1984
+ const commit = ui.commits.findLast(
1985
+ (item) => item.kind === "tool" && item.partID === "question-race-tool-1" && item.toolState === "completed",
1986
+ )
1987
+ return commit ? true : undefined
1988
+ })
1989
+ pending.resolve(ok([request]))
1990
+
1991
+ await Bun.sleep(50)
1992
+
1993
+ expect(
1994
+ ui.events.some(
1995
+ (event) =>
1996
+ event.type === "stream.view" && event.view.type === "question" && event.view.request.id === request.id,
1997
+ ),
1998
+ ).toBe(false)
1999
+
2000
+ ctrl.abort()
2001
+ await run
2002
+ } finally {
2003
+ src.close()
2004
+ await transport.close()
2005
+ }
2006
+ })
2007
+
2008
+ test("respects the includeFiles flag when building prompt payloads", async () => {
2009
+ const src = eventFeed()
2010
+ const ui = footer()
2011
+ const seen: unknown[] = []
2012
+ const file: RunFilePart = {
2013
+ type: "file",
2014
+ url: "file:///tmp/a.ts",
2015
+ filename: "a.ts",
2016
+ mime: "text/plain",
2017
+ }
2018
+
2019
+ const transport = await createSessionTransport({
2020
+ sdk: sdk({
2021
+ stream: src.stream,
2022
+ promptAsync: async (input) => {
2023
+ seen.push(input)
2024
+ queueMicrotask(() => {
2025
+ src.push(busy())
2026
+ src.push(idle())
2027
+ })
2028
+ return ok(undefined)
2029
+ },
2030
+ }),
2031
+ sessionID: "session-1",
2032
+ thinking: true,
2033
+ limits: () => ({}),
2034
+ footer: ui.api,
2035
+ })
2036
+
2037
+ try {
2038
+ await transport.runPromptTurn({
2039
+ agent: undefined,
2040
+ model: undefined,
2041
+ variant: undefined,
2042
+ prompt: { text: "hello", parts: [] },
2043
+ files: [file],
2044
+ includeFiles: true,
2045
+ })
2046
+
2047
+ await transport.runPromptTurn({
2048
+ agent: undefined,
2049
+ model: undefined,
2050
+ variant: undefined,
2051
+ prompt: { text: "again", parts: [] },
2052
+ files: [file],
2053
+ includeFiles: false,
2054
+ })
2055
+
2056
+ expect(seen).toEqual([
2057
+ expect.objectContaining({
2058
+ parts: [file, { type: "text", text: "hello" }],
2059
+ }),
2060
+ expect.objectContaining({
2061
+ parts: [{ type: "text", text: "again" }],
2062
+ }),
2063
+ ])
2064
+ } finally {
2065
+ src.close()
2066
+ await transport.close()
2067
+ }
2068
+ })
2069
+
2070
+ test("falls back to session status polling when idle events are missing", async () => {
2071
+ const src = eventFeed()
2072
+ const ui = footer()
2073
+ let busy = true
2074
+ const transport = await createSessionTransport({
2075
+ sdk: sdk({
2076
+ stream: src.stream,
2077
+ promptAsync: async () => {
2078
+ queueMicrotask(() => {
2079
+ src.push(assistant("msg-1"))
2080
+ busy = false
2081
+ })
2082
+ return ok(undefined)
2083
+ },
2084
+ status: async () => ok(statusMap(busy)),
2085
+ }),
2086
+ sessionID: "session-1",
2087
+ thinking: true,
2088
+ limits: () => ({}),
2089
+ footer: ui.api,
2090
+ })
2091
+
2092
+ try {
2093
+ await Promise.race([
2094
+ transport.runPromptTurn({
2095
+ agent: undefined,
2096
+ model: undefined,
2097
+ variant: undefined,
2098
+ prompt: { text: "hello", parts: [] },
2099
+ files: [],
2100
+ includeFiles: false,
2101
+ }),
2102
+ new Promise((_, reject) => setTimeout(() => reject(new Error("turn timed out")), 1_000)),
2103
+ ])
2104
+ } finally {
2105
+ src.close()
2106
+ await transport.close()
2107
+ }
2108
+ })
2109
+
2110
+ test("flushes interrupted output when the active turn aborts", async () => {
2111
+ const src = eventFeed()
2112
+ const seen = defer()
2113
+ const ui = footer((commit) => {
2114
+ if (commit.kind === "assistant" && commit.phase === "progress") {
2115
+ seen.resolve()
2116
+ }
2117
+ })
2118
+ const transport = await createSessionTransport({
2119
+ sdk: sdk({
2120
+ stream: src.stream,
2121
+ promptAsync: async () => {
2122
+ queueMicrotask(() => {
2123
+ src.push(busy())
2124
+ src.push(assistant("msg-1"))
2125
+ src.push(textUpdated(textPart("txt-1", "msg-1", "")))
2126
+ src.push(textDelta("msg-1", "txt-1", "unfinished"))
2127
+ })
2128
+ return ok(undefined)
2129
+ },
2130
+ }),
2131
+ sessionID: "session-1",
2132
+ thinking: true,
2133
+ limits: () => ({}),
2134
+ footer: ui.api,
2135
+ })
2136
+
2137
+ const ctrl = new AbortController()
2138
+
2139
+ try {
2140
+ const task = transport.runPromptTurn({
2141
+ agent: undefined,
2142
+ model: undefined,
2143
+ variant: undefined,
2144
+ prompt: { text: "hello", parts: [] },
2145
+ files: [],
2146
+ includeFiles: false,
2147
+ signal: ctrl.signal,
2148
+ })
2149
+
2150
+ await seen.promise
2151
+ ctrl.abort()
2152
+ await task
2153
+
2154
+ expect(ui.commits).toEqual([
2155
+ {
2156
+ kind: "assistant",
2157
+ text: "unfinished",
2158
+ phase: "progress",
2159
+ source: "assistant",
2160
+ messageID: "msg-1",
2161
+ partID: "txt-1",
2162
+ },
2163
+ {
2164
+ kind: "assistant",
2165
+ text: "",
2166
+ phase: "final",
2167
+ source: "assistant",
2168
+ messageID: "msg-1",
2169
+ partID: "txt-1",
2170
+ interrupted: true,
2171
+ },
2172
+ ])
2173
+ } finally {
2174
+ src.close()
2175
+ await transport.close()
2176
+ }
2177
+ })
2178
+
2179
+ test("closes an active turn without rejecting it", async () => {
2180
+ const src = eventFeed()
2181
+ const ui = footer()
2182
+ const ready = defer()
2183
+ let aborted = false
2184
+
2185
+ const transport = await createSessionTransport({
2186
+ sdk: sdk({
2187
+ stream: src.stream,
2188
+ promptAsync: async (_input, opt) => {
2189
+ ready.resolve()
2190
+ await new Promise<void>((resolve) => {
2191
+ const onAbort = () => {
2192
+ aborted = true
2193
+ opt?.signal?.removeEventListener("abort", onAbort)
2194
+ resolve()
2195
+ }
2196
+
2197
+ opt?.signal?.addEventListener("abort", onAbort, { once: true })
2198
+ })
2199
+ return ok(undefined)
2200
+ },
2201
+ }),
2202
+ sessionID: "session-1",
2203
+ thinking: true,
2204
+ limits: () => ({}),
2205
+ footer: ui.api,
2206
+ })
2207
+
2208
+ try {
2209
+ const task = transport.runPromptTurn({
2210
+ agent: undefined,
2211
+ model: undefined,
2212
+ variant: undefined,
2213
+ prompt: { text: "hello", parts: [] },
2214
+ files: [],
2215
+ includeFiles: false,
2216
+ })
2217
+
2218
+ await ready.promise
2219
+ await transport.close()
2220
+ await task
2221
+
2222
+ expect(aborted).toBe(true)
2223
+ } finally {
2224
+ src.close()
2225
+ await transport.close()
2226
+ }
2227
+ })
2228
+
2229
+ test("rejects the active turn when the event stream faults", async () => {
2230
+ const ui = footer()
2231
+ const ready = defer()
2232
+
2233
+ const transport = await createSessionTransport({
2234
+ sdk: sdk({
2235
+ globalEvent: () =>
2236
+ globalSse(
2237
+ (async function* (): AsyncGenerator<GlobalEvent> {
2238
+ await ready.promise
2239
+ yield globalEvent(busy())
2240
+ throw new Error("boom")
2241
+ })(),
2242
+ ),
2243
+ promptAsync: async () => {
2244
+ ready.resolve()
2245
+ return ok(undefined)
2246
+ },
2247
+ status: async () => ok({ "session-1": { type: "busy" } }),
2248
+ }),
2249
+ sessionID: "session-1",
2250
+ thinking: true,
2251
+ limits: () => ({}),
2252
+ footer: ui.api,
2253
+ })
2254
+
2255
+ try {
2256
+ await expect(
2257
+ transport.runPromptTurn({
2258
+ agent: undefined,
2259
+ model: undefined,
2260
+ variant: undefined,
2261
+ prompt: { text: "hello", parts: [] },
2262
+ files: [],
2263
+ includeFiles: false,
2264
+ }),
2265
+ ).rejects.toThrow("boom")
2266
+ } finally {
2267
+ await transport.close()
2268
+ }
2269
+ })
2270
+
2271
+ test("rejects the active turn when the backing instance is disposed", async () => {
2272
+ const ui = footer()
2273
+ const ready = defer()
2274
+
2275
+ const transport = await createSessionTransport({
2276
+ sdk: sdk({
2277
+ globalEvent: () =>
2278
+ globalSse(
2279
+ (async function* (): AsyncGenerator<GlobalEvent> {
2280
+ await ready.promise
2281
+ yield globalEvent({
2282
+ id: "evt-disposed",
2283
+ type: "server.instance.disposed",
2284
+ properties: {
2285
+ directory: "/tmp",
2286
+ },
2287
+ })
2288
+ })(),
2289
+ ),
2290
+ promptAsync: async () => {
2291
+ ready.resolve()
2292
+ return ok(undefined)
2293
+ },
2294
+ status: async () => ok({}),
2295
+ }),
2296
+ directory: "/tmp",
2297
+ sessionID: "session-1",
2298
+ thinking: true,
2299
+ limits: () => ({}),
2300
+ footer: ui.api,
2301
+ })
2302
+
2303
+ try {
2304
+ await expect(
2305
+ transport.runPromptTurn({
2306
+ agent: undefined,
2307
+ model: undefined,
2308
+ variant: undefined,
2309
+ prompt: { text: "hello", parts: [] },
2310
+ files: [],
2311
+ includeFiles: false,
2312
+ }),
2313
+ ).rejects.toThrow("instance disposed")
2314
+ } finally {
2315
+ await transport.close()
2316
+ }
2317
+ })
2318
+
2319
+ test("rejects concurrent turns", async () => {
2320
+ const src = eventFeed()
2321
+ const ui = footer()
2322
+ const transport = await createSessionTransport({
2323
+ sdk: sdk({
2324
+ stream: src.stream,
2325
+ }),
2326
+ sessionID: "session-1",
2327
+ thinking: true,
2328
+ limits: () => ({}),
2329
+ footer: ui.api,
2330
+ })
2331
+
2332
+ const ctrl = new AbortController()
2333
+
2334
+ try {
2335
+ const task = transport.runPromptTurn({
2336
+ agent: undefined,
2337
+ model: undefined,
2338
+ variant: undefined,
2339
+ prompt: { text: "one", parts: [] },
2340
+ files: [],
2341
+ includeFiles: false,
2342
+ signal: ctrl.signal,
2343
+ })
2344
+
2345
+ await expect(
2346
+ transport.runPromptTurn({
2347
+ agent: undefined,
2348
+ model: undefined,
2349
+ variant: undefined,
2350
+ prompt: { text: "two", parts: [] },
2351
+ files: [],
2352
+ includeFiles: false,
2353
+ }),
2354
+ ).rejects.toThrow("prompt already running")
2355
+
2356
+ ctrl.abort()
2357
+ await task
2358
+ } finally {
2359
+ src.close()
2360
+ await transport.close()
2361
+ }
2362
+ })
2363
+ })