@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,372 @@
1
+ // Ripgrep utility functions
2
+ import path from "path"
3
+ import { Global } from "../global"
4
+ import fs from "fs/promises"
5
+ import z from "zod"
6
+ import { NamedError } from "@opencode-ai/util/error"
7
+ import { lazy } from "../util/lazy"
8
+ import { $ } from "bun"
9
+ import { Filesystem } from "../util/filesystem"
10
+ import { Process } from "../util/process"
11
+ import { text } from "node:stream/consumers"
12
+
13
+ import { ZipReader, BlobReader, BlobWriter } from "@zip.js/zip.js"
14
+ import { Log } from "@/util/log"
15
+
16
+ export namespace Ripgrep {
17
+ const log = Log.create({ service: "ripgrep" })
18
+ const Stats = z.object({
19
+ elapsed: z.object({
20
+ secs: z.number(),
21
+ nanos: z.number(),
22
+ human: z.string(),
23
+ }),
24
+ searches: z.number(),
25
+ searches_with_match: z.number(),
26
+ bytes_searched: z.number(),
27
+ bytes_printed: z.number(),
28
+ matched_lines: z.number(),
29
+ matches: z.number(),
30
+ })
31
+
32
+ const Begin = z.object({
33
+ type: z.literal("begin"),
34
+ data: z.object({
35
+ path: z.object({
36
+ text: z.string(),
37
+ }),
38
+ }),
39
+ })
40
+
41
+ export const Match = z.object({
42
+ type: z.literal("match"),
43
+ data: z.object({
44
+ path: z.object({
45
+ text: z.string(),
46
+ }),
47
+ lines: z.object({
48
+ text: z.string(),
49
+ }),
50
+ line_number: z.number(),
51
+ absolute_offset: z.number(),
52
+ submatches: z.array(
53
+ z.object({
54
+ match: z.object({
55
+ text: z.string(),
56
+ }),
57
+ start: z.number(),
58
+ end: z.number(),
59
+ }),
60
+ ),
61
+ }),
62
+ })
63
+
64
+ const End = z.object({
65
+ type: z.literal("end"),
66
+ data: z.object({
67
+ path: z.object({
68
+ text: z.string(),
69
+ }),
70
+ binary_offset: z.number().nullable(),
71
+ stats: Stats,
72
+ }),
73
+ })
74
+
75
+ const Summary = z.object({
76
+ type: z.literal("summary"),
77
+ data: z.object({
78
+ elapsed_total: z.object({
79
+ human: z.string(),
80
+ nanos: z.number(),
81
+ secs: z.number(),
82
+ }),
83
+ stats: Stats,
84
+ }),
85
+ })
86
+
87
+ const Result = z.union([Begin, Match, End, Summary])
88
+
89
+ export type Result = z.infer<typeof Result>
90
+ export type Match = z.infer<typeof Match>
91
+ export type Begin = z.infer<typeof Begin>
92
+ export type End = z.infer<typeof End>
93
+ export type Summary = z.infer<typeof Summary>
94
+ const PLATFORM = {
95
+ "arm64-darwin": { platform: "aarch64-apple-darwin", extension: "tar.gz" },
96
+ "arm64-linux": {
97
+ platform: "aarch64-unknown-linux-gnu",
98
+ extension: "tar.gz",
99
+ },
100
+ "x64-darwin": { platform: "x86_64-apple-darwin", extension: "tar.gz" },
101
+ "x64-linux": { platform: "x86_64-unknown-linux-musl", extension: "tar.gz" },
102
+ "x64-win32": { platform: "x86_64-pc-windows-msvc", extension: "zip" },
103
+ } as const
104
+
105
+ export const ExtractionFailedError = NamedError.create(
106
+ "RipgrepExtractionFailedError",
107
+ z.object({
108
+ filepath: z.string(),
109
+ stderr: z.string(),
110
+ }),
111
+ )
112
+
113
+ export const UnsupportedPlatformError = NamedError.create(
114
+ "RipgrepUnsupportedPlatformError",
115
+ z.object({
116
+ platform: z.string(),
117
+ }),
118
+ )
119
+
120
+ export const DownloadFailedError = NamedError.create(
121
+ "RipgrepDownloadFailedError",
122
+ z.object({
123
+ url: z.string(),
124
+ status: z.number(),
125
+ }),
126
+ )
127
+
128
+ const state = lazy(async () => {
129
+ const system = Bun.which("rg")
130
+ if (system) {
131
+ const stat = await fs.stat(system).catch(() => undefined)
132
+ if (stat?.isFile()) return { filepath: system }
133
+ log.warn("bun.which returned invalid rg path", { filepath: system })
134
+ }
135
+ const filepath = path.join(Global.Path.bin, "rg" + (process.platform === "win32" ? ".exe" : ""))
136
+
137
+ if (!(await Filesystem.exists(filepath))) {
138
+ const platformKey = `${process.arch}-${process.platform}` as keyof typeof PLATFORM
139
+ const config = PLATFORM[platformKey]
140
+ if (!config) throw new UnsupportedPlatformError({ platform: platformKey })
141
+
142
+ const version = "14.1.1"
143
+ const filename = `ripgrep-${version}-${config.platform}.${config.extension}`
144
+ const url = `https://github.com/BurntSushi/ripgrep/releases/download/${version}/${filename}`
145
+
146
+ const response = await fetch(url)
147
+ if (!response.ok) throw new DownloadFailedError({ url, status: response.status })
148
+
149
+ const arrayBuffer = await response.arrayBuffer()
150
+ const archivePath = path.join(Global.Path.bin, filename)
151
+ await Filesystem.write(archivePath, Buffer.from(arrayBuffer))
152
+ if (config.extension === "tar.gz") {
153
+ const args = ["tar", "-xzf", archivePath, "--strip-components=1"]
154
+
155
+ if (platformKey.endsWith("-darwin")) args.push("--include=*/rg")
156
+ if (platformKey.endsWith("-linux")) args.push("--wildcards", "*/rg")
157
+
158
+ const proc = Process.spawn(args, {
159
+ cwd: Global.Path.bin,
160
+ stderr: "pipe",
161
+ stdout: "pipe",
162
+ })
163
+ const exit = await proc.exited
164
+ if (exit !== 0) {
165
+ const stderr = proc.stderr ? await text(proc.stderr) : ""
166
+ throw new ExtractionFailedError({
167
+ filepath,
168
+ stderr,
169
+ })
170
+ }
171
+ }
172
+ if (config.extension === "zip") {
173
+ const zipFileReader = new ZipReader(new BlobReader(new Blob([arrayBuffer])))
174
+ const entries = await zipFileReader.getEntries()
175
+ let rgEntry: any
176
+ for (const entry of entries) {
177
+ if (entry.filename.endsWith("rg.exe")) {
178
+ rgEntry = entry
179
+ break
180
+ }
181
+ }
182
+
183
+ if (!rgEntry) {
184
+ throw new ExtractionFailedError({
185
+ filepath: archivePath,
186
+ stderr: "rg.exe not found in zip archive",
187
+ })
188
+ }
189
+
190
+ const rgBlob = await rgEntry.getData(new BlobWriter())
191
+ if (!rgBlob) {
192
+ throw new ExtractionFailedError({
193
+ filepath: archivePath,
194
+ stderr: "Failed to extract rg.exe from zip archive",
195
+ })
196
+ }
197
+ await Filesystem.write(filepath, Buffer.from(await rgBlob.arrayBuffer()))
198
+ await zipFileReader.close()
199
+ }
200
+ await fs.unlink(archivePath)
201
+ if (!platformKey.endsWith("-win32")) await fs.chmod(filepath, 0o755)
202
+ }
203
+
204
+ return {
205
+ filepath,
206
+ }
207
+ })
208
+
209
+ export async function filepath() {
210
+ const { filepath } = await state()
211
+ return filepath
212
+ }
213
+
214
+ export async function* files(input: {
215
+ cwd: string
216
+ glob?: string[]
217
+ hidden?: boolean
218
+ follow?: boolean
219
+ maxDepth?: number
220
+ signal?: AbortSignal
221
+ }) {
222
+ input.signal?.throwIfAborted()
223
+
224
+ const args = [await filepath(), "--files", "--glob=!.git/*"]
225
+ if (input.follow) args.push("--follow")
226
+ if (input.hidden !== false) args.push("--hidden")
227
+ if (input.maxDepth !== undefined) args.push(`--max-depth=${input.maxDepth}`)
228
+ if (input.glob) {
229
+ for (const g of input.glob) {
230
+ args.push(`--glob=${g}`)
231
+ }
232
+ }
233
+
234
+ // Guard against invalid cwd to provide a consistent ENOENT error.
235
+ if (!(await fs.stat(input.cwd).catch(() => undefined))?.isDirectory()) {
236
+ throw Object.assign(new Error(`No such file or directory: '${input.cwd}'`), {
237
+ code: "ENOENT",
238
+ errno: -2,
239
+ path: input.cwd,
240
+ })
241
+ }
242
+
243
+ const proc = Process.spawn(args, {
244
+ cwd: input.cwd,
245
+ stdout: "pipe",
246
+ stderr: "ignore",
247
+ abort: input.signal,
248
+ })
249
+
250
+ if (!proc.stdout) {
251
+ throw new Error("Process output not available")
252
+ }
253
+
254
+ let buffer = ""
255
+ const stream = proc.stdout as AsyncIterable<Buffer | string>
256
+ for await (const chunk of stream) {
257
+ input.signal?.throwIfAborted()
258
+
259
+ buffer += typeof chunk === "string" ? chunk : chunk.toString()
260
+ // Handle both Unix (\n) and Windows (\r\n) line endings
261
+ const lines = buffer.split(/\r?\n/)
262
+ buffer = lines.pop() || ""
263
+
264
+ for (const line of lines) {
265
+ if (line) yield line
266
+ }
267
+ }
268
+
269
+ if (buffer) yield buffer
270
+ await proc.exited
271
+
272
+ input.signal?.throwIfAborted()
273
+ }
274
+
275
+ export async function tree(input: { cwd: string; limit?: number; signal?: AbortSignal }) {
276
+ log.info("tree", input)
277
+ const files = await Array.fromAsync(Ripgrep.files({ cwd: input.cwd, signal: input.signal }))
278
+ interface Node {
279
+ name: string
280
+ children: Map<string, Node>
281
+ }
282
+
283
+ function dir(node: Node, name: string) {
284
+ const existing = node.children.get(name)
285
+ if (existing) return existing
286
+ const next = { name, children: new Map() }
287
+ node.children.set(name, next)
288
+ return next
289
+ }
290
+
291
+ const root: Node = { name: "", children: new Map() }
292
+ for (const file of files) {
293
+ if (file.includes(".opencode")) continue
294
+ const parts = file.split(path.sep)
295
+ if (parts.length < 2) continue
296
+ let node = root
297
+ for (const part of parts.slice(0, -1)) {
298
+ node = dir(node, part)
299
+ }
300
+ }
301
+
302
+ function count(node: Node): number {
303
+ let total = 0
304
+ for (const child of node.children.values()) {
305
+ total += 1 + count(child)
306
+ }
307
+ return total
308
+ }
309
+
310
+ const total = count(root)
311
+ const limit = input.limit ?? total
312
+ const lines: string[] = []
313
+ const queue: { node: Node; path: string }[] = []
314
+ for (const child of Array.from(root.children.values()).sort((a, b) => a.name.localeCompare(b.name))) {
315
+ queue.push({ node: child, path: child.name })
316
+ }
317
+
318
+ let used = 0
319
+ for (let i = 0; i < queue.length && used < limit; i++) {
320
+ const { node, path } = queue[i]
321
+ lines.push(path)
322
+ used++
323
+ for (const child of Array.from(node.children.values()).sort((a, b) => a.name.localeCompare(b.name))) {
324
+ queue.push({ node: child, path: `${path}/${child.name}` })
325
+ }
326
+ }
327
+
328
+ if (total > used) lines.push(`[${total - used} truncated]`)
329
+
330
+ return lines.join("\n")
331
+ }
332
+
333
+ export async function search(input: {
334
+ cwd: string
335
+ pattern: string
336
+ glob?: string[]
337
+ limit?: number
338
+ follow?: boolean
339
+ }) {
340
+ const args = [`${await filepath()}`, "--json", "--hidden", "--glob='!.git/*'"]
341
+ if (input.follow) args.push("--follow")
342
+
343
+ if (input.glob) {
344
+ for (const g of input.glob) {
345
+ args.push(`--glob=${g}`)
346
+ }
347
+ }
348
+
349
+ if (input.limit) {
350
+ args.push(`--max-count=${input.limit}`)
351
+ }
352
+
353
+ args.push("--")
354
+ args.push(input.pattern)
355
+
356
+ const command = args.join(" ")
357
+ const result = await $`${{ raw: command }}`.cwd(input.cwd).quiet().nothrow()
358
+ if (result.exitCode !== 0) {
359
+ return []
360
+ }
361
+
362
+ // Handle both Unix (\n) and Windows (\r\n) line endings
363
+ const lines = result.text().trim().split(/\r?\n/).filter(Boolean)
364
+ // Parse JSON lines from ripgrep output
365
+
366
+ return lines
367
+ .map((line) => JSON.parse(line))
368
+ .map((parsed) => Result.parse(parsed))
369
+ .filter((r) => r.type === "match")
370
+ .map((r) => r.data)
371
+ }
372
+ }
@@ -0,0 +1,71 @@
1
+ import { Instance } from "../project/instance"
2
+ import { Log } from "../util/log"
3
+ import { Flag } from "../flag/flag"
4
+ import { Filesystem } from "../util/filesystem"
5
+
6
+ export namespace FileTime {
7
+ const log = Log.create({ service: "file.time" })
8
+ // Per-session read times plus per-file write locks.
9
+ // All tools that overwrite existing files should run their
10
+ // assert/read/write/update sequence inside withLock(filepath, ...)
11
+ // so concurrent writes to the same file are serialized.
12
+ export const state = Instance.state(() => {
13
+ const read: {
14
+ [sessionID: string]: {
15
+ [path: string]: Date | undefined
16
+ }
17
+ } = {}
18
+ const locks = new Map<string, Promise<void>>()
19
+ return {
20
+ read,
21
+ locks,
22
+ }
23
+ })
24
+
25
+ export function read(sessionID: string, file: string) {
26
+ log.info("read", { sessionID, file })
27
+ const { read } = state()
28
+ read[sessionID] = read[sessionID] || {}
29
+ read[sessionID][file] = new Date()
30
+ }
31
+
32
+ export function get(sessionID: string, file: string) {
33
+ return state().read[sessionID]?.[file]
34
+ }
35
+
36
+ export async function withLock<T>(filepath: string, fn: () => Promise<T>): Promise<T> {
37
+ const current = state()
38
+ const currentLock = current.locks.get(filepath) ?? Promise.resolve()
39
+ let release: () => void = () => {}
40
+ const nextLock = new Promise<void>((resolve) => {
41
+ release = resolve
42
+ })
43
+ const chained = currentLock.then(() => nextLock)
44
+ current.locks.set(filepath, chained)
45
+ await currentLock
46
+ try {
47
+ return await fn()
48
+ } finally {
49
+ release()
50
+ if (current.locks.get(filepath) === chained) {
51
+ current.locks.delete(filepath)
52
+ }
53
+ }
54
+ }
55
+
56
+ export async function assert(sessionID: string, filepath: string) {
57
+ if (Flag.OPENCODE_DISABLE_FILETIME_CHECK === true) {
58
+ return
59
+ }
60
+
61
+ const time = get(sessionID, filepath)
62
+ if (!time) throw new Error(`You must read file ${filepath} before overwriting it. Use the Read tool first`)
63
+ const mtime = Filesystem.stat(filepath)?.mtime
64
+ // Allow a 50ms tolerance for Windows NTFS timestamp fuzziness / async flushing
65
+ if (mtime && mtime.getTime() > time.getTime() + 50) {
66
+ throw new Error(
67
+ `File ${filepath} has been modified since it was last read.\nLast modification: ${mtime.toISOString()}\nLast read: ${time.toISOString()}\n\nPlease read the file again before modifying it.`,
68
+ )
69
+ }
70
+ }
71
+ }
@@ -0,0 +1,128 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { Bus } from "@/bus"
3
+ import z from "zod"
4
+ import { Instance } from "../project/instance"
5
+ import { Log } from "../util/log"
6
+ import { FileIgnore } from "./ignore"
7
+ import { Config } from "../config/config"
8
+ import path from "path"
9
+ // @ts-ignore
10
+ import { createWrapper } from "@parcel/watcher/wrapper"
11
+ import { lazy } from "@/util/lazy"
12
+ import { withTimeout } from "@/util/timeout"
13
+ import type ParcelWatcher from "@parcel/watcher"
14
+ import { $ } from "bun"
15
+ import { Flag } from "@/flag/flag"
16
+ import { readdir } from "fs/promises"
17
+
18
+ const SUBSCRIBE_TIMEOUT_MS = 10_000
19
+
20
+ declare const OPENCODE_LIBC: string | undefined
21
+
22
+ export namespace FileWatcher {
23
+ const log = Log.create({ service: "file.watcher" })
24
+
25
+ export const Event = {
26
+ Updated: BusEvent.define(
27
+ "file.watcher.updated",
28
+ z.object({
29
+ file: z.string(),
30
+ event: z.union([z.literal("add"), z.literal("change"), z.literal("unlink")]),
31
+ }),
32
+ ),
33
+ }
34
+
35
+ const watcher = lazy((): typeof import("@parcel/watcher") | undefined => {
36
+ try {
37
+ const binding = require(
38
+ `@parcel/watcher-${process.platform}-${process.arch}${process.platform === "linux" ? `-${OPENCODE_LIBC || "glibc"}` : ""}`,
39
+ )
40
+ return createWrapper(binding) as typeof import("@parcel/watcher")
41
+ } catch (error) {
42
+ log.error("failed to load watcher binding", { error })
43
+ return
44
+ }
45
+ })
46
+
47
+ const state = Instance.state(
48
+ async () => {
49
+ log.info("init")
50
+ const cfg = await Config.get()
51
+ const backend = (() => {
52
+ if (process.platform === "win32") return "windows"
53
+ if (process.platform === "darwin") return "fs-events"
54
+ if (process.platform === "linux") return "inotify"
55
+ })()
56
+ if (!backend) {
57
+ log.error("watcher backend not supported", { platform: process.platform })
58
+ return {}
59
+ }
60
+ log.info("watcher backend", { platform: process.platform, backend })
61
+
62
+ const w = watcher()
63
+ if (!w) return {}
64
+
65
+ const subscribe: ParcelWatcher.SubscribeCallback = (err, evts) => {
66
+ if (err) return
67
+ for (const evt of evts) {
68
+ if (evt.type === "create") Bus.publish(Event.Updated, { file: evt.path, event: "add" })
69
+ if (evt.type === "update") Bus.publish(Event.Updated, { file: evt.path, event: "change" })
70
+ if (evt.type === "delete") Bus.publish(Event.Updated, { file: evt.path, event: "unlink" })
71
+ }
72
+ }
73
+
74
+ const subs: ParcelWatcher.AsyncSubscription[] = []
75
+ const cfgIgnores = cfg.watcher?.ignore ?? []
76
+
77
+ if (Flag.OPENCODE_EXPERIMENTAL_FILEWATCHER) {
78
+ const pending = w.subscribe(Instance.directory, subscribe, {
79
+ ignore: [...FileIgnore.PATTERNS, ...cfgIgnores],
80
+ backend,
81
+ })
82
+ const sub = await withTimeout(pending, SUBSCRIBE_TIMEOUT_MS).catch((err) => {
83
+ log.error("failed to subscribe to Instance.directory", { error: err })
84
+ pending.then((s) => s.unsubscribe()).catch(() => {})
85
+ return undefined
86
+ })
87
+ if (sub) subs.push(sub)
88
+ }
89
+
90
+ if (Instance.project.vcs === "git") {
91
+ const vcsDir = await $`git rev-parse --git-dir`
92
+ .quiet()
93
+ .nothrow()
94
+ .cwd(Instance.worktree)
95
+ .text()
96
+ .then((x) => path.resolve(Instance.worktree, x.trim()))
97
+ .catch(() => undefined)
98
+ if (vcsDir && !cfgIgnores.includes(".git") && !cfgIgnores.includes(vcsDir)) {
99
+ const gitDirContents = await readdir(vcsDir).catch(() => [])
100
+ const ignoreList = gitDirContents.filter((entry) => entry !== "HEAD")
101
+ const pending = w.subscribe(vcsDir, subscribe, {
102
+ ignore: ignoreList,
103
+ backend,
104
+ })
105
+ const sub = await withTimeout(pending, SUBSCRIBE_TIMEOUT_MS).catch((err) => {
106
+ log.error("failed to subscribe to vcsDir", { error: err })
107
+ pending.then((s) => s.unsubscribe()).catch(() => {})
108
+ return undefined
109
+ })
110
+ if (sub) subs.push(sub)
111
+ }
112
+ }
113
+
114
+ return { subs }
115
+ },
116
+ async (state) => {
117
+ if (!state.subs) return
118
+ await Promise.all(state.subs.map((sub) => sub?.unsubscribe()))
119
+ },
120
+ )
121
+
122
+ export function init() {
123
+ if (Flag.OPENCODE_EXPERIMENTAL_DISABLE_FILEWATCHER) {
124
+ return
125
+ }
126
+ state()
127
+ }
128
+ }
@@ -0,0 +1,109 @@
1
+ function truthy(key: string) {
2
+ const value = process.env[key]?.toLowerCase()
3
+ return value === "true" || value === "1"
4
+ }
5
+
6
+ export namespace Flag {
7
+ export const OPENCODE_AUTO_SHARE = truthy("OPENCODE_AUTO_SHARE")
8
+ export const OPENCODE_GIT_BASH_PATH = process.env["OPENCODE_GIT_BASH_PATH"]
9
+ export const OPENCODE_CONFIG = process.env["OPENCODE_CONFIG"]
10
+ export declare const OPENCODE_TUI_CONFIG: string | undefined
11
+ export declare const OPENCODE_CONFIG_DIR: string | undefined
12
+ export const OPENCODE_CONFIG_CONTENT = process.env["OPENCODE_CONFIG_CONTENT"]
13
+ export const OPENCODE_DISABLE_AUTOUPDATE = truthy("OPENCODE_DISABLE_AUTOUPDATE")
14
+ export const OPENCODE_DISABLE_PRUNE = truthy("OPENCODE_DISABLE_PRUNE")
15
+ export const OPENCODE_DISABLE_TERMINAL_TITLE = truthy("OPENCODE_DISABLE_TERMINAL_TITLE")
16
+ export const OPENCODE_PERMISSION = process.env["OPENCODE_PERMISSION"]
17
+ export const OPENCODE_DISABLE_DEFAULT_PLUGINS = truthy("OPENCODE_DISABLE_DEFAULT_PLUGINS")
18
+ export const OPENCODE_DISABLE_LSP_DOWNLOAD = truthy("OPENCODE_DISABLE_LSP_DOWNLOAD")
19
+ export const OPENCODE_ENABLE_EXPERIMENTAL_MODELS = truthy("OPENCODE_ENABLE_EXPERIMENTAL_MODELS")
20
+ export const OPENCODE_DISABLE_AUTOCOMPACT = truthy("OPENCODE_DISABLE_AUTOCOMPACT")
21
+ export const OPENCODE_DISABLE_MODELS_FETCH = truthy("OPENCODE_DISABLE_MODELS_FETCH")
22
+ export const OPENCODE_DISABLE_CLAUDE_CODE = truthy("OPENCODE_DISABLE_CLAUDE_CODE")
23
+ export const OPENCODE_DISABLE_CLAUDE_CODE_PROMPT =
24
+ OPENCODE_DISABLE_CLAUDE_CODE || truthy("OPENCODE_DISABLE_CLAUDE_CODE_PROMPT")
25
+ export const OPENCODE_DISABLE_CLAUDE_CODE_SKILLS =
26
+ OPENCODE_DISABLE_CLAUDE_CODE || truthy("OPENCODE_DISABLE_CLAUDE_CODE_SKILLS")
27
+ export const OPENCODE_DISABLE_EXTERNAL_SKILLS =
28
+ OPENCODE_DISABLE_CLAUDE_CODE_SKILLS || truthy("OPENCODE_DISABLE_EXTERNAL_SKILLS")
29
+ export declare const OPENCODE_DISABLE_PROJECT_CONFIG: boolean
30
+ export const OPENCODE_FAKE_VCS = process.env["OPENCODE_FAKE_VCS"]
31
+ export declare const OPENCODE_CLIENT: string
32
+ export const OPENCODE_SERVER_PASSWORD = process.env["OPENCODE_SERVER_PASSWORD"]
33
+ export const OPENCODE_SERVER_USERNAME = process.env["OPENCODE_SERVER_USERNAME"]
34
+ export const OPENCODE_ENABLE_QUESTION_TOOL = truthy("OPENCODE_ENABLE_QUESTION_TOOL")
35
+
36
+ // Experimental
37
+ export const OPENCODE_EXPERIMENTAL = truthy("OPENCODE_EXPERIMENTAL")
38
+ export const OPENCODE_EXPERIMENTAL_FILEWATCHER = truthy("OPENCODE_EXPERIMENTAL_FILEWATCHER")
39
+ export const OPENCODE_EXPERIMENTAL_DISABLE_FILEWATCHER = truthy("OPENCODE_EXPERIMENTAL_DISABLE_FILEWATCHER")
40
+ export const OPENCODE_EXPERIMENTAL_ICON_DISCOVERY =
41
+ OPENCODE_EXPERIMENTAL || truthy("OPENCODE_EXPERIMENTAL_ICON_DISCOVERY")
42
+
43
+ const copy = process.env["OPENCODE_EXPERIMENTAL_DISABLE_COPY_ON_SELECT"]
44
+ export const OPENCODE_EXPERIMENTAL_DISABLE_COPY_ON_SELECT =
45
+ copy === undefined ? process.platform === "win32" : truthy("OPENCODE_EXPERIMENTAL_DISABLE_COPY_ON_SELECT")
46
+ export const OPENCODE_ENABLE_EXA =
47
+ truthy("OPENCODE_ENABLE_EXA") || OPENCODE_EXPERIMENTAL || truthy("OPENCODE_EXPERIMENTAL_EXA")
48
+ export const OPENCODE_EXPERIMENTAL_BASH_DEFAULT_TIMEOUT_MS = number("OPENCODE_EXPERIMENTAL_BASH_DEFAULT_TIMEOUT_MS")
49
+ export const OPENCODE_EXPERIMENTAL_OUTPUT_TOKEN_MAX = number("OPENCODE_EXPERIMENTAL_OUTPUT_TOKEN_MAX")
50
+ export const OPENCODE_EXPERIMENTAL_OXFMT = OPENCODE_EXPERIMENTAL || truthy("OPENCODE_EXPERIMENTAL_OXFMT")
51
+ export const OPENCODE_EXPERIMENTAL_LSP_TY = truthy("OPENCODE_EXPERIMENTAL_LSP_TY")
52
+ export const OPENCODE_EXPERIMENTAL_LSP_TOOL = OPENCODE_EXPERIMENTAL || truthy("OPENCODE_EXPERIMENTAL_LSP_TOOL")
53
+ export const OPENCODE_DISABLE_FILETIME_CHECK = truthy("OPENCODE_DISABLE_FILETIME_CHECK")
54
+ export const OPENCODE_EXPERIMENTAL_PLAN_MODE = OPENCODE_EXPERIMENTAL || truthy("OPENCODE_EXPERIMENTAL_PLAN_MODE")
55
+ export const OPENCODE_EXPERIMENTAL_MARKDOWN = truthy("OPENCODE_EXPERIMENTAL_MARKDOWN")
56
+ export const OPENCODE_MODELS_URL = process.env["OPENCODE_MODELS_URL"]
57
+ export const OPENCODE_MODELS_PATH = process.env["OPENCODE_MODELS_PATH"]
58
+
59
+ function number(key: string) {
60
+ const value = process.env[key]
61
+ if (!value) return undefined
62
+ const parsed = Number(value)
63
+ return Number.isInteger(parsed) && parsed > 0 ? parsed : undefined
64
+ }
65
+ }
66
+
67
+ // Dynamic getter for OPENCODE_DISABLE_PROJECT_CONFIG
68
+ // This must be evaluated at access time, not module load time,
69
+ // because external tooling may set this env var at runtime
70
+ Object.defineProperty(Flag, "OPENCODE_DISABLE_PROJECT_CONFIG", {
71
+ get() {
72
+ return truthy("OPENCODE_DISABLE_PROJECT_CONFIG")
73
+ },
74
+ enumerable: true,
75
+ configurable: false,
76
+ })
77
+
78
+ // Dynamic getter for OPENCODE_TUI_CONFIG
79
+ // This must be evaluated at access time, not module load time,
80
+ // because tests and external tooling may set this env var at runtime
81
+ Object.defineProperty(Flag, "OPENCODE_TUI_CONFIG", {
82
+ get() {
83
+ return process.env["OPENCODE_TUI_CONFIG"]
84
+ },
85
+ enumerable: true,
86
+ configurable: false,
87
+ })
88
+
89
+ // Dynamic getter for OPENCODE_CONFIG_DIR
90
+ // This must be evaluated at access time, not module load time,
91
+ // because external tooling may set this env var at runtime
92
+ Object.defineProperty(Flag, "OPENCODE_CONFIG_DIR", {
93
+ get() {
94
+ return process.env["OPENCODE_CONFIG_DIR"]
95
+ },
96
+ enumerable: true,
97
+ configurable: false,
98
+ })
99
+
100
+ // Dynamic getter for OPENCODE_CLIENT
101
+ // This must be evaluated at access time, not module load time,
102
+ // because some commands override the client at runtime
103
+ Object.defineProperty(Flag, "OPENCODE_CLIENT", {
104
+ get() {
105
+ return process.env["OPENCODE_CLIENT"] ?? "cli"
106
+ },
107
+ enumerable: true,
108
+ configurable: false,
109
+ })