@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,395 @@
1
+ import { text } from "node:stream/consumers"
2
+ import { BunProc } from "../bun"
3
+ import { Instance } from "../project/instance"
4
+ import { Filesystem } from "../util/filesystem"
5
+ import { Process } from "../util/process"
6
+ import { Flag } from "@/flag/flag"
7
+
8
+ export interface Info {
9
+ name: string
10
+ command: string[]
11
+ environment?: Record<string, string>
12
+ extensions: string[]
13
+ enabled(): Promise<boolean>
14
+ }
15
+
16
+ export const gofmt: Info = {
17
+ name: "gofmt",
18
+ command: ["gofmt", "-w", "$FILE"],
19
+ extensions: [".go"],
20
+ async enabled() {
21
+ return Bun.which("gofmt") !== null
22
+ },
23
+ }
24
+
25
+ export const mix: Info = {
26
+ name: "mix",
27
+ command: ["mix", "format", "$FILE"],
28
+ extensions: [".ex", ".exs", ".eex", ".heex", ".leex", ".neex", ".sface"],
29
+ async enabled() {
30
+ return Bun.which("mix") !== null
31
+ },
32
+ }
33
+
34
+ export const prettier: Info = {
35
+ name: "prettier",
36
+ command: [BunProc.which(), "x", "prettier", "--write", "$FILE"],
37
+ environment: {
38
+ BUN_BE_BUN: "1",
39
+ },
40
+ extensions: [
41
+ ".js",
42
+ ".jsx",
43
+ ".mjs",
44
+ ".cjs",
45
+ ".ts",
46
+ ".tsx",
47
+ ".mts",
48
+ ".cts",
49
+ ".html",
50
+ ".htm",
51
+ ".css",
52
+ ".scss",
53
+ ".sass",
54
+ ".less",
55
+ ".vue",
56
+ ".svelte",
57
+ ".json",
58
+ ".jsonc",
59
+ ".yaml",
60
+ ".yml",
61
+ ".toml",
62
+ ".xml",
63
+ ".md",
64
+ ".mdx",
65
+ ".graphql",
66
+ ".gql",
67
+ ],
68
+ async enabled() {
69
+ const items = await Filesystem.findUp("package.json", Instance.directory, Instance.worktree)
70
+ for (const item of items) {
71
+ const json = await Filesystem.readJson<{
72
+ dependencies?: Record<string, string>
73
+ devDependencies?: Record<string, string>
74
+ }>(item)
75
+ if (json.dependencies?.prettier) return true
76
+ if (json.devDependencies?.prettier) return true
77
+ }
78
+ return false
79
+ },
80
+ }
81
+
82
+ export const oxfmt: Info = {
83
+ name: "oxfmt",
84
+ command: [BunProc.which(), "x", "oxfmt", "$FILE"],
85
+ environment: {
86
+ BUN_BE_BUN: "1",
87
+ },
88
+ extensions: [".js", ".jsx", ".mjs", ".cjs", ".ts", ".tsx", ".mts", ".cts"],
89
+ async enabled() {
90
+ if (!Flag.OPENCODE_EXPERIMENTAL_OXFMT) return false
91
+ const items = await Filesystem.findUp("package.json", Instance.directory, Instance.worktree)
92
+ for (const item of items) {
93
+ const json = await Filesystem.readJson<{
94
+ dependencies?: Record<string, string>
95
+ devDependencies?: Record<string, string>
96
+ }>(item)
97
+ if (json.dependencies?.oxfmt) return true
98
+ if (json.devDependencies?.oxfmt) return true
99
+ }
100
+ return false
101
+ },
102
+ }
103
+
104
+ export const biome: Info = {
105
+ name: "biome",
106
+ command: [BunProc.which(), "x", "@biomejs/biome", "check", "--write", "$FILE"],
107
+ environment: {
108
+ BUN_BE_BUN: "1",
109
+ },
110
+ extensions: [
111
+ ".js",
112
+ ".jsx",
113
+ ".mjs",
114
+ ".cjs",
115
+ ".ts",
116
+ ".tsx",
117
+ ".mts",
118
+ ".cts",
119
+ ".html",
120
+ ".htm",
121
+ ".css",
122
+ ".scss",
123
+ ".sass",
124
+ ".less",
125
+ ".vue",
126
+ ".svelte",
127
+ ".json",
128
+ ".jsonc",
129
+ ".yaml",
130
+ ".yml",
131
+ ".toml",
132
+ ".xml",
133
+ ".md",
134
+ ".mdx",
135
+ ".graphql",
136
+ ".gql",
137
+ ],
138
+ async enabled() {
139
+ const configs = ["biome.json", "biome.jsonc"]
140
+ for (const config of configs) {
141
+ const found = await Filesystem.findUp(config, Instance.directory, Instance.worktree)
142
+ if (found.length > 0) {
143
+ return true
144
+ }
145
+ }
146
+ return false
147
+ },
148
+ }
149
+
150
+ export const zig: Info = {
151
+ name: "zig",
152
+ command: ["zig", "fmt", "$FILE"],
153
+ extensions: [".zig", ".zon"],
154
+ async enabled() {
155
+ return Bun.which("zig") !== null
156
+ },
157
+ }
158
+
159
+ export const clang: Info = {
160
+ name: "clang-format",
161
+ command: ["clang-format", "-i", "$FILE"],
162
+ extensions: [".c", ".cc", ".cpp", ".cxx", ".c++", ".h", ".hh", ".hpp", ".hxx", ".h++", ".ino", ".C", ".H"],
163
+ async enabled() {
164
+ const items = await Filesystem.findUp(".clang-format", Instance.directory, Instance.worktree)
165
+ return items.length > 0
166
+ },
167
+ }
168
+
169
+ export const ktlint: Info = {
170
+ name: "ktlint",
171
+ command: ["ktlint", "-F", "$FILE"],
172
+ extensions: [".kt", ".kts"],
173
+ async enabled() {
174
+ return Bun.which("ktlint") !== null
175
+ },
176
+ }
177
+
178
+ export const ruff: Info = {
179
+ name: "ruff",
180
+ command: ["ruff", "format", "$FILE"],
181
+ extensions: [".py", ".pyi"],
182
+ async enabled() {
183
+ if (!Bun.which("ruff")) return false
184
+ const configs = ["pyproject.toml", "ruff.toml", ".ruff.toml"]
185
+ for (const config of configs) {
186
+ const found = await Filesystem.findUp(config, Instance.directory, Instance.worktree)
187
+ if (found.length > 0) {
188
+ if (config === "pyproject.toml") {
189
+ const content = await Filesystem.readText(found[0])
190
+ if (content.includes("[tool.ruff]")) return true
191
+ } else {
192
+ return true
193
+ }
194
+ }
195
+ }
196
+ const deps = ["requirements.txt", "pyproject.toml", "Pipfile"]
197
+ for (const dep of deps) {
198
+ const found = await Filesystem.findUp(dep, Instance.directory, Instance.worktree)
199
+ if (found.length > 0) {
200
+ const content = await Filesystem.readText(found[0])
201
+ if (content.includes("ruff")) return true
202
+ }
203
+ }
204
+ return false
205
+ },
206
+ }
207
+
208
+ export const rlang: Info = {
209
+ name: "air",
210
+ command: ["air", "format", "$FILE"],
211
+ extensions: [".R"],
212
+ async enabled() {
213
+ const airPath = Bun.which("air")
214
+ if (airPath == null) return false
215
+
216
+ try {
217
+ const proc = Process.spawn(["air", "--help"], {
218
+ stdout: "pipe",
219
+ stderr: "pipe",
220
+ })
221
+ await proc.exited
222
+ if (!proc.stdout) return false
223
+ const output = await text(proc.stdout)
224
+
225
+ // Check for "Air: An R language server and formatter"
226
+ const firstLine = output.split("\n")[0]
227
+ const hasR = firstLine.includes("R language")
228
+ const hasFormatter = firstLine.includes("formatter")
229
+ return hasR && hasFormatter
230
+ } catch (error) {
231
+ return false
232
+ }
233
+ },
234
+ }
235
+
236
+ export const uvformat: Info = {
237
+ name: "uv",
238
+ command: ["uv", "format", "--", "$FILE"],
239
+ extensions: [".py", ".pyi"],
240
+ async enabled() {
241
+ if (await ruff.enabled()) return false
242
+ if (Bun.which("uv") !== null) {
243
+ const proc = Process.spawn(["uv", "format", "--help"], { stderr: "pipe", stdout: "pipe" })
244
+ const code = await proc.exited
245
+ return code === 0
246
+ }
247
+ return false
248
+ },
249
+ }
250
+
251
+ export const rubocop: Info = {
252
+ name: "rubocop",
253
+ command: ["rubocop", "--autocorrect", "$FILE"],
254
+ extensions: [".rb", ".rake", ".gemspec", ".ru"],
255
+ async enabled() {
256
+ return Bun.which("rubocop") !== null
257
+ },
258
+ }
259
+
260
+ export const standardrb: Info = {
261
+ name: "standardrb",
262
+ command: ["standardrb", "--fix", "$FILE"],
263
+ extensions: [".rb", ".rake", ".gemspec", ".ru"],
264
+ async enabled() {
265
+ return Bun.which("standardrb") !== null
266
+ },
267
+ }
268
+
269
+ export const htmlbeautifier: Info = {
270
+ name: "htmlbeautifier",
271
+ command: ["htmlbeautifier", "$FILE"],
272
+ extensions: [".erb", ".html.erb"],
273
+ async enabled() {
274
+ return Bun.which("htmlbeautifier") !== null
275
+ },
276
+ }
277
+
278
+ export const dart: Info = {
279
+ name: "dart",
280
+ command: ["dart", "format", "$FILE"],
281
+ extensions: [".dart"],
282
+ async enabled() {
283
+ return Bun.which("dart") !== null
284
+ },
285
+ }
286
+
287
+ export const ocamlformat: Info = {
288
+ name: "ocamlformat",
289
+ command: ["ocamlformat", "-i", "$FILE"],
290
+ extensions: [".ml", ".mli"],
291
+ async enabled() {
292
+ if (!Bun.which("ocamlformat")) return false
293
+ const items = await Filesystem.findUp(".ocamlformat", Instance.directory, Instance.worktree)
294
+ return items.length > 0
295
+ },
296
+ }
297
+
298
+ export const terraform: Info = {
299
+ name: "terraform",
300
+ command: ["terraform", "fmt", "$FILE"],
301
+ extensions: [".tf", ".tfvars"],
302
+ async enabled() {
303
+ return Bun.which("terraform") !== null
304
+ },
305
+ }
306
+
307
+ export const latexindent: Info = {
308
+ name: "latexindent",
309
+ command: ["latexindent", "-w", "-s", "$FILE"],
310
+ extensions: [".tex"],
311
+ async enabled() {
312
+ return Bun.which("latexindent") !== null
313
+ },
314
+ }
315
+
316
+ export const gleam: Info = {
317
+ name: "gleam",
318
+ command: ["gleam", "format", "$FILE"],
319
+ extensions: [".gleam"],
320
+ async enabled() {
321
+ return Bun.which("gleam") !== null
322
+ },
323
+ }
324
+
325
+ export const shfmt: Info = {
326
+ name: "shfmt",
327
+ command: ["shfmt", "-w", "$FILE"],
328
+ extensions: [".sh", ".bash"],
329
+ async enabled() {
330
+ return Bun.which("shfmt") !== null
331
+ },
332
+ }
333
+
334
+ export const nixfmt: Info = {
335
+ name: "nixfmt",
336
+ command: ["nixfmt", "$FILE"],
337
+ extensions: [".nix"],
338
+ async enabled() {
339
+ return Bun.which("nixfmt") !== null
340
+ },
341
+ }
342
+
343
+ export const rustfmt: Info = {
344
+ name: "rustfmt",
345
+ command: ["rustfmt", "$FILE"],
346
+ extensions: [".rs"],
347
+ async enabled() {
348
+ return Bun.which("rustfmt") !== null
349
+ },
350
+ }
351
+
352
+ export const pint: Info = {
353
+ name: "pint",
354
+ command: ["./vendor/bin/pint", "$FILE"],
355
+ extensions: [".php"],
356
+ async enabled() {
357
+ const items = await Filesystem.findUp("composer.json", Instance.directory, Instance.worktree)
358
+ for (const item of items) {
359
+ const json = await Filesystem.readJson<{
360
+ require?: Record<string, string>
361
+ "require-dev"?: Record<string, string>
362
+ }>(item)
363
+ if (json.require?.["laravel/pint"]) return true
364
+ if (json["require-dev"]?.["laravel/pint"]) return true
365
+ }
366
+ return false
367
+ },
368
+ }
369
+
370
+ export const ormolu: Info = {
371
+ name: "ormolu",
372
+ command: ["ormolu", "-i", "$FILE"],
373
+ extensions: [".hs"],
374
+ async enabled() {
375
+ return Bun.which("ormolu") !== null
376
+ },
377
+ }
378
+
379
+ export const cljfmt: Info = {
380
+ name: "cljfmt",
381
+ command: ["cljfmt", "fix", "--quiet", "$FILE"],
382
+ extensions: [".clj", ".cljs", ".cljc", ".edn"],
383
+ async enabled() {
384
+ return Bun.which("cljfmt") !== null
385
+ },
386
+ }
387
+
388
+ export const dfmt: Info = {
389
+ name: "dfmt",
390
+ command: ["dfmt", "-i", "$FILE"],
391
+ extensions: [".d"],
392
+ async enabled() {
393
+ return Bun.which("dfmt") !== null
394
+ },
395
+ }
@@ -0,0 +1,140 @@
1
+ import { Bus } from "../bus"
2
+ import { File } from "../file"
3
+ import { Log } from "../util/log"
4
+ import path from "path"
5
+ import z from "zod"
6
+
7
+ import * as Formatter from "./formatter"
8
+ import { Config } from "../config/config"
9
+ import { mergeDeep } from "remeda"
10
+ import { Instance } from "../project/instance"
11
+ import { Process } from "../util/process"
12
+
13
+ export namespace Format {
14
+ const log = Log.create({ service: "format" })
15
+
16
+ export const Status = z
17
+ .object({
18
+ name: z.string(),
19
+ extensions: z.string().array(),
20
+ enabled: z.boolean(),
21
+ })
22
+ .meta({
23
+ ref: "FormatterStatus",
24
+ })
25
+ export type Status = z.infer<typeof Status>
26
+
27
+ const state = Instance.state(async () => {
28
+ const enabled: Record<string, boolean> = {}
29
+ const cfg = await Config.get()
30
+
31
+ const formatters: Record<string, Formatter.Info> = {}
32
+ if (cfg.formatter === false) {
33
+ log.info("all formatters are disabled")
34
+ return {
35
+ enabled,
36
+ formatters,
37
+ }
38
+ }
39
+
40
+ for (const item of Object.values(Formatter)) {
41
+ formatters[item.name] = item
42
+ }
43
+ for (const [name, item] of Object.entries(cfg.formatter ?? {})) {
44
+ if (item.disabled) {
45
+ delete formatters[name]
46
+ continue
47
+ }
48
+ const result: Formatter.Info = mergeDeep(formatters[name] ?? {}, {
49
+ command: [],
50
+ extensions: [],
51
+ ...item,
52
+ })
53
+
54
+ if (result.command.length === 0) continue
55
+
56
+ result.enabled = async () => true
57
+ result.name = name
58
+ formatters[name] = result
59
+ }
60
+
61
+ return {
62
+ enabled,
63
+ formatters,
64
+ }
65
+ })
66
+
67
+ async function isEnabled(item: Formatter.Info) {
68
+ const s = await state()
69
+ let status = s.enabled[item.name]
70
+ if (status === undefined) {
71
+ status = await item.enabled()
72
+ s.enabled[item.name] = status
73
+ }
74
+ return status
75
+ }
76
+
77
+ async function getFormatter(ext: string) {
78
+ const formatters = await state().then((x) => x.formatters)
79
+ const result = []
80
+ for (const item of Object.values(formatters)) {
81
+ log.info("checking", { name: item.name, ext })
82
+ if (!item.extensions.includes(ext)) continue
83
+ if (!(await isEnabled(item))) continue
84
+ log.info("enabled", { name: item.name, ext })
85
+ result.push(item)
86
+ }
87
+ return result
88
+ }
89
+
90
+ export async function status() {
91
+ const s = await state()
92
+ const result: Status[] = []
93
+ for (const formatter of Object.values(s.formatters)) {
94
+ const enabled = await isEnabled(formatter)
95
+ result.push({
96
+ name: formatter.name,
97
+ extensions: formatter.extensions,
98
+ enabled,
99
+ })
100
+ }
101
+ return result
102
+ }
103
+
104
+ export function init() {
105
+ log.info("init")
106
+ Bus.subscribe(File.Event.Edited, async (payload) => {
107
+ const file = payload.properties.file
108
+ log.info("formatting", { file })
109
+ const ext = path.extname(file)
110
+
111
+ for (const item of await getFormatter(ext)) {
112
+ log.info("running", { command: item.command })
113
+ try {
114
+ const proc = Process.spawn(
115
+ item.command.map((x) => x.replace("$FILE", file)),
116
+ {
117
+ cwd: Instance.directory,
118
+ env: { ...process.env, ...item.environment },
119
+ stdout: "ignore",
120
+ stderr: "ignore",
121
+ },
122
+ )
123
+ const exit = await proc.exited
124
+ if (exit !== 0)
125
+ log.error("failed", {
126
+ command: item.command,
127
+ ...item.environment,
128
+ })
129
+ } catch (error) {
130
+ log.error("failed to format file", {
131
+ error,
132
+ command: item.command,
133
+ ...item.environment,
134
+ file,
135
+ })
136
+ }
137
+ }
138
+ })
139
+ }
140
+ }
@@ -0,0 +1,54 @@
1
+ import fs from "fs/promises"
2
+ import { xdgData, xdgCache, xdgConfig, xdgState } from "xdg-basedir"
3
+ import path from "path"
4
+ import os from "os"
5
+ import { Filesystem } from "../util/filesystem"
6
+
7
+ const app = "opencode"
8
+
9
+ const data = path.join(xdgData!, app)
10
+ const cache = path.join(xdgCache!, app)
11
+ const config = path.join(xdgConfig!, app)
12
+ const state = path.join(xdgState!, app)
13
+
14
+ export namespace Global {
15
+ export const Path = {
16
+ // Allow override via OPENCODE_TEST_HOME for test isolation
17
+ get home() {
18
+ return process.env.OPENCODE_TEST_HOME || os.homedir()
19
+ },
20
+ data,
21
+ bin: path.join(data, "bin"),
22
+ log: path.join(data, "log"),
23
+ cache,
24
+ config,
25
+ state,
26
+ }
27
+ }
28
+
29
+ await Promise.all([
30
+ fs.mkdir(Global.Path.data, { recursive: true }),
31
+ fs.mkdir(Global.Path.config, { recursive: true }),
32
+ fs.mkdir(Global.Path.state, { recursive: true }),
33
+ fs.mkdir(Global.Path.log, { recursive: true }),
34
+ fs.mkdir(Global.Path.bin, { recursive: true }),
35
+ ])
36
+
37
+ const CACHE_VERSION = "21"
38
+
39
+ const version = await Filesystem.readText(path.join(Global.Path.cache, "version")).catch(() => "0")
40
+
41
+ if (version !== CACHE_VERSION) {
42
+ try {
43
+ const contents = await fs.readdir(Global.Path.cache)
44
+ await Promise.all(
45
+ contents.map((item) =>
46
+ fs.rm(path.join(Global.Path.cache, item), {
47
+ recursive: true,
48
+ force: true,
49
+ }),
50
+ ),
51
+ )
52
+ } catch (e) {}
53
+ await Filesystem.write(path.join(Global.Path.cache, "version"), CACHE_VERSION)
54
+ }
package/src/id/id.ts ADDED
@@ -0,0 +1,84 @@
1
+ import z from "zod"
2
+ import { randomBytes } from "crypto"
3
+
4
+ export namespace Identifier {
5
+ const prefixes = {
6
+ session: "ses",
7
+ message: "msg",
8
+ permission: "per",
9
+ question: "que",
10
+ user: "usr",
11
+ part: "prt",
12
+ pty: "pty",
13
+ tool: "tool",
14
+ workspace: "wrk",
15
+ } as const
16
+
17
+ export function schema(prefix: keyof typeof prefixes) {
18
+ return z.string().startsWith(prefixes[prefix])
19
+ }
20
+
21
+ const LENGTH = 26
22
+
23
+ // State for monotonic ID generation
24
+ let lastTimestamp = 0
25
+ let counter = 0
26
+
27
+ export function ascending(prefix: keyof typeof prefixes, given?: string) {
28
+ return generateID(prefix, false, given)
29
+ }
30
+
31
+ export function descending(prefix: keyof typeof prefixes, given?: string) {
32
+ return generateID(prefix, true, given)
33
+ }
34
+
35
+ function generateID(prefix: keyof typeof prefixes, descending: boolean, given?: string): string {
36
+ if (!given) {
37
+ return create(prefix, descending)
38
+ }
39
+
40
+ if (!given.startsWith(prefixes[prefix])) {
41
+ throw new Error(`ID ${given} does not start with ${prefixes[prefix]}`)
42
+ }
43
+ return given
44
+ }
45
+
46
+ function randomBase62(length: number): string {
47
+ const chars = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"
48
+ let result = ""
49
+ const bytes = randomBytes(length)
50
+ for (let i = 0; i < length; i++) {
51
+ result += chars[bytes[i] % 62]
52
+ }
53
+ return result
54
+ }
55
+
56
+ export function create(prefix: keyof typeof prefixes, descending: boolean, timestamp?: number): string {
57
+ const currentTimestamp = timestamp ?? Date.now()
58
+
59
+ if (currentTimestamp !== lastTimestamp) {
60
+ lastTimestamp = currentTimestamp
61
+ counter = 0
62
+ }
63
+ counter++
64
+
65
+ let now = BigInt(currentTimestamp) * BigInt(0x1000) + BigInt(counter)
66
+
67
+ now = descending ? ~now : now
68
+
69
+ const timeBytes = Buffer.alloc(6)
70
+ for (let i = 0; i < 6; i++) {
71
+ timeBytes[i] = Number((now >> BigInt(40 - 8 * i)) & BigInt(0xff))
72
+ }
73
+
74
+ return prefixes[prefix] + "_" + timeBytes.toString("hex") + randomBase62(LENGTH - 12)
75
+ }
76
+
77
+ /** Extract timestamp from an ascending ID. Does not work with descending IDs. */
78
+ export function timestamp(id: string): number {
79
+ const prefix = id.split("_")[0]
80
+ const hex = id.slice(prefix.length + 1, prefix.length + 13)
81
+ const encoded = BigInt("0x" + hex)
82
+ return Number(encoded / BigInt(0x1000))
83
+ }
84
+ }