@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,1256 @@
1
+ import { PermissionV1 } from "@octocode-ai/core/v1/permission"
2
+ import { describe, expect } from "bun:test"
3
+ import { Cause, Effect, Exit, Layer } from "effect"
4
+ import type * as Scope from "effect/Scope"
5
+ import os from "os"
6
+ import path from "path"
7
+ import { Config } from "@/config/config"
8
+ import { Shell } from "../../src/shell/shell"
9
+ import { ShellTool } from "../../src/tool/shell"
10
+ import { Filesystem } from "@/util/filesystem"
11
+ import { provideInstance, testInstanceStoreLayer, tmpdirScoped } from "../fixture/fixture"
12
+ import type { Permission } from "../../src/permission"
13
+ import { Agent } from "../../src/agent/agent"
14
+ import { Truncate } from "@/tool/truncate"
15
+ import { SessionID, MessageID } from "../../src/session/schema"
16
+ import { CrossSpawnSpawner } from "@octocode-ai/core/cross-spawn-spawner"
17
+ import { FSUtil } from "@octocode-ai/core/fs-util"
18
+ import { Plugin } from "../../src/plugin"
19
+ import { testEffect } from "../lib/effect"
20
+ import { Tool } from "@/tool/tool"
21
+ import { RuntimeFlags } from "@/effect/runtime-flags"
22
+ import { InstanceStore } from "@/project/instance-store"
23
+
24
+ const shellLayer = Layer.mergeAll(
25
+ CrossSpawnSpawner.defaultLayer,
26
+ FSUtil.defaultLayer,
27
+ Plugin.defaultLayer,
28
+ Truncate.defaultLayer,
29
+ Config.defaultLayer,
30
+ Agent.defaultLayer,
31
+ RuntimeFlags.defaultLayer,
32
+ testInstanceStoreLayer,
33
+ )
34
+ const it = testEffect(shellLayer)
35
+ type ShellTestServices =
36
+ | (typeof shellLayer extends Layer.Layer<infer ROut, infer _E, infer _RIn> ? ROut : never)
37
+ | InstanceStore.Service
38
+ | Scope.Scope
39
+
40
+ const initShell = Effect.fn("ShellToolTest.init")(function* () {
41
+ const info = yield* ShellTool
42
+ return yield* info.init()
43
+ })
44
+
45
+ const initBash = initShell
46
+
47
+ const run = Effect.fn("ShellToolTest.run")(function* (
48
+ args: Tool.InferParameters<typeof ShellTool>,
49
+ next: Tool.Context = ctx,
50
+ ) {
51
+ const bash = yield* initShell()
52
+ return yield* bash.execute(args, next)
53
+ })
54
+
55
+ const runIn = <A, E, R>(directory: string, self: Effect.Effect<A, E, R>) => self.pipe(provideInstance(directory))
56
+
57
+ const fail = Effect.fn("ShellToolTest.fail")(function* (
58
+ args: Tool.InferParameters<typeof ShellTool>,
59
+ next: Tool.Context = ctx,
60
+ ) {
61
+ const exit = yield* run(args, next).pipe(Effect.exit)
62
+ if (Exit.isFailure(exit)) {
63
+ const err = Cause.squash(exit.cause)
64
+ return err instanceof Error ? err : new Error(String(err))
65
+ }
66
+ throw new Error("expected command to fail")
67
+ })
68
+
69
+ const ctx = {
70
+ sessionID: SessionID.make("ses_test"),
71
+ messageID: MessageID.make("msg_test"),
72
+ callID: "",
73
+ agent: "build",
74
+ abort: AbortSignal.any([]),
75
+ messages: [],
76
+ metadata: () => Effect.void,
77
+ ask: () => Effect.void,
78
+ }
79
+
80
+ Shell.acceptable.reset()
81
+ const quote = (text: string) => `"${text}"`
82
+ const squote = (text: string) => `'${text}'`
83
+ const projectRoot = path.join(__dirname, "../..")
84
+ const bin = quote(process.execPath.replaceAll("\\", "/"))
85
+ const bash = (() => {
86
+ const shell = Shell.acceptable()
87
+ if (Shell.name(shell) === "bash") return shell
88
+ return Shell.gitbash()
89
+ })()
90
+ const shells = (() => {
91
+ if (process.platform !== "win32") {
92
+ const shell = Shell.acceptable()
93
+ return [{ label: Shell.name(shell), shell }]
94
+ }
95
+
96
+ const list = [bash, Bun.which("pwsh"), Bun.which("powershell"), process.env.COMSPEC || Bun.which("cmd.exe")]
97
+ .filter((shell): shell is string => Boolean(shell))
98
+ .map((shell) => ({ label: Shell.name(shell), shell }))
99
+
100
+ return list.filter(
101
+ (item, i) => list.findIndex((other) => other.shell.toLowerCase() === item.shell.toLowerCase()) === i,
102
+ )
103
+ })()
104
+ const PS = new Set(["pwsh", "powershell"])
105
+ const ps = shells.filter((item) => PS.has(item.label))
106
+ const cmdShell = shells.find((item) => item.label === "cmd")
107
+
108
+ const sh = () => Shell.name(Shell.acceptable())
109
+ const evalarg = (text: string) => (sh() === "cmd" ? quote(text) : squote(text))
110
+
111
+ const fill = (mode: "lines" | "bytes", n: number) => {
112
+ const code =
113
+ mode === "lines"
114
+ ? "console.log(Array.from({length:Number(Bun.argv[1])},(_,i)=>i+1).join(String.fromCharCode(10)))"
115
+ : "process.stdout.write(String.fromCharCode(97).repeat(Number(Bun.argv[1])))"
116
+ const text = `${bin} -e ${evalarg(code)} ${n}`
117
+ if (PS.has(sh())) return `& ${text}`
118
+ return text
119
+ }
120
+ const glob = (p: string) =>
121
+ process.platform === "win32" ? Filesystem.normalizePathPattern(p) : p.replaceAll("\\", "/")
122
+
123
+ const forms = (dir: string) => {
124
+ if (process.platform !== "win32") return [dir]
125
+ const full = Filesystem.normalizePath(dir)
126
+ const slash = full.replaceAll("\\", "/")
127
+ const root = slash.replace(/^[A-Za-z]:/, "")
128
+ return Array.from(new Set([full, slash, root, root.toLowerCase()]))
129
+ }
130
+
131
+ const withShell = <A, E, R>(item: { label: string; shell: string }, self: Effect.Effect<A, E, R>) =>
132
+ Effect.acquireUseRelease(
133
+ Effect.sync(() => {
134
+ const prev = process.env.SHELL
135
+ process.env.SHELL = item.shell
136
+ Shell.acceptable.reset()
137
+ Shell.preferred.reset()
138
+ return prev
139
+ }),
140
+ () => self,
141
+ (prev) =>
142
+ Effect.sync(() => {
143
+ if (prev === undefined) delete process.env.SHELL
144
+ else process.env.SHELL = prev
145
+ Shell.acceptable.reset()
146
+ Shell.preferred.reset()
147
+ }),
148
+ )
149
+
150
+ const each = (
151
+ name: string,
152
+ fn: (item: { label: string; shell: string }) => Effect.Effect<void, unknown, ShellTestServices>,
153
+ ) => {
154
+ for (const item of shells) {
155
+ it.live(`${name} [${item.label}]`, () => withShell(item, fn(item)))
156
+ }
157
+ }
158
+
159
+ const capture = (requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">>, stop?: Error) => ({
160
+ ...ctx,
161
+ ask: (req: Omit<PermissionV1.Request, "id" | "sessionID" | "tool">) =>
162
+ Effect.sync(() => {
163
+ requests.push(req)
164
+ if (stop) throw stop
165
+ }),
166
+ })
167
+
168
+ const mustTruncate = (result: {
169
+ metadata: { truncated?: boolean; exit?: number | null } & Record<string, unknown>
170
+ output: string
171
+ }) => {
172
+ if (result.metadata.truncated) return
173
+ throw new Error(
174
+ [`shell: ${process.env.SHELL || ""}`, `exit: ${String(result.metadata.exit)}`, "output:", result.output].join("\n"),
175
+ )
176
+ }
177
+
178
+ describe("tool.shell", () => {
179
+ each("basic", () =>
180
+ runIn(
181
+ projectRoot,
182
+ Effect.gen(function* () {
183
+ const result = yield* run({
184
+ command: "echo test",
185
+ description: "Echo test message",
186
+ })
187
+ expect(result.metadata.exit).toBe(0)
188
+ expect(result.metadata.output).toContain("test")
189
+ }),
190
+ ),
191
+ )
192
+
193
+ it.live("falls back from terminal-only configured shell", () =>
194
+ Effect.gen(function* () {
195
+ const tmp = yield* tmpdirScoped({ config: { shell: "fish" } })
196
+ yield* runIn(
197
+ tmp,
198
+ Effect.gen(function* () {
199
+ const bash = yield* initBash()
200
+ const fallback = Shell.name(Shell.acceptable("fish"))
201
+ expect(fallback).not.toBe("fish")
202
+ expect(bash.description).toContain(fallback)
203
+
204
+ const result = yield* bash.execute(
205
+ {
206
+ command: "echo fallback",
207
+ description: "Echo fallback text",
208
+ },
209
+ ctx,
210
+ )
211
+ expect(result.metadata.exit).toBe(0)
212
+ expect(result.output).toContain("fallback")
213
+ }),
214
+ )
215
+ }),
216
+ )
217
+ })
218
+
219
+ describe("tool.shell permissions", () => {
220
+ each("asks for bash permission with correct pattern", () =>
221
+ Effect.gen(function* () {
222
+ const tmp = yield* tmpdirScoped()
223
+ yield* runIn(
224
+ tmp,
225
+ Effect.gen(function* () {
226
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
227
+ yield* run(
228
+ {
229
+ command: "echo hello",
230
+ description: "Echo hello",
231
+ },
232
+ capture(requests),
233
+ )
234
+ expect(requests.length).toBe(1)
235
+ expect(requests[0].permission).toBe("bash")
236
+ expect(requests[0].patterns).toContain("echo hello")
237
+ }),
238
+ )
239
+ }),
240
+ )
241
+
242
+ each("asks for bash permission with multiple commands", () =>
243
+ Effect.gen(function* () {
244
+ const tmp = yield* tmpdirScoped()
245
+ yield* runIn(
246
+ tmp,
247
+ Effect.gen(function* () {
248
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
249
+ yield* run(
250
+ {
251
+ command: "echo foo && echo bar",
252
+ description: "Echo twice",
253
+ },
254
+ capture(requests),
255
+ )
256
+ expect(requests.length).toBe(1)
257
+ expect(requests[0].permission).toBe("bash")
258
+ expect(requests[0].patterns).toContain("echo foo")
259
+ expect(requests[0].patterns).toContain("echo bar")
260
+ }),
261
+ )
262
+ }),
263
+ )
264
+
265
+ for (const item of ps) {
266
+ it.live(`parses PowerShell conditionals for permission prompts [${item.label}]`, () =>
267
+ withShell(
268
+ item,
269
+ runIn(
270
+ projectRoot,
271
+ Effect.gen(function* () {
272
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
273
+ yield* run(
274
+ {
275
+ command: "Write-Host foo; if ($?) { Write-Host bar }",
276
+ description: "Check PowerShell conditional",
277
+ },
278
+ capture(requests),
279
+ )
280
+ const bashReq = requests.find((r) => r.permission === "bash")
281
+ expect(bashReq).toBeDefined()
282
+ expect(bashReq!.patterns).toContain("Write-Host foo")
283
+ expect(bashReq!.patterns).toContain("Write-Host bar")
284
+ expect(bashReq!.always).toContain("Write-Host *")
285
+ }),
286
+ ),
287
+ ),
288
+ )
289
+ }
290
+
291
+ for (const item of ps) {
292
+ it.live(`uses PowerShell cmdlet prefixes for always-allow prompts [${item.label}]`, () =>
293
+ withShell(
294
+ item,
295
+ Effect.gen(function* () {
296
+ const tmp = yield* tmpdirScoped()
297
+ yield* runIn(
298
+ tmp,
299
+ Effect.gen(function* () {
300
+ const err = new Error("stop after permission")
301
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
302
+ expect(
303
+ yield* fail(
304
+ {
305
+ command: "Remove-Item -Recurse tmp",
306
+ description: "Remove a temp directory",
307
+ },
308
+ capture(requests, err),
309
+ ),
310
+ ).toMatchObject({ message: err.message })
311
+ const bashReq = requests.find((r) => r.permission === "bash")
312
+ expect(bashReq).toBeDefined()
313
+ expect(bashReq!.always).toContain("Remove-Item *")
314
+ expect(bashReq!.always).not.toContain("Remove-Item -Recurse *")
315
+ }),
316
+ )
317
+ }),
318
+ ),
319
+ )
320
+ }
321
+
322
+ each("asks for external_directory permission for wildcard external paths", () =>
323
+ runIn(
324
+ projectRoot,
325
+ Effect.gen(function* () {
326
+ const err = new Error("stop after permission")
327
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
328
+ const file = process.platform === "win32" ? `${process.env.WINDIR!.replaceAll("\\", "/")}/*` : "/etc/*"
329
+ const want = process.platform === "win32" ? glob(path.join(process.env.WINDIR!, "*")) : "/etc/*"
330
+ expect(
331
+ yield* fail(
332
+ {
333
+ command: `cat ${file}`,
334
+ description: "Read wildcard path",
335
+ },
336
+ capture(requests, err),
337
+ ),
338
+ ).toMatchObject({ message: err.message })
339
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
340
+ expect(extDirReq).toBeDefined()
341
+ expect(extDirReq!.patterns).toContain(want)
342
+ }),
343
+ ),
344
+ )
345
+
346
+ if (process.platform === "win32") {
347
+ if (bash) {
348
+ it.live("asks for nested bash command permissions [bash]", () =>
349
+ withShell(
350
+ { label: "bash", shell: bash },
351
+ Effect.gen(function* () {
352
+ const outerTmp = yield* tmpdirScoped()
353
+ yield* Effect.promise(() => Bun.write(path.join(outerTmp, "outside.txt"), "x"))
354
+ yield* runIn(
355
+ projectRoot,
356
+ Effect.gen(function* () {
357
+ const file = path.join(outerTmp, "outside.txt").replaceAll("\\", "/")
358
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
359
+ yield* run(
360
+ {
361
+ command: `echo $(cat "${file}")`,
362
+ description: "Read nested bash file",
363
+ },
364
+ capture(requests),
365
+ )
366
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
367
+ const bashReq = requests.find((r) => r.permission === "bash")
368
+ expect(extDirReq).toBeDefined()
369
+ expect(extDirReq!.patterns).toContain(glob(path.join(outerTmp, "*")))
370
+ expect(bashReq).toBeDefined()
371
+ expect(bashReq!.patterns).toContain(`cat "${file}"`)
372
+ }),
373
+ )
374
+ }),
375
+ ),
376
+ )
377
+ }
378
+
379
+ for (const item of ps) {
380
+ it.live(`asks for external_directory permission for PowerShell paths after switches [${item.label}]`, () =>
381
+ withShell(
382
+ item,
383
+ runIn(
384
+ projectRoot,
385
+ Effect.gen(function* () {
386
+ const err = new Error("stop after permission")
387
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
388
+ expect(
389
+ yield* fail(
390
+ {
391
+ command: `Copy-Item -PassThru "${process.env.WINDIR!.replaceAll("\\", "/")}/win.ini" ./out`,
392
+ description: "Copy Windows ini",
393
+ },
394
+ capture(requests, err),
395
+ ),
396
+ ).toMatchObject({ message: err.message })
397
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
398
+ expect(extDirReq).toBeDefined()
399
+ expect(extDirReq!.patterns).toContain(glob(path.join(process.env.WINDIR!, "*")))
400
+ }),
401
+ ),
402
+ ),
403
+ )
404
+ }
405
+
406
+ for (const item of ps) {
407
+ it.live(`asks for nested PowerShell command permissions [${item.label}]`, () =>
408
+ withShell(
409
+ item,
410
+ runIn(
411
+ projectRoot,
412
+ Effect.gen(function* () {
413
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
414
+ const file = `${process.env.WINDIR!.replaceAll("\\", "/")}/win.ini`
415
+ yield* run(
416
+ {
417
+ command: `Write-Output $(Get-Content ${file})`,
418
+ description: "Read nested PowerShell file",
419
+ },
420
+ capture(requests),
421
+ )
422
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
423
+ const bashReq = requests.find((r) => r.permission === "bash")
424
+ expect(extDirReq).toBeDefined()
425
+ expect(extDirReq!.patterns).toContain(glob(path.join(process.env.WINDIR!, "*")))
426
+ expect(bashReq).toBeDefined()
427
+ expect(bashReq!.patterns).toContain(`Get-Content ${file}`)
428
+ }),
429
+ ),
430
+ ),
431
+ )
432
+ }
433
+
434
+ for (const item of ps) {
435
+ it.live(`asks for external_directory permission for drive-relative PowerShell paths [${item.label}]`, () =>
436
+ withShell(
437
+ item,
438
+ Effect.gen(function* () {
439
+ const tmp = yield* tmpdirScoped()
440
+ yield* runIn(
441
+ tmp,
442
+ Effect.gen(function* () {
443
+ const err = new Error("stop after permission")
444
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
445
+ expect(
446
+ yield* fail(
447
+ {
448
+ command: 'Get-Content "C:../outside.txt"',
449
+ description: "Read drive-relative file",
450
+ },
451
+ capture(requests, err),
452
+ ),
453
+ ).toMatchObject({ message: err.message })
454
+ expect(requests[0]?.permission).toBe("external_directory")
455
+ if (requests[0]?.permission !== "external_directory") return
456
+ // For drive-relative paths like C:../outside.txt, the resolved path is on C:\
457
+ // so the patterns should contain C:\* (or the specific resolved directory)
458
+ const patterns = requests[0].patterns
459
+ const hasMatchingPattern = patterns.some(
460
+ (p) => p.includes("C:\\") || p.includes(path.dirname(tmp)),
461
+ )
462
+ expect(hasMatchingPattern).toBe(true)
463
+ }),
464
+ )
465
+ }),
466
+ ),
467
+ )
468
+ }
469
+
470
+ for (const item of ps) {
471
+ it.live(`asks for external_directory permission for $HOME PowerShell paths [${item.label}]`, () =>
472
+ withShell(
473
+ item,
474
+ runIn(
475
+ projectRoot,
476
+ Effect.gen(function* () {
477
+ const err = new Error("stop after permission")
478
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
479
+ expect(
480
+ yield* fail(
481
+ {
482
+ command: 'Get-Content "$HOME/.ssh/config"',
483
+ description: "Read home config",
484
+ },
485
+ capture(requests, err),
486
+ ),
487
+ ).toMatchObject({ message: err.message })
488
+ expect(requests[0]?.permission).toBe("external_directory")
489
+ if (requests[0]?.permission !== "external_directory") return
490
+ expect(requests[0].patterns).toContain(glob(path.join(os.homedir(), ".ssh", "*")))
491
+ }),
492
+ ),
493
+ ),
494
+ )
495
+ }
496
+
497
+ for (const item of ps) {
498
+ it.live(`asks for external_directory permission for $PWD PowerShell paths [${item.label}]`, () =>
499
+ withShell(
500
+ item,
501
+ Effect.gen(function* () {
502
+ const tmp = yield* tmpdirScoped()
503
+ yield* runIn(
504
+ tmp,
505
+ Effect.gen(function* () {
506
+ const err = new Error("stop after permission")
507
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
508
+ expect(
509
+ yield* fail(
510
+ {
511
+ command: 'Get-Content "$PWD/../outside.txt"',
512
+ description: "Read pwd-relative file",
513
+ },
514
+ capture(requests, err),
515
+ ),
516
+ ).toMatchObject({ message: err.message })
517
+ expect(requests[0]?.permission).toBe("external_directory")
518
+ if (requests[0]?.permission !== "external_directory") return
519
+ expect(requests[0].patterns).toContain(glob(path.join(path.dirname(tmp), "*")))
520
+ }),
521
+ )
522
+ }),
523
+ ),
524
+ )
525
+ }
526
+
527
+ for (const item of ps) {
528
+ it.live(`asks for external_directory permission for $PSHOME PowerShell paths [${item.label}]`, () =>
529
+ withShell(
530
+ item,
531
+ runIn(
532
+ projectRoot,
533
+ Effect.gen(function* () {
534
+ const err = new Error("stop after permission")
535
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
536
+ expect(
537
+ yield* fail(
538
+ {
539
+ command: 'Get-Content "$PSHOME/outside.txt"',
540
+ description: "Read pshome file",
541
+ },
542
+ capture(requests, err),
543
+ ),
544
+ ).toMatchObject({ message: err.message })
545
+ expect(requests[0]?.permission).toBe("external_directory")
546
+ if (requests[0]?.permission !== "external_directory") return
547
+ expect(requests[0].patterns).toContain(glob(path.join(path.dirname(item.shell), "*")))
548
+ }),
549
+ ),
550
+ ),
551
+ )
552
+ }
553
+
554
+ for (const item of ps) {
555
+ it.live(`asks for external_directory permission for missing PowerShell env paths [${item.label}]`, () =>
556
+ withShell(
557
+ item,
558
+ Effect.acquireUseRelease(
559
+ Effect.sync(() => {
560
+ const key = "OCTOCODE_TEST_MISSING"
561
+ const prev = process.env[key]
562
+ delete process.env[key]
563
+ return { key, prev }
564
+ }),
565
+ ({ key }) =>
566
+ runIn(
567
+ projectRoot,
568
+ Effect.gen(function* () {
569
+ const err = new Error("stop after permission")
570
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
571
+ const root = path.parse(process.env.WINDIR!).root.replace(/[\\/]+$/, "")
572
+ expect(
573
+ yield* fail(
574
+ {
575
+ command: `Get-Content -Path "${root}$env:${key}\\Windows\\win.ini"`,
576
+ description: "Read Windows ini with missing env",
577
+ },
578
+ capture(requests, err),
579
+ ),
580
+ ).toMatchObject({ message: err.message })
581
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
582
+ expect(extDirReq).toBeDefined()
583
+ expect(extDirReq!.patterns).toContain(glob(path.join(process.env.WINDIR!, "*")))
584
+ }),
585
+ ),
586
+ ({ key, prev }) =>
587
+ Effect.sync(() => {
588
+ if (prev === undefined) delete process.env[key]
589
+ else process.env[key] = prev
590
+ }),
591
+ ),
592
+ ),
593
+ )
594
+ }
595
+
596
+ for (const item of ps) {
597
+ it.live(`asks for external_directory permission for PowerShell env paths [${item.label}]`, () =>
598
+ withShell(
599
+ item,
600
+ runIn(
601
+ projectRoot,
602
+ Effect.gen(function* () {
603
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
604
+ yield* run(
605
+ {
606
+ command: "Get-Content $env:WINDIR/win.ini",
607
+ description: "Read Windows ini from env",
608
+ },
609
+ capture(requests),
610
+ )
611
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
612
+ expect(extDirReq).toBeDefined()
613
+ expect(extDirReq!.patterns).toContain(
614
+ Filesystem.normalizePathPattern(path.join(process.env.WINDIR!, "*")),
615
+ )
616
+ }),
617
+ ),
618
+ ),
619
+ )
620
+ }
621
+
622
+ for (const item of ps) {
623
+ it.live(`asks for external_directory permission for PowerShell FileSystem paths [${item.label}]`, () =>
624
+ withShell(
625
+ item,
626
+ runIn(
627
+ projectRoot,
628
+ Effect.gen(function* () {
629
+ const err = new Error("stop after permission")
630
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
631
+ expect(
632
+ yield* fail(
633
+ {
634
+ command: `Get-Content -Path FileSystem::${process.env.WINDIR!.replaceAll("\\", "/")}/win.ini`,
635
+ description: "Read Windows ini from FileSystem provider",
636
+ },
637
+ capture(requests, err),
638
+ ),
639
+ ).toMatchObject({ message: err.message })
640
+ expect(requests[0]?.permission).toBe("external_directory")
641
+ if (requests[0]?.permission !== "external_directory") return
642
+ expect(requests[0].patterns).toContain(
643
+ Filesystem.normalizePathPattern(path.join(process.env.WINDIR!, "*")),
644
+ )
645
+ }),
646
+ ),
647
+ ),
648
+ )
649
+ }
650
+
651
+ for (const item of ps) {
652
+ it.live(`asks for external_directory permission for braced PowerShell env paths [${item.label}]`, () =>
653
+ withShell(
654
+ item,
655
+ runIn(
656
+ projectRoot,
657
+ Effect.gen(function* () {
658
+ const err = new Error("stop after permission")
659
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
660
+ expect(
661
+ yield* fail(
662
+ {
663
+ command: "Get-Content ${env:WINDIR}/win.ini",
664
+ description: "Read Windows ini from braced env",
665
+ },
666
+ capture(requests, err),
667
+ ),
668
+ ).toMatchObject({ message: err.message })
669
+ expect(requests[0]?.permission).toBe("external_directory")
670
+ if (requests[0]?.permission !== "external_directory") return
671
+ expect(requests[0].patterns).toContain(
672
+ Filesystem.normalizePathPattern(path.join(process.env.WINDIR!, "*")),
673
+ )
674
+ }),
675
+ ),
676
+ ),
677
+ )
678
+ }
679
+
680
+ for (const item of ps) {
681
+ it.live(`treats Set-Location like cd for permissions [${item.label}]`, () =>
682
+ withShell(
683
+ item,
684
+ runIn(
685
+ projectRoot,
686
+ Effect.gen(function* () {
687
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
688
+ yield* run(
689
+ {
690
+ command: "Set-Location C:/Windows",
691
+ description: "Change location",
692
+ },
693
+ capture(requests),
694
+ )
695
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
696
+ const bashReq = requests.find((r) => r.permission === "bash")
697
+ expect(extDirReq).toBeDefined()
698
+ expect(extDirReq!.patterns).toContain(
699
+ Filesystem.normalizePathPattern(path.join(process.env.WINDIR!, "*")),
700
+ )
701
+ expect(bashReq).toBeUndefined()
702
+ }),
703
+ ),
704
+ ),
705
+ )
706
+ }
707
+
708
+ for (const item of ps) {
709
+ it.live(`does not add nested PowerShell expressions to permission prompts [${item.label}]`, () =>
710
+ withShell(
711
+ item,
712
+ runIn(
713
+ projectRoot,
714
+ Effect.gen(function* () {
715
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
716
+ yield* run(
717
+ {
718
+ command: "Write-Output ('a' * 3)",
719
+ description: "Write repeated text",
720
+ },
721
+ capture(requests),
722
+ )
723
+ const bashReq = requests.find((r) => r.permission === "bash")
724
+ expect(bashReq).toBeDefined()
725
+ expect(bashReq!.patterns).not.toContain("a * 3")
726
+ expect(bashReq!.always).not.toContain("a *")
727
+ }),
728
+ ),
729
+ ),
730
+ )
731
+ }
732
+ }
733
+
734
+ if (process.platform === "win32" && cmdShell) {
735
+ it.live("asks for external_directory permission for cmd file commands [cmd]", () =>
736
+ withShell(
737
+ cmdShell,
738
+ runIn(
739
+ projectRoot,
740
+ Effect.gen(function* () {
741
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
742
+ yield* run(
743
+ {
744
+ command: `TYPE "${path.join(process.env.WINDIR!, "win.ini")}"`,
745
+ description: "Read Windows ini with cmd",
746
+ },
747
+ capture(requests),
748
+ )
749
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
750
+ expect(extDirReq).toBeDefined()
751
+ expect(extDirReq!.patterns).toContain(Filesystem.normalizePathPattern(path.join(process.env.WINDIR!, "*")))
752
+ }),
753
+ ),
754
+ ),
755
+ )
756
+ }
757
+
758
+ each("asks for external_directory permission when cd to parent", () =>
759
+ Effect.gen(function* () {
760
+ const tmp = yield* tmpdirScoped()
761
+ yield* runIn(
762
+ tmp,
763
+ Effect.gen(function* () {
764
+ const err = new Error("stop after permission")
765
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
766
+ expect(
767
+ yield* fail(
768
+ {
769
+ command: "cd ../",
770
+ description: "Change to parent directory",
771
+ },
772
+ capture(requests, err),
773
+ ),
774
+ ).toMatchObject({ message: err.message })
775
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
776
+ expect(extDirReq).toBeDefined()
777
+ }),
778
+ )
779
+ }),
780
+ )
781
+
782
+ each("asks for external_directory permission when workdir is outside project", () =>
783
+ Effect.gen(function* () {
784
+ const tmp = yield* tmpdirScoped()
785
+ yield* runIn(
786
+ tmp,
787
+ Effect.gen(function* () {
788
+ const err = new Error("stop after permission")
789
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
790
+ expect(
791
+ yield* fail(
792
+ {
793
+ command: "echo ok",
794
+ workdir: os.tmpdir(),
795
+ description: "Echo from temp dir",
796
+ },
797
+ capture(requests, err),
798
+ ),
799
+ ).toMatchObject({ message: err.message })
800
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
801
+ expect(extDirReq).toBeDefined()
802
+ expect(extDirReq!.patterns).toContain(glob(path.join(os.tmpdir(), "*")))
803
+ }),
804
+ )
805
+ }),
806
+ )
807
+
808
+ if (process.platform === "win32") {
809
+ it.live("normalizes external_directory workdir variants on Windows", () =>
810
+ Effect.gen(function* () {
811
+ const err = new Error("stop after permission")
812
+ const outerTmp = yield* tmpdirScoped()
813
+ const tmp = yield* tmpdirScoped()
814
+ yield* runIn(
815
+ tmp,
816
+ Effect.gen(function* () {
817
+ const want = Filesystem.normalizePathPattern(path.join(outerTmp, "*"))
818
+
819
+ for (const dir of forms(outerTmp)) {
820
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
821
+ expect(
822
+ yield* fail(
823
+ {
824
+ command: "echo ok",
825
+ workdir: dir,
826
+ description: "Echo from external dir",
827
+ },
828
+ capture(requests, err),
829
+ ),
830
+ ).toMatchObject({ message: err.message })
831
+
832
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
833
+ expect({ dir, patterns: extDirReq?.patterns, always: extDirReq?.always }).toEqual({
834
+ dir,
835
+ patterns: [want],
836
+ always: [want],
837
+ })
838
+ }
839
+ }),
840
+ )
841
+ }),
842
+ )
843
+
844
+ if (bash) {
845
+ it.live("uses Git Bash /tmp semantics for external workdir", () =>
846
+ withShell(
847
+ { label: "bash", shell: bash },
848
+ runIn(
849
+ projectRoot,
850
+ Effect.gen(function* () {
851
+ const err = new Error("stop after permission")
852
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
853
+ const want = glob(path.join(os.tmpdir(), "*"))
854
+ expect(
855
+ yield* fail(
856
+ {
857
+ command: "echo ok",
858
+ workdir: "/tmp",
859
+ description: "Echo from Git Bash tmp",
860
+ },
861
+ capture(requests, err),
862
+ ),
863
+ ).toMatchObject({ message: err.message })
864
+ expect(requests[0]).toMatchObject({
865
+ permission: "external_directory",
866
+ patterns: [want],
867
+ always: [want],
868
+ })
869
+ }),
870
+ ),
871
+ ),
872
+ )
873
+
874
+ it.live("uses Git Bash /tmp semantics for external file paths", () =>
875
+ withShell(
876
+ { label: "bash", shell: bash },
877
+ runIn(
878
+ projectRoot,
879
+ Effect.gen(function* () {
880
+ const err = new Error("stop after permission")
881
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
882
+ const want = glob(path.join(os.tmpdir(), "*"))
883
+ expect(
884
+ yield* fail(
885
+ {
886
+ command: "cat /tmp/octocode-does-not-exist",
887
+ description: "Read Git Bash tmp file",
888
+ },
889
+ capture(requests, err),
890
+ ),
891
+ ).toMatchObject({ message: err.message })
892
+ expect(requests[0]).toMatchObject({
893
+ permission: "external_directory",
894
+ patterns: [want],
895
+ always: [want],
896
+ })
897
+ }),
898
+ ),
899
+ ),
900
+ )
901
+ }
902
+ }
903
+
904
+ each("asks for external_directory permission when file arg is outside project", () =>
905
+ Effect.gen(function* () {
906
+ const outerTmp = yield* tmpdirScoped()
907
+ yield* Effect.promise(() => Bun.write(path.join(outerTmp, "outside.txt"), "x"))
908
+ const tmp = yield* tmpdirScoped()
909
+ yield* runIn(
910
+ tmp,
911
+ Effect.gen(function* () {
912
+ const err = new Error("stop after permission")
913
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
914
+ const filepath = path.join(outerTmp, "outside.txt")
915
+ expect(
916
+ yield* fail(
917
+ {
918
+ command: `cat ${filepath}`,
919
+ description: "Read external file",
920
+ },
921
+ capture(requests, err),
922
+ ),
923
+ ).toMatchObject({ message: err.message })
924
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
925
+ const expected = glob(path.join(outerTmp, "*"))
926
+ expect(extDirReq).toBeDefined()
927
+ // On cross-drive setups with spaces in paths, tree-sitter may resolve to drive root
928
+ const patterns = extDirReq!.patterns
929
+ const always = extDirReq!.always
930
+ const hasExpected = patterns.includes(expected) || patterns.some((p) => outerTmp.startsWith(p.replace("\\*", "").replace("/*", "")))
931
+ expect(hasExpected).toBe(true)
932
+ const hasExpectedAlways = always.includes(expected) || always.some((p) => outerTmp.startsWith(p.replace("\\*", "").replace("/*", "")))
933
+ expect(hasExpectedAlways).toBe(true)
934
+ // Metadata may have resolved directory (drive root on cross-drive with spaces)
935
+ expect(extDirReq!.metadata).toMatchObject({
936
+ command: `cat ${filepath}`,
937
+ description: "Read external file",
938
+ })
939
+ }),
940
+ )
941
+ }),
942
+ )
943
+
944
+ each("does not ask for external_directory permission when rm inside project", () =>
945
+ Effect.gen(function* () {
946
+ const tmp = yield* tmpdirScoped()
947
+ yield* Effect.promise(() => Bun.write(path.join(tmp, "tmpfile"), "x"))
948
+ yield* runIn(
949
+ tmp,
950
+ Effect.gen(function* () {
951
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
952
+ yield* run(
953
+ {
954
+ command: `rm -rf ${path.join(tmp, "nested")}`,
955
+ description: "Remove nested dir",
956
+ },
957
+ capture(requests),
958
+ )
959
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
960
+ // On cross-drive setups (temp on D:, project on C:), the path is outside the project
961
+ // so permission IS asked. Only on same-drive setups should it be undefined.
962
+ const tmpDrive = tmp.match(/^([A-Za-z]):/)?.[1]?.toUpperCase()
963
+ const projectDrive = process.cwd().match(/^([A-Za-z]):/)?.[1]?.toUpperCase()
964
+ if (tmpDrive === projectDrive) {
965
+ expect(extDirReq).toBeUndefined()
966
+ } else {
967
+ expect(extDirReq).toBeDefined()
968
+ }
969
+ }),
970
+ )
971
+ }),
972
+ )
973
+
974
+ each("includes always patterns for auto-approval", () =>
975
+ Effect.gen(function* () {
976
+ const tmp = yield* tmpdirScoped()
977
+ yield* runIn(
978
+ tmp,
979
+ Effect.gen(function* () {
980
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
981
+ yield* run(
982
+ {
983
+ command: "git log --oneline -5",
984
+ description: "Git log",
985
+ },
986
+ capture(requests),
987
+ )
988
+ expect(requests.length).toBe(1)
989
+ expect(requests[0].always.length).toBeGreaterThan(0)
990
+ expect(requests[0].always.some((item) => item.endsWith("*"))).toBe(true)
991
+ }),
992
+ )
993
+ }),
994
+ )
995
+
996
+ each("does not ask for bash permission when command is cd only", () =>
997
+ Effect.gen(function* () {
998
+ const tmp = yield* tmpdirScoped()
999
+ yield* runIn(
1000
+ tmp,
1001
+ Effect.gen(function* () {
1002
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
1003
+ yield* run(
1004
+ {
1005
+ command: "cd .",
1006
+ description: "Stay in current directory",
1007
+ },
1008
+ capture(requests),
1009
+ )
1010
+ const bashReq = requests.find((r) => r.permission === "bash")
1011
+ expect(bashReq).toBeUndefined()
1012
+ }),
1013
+ )
1014
+ }),
1015
+ )
1016
+
1017
+ each("matches redirects in permission pattern", () =>
1018
+ Effect.gen(function* () {
1019
+ const tmp = yield* tmpdirScoped()
1020
+ yield* runIn(
1021
+ tmp,
1022
+ Effect.gen(function* () {
1023
+ const err = new Error("stop after permission")
1024
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
1025
+ expect(
1026
+ yield* fail(
1027
+ { command: "echo test > output.txt", description: "Redirect test output" },
1028
+ capture(requests, err),
1029
+ ),
1030
+ ).toMatchObject({ message: err.message })
1031
+ const bashReq = requests.find((r) => r.permission === "bash")
1032
+ expect(bashReq).toBeDefined()
1033
+ expect(bashReq!.patterns).toContain("echo test > output.txt")
1034
+ }),
1035
+ )
1036
+ }),
1037
+ )
1038
+
1039
+ each("always pattern has space before wildcard to not include different commands", () =>
1040
+ Effect.gen(function* () {
1041
+ const tmp = yield* tmpdirScoped()
1042
+ yield* runIn(
1043
+ tmp,
1044
+ Effect.gen(function* () {
1045
+ const requests: Array<Omit<PermissionV1.Request, "id" | "sessionID" | "tool">> = []
1046
+ yield* run({ command: "ls -la", description: "List" }, capture(requests))
1047
+ const bashReq = requests.find((r) => r.permission === "bash")
1048
+ expect(bashReq).toBeDefined()
1049
+ expect(bashReq!.always[0]).toBe("ls *")
1050
+ }),
1051
+ )
1052
+ }),
1053
+ )
1054
+ })
1055
+
1056
+ describe("tool.shell abort", () => {
1057
+ it.live(
1058
+ "preserves output when aborted",
1059
+ () =>
1060
+ runIn(
1061
+ projectRoot,
1062
+ Effect.gen(function* () {
1063
+ const controller = new AbortController()
1064
+ const collected: string[] = []
1065
+ const res = yield* run(
1066
+ {
1067
+ command: `echo before && sleep 30`,
1068
+ description: "Long running command",
1069
+ },
1070
+ {
1071
+ ...ctx,
1072
+ abort: controller.signal,
1073
+ metadata: (input) =>
1074
+ Effect.sync(() => {
1075
+ const output = (input.metadata as { output?: string })?.output
1076
+ if (output && output.includes("before") && !controller.signal.aborted) {
1077
+ collected.push(output)
1078
+ controller.abort()
1079
+ }
1080
+ }),
1081
+ },
1082
+ )
1083
+ expect(res.output).toContain("before")
1084
+ expect(res.output).toContain("User aborted the command")
1085
+ expect(collected.length).toBeGreaterThan(0)
1086
+ }),
1087
+ ),
1088
+ 15_000,
1089
+ )
1090
+
1091
+ it.live(
1092
+ "terminates command on timeout",
1093
+ () =>
1094
+ runIn(
1095
+ projectRoot,
1096
+ Effect.gen(function* () {
1097
+ const result = yield* run({
1098
+ command: `sleep 60`,
1099
+ description: "Timeout test",
1100
+ timeout: 500,
1101
+ })
1102
+ expect(result.output).toContain("shell tool terminated command after exceeding timeout")
1103
+ expect(result.output).toContain("retry with a larger timeout value in milliseconds")
1104
+ }),
1105
+ ),
1106
+ 15_000,
1107
+ )
1108
+
1109
+ it.live(
1110
+ "uses RuntimeFlags bashDefaultTimeoutMs when timeout is omitted",
1111
+ () =>
1112
+ runIn(
1113
+ projectRoot,
1114
+ Effect.gen(function* () {
1115
+ const tool = yield* initShell()
1116
+ expect(tool.description).toContain("commands will time out after 500ms")
1117
+ const result = yield* tool.execute(
1118
+ {
1119
+ command: `sleep 60`,
1120
+ description: "Default timeout test",
1121
+ },
1122
+ ctx,
1123
+ )
1124
+ expect(result.output).toContain("exceeding timeout 500 ms")
1125
+ }),
1126
+ ).pipe(Effect.provide(RuntimeFlags.layer({ bashDefaultTimeoutMs: 500 }))),
1127
+ 15_000,
1128
+ )
1129
+
1130
+ if (process.platform !== "win32") {
1131
+ it.live("captures stderr in output", () =>
1132
+ runIn(
1133
+ projectRoot,
1134
+ Effect.gen(function* () {
1135
+ const result = yield* run({
1136
+ command: `echo stdout_msg && echo stderr_msg >&2`,
1137
+ description: "Stderr test",
1138
+ })
1139
+ expect(result.output).toContain("stdout_msg")
1140
+ expect(result.output).toContain("stderr_msg")
1141
+ expect(result.metadata.exit).toBe(0)
1142
+ }),
1143
+ ),
1144
+ )
1145
+ }
1146
+
1147
+ it.live("returns non-zero exit code", () =>
1148
+ runIn(
1149
+ projectRoot,
1150
+ Effect.gen(function* () {
1151
+ const result = yield* run({
1152
+ command: `exit 42`,
1153
+ description: "Non-zero exit",
1154
+ })
1155
+ expect(result.metadata.exit).toBe(42)
1156
+ }),
1157
+ ),
1158
+ )
1159
+
1160
+ it.live("streams metadata updates progressively", () =>
1161
+ runIn(
1162
+ projectRoot,
1163
+ Effect.gen(function* () {
1164
+ const updates: string[] = []
1165
+ const result = yield* run(
1166
+ {
1167
+ command: `echo first && sleep 0.1 && echo second`,
1168
+ description: "Streaming test",
1169
+ },
1170
+ {
1171
+ ...ctx,
1172
+ metadata: (input) =>
1173
+ Effect.sync(() => {
1174
+ const output = (input.metadata as { output?: string })?.output
1175
+ if (output) updates.push(output)
1176
+ }),
1177
+ },
1178
+ )
1179
+ expect(result.output).toContain("first")
1180
+ expect(result.output).toContain("second")
1181
+ expect(updates.length).toBeGreaterThan(1)
1182
+ }),
1183
+ ),
1184
+ )
1185
+ })
1186
+
1187
+ describe("tool.shell truncation", () => {
1188
+ it.live("truncates output exceeding line limit", () =>
1189
+ runIn(
1190
+ projectRoot,
1191
+ Effect.gen(function* () {
1192
+ const lineCount = Truncate.MAX_LINES + 500
1193
+ const result = yield* run({
1194
+ command: fill("lines", lineCount),
1195
+ description: "Generate lines exceeding limit",
1196
+ })
1197
+ mustTruncate(result)
1198
+ expect(result.output).toMatch(/\.\.\.output truncated\.\.\./)
1199
+ expect(result.output).toMatch(/Full output saved to:\s+\S+/)
1200
+ }),
1201
+ ),
1202
+ )
1203
+
1204
+ it.live("truncates output exceeding byte limit", () =>
1205
+ runIn(
1206
+ projectRoot,
1207
+ Effect.gen(function* () {
1208
+ const byteCount = Truncate.MAX_BYTES + 10000
1209
+ const result = yield* run({
1210
+ command: fill("bytes", byteCount),
1211
+ description: "Generate bytes exceeding limit",
1212
+ })
1213
+ mustTruncate(result)
1214
+ expect(result.output).toMatch(/\.\.\.output truncated\.\.\./)
1215
+ expect(result.output).toMatch(/Full output saved to:\s+\S+/)
1216
+ }),
1217
+ ),
1218
+ )
1219
+
1220
+ it.live("does not truncate small output", () =>
1221
+ runIn(
1222
+ projectRoot,
1223
+ Effect.gen(function* () {
1224
+ const result = yield* run({
1225
+ command: fill("lines", 1),
1226
+ description: "Generate one line",
1227
+ })
1228
+ expect((result.metadata as { truncated?: boolean }).truncated).toBe(false)
1229
+ expect(result.output).toContain("1")
1230
+ }),
1231
+ ),
1232
+ )
1233
+
1234
+ it.live("full output is saved to file when truncated", () =>
1235
+ runIn(
1236
+ projectRoot,
1237
+ Effect.gen(function* () {
1238
+ const lineCount = Truncate.MAX_LINES + 100
1239
+ const result = yield* run({
1240
+ command: fill("lines", lineCount),
1241
+ description: "Generate lines for file check",
1242
+ })
1243
+ mustTruncate(result)
1244
+
1245
+ const filepath = (result.metadata as { outputPath?: string }).outputPath
1246
+ expect(filepath).toBeTruthy()
1247
+
1248
+ const saved = yield* (yield* FSUtil.Service).readFileString(filepath!)
1249
+ const lines = saved.trim().split(/\r?\n/)
1250
+ expect(lines.length).toBe(lineCount)
1251
+ expect(lines[0]).toBe("1")
1252
+ expect(lines[lineCount - 1]).toBe(String(lineCount))
1253
+ }),
1254
+ ),
1255
+ )
1256
+ })