@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,643 @@
1
+ import { $ } from "bun"
2
+ import fs from "fs/promises"
3
+ import path from "path"
4
+ import z from "zod"
5
+ import { NamedError } from "@opencode-ai/util/error"
6
+ import { Global } from "../global"
7
+ import { Instance } from "../project/instance"
8
+ import { InstanceBootstrap } from "../project/bootstrap"
9
+ import { Project } from "../project/project"
10
+ import { Database, eq } from "../storage/db"
11
+ import { ProjectTable } from "../project/project.sql"
12
+ import { fn } from "../util/fn"
13
+ import { Log } from "../util/log"
14
+ import { BusEvent } from "@/bus/bus-event"
15
+ import { GlobalBus } from "@/bus/global"
16
+
17
+ export namespace Worktree {
18
+ const log = Log.create({ service: "worktree" })
19
+
20
+ export const Event = {
21
+ Ready: BusEvent.define(
22
+ "worktree.ready",
23
+ z.object({
24
+ name: z.string(),
25
+ branch: z.string(),
26
+ }),
27
+ ),
28
+ Failed: BusEvent.define(
29
+ "worktree.failed",
30
+ z.object({
31
+ message: z.string(),
32
+ }),
33
+ ),
34
+ }
35
+
36
+ export const Info = z
37
+ .object({
38
+ name: z.string(),
39
+ branch: z.string(),
40
+ directory: z.string(),
41
+ })
42
+ .meta({
43
+ ref: "Worktree",
44
+ })
45
+
46
+ export type Info = z.infer<typeof Info>
47
+
48
+ export const CreateInput = z
49
+ .object({
50
+ name: z.string().optional(),
51
+ startCommand: z
52
+ .string()
53
+ .optional()
54
+ .describe("Additional startup script to run after the project's start command"),
55
+ })
56
+ .meta({
57
+ ref: "WorktreeCreateInput",
58
+ })
59
+
60
+ export type CreateInput = z.infer<typeof CreateInput>
61
+
62
+ export const RemoveInput = z
63
+ .object({
64
+ directory: z.string(),
65
+ })
66
+ .meta({
67
+ ref: "WorktreeRemoveInput",
68
+ })
69
+
70
+ export type RemoveInput = z.infer<typeof RemoveInput>
71
+
72
+ export const ResetInput = z
73
+ .object({
74
+ directory: z.string(),
75
+ })
76
+ .meta({
77
+ ref: "WorktreeResetInput",
78
+ })
79
+
80
+ export type ResetInput = z.infer<typeof ResetInput>
81
+
82
+ export const NotGitError = NamedError.create(
83
+ "WorktreeNotGitError",
84
+ z.object({
85
+ message: z.string(),
86
+ }),
87
+ )
88
+
89
+ export const NameGenerationFailedError = NamedError.create(
90
+ "WorktreeNameGenerationFailedError",
91
+ z.object({
92
+ message: z.string(),
93
+ }),
94
+ )
95
+
96
+ export const CreateFailedError = NamedError.create(
97
+ "WorktreeCreateFailedError",
98
+ z.object({
99
+ message: z.string(),
100
+ }),
101
+ )
102
+
103
+ export const StartCommandFailedError = NamedError.create(
104
+ "WorktreeStartCommandFailedError",
105
+ z.object({
106
+ message: z.string(),
107
+ }),
108
+ )
109
+
110
+ export const RemoveFailedError = NamedError.create(
111
+ "WorktreeRemoveFailedError",
112
+ z.object({
113
+ message: z.string(),
114
+ }),
115
+ )
116
+
117
+ export const ResetFailedError = NamedError.create(
118
+ "WorktreeResetFailedError",
119
+ z.object({
120
+ message: z.string(),
121
+ }),
122
+ )
123
+
124
+ const ADJECTIVES = [
125
+ "brave",
126
+ "calm",
127
+ "clever",
128
+ "cosmic",
129
+ "crisp",
130
+ "curious",
131
+ "eager",
132
+ "gentle",
133
+ "glowing",
134
+ "happy",
135
+ "hidden",
136
+ "jolly",
137
+ "kind",
138
+ "lucky",
139
+ "mighty",
140
+ "misty",
141
+ "neon",
142
+ "nimble",
143
+ "playful",
144
+ "proud",
145
+ "quick",
146
+ "quiet",
147
+ "shiny",
148
+ "silent",
149
+ "stellar",
150
+ "sunny",
151
+ "swift",
152
+ "tidy",
153
+ "witty",
154
+ ] as const
155
+
156
+ const NOUNS = [
157
+ "cabin",
158
+ "cactus",
159
+ "canyon",
160
+ "circuit",
161
+ "comet",
162
+ "eagle",
163
+ "engine",
164
+ "falcon",
165
+ "forest",
166
+ "garden",
167
+ "harbor",
168
+ "island",
169
+ "knight",
170
+ "lagoon",
171
+ "meadow",
172
+ "moon",
173
+ "mountain",
174
+ "nebula",
175
+ "orchid",
176
+ "otter",
177
+ "panda",
178
+ "pixel",
179
+ "planet",
180
+ "river",
181
+ "rocket",
182
+ "sailor",
183
+ "squid",
184
+ "star",
185
+ "tiger",
186
+ "wizard",
187
+ "wolf",
188
+ ] as const
189
+
190
+ function pick<const T extends readonly string[]>(list: T) {
191
+ return list[Math.floor(Math.random() * list.length)]
192
+ }
193
+
194
+ function slug(input: string) {
195
+ return input
196
+ .trim()
197
+ .toLowerCase()
198
+ .replace(/[^a-z0-9]+/g, "-")
199
+ .replace(/^-+/, "")
200
+ .replace(/-+$/, "")
201
+ }
202
+
203
+ function randomName() {
204
+ return `${pick(ADJECTIVES)}-${pick(NOUNS)}`
205
+ }
206
+
207
+ async function exists(target: string) {
208
+ return fs
209
+ .stat(target)
210
+ .then(() => true)
211
+ .catch(() => false)
212
+ }
213
+
214
+ function outputText(input: Uint8Array | undefined) {
215
+ if (!input?.length) return ""
216
+ return new TextDecoder().decode(input).trim()
217
+ }
218
+
219
+ function errorText(result: { stdout?: Uint8Array; stderr?: Uint8Array }) {
220
+ return [outputText(result.stderr), outputText(result.stdout)].filter(Boolean).join("\n")
221
+ }
222
+
223
+ function failed(result: { stdout?: Uint8Array; stderr?: Uint8Array }) {
224
+ return [outputText(result.stderr), outputText(result.stdout)].filter(Boolean).flatMap((chunk) =>
225
+ chunk
226
+ .split("\n")
227
+ .map((line) => line.trim())
228
+ .flatMap((line) => {
229
+ const match = line.match(/^warning:\s+failed to remove\s+(.+):\s+/i)
230
+ if (!match) return []
231
+ const value = match[1]?.trim().replace(/^['"]|['"]$/g, "")
232
+ if (!value) return []
233
+ return [value]
234
+ }),
235
+ )
236
+ }
237
+
238
+ async function prune(root: string, entries: string[]) {
239
+ const base = await canonical(root)
240
+ await Promise.all(
241
+ entries.map(async (entry) => {
242
+ const target = await canonical(path.resolve(root, entry))
243
+ if (target === base) return
244
+ if (!target.startsWith(`${base}${path.sep}`)) return
245
+ await fs.rm(target, { recursive: true, force: true }).catch(() => undefined)
246
+ }),
247
+ )
248
+ }
249
+
250
+ async function sweep(root: string) {
251
+ const first = await $`git clean -ffdx`.quiet().nothrow().cwd(root)
252
+ if (first.exitCode === 0) return first
253
+
254
+ const entries = failed(first)
255
+ if (!entries.length) return first
256
+
257
+ await prune(root, entries)
258
+ return $`git clean -ffdx`.quiet().nothrow().cwd(root)
259
+ }
260
+
261
+ async function canonical(input: string) {
262
+ const abs = path.resolve(input)
263
+ const real = await fs.realpath(abs).catch(() => abs)
264
+ const normalized = path.normalize(real)
265
+ return process.platform === "win32" ? normalized.toLowerCase() : normalized
266
+ }
267
+
268
+ async function candidate(root: string, base?: string) {
269
+ for (const attempt of Array.from({ length: 26 }, (_, i) => i)) {
270
+ const name = base ? (attempt === 0 ? base : `${base}-${randomName()}`) : randomName()
271
+ const branch = `opencode/${name}`
272
+ const directory = path.join(root, name)
273
+
274
+ if (await exists(directory)) continue
275
+
276
+ const ref = `refs/heads/${branch}`
277
+ const branchCheck = await $`git show-ref --verify --quiet ${ref}`.quiet().nothrow().cwd(Instance.worktree)
278
+ if (branchCheck.exitCode === 0) continue
279
+
280
+ return Info.parse({ name, branch, directory })
281
+ }
282
+
283
+ throw new NameGenerationFailedError({ message: "Failed to generate a unique worktree name" })
284
+ }
285
+
286
+ async function runStartCommand(directory: string, cmd: string) {
287
+ if (process.platform === "win32") {
288
+ return $`cmd /c ${cmd}`.nothrow().cwd(directory)
289
+ }
290
+ return $`bash -lc ${cmd}`.nothrow().cwd(directory)
291
+ }
292
+
293
+ type StartKind = "project" | "worktree"
294
+
295
+ async function runStartScript(directory: string, cmd: string, kind: StartKind) {
296
+ const text = cmd.trim()
297
+ if (!text) return true
298
+
299
+ const ran = await runStartCommand(directory, text)
300
+ if (ran.exitCode === 0) return true
301
+
302
+ log.error("worktree start command failed", {
303
+ kind,
304
+ directory,
305
+ message: errorText(ran),
306
+ })
307
+ return false
308
+ }
309
+
310
+ async function runStartScripts(directory: string, input: { projectID: string; extra?: string }) {
311
+ const row = Database.use((db) => db.select().from(ProjectTable).where(eq(ProjectTable.id, input.projectID)).get())
312
+ const project = row ? Project.fromRow(row) : undefined
313
+ const startup = project?.commands?.start?.trim() ?? ""
314
+ const ok = await runStartScript(directory, startup, "project")
315
+ if (!ok) return false
316
+
317
+ const extra = input.extra ?? ""
318
+ await runStartScript(directory, extra, "worktree")
319
+ return true
320
+ }
321
+
322
+ function queueStartScripts(directory: string, input: { projectID: string; extra?: string }) {
323
+ setTimeout(() => {
324
+ const start = async () => {
325
+ await runStartScripts(directory, input)
326
+ }
327
+
328
+ void start().catch((error) => {
329
+ log.error("worktree start task failed", { directory, error })
330
+ })
331
+ }, 0)
332
+ }
333
+
334
+ export const create = fn(CreateInput.optional(), async (input) => {
335
+ if (Instance.project.vcs !== "git") {
336
+ throw new NotGitError({ message: "Worktrees are only supported for git projects" })
337
+ }
338
+
339
+ const root = path.join(Global.Path.data, "worktree", Instance.project.id)
340
+ await fs.mkdir(root, { recursive: true })
341
+
342
+ const base = input?.name ? slug(input.name) : ""
343
+ const info = await candidate(root, base || undefined)
344
+
345
+ const created = await $`git worktree add --no-checkout -b ${info.branch} ${info.directory}`
346
+ .quiet()
347
+ .nothrow()
348
+ .cwd(Instance.worktree)
349
+ if (created.exitCode !== 0) {
350
+ throw new CreateFailedError({ message: errorText(created) || "Failed to create git worktree" })
351
+ }
352
+
353
+ await Project.addSandbox(Instance.project.id, info.directory).catch(() => undefined)
354
+
355
+ const projectID = Instance.project.id
356
+ const extra = input?.startCommand?.trim()
357
+ setTimeout(() => {
358
+ const start = async () => {
359
+ const populated = await $`git reset --hard`.quiet().nothrow().cwd(info.directory)
360
+ if (populated.exitCode !== 0) {
361
+ const message = errorText(populated) || "Failed to populate worktree"
362
+ log.error("worktree checkout failed", { directory: info.directory, message })
363
+ GlobalBus.emit("event", {
364
+ directory: info.directory,
365
+ payload: {
366
+ type: Event.Failed.type,
367
+ properties: {
368
+ message,
369
+ },
370
+ },
371
+ })
372
+ return
373
+ }
374
+
375
+ const booted = await Instance.provide({
376
+ directory: info.directory,
377
+ init: InstanceBootstrap,
378
+ fn: () => undefined,
379
+ })
380
+ .then(() => true)
381
+ .catch((error) => {
382
+ const message = error instanceof Error ? error.message : String(error)
383
+ log.error("worktree bootstrap failed", { directory: info.directory, message })
384
+ GlobalBus.emit("event", {
385
+ directory: info.directory,
386
+ payload: {
387
+ type: Event.Failed.type,
388
+ properties: {
389
+ message,
390
+ },
391
+ },
392
+ })
393
+ return false
394
+ })
395
+ if (!booted) return
396
+
397
+ GlobalBus.emit("event", {
398
+ directory: info.directory,
399
+ payload: {
400
+ type: Event.Ready.type,
401
+ properties: {
402
+ name: info.name,
403
+ branch: info.branch,
404
+ },
405
+ },
406
+ })
407
+
408
+ await runStartScripts(info.directory, { projectID, extra })
409
+ }
410
+
411
+ void start().catch((error) => {
412
+ log.error("worktree start task failed", { directory: info.directory, error })
413
+ })
414
+ }, 0)
415
+
416
+ return info
417
+ })
418
+
419
+ export const remove = fn(RemoveInput, async (input) => {
420
+ if (Instance.project.vcs !== "git") {
421
+ throw new NotGitError({ message: "Worktrees are only supported for git projects" })
422
+ }
423
+
424
+ const directory = await canonical(input.directory)
425
+ const locate = async (stdout: Uint8Array | undefined) => {
426
+ const lines = outputText(stdout)
427
+ .split("\n")
428
+ .map((line) => line.trim())
429
+ const entries = lines.reduce<{ path?: string; branch?: string }[]>((acc, line) => {
430
+ if (!line) return acc
431
+ if (line.startsWith("worktree ")) {
432
+ acc.push({ path: line.slice("worktree ".length).trim() })
433
+ return acc
434
+ }
435
+ const current = acc[acc.length - 1]
436
+ if (!current) return acc
437
+ if (line.startsWith("branch ")) {
438
+ current.branch = line.slice("branch ".length).trim()
439
+ }
440
+ return acc
441
+ }, [])
442
+
443
+ return (async () => {
444
+ for (const item of entries) {
445
+ if (!item.path) continue
446
+ const key = await canonical(item.path)
447
+ if (key === directory) return item
448
+ }
449
+ })()
450
+ }
451
+
452
+ const clean = (target: string) =>
453
+ fs
454
+ .rm(target, {
455
+ recursive: true,
456
+ force: true,
457
+ maxRetries: 5,
458
+ retryDelay: 100,
459
+ })
460
+ .catch((error) => {
461
+ const message = error instanceof Error ? error.message : String(error)
462
+ throw new RemoveFailedError({ message: message || "Failed to remove git worktree directory" })
463
+ })
464
+
465
+ const list = await $`git worktree list --porcelain`.quiet().nothrow().cwd(Instance.worktree)
466
+ if (list.exitCode !== 0) {
467
+ throw new RemoveFailedError({ message: errorText(list) || "Failed to read git worktrees" })
468
+ }
469
+
470
+ const entry = await locate(list.stdout)
471
+
472
+ if (!entry?.path) {
473
+ const directoryExists = await exists(directory)
474
+ if (directoryExists) {
475
+ await clean(directory)
476
+ }
477
+ return true
478
+ }
479
+
480
+ const removed = await $`git worktree remove --force ${entry.path}`.quiet().nothrow().cwd(Instance.worktree)
481
+ if (removed.exitCode !== 0) {
482
+ const next = await $`git worktree list --porcelain`.quiet().nothrow().cwd(Instance.worktree)
483
+ if (next.exitCode !== 0) {
484
+ throw new RemoveFailedError({
485
+ message: errorText(removed) || errorText(next) || "Failed to remove git worktree",
486
+ })
487
+ }
488
+
489
+ const stale = await locate(next.stdout)
490
+ if (stale?.path) {
491
+ throw new RemoveFailedError({ message: errorText(removed) || "Failed to remove git worktree" })
492
+ }
493
+ }
494
+
495
+ await clean(entry.path)
496
+
497
+ const branch = entry.branch?.replace(/^refs\/heads\//, "")
498
+ if (branch) {
499
+ const deleted = await $`git branch -D ${branch}`.quiet().nothrow().cwd(Instance.worktree)
500
+ if (deleted.exitCode !== 0) {
501
+ throw new RemoveFailedError({ message: errorText(deleted) || "Failed to delete worktree branch" })
502
+ }
503
+ }
504
+
505
+ return true
506
+ })
507
+
508
+ export const reset = fn(ResetInput, async (input) => {
509
+ if (Instance.project.vcs !== "git") {
510
+ throw new NotGitError({ message: "Worktrees are only supported for git projects" })
511
+ }
512
+
513
+ const directory = await canonical(input.directory)
514
+ const primary = await canonical(Instance.worktree)
515
+ if (directory === primary) {
516
+ throw new ResetFailedError({ message: "Cannot reset the primary workspace" })
517
+ }
518
+
519
+ const list = await $`git worktree list --porcelain`.quiet().nothrow().cwd(Instance.worktree)
520
+ if (list.exitCode !== 0) {
521
+ throw new ResetFailedError({ message: errorText(list) || "Failed to read git worktrees" })
522
+ }
523
+
524
+ const lines = outputText(list.stdout)
525
+ .split("\n")
526
+ .map((line) => line.trim())
527
+ const entries = lines.reduce<{ path?: string; branch?: string }[]>((acc, line) => {
528
+ if (!line) return acc
529
+ if (line.startsWith("worktree ")) {
530
+ acc.push({ path: line.slice("worktree ".length).trim() })
531
+ return acc
532
+ }
533
+ const current = acc[acc.length - 1]
534
+ if (!current) return acc
535
+ if (line.startsWith("branch ")) {
536
+ current.branch = line.slice("branch ".length).trim()
537
+ }
538
+ return acc
539
+ }, [])
540
+
541
+ const entry = await (async () => {
542
+ for (const item of entries) {
543
+ if (!item.path) continue
544
+ const key = await canonical(item.path)
545
+ if (key === directory) return item
546
+ }
547
+ })()
548
+ if (!entry?.path) {
549
+ throw new ResetFailedError({ message: "Worktree not found" })
550
+ }
551
+
552
+ const remoteList = await $`git remote`.quiet().nothrow().cwd(Instance.worktree)
553
+ if (remoteList.exitCode !== 0) {
554
+ throw new ResetFailedError({ message: errorText(remoteList) || "Failed to list git remotes" })
555
+ }
556
+
557
+ const remotes = outputText(remoteList.stdout)
558
+ .split("\n")
559
+ .map((line) => line.trim())
560
+ .filter(Boolean)
561
+
562
+ const remote = remotes.includes("origin")
563
+ ? "origin"
564
+ : remotes.length === 1
565
+ ? remotes[0]
566
+ : remotes.includes("upstream")
567
+ ? "upstream"
568
+ : ""
569
+
570
+ const remoteHead = remote
571
+ ? await $`git symbolic-ref refs/remotes/${remote}/HEAD`.quiet().nothrow().cwd(Instance.worktree)
572
+ : { exitCode: 1, stdout: undefined, stderr: undefined }
573
+
574
+ const remoteRef = remoteHead.exitCode === 0 ? outputText(remoteHead.stdout) : ""
575
+ const remoteTarget = remoteRef ? remoteRef.replace(/^refs\/remotes\//, "") : ""
576
+ const remoteBranch = remote && remoteTarget.startsWith(`${remote}/`) ? remoteTarget.slice(`${remote}/`.length) : ""
577
+
578
+ const mainCheck = await $`git show-ref --verify --quiet refs/heads/main`.quiet().nothrow().cwd(Instance.worktree)
579
+ const masterCheck = await $`git show-ref --verify --quiet refs/heads/master`
580
+ .quiet()
581
+ .nothrow()
582
+ .cwd(Instance.worktree)
583
+ const localBranch = mainCheck.exitCode === 0 ? "main" : masterCheck.exitCode === 0 ? "master" : ""
584
+
585
+ const target = remoteBranch ? `${remote}/${remoteBranch}` : localBranch
586
+ if (!target) {
587
+ throw new ResetFailedError({ message: "Default branch not found" })
588
+ }
589
+
590
+ if (remoteBranch) {
591
+ const fetch = await $`git fetch ${remote} ${remoteBranch}`.quiet().nothrow().cwd(Instance.worktree)
592
+ if (fetch.exitCode !== 0) {
593
+ throw new ResetFailedError({ message: errorText(fetch) || `Failed to fetch ${target}` })
594
+ }
595
+ }
596
+
597
+ if (!entry.path) {
598
+ throw new ResetFailedError({ message: "Worktree path not found" })
599
+ }
600
+
601
+ const worktreePath = entry.path
602
+
603
+ const resetToTarget = await $`git reset --hard ${target}`.quiet().nothrow().cwd(worktreePath)
604
+ if (resetToTarget.exitCode !== 0) {
605
+ throw new ResetFailedError({ message: errorText(resetToTarget) || "Failed to reset worktree to target" })
606
+ }
607
+
608
+ const clean = await sweep(worktreePath)
609
+ if (clean.exitCode !== 0) {
610
+ throw new ResetFailedError({ message: errorText(clean) || "Failed to clean worktree" })
611
+ }
612
+
613
+ const update = await $`git submodule update --init --recursive --force`.quiet().nothrow().cwd(worktreePath)
614
+ if (update.exitCode !== 0) {
615
+ throw new ResetFailedError({ message: errorText(update) || "Failed to update submodules" })
616
+ }
617
+
618
+ const subReset = await $`git submodule foreach --recursive git reset --hard`.quiet().nothrow().cwd(worktreePath)
619
+ if (subReset.exitCode !== 0) {
620
+ throw new ResetFailedError({ message: errorText(subReset) || "Failed to reset submodules" })
621
+ }
622
+
623
+ const subClean = await $`git submodule foreach --recursive git clean -fdx`.quiet().nothrow().cwd(worktreePath)
624
+ if (subClean.exitCode !== 0) {
625
+ throw new ResetFailedError({ message: errorText(subClean) || "Failed to clean submodules" })
626
+ }
627
+
628
+ const status = await $`git status --porcelain=v1`.quiet().nothrow().cwd(worktreePath)
629
+ if (status.exitCode !== 0) {
630
+ throw new ResetFailedError({ message: errorText(status) || "Failed to read git status" })
631
+ }
632
+
633
+ const dirty = outputText(status.stdout)
634
+ if (dirty) {
635
+ throw new ResetFailedError({ message: `Worktree reset left local changes:\n${dirty}` })
636
+ }
637
+
638
+ const projectID = Instance.project.id
639
+ queueStartScripts(worktreePath, { projectID })
640
+
641
+ return true
642
+ })
643
+ }
package/sst-env.d.ts ADDED
@@ -0,0 +1,10 @@
1
+ /* This file is auto-generated by SST. Do not edit. */
2
+ /* tslint:disable */
3
+ /* eslint-disable */
4
+ /* deno-fmt-ignore-file */
5
+ /* biome-ignore-all lint: auto-generated */
6
+
7
+ /// <reference path="../../sst-env.d.ts" />
8
+
9
+ import "sst"
10
+ export {}