@stonerzju/opencode 1.2.16-offline.1

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 (496) hide show
  1. package/AGENTS.md +10 -0
  2. package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
  3. package/Dockerfile +18 -0
  4. package/README.md +15 -0
  5. package/bin/opencode +179 -0
  6. package/bunfig.toml +7 -0
  7. package/drizzle.config.ts +10 -0
  8. package/migration/20260127222353_familiar_lady_ursula/migration.sql +90 -0
  9. package/migration/20260127222353_familiar_lady_ursula/snapshot.json +796 -0
  10. package/migration/20260211171708_add_project_commands/migration.sql +1 -0
  11. package/migration/20260211171708_add_project_commands/snapshot.json +806 -0
  12. package/migration/20260213144116_wakeful_the_professor/migration.sql +11 -0
  13. package/migration/20260213144116_wakeful_the_professor/snapshot.json +897 -0
  14. package/migration/20260225215848_workspace/migration.sql +7 -0
  15. package/migration/20260225215848_workspace/snapshot.json +959 -0
  16. package/package.json +140 -0
  17. package/package.json.bak +140 -0
  18. package/parsers-config.ts +254 -0
  19. package/script/build.ts +224 -0
  20. package/script/check-migrations.ts +16 -0
  21. package/script/postinstall.mjs +131 -0
  22. package/script/publish.ts +181 -0
  23. package/script/schema.ts +63 -0
  24. package/script/seed-e2e.ts +50 -0
  25. package/src/acp/README.md +174 -0
  26. package/src/acp/agent.ts +1741 -0
  27. package/src/acp/session.ts +116 -0
  28. package/src/acp/types.ts +23 -0
  29. package/src/agent/agent.ts +339 -0
  30. package/src/agent/generate.txt +75 -0
  31. package/src/agent/prompt/compaction.txt +14 -0
  32. package/src/agent/prompt/explore.txt +18 -0
  33. package/src/agent/prompt/summary.txt +11 -0
  34. package/src/agent/prompt/title.txt +44 -0
  35. package/src/auth/index.ts +68 -0
  36. package/src/bun/index.ts +131 -0
  37. package/src/bun/registry.ts +50 -0
  38. package/src/bus/bus-event.ts +43 -0
  39. package/src/bus/global.ts +10 -0
  40. package/src/bus/index.ts +105 -0
  41. package/src/cli/bootstrap.ts +17 -0
  42. package/src/cli/cmd/acp.ts +70 -0
  43. package/src/cli/cmd/agent.ts +257 -0
  44. package/src/cli/cmd/auth.ts +449 -0
  45. package/src/cli/cmd/cmd.ts +7 -0
  46. package/src/cli/cmd/db.ts +118 -0
  47. package/src/cli/cmd/debug/agent.ts +167 -0
  48. package/src/cli/cmd/debug/config.ts +16 -0
  49. package/src/cli/cmd/debug/file.ts +97 -0
  50. package/src/cli/cmd/debug/index.ts +48 -0
  51. package/src/cli/cmd/debug/lsp.ts +52 -0
  52. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  53. package/src/cli/cmd/debug/scrap.ts +16 -0
  54. package/src/cli/cmd/debug/skill.ts +16 -0
  55. package/src/cli/cmd/debug/snapshot.ts +52 -0
  56. package/src/cli/cmd/export.ts +88 -0
  57. package/src/cli/cmd/generate.ts +38 -0
  58. package/src/cli/cmd/github.ts +1631 -0
  59. package/src/cli/cmd/import.ts +170 -0
  60. package/src/cli/cmd/mcp.ts +754 -0
  61. package/src/cli/cmd/models.ts +77 -0
  62. package/src/cli/cmd/pr.ts +112 -0
  63. package/src/cli/cmd/run.ts +625 -0
  64. package/src/cli/cmd/serve.ts +31 -0
  65. package/src/cli/cmd/session.ts +156 -0
  66. package/src/cli/cmd/stats.ts +410 -0
  67. package/src/cli/cmd/tui/app.tsx +845 -0
  68. package/src/cli/cmd/tui/attach.ts +88 -0
  69. package/src/cli/cmd/tui/component/border.tsx +21 -0
  70. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  71. package/src/cli/cmd/tui/component/dialog-command.tsx +147 -0
  72. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  73. package/src/cli/cmd/tui/component/dialog-model.tsx +165 -0
  74. package/src/cli/cmd/tui/component/dialog-provider.tsx +259 -0
  75. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  76. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  77. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  78. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  79. package/src/cli/cmd/tui/component/dialog-status.tsx +167 -0
  80. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  81. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  82. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  83. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +667 -0
  84. package/src/cli/cmd/tui/component/prompt/frecency.tsx +90 -0
  85. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  86. package/src/cli/cmd/tui/component/prompt/index.tsx +1155 -0
  87. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  88. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  89. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  90. package/src/cli/cmd/tui/component/tips.tsx +152 -0
  91. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  92. package/src/cli/cmd/tui/context/args.tsx +15 -0
  93. package/src/cli/cmd/tui/context/directory.ts +13 -0
  94. package/src/cli/cmd/tui/context/exit.tsx +53 -0
  95. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  96. package/src/cli/cmd/tui/context/keybind.tsx +102 -0
  97. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  98. package/src/cli/cmd/tui/context/local.tsx +406 -0
  99. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  100. package/src/cli/cmd/tui/context/route.tsx +46 -0
  101. package/src/cli/cmd/tui/context/sdk.tsx +101 -0
  102. package/src/cli/cmd/tui/context/sync.tsx +488 -0
  103. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  104. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  105. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  106. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  107. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  108. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  109. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  110. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  111. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  112. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  113. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  114. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  115. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  116. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  117. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  118. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  119. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  120. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  121. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  122. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  123. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  124. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  125. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  126. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  127. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  128. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  129. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  130. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  131. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  132. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  133. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  134. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  135. package/src/cli/cmd/tui/context/theme.tsx +1152 -0
  136. package/src/cli/cmd/tui/context/tui-config.tsx +9 -0
  137. package/src/cli/cmd/tui/event.ts +48 -0
  138. package/src/cli/cmd/tui/routes/home.tsx +145 -0
  139. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  140. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  141. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  142. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  143. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  144. package/src/cli/cmd/tui/routes/session/header.tsx +135 -0
  145. package/src/cli/cmd/tui/routes/session/index.tsx +2219 -0
  146. package/src/cli/cmd/tui/routes/session/permission.tsx +685 -0
  147. package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
  148. package/src/cli/cmd/tui/routes/session/sidebar.tsx +321 -0
  149. package/src/cli/cmd/tui/thread.ts +199 -0
  150. package/src/cli/cmd/tui/ui/dialog-alert.tsx +59 -0
  151. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +85 -0
  152. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +207 -0
  153. package/src/cli/cmd/tui/ui/dialog-help.tsx +40 -0
  154. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +80 -0
  155. package/src/cli/cmd/tui/ui/dialog-select.tsx +401 -0
  156. package/src/cli/cmd/tui/ui/dialog.tsx +182 -0
  157. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  158. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  159. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  160. package/src/cli/cmd/tui/util/clipboard.ts +164 -0
  161. package/src/cli/cmd/tui/util/editor.ts +33 -0
  162. package/src/cli/cmd/tui/util/selection.ts +25 -0
  163. package/src/cli/cmd/tui/util/signal.ts +7 -0
  164. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  165. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  166. package/src/cli/cmd/tui/win32.ts +129 -0
  167. package/src/cli/cmd/tui/worker.ts +157 -0
  168. package/src/cli/cmd/uninstall.ts +356 -0
  169. package/src/cli/cmd/upgrade.ts +73 -0
  170. package/src/cli/cmd/web.ts +81 -0
  171. package/src/cli/cmd/workspace-serve.ts +16 -0
  172. package/src/cli/error.ts +57 -0
  173. package/src/cli/logo.ts +6 -0
  174. package/src/cli/network.ts +60 -0
  175. package/src/cli/ui.ts +116 -0
  176. package/src/cli/upgrade.ts +25 -0
  177. package/src/command/index.ts +150 -0
  178. package/src/command/template/initialize.txt +10 -0
  179. package/src/command/template/review.txt +101 -0
  180. package/src/config/config.ts +1408 -0
  181. package/src/config/markdown.ts +99 -0
  182. package/src/config/migrate-tui-config.ts +155 -0
  183. package/src/config/paths.ts +174 -0
  184. package/src/config/tui-schema.ts +34 -0
  185. package/src/config/tui.ts +118 -0
  186. package/src/control/control.sql.ts +22 -0
  187. package/src/control/index.ts +67 -0
  188. package/src/control-plane/adaptors/index.ts +10 -0
  189. package/src/control-plane/adaptors/types.ts +7 -0
  190. package/src/control-plane/adaptors/worktree.ts +26 -0
  191. package/src/control-plane/config.ts +10 -0
  192. package/src/control-plane/session-proxy-middleware.ts +46 -0
  193. package/src/control-plane/sse.ts +66 -0
  194. package/src/control-plane/workspace-server/routes.ts +33 -0
  195. package/src/control-plane/workspace-server/server.ts +24 -0
  196. package/src/control-plane/workspace.sql.ts +12 -0
  197. package/src/control-plane/workspace.ts +160 -0
  198. package/src/env/index.ts +28 -0
  199. package/src/file/ignore.ts +82 -0
  200. package/src/file/index.ts +646 -0
  201. package/src/file/ripgrep.ts +372 -0
  202. package/src/file/time.ts +71 -0
  203. package/src/file/watcher.ts +128 -0
  204. package/src/flag/flag.ts +109 -0
  205. package/src/format/formatter.ts +395 -0
  206. package/src/format/index.ts +140 -0
  207. package/src/global/index.ts +54 -0
  208. package/src/id/id.ts +84 -0
  209. package/src/ide/index.ts +76 -0
  210. package/src/index.ts +210 -0
  211. package/src/installation/index.ts +266 -0
  212. package/src/lsp/client.ts +251 -0
  213. package/src/lsp/index.ts +485 -0
  214. package/src/lsp/language.ts +120 -0
  215. package/src/lsp/server.ts +2142 -0
  216. package/src/mcp/auth.ts +130 -0
  217. package/src/mcp/index.ts +937 -0
  218. package/src/mcp/oauth-callback.ts +200 -0
  219. package/src/mcp/oauth-provider.ts +176 -0
  220. package/src/patch/index.ts +680 -0
  221. package/src/permission/arity.ts +163 -0
  222. package/src/permission/index.ts +210 -0
  223. package/src/permission/next.ts +286 -0
  224. package/src/plugin/codex.ts +624 -0
  225. package/src/plugin/copilot.ts +327 -0
  226. package/src/plugin/index.ts +143 -0
  227. package/src/project/bootstrap.ts +33 -0
  228. package/src/project/instance.ts +114 -0
  229. package/src/project/project.sql.ts +15 -0
  230. package/src/project/project.ts +441 -0
  231. package/src/project/state.ts +70 -0
  232. package/src/project/vcs.ts +76 -0
  233. package/src/provider/auth.ts +147 -0
  234. package/src/provider/error.ts +189 -0
  235. package/src/provider/models.ts +146 -0
  236. package/src/provider/provider.ts +1338 -0
  237. package/src/provider/sdk/copilot/README.md +5 -0
  238. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
  239. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  240. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
  241. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  242. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
  243. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  244. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  245. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
  246. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  247. package/src/provider/sdk/copilot/index.ts +2 -0
  248. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  249. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
  250. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  251. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  252. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  253. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
  254. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
  255. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
  256. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  257. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
  258. package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
  259. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
  260. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
  261. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
  262. package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
  263. package/src/provider/transform.ts +955 -0
  264. package/src/pty/index.ts +324 -0
  265. package/src/question/index.ts +171 -0
  266. package/src/scheduler/index.ts +61 -0
  267. package/src/server/error.ts +36 -0
  268. package/src/server/event.ts +7 -0
  269. package/src/server/mdns.ts +60 -0
  270. package/src/server/routes/config.ts +92 -0
  271. package/src/server/routes/experimental.ts +270 -0
  272. package/src/server/routes/file.ts +197 -0
  273. package/src/server/routes/global.ts +185 -0
  274. package/src/server/routes/mcp.ts +225 -0
  275. package/src/server/routes/permission.ts +68 -0
  276. package/src/server/routes/project.ts +82 -0
  277. package/src/server/routes/provider.ts +165 -0
  278. package/src/server/routes/pty.ts +200 -0
  279. package/src/server/routes/question.ts +98 -0
  280. package/src/server/routes/session.ts +974 -0
  281. package/src/server/routes/tui.ts +379 -0
  282. package/src/server/routes/workspace.ts +104 -0
  283. package/src/server/server.ts +623 -0
  284. package/src/session/compaction.ts +261 -0
  285. package/src/session/index.ts +877 -0
  286. package/src/session/instruction.ts +192 -0
  287. package/src/session/llm.ts +279 -0
  288. package/src/session/message-v2.ts +899 -0
  289. package/src/session/message.ts +189 -0
  290. package/src/session/processor.ts +421 -0
  291. package/src/session/prompt/anthropic-20250930.txt +166 -0
  292. package/src/session/prompt/anthropic.txt +105 -0
  293. package/src/session/prompt/beast.txt +147 -0
  294. package/src/session/prompt/build-switch.txt +5 -0
  295. package/src/session/prompt/codex_header.txt +79 -0
  296. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  297. package/src/session/prompt/gemini.txt +155 -0
  298. package/src/session/prompt/max-steps.txt +16 -0
  299. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  300. package/src/session/prompt/plan.txt +26 -0
  301. package/src/session/prompt/qwen.txt +109 -0
  302. package/src/session/prompt/trinity.txt +97 -0
  303. package/src/session/prompt.ts +1959 -0
  304. package/src/session/retry.ts +101 -0
  305. package/src/session/revert.ts +138 -0
  306. package/src/session/session.sql.ts +88 -0
  307. package/src/session/status.ts +76 -0
  308. package/src/session/summary.ts +161 -0
  309. package/src/session/system.ts +54 -0
  310. package/src/session/todo.ts +56 -0
  311. package/src/share/share-next.ts +210 -0
  312. package/src/share/share.sql.ts +13 -0
  313. package/src/shell/shell.ts +68 -0
  314. package/src/skill/discovery.ts +98 -0
  315. package/src/skill/index.ts +1 -0
  316. package/src/skill/skill.ts +189 -0
  317. package/src/snapshot/index.ts +297 -0
  318. package/src/sql.d.ts +4 -0
  319. package/src/storage/db.ts +155 -0
  320. package/src/storage/json-migration.ts +425 -0
  321. package/src/storage/schema.sql.ts +10 -0
  322. package/src/storage/schema.ts +5 -0
  323. package/src/storage/storage.ts +220 -0
  324. package/src/tool/apply_patch.ts +281 -0
  325. package/src/tool/apply_patch.txt +33 -0
  326. package/src/tool/bash.ts +274 -0
  327. package/src/tool/bash.txt +115 -0
  328. package/src/tool/batch.ts +181 -0
  329. package/src/tool/batch.txt +24 -0
  330. package/src/tool/codesearch.ts +132 -0
  331. package/src/tool/codesearch.txt +12 -0
  332. package/src/tool/edit.ts +654 -0
  333. package/src/tool/edit.txt +10 -0
  334. package/src/tool/external-directory.ts +32 -0
  335. package/src/tool/glob.ts +78 -0
  336. package/src/tool/glob.txt +6 -0
  337. package/src/tool/grep.ts +156 -0
  338. package/src/tool/grep.txt +8 -0
  339. package/src/tool/invalid.ts +17 -0
  340. package/src/tool/ls.ts +121 -0
  341. package/src/tool/ls.txt +1 -0
  342. package/src/tool/lsp.ts +97 -0
  343. package/src/tool/lsp.txt +19 -0
  344. package/src/tool/multiedit.ts +46 -0
  345. package/src/tool/multiedit.txt +41 -0
  346. package/src/tool/plan-enter.txt +14 -0
  347. package/src/tool/plan-exit.txt +13 -0
  348. package/src/tool/plan.ts +131 -0
  349. package/src/tool/question.ts +33 -0
  350. package/src/tool/question.txt +10 -0
  351. package/src/tool/read.ts +293 -0
  352. package/src/tool/read.txt +14 -0
  353. package/src/tool/registry.ts +173 -0
  354. package/src/tool/skill.ts +123 -0
  355. package/src/tool/task.ts +165 -0
  356. package/src/tool/task.txt +60 -0
  357. package/src/tool/todo.ts +53 -0
  358. package/src/tool/todoread.txt +14 -0
  359. package/src/tool/todowrite.txt +167 -0
  360. package/src/tool/tool.ts +89 -0
  361. package/src/tool/truncation.ts +107 -0
  362. package/src/tool/webfetch.ts +206 -0
  363. package/src/tool/webfetch.txt +13 -0
  364. package/src/tool/websearch.ts +150 -0
  365. package/src/tool/websearch.txt +14 -0
  366. package/src/tool/write.ts +84 -0
  367. package/src/tool/write.txt +8 -0
  368. package/src/util/abort.ts +35 -0
  369. package/src/util/archive.ts +16 -0
  370. package/src/util/color.ts +19 -0
  371. package/src/util/context.ts +25 -0
  372. package/src/util/defer.ts +12 -0
  373. package/src/util/eventloop.ts +20 -0
  374. package/src/util/filesystem.ts +189 -0
  375. package/src/util/fn.ts +11 -0
  376. package/src/util/format.ts +20 -0
  377. package/src/util/git.ts +35 -0
  378. package/src/util/glob.ts +34 -0
  379. package/src/util/iife.ts +3 -0
  380. package/src/util/keybind.ts +103 -0
  381. package/src/util/lazy.ts +23 -0
  382. package/src/util/locale.ts +81 -0
  383. package/src/util/lock.ts +98 -0
  384. package/src/util/log.ts +182 -0
  385. package/src/util/process.ts +126 -0
  386. package/src/util/proxied.ts +3 -0
  387. package/src/util/queue.ts +32 -0
  388. package/src/util/rpc.ts +66 -0
  389. package/src/util/scrap.ts +10 -0
  390. package/src/util/signal.ts +12 -0
  391. package/src/util/timeout.ts +14 -0
  392. package/src/util/token.ts +7 -0
  393. package/src/util/wildcard.ts +59 -0
  394. package/src/worktree/index.ts +643 -0
  395. package/sst-env.d.ts +10 -0
  396. package/test/AGENTS.md +81 -0
  397. package/test/acp/agent-interface.test.ts +51 -0
  398. package/test/acp/event-subscription.test.ts +683 -0
  399. package/test/agent/agent.test.ts +689 -0
  400. package/test/bun.test.ts +53 -0
  401. package/test/cli/github-action.test.ts +197 -0
  402. package/test/cli/github-remote.test.ts +80 -0
  403. package/test/cli/import.test.ts +38 -0
  404. package/test/cli/plugin-auth-picker.test.ts +120 -0
  405. package/test/cli/tui/transcript.test.ts +322 -0
  406. package/test/config/agent-color.test.ts +71 -0
  407. package/test/config/config.test.ts +1886 -0
  408. package/test/config/fixtures/empty-frontmatter.md +4 -0
  409. package/test/config/fixtures/frontmatter.md +28 -0
  410. package/test/config/fixtures/markdown-header.md +11 -0
  411. package/test/config/fixtures/no-frontmatter.md +1 -0
  412. package/test/config/fixtures/weird-model-id.md +13 -0
  413. package/test/config/markdown.test.ts +228 -0
  414. package/test/config/tui.test.ts +510 -0
  415. package/test/control-plane/session-proxy-middleware.test.ts +147 -0
  416. package/test/control-plane/sse.test.ts +56 -0
  417. package/test/control-plane/workspace-server-sse.test.ts +65 -0
  418. package/test/control-plane/workspace-sync.test.ts +97 -0
  419. package/test/file/ignore.test.ts +10 -0
  420. package/test/file/index.test.ts +394 -0
  421. package/test/file/path-traversal.test.ts +198 -0
  422. package/test/file/ripgrep.test.ts +39 -0
  423. package/test/file/time.test.ts +361 -0
  424. package/test/fixture/db.ts +11 -0
  425. package/test/fixture/fixture.ts +45 -0
  426. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  427. package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
  428. package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
  429. package/test/fixture/skills/cloudflare/SKILL.md +211 -0
  430. package/test/fixture/skills/index.json +6 -0
  431. package/test/ide/ide.test.ts +82 -0
  432. package/test/keybind.test.ts +421 -0
  433. package/test/lsp/client.test.ts +95 -0
  434. package/test/mcp/headers.test.ts +153 -0
  435. package/test/mcp/oauth-browser.test.ts +249 -0
  436. package/test/memory/abort-leak.test.ts +136 -0
  437. package/test/patch/patch.test.ts +348 -0
  438. package/test/permission/arity.test.ts +33 -0
  439. package/test/permission/next.test.ts +689 -0
  440. package/test/permission-task.test.ts +319 -0
  441. package/test/plugin/auth-override.test.ts +44 -0
  442. package/test/plugin/codex.test.ts +123 -0
  443. package/test/preload.ts +80 -0
  444. package/test/project/project.test.ts +348 -0
  445. package/test/project/worktree-remove.test.ts +65 -0
  446. package/test/provider/amazon-bedrock.test.ts +446 -0
  447. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  448. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  449. package/test/provider/gitlab-duo.test.ts +262 -0
  450. package/test/provider/provider.test.ts +2220 -0
  451. package/test/provider/transform.test.ts +2353 -0
  452. package/test/pty/pty-output-isolation.test.ts +140 -0
  453. package/test/question/question.test.ts +300 -0
  454. package/test/scheduler.test.ts +73 -0
  455. package/test/server/global-session-list.test.ts +89 -0
  456. package/test/server/session-list.test.ts +90 -0
  457. package/test/server/session-select.test.ts +78 -0
  458. package/test/session/compaction.test.ts +423 -0
  459. package/test/session/instruction.test.ts +170 -0
  460. package/test/session/llm.test.ts +667 -0
  461. package/test/session/message-v2.test.ts +924 -0
  462. package/test/session/prompt.test.ts +211 -0
  463. package/test/session/retry.test.ts +188 -0
  464. package/test/session/revert-compact.test.ts +285 -0
  465. package/test/session/session.test.ts +71 -0
  466. package/test/session/structured-output-integration.test.ts +233 -0
  467. package/test/session/structured-output.test.ts +385 -0
  468. package/test/skill/discovery.test.ts +110 -0
  469. package/test/skill/skill.test.ts +388 -0
  470. package/test/snapshot/snapshot.test.ts +1180 -0
  471. package/test/storage/json-migration.test.ts +846 -0
  472. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  473. package/test/tool/apply_patch.test.ts +566 -0
  474. package/test/tool/bash.test.ts +402 -0
  475. package/test/tool/edit.test.ts +496 -0
  476. package/test/tool/external-directory.test.ts +127 -0
  477. package/test/tool/fixtures/large-image.png +0 -0
  478. package/test/tool/fixtures/models-api.json +38413 -0
  479. package/test/tool/grep.test.ts +110 -0
  480. package/test/tool/question.test.ts +107 -0
  481. package/test/tool/read.test.ts +504 -0
  482. package/test/tool/registry.test.ts +122 -0
  483. package/test/tool/skill.test.ts +112 -0
  484. package/test/tool/truncation.test.ts +160 -0
  485. package/test/tool/webfetch.test.ts +100 -0
  486. package/test/tool/write.test.ts +348 -0
  487. package/test/util/filesystem.test.ts +443 -0
  488. package/test/util/format.test.ts +59 -0
  489. package/test/util/glob.test.ts +164 -0
  490. package/test/util/iife.test.ts +36 -0
  491. package/test/util/lazy.test.ts +50 -0
  492. package/test/util/lock.test.ts +72 -0
  493. package/test/util/process.test.ts +59 -0
  494. package/test/util/timeout.test.ts +21 -0
  495. package/test/util/wildcard.test.ts +90 -0
  496. package/tsconfig.json +16 -0
@@ -0,0 +1,281 @@
1
+ import z from "zod"
2
+ import * as path from "path"
3
+ import * as fs from "fs/promises"
4
+ import { Tool } from "./tool"
5
+ import { Bus } from "../bus"
6
+ import { FileWatcher } from "../file/watcher"
7
+ import { Instance } from "../project/instance"
8
+ import { Patch } from "../patch"
9
+ import { createTwoFilesPatch, diffLines } from "diff"
10
+ import { assertExternalDirectory } from "./external-directory"
11
+ import { trimDiff } from "./edit"
12
+ import { LSP } from "../lsp"
13
+ import { Filesystem } from "../util/filesystem"
14
+ import DESCRIPTION from "./apply_patch.txt"
15
+ import { File } from "../file"
16
+
17
+ const PatchParams = z.object({
18
+ patchText: z.string().describe("The full patch text that describes all changes to be made"),
19
+ })
20
+
21
+ export const ApplyPatchTool = Tool.define("apply_patch", {
22
+ description: DESCRIPTION,
23
+ parameters: PatchParams,
24
+ async execute(params, ctx) {
25
+ if (!params.patchText) {
26
+ throw new Error("patchText is required")
27
+ }
28
+
29
+ // Parse the patch to get hunks
30
+ let hunks: Patch.Hunk[]
31
+ try {
32
+ const parseResult = Patch.parsePatch(params.patchText)
33
+ hunks = parseResult.hunks
34
+ } catch (error) {
35
+ throw new Error(`apply_patch verification failed: ${error}`)
36
+ }
37
+
38
+ if (hunks.length === 0) {
39
+ const normalized = params.patchText.replace(/\r\n/g, "\n").replace(/\r/g, "\n").trim()
40
+ if (normalized === "*** Begin Patch\n*** End Patch") {
41
+ throw new Error("patch rejected: empty patch")
42
+ }
43
+ throw new Error("apply_patch verification failed: no hunks found")
44
+ }
45
+
46
+ // Validate file paths and check permissions
47
+ const fileChanges: Array<{
48
+ filePath: string
49
+ oldContent: string
50
+ newContent: string
51
+ type: "add" | "update" | "delete" | "move"
52
+ movePath?: string
53
+ diff: string
54
+ additions: number
55
+ deletions: number
56
+ }> = []
57
+
58
+ let totalDiff = ""
59
+
60
+ for (const hunk of hunks) {
61
+ const filePath = path.resolve(Instance.directory, hunk.path)
62
+ await assertExternalDirectory(ctx, filePath)
63
+
64
+ switch (hunk.type) {
65
+ case "add": {
66
+ const oldContent = ""
67
+ const newContent =
68
+ hunk.contents.length === 0 || hunk.contents.endsWith("\n") ? hunk.contents : `${hunk.contents}\n`
69
+ const diff = trimDiff(createTwoFilesPatch(filePath, filePath, oldContent, newContent))
70
+
71
+ let additions = 0
72
+ let deletions = 0
73
+ for (const change of diffLines(oldContent, newContent)) {
74
+ if (change.added) additions += change.count || 0
75
+ if (change.removed) deletions += change.count || 0
76
+ }
77
+
78
+ fileChanges.push({
79
+ filePath,
80
+ oldContent,
81
+ newContent,
82
+ type: "add",
83
+ diff,
84
+ additions,
85
+ deletions,
86
+ })
87
+
88
+ totalDiff += diff + "\n"
89
+ break
90
+ }
91
+
92
+ case "update": {
93
+ // Check if file exists for update
94
+ const stats = await fs.stat(filePath).catch(() => null)
95
+ if (!stats || stats.isDirectory()) {
96
+ throw new Error(`apply_patch verification failed: Failed to read file to update: ${filePath}`)
97
+ }
98
+
99
+ const oldContent = await fs.readFile(filePath, "utf-8")
100
+ let newContent = oldContent
101
+
102
+ // Apply the update chunks to get new content
103
+ try {
104
+ const fileUpdate = Patch.deriveNewContentsFromChunks(filePath, hunk.chunks)
105
+ newContent = fileUpdate.content
106
+ } catch (error) {
107
+ throw new Error(`apply_patch verification failed: ${error}`)
108
+ }
109
+
110
+ const diff = trimDiff(createTwoFilesPatch(filePath, filePath, oldContent, newContent))
111
+
112
+ let additions = 0
113
+ let deletions = 0
114
+ for (const change of diffLines(oldContent, newContent)) {
115
+ if (change.added) additions += change.count || 0
116
+ if (change.removed) deletions += change.count || 0
117
+ }
118
+
119
+ const movePath = hunk.move_path ? path.resolve(Instance.directory, hunk.move_path) : undefined
120
+ await assertExternalDirectory(ctx, movePath)
121
+
122
+ fileChanges.push({
123
+ filePath,
124
+ oldContent,
125
+ newContent,
126
+ type: hunk.move_path ? "move" : "update",
127
+ movePath,
128
+ diff,
129
+ additions,
130
+ deletions,
131
+ })
132
+
133
+ totalDiff += diff + "\n"
134
+ break
135
+ }
136
+
137
+ case "delete": {
138
+ const contentToDelete = await fs.readFile(filePath, "utf-8").catch((error) => {
139
+ throw new Error(`apply_patch verification failed: ${error}`)
140
+ })
141
+ const deleteDiff = trimDiff(createTwoFilesPatch(filePath, filePath, contentToDelete, ""))
142
+
143
+ const deletions = contentToDelete.split("\n").length
144
+
145
+ fileChanges.push({
146
+ filePath,
147
+ oldContent: contentToDelete,
148
+ newContent: "",
149
+ type: "delete",
150
+ diff: deleteDiff,
151
+ additions: 0,
152
+ deletions,
153
+ })
154
+
155
+ totalDiff += deleteDiff + "\n"
156
+ break
157
+ }
158
+ }
159
+ }
160
+
161
+ // Build per-file metadata for UI rendering (used for both permission and result)
162
+ const files = fileChanges.map((change) => ({
163
+ filePath: change.filePath,
164
+ relativePath: path.relative(Instance.worktree, change.movePath ?? change.filePath).replaceAll("\\", "/"),
165
+ type: change.type,
166
+ diff: change.diff,
167
+ before: change.oldContent,
168
+ after: change.newContent,
169
+ additions: change.additions,
170
+ deletions: change.deletions,
171
+ movePath: change.movePath,
172
+ }))
173
+
174
+ // Check permissions if needed
175
+ const relativePaths = fileChanges.map((c) => path.relative(Instance.worktree, c.filePath).replaceAll("\\", "/"))
176
+ await ctx.ask({
177
+ permission: "edit",
178
+ patterns: relativePaths,
179
+ always: ["*"],
180
+ metadata: {
181
+ filepath: relativePaths.join(", "),
182
+ diff: totalDiff,
183
+ files,
184
+ },
185
+ })
186
+
187
+ // Apply the changes
188
+ const updates: Array<{ file: string; event: "add" | "change" | "unlink" }> = []
189
+
190
+ for (const change of fileChanges) {
191
+ const edited = change.type === "delete" ? undefined : (change.movePath ?? change.filePath)
192
+ switch (change.type) {
193
+ case "add":
194
+ // Create parent directories (recursive: true is safe on existing/root dirs)
195
+ await fs.mkdir(path.dirname(change.filePath), { recursive: true })
196
+ await fs.writeFile(change.filePath, change.newContent, "utf-8")
197
+ updates.push({ file: change.filePath, event: "add" })
198
+ break
199
+
200
+ case "update":
201
+ await fs.writeFile(change.filePath, change.newContent, "utf-8")
202
+ updates.push({ file: change.filePath, event: "change" })
203
+ break
204
+
205
+ case "move":
206
+ if (change.movePath) {
207
+ // Create parent directories (recursive: true is safe on existing/root dirs)
208
+ await fs.mkdir(path.dirname(change.movePath), { recursive: true })
209
+ await fs.writeFile(change.movePath, change.newContent, "utf-8")
210
+ await fs.unlink(change.filePath)
211
+ updates.push({ file: change.filePath, event: "unlink" })
212
+ updates.push({ file: change.movePath, event: "add" })
213
+ }
214
+ break
215
+
216
+ case "delete":
217
+ await fs.unlink(change.filePath)
218
+ updates.push({ file: change.filePath, event: "unlink" })
219
+ break
220
+ }
221
+
222
+ if (edited) {
223
+ await Bus.publish(File.Event.Edited, {
224
+ file: edited,
225
+ })
226
+ }
227
+ }
228
+
229
+ // Publish file change events
230
+ for (const update of updates) {
231
+ await Bus.publish(FileWatcher.Event.Updated, update)
232
+ }
233
+
234
+ // Notify LSP of file changes and collect diagnostics
235
+ for (const change of fileChanges) {
236
+ if (change.type === "delete") continue
237
+ const target = change.movePath ?? change.filePath
238
+ await LSP.touchFile(target, true)
239
+ }
240
+ const diagnostics = await LSP.diagnostics()
241
+
242
+ // Generate output summary
243
+ const summaryLines = fileChanges.map((change) => {
244
+ if (change.type === "add") {
245
+ return `A ${path.relative(Instance.worktree, change.filePath).replaceAll("\\", "/")}`
246
+ }
247
+ if (change.type === "delete") {
248
+ return `D ${path.relative(Instance.worktree, change.filePath).replaceAll("\\", "/")}`
249
+ }
250
+ const target = change.movePath ?? change.filePath
251
+ return `M ${path.relative(Instance.worktree, target).replaceAll("\\", "/")}`
252
+ })
253
+ let output = `Success. Updated the following files:\n${summaryLines.join("\n")}`
254
+
255
+ // Report LSP errors for changed files
256
+ const MAX_DIAGNOSTICS_PER_FILE = 20
257
+ for (const change of fileChanges) {
258
+ if (change.type === "delete") continue
259
+ const target = change.movePath ?? change.filePath
260
+ const normalized = Filesystem.normalizePath(target)
261
+ const issues = diagnostics[normalized] ?? []
262
+ const errors = issues.filter((item) => item.severity === 1)
263
+ if (errors.length > 0) {
264
+ const limited = errors.slice(0, MAX_DIAGNOSTICS_PER_FILE)
265
+ const suffix =
266
+ errors.length > MAX_DIAGNOSTICS_PER_FILE ? `\n... and ${errors.length - MAX_DIAGNOSTICS_PER_FILE} more` : ""
267
+ output += `\n\nLSP errors detected in ${path.relative(Instance.worktree, target).replaceAll("\\", "/")}, please fix:\n<diagnostics file="${target}">\n${limited.map(LSP.Diagnostic.pretty).join("\n")}${suffix}\n</diagnostics>`
268
+ }
269
+ }
270
+
271
+ return {
272
+ title: output,
273
+ metadata: {
274
+ diff: totalDiff,
275
+ files,
276
+ diagnostics,
277
+ },
278
+ output,
279
+ }
280
+ },
281
+ })
@@ -0,0 +1,33 @@
1
+ Use the `apply_patch` tool to edit files. Your patch language is a stripped‑down, file‑oriented diff format designed to be easy to parse and safe to apply. You can think of it as a high‑level envelope:
2
+
3
+ *** Begin Patch
4
+ [ one or more file sections ]
5
+ *** End Patch
6
+
7
+ Within that envelope, you get a sequence of file operations.
8
+ You MUST include a header to specify the action you are taking.
9
+ Each operation starts with one of three headers:
10
+
11
+ *** Add File: <path> - create a new file. Every following line is a + line (the initial contents).
12
+ *** Delete File: <path> - remove an existing file. Nothing follows.
13
+ *** Update File: <path> - patch an existing file in place (optionally with a rename).
14
+
15
+ Example patch:
16
+
17
+ ```
18
+ *** Begin Patch
19
+ *** Add File: hello.txt
20
+ +Hello world
21
+ *** Update File: src/app.py
22
+ *** Move to: src/main.py
23
+ @@ def greet():
24
+ -print("Hi")
25
+ +print("Hello, world!")
26
+ *** Delete File: obsolete.txt
27
+ *** End Patch
28
+ ```
29
+
30
+ It is important to remember:
31
+
32
+ - You must include a header with your intended action (Add/Delete/Update)
33
+ - You must prefix new lines with `+` even when creating a new file
@@ -0,0 +1,274 @@
1
+ import z from "zod"
2
+ import { spawn } from "child_process"
3
+ import { Tool } from "./tool"
4
+ import path from "path"
5
+ import DESCRIPTION from "./bash.txt"
6
+ import { Log } from "../util/log"
7
+ import { Instance } from "../project/instance"
8
+ import { lazy } from "@/util/lazy"
9
+ import { Language } from "web-tree-sitter"
10
+
11
+ import { $ } from "bun"
12
+ import { Filesystem } from "@/util/filesystem"
13
+ import { fileURLToPath } from "url"
14
+ import { Flag } from "@/flag/flag.ts"
15
+ import { Shell } from "@/shell/shell"
16
+
17
+ import { BashArity } from "@/permission/arity"
18
+ import { Truncate } from "./truncation"
19
+ import { Plugin } from "@/plugin"
20
+
21
+ const MAX_METADATA_LENGTH = 30_000
22
+ const DEFAULT_TIMEOUT = Flag.OPENCODE_EXPERIMENTAL_BASH_DEFAULT_TIMEOUT_MS || 2 * 60 * 1000
23
+
24
+ export const log = Log.create({ service: "bash-tool" })
25
+
26
+ const resolveWasm = (asset: string) => {
27
+ if (asset.startsWith("file://")) return fileURLToPath(asset)
28
+ if (asset.startsWith("/") || /^[a-z]:/i.test(asset)) return asset
29
+ const url = new URL(asset, import.meta.url)
30
+ return fileURLToPath(url)
31
+ }
32
+
33
+ const parser = lazy(async () => {
34
+ const { Parser } = await import("web-tree-sitter")
35
+ const { default: treeWasm } = await import("web-tree-sitter/tree-sitter.wasm" as string, {
36
+ with: { type: "wasm" },
37
+ })
38
+ const treePath = resolveWasm(treeWasm)
39
+ await Parser.init({
40
+ locateFile() {
41
+ return treePath
42
+ },
43
+ })
44
+ const { default: bashWasm } = await import("tree-sitter-bash/tree-sitter-bash.wasm" as string, {
45
+ with: { type: "wasm" },
46
+ })
47
+ const bashPath = resolveWasm(bashWasm)
48
+ const bashLanguage = await Language.load(bashPath)
49
+ const p = new Parser()
50
+ p.setLanguage(bashLanguage)
51
+ return p
52
+ })
53
+
54
+ // TODO: we may wanna rename this tool so it works better on other shells
55
+ export const BashTool = Tool.define("bash", async () => {
56
+ const shell = Shell.acceptable()
57
+ log.info("bash tool using shell", { shell })
58
+
59
+ return {
60
+ description: DESCRIPTION.replaceAll("${directory}", Instance.directory)
61
+ .replaceAll("${maxLines}", String(Truncate.MAX_LINES))
62
+ .replaceAll("${maxBytes}", String(Truncate.MAX_BYTES)),
63
+ parameters: z.object({
64
+ command: z.string().describe("The command to execute"),
65
+ timeout: z.number().describe("Optional timeout in milliseconds").optional(),
66
+ workdir: z
67
+ .string()
68
+ .describe(
69
+ `The working directory to run the command in. Defaults to ${Instance.directory}. Use this instead of 'cd' commands.`,
70
+ )
71
+ .optional(),
72
+ description: z
73
+ .string()
74
+ .describe(
75
+ "Clear, concise description of what this command does in 5-10 words. Examples:\nInput: ls\nOutput: Lists files in current directory\n\nInput: git status\nOutput: Shows working tree status\n\nInput: npm install\nOutput: Installs package dependencies\n\nInput: mkdir foo\nOutput: Creates directory 'foo'",
76
+ ),
77
+ }),
78
+ async execute(params, ctx) {
79
+ const cwd = params.workdir || Instance.directory
80
+ if (params.timeout !== undefined && params.timeout < 0) {
81
+ throw new Error(`Invalid timeout value: ${params.timeout}. Timeout must be a positive number.`)
82
+ }
83
+ const timeout = params.timeout ?? DEFAULT_TIMEOUT
84
+ const tree = await parser().then((p) => p.parse(params.command))
85
+ if (!tree) {
86
+ throw new Error("Failed to parse command")
87
+ }
88
+ const directories = new Set<string>()
89
+ if (!Instance.containsPath(cwd)) directories.add(cwd)
90
+ const patterns = new Set<string>()
91
+ const always = new Set<string>()
92
+
93
+ for (const node of tree.rootNode.descendantsOfType("command")) {
94
+ if (!node) continue
95
+
96
+ // Get full command text including redirects if present
97
+ let commandText = node.parent?.type === "redirected_statement" ? node.parent.text : node.text
98
+
99
+ const command = []
100
+ for (let i = 0; i < node.childCount; i++) {
101
+ const child = node.child(i)
102
+ if (!child) continue
103
+ if (
104
+ child.type !== "command_name" &&
105
+ child.type !== "word" &&
106
+ child.type !== "string" &&
107
+ child.type !== "raw_string" &&
108
+ child.type !== "concatenation"
109
+ ) {
110
+ continue
111
+ }
112
+ command.push(child.text)
113
+ }
114
+
115
+ // not an exhaustive list, but covers most common cases
116
+ if (["cd", "rm", "cp", "mv", "mkdir", "touch", "chmod", "chown", "cat"].includes(command[0])) {
117
+ for (const arg of command.slice(1)) {
118
+ if (arg.startsWith("-") || (command[0] === "chmod" && arg.startsWith("+"))) continue
119
+ const resolved = await $`realpath ${arg}`
120
+ .cwd(cwd)
121
+ .quiet()
122
+ .nothrow()
123
+ .text()
124
+ .then((x) => x.trim())
125
+ log.info("resolved path", { arg, resolved })
126
+ if (resolved) {
127
+ const normalized =
128
+ process.platform === "win32" ? Filesystem.windowsPath(resolved).replace(/\//g, "\\") : resolved
129
+ if (!Instance.containsPath(normalized)) {
130
+ const dir = (await Filesystem.isDir(normalized)) ? normalized : path.dirname(normalized)
131
+ directories.add(dir)
132
+ }
133
+ }
134
+ }
135
+ }
136
+
137
+ // cd covered by above check
138
+ if (command.length && command[0] !== "cd") {
139
+ patterns.add(commandText)
140
+ always.add(BashArity.prefix(command).join(" ") + " *")
141
+ }
142
+ }
143
+
144
+ if (directories.size > 0) {
145
+ const globs = Array.from(directories).map((dir) => {
146
+ // Preserve POSIX-looking paths with /s, even on Windows
147
+ if (dir.startsWith("/")) return `${dir.replace(/[\\/]+$/, "")}/*`
148
+ return path.join(dir, "*")
149
+ })
150
+ await ctx.ask({
151
+ permission: "external_directory",
152
+ patterns: globs,
153
+ always: globs,
154
+ metadata: {},
155
+ })
156
+ }
157
+
158
+ if (patterns.size > 0) {
159
+ await ctx.ask({
160
+ permission: "bash",
161
+ patterns: Array.from(patterns),
162
+ always: Array.from(always),
163
+ metadata: {},
164
+ })
165
+ }
166
+
167
+ const shellEnv = await Plugin.trigger(
168
+ "shell.env",
169
+ { cwd, sessionID: ctx.sessionID, callID: ctx.callID },
170
+ { env: {} },
171
+ )
172
+ const proc = spawn(params.command, {
173
+ shell,
174
+ cwd,
175
+ env: {
176
+ ...process.env,
177
+ ...shellEnv.env,
178
+ },
179
+ stdio: ["ignore", "pipe", "pipe"],
180
+ detached: process.platform !== "win32",
181
+ })
182
+
183
+ let output = ""
184
+
185
+ // Initialize metadata with empty output
186
+ ctx.metadata({
187
+ metadata: {
188
+ output: "",
189
+ description: params.description,
190
+ },
191
+ })
192
+
193
+ const append = (chunk: Buffer) => {
194
+ output += chunk.toString()
195
+ ctx.metadata({
196
+ metadata: {
197
+ // truncate the metadata to avoid GIANT blobs of data (has nothing to do w/ what agent can access)
198
+ output: output.length > MAX_METADATA_LENGTH ? output.slice(0, MAX_METADATA_LENGTH) + "\n\n..." : output,
199
+ description: params.description,
200
+ },
201
+ })
202
+ }
203
+
204
+ proc.stdout?.on("data", append)
205
+ proc.stderr?.on("data", append)
206
+
207
+ let timedOut = false
208
+ let aborted = false
209
+ let exited = false
210
+
211
+ const kill = () => Shell.killTree(proc, { exited: () => exited })
212
+
213
+ if (ctx.abort.aborted) {
214
+ aborted = true
215
+ await kill()
216
+ }
217
+
218
+ const abortHandler = () => {
219
+ aborted = true
220
+ void kill()
221
+ }
222
+
223
+ ctx.abort.addEventListener("abort", abortHandler, { once: true })
224
+
225
+ const timeoutTimer = setTimeout(() => {
226
+ timedOut = true
227
+ void kill()
228
+ }, timeout + 100)
229
+
230
+ await new Promise<void>((resolve, reject) => {
231
+ const cleanup = () => {
232
+ clearTimeout(timeoutTimer)
233
+ ctx.abort.removeEventListener("abort", abortHandler)
234
+ }
235
+
236
+ proc.once("exit", () => {
237
+ exited = true
238
+ cleanup()
239
+ resolve()
240
+ })
241
+
242
+ proc.once("error", (error) => {
243
+ exited = true
244
+ cleanup()
245
+ reject(error)
246
+ })
247
+ })
248
+
249
+ const resultMetadata: string[] = []
250
+
251
+ if (timedOut) {
252
+ resultMetadata.push(`bash tool terminated command after exceeding timeout ${timeout} ms`)
253
+ }
254
+
255
+ if (aborted) {
256
+ resultMetadata.push("User aborted the command")
257
+ }
258
+
259
+ if (resultMetadata.length > 0) {
260
+ output += "\n\n<bash_metadata>\n" + resultMetadata.join("\n") + "\n</bash_metadata>"
261
+ }
262
+
263
+ return {
264
+ title: params.description,
265
+ metadata: {
266
+ output: output.length > MAX_METADATA_LENGTH ? output.slice(0, MAX_METADATA_LENGTH) + "\n\n..." : output,
267
+ exit: proc.exitCode,
268
+ description: params.description,
269
+ },
270
+ output,
271
+ }
272
+ },
273
+ }
274
+ })