@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,78 @@
1
+ import { describe, expect, test } from "bun:test"
2
+ import path from "path"
3
+ import { Session } from "../../src/session"
4
+ import { Log } from "../../src/util/log"
5
+ import { Instance } from "../../src/project/instance"
6
+ import { Server } from "../../src/server/server"
7
+
8
+ const projectRoot = path.join(__dirname, "../..")
9
+ Log.init({ print: false })
10
+
11
+ describe("tui.selectSession endpoint", () => {
12
+ test("should return 200 when called with valid session", async () => {
13
+ await Instance.provide({
14
+ directory: projectRoot,
15
+ fn: async () => {
16
+ // #given
17
+ const session = await Session.create({})
18
+
19
+ // #when
20
+ const app = Server.App()
21
+ const response = await app.request("/tui/select-session", {
22
+ method: "POST",
23
+ headers: { "Content-Type": "application/json" },
24
+ body: JSON.stringify({ sessionID: session.id }),
25
+ })
26
+
27
+ // #then
28
+ expect(response.status).toBe(200)
29
+ const body = await response.json()
30
+ expect(body).toBe(true)
31
+
32
+ await Session.remove(session.id)
33
+ },
34
+ })
35
+ })
36
+
37
+ test("should return 404 when session does not exist", async () => {
38
+ await Instance.provide({
39
+ directory: projectRoot,
40
+ fn: async () => {
41
+ // #given
42
+ const nonExistentSessionID = "ses_nonexistent123"
43
+
44
+ // #when
45
+ const app = Server.App()
46
+ const response = await app.request("/tui/select-session", {
47
+ method: "POST",
48
+ headers: { "Content-Type": "application/json" },
49
+ body: JSON.stringify({ sessionID: nonExistentSessionID }),
50
+ })
51
+
52
+ // #then
53
+ expect(response.status).toBe(404)
54
+ },
55
+ })
56
+ })
57
+
58
+ test("should return 400 when session ID format is invalid", async () => {
59
+ await Instance.provide({
60
+ directory: projectRoot,
61
+ fn: async () => {
62
+ // #given
63
+ const invalidSessionID = "invalid_session_id"
64
+
65
+ // #when
66
+ const app = Server.App()
67
+ const response = await app.request("/tui/select-session", {
68
+ method: "POST",
69
+ headers: { "Content-Type": "application/json" },
70
+ body: JSON.stringify({ sessionID: invalidSessionID }),
71
+ })
72
+
73
+ // #then
74
+ expect(response.status).toBe(400)
75
+ },
76
+ })
77
+ })
78
+ })
@@ -0,0 +1,423 @@
1
+ import { describe, expect, test } from "bun:test"
2
+ import path from "path"
3
+ import { SessionCompaction } from "../../src/session/compaction"
4
+ import { Token } from "../../src/util/token"
5
+ import { Instance } from "../../src/project/instance"
6
+ import { Log } from "../../src/util/log"
7
+ import { tmpdir } from "../fixture/fixture"
8
+ import { Session } from "../../src/session"
9
+ import type { Provider } from "../../src/provider/provider"
10
+
11
+ Log.init({ print: false })
12
+
13
+ function createModel(opts: {
14
+ context: number
15
+ output: number
16
+ input?: number
17
+ cost?: Provider.Model["cost"]
18
+ npm?: string
19
+ }): Provider.Model {
20
+ return {
21
+ id: "test-model",
22
+ providerID: "test",
23
+ name: "Test",
24
+ limit: {
25
+ context: opts.context,
26
+ input: opts.input,
27
+ output: opts.output,
28
+ },
29
+ cost: opts.cost ?? { input: 0, output: 0, cache: { read: 0, write: 0 } },
30
+ capabilities: {
31
+ toolcall: true,
32
+ attachment: false,
33
+ reasoning: false,
34
+ temperature: true,
35
+ input: { text: true, image: false, audio: false, video: false },
36
+ output: { text: true, image: false, audio: false, video: false },
37
+ },
38
+ api: { npm: opts.npm ?? "@ai-sdk/anthropic" },
39
+ options: {},
40
+ } as Provider.Model
41
+ }
42
+
43
+ describe("session.compaction.isOverflow", () => {
44
+ test("returns true when token count exceeds usable context", async () => {
45
+ await using tmp = await tmpdir()
46
+ await Instance.provide({
47
+ directory: tmp.path,
48
+ fn: async () => {
49
+ const model = createModel({ context: 100_000, output: 32_000 })
50
+ const tokens = { input: 75_000, output: 5_000, reasoning: 0, cache: { read: 0, write: 0 } }
51
+ expect(await SessionCompaction.isOverflow({ tokens, model })).toBe(true)
52
+ },
53
+ })
54
+ })
55
+
56
+ test("returns false when token count within usable context", async () => {
57
+ await using tmp = await tmpdir()
58
+ await Instance.provide({
59
+ directory: tmp.path,
60
+ fn: async () => {
61
+ const model = createModel({ context: 200_000, output: 32_000 })
62
+ const tokens = { input: 100_000, output: 10_000, reasoning: 0, cache: { read: 0, write: 0 } }
63
+ expect(await SessionCompaction.isOverflow({ tokens, model })).toBe(false)
64
+ },
65
+ })
66
+ })
67
+
68
+ test("includes cache.read in token count", async () => {
69
+ await using tmp = await tmpdir()
70
+ await Instance.provide({
71
+ directory: tmp.path,
72
+ fn: async () => {
73
+ const model = createModel({ context: 100_000, output: 32_000 })
74
+ const tokens = { input: 60_000, output: 10_000, reasoning: 0, cache: { read: 10_000, write: 0 } }
75
+ expect(await SessionCompaction.isOverflow({ tokens, model })).toBe(true)
76
+ },
77
+ })
78
+ })
79
+
80
+ test("respects input limit for input caps", async () => {
81
+ await using tmp = await tmpdir()
82
+ await Instance.provide({
83
+ directory: tmp.path,
84
+ fn: async () => {
85
+ const model = createModel({ context: 400_000, input: 272_000, output: 128_000 })
86
+ const tokens = { input: 271_000, output: 1_000, reasoning: 0, cache: { read: 2_000, write: 0 } }
87
+ expect(await SessionCompaction.isOverflow({ tokens, model })).toBe(true)
88
+ },
89
+ })
90
+ })
91
+
92
+ test("returns false when input/output are within input caps", async () => {
93
+ await using tmp = await tmpdir()
94
+ await Instance.provide({
95
+ directory: tmp.path,
96
+ fn: async () => {
97
+ const model = createModel({ context: 400_000, input: 272_000, output: 128_000 })
98
+ const tokens = { input: 200_000, output: 20_000, reasoning: 0, cache: { read: 10_000, write: 0 } }
99
+ expect(await SessionCompaction.isOverflow({ tokens, model })).toBe(false)
100
+ },
101
+ })
102
+ })
103
+
104
+ test("returns false when output within limit with input caps", async () => {
105
+ await using tmp = await tmpdir()
106
+ await Instance.provide({
107
+ directory: tmp.path,
108
+ fn: async () => {
109
+ const model = createModel({ context: 200_000, input: 120_000, output: 10_000 })
110
+ const tokens = { input: 50_000, output: 9_999, reasoning: 0, cache: { read: 0, write: 0 } }
111
+ expect(await SessionCompaction.isOverflow({ tokens, model })).toBe(false)
112
+ },
113
+ })
114
+ })
115
+
116
+ // ─── Bug reproduction tests ───────────────────────────────────────────
117
+ // These tests demonstrate that when limit.input is set, isOverflow()
118
+ // does not subtract any headroom for the next model response. This means
119
+ // compaction only triggers AFTER we've already consumed the full input
120
+ // budget, leaving zero room for the next API call's output tokens.
121
+ //
122
+ // Compare: without limit.input, usable = context - output (reserves space).
123
+ // With limit.input, usable = limit.input (reserves nothing).
124
+ //
125
+ // Related issues: #10634, #8089, #11086, #12621
126
+ // Open PRs: #6875, #12924
127
+
128
+ test("BUG: no headroom when limit.input is set — compaction should trigger near boundary but does not", async () => {
129
+ await using tmp = await tmpdir()
130
+ await Instance.provide({
131
+ directory: tmp.path,
132
+ fn: async () => {
133
+ // Simulate Claude with prompt caching: input limit = 200K, output limit = 32K
134
+ const model = createModel({ context: 200_000, input: 200_000, output: 32_000 })
135
+
136
+ // We've used 198K tokens total. Only 2K under the input limit.
137
+ // On the next turn, the full conversation (198K) becomes input,
138
+ // plus the model needs room to generate output — this WILL overflow.
139
+ const tokens = { input: 180_000, output: 15_000, reasoning: 0, cache: { read: 3_000, write: 0 } }
140
+ // count = 180K + 3K + 15K = 198K
141
+ // usable = limit.input = 200K (no output subtracted!)
142
+ // 198K > 200K = false → no compaction triggered
143
+
144
+ // WITHOUT limit.input: usable = 200K - 32K = 168K, and 198K > 168K = true ✓
145
+ // WITH limit.input: usable = 200K, and 198K > 200K = false ✗
146
+
147
+ // With 198K used and only 2K headroom, the next turn will overflow.
148
+ // Compaction MUST trigger here.
149
+ expect(await SessionCompaction.isOverflow({ tokens, model })).toBe(true)
150
+ },
151
+ })
152
+ })
153
+
154
+ test("BUG: without limit.input, same token count correctly triggers compaction", async () => {
155
+ await using tmp = await tmpdir()
156
+ await Instance.provide({
157
+ directory: tmp.path,
158
+ fn: async () => {
159
+ // Same model but without limit.input — uses context - output instead
160
+ const model = createModel({ context: 200_000, output: 32_000 })
161
+
162
+ // Same token usage as above
163
+ const tokens = { input: 180_000, output: 15_000, reasoning: 0, cache: { read: 3_000, write: 0 } }
164
+ // count = 198K
165
+ // usable = context - output = 200K - 32K = 168K
166
+ // 198K > 168K = true → compaction correctly triggered
167
+
168
+ const result = await SessionCompaction.isOverflow({ tokens, model })
169
+ expect(result).toBe(true) // ← Correct: headroom is reserved
170
+ },
171
+ })
172
+ })
173
+
174
+ test("BUG: asymmetry — limit.input model allows 30K more usage before compaction than equivalent model without it", async () => {
175
+ await using tmp = await tmpdir()
176
+ await Instance.provide({
177
+ directory: tmp.path,
178
+ fn: async () => {
179
+ // Two models with identical context/output limits, differing only in limit.input
180
+ const withInputLimit = createModel({ context: 200_000, input: 200_000, output: 32_000 })
181
+ const withoutInputLimit = createModel({ context: 200_000, output: 32_000 })
182
+
183
+ // 170K total tokens — well above context-output (168K) but below input limit (200K)
184
+ const tokens = { input: 166_000, output: 10_000, reasoning: 0, cache: { read: 5_000, write: 0 } }
185
+
186
+ const withLimit = await SessionCompaction.isOverflow({ tokens, model: withInputLimit })
187
+ const withoutLimit = await SessionCompaction.isOverflow({ tokens, model: withoutInputLimit })
188
+
189
+ // Both models have identical real capacity — they should agree:
190
+ expect(withLimit).toBe(true) // should compact (170K leaves no room for 32K output)
191
+ expect(withoutLimit).toBe(true) // correctly compacts (170K > 168K)
192
+ },
193
+ })
194
+ })
195
+
196
+ test("returns false when model context limit is 0", async () => {
197
+ await using tmp = await tmpdir()
198
+ await Instance.provide({
199
+ directory: tmp.path,
200
+ fn: async () => {
201
+ const model = createModel({ context: 0, output: 32_000 })
202
+ const tokens = { input: 100_000, output: 10_000, reasoning: 0, cache: { read: 0, write: 0 } }
203
+ expect(await SessionCompaction.isOverflow({ tokens, model })).toBe(false)
204
+ },
205
+ })
206
+ })
207
+
208
+ test("returns false when compaction.auto is disabled", async () => {
209
+ await using tmp = await tmpdir({
210
+ init: async (dir) => {
211
+ await Bun.write(
212
+ path.join(dir, "opencode.json"),
213
+ JSON.stringify({
214
+ compaction: { auto: false },
215
+ }),
216
+ )
217
+ },
218
+ })
219
+ await Instance.provide({
220
+ directory: tmp.path,
221
+ fn: async () => {
222
+ const model = createModel({ context: 100_000, output: 32_000 })
223
+ const tokens = { input: 75_000, output: 5_000, reasoning: 0, cache: { read: 0, write: 0 } }
224
+ expect(await SessionCompaction.isOverflow({ tokens, model })).toBe(false)
225
+ },
226
+ })
227
+ })
228
+ })
229
+
230
+ describe("util.token.estimate", () => {
231
+ test("estimates tokens from text (4 chars per token)", () => {
232
+ const text = "x".repeat(4000)
233
+ expect(Token.estimate(text)).toBe(1000)
234
+ })
235
+
236
+ test("estimates tokens from larger text", () => {
237
+ const text = "y".repeat(20_000)
238
+ expect(Token.estimate(text)).toBe(5000)
239
+ })
240
+
241
+ test("returns 0 for empty string", () => {
242
+ expect(Token.estimate("")).toBe(0)
243
+ })
244
+ })
245
+
246
+ describe("session.getUsage", () => {
247
+ test("normalizes standard usage to token format", () => {
248
+ const model = createModel({ context: 100_000, output: 32_000 })
249
+ const result = Session.getUsage({
250
+ model,
251
+ usage: {
252
+ inputTokens: 1000,
253
+ outputTokens: 500,
254
+ totalTokens: 1500,
255
+ },
256
+ })
257
+
258
+ expect(result.tokens.input).toBe(1000)
259
+ expect(result.tokens.output).toBe(500)
260
+ expect(result.tokens.reasoning).toBe(0)
261
+ expect(result.tokens.cache.read).toBe(0)
262
+ expect(result.tokens.cache.write).toBe(0)
263
+ })
264
+
265
+ test("extracts cached tokens to cache.read", () => {
266
+ const model = createModel({ context: 100_000, output: 32_000 })
267
+ const result = Session.getUsage({
268
+ model,
269
+ usage: {
270
+ inputTokens: 1000,
271
+ outputTokens: 500,
272
+ totalTokens: 1500,
273
+ cachedInputTokens: 200,
274
+ },
275
+ })
276
+
277
+ expect(result.tokens.input).toBe(800)
278
+ expect(result.tokens.cache.read).toBe(200)
279
+ })
280
+
281
+ test("handles anthropic cache write metadata", () => {
282
+ const model = createModel({ context: 100_000, output: 32_000 })
283
+ const result = Session.getUsage({
284
+ model,
285
+ usage: {
286
+ inputTokens: 1000,
287
+ outputTokens: 500,
288
+ totalTokens: 1500,
289
+ },
290
+ metadata: {
291
+ anthropic: {
292
+ cacheCreationInputTokens: 300,
293
+ },
294
+ },
295
+ })
296
+
297
+ expect(result.tokens.cache.write).toBe(300)
298
+ })
299
+
300
+ test("does not subtract cached tokens for anthropic provider", () => {
301
+ const model = createModel({ context: 100_000, output: 32_000 })
302
+ const result = Session.getUsage({
303
+ model,
304
+ usage: {
305
+ inputTokens: 1000,
306
+ outputTokens: 500,
307
+ totalTokens: 1500,
308
+ cachedInputTokens: 200,
309
+ },
310
+ metadata: {
311
+ anthropic: {},
312
+ },
313
+ })
314
+
315
+ expect(result.tokens.input).toBe(1000)
316
+ expect(result.tokens.cache.read).toBe(200)
317
+ })
318
+
319
+ test("handles reasoning tokens", () => {
320
+ const model = createModel({ context: 100_000, output: 32_000 })
321
+ const result = Session.getUsage({
322
+ model,
323
+ usage: {
324
+ inputTokens: 1000,
325
+ outputTokens: 500,
326
+ totalTokens: 1500,
327
+ reasoningTokens: 100,
328
+ },
329
+ })
330
+
331
+ expect(result.tokens.reasoning).toBe(100)
332
+ })
333
+
334
+ test("handles undefined optional values gracefully", () => {
335
+ const model = createModel({ context: 100_000, output: 32_000 })
336
+ const result = Session.getUsage({
337
+ model,
338
+ usage: {
339
+ inputTokens: 0,
340
+ outputTokens: 0,
341
+ totalTokens: 0,
342
+ },
343
+ })
344
+
345
+ expect(result.tokens.input).toBe(0)
346
+ expect(result.tokens.output).toBe(0)
347
+ expect(result.tokens.reasoning).toBe(0)
348
+ expect(result.tokens.cache.read).toBe(0)
349
+ expect(result.tokens.cache.write).toBe(0)
350
+ expect(Number.isNaN(result.cost)).toBe(false)
351
+ })
352
+
353
+ test("calculates cost correctly", () => {
354
+ const model = createModel({
355
+ context: 100_000,
356
+ output: 32_000,
357
+ cost: {
358
+ input: 3,
359
+ output: 15,
360
+ cache: { read: 0.3, write: 3.75 },
361
+ },
362
+ })
363
+ const result = Session.getUsage({
364
+ model,
365
+ usage: {
366
+ inputTokens: 1_000_000,
367
+ outputTokens: 100_000,
368
+ totalTokens: 1_100_000,
369
+ },
370
+ })
371
+
372
+ expect(result.cost).toBe(3 + 1.5)
373
+ })
374
+
375
+ test.each(["@ai-sdk/anthropic", "@ai-sdk/amazon-bedrock", "@ai-sdk/google-vertex/anthropic"])(
376
+ "computes total from components for %s models",
377
+ (npm) => {
378
+ const model = createModel({ context: 100_000, output: 32_000, npm })
379
+ const usage = {
380
+ inputTokens: 1000,
381
+ outputTokens: 500,
382
+ // These providers typically report total as input + output only,
383
+ // excluding cache read/write.
384
+ totalTokens: 1500,
385
+ cachedInputTokens: 200,
386
+ }
387
+ if (npm === "@ai-sdk/amazon-bedrock") {
388
+ const result = Session.getUsage({
389
+ model,
390
+ usage,
391
+ metadata: {
392
+ bedrock: {
393
+ usage: {
394
+ cacheWriteInputTokens: 300,
395
+ },
396
+ },
397
+ },
398
+ })
399
+
400
+ expect(result.tokens.input).toBe(1000)
401
+ expect(result.tokens.cache.read).toBe(200)
402
+ expect(result.tokens.cache.write).toBe(300)
403
+ expect(result.tokens.total).toBe(2000)
404
+ return
405
+ }
406
+
407
+ const result = Session.getUsage({
408
+ model,
409
+ usage,
410
+ metadata: {
411
+ anthropic: {
412
+ cacheCreationInputTokens: 300,
413
+ },
414
+ },
415
+ })
416
+
417
+ expect(result.tokens.input).toBe(1000)
418
+ expect(result.tokens.cache.read).toBe(200)
419
+ expect(result.tokens.cache.write).toBe(300)
420
+ expect(result.tokens.total).toBe(2000)
421
+ },
422
+ )
423
+ })
@@ -0,0 +1,170 @@
1
+ import { afterEach, beforeEach, describe, expect, test } from "bun:test"
2
+ import path from "path"
3
+ import { InstructionPrompt } from "../../src/session/instruction"
4
+ import { Instance } from "../../src/project/instance"
5
+ import { Global } from "../../src/global"
6
+ import { tmpdir } from "../fixture/fixture"
7
+
8
+ describe("InstructionPrompt.resolve", () => {
9
+ test("returns empty when AGENTS.md is at project root (already in systemPaths)", async () => {
10
+ await using tmp = await tmpdir({
11
+ init: async (dir) => {
12
+ await Bun.write(path.join(dir, "AGENTS.md"), "# Root Instructions")
13
+ await Bun.write(path.join(dir, "src", "file.ts"), "const x = 1")
14
+ },
15
+ })
16
+ await Instance.provide({
17
+ directory: tmp.path,
18
+ fn: async () => {
19
+ const system = await InstructionPrompt.systemPaths()
20
+ expect(system.has(path.join(tmp.path, "AGENTS.md"))).toBe(true)
21
+
22
+ const results = await InstructionPrompt.resolve([], path.join(tmp.path, "src", "file.ts"), "test-message-1")
23
+ expect(results).toEqual([])
24
+ },
25
+ })
26
+ })
27
+
28
+ test("returns AGENTS.md from subdirectory (not in systemPaths)", async () => {
29
+ await using tmp = await tmpdir({
30
+ init: async (dir) => {
31
+ await Bun.write(path.join(dir, "subdir", "AGENTS.md"), "# Subdir Instructions")
32
+ await Bun.write(path.join(dir, "subdir", "nested", "file.ts"), "const x = 1")
33
+ },
34
+ })
35
+ await Instance.provide({
36
+ directory: tmp.path,
37
+ fn: async () => {
38
+ const system = await InstructionPrompt.systemPaths()
39
+ expect(system.has(path.join(tmp.path, "subdir", "AGENTS.md"))).toBe(false)
40
+
41
+ const results = await InstructionPrompt.resolve(
42
+ [],
43
+ path.join(tmp.path, "subdir", "nested", "file.ts"),
44
+ "test-message-2",
45
+ )
46
+ expect(results.length).toBe(1)
47
+ expect(results[0].filepath).toBe(path.join(tmp.path, "subdir", "AGENTS.md"))
48
+ },
49
+ })
50
+ })
51
+
52
+ test("doesn't reload AGENTS.md when reading it directly", async () => {
53
+ await using tmp = await tmpdir({
54
+ init: async (dir) => {
55
+ await Bun.write(path.join(dir, "subdir", "AGENTS.md"), "# Subdir Instructions")
56
+ await Bun.write(path.join(dir, "subdir", "nested", "file.ts"), "const x = 1")
57
+ },
58
+ })
59
+ await Instance.provide({
60
+ directory: tmp.path,
61
+ fn: async () => {
62
+ const filepath = path.join(tmp.path, "subdir", "AGENTS.md")
63
+ const system = await InstructionPrompt.systemPaths()
64
+ expect(system.has(filepath)).toBe(false)
65
+
66
+ const results = await InstructionPrompt.resolve([], filepath, "test-message-2")
67
+ expect(results).toEqual([])
68
+ },
69
+ })
70
+ })
71
+ })
72
+
73
+ describe("InstructionPrompt.systemPaths OPENCODE_CONFIG_DIR", () => {
74
+ let originalConfigDir: string | undefined
75
+
76
+ beforeEach(() => {
77
+ originalConfigDir = process.env["OPENCODE_CONFIG_DIR"]
78
+ })
79
+
80
+ afterEach(() => {
81
+ if (originalConfigDir === undefined) {
82
+ delete process.env["OPENCODE_CONFIG_DIR"]
83
+ } else {
84
+ process.env["OPENCODE_CONFIG_DIR"] = originalConfigDir
85
+ }
86
+ })
87
+
88
+ test("prefers OPENCODE_CONFIG_DIR AGENTS.md over global when both exist", async () => {
89
+ await using profileTmp = await tmpdir({
90
+ init: async (dir) => {
91
+ await Bun.write(path.join(dir, "AGENTS.md"), "# Profile Instructions")
92
+ },
93
+ })
94
+ await using globalTmp = await tmpdir({
95
+ init: async (dir) => {
96
+ await Bun.write(path.join(dir, "AGENTS.md"), "# Global Instructions")
97
+ },
98
+ })
99
+ await using projectTmp = await tmpdir()
100
+
101
+ process.env["OPENCODE_CONFIG_DIR"] = profileTmp.path
102
+ const originalGlobalConfig = Global.Path.config
103
+ ;(Global.Path as { config: string }).config = globalTmp.path
104
+
105
+ try {
106
+ await Instance.provide({
107
+ directory: projectTmp.path,
108
+ fn: async () => {
109
+ const paths = await InstructionPrompt.systemPaths()
110
+ expect(paths.has(path.join(profileTmp.path, "AGENTS.md"))).toBe(true)
111
+ expect(paths.has(path.join(globalTmp.path, "AGENTS.md"))).toBe(false)
112
+ },
113
+ })
114
+ } finally {
115
+ ;(Global.Path as { config: string }).config = originalGlobalConfig
116
+ }
117
+ })
118
+
119
+ test("falls back to global AGENTS.md when OPENCODE_CONFIG_DIR has no AGENTS.md", async () => {
120
+ await using profileTmp = await tmpdir()
121
+ await using globalTmp = await tmpdir({
122
+ init: async (dir) => {
123
+ await Bun.write(path.join(dir, "AGENTS.md"), "# Global Instructions")
124
+ },
125
+ })
126
+ await using projectTmp = await tmpdir()
127
+
128
+ process.env["OPENCODE_CONFIG_DIR"] = profileTmp.path
129
+ const originalGlobalConfig = Global.Path.config
130
+ ;(Global.Path as { config: string }).config = globalTmp.path
131
+
132
+ try {
133
+ await Instance.provide({
134
+ directory: projectTmp.path,
135
+ fn: async () => {
136
+ const paths = await InstructionPrompt.systemPaths()
137
+ expect(paths.has(path.join(profileTmp.path, "AGENTS.md"))).toBe(false)
138
+ expect(paths.has(path.join(globalTmp.path, "AGENTS.md"))).toBe(true)
139
+ },
140
+ })
141
+ } finally {
142
+ ;(Global.Path as { config: string }).config = originalGlobalConfig
143
+ }
144
+ })
145
+
146
+ test("uses global AGENTS.md when OPENCODE_CONFIG_DIR is not set", async () => {
147
+ await using globalTmp = await tmpdir({
148
+ init: async (dir) => {
149
+ await Bun.write(path.join(dir, "AGENTS.md"), "# Global Instructions")
150
+ },
151
+ })
152
+ await using projectTmp = await tmpdir()
153
+
154
+ delete process.env["OPENCODE_CONFIG_DIR"]
155
+ const originalGlobalConfig = Global.Path.config
156
+ ;(Global.Path as { config: string }).config = globalTmp.path
157
+
158
+ try {
159
+ await Instance.provide({
160
+ directory: projectTmp.path,
161
+ fn: async () => {
162
+ const paths = await InstructionPrompt.systemPaths()
163
+ expect(paths.has(path.join(globalTmp.path, "AGENTS.md"))).toBe(true)
164
+ },
165
+ })
166
+ } finally {
167
+ ;(Global.Path as { config: string }).config = originalGlobalConfig
168
+ }
169
+ })
170
+ })