@farhanic017/octocode 3.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (716) hide show
  1. package/.swarm_lessons.json +85 -0
  2. package/AGENTS.md +151 -0
  3. package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
  4. package/Dockerfile +18 -0
  5. package/README.md +15 -0
  6. package/bin/octocode +31 -0
  7. package/bunfig.toml +7 -0
  8. package/git +0 -0
  9. package/package.json +151 -0
  10. package/parsers-config.ts +1 -0
  11. package/script/bench-search.ts +115 -0
  12. package/script/bench-test-suite.ts +52 -0
  13. package/script/build.ts +243 -0
  14. package/script/generate.ts +14 -0
  15. package/script/httpapi-exercise.ts +1 -0
  16. package/script/octo-shim.mjs +34 -0
  17. package/script/postinstall.mjs +198 -0
  18. package/script/profile-test-files.ts +42 -0
  19. package/script/publish.ts +211 -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 +154 -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 +345 -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/tool.ts +321 -0
  56. package/src/acp/usage.ts +239 -0
  57. package/src/agent/agent.ts +451 -0
  58. package/src/agent/generate.txt +75 -0
  59. package/src/agent/prompt/compaction.txt +30 -0
  60. package/src/agent/prompt/explore.txt +18 -0
  61. package/src/agent/prompt/summary.txt +11 -0
  62. package/src/agent/prompt/title.txt +50 -0
  63. package/src/agent/subagent-permissions.ts +35 -0
  64. package/src/audio.d.ts +14 -0
  65. package/src/auth/index.ts +96 -0
  66. package/src/background/job.ts +36 -0
  67. package/src/bus/global.ts +22 -0
  68. package/src/cli/bootstrap.ts +11 -0
  69. package/src/cli/cmd/account.ts +264 -0
  70. package/src/cli/cmd/acp.ts +76 -0
  71. package/src/cli/cmd/agent.ts +259 -0
  72. package/src/cli/cmd/attach.ts +97 -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 +188 -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 +1594 -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 +859 -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 +1 -0
  98. package/src/cli/cmd/providers.ts +550 -0
  99. package/src/cli/cmd/run/demo.ts +1274 -0
  100. package/src/cli/cmd/run/entry.body.ts +205 -0
  101. package/src/cli/cmd/run/footer.command.tsx +1088 -0
  102. package/src/cli/cmd/run/footer.menu.tsx +353 -0
  103. package/src/cli/cmd/run/footer.permission.tsx +482 -0
  104. package/src/cli/cmd/run/footer.prompt.tsx +1351 -0
  105. package/src/cli/cmd/run/footer.question.tsx +580 -0
  106. package/src/cli/cmd/run/footer.subagent.tsx +180 -0
  107. package/src/cli/cmd/run/footer.ts +1153 -0
  108. package/src/cli/cmd/run/footer.view.tsx +992 -0
  109. package/src/cli/cmd/run/footer.width.ts +27 -0
  110. package/src/cli/cmd/run/otel.ts +117 -0
  111. package/src/cli/cmd/run/permission.shared.ts +256 -0
  112. package/src/cli/cmd/run/prompt.editor.ts +157 -0
  113. package/src/cli/cmd/run/prompt.shared.ts +153 -0
  114. package/src/cli/cmd/run/question.shared.ts +340 -0
  115. package/src/cli/cmd/run/runtime.boot.ts +202 -0
  116. package/src/cli/cmd/run/runtime.lifecycle.ts +431 -0
  117. package/src/cli/cmd/run/runtime.queue.ts +349 -0
  118. package/src/cli/cmd/run/runtime.shared.ts +17 -0
  119. package/src/cli/cmd/run/runtime.stdin.ts +37 -0
  120. package/src/cli/cmd/run/runtime.ts +905 -0
  121. package/src/cli/cmd/run/scrollback.shared.ts +92 -0
  122. package/src/cli/cmd/run/scrollback.surface.ts +448 -0
  123. package/src/cli/cmd/run/scrollback.writer.tsx +353 -0
  124. package/src/cli/cmd/run/splash.ts +284 -0
  125. package/src/cli/cmd/run/stream.transport.ts +1465 -0
  126. package/src/cli/cmd/run/stream.ts +175 -0
  127. package/src/cli/cmd/run/subagent-data.ts +876 -0
  128. package/src/cli/cmd/run/theme.ts +690 -0
  129. package/src/cli/cmd/run/tool.ts +1489 -0
  130. package/src/cli/cmd/run/trace.ts +94 -0
  131. package/src/cli/cmd/run/turn-summary.ts +47 -0
  132. package/src/cli/cmd/run/types.ts +350 -0
  133. package/src/cli/cmd/run/variant.shared.ts +215 -0
  134. package/src/cli/cmd/run.ts +897 -0
  135. package/src/cli/cmd/serve.ts +24 -0
  136. package/src/cli/cmd/stats.ts +393 -0
  137. package/src/cli/cmd/tui.ts +264 -0
  138. package/src/cli/cmd/uninstall.ts +353 -0
  139. package/src/cli/cmd/upgrade.ts +74 -0
  140. package/src/cli/cmd/web.ts +84 -0
  141. package/src/cli/effect/prompt.ts +37 -0
  142. package/src/cli/effect-cmd.ts +96 -0
  143. package/src/cli/error.ts +118 -0
  144. package/src/cli/heap.ts +66 -0
  145. package/src/cli/logo.ts +1 -0
  146. package/src/cli/network.ts +64 -0
  147. package/src/cli/tui/layer.ts +7 -0
  148. package/src/cli/tui/photon_rs_bg-bq08arze.wasm +0 -0
  149. package/src/cli/tui/tree-sitter-3jzf13jk.wasm +0 -0
  150. package/src/cli/tui/tree-sitter-bash-hq5s6fxb.wasm +0 -0
  151. package/src/cli/tui/tree-sitter-powershell-ryb2ffqs.wasm +0 -0
  152. package/src/cli/tui/worker.ts +99 -0
  153. package/src/cli/ui.ts +144 -0
  154. package/src/cli/upgrade.ts +53 -0
  155. package/src/command/index.ts +189 -0
  156. package/src/command/template/initialize.txt +66 -0
  157. package/src/command/template/review.txt +101 -0
  158. package/src/command/terminal-ai-commands.ts +567 -0
  159. package/src/config/agent.ts +68 -0
  160. package/src/config/command.ts +45 -0
  161. package/src/config/config.ts +686 -0
  162. package/src/config/entry-name.ts +19 -0
  163. package/src/config/managed.ts +77 -0
  164. package/src/config/markdown.ts +36 -0
  165. package/src/config/parse.ts +79 -0
  166. package/src/config/paths.ts +47 -0
  167. package/src/config/plugin.ts +79 -0
  168. package/src/config/reference.ts +48 -0
  169. package/src/config/tui-cwd.ts +5 -0
  170. package/src/config/tui-host-attention.ts +21 -0
  171. package/src/config/tui-migrate.ts +156 -0
  172. package/src/config/tui.ts +294 -0
  173. package/src/config/variable.ts +91 -0
  174. package/src/control-plane/adapters/index.ts +41 -0
  175. package/src/control-plane/adapters/worktree.ts +96 -0
  176. package/src/control-plane/dev/README.md +19 -0
  177. package/src/control-plane/dev/debug-workspace-plugin.ts +73 -0
  178. package/src/control-plane/types.ts +59 -0
  179. package/src/control-plane/util.ts +39 -0
  180. package/src/control-plane/workspace-adapter-runtime.ts +51 -0
  181. package/src/control-plane/workspace-context.ts +26 -0
  182. package/src/control-plane/workspace.ts +1075 -0
  183. package/src/desktop/agent.ts +147 -0
  184. package/src/desktop/index.ts +247 -0
  185. package/src/desktop/platform.ts +97 -0
  186. package/src/desktop/stream.ts +64 -0
  187. package/src/desktop/vision.ts +52 -0
  188. package/src/desktop/websocket.ts +120 -0
  189. package/src/effect/app-runtime.ts +133 -0
  190. package/src/effect/bootstrap-runtime.ts +23 -0
  191. package/src/effect/bridge.ts +84 -0
  192. package/src/effect/config-service.ts +67 -0
  193. package/src/effect/instance-ref.ts +11 -0
  194. package/src/effect/instance-registry.ts +12 -0
  195. package/src/effect/instance-state.ts +72 -0
  196. package/src/effect/promise.ts +17 -0
  197. package/src/effect/run-service.ts +47 -0
  198. package/src/effect/runner.ts +217 -0
  199. package/src/effect/runtime-flags.ts +76 -0
  200. package/src/env/index.ts +40 -0
  201. package/src/event-v2-bridge.ts +76 -0
  202. package/src/format/formatter.ts +404 -0
  203. package/src/format/index.ts +212 -0
  204. package/src/git/index.ts +347 -0
  205. package/src/id/id.ts +80 -0
  206. package/src/ide/index.ts +70 -0
  207. package/src/image/image.ts +177 -0
  208. package/src/index.ts +208 -0
  209. package/src/installation/index.ts +350 -0
  210. package/src/lsp/client.ts +686 -0
  211. package/src/lsp/diagnostic.ts +29 -0
  212. package/src/lsp/language.ts +121 -0
  213. package/src/lsp/launch.ts +21 -0
  214. package/src/lsp/lsp.ts +517 -0
  215. package/src/lsp/server.ts +2116 -0
  216. package/src/markdown.d.ts +4 -0
  217. package/src/mcp/auth.ts +171 -0
  218. package/src/mcp/builtin.ts +92 -0
  219. package/src/mcp/index.ts +1000 -0
  220. package/src/mcp/oauth-callback.ts +232 -0
  221. package/src/mcp/oauth-provider.ts +217 -0
  222. package/src/node.ts +5 -0
  223. package/src/patch/index.ts +689 -0
  224. package/src/permission/arity.ts +163 -0
  225. package/src/permission/evaluate.ts +1 -0
  226. package/src/permission/index.ts +230 -0
  227. package/src/plugin/azure.ts +26 -0
  228. package/src/plugin/cloudflare.ts +76 -0
  229. package/src/plugin/digitalocean.ts +391 -0
  230. package/src/plugin/github-copilot/copilot.ts +417 -0
  231. package/src/plugin/github-copilot/models.ts +246 -0
  232. package/src/plugin/index.ts +320 -0
  233. package/src/plugin/install.ts +439 -0
  234. package/src/plugin/loader.ts +237 -0
  235. package/src/plugin/meta.ts +188 -0
  236. package/src/plugin/openai/README.md +31 -0
  237. package/src/plugin/openai/codex.ts +647 -0
  238. package/src/plugin/openai/ws-pool.ts +290 -0
  239. package/src/plugin/openai/ws.ts +381 -0
  240. package/src/plugin/shared.ts +323 -0
  241. package/src/plugin/tui/internal.ts +12 -0
  242. package/src/plugin/tui/runtime.ts +1174 -0
  243. package/src/plugin/xai.ts +742 -0
  244. package/src/project/bootstrap-service.ts +9 -0
  245. package/src/project/bootstrap.ts +80 -0
  246. package/src/project/instance-context.ts +24 -0
  247. package/src/project/instance-layer.ts +11 -0
  248. package/src/project/instance-runtime.ts +16 -0
  249. package/src/project/instance-store.ts +207 -0
  250. package/src/project/project.ts +520 -0
  251. package/src/project/vcs.ts +435 -0
  252. package/src/provider/auth.ts +230 -0
  253. package/src/provider/error.ts +188 -0
  254. package/src/provider/model-status.ts +8 -0
  255. package/src/provider/provider.ts +2037 -0
  256. package/src/provider/transform.ts +1367 -0
  257. package/src/pty-preparation.ts +30 -0
  258. package/src/question/index.ts +229 -0
  259. package/src/question/schema.ts +10 -0
  260. package/src/reference/reference.ts +239 -0
  261. package/src/reference/repository-cache.ts +320 -0
  262. package/src/server/auth.ts +48 -0
  263. package/src/server/cors.ts +34 -0
  264. package/src/server/event.ts +13 -0
  265. package/src/server/global-lifecycle.ts +37 -0
  266. package/src/server/init-projectors.ts +3 -0
  267. package/src/server/mdns.ts +60 -0
  268. package/src/server/projectors.ts +1 -0
  269. package/src/server/proxy-util.ts +48 -0
  270. package/src/server/routes/instance/httpapi/AGENTS.md +39 -0
  271. package/src/server/routes/instance/httpapi/api.ts +80 -0
  272. package/src/server/routes/instance/httpapi/errors.ts +193 -0
  273. package/src/server/routes/instance/httpapi/groups/config.ts +65 -0
  274. package/src/server/routes/instance/httpapi/groups/control-plane.ts +35 -0
  275. package/src/server/routes/instance/httpapi/groups/control.ts +76 -0
  276. package/src/server/routes/instance/httpapi/groups/event.ts +29 -0
  277. package/src/server/routes/instance/httpapi/groups/experimental.ts +260 -0
  278. package/src/server/routes/instance/httpapi/groups/file.ts +172 -0
  279. package/src/server/routes/instance/httpapi/groups/global.ts +138 -0
  280. package/src/server/routes/instance/httpapi/groups/instance.ts +206 -0
  281. package/src/server/routes/instance/httpapi/groups/mcp.ts +156 -0
  282. package/src/server/routes/instance/httpapi/groups/metadata.ts +18 -0
  283. package/src/server/routes/instance/httpapi/groups/permission.ts +61 -0
  284. package/src/server/routes/instance/httpapi/groups/project-copy.ts +88 -0
  285. package/src/server/routes/instance/httpapi/groups/project.ts +93 -0
  286. package/src/server/routes/instance/httpapi/groups/provider.ts +101 -0
  287. package/src/server/routes/instance/httpapi/groups/pty.ts +172 -0
  288. package/src/server/routes/instance/httpapi/groups/query.ts +12 -0
  289. package/src/server/routes/instance/httpapi/groups/question.ts +74 -0
  290. package/src/server/routes/instance/httpapi/groups/reference.ts +60 -0
  291. package/src/server/routes/instance/httpapi/groups/sync.ts +113 -0
  292. package/src/server/routes/instance/httpapi/groups/tui.ts +208 -0
  293. package/src/server/routes/instance/httpapi/groups/workspace.ts +141 -0
  294. package/src/server/routes/instance/httpapi/handlers/config.ts +34 -0
  295. package/src/server/routes/instance/httpapi/handlers/control-plane.ts +37 -0
  296. package/src/server/routes/instance/httpapi/handlers/control.ts +37 -0
  297. package/src/server/routes/instance/httpapi/handlers/event.ts +102 -0
  298. package/src/server/routes/instance/httpapi/handlers/experimental.ts +187 -0
  299. package/src/server/routes/instance/httpapi/handlers/file.ts +128 -0
  300. package/src/server/routes/instance/httpapi/handlers/global.ts +157 -0
  301. package/src/server/routes/instance/httpapi/handlers/instance.ts +110 -0
  302. package/src/server/routes/instance/httpapi/handlers/mcp.ts +111 -0
  303. package/src/server/routes/instance/httpapi/handlers/permission.ts +41 -0
  304. package/src/server/routes/instance/httpapi/handlers/project-copy.ts +157 -0
  305. package/src/server/routes/instance/httpapi/handlers/project.ts +63 -0
  306. package/src/server/routes/instance/httpapi/handlers/provider.ts +113 -0
  307. package/src/server/routes/instance/httpapi/handlers/pty.ts +258 -0
  308. package/src/server/routes/instance/httpapi/handlers/question.ts +54 -0
  309. package/src/server/routes/instance/httpapi/handlers/reference.ts +27 -0
  310. package/src/server/routes/instance/httpapi/handlers/sync.ts +95 -0
  311. package/src/server/routes/instance/httpapi/handlers/tui.ts +131 -0
  312. package/src/server/routes/instance/httpapi/handlers/workspace.ts +102 -0
  313. package/src/server/routes/instance/httpapi/lifecycle.ts +57 -0
  314. package/src/server/routes/instance/httpapi/middleware/authorization.ts +150 -0
  315. package/src/server/routes/instance/httpapi/middleware/compression.ts +64 -0
  316. package/src/server/routes/instance/httpapi/middleware/cors-vary.ts +29 -0
  317. package/src/server/routes/instance/httpapi/middleware/error.ts +36 -0
  318. package/src/server/routes/instance/httpapi/middleware/fence.ts +25 -0
  319. package/src/server/routes/instance/httpapi/middleware/instance-context.ts +43 -0
  320. package/src/server/routes/instance/httpapi/middleware/proxy.ts +108 -0
  321. package/src/server/routes/instance/httpapi/middleware/schema-error.ts +42 -0
  322. package/src/server/routes/instance/httpapi/middleware/workspace-routing.ts +250 -0
  323. package/src/server/routes/instance/httpapi/public.ts +535 -0
  324. package/src/server/routes/instance/httpapi/server.ts +281 -0
  325. package/src/server/routes/instance/httpapi/websocket-tracker.ts +57 -0
  326. package/src/server/server.ts +218 -0
  327. package/src/server/shared/fence.ts +68 -0
  328. package/src/server/shared/pty-ticket.ts +15 -0
  329. package/src/server/shared/public-ui.ts +12 -0
  330. package/src/server/shared/tui-control.ts +28 -0
  331. package/src/server/shared/ui.ts +108 -0
  332. package/src/server/shared/workspace-routing.ts +38 -0
  333. package/src/server/tui-event.ts +53 -0
  334. package/src/share/share-next.ts +379 -0
  335. package/src/shell/shell.ts +215 -0
  336. package/src/skill/discovery.ts +115 -0
  337. package/src/skill/index.ts +382 -0
  338. package/src/snapshot/index.ts +762 -0
  339. package/src/sql.d.ts +4 -0
  340. package/src/storage/schema.ts +5 -0
  341. package/src/storage/storage.ts +329 -0
  342. package/src/swarm/evolution.ts +304 -0
  343. package/src/swarm/index.ts +7 -0
  344. package/src/swarm/learning.ts +736 -0
  345. package/src/swarm/vision.ts +131 -0
  346. package/src/sync/README.md +179 -0
  347. package/src/sync/schema.ts +11 -0
  348. package/src/temporary.ts +33 -0
  349. package/src/tool/apply_patch.ts +313 -0
  350. package/src/tool/apply_patch.txt +33 -0
  351. package/src/tool/browser_click.ts +51 -0
  352. package/src/tool/browser_drag.ts +76 -0
  353. package/src/tool/browser_evaluate.ts +51 -0
  354. package/src/tool/browser_hover.ts +51 -0
  355. package/src/tool/browser_navigate.ts +45 -0
  356. package/src/tool/browser_screenshot.ts +66 -0
  357. package/src/tool/browser_select.ts +52 -0
  358. package/src/tool/browser_type.ts +52 -0
  359. package/src/tool/browser_wait.ts +60 -0
  360. package/src/tool/desktop_clipboard.ts +82 -0
  361. package/src/tool/desktop_control.ts +160 -0
  362. package/src/tool/desktop_record.ts +93 -0
  363. package/src/tool/desktop_replay.ts +99 -0
  364. package/src/tool/desktop_screen_record.ts +143 -0
  365. package/src/tool/desktop_window.ts +142 -0
  366. package/src/tool/desktop_workflow.ts +81 -0
  367. package/src/tool/edit.ts +737 -0
  368. package/src/tool/edit.txt +10 -0
  369. package/src/tool/external-directory.ts +49 -0
  370. package/src/tool/glob.ts +84 -0
  371. package/src/tool/glob.txt +6 -0
  372. package/src/tool/grep.ts +140 -0
  373. package/src/tool/grep.txt +8 -0
  374. package/src/tool/invalid.ts +21 -0
  375. package/src/tool/json-schema.ts +164 -0
  376. package/src/tool/lsp.ts +113 -0
  377. package/src/tool/lsp.txt +24 -0
  378. package/src/tool/mcp-websearch.ts +96 -0
  379. package/src/tool/open_app.ts +55 -0
  380. package/src/tool/open_terminal.ts +69 -0
  381. package/src/tool/plan-enter.txt +14 -0
  382. package/src/tool/plan-exit.txt +13 -0
  383. package/src/tool/plan.ts +79 -0
  384. package/src/tool/question.ts +44 -0
  385. package/src/tool/question.txt +10 -0
  386. package/src/tool/read.ts +392 -0
  387. package/src/tool/read.txt +14 -0
  388. package/src/tool/registry.ts +527 -0
  389. package/src/tool/schema.ts +14 -0
  390. package/src/tool/screenshot.ts +72 -0
  391. package/src/tool/shell/id.ts +19 -0
  392. package/src/tool/shell/prompt.ts +307 -0
  393. package/src/tool/shell/shell.txt +21 -0
  394. package/src/tool/shell.ts +669 -0
  395. package/src/tool/skill.ts +72 -0
  396. package/src/tool/skill.txt +5 -0
  397. package/src/tool/task.ts +339 -0
  398. package/src/tool/task.txt +19 -0
  399. package/src/tool/todo.ts +57 -0
  400. package/src/tool/todowrite.txt +44 -0
  401. package/src/tool/tool.ts +180 -0
  402. package/src/tool/truncate.ts +160 -0
  403. package/src/tool/truncation-dir.ts +4 -0
  404. package/src/tool/visual_qa.ts +77 -0
  405. package/src/tool/webfetch.ts +192 -0
  406. package/src/tool/webfetch.txt +13 -0
  407. package/src/tool/websearch.ts +143 -0
  408. package/src/tool/websearch.txt +14 -0
  409. package/src/tool/write.ts +104 -0
  410. package/src/tool/write.txt +8 -0
  411. package/src/util/archive.ts +17 -0
  412. package/src/util/bom.ts +27 -0
  413. package/src/util/data-url.ts +9 -0
  414. package/src/util/defer.ts +10 -0
  415. package/src/util/effect-http-client.ts +11 -0
  416. package/src/util/error.ts +1 -0
  417. package/src/util/filesystem.ts +251 -0
  418. package/src/util/iife.ts +3 -0
  419. package/src/util/lazy.ts +20 -0
  420. package/src/util/local-context.ts +25 -0
  421. package/src/util/locale.ts +2 -0
  422. package/src/util/media.ts +26 -0
  423. package/src/util/process.ts +177 -0
  424. package/src/util/proxy-env.ts +72 -0
  425. package/src/util/queue.ts +32 -0
  426. package/src/util/record.ts +1 -0
  427. package/src/util/repository.ts +232 -0
  428. package/src/util/rpc.ts +66 -0
  429. package/src/util/signal.ts +12 -0
  430. package/src/util/sys-monitor.ts +223 -0
  431. package/src/util/timeout.ts +13 -0
  432. package/src/util/wildcard.ts +59 -0
  433. package/src/worktree/index.ts +645 -0
  434. package/sst-env.d.ts +10 -0
  435. package/test/AGENTS.md +204 -0
  436. package/test/EFFECT_TEST_MIGRATION.md +169 -0
  437. package/test/account/repo.test.ts +353 -0
  438. package/test/account/service.test.ts +453 -0
  439. package/test/acp/config-option.test.ts +229 -0
  440. package/test/acp/content.test.ts +201 -0
  441. package/test/acp/directory.test.ts +186 -0
  442. package/test/acp/error.test.ts +67 -0
  443. package/test/acp/event.test.ts +743 -0
  444. package/test/acp/permission.test.ts +273 -0
  445. package/test/acp/tool.test.ts +210 -0
  446. package/test/acp/usage.test.ts +315 -0
  447. package/test/agent/agent.test.ts +711 -0
  448. package/test/agent/plan-mode-subagent-bypass.test.ts +213 -0
  449. package/test/agent/plugin-agent-regression.test.ts +62 -0
  450. package/test/auth/auth.test.ts +77 -0
  451. package/test/background/job.test.ts +243 -0
  452. package/test/cli/account.test.ts +30 -0
  453. package/test/cli/acp/acp-test-client.ts +97 -0
  454. package/test/cli/acp/config-options.test.ts +103 -0
  455. package/test/cli/acp/helpers.ts +96 -0
  456. package/test/cli/acp/initialize-auth.test.ts +61 -0
  457. package/test/cli/acp/lifecycle.test.ts +118 -0
  458. package/test/cli/acp/prompt-content.test.ts +97 -0
  459. package/test/cli/acp/skills.test.ts +38 -0
  460. package/test/cli/cmd/tui/attention.test.ts +484 -0
  461. package/test/cli/effect-cmd-instance-als.test.ts +39 -0
  462. package/test/cli/error.test.ts +95 -0
  463. package/test/cli/github-action.test.ts +199 -0
  464. package/test/cli/github-remote.test.ts +90 -0
  465. package/test/cli/help/__snapshots__/help-snapshots.test.ts.snap +631 -0
  466. package/test/cli/help/help-snapshots.test.ts +137 -0
  467. package/test/cli/import.test.ts +54 -0
  468. package/test/cli/mcp-add.test.ts +74 -0
  469. package/test/cli/plugin-auth-picker.test.ts +120 -0
  470. package/test/cli/run/entry.body.test.ts +536 -0
  471. package/test/cli/run/footer.menu.test.ts +43 -0
  472. package/test/cli/run/footer.view.test.tsx +1336 -0
  473. package/test/cli/run/footer.width.test.ts +35 -0
  474. package/test/cli/run/permission.shared.test.ts +144 -0
  475. package/test/cli/run/prompt.editor.test.ts +101 -0
  476. package/test/cli/run/prompt.shared.test.ts +101 -0
  477. package/test/cli/run/question.shared.test.ts +115 -0
  478. package/test/cli/run/run-process.test.ts +84 -0
  479. package/test/cli/run/runtime.boot.test.ts +283 -0
  480. package/test/cli/run/runtime.queue.test.ts +481 -0
  481. package/test/cli/run/runtime.stdin.test.ts +71 -0
  482. package/test/cli/run/runtime.test.ts +238 -0
  483. package/test/cli/run/scrollback.surface.test.ts +1065 -0
  484. package/test/cli/run/stream.test.ts +56 -0
  485. package/test/cli/run/stream.transport.test.ts +2363 -0
  486. package/test/cli/run/subagent-data.test.ts +547 -0
  487. package/test/cli/run/theme.test.ts +177 -0
  488. package/test/cli/run/variant.shared.test.ts +217 -0
  489. package/test/cli/serve/serve-process.test.ts +61 -0
  490. package/test/cli/smokes/read-only.test.ts +115 -0
  491. package/test/cli/tui/attach.test.ts +11 -0
  492. package/test/cli/tui/editor-context-zed.test.ts +379 -0
  493. package/test/cli/tui/editor-context.test.tsx +297 -0
  494. package/test/cli/tui/plugin-add.test.ts +110 -0
  495. package/test/cli/tui/plugin-install.test.ts +87 -0
  496. package/test/cli/tui/plugin-lifecycle.test.ts +224 -0
  497. package/test/cli/tui/plugin-loader-entrypoint.test.ts +485 -0
  498. package/test/cli/tui/plugin-loader-pure.test.ts +72 -0
  499. package/test/cli/tui/plugin-loader.test.ts +1332 -0
  500. package/test/cli/tui/plugin-toggle.test.ts +264 -0
  501. package/test/cli/tui/thread.test.ts +36 -0
  502. package/test/cli.test.ts +7 -0
  503. package/test/command/acp-slash-detect.test.ts +202 -0
  504. package/test/command/demo-slash.test.ts +213 -0
  505. package/test/command/slash-head.test.ts +232 -0
  506. package/test/command/slash-parsing.test.ts +193 -0
  507. package/test/command/terminal-ai-commands.test.ts +536 -0
  508. package/test/config/agent-color.test.ts +47 -0
  509. package/test/config/config.test.ts +1994 -0
  510. package/test/config/entry-name.test.ts +57 -0
  511. package/test/config/fixtures/empty-frontmatter.md +4 -0
  512. package/test/config/fixtures/frontmatter.md +28 -0
  513. package/test/config/fixtures/markdown-header.md +11 -0
  514. package/test/config/fixtures/no-frontmatter.md +1 -0
  515. package/test/config/fixtures/weird-model-id.md +13 -0
  516. package/test/config/lsp.test.ts +69 -0
  517. package/test/config/markdown.test.ts +228 -0
  518. package/test/config/plugin.test.ts +0 -0
  519. package/test/config/tui.test.ts +886 -0
  520. package/test/control-plane/adapters.test.ts +71 -0
  521. package/test/control-plane/workspace.test.ts +1704 -0
  522. package/test/effect/app-runtime-logger.test.ts +105 -0
  523. package/test/effect/config-service.test.ts +65 -0
  524. package/test/effect/instance-state.test.ts +391 -0
  525. package/test/effect/run-service.test.ts +89 -0
  526. package/test/effect/runner.test.ts +514 -0
  527. package/test/effect/runtime-flags.test.ts +373 -0
  528. package/test/fake/account.ts +9 -0
  529. package/test/fake/auth.ts +8 -0
  530. package/test/fake/npm.ts +8 -0
  531. package/test/fake/provider.ts +82 -0
  532. package/test/fake/skill.ts +8 -0
  533. package/test/filesystem/filesystem.test.ts +319 -0
  534. package/test/fixture/agent-plugin.constants.ts +6 -0
  535. package/test/fixture/agent-plugin.ts +12 -0
  536. package/test/fixture/config.ts +23 -0
  537. package/test/fixture/db.ts +11 -0
  538. package/test/fixture/fixture.test.ts +26 -0
  539. package/test/fixture/fixture.ts +224 -0
  540. package/test/fixture/flag.ts +20 -0
  541. package/test/fixture/flock-worker.ts +72 -0
  542. package/test/fixture/lsp/fake-lsp-server.js +249 -0
  543. package/test/fixture/plug-worker.ts +96 -0
  544. package/test/fixture/plugin-meta-worker.ts +19 -0
  545. package/test/fixture/plugin.ts +10 -0
  546. package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
  547. package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
  548. package/test/fixture/skills/cloudflare/SKILL.md +211 -0
  549. package/test/fixture/skills/index.json +6 -0
  550. package/test/fixture/tui-environment.tsx +32 -0
  551. package/test/fixture/tui-plugin.ts +357 -0
  552. package/test/fixture/tui-runtime.ts +56 -0
  553. package/test/fixture/tui-sdk.ts +82 -0
  554. package/test/fixture/workspace.ts +30 -0
  555. package/test/format/format.test.ts +228 -0
  556. package/test/git/git.test.ts +178 -0
  557. package/test/ide/ide.test.ts +82 -0
  558. package/test/image/fixtures/picture-5mb-base64.png +0 -0
  559. package/test/image/image.test.ts +123 -0
  560. package/test/installation/installation.test.ts +231 -0
  561. package/test/lib/cli-process.ts +459 -0
  562. package/test/lib/effect.ts +177 -0
  563. package/test/lib/filesystem.ts +10 -0
  564. package/test/lib/llm-server.ts +771 -0
  565. package/test/lib/snapshot.ts +73 -0
  566. package/test/lib/test-provider.ts +37 -0
  567. package/test/lib/websocket.ts +46 -0
  568. package/test/lsp/client.test.ts +493 -0
  569. package/test/lsp/index.test.ts +232 -0
  570. package/test/lsp/jdtls-root.test.ts +459 -0
  571. package/test/lsp/launch.test.ts +22 -0
  572. package/test/lsp/lifecycle.test.ts +160 -0
  573. package/test/mcp/auth.test.ts +78 -0
  574. package/test/mcp/builtin.test.ts +95 -0
  575. package/test/mcp/headers.test.ts +126 -0
  576. package/test/mcp/lifecycle.test.ts +999 -0
  577. package/test/mcp/oauth-auto-connect.test.ts +274 -0
  578. package/test/mcp/oauth-browser.test.ts +232 -0
  579. package/test/mcp/oauth-callback.test.ts +34 -0
  580. package/test/mcp/oauth-provider.test.ts +61 -0
  581. package/test/patch/patch.test.ts +383 -0
  582. package/test/permission/arity.test.ts +33 -0
  583. package/test/permission/next.test.ts +1176 -0
  584. package/test/permission-task.test.ts +318 -0
  585. package/test/plugin/auth-override.test.ts +105 -0
  586. package/test/plugin/cloudflare.test.ts +68 -0
  587. package/test/plugin/codex.test.ts +247 -0
  588. package/test/plugin/github-copilot-models.test.ts +332 -0
  589. package/test/plugin/install-concurrency.test.ts +140 -0
  590. package/test/plugin/install.test.ts +573 -0
  591. package/test/plugin/loader-shared.test.ts +1303 -0
  592. package/test/plugin/meta.test.ts +137 -0
  593. package/test/plugin/openai-rollout.test.ts +17 -0
  594. package/test/plugin/openai-ws.test.ts +877 -0
  595. package/test/plugin/shared.test.ts +88 -0
  596. package/test/plugin/trigger.test.ts +120 -0
  597. package/test/plugin/workspace-adapter.test.ts +137 -0
  598. package/test/plugin/xai.test.ts +634 -0
  599. package/test/preload.ts +99 -0
  600. package/test/project/instance-bootstrap.test.ts +110 -0
  601. package/test/project/instance.test.ts +245 -0
  602. package/test/project/migrate-global.test.ts +170 -0
  603. package/test/project/project-directory.test.ts +169 -0
  604. package/test/project/project.test.ts +818 -0
  605. package/test/project/vcs.test.ts +336 -0
  606. package/test/project/worktree-remove.test.ts +126 -0
  607. package/test/project/worktree.test.ts +320 -0
  608. package/test/provider/amazon-bedrock.test.ts +360 -0
  609. package/test/provider/cf-ai-gateway-e2e.test.ts +132 -0
  610. package/test/provider/digitalocean.test.ts +123 -0
  611. package/test/provider/gitlab-duo.test.ts +412 -0
  612. package/test/provider/header-timeout.test.ts +233 -0
  613. package/test/provider/model-status.test.ts +61 -0
  614. package/test/provider/provider.test.ts +1795 -0
  615. package/test/provider/transform.test.ts +3937 -0
  616. package/test/pty/pty-shell.test.ts +102 -0
  617. package/test/question/question.test.ts +465 -0
  618. package/test/reference/reference.test.ts +311 -0
  619. package/test/server/AGENTS.md +15 -0
  620. package/test/server/auth.test.ts +59 -0
  621. package/test/server/global-bus.ts +31 -0
  622. package/test/server/httpapi-authorization.test.ts +174 -0
  623. package/test/server/httpapi-compression.test.ts +154 -0
  624. package/test/server/httpapi-config.test.ts +113 -0
  625. package/test/server/httpapi-control-plane.test.ts +63 -0
  626. package/test/server/httpapi-cors-vary.test.ts +66 -0
  627. package/test/server/httpapi-cors.test.ts +122 -0
  628. package/test/server/httpapi-error-middleware.test.ts +96 -0
  629. package/test/server/httpapi-event.test.ts +97 -0
  630. package/test/server/httpapi-exercise/assertions.ts +64 -0
  631. package/test/server/httpapi-exercise/backend.ts +144 -0
  632. package/test/server/httpapi-exercise/dsl.ts +210 -0
  633. package/test/server/httpapi-exercise/environment.ts +40 -0
  634. package/test/server/httpapi-exercise/index.ts +1538 -0
  635. package/test/server/httpapi-exercise/report.ts +66 -0
  636. package/test/server/httpapi-exercise/routing.ts +96 -0
  637. package/test/server/httpapi-exercise/runner.ts +267 -0
  638. package/test/server/httpapi-exercise/runtime.ts +52 -0
  639. package/test/server/httpapi-exercise/types.ts +123 -0
  640. package/test/server/httpapi-experimental.test.ts +300 -0
  641. package/test/server/httpapi-file.test.ts +76 -0
  642. package/test/server/httpapi-global.test.ts +66 -0
  643. package/test/server/httpapi-instance-context.test.ts +347 -0
  644. package/test/server/httpapi-instance-route-auth.test.ts +84 -0
  645. package/test/server/httpapi-instance.test.ts +265 -0
  646. package/test/server/httpapi-layer.ts +33 -0
  647. package/test/server/httpapi-listen.test.ts +415 -0
  648. package/test/server/httpapi-mcp-oauth.test.ts +73 -0
  649. package/test/server/httpapi-mcp.test.ts +234 -0
  650. package/test/server/httpapi-mdns.test.ts +82 -0
  651. package/test/server/httpapi-promptasync-context.test.ts +222 -0
  652. package/test/server/httpapi-provider.test.ts +403 -0
  653. package/test/server/httpapi-pty.test.ts +275 -0
  654. package/test/server/httpapi-public-openapi.test.ts +297 -0
  655. package/test/server/httpapi-query-schema-drift.test.ts +330 -0
  656. package/test/server/httpapi-reference.test.ts +56 -0
  657. package/test/server/httpapi-schema-error-body.test.ts +165 -0
  658. package/test/server/httpapi-sdk.test.ts +909 -0
  659. package/test/server/httpapi-sync.test.ts +154 -0
  660. package/test/server/httpapi-ui.test.ts +456 -0
  661. package/test/server/httpapi-v2-location.test.ts +85 -0
  662. package/test/server/httpapi-workspace-routing.test.ts +554 -0
  663. package/test/server/httpapi-workspace.test.ts +515 -0
  664. package/test/server/project-copy.test.ts +101 -0
  665. package/test/server/project-init-git.test.ts +117 -0
  666. package/test/server/proxy-util.test.ts +113 -0
  667. package/test/server/sdk-error-shape.test.ts +84 -0
  668. package/test/server/sdk-v1-smoke.test.ts +60 -0
  669. package/test/server/workspace-proxy.test.ts +181 -0
  670. package/test/server/workspace-routing.test.ts +94 -0
  671. package/test/server/worktree-endpoint-repro.test.ts +307 -0
  672. package/test/share/share-next.test.ts +344 -0
  673. package/test/shell/shell.test.ts +99 -0
  674. package/test/skill/discovery.test.ts +139 -0
  675. package/test/skill/skill.test.ts +571 -0
  676. package/test/snapshot/snapshot.test.ts +1121 -0
  677. package/test/storage/storage.test.ts +296 -0
  678. package/test/storage/workspace-time-migration.test.ts +50 -0
  679. package/test/tool/__snapshots__/parameters.test.ts.snap +484 -0
  680. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  681. package/test/tool/apply_patch.test.ts +533 -0
  682. package/test/tool/browser.integration.test.ts +141 -0
  683. package/test/tool/desktop.integration.test.ts +129 -0
  684. package/test/tool/desktop.test.ts +85 -0
  685. package/test/tool/edit.test.ts +578 -0
  686. package/test/tool/external-directory.test.ts +167 -0
  687. package/test/tool/fixtures/large-image.png +0 -0
  688. package/test/tool/fixtures/models-api.json +117299 -0
  689. package/test/tool/glob.test.ts +188 -0
  690. package/test/tool/grep.test.ts +266 -0
  691. package/test/tool/lsp.test.ts +181 -0
  692. package/test/tool/parameters.test.ts +293 -0
  693. package/test/tool/question.test.ts +138 -0
  694. package/test/tool/read.test.ts +659 -0
  695. package/test/tool/registry.test.ts +539 -0
  696. package/test/tool/shell.test.ts +1256 -0
  697. package/test/tool/skill.test.ts +135 -0
  698. package/test/tool/task.test.ts +901 -0
  699. package/test/tool/tool-define.test.ts +153 -0
  700. package/test/tool/truncation.test.ts +266 -0
  701. package/test/tool/webfetch.test.ts +113 -0
  702. package/test/tool/websearch.test.ts +99 -0
  703. package/test/tool/write.test.ts +276 -0
  704. package/test/util/data-url.test.ts +14 -0
  705. package/test/util/error.test.ts +16 -0
  706. package/test/util/filesystem.test.ts +656 -0
  707. package/test/util/glob.test.ts +164 -0
  708. package/test/util/iife.test.ts +36 -0
  709. package/test/util/lazy.test.ts +50 -0
  710. package/test/util/log.test.ts +77 -0
  711. package/test/util/module.test.ts +59 -0
  712. package/test/util/process.test.ts +128 -0
  713. package/test/util/repository.test.ts +93 -0
  714. package/test/util/timeout.test.ts +21 -0
  715. package/test/util/wildcard.test.ts +90 -0
  716. package/tsconfig.json +16 -0
@@ -0,0 +1,1336 @@
1
+ /** @jsxImportSource @opentui/solid */
2
+ import { expect, test } from "bun:test"
3
+ import { RGBA, type BoxRenderable } from "@opentui/core"
4
+ import { testRender, useRenderer } from "@opentui/solid"
5
+ import { createSignal } from "solid-js"
6
+ import { createDefaultOpenTuiKeymap } from "@opentui/keymap/opentui"
7
+ import type { QuestionRequest } from "@octocode-ai/sdk/v2"
8
+ import { OctocodeKeymapProvider, registerOctocodeKeymap } from "@octocode-ai/tui/keymap"
9
+ import {
10
+ RUN_COMMAND_PANEL_ROWS,
11
+ RUN_SUBAGENT_PANEL_ROWS,
12
+ RunCommandMenuBody,
13
+ RunModelSelectBody,
14
+ RunQueuedPromptSelectBody,
15
+ RunSkillSelectBody,
16
+ RunSubagentSelectBody,
17
+ RunVariantSelectBody,
18
+ } from "@/cli/cmd/run/footer.command"
19
+ import { RunFooterView } from "@/cli/cmd/run/footer.view"
20
+ import { RunEntryContent } from "@/cli/cmd/run/scrollback.writer"
21
+ import { RUN_THEME_FALLBACK, type RunTheme } from "@/cli/cmd/run/theme"
22
+ import type {
23
+ FooterState,
24
+ FooterSubagentState,
25
+ FooterSubagentTab,
26
+ FooterView,
27
+ RunCommand,
28
+ RunInput,
29
+ RunPrompt,
30
+ RunProvider,
31
+ RunTuiConfig,
32
+ StreamCommit,
33
+ } from "@/cli/cmd/run/types"
34
+ import { RunQuestionBody } from "@/cli/cmd/run/footer.question"
35
+ import { RejectField } from "@/cli/cmd/run/footer.permission"
36
+ import { createTuiResolvedConfig } from "../../fixture/tui-runtime"
37
+
38
+ const tuiConfig = createTuiResolvedConfig()
39
+
40
+ function command(input: { name: string; description: string; source?: "command" | "mcp" | "skill" }) {
41
+ return {
42
+ name: input.name,
43
+ description: input.description,
44
+ source: input.source,
45
+ template: "",
46
+ hints: [],
47
+ } satisfies RunCommand
48
+ }
49
+
50
+ function model(input: {
51
+ id: string
52
+ name: string
53
+ status?: "active" | "deprecated"
54
+ cost?: number
55
+ variants?: Record<string, Record<string, never>>
56
+ }) {
57
+ return {
58
+ id: input.id,
59
+ providerID: "octo",
60
+ api: {
61
+ id: "octo",
62
+ url: "https://octocode.ai",
63
+ npm: "@ai-sdk/openai-compatible",
64
+ },
65
+ name: input.name,
66
+ capabilities: {
67
+ temperature: true,
68
+ reasoning: true,
69
+ attachment: true,
70
+ toolcall: true,
71
+ input: {
72
+ text: true,
73
+ audio: false,
74
+ image: true,
75
+ video: false,
76
+ pdf: true,
77
+ },
78
+ output: {
79
+ text: true,
80
+ audio: false,
81
+ image: false,
82
+ video: false,
83
+ pdf: false,
84
+ },
85
+ interleaved: false,
86
+ },
87
+ cost: {
88
+ input: input.cost ?? 1,
89
+ output: 1,
90
+ cache: {
91
+ read: 0,
92
+ write: 0,
93
+ },
94
+ },
95
+ limit: {
96
+ context: 128000,
97
+ output: 8192,
98
+ },
99
+ status: input.status ?? "active",
100
+ options: {},
101
+ headers: {},
102
+ release_date: "2026-01-01",
103
+ variants: input.variants,
104
+ } satisfies RunProvider["models"][string]
105
+ }
106
+
107
+ function provider() {
108
+ return {
109
+ id: "octo",
110
+ name: "octo",
111
+ source: "api",
112
+ env: [],
113
+ options: {},
114
+ models: {
115
+ "gpt-5": model({ id: "gpt-5", name: "GPT-5", variants: { high: {}, minimal: {} } }),
116
+ "gpt-free": model({ id: "gpt-free", name: "GPT Free", cost: 0 }),
117
+ old: model({ id: "old", name: "Old Model", status: "deprecated" }),
118
+ },
119
+ } satisfies RunProvider
120
+ }
121
+
122
+ function subagent(input: {
123
+ sessionID: string
124
+ label: string
125
+ description: string
126
+ status?: FooterSubagentTab["status"]
127
+ }) {
128
+ return {
129
+ sessionID: input.sessionID,
130
+ partID: `part-${input.sessionID}`,
131
+ callID: `call-${input.sessionID}`,
132
+ label: input.label,
133
+ description: input.description,
134
+ status: input.status ?? "running",
135
+ lastUpdatedAt: 1,
136
+ } satisfies FooterSubagentTab
137
+ }
138
+
139
+ function footerState(input: Partial<FooterState> = {}) {
140
+ return createSignal<FooterState>({
141
+ phase: "idle",
142
+ status: "",
143
+ queue: 0,
144
+ model: "gpt-5",
145
+ duration: "",
146
+ usage: "",
147
+ first: false,
148
+ interrupt: 0,
149
+ exit: 0,
150
+ ...input,
151
+ })[0]
152
+ }
153
+
154
+ async function renderFooter(
155
+ input: {
156
+ tuiConfig?: RunTuiConfig
157
+ commands?: RunCommand[]
158
+ theme?: () => RunTheme
159
+ providers?: RunProvider[]
160
+ currentModel?: RunInput["model"]
161
+ currentVariant?: string
162
+ subagents?: FooterSubagentState
163
+ backgroundSubagents?: boolean
164
+ width?: number
165
+ height?: number
166
+ state?: Partial<FooterState>
167
+ onCycle?: () => void
168
+ onSubmit?: (prompt: RunPrompt) => boolean
169
+ } = {},
170
+ ) {
171
+ const [view] = createSignal<FooterView>({ type: "prompt" })
172
+ const [subagents] = createSignal<FooterSubagentState>(
173
+ input.subagents ?? { tabs: [], details: {}, permissions: [], questions: [] },
174
+ )
175
+ const state = footerState(input.state)
176
+ const config = input.tuiConfig ?? tuiConfig
177
+ let offKeymap: (() => void) | undefined
178
+
179
+ function Harness() {
180
+ const renderer = useRenderer()
181
+ const keymap = createDefaultOpenTuiKeymap(renderer)
182
+ offKeymap = registerOctocodeKeymap(keymap, renderer, config)
183
+
184
+ return (
185
+ <OctocodeKeymapProvider keymap={keymap}>
186
+ <RunFooterView
187
+ directory="/tmp"
188
+ findFiles={async () => []}
189
+ agents={() => []}
190
+ resources={() => []}
191
+ commands={() => input.commands ?? []}
192
+ providers={() => input.providers}
193
+ currentModel={() => input.currentModel}
194
+ variants={() => []}
195
+ currentVariant={() => input.currentVariant}
196
+ state={state}
197
+ view={view}
198
+ subagent={subagents}
199
+ theme={input.theme ?? (() => RUN_THEME_FALLBACK)}
200
+ tuiConfig={config}
201
+ backgroundSubagents={input.backgroundSubagents ?? true}
202
+ agent="octo"
203
+ onSubmit={input.onSubmit ?? (() => true)}
204
+ onPermissionReply={() => {}}
205
+ onQuestionReply={() => {}}
206
+ onQuestionReject={() => {}}
207
+ onCycle={input.onCycle ?? (() => {})}
208
+ onInterrupt={() => false}
209
+ onEditorOpen={async () => undefined}
210
+ onInputClear={() => {}}
211
+ onExit={() => {}}
212
+ onModelSelect={() => {}}
213
+ onVariantSelect={() => {}}
214
+ onRows={() => {}}
215
+ onLayout={() => {}}
216
+ onStatus={() => {}}
217
+ onQueuedRemove={async () => true}
218
+ />
219
+ </OctocodeKeymapProvider>
220
+ )
221
+ }
222
+
223
+ const app = await testRender(
224
+ () => (
225
+ <box width={input.width ?? 100} height={input.height ?? 8}>
226
+ <Harness />
227
+ </box>
228
+ ),
229
+ { width: input.width ?? 100, height: input.height ?? 8, kittyKeyboard: true },
230
+ )
231
+
232
+ return {
233
+ ...app,
234
+ cleanup() {
235
+ app.renderer.currentFocusedRenderable?.blur()
236
+ app.renderer.currentFocusedEditor?.blur()
237
+ offKeymap?.()
238
+ offKeymap = undefined
239
+ app.renderer.destroy()
240
+ },
241
+ }
242
+ }
243
+
244
+ function expectPaletteList(list: BoxRenderable, selectedIndex: number) {
245
+ expect(list.backgroundColor.toInts()).toEqual((RUN_THEME_FALLBACK.footer.shade as RGBA).toInts())
246
+ expect((list.getChildren()[selectedIndex] as BoxRenderable).backgroundColor.toInts()).toEqual(
247
+ (RUN_THEME_FALLBACK.footer.selected as RGBA).toInts(),
248
+ )
249
+ }
250
+
251
+ test("direct footer composer area does not adopt footer surface", async () => {
252
+ const surface = RGBA.fromHex("#123456")
253
+ const [theme, setTheme] = createSignal(RUN_THEME_FALLBACK)
254
+ const app = await renderFooter({ theme })
255
+
256
+ try {
257
+ await app.renderOnce()
258
+ const area = app.renderer.root.findDescendantById("run-direct-footer-composer-area") as BoxRenderable
259
+
260
+ expect(area.backgroundColor.toInts()).not.toEqual(surface.toInts())
261
+ setTheme({
262
+ ...RUN_THEME_FALLBACK,
263
+ footer: {
264
+ ...RUN_THEME_FALLBACK.footer,
265
+ surface,
266
+ },
267
+ })
268
+ await app.renderOnce()
269
+
270
+ expect(area.backgroundColor.toInts()).not.toEqual(surface.toInts())
271
+ } finally {
272
+ app.cleanup()
273
+ }
274
+ })
275
+
276
+ test("run entry content updates when live commit text changes", async () => {
277
+ const [commit, setCommit] = createSignal<StreamCommit>({
278
+ kind: "tool",
279
+ text: "I",
280
+ phase: "progress",
281
+ source: "tool",
282
+ messageID: "msg-1",
283
+ partID: "part-1",
284
+ tool: "bash",
285
+ })
286
+
287
+ const app = await testRender(
288
+ () => (
289
+ <box width={80} height={4}>
290
+ <RunEntryContent commit={commit()} theme={RUN_THEME_FALLBACK} width={80} />
291
+ </box>
292
+ ),
293
+ {
294
+ width: 80,
295
+ height: 4,
296
+ },
297
+ )
298
+
299
+ try {
300
+ await app.renderOnce()
301
+ expect(app.captureCharFrame()).toContain("I")
302
+
303
+ setCommit({
304
+ kind: "tool",
305
+ text: "I need to inspect the codebase",
306
+ phase: "progress",
307
+ source: "tool",
308
+ messageID: "msg-1",
309
+ partID: "part-1",
310
+ tool: "bash",
311
+ })
312
+ await app.renderOnce()
313
+
314
+ expect(app.captureCharFrame()).toContain("I need to inspect the codebase")
315
+ } finally {
316
+ app.renderer.destroy()
317
+ }
318
+ })
319
+
320
+ test("direct command panel renders grouped command palette", async () => {
321
+ const [commands] = createSignal<RunCommand[] | undefined>([
322
+ command({ name: "review", description: "Review code" }),
323
+ command({ name: "deploy", description: "Deploy prompt", source: "mcp" }),
324
+ command({ name: "internal", description: "Skill command", source: "skill" }),
325
+ ])
326
+ const [subagents] = createSignal([])
327
+ const [variants] = createSignal(["high", "minimal"])
328
+
329
+ const app = await testRender(
330
+ () => (
331
+ <box width={100} height={RUN_COMMAND_PANEL_ROWS}>
332
+ <RunCommandMenuBody
333
+ theme={() => RUN_THEME_FALLBACK.footer}
334
+ commands={commands}
335
+ subagents={subagents}
336
+ queued={() => []}
337
+ variants={variants}
338
+ variantCycle="ctrl+t"
339
+ onClose={() => {}}
340
+ onModel={() => {}}
341
+ onEditor={() => {}}
342
+ onSkill={() => {}}
343
+ onSubagent={() => {}}
344
+ onQueued={() => {}}
345
+ onVariant={() => {}}
346
+ onVariantCycle={() => {}}
347
+ onCommand={() => {}}
348
+ onNew={() => {}}
349
+ onExit={() => {}}
350
+ />
351
+ </box>
352
+ ),
353
+ {
354
+ width: 100,
355
+ height: RUN_COMMAND_PANEL_ROWS,
356
+ },
357
+ )
358
+
359
+ try {
360
+ await app.renderOnce()
361
+ const frame = app.captureCharFrame()
362
+
363
+ expect(frame).toContain("Commands")
364
+ expect(frame).toContain("Search")
365
+ expect(frame).toContain("Session")
366
+ expect(frame).toContain("Agent")
367
+ expect(frame).toContain("Prompt")
368
+ expect(frame).toContain("Open editor")
369
+ expect(frame).toContain("/editor")
370
+ expect(frame).toContain("Switch model")
371
+ expect(frame).toContain("Skills")
372
+ expect(frame).toContain("/skills")
373
+ expect(frame.match(/\bAgent\b/g)?.length).toBe(1)
374
+ expect(frame).not.toContain("┌")
375
+ expect(frame).not.toContain("┃")
376
+ expect(frame).not.toContain("/internal")
377
+ expect(frame).not.toContain("Choose model for future turns")
378
+ expect(frame).not.toContain("Cycle reasoning effort for future turns")
379
+ expect(frame).not.toContain("Review code")
380
+ expect(frame).not.toContain("Commands 8")
381
+ } finally {
382
+ app.renderer.destroy()
383
+ }
384
+ })
385
+
386
+ test("direct skill panel renders searchable skill list", async () => {
387
+ const [commands] = createSignal<RunCommand[] | undefined>([
388
+ command({ name: "review", description: "Review code" }),
389
+ command({ name: "internal", description: "Skill command", source: "skill" }),
390
+ command({ name: "formatter", description: "Apply formatter fixes", source: "skill" }),
391
+ ])
392
+
393
+ const app = await testRender(
394
+ () => (
395
+ <box width={100} height={RUN_COMMAND_PANEL_ROWS}>
396
+ <RunSkillSelectBody
397
+ theme={() => RUN_THEME_FALLBACK.footer}
398
+ commands={commands}
399
+ onClose={() => {}}
400
+ onSelect={() => {}}
401
+ />
402
+ </box>
403
+ ),
404
+ {
405
+ width: 100,
406
+ height: RUN_COMMAND_PANEL_ROWS,
407
+ },
408
+ )
409
+
410
+ try {
411
+ await app.renderOnce()
412
+ const frame = app.captureCharFrame()
413
+
414
+ expect(frame).toContain("Skills")
415
+ expect(frame).toContain("Search")
416
+ expect(frame).toContain("internal")
417
+ expect(frame).not.toContain("/internal")
418
+ expect(frame).toContain("formatter")
419
+ expect(frame).toContain("Apply formatter fixes")
420
+ expect(frame).not.toContain("review")
421
+ } finally {
422
+ app.renderer.destroy()
423
+ }
424
+ })
425
+
426
+ test("direct skill panel truncates long descriptions from the end", async () => {
427
+ const [commands] = createSignal<RunCommand[] | undefined>([
428
+ command({
429
+ name: "terminal-control",
430
+ description:
431
+ "Control and test terminal applications, REPLs, interactive CLIs, shell processes, OpenTUI applications, or other terminal-backed workflows.",
432
+ source: "skill",
433
+ }),
434
+ ])
435
+
436
+ const app = await testRender(
437
+ () => (
438
+ <box width={100} height={RUN_COMMAND_PANEL_ROWS}>
439
+ <RunSkillSelectBody
440
+ theme={() => RUN_THEME_FALLBACK.footer}
441
+ commands={commands}
442
+ onClose={() => {}}
443
+ onSelect={() => {}}
444
+ />
445
+ </box>
446
+ ),
447
+ {
448
+ width: 100,
449
+ height: RUN_COMMAND_PANEL_ROWS,
450
+ },
451
+ )
452
+
453
+ try {
454
+ await app.renderOnce()
455
+ const frame = app.captureCharFrame()
456
+
457
+ expect(frame).toContain("terminal-control")
458
+ expect(frame).toContain("Control and test terminal applications")
459
+ expect(frame).not.toMatch(/application(?:…|\.\.\.)ocess/)
460
+ } finally {
461
+ app.renderer.destroy()
462
+ }
463
+ })
464
+
465
+ test("direct command panel shows subagent entry when available", async () => {
466
+ const [commands] = createSignal<RunCommand[] | undefined>([])
467
+ const [subagents] = createSignal([subagent({ sessionID: "s-1", label: "Explore", description: "Inspect auth flow" })])
468
+ const [variants] = createSignal<string[]>([])
469
+
470
+ const app = await testRender(
471
+ () => (
472
+ <box width={100} height={RUN_COMMAND_PANEL_ROWS}>
473
+ <RunCommandMenuBody
474
+ theme={() => RUN_THEME_FALLBACK.footer}
475
+ commands={commands}
476
+ subagents={subagents}
477
+ queued={() => []}
478
+ variants={variants}
479
+ variantCycle="ctrl+t"
480
+ onClose={() => {}}
481
+ onModel={() => {}}
482
+ onEditor={() => {}}
483
+ onSkill={() => {}}
484
+ onSubagent={() => {}}
485
+ onQueued={() => {}}
486
+ onVariant={() => {}}
487
+ onVariantCycle={() => {}}
488
+ onCommand={() => {}}
489
+ onNew={() => {}}
490
+ onExit={() => {}}
491
+ />
492
+ </box>
493
+ ),
494
+ {
495
+ width: 100,
496
+ height: RUN_COMMAND_PANEL_ROWS,
497
+ },
498
+ )
499
+
500
+ try {
501
+ await app.renderOnce()
502
+ const frame = app.captureCharFrame()
503
+
504
+ expect(frame).toContain("View subagents")
505
+ expect(frame).toContain("1 active")
506
+ } finally {
507
+ app.renderer.destroy()
508
+ }
509
+ })
510
+
511
+ test("direct command panel keeps completed subagents available", async () => {
512
+ const [commands] = createSignal<RunCommand[] | undefined>([])
513
+ const [subagents] = createSignal([
514
+ subagent({ sessionID: "s-1", label: "Explore", description: "Inspect auth flow", status: "completed" }),
515
+ ])
516
+ const [variants] = createSignal<string[]>([])
517
+
518
+ const app = await testRender(
519
+ () => (
520
+ <box width={100} height={RUN_COMMAND_PANEL_ROWS}>
521
+ <RunCommandMenuBody
522
+ theme={() => RUN_THEME_FALLBACK.footer}
523
+ commands={commands}
524
+ subagents={subagents}
525
+ queued={() => []}
526
+ variants={variants}
527
+ variantCycle="ctrl+t"
528
+ onClose={() => {}}
529
+ onModel={() => {}}
530
+ onEditor={() => {}}
531
+ onSkill={() => {}}
532
+ onSubagent={() => {}}
533
+ onQueued={() => {}}
534
+ onVariant={() => {}}
535
+ onVariantCycle={() => {}}
536
+ onCommand={() => {}}
537
+ onNew={() => {}}
538
+ onExit={() => {}}
539
+ />
540
+ </box>
541
+ ),
542
+ {
543
+ width: 100,
544
+ height: RUN_COMMAND_PANEL_ROWS,
545
+ },
546
+ )
547
+
548
+ try {
549
+ await app.renderOnce()
550
+ const frame = app.captureCharFrame()
551
+
552
+ expect(frame).toContain("View subagents")
553
+ expect(frame).toContain("1 recent")
554
+ } finally {
555
+ app.renderer.destroy()
556
+ }
557
+ })
558
+
559
+ test("direct subagent panel renders active subagents", async () => {
560
+ const [tabs] = createSignal([
561
+ subagent({ sessionID: "s-1", label: "Explore", description: "Inspect auth flow" }),
562
+ subagent({ sessionID: "s-2", label: "General", description: "Write migration plan", status: "completed" }),
563
+ ])
564
+ const [current] = createSignal<string | undefined>("s-1")
565
+ let rows = 0
566
+
567
+ const app = await testRender(
568
+ () => (
569
+ <box width={100} height={RUN_SUBAGENT_PANEL_ROWS}>
570
+ <RunSubagentSelectBody
571
+ theme={() => RUN_THEME_FALLBACK.footer}
572
+ tabs={tabs}
573
+ current={current}
574
+ onClose={() => {}}
575
+ onSelect={() => {}}
576
+ onRows={(value) => {
577
+ rows = value
578
+ }}
579
+ />
580
+ </box>
581
+ ),
582
+ {
583
+ width: 100,
584
+ height: RUN_SUBAGENT_PANEL_ROWS,
585
+ },
586
+ )
587
+
588
+ try {
589
+ await app.renderOnce()
590
+ const frame = app.captureCharFrame()
591
+ const list = app.renderer.root.findDescendantById("run-direct-footer-subagent-list") as BoxRenderable
592
+
593
+ expect(frame).toContain("Select subagent")
594
+ expect(frame).toContain("Inspect auth flow")
595
+ expect(frame).toContain("Write migration plan")
596
+ expect(frame).toContain("done")
597
+ expect(frame).not.toContain("┌")
598
+ expect(frame).not.toContain("┃")
599
+ expectPaletteList(list, 0)
600
+ expect(rows).toBe(8)
601
+ } finally {
602
+ app.renderer.destroy()
603
+ }
604
+ })
605
+
606
+ test("direct queued prompt panel renders pending prompt actions", async () => {
607
+ const [prompts] = createSignal([
608
+ { messageID: "m-1", partID: "p-1", prompt: { text: "fix the auth test", parts: [] } },
609
+ ])
610
+
611
+ const app = await testRender(
612
+ () => (
613
+ <box width={100} height={RUN_SUBAGENT_PANEL_ROWS}>
614
+ <RunQueuedPromptSelectBody
615
+ theme={() => RUN_THEME_FALLBACK.footer}
616
+ prompts={prompts}
617
+ onClose={() => {}}
618
+ onEdit={() => {}}
619
+ onDelete={() => {}}
620
+ />
621
+ </box>
622
+ ),
623
+ { width: 100, height: RUN_SUBAGENT_PANEL_ROWS },
624
+ )
625
+
626
+ try {
627
+ await app.renderOnce()
628
+ const frame = app.captureCharFrame()
629
+ const list = app.renderer.root.findDescendantById("run-direct-footer-queued-list") as BoxRenderable
630
+
631
+ expect(frame).toContain("Queued prompts")
632
+ expect(frame).toContain("fix the auth test")
633
+ expect(frame).toContain("queued")
634
+ expect(frame).not.toContain("┌")
635
+ expect(frame).not.toContain("┃")
636
+ expectPaletteList(list, 0)
637
+ } finally {
638
+ app.renderer.destroy()
639
+ }
640
+ })
641
+
642
+ // OpenTUI currently crashes Bun in the full `test/cli/run` directory run here.
643
+ // Re-enable after the upstream OpenTUI fix lands in this repo.
644
+ test.skip("direct footer recreates the frame across command panel transitions", async () => {
645
+ const app = await renderFooter()
646
+
647
+ try {
648
+ await app.renderOnce()
649
+
650
+ for (let index = 0; index < 3; index++) {
651
+ const composerFrame = app.renderer.root.findDescendantById("run-direct-footer-composer-frame") as BoxRenderable
652
+ app.mockInput.pressKey("p", { ctrl: true })
653
+ await app.renderOnce()
654
+
655
+ expect(app.captureCharFrame()).toContain("Commands")
656
+ expect(app.renderer.root.findDescendantById("run-direct-footer-composer-frame")).not.toBe(composerFrame)
657
+ app.mockInput.pressKey("c", { ctrl: true })
658
+ await app.renderOnce()
659
+ expect(app.captureCharFrame()).not.toContain("Commands")
660
+ expect(app.captureCharFrame()).not.toContain("┃")
661
+ expect(app.captureCharFrame()).not.toContain("█")
662
+ }
663
+ } finally {
664
+ app.cleanup()
665
+ }
666
+ })
667
+
668
+ test.skip("direct footer dispatches leader variant binding only when leader is registered", async () => {
669
+ const calls: string[] = []
670
+ const app = await renderFooter({
671
+ tuiConfig: createTuiResolvedConfig({ keybinds: { leader: "ctrl+x", variant_cycle: "<leader>t" } }),
672
+ onCycle: () => calls.push("cycle"),
673
+ })
674
+
675
+ try {
676
+ await app.renderOnce()
677
+ app.mockInput.pressKey("t")
678
+ expect(calls).toEqual([])
679
+
680
+ app.mockInput.pressKey("x", { ctrl: true })
681
+ app.mockInput.pressKey("t")
682
+ expect(calls).toEqual(["cycle"])
683
+ } finally {
684
+ app.cleanup()
685
+ }
686
+ })
687
+
688
+ test("direct footer keeps leader variant binding inactive when leader is disabled", async () => {
689
+ const calls: string[] = []
690
+ const app = await renderFooter({
691
+ tuiConfig: createTuiResolvedConfig({ keybinds: { leader: "none", variant_cycle: "<leader>t" } }),
692
+ onCycle: () => calls.push("cycle"),
693
+ })
694
+
695
+ try {
696
+ await app.renderOnce()
697
+ app.mockInput.pressKey("t")
698
+ app.mockInput.pressKey("x", { ctrl: true })
699
+ app.mockInput.pressKey("t")
700
+
701
+ expect(calls).toEqual([])
702
+ } finally {
703
+ app.cleanup()
704
+ }
705
+ })
706
+
707
+ test("direct footer submits slash autocomplete selections without dispatching shell completions", async () => {
708
+ const submits: RunPrompt[] = []
709
+ const app = await renderFooter({
710
+ commands: [command({ name: "review", description: "Review code" })],
711
+ onSubmit(prompt) {
712
+ submits.push(prompt)
713
+ return true
714
+ },
715
+ })
716
+
717
+ try {
718
+ await app.renderOnce()
719
+ "/rev".split("").forEach((key) => app.mockInput.pressKey(key))
720
+ await app.renderOnce()
721
+ app.mockInput.pressEnter()
722
+ await app.renderOnce()
723
+
724
+ "/rev".split("").forEach((key) => app.mockInput.pressKey(key))
725
+ await app.renderOnce()
726
+ app.mockInput.pressKey("TAB")
727
+ await app.renderOnce()
728
+
729
+ "/re branch".split("").forEach((key) => app.mockInput.pressKey(key))
730
+ Array.from({ length: 7 }).forEach(() => app.mockInput.pressKey("ARROW_LEFT"))
731
+ app.mockInput.pressKey("v")
732
+ await app.renderOnce()
733
+ app.mockInput.pressEnter()
734
+ await app.renderOnce()
735
+
736
+ "/nx".split("").forEach((key) => app.mockInput.pressKey(key))
737
+ app.mockInput.pressKey("ARROW_LEFT")
738
+ app.mockInput.pressKey("e")
739
+ await app.renderOnce()
740
+ app.mockInput.pressEnter()
741
+ await app.renderOnce()
742
+
743
+ "/n scratch".split("").forEach((key) => app.mockInput.pressKey(key))
744
+ Array.from({ length: 8 }).forEach(() => app.mockInput.pressKey("ARROW_LEFT"))
745
+ app.mockInput.pressKey("e")
746
+ await app.renderOnce()
747
+ app.mockInput.pressEnter()
748
+ await app.renderOnce()
749
+
750
+ app.mockInput.pressKey("!")
751
+ "/rev".split("").forEach((key) => app.mockInput.pressKey(key))
752
+ await app.renderOnce()
753
+ app.mockInput.pressEnter()
754
+ await app.renderOnce()
755
+
756
+ expect(submits).toEqual([
757
+ { text: "/review ", parts: [], command: { name: "review", arguments: "" } },
758
+ { text: "/review ", parts: [], command: { name: "review", arguments: "" } },
759
+ { text: "/review branch", parts: [], command: { name: "review", arguments: "branch" } },
760
+ { text: "/new ", parts: [] },
761
+ { text: "/new ", parts: [] },
762
+ ])
763
+ expect(app.captureCharFrame()).toContain("/review")
764
+ } finally {
765
+ app.cleanup()
766
+ }
767
+ })
768
+
769
+ test("direct footer slash autocomplete keeps a real skills command", async () => {
770
+ const submits: RunPrompt[] = []
771
+ const app = await renderFooter({
772
+ commands: [
773
+ command({ name: "skills", description: "Run the real skills command" }),
774
+ command({ name: "formatter", description: "Apply formatter fixes", source: "skill" }),
775
+ ],
776
+ onSubmit(prompt) {
777
+ submits.push(prompt)
778
+ return true
779
+ },
780
+ })
781
+
782
+ try {
783
+ await app.renderOnce()
784
+ "/skills".split("").forEach((key) => app.mockInput.pressKey(key))
785
+ await app.renderOnce()
786
+ app.mockInput.pressEnter()
787
+ await app.renderOnce()
788
+
789
+ expect(submits).toEqual([{ text: "/skills ", parts: [], command: { name: "skills", arguments: "" } }])
790
+ expect(app.captureCharFrame()).not.toContain("Apply formatter fixes")
791
+ } finally {
792
+ app.cleanup()
793
+ }
794
+ })
795
+
796
+ // OpenTUI currently segfaults Bun while tearing down this composer-to-skill-panel transition.
797
+ // Re-enable after the upstream renderer teardown fix lands.
798
+ test.skip("direct footer skill picker inserts an editable bound skill command", async () => {
799
+ const submits: RunPrompt[] = []
800
+ const app = await renderFooter({
801
+ commands: [command({ name: "new", description: "Skill named new", source: "skill" })],
802
+ onSubmit(prompt) {
803
+ submits.push(prompt)
804
+ return true
805
+ },
806
+ })
807
+
808
+ try {
809
+ await app.renderOnce()
810
+ "/skills".split("").forEach((key) => app.mockInput.pressKey(key))
811
+ await app.renderOnce()
812
+ app.mockInput.pressEnter()
813
+ await app.renderOnce()
814
+
815
+ expect(app.captureCharFrame()).toContain("Skill named new")
816
+
817
+ app.mockInput.pressEnter()
818
+ await app.renderOnce()
819
+
820
+ expect(submits).toEqual([])
821
+ expect(app.captureCharFrame()).toContain("/new")
822
+
823
+ "task".split("").forEach((key) => app.mockInput.pressKey(key))
824
+ await app.renderOnce()
825
+ app.mockInput.pressEnter()
826
+ await app.renderOnce()
827
+
828
+ expect(submits).toEqual([{ text: "/new task", parts: [], command: { name: "new", arguments: "task" } }])
829
+ } finally {
830
+ app.cleanup()
831
+ }
832
+ })
833
+
834
+ // OpenTUI currently segfaults Bun while tearing down this skill-panel close transition.
835
+ // Re-enable after the upstream renderer teardown fix lands.
836
+ test.skip("direct footer clears the synthetic skills draft when the panel closes", async () => {
837
+ const submits: RunPrompt[] = []
838
+ const app = await renderFooter({
839
+ commands: [command({ name: "formatter", description: "Apply formatter fixes", source: "skill" })],
840
+ onSubmit(prompt) {
841
+ submits.push(prompt)
842
+ return true
843
+ },
844
+ })
845
+
846
+ try {
847
+ await app.renderOnce()
848
+ "/skills".split("").forEach((key) => app.mockInput.pressKey(key))
849
+ await app.renderOnce()
850
+ app.mockInput.pressEnter()
851
+ await app.renderOnce()
852
+
853
+ expect(app.captureCharFrame()).toContain("Apply formatter fixes")
854
+
855
+ app.mockInput.pressKey("c", { ctrl: true })
856
+ await app.renderOnce()
857
+ app.mockInput.pressEnter()
858
+ await app.renderOnce()
859
+
860
+ expect(submits).toEqual([])
861
+ expect(app.captureCharFrame()).not.toContain("/skills")
862
+ } finally {
863
+ app.cleanup()
864
+ }
865
+ })
866
+
867
+ test("direct footer shows editable prompts and additional queued work while running", async () => {
868
+ const [state] = createSignal<FooterState>({
869
+ phase: "running",
870
+ status: "",
871
+ queue: 3,
872
+ model: "gpt-5",
873
+ duration: "",
874
+ usage: "",
875
+ first: false,
876
+ interrupt: 0,
877
+ exit: 0,
878
+ })
879
+ const [view] = createSignal<FooterView>({ type: "prompt" })
880
+ const [subagents] = createSignal<FooterSubagentState>({
881
+ tabs: [subagent({ sessionID: "s-1", label: "Explore", description: "Inspect auth flow" })],
882
+ details: {},
883
+ permissions: [],
884
+ questions: [],
885
+ })
886
+ let offKeymap: (() => void) | undefined
887
+ function Harness() {
888
+ const renderer = useRenderer()
889
+ const keymap = createDefaultOpenTuiKeymap(renderer)
890
+ offKeymap = registerOctocodeKeymap(keymap, renderer, tuiConfig)
891
+
892
+ return (
893
+ <OctocodeKeymapProvider keymap={keymap}>
894
+ <RunFooterView
895
+ directory="/tmp"
896
+ findFiles={async () => []}
897
+ agents={() => []}
898
+ resources={() => []}
899
+ commands={() => []}
900
+ providers={() => undefined}
901
+ currentModel={() => ({
902
+ providerID: "octo",
903
+ modelID: "a-model-name-long-enough-to-force-responsive-truncation",
904
+ })}
905
+ variants={() => []}
906
+ currentVariant={() => undefined}
907
+ state={state}
908
+ view={view}
909
+ subagent={subagents}
910
+ queuedPrompts={() => [
911
+ { messageID: "m-queued", partID: "p-queued", prompt: { text: "follow up", parts: [] } },
912
+ ]}
913
+ theme={() => RUN_THEME_FALLBACK}
914
+ tuiConfig={tuiConfig}
915
+ backgroundSubagents={true}
916
+ agent="octo"
917
+ onSubmit={() => true}
918
+ onPermissionReply={() => {}}
919
+ onQuestionReply={() => {}}
920
+ onQuestionReject={() => {}}
921
+ onCycle={() => {}}
922
+ onInterrupt={() => false}
923
+ onEditorOpen={async () => undefined}
924
+ onInputClear={() => {}}
925
+ onExit={() => {}}
926
+ onModelSelect={() => {}}
927
+ onVariantSelect={() => {}}
928
+ onRows={() => {}}
929
+ onLayout={() => {}}
930
+ onStatus={() => {}}
931
+ onQueuedRemove={async () => true}
932
+ />
933
+ </OctocodeKeymapProvider>
934
+ )
935
+ }
936
+
937
+ const app = await testRender(
938
+ () => (
939
+ <box width={160} height={8}>
940
+ <Harness />
941
+ </box>
942
+ ),
943
+ {
944
+ width: 160,
945
+ height: 8,
946
+ },
947
+ )
948
+
949
+ try {
950
+ await app.renderOnce()
951
+ const frame = app.captureCharFrame()
952
+ const transparent = RGBA.fromValues(0, 0, 0, 0).toInts()
953
+ const tinted = (RUN_THEME_FALLBACK.footer.status as RGBA).toInts()
954
+ const accent = (RUN_THEME_FALLBACK.footer.statusAccent as RGBA).toInts()
955
+ const statusline = app.renderer.root.findDescendantById("run-direct-footer-statusline") as BoxRenderable
956
+ const mode = app.renderer.root.findDescendantById("run-direct-footer-statusline-mode") as BoxRenderable
957
+ const main = app.renderer.root.findDescendantById("run-direct-footer-statusline-main") as BoxRenderable
958
+ const spinner = app.renderer.root.findDescendantById("run-direct-footer-status-spinner")
959
+ const model = app.renderer.root.findDescendantById("run-direct-footer-statusline-model") as BoxRenderable
960
+ const queued = app.renderer.root.findDescendantById("run-direct-footer-statusline-queued") as BoxRenderable
961
+ const hint = app.renderer.root.findDescendantById("run-direct-footer-statusline-hint") as BoxRenderable
962
+
963
+ expect(spinner).toBeDefined()
964
+ expect(frame).toContain("a-model-name-long-enough-to-force-responsive-truncation")
965
+ expect(frame).toContain("3 queued")
966
+ expect(frame).toContain("ctrl+b background")
967
+ expect(frame).toContain("ctrl+x q 3 queued")
968
+ expect(frame).toContain("ctrl+x down subagents")
969
+ expect(frame).toContain("ctrl+p cmd")
970
+ expect(frame).toContain("a-model-name-long-enough-to-force-responsive-truncation")
971
+ expect(frame).toContain("subagents · ctrl+p cmd")
972
+ expect(frame).not.toContain("1 agent")
973
+ expect(statusline.backgroundColor.toInts()).toEqual(tinted)
974
+ expect(mode.backgroundColor.toInts()).toEqual(accent)
975
+ expect(main.backgroundColor.toInts()).toEqual(transparent)
976
+ expect(model.backgroundColor.toInts()).toEqual(transparent)
977
+ expect(queued.backgroundColor.toInts()).toEqual(transparent)
978
+ expect(hint.backgroundColor.toInts()).toEqual(transparent)
979
+ } finally {
980
+ app.renderer.currentFocusedRenderable?.blur()
981
+ app.renderer.currentFocusedEditor?.blur()
982
+ offKeymap?.()
983
+ app.renderer.destroy()
984
+ }
985
+ })
986
+
987
+ test("direct footer separates a lone context hint from model and command hint", async () => {
988
+ const app = await renderFooter({
989
+ providers: [provider()],
990
+ currentModel: { providerID: "octo", modelID: "gpt-5" },
991
+ currentVariant: "xhigh",
992
+ subagents: {
993
+ tabs: [subagent({ sessionID: "s-1", label: "Explore", description: "Inspect auth flow" })],
994
+ details: {},
995
+ permissions: [],
996
+ questions: [],
997
+ },
998
+ backgroundSubagents: false,
999
+ width: 160,
1000
+ })
1001
+
1002
+ try {
1003
+ await app.renderOnce()
1004
+ const frame = app.captureCharFrame()
1005
+
1006
+ expect(frame).toContain("GPT-5")
1007
+ expect(frame).toContain("xhigh · ctrl+x down subagents · ctrl+p cmd")
1008
+ expect(frame).not.toContain("ctrl+b background")
1009
+ expect(frame).not.toContain("queued")
1010
+ } finally {
1011
+ app.cleanup()
1012
+ }
1013
+ })
1014
+
1015
+ test("direct footer hides the subagent hint when only completed subagents remain", async () => {
1016
+ const app = await renderFooter({
1017
+ providers: [provider()],
1018
+ currentModel: { providerID: "octo", modelID: "gpt-5" },
1019
+ currentVariant: "xhigh",
1020
+ subagents: {
1021
+ tabs: [subagent({ sessionID: "s-1", label: "Explore", description: "Inspect auth flow", status: "completed" })],
1022
+ details: {},
1023
+ permissions: [],
1024
+ questions: [],
1025
+ },
1026
+ backgroundSubagents: false,
1027
+ width: 160,
1028
+ })
1029
+
1030
+ try {
1031
+ await app.renderOnce()
1032
+ const frame = app.captureCharFrame()
1033
+
1034
+ expect(frame).toContain("GPT-5")
1035
+ expect(frame).toContain("xhigh · ctrl+p cmd")
1036
+ expect(frame).not.toContain("ctrl+x down subagents")
1037
+ } finally {
1038
+ app.cleanup()
1039
+ }
1040
+ })
1041
+
1042
+ test("direct footer omits interrupt key hint when interrupt is unbound", async () => {
1043
+ const app = await renderFooter({
1044
+ tuiConfig: createTuiResolvedConfig({ keybinds: { session_interrupt: "none", input_clear: "ctrl+l" } }),
1045
+ state: { phase: "running" },
1046
+ })
1047
+
1048
+ try {
1049
+ await app.renderOnce()
1050
+ const frame = app.captureCharFrame()
1051
+
1052
+ expect(frame).toContain("interrupt")
1053
+ expect(frame).not.toContain("ctrl+l")
1054
+ } finally {
1055
+ app.cleanup()
1056
+ }
1057
+ })
1058
+
1059
+ test("direct footer shows full usage metadata when room is available", async () => {
1060
+ const app = await renderFooter({
1061
+ state: { usage: "159.6K (16%) · $4.23" },
1062
+ })
1063
+
1064
+ try {
1065
+ await app.renderOnce()
1066
+ const frame = app.captureCharFrame()
1067
+
1068
+ expect(frame).toContain("159.6K (16%) · $4.23")
1069
+ } finally {
1070
+ app.cleanup()
1071
+ }
1072
+ })
1073
+
1074
+ test("direct footer mode label keeps left padding without a status pill", async () => {
1075
+ const app = await renderFooter()
1076
+
1077
+ try {
1078
+ await app.renderOnce()
1079
+ const statusline = app
1080
+ .captureCharFrame()
1081
+ .split("\n")
1082
+ .find((line) => line.includes("BUILD") && line.includes("cmd"))
1083
+
1084
+ expect(statusline).toBeDefined()
1085
+ expect(statusline?.startsWith(" BUILD ")).toBe(true)
1086
+ } finally {
1087
+ app.cleanup()
1088
+ }
1089
+ })
1090
+
1091
+ test("direct question body separates single-select checkmark from label", async () => {
1092
+ const request = {
1093
+ id: "question-1",
1094
+ sessionID: "session-1",
1095
+ questions: [
1096
+ {
1097
+ question: "Which categorical concept is often described as a universal way to combine two objects?",
1098
+ header: "Universal Product",
1099
+ options: [
1100
+ { label: "Product", description: "A product comes with projections." },
1101
+ { label: "Equalizer", description: "An equalizer selects morphisms where arrows agree." },
1102
+ ],
1103
+ },
1104
+ ],
1105
+ } satisfies QuestionRequest
1106
+ const replies: unknown[] = []
1107
+
1108
+ const app = await testRender(
1109
+ () => (
1110
+ <box width={100} height={12}>
1111
+ <RunQuestionBody
1112
+ request={request}
1113
+ theme={RUN_THEME_FALLBACK.footer}
1114
+ onReply={(input) => {
1115
+ replies.push(input)
1116
+ }}
1117
+ onReject={() => {}}
1118
+ />
1119
+ </box>
1120
+ ),
1121
+ {
1122
+ width: 100,
1123
+ height: 12,
1124
+ },
1125
+ )
1126
+
1127
+ try {
1128
+ app.mockInput.pressEnter()
1129
+ await app.renderOnce()
1130
+
1131
+ expect(replies).toHaveLength(1)
1132
+ expect(app.captureCharFrame()).toContain("Product ✓")
1133
+ } finally {
1134
+ app.renderer.destroy()
1135
+ }
1136
+ })
1137
+
1138
+ // OpenTUI currently segfaults while tearing down this textarea-backed keymap renderer.
1139
+ // Re-enable after the runtime fix.
1140
+ test.skip("direct custom answer submits through keymap return binding", async () => {
1141
+ const question = {
1142
+ id: "question-1",
1143
+ sessionID: "session-1",
1144
+ questions: [
1145
+ {
1146
+ question: "Which answer should I use?",
1147
+ header: "Answer",
1148
+ options: [{ label: "Provided", description: "Use the listed answer." }],
1149
+ custom: true,
1150
+ },
1151
+ ],
1152
+ } satisfies QuestionRequest
1153
+ const questions: unknown[] = []
1154
+ let off: (() => void) | undefined
1155
+
1156
+ function Harness() {
1157
+ const renderer = useRenderer()
1158
+ const keymap = createDefaultOpenTuiKeymap(renderer)
1159
+ off = registerOctocodeKeymap(keymap, renderer, tuiConfig)
1160
+
1161
+ return (
1162
+ <OctocodeKeymapProvider keymap={keymap}>
1163
+ <RunQuestionBody
1164
+ request={question}
1165
+ theme={RUN_THEME_FALLBACK.footer}
1166
+ onReply={(input) => {
1167
+ questions.push(input)
1168
+ }}
1169
+ onReject={() => {}}
1170
+ />
1171
+ </OctocodeKeymapProvider>
1172
+ )
1173
+ }
1174
+
1175
+ const app = await testRender(
1176
+ () => (
1177
+ <box width={100} height={18}>
1178
+ <Harness />
1179
+ </box>
1180
+ ),
1181
+ { width: 100, height: 18, kittyKeyboard: true },
1182
+ )
1183
+
1184
+ try {
1185
+ await app.renderOnce()
1186
+ app.mockInput.pressKey("2")
1187
+ await app.renderOnce()
1188
+ "typed".split("").forEach((key) => app.mockInput.pressKey(key))
1189
+ await app.renderOnce()
1190
+ app.mockInput.pressEnter()
1191
+ await app.renderOnce()
1192
+ expect(questions).toEqual([{ requestID: "question-1", answers: [["typed"]] }])
1193
+ } finally {
1194
+ app.renderer.currentFocusedRenderable?.blur()
1195
+ app.renderer.currentFocusedEditor?.blur()
1196
+ off?.()
1197
+ app.renderer.destroy()
1198
+ }
1199
+ })
1200
+
1201
+ test("direct permission rejection submits through keymap return binding", async () => {
1202
+ let text = ""
1203
+ const submits: string[] = []
1204
+ let off: (() => void) | undefined
1205
+
1206
+ function Harness() {
1207
+ const renderer = useRenderer()
1208
+ const keymap = createDefaultOpenTuiKeymap(renderer)
1209
+ off = registerOctocodeKeymap(keymap, renderer, tuiConfig)
1210
+
1211
+ return (
1212
+ <OctocodeKeymapProvider keymap={keymap}>
1213
+ <RejectField
1214
+ theme={RUN_THEME_FALLBACK.footer}
1215
+ text=""
1216
+ disabled={false}
1217
+ onChange={(input) => {
1218
+ text = input
1219
+ }}
1220
+ onConfirm={() => {
1221
+ submits.push(text)
1222
+ }}
1223
+ onCancel={() => {}}
1224
+ />
1225
+ </OctocodeKeymapProvider>
1226
+ )
1227
+ }
1228
+
1229
+ const app = await testRender(
1230
+ () => (
1231
+ <box width={100} height={18}>
1232
+ <Harness />
1233
+ </box>
1234
+ ),
1235
+ { width: 100, height: 18, kittyKeyboard: true },
1236
+ )
1237
+
1238
+ try {
1239
+ await app.renderOnce()
1240
+ "retry".split("").forEach((key) => app.mockInput.pressKey(key))
1241
+ await app.renderOnce()
1242
+ expect(app.captureCharFrame()).toContain("retry")
1243
+ app.mockInput.pressEnter()
1244
+ await app.renderOnce()
1245
+ expect(submits).toEqual(["retry"])
1246
+ } finally {
1247
+ app.renderer.currentFocusedRenderable?.blur()
1248
+ app.renderer.currentFocusedEditor?.blur()
1249
+ off?.()
1250
+ app.renderer.destroy()
1251
+ }
1252
+ })
1253
+
1254
+ test("direct model panel renders current model selector", async () => {
1255
+ const [providers] = createSignal<RunProvider[] | undefined>([provider()])
1256
+ const [current] = createSignal<RunInput["model"]>({ providerID: "octo", modelID: "gpt-5" })
1257
+
1258
+ const app = await testRender(
1259
+ () => (
1260
+ <box width={100} height={RUN_COMMAND_PANEL_ROWS}>
1261
+ <RunModelSelectBody
1262
+ theme={() => RUN_THEME_FALLBACK.footer}
1263
+ providers={providers}
1264
+ current={current}
1265
+ onClose={() => {}}
1266
+ onSelect={() => {}}
1267
+ />
1268
+ </box>
1269
+ ),
1270
+ {
1271
+ width: 100,
1272
+ height: RUN_COMMAND_PANEL_ROWS,
1273
+ },
1274
+ )
1275
+
1276
+ try {
1277
+ await app.renderOnce()
1278
+ const frame = app.captureCharFrame()
1279
+ const list = app.renderer.root.findDescendantById("run-direct-footer-model-list") as BoxRenderable
1280
+
1281
+ expect(frame).toContain("Select model")
1282
+ expect(frame).toContain("Search")
1283
+ expect(frame).toContain("octo")
1284
+ expect(frame).toContain("GPT-5")
1285
+ expect(frame).toContain("current")
1286
+ expect(frame).toContain("GPT Free")
1287
+ expect(frame).toContain("Free")
1288
+ expect(frame).not.toContain("┌")
1289
+ expect(frame).not.toContain("┃")
1290
+ expect(frame).not.toContain("Old Model")
1291
+ expectPaletteList(list, 2)
1292
+ } finally {
1293
+ app.renderer.destroy()
1294
+ }
1295
+ })
1296
+
1297
+ test("direct variant panel renders current variant selector", async () => {
1298
+ const [variants] = createSignal(["high", "minimal"])
1299
+ const [current] = createSignal<string | undefined>("high")
1300
+
1301
+ const app = await testRender(
1302
+ () => (
1303
+ <box width={100} height={RUN_COMMAND_PANEL_ROWS}>
1304
+ <RunVariantSelectBody
1305
+ theme={() => RUN_THEME_FALLBACK.footer}
1306
+ variants={variants}
1307
+ current={current}
1308
+ onClose={() => {}}
1309
+ onSelect={() => {}}
1310
+ />
1311
+ </box>
1312
+ ),
1313
+ {
1314
+ width: 100,
1315
+ height: RUN_COMMAND_PANEL_ROWS,
1316
+ },
1317
+ )
1318
+
1319
+ try {
1320
+ await app.renderOnce()
1321
+ const frame = app.captureCharFrame()
1322
+ const list = app.renderer.root.findDescendantById("run-direct-footer-variant-list") as BoxRenderable
1323
+
1324
+ expect(frame).toContain("Select variant")
1325
+ expect(frame).toContain("Default")
1326
+ expect(frame).toContain("high")
1327
+ expect(frame).toContain("minimal")
1328
+ expect(frame).toContain("current")
1329
+ expect(frame).not.toContain("┌")
1330
+ expect(frame).not.toContain("┃")
1331
+ expectPaletteList(list, 1)
1332
+ } finally {
1333
+ app.renderer.destroy()
1334
+ }
1335
+ })
1336
+