innocode 1.0.0

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 (434) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/innocode +84 -0
  5. package/bin/opencode +84 -0
  6. package/bunfig.toml +5 -0
  7. package/package.json +126 -0
  8. package/parsers-config.ts +253 -0
  9. package/script/build.ts +198 -0
  10. package/script/postinstall.mjs +125 -0
  11. package/script/publish.ts +186 -0
  12. package/script/schema.ts +47 -0
  13. package/script/seed-e2e.ts +50 -0
  14. package/src/acp/README.md +164 -0
  15. package/src/acp/agent.ts +1676 -0
  16. package/src/acp/session.ts +117 -0
  17. package/src/acp/types.ts +23 -0
  18. package/src/agent/agent.ts +338 -0
  19. package/src/agent/generate.txt +75 -0
  20. package/src/agent/prompt/compaction.txt +14 -0
  21. package/src/agent/prompt/explore.txt +18 -0
  22. package/src/agent/prompt/summary.txt +11 -0
  23. package/src/agent/prompt/title.txt +44 -0
  24. package/src/auth/index.ts +70 -0
  25. package/src/bun/index.ts +137 -0
  26. package/src/bun/registry.ts +48 -0
  27. package/src/bus/bus-event.ts +43 -0
  28. package/src/bus/global.ts +10 -0
  29. package/src/bus/index.ts +105 -0
  30. package/src/cli/bootstrap.ts +17 -0
  31. package/src/cli/cmd/acp.ts +70 -0
  32. package/src/cli/cmd/agent.ts +257 -0
  33. package/src/cli/cmd/auth.ts +400 -0
  34. package/src/cli/cmd/cmd.ts +7 -0
  35. package/src/cli/cmd/debug/agent.ts +167 -0
  36. package/src/cli/cmd/debug/config.ts +16 -0
  37. package/src/cli/cmd/debug/file.ts +97 -0
  38. package/src/cli/cmd/debug/index.ts +48 -0
  39. package/src/cli/cmd/debug/lsp.ts +52 -0
  40. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  41. package/src/cli/cmd/debug/scrap.ts +16 -0
  42. package/src/cli/cmd/debug/skill.ts +16 -0
  43. package/src/cli/cmd/debug/snapshot.ts +52 -0
  44. package/src/cli/cmd/export.ts +88 -0
  45. package/src/cli/cmd/generate.ts +38 -0
  46. package/src/cli/cmd/github.ts +1540 -0
  47. package/src/cli/cmd/import.ts +147 -0
  48. package/src/cli/cmd/mcp.ts +765 -0
  49. package/src/cli/cmd/models.ts +77 -0
  50. package/src/cli/cmd/pr.ts +113 -0
  51. package/src/cli/cmd/run.ts +598 -0
  52. package/src/cli/cmd/serve.ts +20 -0
  53. package/src/cli/cmd/session.ts +135 -0
  54. package/src/cli/cmd/stats.ts +426 -0
  55. package/src/cli/cmd/tui/app.tsx +812 -0
  56. package/src/cli/cmd/tui/attach.ts +60 -0
  57. package/src/cli/cmd/tui/component/border.tsx +21 -0
  58. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  59. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  60. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  61. package/src/cli/cmd/tui/component/dialog-model.tsx +165 -0
  62. package/src/cli/cmd/tui/component/dialog-provider.tsx +243 -0
  63. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  64. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  65. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  66. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  67. package/src/cli/cmd/tui/component/dialog-status.tsx +167 -0
  68. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  69. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  70. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  71. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +666 -0
  72. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  73. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  74. package/src/cli/cmd/tui/component/prompt/index.tsx +1153 -0
  75. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  76. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  77. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  78. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  79. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  80. package/src/cli/cmd/tui/context/args.tsx +15 -0
  81. package/src/cli/cmd/tui/context/directory.ts +13 -0
  82. package/src/cli/cmd/tui/context/exit.tsx +54 -0
  83. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  84. package/src/cli/cmd/tui/context/keybind.tsx +100 -0
  85. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  86. package/src/cli/cmd/tui/context/local.tsx +409 -0
  87. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  88. package/src/cli/cmd/tui/context/route.tsx +46 -0
  89. package/src/cli/cmd/tui/context/sdk.tsx +101 -0
  90. package/src/cli/cmd/tui/context/sync.tsx +470 -0
  91. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  92. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  93. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  94. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  95. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  96. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  97. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  98. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  99. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  100. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  101. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  102. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  103. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  104. package/src/cli/cmd/tui/context/theme/innocode.json +245 -0
  105. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  106. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  107. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  108. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  109. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  110. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  111. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  112. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  113. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  114. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  115. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  116. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  117. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  118. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  119. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  120. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  121. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  122. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  123. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  124. package/src/cli/cmd/tui/context/theme.tsx +1154 -0
  125. package/src/cli/cmd/tui/event.ts +48 -0
  126. package/src/cli/cmd/tui/routes/home.tsx +145 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  128. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  129. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  130. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  131. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  132. package/src/cli/cmd/tui/routes/session/header.tsx +135 -0
  133. package/src/cli/cmd/tui/routes/session/index.tsx +2139 -0
  134. package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
  135. package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
  136. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  137. package/src/cli/cmd/tui/thread.ts +188 -0
  138. package/src/cli/cmd/tui/ui/dialog-alert.tsx +59 -0
  139. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +85 -0
  140. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +207 -0
  141. package/src/cli/cmd/tui/ui/dialog-help.tsx +40 -0
  142. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +80 -0
  143. package/src/cli/cmd/tui/ui/dialog-select.tsx +401 -0
  144. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  145. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  146. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  147. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  148. package/src/cli/cmd/tui/util/clipboard.ts +159 -0
  149. package/src/cli/cmd/tui/util/editor.ts +32 -0
  150. package/src/cli/cmd/tui/util/signal.ts +7 -0
  151. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  152. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  153. package/src/cli/cmd/tui/win32.ts +129 -0
  154. package/src/cli/cmd/tui/worker.ts +152 -0
  155. package/src/cli/cmd/uninstall.ts +363 -0
  156. package/src/cli/cmd/upgrade.ts +73 -0
  157. package/src/cli/cmd/web.ts +81 -0
  158. package/src/cli/error.ts +57 -0
  159. package/src/cli/logo.ts +6 -0
  160. package/src/cli/network.ts +60 -0
  161. package/src/cli/ui.ts +113 -0
  162. package/src/cli/upgrade.ts +25 -0
  163. package/src/command/index.ts +150 -0
  164. package/src/command/template/initialize.txt +10 -0
  165. package/src/command/template/review.txt +101 -0
  166. package/src/config/config.ts +1517 -0
  167. package/src/config/markdown.ts +98 -0
  168. package/src/env/index.ts +28 -0
  169. package/src/file/ignore.ts +83 -0
  170. package/src/file/index.ts +583 -0
  171. package/src/file/ripgrep.ts +375 -0
  172. package/src/file/time.ts +69 -0
  173. package/src/file/watcher.ts +127 -0
  174. package/src/flag/flag.ts +148 -0
  175. package/src/format/formatter.ts +366 -0
  176. package/src/format/index.ts +137 -0
  177. package/src/global/index.ts +80 -0
  178. package/src/id/id.ts +83 -0
  179. package/src/ide/index.ts +76 -0
  180. package/src/index.ts +160 -0
  181. package/src/installation/index.ts +268 -0
  182. package/src/lsp/client.ts +252 -0
  183. package/src/lsp/index.ts +485 -0
  184. package/src/lsp/language.ts +119 -0
  185. package/src/lsp/server.ts +2046 -0
  186. package/src/mcp/auth.ts +132 -0
  187. package/src/mcp/index.ts +937 -0
  188. package/src/mcp/oauth-callback.ts +200 -0
  189. package/src/mcp/oauth-provider.ts +154 -0
  190. package/src/patch/index.ts +680 -0
  191. package/src/permission/arity.ts +163 -0
  192. package/src/permission/index.ts +210 -0
  193. package/src/permission/next.ts +280 -0
  194. package/src/plugin/codex.ts +624 -0
  195. package/src/plugin/copilot.ts +327 -0
  196. package/src/plugin/index.ts +138 -0
  197. package/src/project/bootstrap.ts +35 -0
  198. package/src/project/instance.ts +114 -0
  199. package/src/project/project.ts +371 -0
  200. package/src/project/state.ts +70 -0
  201. package/src/project/vcs.ts +76 -0
  202. package/src/provider/auth.ts +147 -0
  203. package/src/provider/error.ts +189 -0
  204. package/src/provider/models.ts +133 -0
  205. package/src/provider/provider.ts +1370 -0
  206. package/src/provider/sdk/copilot/README.md +5 -0
  207. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
  208. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  209. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
  210. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  211. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
  212. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  213. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  214. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
  215. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  216. package/src/provider/sdk/copilot/index.ts +2 -0
  217. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  218. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
  219. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  220. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  221. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  222. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
  223. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
  224. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
  225. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  226. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
  227. package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
  228. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
  229. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
  230. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
  231. package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
  232. package/src/provider/transform.ts +806 -0
  233. package/src/pty/index.ts +286 -0
  234. package/src/question/index.ts +171 -0
  235. package/src/scheduler/index.ts +61 -0
  236. package/src/server/error.ts +36 -0
  237. package/src/server/event.ts +7 -0
  238. package/src/server/mdns.ts +60 -0
  239. package/src/server/routes/config.ts +92 -0
  240. package/src/server/routes/experimental.ts +208 -0
  241. package/src/server/routes/file.ts +197 -0
  242. package/src/server/routes/global.ts +183 -0
  243. package/src/server/routes/mcp.ts +225 -0
  244. package/src/server/routes/permission.ts +68 -0
  245. package/src/server/routes/project.ts +82 -0
  246. package/src/server/routes/provider.ts +179 -0
  247. package/src/server/routes/pty.ts +176 -0
  248. package/src/server/routes/question.ts +98 -0
  249. package/src/server/routes/session.ts +939 -0
  250. package/src/server/routes/tui.ts +379 -0
  251. package/src/server/server.ts +621 -0
  252. package/src/session/compaction.ts +261 -0
  253. package/src/session/index.ts +543 -0
  254. package/src/session/instruction.ts +197 -0
  255. package/src/session/llm.ts +283 -0
  256. package/src/session/message-v2.ts +841 -0
  257. package/src/session/message.ts +189 -0
  258. package/src/session/processor.ts +410 -0
  259. package/src/session/prompt/anthropic-20250930.txt +166 -0
  260. package/src/session/prompt/anthropic.txt +105 -0
  261. package/src/session/prompt/beast.txt +147 -0
  262. package/src/session/prompt/build-switch.txt +5 -0
  263. package/src/session/prompt/codex_header.txt +79 -0
  264. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  265. package/src/session/prompt/gemini.txt +155 -0
  266. package/src/session/prompt/max-steps.txt +16 -0
  267. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  268. package/src/session/prompt/plan.txt +26 -0
  269. package/src/session/prompt/qwen.txt +109 -0
  270. package/src/session/prompt/trinity.txt +97 -0
  271. package/src/session/prompt.ts +1964 -0
  272. package/src/session/retry.ts +101 -0
  273. package/src/session/revert.ts +121 -0
  274. package/src/session/status.ts +76 -0
  275. package/src/session/summary.ts +203 -0
  276. package/src/session/system.ts +54 -0
  277. package/src/session/todo.ts +37 -0
  278. package/src/share/share-next.ts +200 -0
  279. package/src/share/share.ts +92 -0
  280. package/src/shell/shell.ts +67 -0
  281. package/src/skill/discovery.ts +97 -0
  282. package/src/skill/index.ts +1 -0
  283. package/src/skill/skill.ts +188 -0
  284. package/src/snapshot/index.ts +255 -0
  285. package/src/storage/storage.ts +227 -0
  286. package/src/tool/apply_patch.ts +281 -0
  287. package/src/tool/apply_patch.txt +33 -0
  288. package/src/tool/bash.ts +269 -0
  289. package/src/tool/bash.txt +115 -0
  290. package/src/tool/batch.ts +175 -0
  291. package/src/tool/batch.txt +24 -0
  292. package/src/tool/codesearch.ts +132 -0
  293. package/src/tool/codesearch.txt +12 -0
  294. package/src/tool/edit.ts +655 -0
  295. package/src/tool/edit.txt +10 -0
  296. package/src/tool/external-directory.ts +32 -0
  297. package/src/tool/glob.ts +80 -0
  298. package/src/tool/glob.txt +6 -0
  299. package/src/tool/grep.ts +150 -0
  300. package/src/tool/grep.txt +8 -0
  301. package/src/tool/invalid.ts +17 -0
  302. package/src/tool/ls.ts +121 -0
  303. package/src/tool/ls.txt +1 -0
  304. package/src/tool/lsp.ts +96 -0
  305. package/src/tool/lsp.txt +19 -0
  306. package/src/tool/multiedit.ts +46 -0
  307. package/src/tool/multiedit.txt +41 -0
  308. package/src/tool/plan-enter.txt +14 -0
  309. package/src/tool/plan-exit.txt +13 -0
  310. package/src/tool/plan.ts +130 -0
  311. package/src/tool/question.ts +33 -0
  312. package/src/tool/question.txt +10 -0
  313. package/src/tool/read.ts +261 -0
  314. package/src/tool/read.txt +14 -0
  315. package/src/tool/registry.ts +160 -0
  316. package/src/tool/skill.ts +123 -0
  317. package/src/tool/task.ts +165 -0
  318. package/src/tool/task.txt +60 -0
  319. package/src/tool/todo.ts +53 -0
  320. package/src/tool/todoread.txt +14 -0
  321. package/src/tool/todowrite.txt +167 -0
  322. package/src/tool/tool.ts +89 -0
  323. package/src/tool/truncation.ts +106 -0
  324. package/src/tool/webfetch.ts +186 -0
  325. package/src/tool/webfetch.txt +13 -0
  326. package/src/tool/websearch.ts +150 -0
  327. package/src/tool/websearch.txt +14 -0
  328. package/src/tool/write.ts +85 -0
  329. package/src/tool/write.txt +8 -0
  330. package/src/util/abort.ts +35 -0
  331. package/src/util/archive.ts +16 -0
  332. package/src/util/color.ts +19 -0
  333. package/src/util/context.ts +25 -0
  334. package/src/util/defer.ts +12 -0
  335. package/src/util/eventloop.ts +20 -0
  336. package/src/util/filesystem.ts +93 -0
  337. package/src/util/fn.ts +11 -0
  338. package/src/util/format.ts +20 -0
  339. package/src/util/iife.ts +3 -0
  340. package/src/util/keybind.ts +103 -0
  341. package/src/util/lazy.ts +18 -0
  342. package/src/util/locale.ts +81 -0
  343. package/src/util/lock.ts +98 -0
  344. package/src/util/log.ts +180 -0
  345. package/src/util/proxied.ts +3 -0
  346. package/src/util/queue.ts +32 -0
  347. package/src/util/rpc.ts +66 -0
  348. package/src/util/scrap.ts +10 -0
  349. package/src/util/signal.ts +12 -0
  350. package/src/util/timeout.ts +14 -0
  351. package/src/util/token.ts +7 -0
  352. package/src/util/wildcard.ts +56 -0
  353. package/src/worktree/index.ts +612 -0
  354. package/sst-env.d.ts +9 -0
  355. package/test/acp/agent-interface.test.ts +51 -0
  356. package/test/acp/event-subscription.test.ts +436 -0
  357. package/test/agent/agent.test.ts +675 -0
  358. package/test/bun.test.ts +53 -0
  359. package/test/cli/github-action.test.ts +161 -0
  360. package/test/cli/github-remote.test.ts +80 -0
  361. package/test/cli/import.test.ts +38 -0
  362. package/test/cli/tui/transcript.test.ts +322 -0
  363. package/test/config/agent-color.test.ts +71 -0
  364. package/test/config/config.test.ts +1802 -0
  365. package/test/config/fixtures/empty-frontmatter.md +4 -0
  366. package/test/config/fixtures/frontmatter.md +28 -0
  367. package/test/config/fixtures/markdown-header.md +11 -0
  368. package/test/config/fixtures/no-frontmatter.md +1 -0
  369. package/test/config/fixtures/weird-model-id.md +13 -0
  370. package/test/config/markdown.test.ts +228 -0
  371. package/test/file/ignore.test.ts +10 -0
  372. package/test/file/path-traversal.test.ts +198 -0
  373. package/test/file/ripgrep.test.ts +39 -0
  374. package/test/fixture/fixture.ts +45 -0
  375. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  376. package/test/ide/ide.test.ts +82 -0
  377. package/test/keybind.test.ts +421 -0
  378. package/test/lsp/client.test.ts +95 -0
  379. package/test/mcp/headers.test.ts +153 -0
  380. package/test/mcp/oauth-browser.test.ts +249 -0
  381. package/test/memory/abort-leak.test.ts +136 -0
  382. package/test/patch/patch.test.ts +348 -0
  383. package/test/permission/arity.test.ts +33 -0
  384. package/test/permission/next.test.ts +690 -0
  385. package/test/permission-task.test.ts +319 -0
  386. package/test/plugin/auth-override.test.ts +44 -0
  387. package/test/plugin/codex.test.ts +123 -0
  388. package/test/preload.ts +63 -0
  389. package/test/project/project.test.ts +120 -0
  390. package/test/provider/amazon-bedrock.test.ts +445 -0
  391. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  392. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  393. package/test/provider/gitlab-duo.test.ts +262 -0
  394. package/test/provider/provider.test.ts +2129 -0
  395. package/test/provider/transform.test.ts +1928 -0
  396. package/test/question/question.test.ts +300 -0
  397. package/test/scheduler.test.ts +73 -0
  398. package/test/server/session-list.test.ts +39 -0
  399. package/test/server/session-select.test.ts +78 -0
  400. package/test/session/compaction.test.ts +423 -0
  401. package/test/session/instruction.test.ts +170 -0
  402. package/test/session/llm.test.ts +667 -0
  403. package/test/session/message-v2.test.ts +924 -0
  404. package/test/session/prompt-missing-file.test.ts +53 -0
  405. package/test/session/prompt-special-chars.test.ts +56 -0
  406. package/test/session/prompt-variant.test.ts +68 -0
  407. package/test/session/retry.test.ts +188 -0
  408. package/test/session/revert-compact.test.ts +285 -0
  409. package/test/session/session.test.ts +71 -0
  410. package/test/session/structured-output-integration.test.ts +233 -0
  411. package/test/session/structured-output.test.ts +385 -0
  412. package/test/skill/discovery.test.ts +60 -0
  413. package/test/skill/skill.test.ts +388 -0
  414. package/test/snapshot/snapshot.test.ts +1040 -0
  415. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  416. package/test/tool/apply_patch.test.ts +559 -0
  417. package/test/tool/bash.test.ts +399 -0
  418. package/test/tool/external-directory.test.ts +127 -0
  419. package/test/tool/fixtures/large-image.png +0 -0
  420. package/test/tool/fixtures/models-api.json +38413 -0
  421. package/test/tool/grep.test.ts +110 -0
  422. package/test/tool/question.test.ts +107 -0
  423. package/test/tool/read.test.ts +421 -0
  424. package/test/tool/registry.test.ts +122 -0
  425. package/test/tool/skill.test.ts +112 -0
  426. package/test/tool/truncation.test.ts +159 -0
  427. package/test/util/filesystem.test.ts +39 -0
  428. package/test/util/format.test.ts +59 -0
  429. package/test/util/iife.test.ts +36 -0
  430. package/test/util/lazy.test.ts +50 -0
  431. package/test/util/lock.test.ts +72 -0
  432. package/test/util/timeout.test.ts +21 -0
  433. package/test/util/wildcard.test.ts +75 -0
  434. package/tsconfig.json +16 -0
@@ -0,0 +1,612 @@
1
+ import { $ } from "bun"
2
+ import fs from "fs/promises"
3
+ import path from "path"
4
+ import z from "zod"
5
+ import { NamedError } from "@opencode-ai/util/error"
6
+ import { Global } from "../global"
7
+ import { Instance } from "../project/instance"
8
+ import { InstanceBootstrap } from "../project/bootstrap"
9
+ import { Project } from "../project/project"
10
+ import { Storage } from "../storage/storage"
11
+ import { fn } from "../util/fn"
12
+ import { Log } from "../util/log"
13
+ import { BusEvent } from "@/bus/bus-event"
14
+ import { GlobalBus } from "@/bus/global"
15
+
16
+ export namespace Worktree {
17
+ const log = Log.create({ service: "worktree" })
18
+
19
+ export const Event = {
20
+ Ready: BusEvent.define(
21
+ "worktree.ready",
22
+ z.object({
23
+ name: z.string(),
24
+ branch: z.string(),
25
+ }),
26
+ ),
27
+ Failed: BusEvent.define(
28
+ "worktree.failed",
29
+ z.object({
30
+ message: z.string(),
31
+ }),
32
+ ),
33
+ }
34
+
35
+ export const Info = z
36
+ .object({
37
+ name: z.string(),
38
+ branch: z.string(),
39
+ directory: z.string(),
40
+ })
41
+ .meta({
42
+ ref: "Worktree",
43
+ })
44
+
45
+ export type Info = z.infer<typeof Info>
46
+
47
+ export const CreateInput = z
48
+ .object({
49
+ name: z.string().optional(),
50
+ startCommand: z
51
+ .string()
52
+ .optional()
53
+ .describe("Additional startup script to run after the project's start command"),
54
+ })
55
+ .meta({
56
+ ref: "WorktreeCreateInput",
57
+ })
58
+
59
+ export type CreateInput = z.infer<typeof CreateInput>
60
+
61
+ export const RemoveInput = z
62
+ .object({
63
+ directory: z.string(),
64
+ })
65
+ .meta({
66
+ ref: "WorktreeRemoveInput",
67
+ })
68
+
69
+ export type RemoveInput = z.infer<typeof RemoveInput>
70
+
71
+ export const ResetInput = z
72
+ .object({
73
+ directory: z.string(),
74
+ })
75
+ .meta({
76
+ ref: "WorktreeResetInput",
77
+ })
78
+
79
+ export type ResetInput = z.infer<typeof ResetInput>
80
+
81
+ export const NotGitError = NamedError.create(
82
+ "WorktreeNotGitError",
83
+ z.object({
84
+ message: z.string(),
85
+ }),
86
+ )
87
+
88
+ export const NameGenerationFailedError = NamedError.create(
89
+ "WorktreeNameGenerationFailedError",
90
+ z.object({
91
+ message: z.string(),
92
+ }),
93
+ )
94
+
95
+ export const CreateFailedError = NamedError.create(
96
+ "WorktreeCreateFailedError",
97
+ z.object({
98
+ message: z.string(),
99
+ }),
100
+ )
101
+
102
+ export const StartCommandFailedError = NamedError.create(
103
+ "WorktreeStartCommandFailedError",
104
+ z.object({
105
+ message: z.string(),
106
+ }),
107
+ )
108
+
109
+ export const RemoveFailedError = NamedError.create(
110
+ "WorktreeRemoveFailedError",
111
+ z.object({
112
+ message: z.string(),
113
+ }),
114
+ )
115
+
116
+ export const ResetFailedError = NamedError.create(
117
+ "WorktreeResetFailedError",
118
+ z.object({
119
+ message: z.string(),
120
+ }),
121
+ )
122
+
123
+ const ADJECTIVES = [
124
+ "brave",
125
+ "calm",
126
+ "clever",
127
+ "cosmic",
128
+ "crisp",
129
+ "curious",
130
+ "eager",
131
+ "gentle",
132
+ "glowing",
133
+ "happy",
134
+ "hidden",
135
+ "jolly",
136
+ "kind",
137
+ "lucky",
138
+ "mighty",
139
+ "misty",
140
+ "neon",
141
+ "nimble",
142
+ "playful",
143
+ "proud",
144
+ "quick",
145
+ "quiet",
146
+ "shiny",
147
+ "silent",
148
+ "stellar",
149
+ "sunny",
150
+ "swift",
151
+ "tidy",
152
+ "witty",
153
+ ] as const
154
+
155
+ const NOUNS = [
156
+ "cabin",
157
+ "cactus",
158
+ "canyon",
159
+ "circuit",
160
+ "comet",
161
+ "eagle",
162
+ "engine",
163
+ "falcon",
164
+ "forest",
165
+ "garden",
166
+ "harbor",
167
+ "island",
168
+ "knight",
169
+ "lagoon",
170
+ "meadow",
171
+ "moon",
172
+ "mountain",
173
+ "nebula",
174
+ "orchid",
175
+ "otter",
176
+ "panda",
177
+ "pixel",
178
+ "planet",
179
+ "river",
180
+ "rocket",
181
+ "sailor",
182
+ "squid",
183
+ "star",
184
+ "tiger",
185
+ "wizard",
186
+ "wolf",
187
+ ] as const
188
+
189
+ function pick<const T extends readonly string[]>(list: T) {
190
+ return list[Math.floor(Math.random() * list.length)]
191
+ }
192
+
193
+ function slug(input: string) {
194
+ return input
195
+ .trim()
196
+ .toLowerCase()
197
+ .replace(/[^a-z0-9]+/g, "-")
198
+ .replace(/^-+/, "")
199
+ .replace(/-+$/, "")
200
+ }
201
+
202
+ function randomName() {
203
+ return `${pick(ADJECTIVES)}-${pick(NOUNS)}`
204
+ }
205
+
206
+ async function exists(target: string) {
207
+ return fs
208
+ .stat(target)
209
+ .then(() => true)
210
+ .catch(() => false)
211
+ }
212
+
213
+ function outputText(input: Uint8Array | undefined) {
214
+ if (!input?.length) return ""
215
+ return new TextDecoder().decode(input).trim()
216
+ }
217
+
218
+ function errorText(result: { stdout?: Uint8Array; stderr?: Uint8Array }) {
219
+ return [outputText(result.stderr), outputText(result.stdout)].filter(Boolean).join("\n")
220
+ }
221
+
222
+ function failed(result: { stdout?: Uint8Array; stderr?: Uint8Array }) {
223
+ return [outputText(result.stderr), outputText(result.stdout)].filter(Boolean).flatMap((chunk) =>
224
+ chunk
225
+ .split("\n")
226
+ .map((line) => line.trim())
227
+ .flatMap((line) => {
228
+ const match = line.match(/^warning:\s+failed to remove\s+(.+):\s+/i)
229
+ if (!match) return []
230
+ const value = match[1]?.trim().replace(/^['"]|['"]$/g, "")
231
+ if (!value) return []
232
+ return [value]
233
+ }),
234
+ )
235
+ }
236
+
237
+ async function prune(root: string, entries: string[]) {
238
+ const base = await canonical(root)
239
+ await Promise.all(
240
+ entries.map(async (entry) => {
241
+ const target = await canonical(path.resolve(root, entry))
242
+ if (target === base) return
243
+ if (!target.startsWith(`${base}${path.sep}`)) return
244
+ await fs.rm(target, { recursive: true, force: true }).catch(() => undefined)
245
+ }),
246
+ )
247
+ }
248
+
249
+ async function sweep(root: string) {
250
+ const first = await $`git clean -ffdx`.quiet().nothrow().cwd(root)
251
+ if (first.exitCode === 0) return first
252
+
253
+ const entries = failed(first)
254
+ if (!entries.length) return first
255
+
256
+ await prune(root, entries)
257
+ return $`git clean -ffdx`.quiet().nothrow().cwd(root)
258
+ }
259
+
260
+ async function canonical(input: string) {
261
+ const abs = path.resolve(input)
262
+ const real = await fs.realpath(abs).catch(() => abs)
263
+ const normalized = path.normalize(real)
264
+ return process.platform === "win32" ? normalized.toLowerCase() : normalized
265
+ }
266
+
267
+ async function candidate(root: string, base?: string) {
268
+ for (const attempt of Array.from({ length: 26 }, (_, i) => i)) {
269
+ const name = base ? (attempt === 0 ? base : `${base}-${randomName()}`) : randomName()
270
+ const branch = `opencode/${name}`
271
+ const directory = path.join(root, name)
272
+
273
+ if (await exists(directory)) continue
274
+
275
+ const ref = `refs/heads/${branch}`
276
+ const branchCheck = await $`git show-ref --verify --quiet ${ref}`.quiet().nothrow().cwd(Instance.worktree)
277
+ if (branchCheck.exitCode === 0) continue
278
+
279
+ return Info.parse({ name, branch, directory })
280
+ }
281
+
282
+ throw new NameGenerationFailedError({ message: "Failed to generate a unique worktree name" })
283
+ }
284
+
285
+ async function runStartCommand(directory: string, cmd: string) {
286
+ if (process.platform === "win32") {
287
+ return $`cmd /c ${cmd}`.nothrow().cwd(directory)
288
+ }
289
+ return $`bash -lc ${cmd}`.nothrow().cwd(directory)
290
+ }
291
+
292
+ type StartKind = "project" | "worktree"
293
+
294
+ async function runStartScript(directory: string, cmd: string, kind: StartKind) {
295
+ const text = cmd.trim()
296
+ if (!text) return true
297
+
298
+ const ran = await runStartCommand(directory, text)
299
+ if (ran.exitCode === 0) return true
300
+
301
+ log.error("worktree start command failed", {
302
+ kind,
303
+ directory,
304
+ message: errorText(ran),
305
+ })
306
+ return false
307
+ }
308
+
309
+ async function runStartScripts(directory: string, input: { projectID: string; extra?: string }) {
310
+ const project = await Storage.read<Project.Info>(["project", input.projectID]).catch(() => undefined)
311
+ const startup = project?.commands?.start?.trim() ?? ""
312
+ const ok = await runStartScript(directory, startup, "project")
313
+ if (!ok) return false
314
+
315
+ const extra = input.extra ?? ""
316
+ await runStartScript(directory, extra, "worktree")
317
+ return true
318
+ }
319
+
320
+ function queueStartScripts(directory: string, input: { projectID: string; extra?: string }) {
321
+ setTimeout(() => {
322
+ const start = async () => {
323
+ await runStartScripts(directory, input)
324
+ }
325
+
326
+ void start().catch((error) => {
327
+ log.error("worktree start task failed", { directory, error })
328
+ })
329
+ }, 0)
330
+ }
331
+
332
+ export const create = fn(CreateInput.optional(), async (input) => {
333
+ if (Instance.project.vcs !== "git") {
334
+ throw new NotGitError({ message: "Worktrees are only supported for git projects" })
335
+ }
336
+
337
+ const root = path.join(Global.Path.data, "worktree", Instance.project.id)
338
+ await fs.mkdir(root, { recursive: true })
339
+
340
+ const base = input?.name ? slug(input.name) : ""
341
+ const info = await candidate(root, base || undefined)
342
+
343
+ const created = await $`git worktree add --no-checkout -b ${info.branch} ${info.directory}`
344
+ .quiet()
345
+ .nothrow()
346
+ .cwd(Instance.worktree)
347
+ if (created.exitCode !== 0) {
348
+ throw new CreateFailedError({ message: errorText(created) || "Failed to create git worktree" })
349
+ }
350
+
351
+ await Project.addSandbox(Instance.project.id, info.directory).catch(() => undefined)
352
+
353
+ const projectID = Instance.project.id
354
+ const extra = input?.startCommand?.trim()
355
+ setTimeout(() => {
356
+ const start = async () => {
357
+ const populated = await $`git reset --hard`.quiet().nothrow().cwd(info.directory)
358
+ if (populated.exitCode !== 0) {
359
+ const message = errorText(populated) || "Failed to populate worktree"
360
+ log.error("worktree checkout failed", { directory: info.directory, message })
361
+ GlobalBus.emit("event", {
362
+ directory: info.directory,
363
+ payload: {
364
+ type: Event.Failed.type,
365
+ properties: {
366
+ message,
367
+ },
368
+ },
369
+ })
370
+ return
371
+ }
372
+
373
+ const booted = await Instance.provide({
374
+ directory: info.directory,
375
+ init: InstanceBootstrap,
376
+ fn: () => undefined,
377
+ })
378
+ .then(() => true)
379
+ .catch((error) => {
380
+ const message = error instanceof Error ? error.message : String(error)
381
+ log.error("worktree bootstrap failed", { directory: info.directory, message })
382
+ GlobalBus.emit("event", {
383
+ directory: info.directory,
384
+ payload: {
385
+ type: Event.Failed.type,
386
+ properties: {
387
+ message,
388
+ },
389
+ },
390
+ })
391
+ return false
392
+ })
393
+ if (!booted) return
394
+
395
+ GlobalBus.emit("event", {
396
+ directory: info.directory,
397
+ payload: {
398
+ type: Event.Ready.type,
399
+ properties: {
400
+ name: info.name,
401
+ branch: info.branch,
402
+ },
403
+ },
404
+ })
405
+
406
+ await runStartScripts(info.directory, { projectID, extra })
407
+ }
408
+
409
+ void start().catch((error) => {
410
+ log.error("worktree start task failed", { directory: info.directory, error })
411
+ })
412
+ }, 0)
413
+
414
+ return info
415
+ })
416
+
417
+ export const remove = fn(RemoveInput, async (input) => {
418
+ if (Instance.project.vcs !== "git") {
419
+ throw new NotGitError({ message: "Worktrees are only supported for git projects" })
420
+ }
421
+
422
+ const directory = await canonical(input.directory)
423
+ const list = await $`git worktree list --porcelain`.quiet().nothrow().cwd(Instance.worktree)
424
+ if (list.exitCode !== 0) {
425
+ throw new RemoveFailedError({ message: errorText(list) || "Failed to read git worktrees" })
426
+ }
427
+
428
+ const lines = outputText(list.stdout)
429
+ .split("\n")
430
+ .map((line) => line.trim())
431
+ const entries = lines.reduce<{ path?: string; branch?: string }[]>((acc, line) => {
432
+ if (!line) return acc
433
+ if (line.startsWith("worktree ")) {
434
+ acc.push({ path: line.slice("worktree ".length).trim() })
435
+ return acc
436
+ }
437
+ const current = acc[acc.length - 1]
438
+ if (!current) return acc
439
+ if (line.startsWith("branch ")) {
440
+ current.branch = line.slice("branch ".length).trim()
441
+ }
442
+ return acc
443
+ }, [])
444
+
445
+ const entry = await (async () => {
446
+ for (const item of entries) {
447
+ if (!item.path) continue
448
+ const key = await canonical(item.path)
449
+ if (key === directory) return item
450
+ }
451
+ })()
452
+
453
+ if (!entry?.path) {
454
+ const directoryExists = await exists(directory)
455
+ if (directoryExists) {
456
+ await fs.rm(directory, { recursive: true, force: true })
457
+ }
458
+ return true
459
+ }
460
+
461
+ const removed = await $`git worktree remove --force ${entry.path}`.quiet().nothrow().cwd(Instance.worktree)
462
+ if (removed.exitCode !== 0) {
463
+ throw new RemoveFailedError({ message: errorText(removed) || "Failed to remove git worktree" })
464
+ }
465
+
466
+ const branch = entry.branch?.replace(/^refs\/heads\//, "")
467
+ if (branch) {
468
+ const deleted = await $`git branch -D ${branch}`.quiet().nothrow().cwd(Instance.worktree)
469
+ if (deleted.exitCode !== 0) {
470
+ throw new RemoveFailedError({ message: errorText(deleted) || "Failed to delete worktree branch" })
471
+ }
472
+ }
473
+
474
+ return true
475
+ })
476
+
477
+ export const reset = fn(ResetInput, async (input) => {
478
+ if (Instance.project.vcs !== "git") {
479
+ throw new NotGitError({ message: "Worktrees are only supported for git projects" })
480
+ }
481
+
482
+ const directory = await canonical(input.directory)
483
+ const primary = await canonical(Instance.worktree)
484
+ if (directory === primary) {
485
+ throw new ResetFailedError({ message: "Cannot reset the primary workspace" })
486
+ }
487
+
488
+ const list = await $`git worktree list --porcelain`.quiet().nothrow().cwd(Instance.worktree)
489
+ if (list.exitCode !== 0) {
490
+ throw new ResetFailedError({ message: errorText(list) || "Failed to read git worktrees" })
491
+ }
492
+
493
+ const lines = outputText(list.stdout)
494
+ .split("\n")
495
+ .map((line) => line.trim())
496
+ const entries = lines.reduce<{ path?: string; branch?: string }[]>((acc, line) => {
497
+ if (!line) return acc
498
+ if (line.startsWith("worktree ")) {
499
+ acc.push({ path: line.slice("worktree ".length).trim() })
500
+ return acc
501
+ }
502
+ const current = acc[acc.length - 1]
503
+ if (!current) return acc
504
+ if (line.startsWith("branch ")) {
505
+ current.branch = line.slice("branch ".length).trim()
506
+ }
507
+ return acc
508
+ }, [])
509
+
510
+ const entry = await (async () => {
511
+ for (const item of entries) {
512
+ if (!item.path) continue
513
+ const key = await canonical(item.path)
514
+ if (key === directory) return item
515
+ }
516
+ })()
517
+ if (!entry?.path) {
518
+ throw new ResetFailedError({ message: "Worktree not found" })
519
+ }
520
+
521
+ const remoteList = await $`git remote`.quiet().nothrow().cwd(Instance.worktree)
522
+ if (remoteList.exitCode !== 0) {
523
+ throw new ResetFailedError({ message: errorText(remoteList) || "Failed to list git remotes" })
524
+ }
525
+
526
+ const remotes = outputText(remoteList.stdout)
527
+ .split("\n")
528
+ .map((line) => line.trim())
529
+ .filter(Boolean)
530
+
531
+ const remote = remotes.includes("origin")
532
+ ? "origin"
533
+ : remotes.length === 1
534
+ ? remotes[0]
535
+ : remotes.includes("upstream")
536
+ ? "upstream"
537
+ : ""
538
+
539
+ const remoteHead = remote
540
+ ? await $`git symbolic-ref refs/remotes/${remote}/HEAD`.quiet().nothrow().cwd(Instance.worktree)
541
+ : { exitCode: 1, stdout: undefined, stderr: undefined }
542
+
543
+ const remoteRef = remoteHead.exitCode === 0 ? outputText(remoteHead.stdout) : ""
544
+ const remoteTarget = remoteRef ? remoteRef.replace(/^refs\/remotes\//, "") : ""
545
+ const remoteBranch = remote && remoteTarget.startsWith(`${remote}/`) ? remoteTarget.slice(`${remote}/`.length) : ""
546
+
547
+ const mainCheck = await $`git show-ref --verify --quiet refs/heads/main`.quiet().nothrow().cwd(Instance.worktree)
548
+ const masterCheck = await $`git show-ref --verify --quiet refs/heads/master`
549
+ .quiet()
550
+ .nothrow()
551
+ .cwd(Instance.worktree)
552
+ const localBranch = mainCheck.exitCode === 0 ? "main" : masterCheck.exitCode === 0 ? "master" : ""
553
+
554
+ const target = remoteBranch ? `${remote}/${remoteBranch}` : localBranch
555
+ if (!target) {
556
+ throw new ResetFailedError({ message: "Default branch not found" })
557
+ }
558
+
559
+ if (remoteBranch) {
560
+ const fetch = await $`git fetch ${remote} ${remoteBranch}`.quiet().nothrow().cwd(Instance.worktree)
561
+ if (fetch.exitCode !== 0) {
562
+ throw new ResetFailedError({ message: errorText(fetch) || `Failed to fetch ${target}` })
563
+ }
564
+ }
565
+
566
+ if (!entry.path) {
567
+ throw new ResetFailedError({ message: "Worktree path not found" })
568
+ }
569
+
570
+ const worktreePath = entry.path
571
+
572
+ const resetToTarget = await $`git reset --hard ${target}`.quiet().nothrow().cwd(worktreePath)
573
+ if (resetToTarget.exitCode !== 0) {
574
+ throw new ResetFailedError({ message: errorText(resetToTarget) || "Failed to reset worktree to target" })
575
+ }
576
+
577
+ const clean = await sweep(worktreePath)
578
+ if (clean.exitCode !== 0) {
579
+ throw new ResetFailedError({ message: errorText(clean) || "Failed to clean worktree" })
580
+ }
581
+
582
+ const update = await $`git submodule update --init --recursive --force`.quiet().nothrow().cwd(worktreePath)
583
+ if (update.exitCode !== 0) {
584
+ throw new ResetFailedError({ message: errorText(update) || "Failed to update submodules" })
585
+ }
586
+
587
+ const subReset = await $`git submodule foreach --recursive git reset --hard`.quiet().nothrow().cwd(worktreePath)
588
+ if (subReset.exitCode !== 0) {
589
+ throw new ResetFailedError({ message: errorText(subReset) || "Failed to reset submodules" })
590
+ }
591
+
592
+ const subClean = await $`git submodule foreach --recursive git clean -fdx`.quiet().nothrow().cwd(worktreePath)
593
+ if (subClean.exitCode !== 0) {
594
+ throw new ResetFailedError({ message: errorText(subClean) || "Failed to clean submodules" })
595
+ }
596
+
597
+ const status = await $`git status --porcelain=v1`.quiet().nothrow().cwd(worktreePath)
598
+ if (status.exitCode !== 0) {
599
+ throw new ResetFailedError({ message: errorText(status) || "Failed to read git status" })
600
+ }
601
+
602
+ const dirty = outputText(status.stdout)
603
+ if (dirty) {
604
+ throw new ResetFailedError({ message: `Worktree reset left local changes:\n${dirty}` })
605
+ }
606
+
607
+ const projectID = Instance.project.id
608
+ queueStartScripts(worktreePath, { projectID })
609
+
610
+ return true
611
+ })
612
+ }
package/sst-env.d.ts ADDED
@@ -0,0 +1,9 @@
1
+ /* This file is auto-generated by SST. Do not edit. */
2
+ /* tslint:disable */
3
+ /* eslint-disable */
4
+ /* deno-fmt-ignore-file */
5
+
6
+ /// <reference path="../../sst-env.d.ts" />
7
+
8
+ import "sst"
9
+ export {}
@@ -0,0 +1,51 @@
1
+ import { describe, expect, test } from "bun:test"
2
+ import { ACP } from "../../src/acp/agent"
3
+ import type { Agent as ACPAgent } from "@agentclientprotocol/sdk"
4
+
5
+ /**
6
+ * Type-level test: This line will fail to compile if ACP.Agent
7
+ * doesn't properly implement the ACPAgent interface.
8
+ *
9
+ * The SDK checks for methods like `agent.unstable_setSessionModel` at runtime
10
+ * and throws "Method not found" if they're missing. TypeScript allows optional
11
+ * interface methods to be omitted, but the SDK still expects them.
12
+ *
13
+ * @see https://github.com/agentclientprotocol/typescript-sdk/commit/7072d3f
14
+ */
15
+ type _AssertAgentImplementsACPAgent = ACP.Agent extends ACPAgent ? true : never
16
+ const _typeCheck: _AssertAgentImplementsACPAgent = true
17
+
18
+ /**
19
+ * Runtime verification that optional methods the SDK expects are actually implemented.
20
+ * The SDK's router checks `if (!agent.methodName)` and throws MethodNotFound if missing.
21
+ */
22
+ describe("acp.agent interface compliance", () => {
23
+ // Extract method names from the ACPAgent interface type
24
+ type ACPAgentMethods = keyof ACPAgent
25
+
26
+ // Methods that the SDK's router explicitly checks for at runtime
27
+ const sdkCheckedMethods: ACPAgentMethods[] = [
28
+ // Required
29
+ "initialize",
30
+ "newSession",
31
+ "prompt",
32
+ "cancel",
33
+ // Optional but checked by SDK router
34
+ "loadSession",
35
+ "setSessionMode",
36
+ "authenticate",
37
+ // Unstable - SDK checks these with unstable_ prefix
38
+ "unstable_listSessions",
39
+ "unstable_forkSession",
40
+ "unstable_resumeSession",
41
+ "unstable_setSessionModel",
42
+ ]
43
+
44
+ test("Agent implements all SDK-checked methods", () => {
45
+ for (const method of sdkCheckedMethods) {
46
+ expect(typeof ACP.Agent.prototype[method as keyof typeof ACP.Agent.prototype], `Missing method: ${method}`).toBe(
47
+ "function",
48
+ )
49
+ }
50
+ })
51
+ })