opencode-v2 1.1.53

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 (439) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/opencode +84 -0
  5. package/bunfig.toml +5 -0
  6. package/package.json +126 -0
  7. package/parsers-config.ts +253 -0
  8. package/script/build.ts +193 -0
  9. package/script/postinstall.mjs +125 -0
  10. package/script/publish.ts +181 -0
  11. package/script/schema.ts +47 -0
  12. package/script/seed-e2e.ts +50 -0
  13. package/src/acp/README.md +164 -0
  14. package/src/acp/agent.ts +1676 -0
  15. package/src/acp/session.ts +117 -0
  16. package/src/acp/types.ts +23 -0
  17. package/src/agent/agent.ts +414 -0
  18. package/src/agent/generate.txt +75 -0
  19. package/src/agent/prompt/compaction.txt +12 -0
  20. package/src/agent/prompt/explore.txt +18 -0
  21. package/src/agent/prompt/summary.txt +11 -0
  22. package/src/agent/prompt/title.txt +44 -0
  23. package/src/auth/index.ts +70 -0
  24. package/src/bun/index.ts +137 -0
  25. package/src/bun/registry.ts +48 -0
  26. package/src/bus/bus-event.ts +43 -0
  27. package/src/bus/global.ts +10 -0
  28. package/src/bus/index.ts +105 -0
  29. package/src/cli/bootstrap.ts +17 -0
  30. package/src/cli/cmd/acp.ts +70 -0
  31. package/src/cli/cmd/agent.ts +257 -0
  32. package/src/cli/cmd/auth.ts +400 -0
  33. package/src/cli/cmd/cmd.ts +7 -0
  34. package/src/cli/cmd/debug/agent.ts +167 -0
  35. package/src/cli/cmd/debug/config.ts +16 -0
  36. package/src/cli/cmd/debug/file.ts +97 -0
  37. package/src/cli/cmd/debug/index.ts +48 -0
  38. package/src/cli/cmd/debug/lsp.ts +52 -0
  39. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  40. package/src/cli/cmd/debug/scrap.ts +16 -0
  41. package/src/cli/cmd/debug/skill.ts +16 -0
  42. package/src/cli/cmd/debug/snapshot.ts +52 -0
  43. package/src/cli/cmd/export.ts +88 -0
  44. package/src/cli/cmd/generate.ts +38 -0
  45. package/src/cli/cmd/github.ts +1540 -0
  46. package/src/cli/cmd/import.ts +147 -0
  47. package/src/cli/cmd/mcp.ts +755 -0
  48. package/src/cli/cmd/models.ts +77 -0
  49. package/src/cli/cmd/pr.ts +112 -0
  50. package/src/cli/cmd/run.ts +617 -0
  51. package/src/cli/cmd/serve.ts +20 -0
  52. package/src/cli/cmd/session.ts +135 -0
  53. package/src/cli/cmd/stats.ts +426 -0
  54. package/src/cli/cmd/tui/app.tsx +801 -0
  55. package/src/cli/cmd/tui/attach.ts +52 -0
  56. package/src/cli/cmd/tui/component/border.tsx +21 -0
  57. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  58. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  59. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  60. package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
  61. package/src/cli/cmd/tui/component/dialog-provider.tsx +266 -0
  62. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  63. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  64. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  65. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  66. package/src/cli/cmd/tui/component/dialog-status.tsx +177 -0
  67. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  68. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  69. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  70. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +666 -0
  71. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  72. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  73. package/src/cli/cmd/tui/component/prompt/index.tsx +1132 -0
  74. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  75. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  76. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  77. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  78. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  79. package/src/cli/cmd/tui/context/args.tsx +15 -0
  80. package/src/cli/cmd/tui/context/directory.ts +13 -0
  81. package/src/cli/cmd/tui/context/exit.tsx +52 -0
  82. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  83. package/src/cli/cmd/tui/context/keybind.tsx +100 -0
  84. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  85. package/src/cli/cmd/tui/context/local.tsx +409 -0
  86. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  87. package/src/cli/cmd/tui/context/route.tsx +46 -0
  88. package/src/cli/cmd/tui/context/sdk.tsx +101 -0
  89. package/src/cli/cmd/tui/context/sync.tsx +470 -0
  90. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  91. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  92. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  93. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  94. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  95. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  96. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  97. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  98. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  99. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  100. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  101. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  102. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  103. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  104. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  105. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  106. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  107. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  108. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  109. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  110. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  111. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  112. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  113. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  114. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  115. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  116. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  117. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  118. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  119. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  120. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  121. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  122. package/src/cli/cmd/tui/context/theme.tsx +1152 -0
  123. package/src/cli/cmd/tui/event.ts +48 -0
  124. package/src/cli/cmd/tui/routes/home.tsx +140 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  126. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  128. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  129. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  130. package/src/cli/cmd/tui/routes/session/header.tsx +142 -0
  131. package/src/cli/cmd/tui/routes/session/index.tsx +2126 -0
  132. package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
  133. package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
  134. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  135. package/src/cli/cmd/tui/thread.ts +175 -0
  136. package/src/cli/cmd/tui/ui/dialog-alert.tsx +68 -0
  137. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +93 -0
  138. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +215 -0
  139. package/src/cli/cmd/tui/ui/dialog-help.tsx +49 -0
  140. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +88 -0
  141. package/src/cli/cmd/tui/ui/dialog-select.tsx +399 -0
  142. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  143. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  144. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  145. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  146. package/src/cli/cmd/tui/util/clipboard.ts +159 -0
  147. package/src/cli/cmd/tui/util/editor.ts +32 -0
  148. package/src/cli/cmd/tui/util/signal.ts +7 -0
  149. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  150. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  151. package/src/cli/cmd/tui/worker.ts +152 -0
  152. package/src/cli/cmd/uninstall.ts +357 -0
  153. package/src/cli/cmd/upgrade.ts +73 -0
  154. package/src/cli/cmd/web.ts +81 -0
  155. package/src/cli/error.ts +57 -0
  156. package/src/cli/logo.ts +6 -0
  157. package/src/cli/network.ts +60 -0
  158. package/src/cli/ui.ts +113 -0
  159. package/src/cli/upgrade.ts +25 -0
  160. package/src/command/index.ts +150 -0
  161. package/src/command/template/initialize.txt +10 -0
  162. package/src/command/template/review.txt +99 -0
  163. package/src/config/config.ts +1477 -0
  164. package/src/config/markdown.ts +98 -0
  165. package/src/env/index.ts +28 -0
  166. package/src/file/ignore.ts +83 -0
  167. package/src/file/index.ts +583 -0
  168. package/src/file/ripgrep.ts +375 -0
  169. package/src/file/time.ts +69 -0
  170. package/src/file/watcher.ts +127 -0
  171. package/src/flag/flag.ts +97 -0
  172. package/src/format/formatter.ts +366 -0
  173. package/src/format/index.ts +137 -0
  174. package/src/global/index.ts +55 -0
  175. package/src/id/id.ts +83 -0
  176. package/src/ide/index.ts +76 -0
  177. package/src/index.ts +159 -0
  178. package/src/installation/index.ts +246 -0
  179. package/src/lsp/client.ts +252 -0
  180. package/src/lsp/index.ts +485 -0
  181. package/src/lsp/language.ts +119 -0
  182. package/src/lsp/server.ts +2046 -0
  183. package/src/mcp/auth.ts +132 -0
  184. package/src/mcp/index.ts +934 -0
  185. package/src/mcp/oauth-callback.ts +200 -0
  186. package/src/mcp/oauth-provider.ts +154 -0
  187. package/src/patch/index.ts +680 -0
  188. package/src/permission/arity.ts +163 -0
  189. package/src/permission/index.ts +210 -0
  190. package/src/permission/next.ts +280 -0
  191. package/src/plugin/codex.ts +624 -0
  192. package/src/plugin/copilot.ts +327 -0
  193. package/src/plugin/index.ts +138 -0
  194. package/src/project/bootstrap.ts +35 -0
  195. package/src/project/instance.ts +114 -0
  196. package/src/project/project.ts +371 -0
  197. package/src/project/state.ts +70 -0
  198. package/src/project/vcs.ts +76 -0
  199. package/src/provider/auth.ts +147 -0
  200. package/src/provider/models.ts +133 -0
  201. package/src/provider/provider.ts +1262 -0
  202. package/src/provider/sdk/copilot/README.md +5 -0
  203. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
  204. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  205. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
  206. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  207. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
  208. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  209. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  210. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
  211. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  212. package/src/provider/sdk/copilot/index.ts +2 -0
  213. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  214. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
  215. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  216. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  217. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  218. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
  219. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
  220. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
  221. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  222. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
  223. package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
  224. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
  225. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
  226. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
  227. package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
  228. package/src/provider/transform.ts +828 -0
  229. package/src/pty/index.ts +250 -0
  230. package/src/question/index.ts +171 -0
  231. package/src/scheduler/index.ts +61 -0
  232. package/src/server/error.ts +36 -0
  233. package/src/server/event.ts +7 -0
  234. package/src/server/mdns.ts +60 -0
  235. package/src/server/routes/config.ts +92 -0
  236. package/src/server/routes/experimental.ts +208 -0
  237. package/src/server/routes/file.ts +197 -0
  238. package/src/server/routes/global.ts +183 -0
  239. package/src/server/routes/mcp.ts +225 -0
  240. package/src/server/routes/permission.ts +68 -0
  241. package/src/server/routes/project.ts +82 -0
  242. package/src/server/routes/provider.ts +165 -0
  243. package/src/server/routes/pty.ts +169 -0
  244. package/src/server/routes/question.ts +98 -0
  245. package/src/server/routes/session.ts +939 -0
  246. package/src/server/routes/tui.ts +379 -0
  247. package/src/server/server.ts +613 -0
  248. package/src/session/compaction.ts +226 -0
  249. package/src/session/index.ts +524 -0
  250. package/src/session/instruction.ts +197 -0
  251. package/src/session/llm.ts +289 -0
  252. package/src/session/message-v2.ts +802 -0
  253. package/src/session/message.ts +189 -0
  254. package/src/session/processor.ts +407 -0
  255. package/src/session/prompt/agent.txt +43 -0
  256. package/src/session/prompt/anthropic-20250930.txt +166 -0
  257. package/src/session/prompt/anthropic.txt +105 -0
  258. package/src/session/prompt/beast.txt +147 -0
  259. package/src/session/prompt/build-switch.txt +5 -0
  260. package/src/session/prompt/codex_header.txt +79 -0
  261. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  262. package/src/session/prompt/gemini.txt +155 -0
  263. package/src/session/prompt/max-steps.txt +16 -0
  264. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  265. package/src/session/prompt/plan.txt +26 -0
  266. package/src/session/prompt/qwen.txt +109 -0
  267. package/src/session/prompt/research.txt +81 -0
  268. package/src/session/prompt/trinity.txt +97 -0
  269. package/src/session/prompt.ts +1952 -0
  270. package/src/session/retry.ts +97 -0
  271. package/src/session/revert.ts +121 -0
  272. package/src/session/status.ts +76 -0
  273. package/src/session/summary.ts +217 -0
  274. package/src/session/system.ts +54 -0
  275. package/src/session/todo.ts +37 -0
  276. package/src/share/share-next.ts +200 -0
  277. package/src/share/share.ts +92 -0
  278. package/src/shell/shell.ts +67 -0
  279. package/src/skill/discovery.ts +97 -0
  280. package/src/skill/index.ts +1 -0
  281. package/src/skill/skill.ts +188 -0
  282. package/src/snapshot/index.ts +255 -0
  283. package/src/storage/storage.ts +227 -0
  284. package/src/tool/agent-enter.txt +1 -0
  285. package/src/tool/agent-exit.txt +1 -0
  286. package/src/tool/agent.ts +237 -0
  287. package/src/tool/apply_patch.ts +281 -0
  288. package/src/tool/apply_patch.txt +33 -0
  289. package/src/tool/bash.ts +269 -0
  290. package/src/tool/bash.txt +115 -0
  291. package/src/tool/batch.ts +175 -0
  292. package/src/tool/batch.txt +24 -0
  293. package/src/tool/chat-enter.txt +15 -0
  294. package/src/tool/chat-exit.txt +7 -0
  295. package/src/tool/chat.ts +217 -0
  296. package/src/tool/codesearch.ts +132 -0
  297. package/src/tool/codesearch.txt +12 -0
  298. package/src/tool/edit.ts +655 -0
  299. package/src/tool/edit.txt +10 -0
  300. package/src/tool/external-directory.ts +32 -0
  301. package/src/tool/glob.ts +78 -0
  302. package/src/tool/glob.txt +6 -0
  303. package/src/tool/grep.ts +147 -0
  304. package/src/tool/grep.txt +8 -0
  305. package/src/tool/invalid.ts +17 -0
  306. package/src/tool/ls.ts +121 -0
  307. package/src/tool/ls.txt +1 -0
  308. package/src/tool/lsp.ts +96 -0
  309. package/src/tool/lsp.txt +19 -0
  310. package/src/tool/multiedit.ts +46 -0
  311. package/src/tool/multiedit.txt +41 -0
  312. package/src/tool/plan-enter.txt +14 -0
  313. package/src/tool/plan-exit.txt +13 -0
  314. package/src/tool/plan.ts +130 -0
  315. package/src/tool/question.ts +33 -0
  316. package/src/tool/question.txt +10 -0
  317. package/src/tool/read.ts +211 -0
  318. package/src/tool/read.txt +12 -0
  319. package/src/tool/registry.ts +167 -0
  320. package/src/tool/research-enter.txt +1 -0
  321. package/src/tool/research-exit.txt +1 -0
  322. package/src/tool/research.ts +134 -0
  323. package/src/tool/skill.ts +123 -0
  324. package/src/tool/task.ts +165 -0
  325. package/src/tool/task.txt +60 -0
  326. package/src/tool/todo.ts +53 -0
  327. package/src/tool/todoread.txt +14 -0
  328. package/src/tool/todowrite.txt +167 -0
  329. package/src/tool/tool.ts +89 -0
  330. package/src/tool/truncation.ts +106 -0
  331. package/src/tool/webfetch.ts +186 -0
  332. package/src/tool/webfetch.txt +13 -0
  333. package/src/tool/websearch.ts +150 -0
  334. package/src/tool/websearch.txt +14 -0
  335. package/src/tool/write.ts +85 -0
  336. package/src/tool/write.txt +8 -0
  337. package/src/util/abort.ts +35 -0
  338. package/src/util/archive.ts +16 -0
  339. package/src/util/color.ts +19 -0
  340. package/src/util/context.ts +25 -0
  341. package/src/util/defer.ts +12 -0
  342. package/src/util/eventloop.ts +20 -0
  343. package/src/util/filesystem.ts +93 -0
  344. package/src/util/fn.ts +11 -0
  345. package/src/util/format.ts +20 -0
  346. package/src/util/iife.ts +3 -0
  347. package/src/util/keybind.ts +103 -0
  348. package/src/util/lazy.ts +18 -0
  349. package/src/util/locale.ts +81 -0
  350. package/src/util/lock.ts +98 -0
  351. package/src/util/log.ts +180 -0
  352. package/src/util/proxied.ts +3 -0
  353. package/src/util/queue.ts +32 -0
  354. package/src/util/rpc.ts +66 -0
  355. package/src/util/scrap.ts +10 -0
  356. package/src/util/signal.ts +12 -0
  357. package/src/util/timeout.ts +14 -0
  358. package/src/util/token.ts +7 -0
  359. package/src/util/wildcard.ts +56 -0
  360. package/src/worktree/index.ts +574 -0
  361. package/sst-env.d.ts +9 -0
  362. package/test/acp/agent-interface.test.ts +51 -0
  363. package/test/acp/event-subscription.test.ts +436 -0
  364. package/test/agent/agent.test.ts +675 -0
  365. package/test/bun.test.ts +53 -0
  366. package/test/cli/github-action.test.ts +161 -0
  367. package/test/cli/github-remote.test.ts +80 -0
  368. package/test/cli/import.test.ts +38 -0
  369. package/test/cli/tui/transcript.test.ts +322 -0
  370. package/test/config/agent-color.test.ts +71 -0
  371. package/test/config/config.test.ts +1802 -0
  372. package/test/config/fixtures/empty-frontmatter.md +4 -0
  373. package/test/config/fixtures/frontmatter.md +28 -0
  374. package/test/config/fixtures/markdown-header.md +11 -0
  375. package/test/config/fixtures/no-frontmatter.md +1 -0
  376. package/test/config/fixtures/weird-model-id.md +13 -0
  377. package/test/config/markdown.test.ts +228 -0
  378. package/test/file/ignore.test.ts +10 -0
  379. package/test/file/path-traversal.test.ts +198 -0
  380. package/test/file/ripgrep.test.ts +39 -0
  381. package/test/fixture/fixture.ts +45 -0
  382. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  383. package/test/ide/ide.test.ts +82 -0
  384. package/test/keybind.test.ts +421 -0
  385. package/test/lsp/client.test.ts +95 -0
  386. package/test/mcp/headers.test.ts +153 -0
  387. package/test/mcp/oauth-browser.test.ts +249 -0
  388. package/test/memory/abort-leak.test.ts +136 -0
  389. package/test/patch/patch.test.ts +348 -0
  390. package/test/permission/arity.test.ts +33 -0
  391. package/test/permission/next.test.ts +690 -0
  392. package/test/permission-task.test.ts +319 -0
  393. package/test/plugin/auth-override.test.ts +44 -0
  394. package/test/plugin/codex.test.ts +123 -0
  395. package/test/preload.ts +63 -0
  396. package/test/project/project.test.ts +120 -0
  397. package/test/provider/amazon-bedrock.test.ts +445 -0
  398. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  399. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  400. package/test/provider/gitlab-duo.test.ts +262 -0
  401. package/test/provider/provider.test.ts +2129 -0
  402. package/test/provider/transform.test.ts +2022 -0
  403. package/test/question/question.test.ts +300 -0
  404. package/test/scheduler.test.ts +73 -0
  405. package/test/server/session-list.test.ts +39 -0
  406. package/test/server/session-select.test.ts +78 -0
  407. package/test/session/compaction.test.ts +293 -0
  408. package/test/session/instruction.test.ts +170 -0
  409. package/test/session/llm.test.ts +691 -0
  410. package/test/session/message-v2.test.ts +786 -0
  411. package/test/session/prompt-missing-file.test.ts +53 -0
  412. package/test/session/prompt-special-chars.test.ts +56 -0
  413. package/test/session/prompt-variant.test.ts +60 -0
  414. package/test/session/retry.test.ts +179 -0
  415. package/test/session/revert-compact.test.ts +285 -0
  416. package/test/session/session.test.ts +71 -0
  417. package/test/skill/discovery.test.ts +60 -0
  418. package/test/skill/skill.test.ts +388 -0
  419. package/test/snapshot/snapshot.test.ts +1040 -0
  420. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  421. package/test/tool/apply_patch.test.ts +559 -0
  422. package/test/tool/bash.test.ts +399 -0
  423. package/test/tool/external-directory.test.ts +127 -0
  424. package/test/tool/fixtures/large-image.png +0 -0
  425. package/test/tool/fixtures/models-api.json +38413 -0
  426. package/test/tool/grep.test.ts +110 -0
  427. package/test/tool/question.test.ts +107 -0
  428. package/test/tool/read.test.ts +358 -0
  429. package/test/tool/registry.test.ts +122 -0
  430. package/test/tool/skill.test.ts +112 -0
  431. package/test/tool/truncation.test.ts +159 -0
  432. package/test/util/filesystem.test.ts +39 -0
  433. package/test/util/format.test.ts +59 -0
  434. package/test/util/iife.test.ts +36 -0
  435. package/test/util/lazy.test.ts +50 -0
  436. package/test/util/lock.test.ts +72 -0
  437. package/test/util/timeout.test.ts +21 -0
  438. package/test/util/wildcard.test.ts +75 -0
  439. package/tsconfig.json +16 -0
@@ -0,0 +1,200 @@
1
+ import { Bus } from "@/bus"
2
+ import { Config } from "@/config/config"
3
+ import { ulid } from "ulid"
4
+ import { Provider } from "@/provider/provider"
5
+ import { Session } from "@/session"
6
+ import { MessageV2 } from "@/session/message-v2"
7
+ import { Storage } from "@/storage/storage"
8
+ import { Log } from "@/util/log"
9
+ import type * as SDK from "@opencode-ai/sdk/v2"
10
+
11
+ export namespace ShareNext {
12
+ const log = Log.create({ service: "share-next" })
13
+
14
+ export async function url() {
15
+ return Config.get().then((x) => x.enterprise?.url ?? "https://opncd.ai")
16
+ }
17
+
18
+ const disabled = process.env["OPENCODE_DISABLE_SHARE"] === "true" || process.env["OPENCODE_DISABLE_SHARE"] === "1"
19
+
20
+ export async function init() {
21
+ if (disabled) return
22
+ Bus.subscribe(Session.Event.Updated, async (evt) => {
23
+ await sync(evt.properties.info.id, [
24
+ {
25
+ type: "session",
26
+ data: evt.properties.info,
27
+ },
28
+ ])
29
+ })
30
+ Bus.subscribe(MessageV2.Event.Updated, async (evt) => {
31
+ await sync(evt.properties.info.sessionID, [
32
+ {
33
+ type: "message",
34
+ data: evt.properties.info,
35
+ },
36
+ ])
37
+ if (evt.properties.info.role === "user") {
38
+ await sync(evt.properties.info.sessionID, [
39
+ {
40
+ type: "model",
41
+ data: [
42
+ await Provider.getModel(evt.properties.info.model.providerID, evt.properties.info.model.modelID).then(
43
+ (m) => m,
44
+ ),
45
+ ],
46
+ },
47
+ ])
48
+ }
49
+ })
50
+ Bus.subscribe(MessageV2.Event.PartUpdated, async (evt) => {
51
+ await sync(evt.properties.part.sessionID, [
52
+ {
53
+ type: "part",
54
+ data: evt.properties.part,
55
+ },
56
+ ])
57
+ })
58
+ Bus.subscribe(Session.Event.Diff, async (evt) => {
59
+ await sync(evt.properties.sessionID, [
60
+ {
61
+ type: "session_diff",
62
+ data: evt.properties.diff,
63
+ },
64
+ ])
65
+ })
66
+ }
67
+
68
+ export async function create(sessionID: string) {
69
+ if (disabled) return { id: "", url: "", secret: "" }
70
+ log.info("creating share", { sessionID })
71
+ const result = await fetch(`${await url()}/api/share`, {
72
+ method: "POST",
73
+ headers: {
74
+ "Content-Type": "application/json",
75
+ },
76
+ body: JSON.stringify({ sessionID: sessionID }),
77
+ })
78
+ .then((x) => x.json())
79
+ .then((x) => x as { id: string; url: string; secret: string })
80
+ await Storage.write(["session_share", sessionID], result)
81
+ fullSync(sessionID)
82
+ return result
83
+ }
84
+
85
+ function get(sessionID: string) {
86
+ return Storage.read<{
87
+ id: string
88
+ secret: string
89
+ url: string
90
+ }>(["session_share", sessionID])
91
+ }
92
+
93
+ type Data =
94
+ | {
95
+ type: "session"
96
+ data: SDK.Session
97
+ }
98
+ | {
99
+ type: "message"
100
+ data: SDK.Message
101
+ }
102
+ | {
103
+ type: "part"
104
+ data: SDK.Part
105
+ }
106
+ | {
107
+ type: "session_diff"
108
+ data: SDK.FileDiff[]
109
+ }
110
+ | {
111
+ type: "model"
112
+ data: SDK.Model[]
113
+ }
114
+
115
+ const queue = new Map<string, { timeout: NodeJS.Timeout; data: Map<string, Data> }>()
116
+ async function sync(sessionID: string, data: Data[]) {
117
+ if (disabled) return
118
+ const existing = queue.get(sessionID)
119
+ if (existing) {
120
+ for (const item of data) {
121
+ existing.data.set("id" in item ? (item.id as string) : ulid(), item)
122
+ }
123
+ return
124
+ }
125
+
126
+ const dataMap = new Map<string, Data>()
127
+ for (const item of data) {
128
+ dataMap.set("id" in item ? (item.id as string) : ulid(), item)
129
+ }
130
+
131
+ const timeout = setTimeout(async () => {
132
+ const queued = queue.get(sessionID)
133
+ if (!queued) return
134
+ queue.delete(sessionID)
135
+ const share = await get(sessionID).catch(() => undefined)
136
+ if (!share) return
137
+
138
+ await fetch(`${await url()}/api/share/${share.id}/sync`, {
139
+ method: "POST",
140
+ headers: {
141
+ "Content-Type": "application/json",
142
+ },
143
+ body: JSON.stringify({
144
+ secret: share.secret,
145
+ data: Array.from(queued.data.values()),
146
+ }),
147
+ })
148
+ }, 1000)
149
+ queue.set(sessionID, { timeout, data: dataMap })
150
+ }
151
+
152
+ export async function remove(sessionID: string) {
153
+ if (disabled) return
154
+ log.info("removing share", { sessionID })
155
+ const share = await get(sessionID)
156
+ if (!share) return
157
+ await fetch(`${await url()}/api/share/${share.id}`, {
158
+ method: "DELETE",
159
+ headers: {
160
+ "Content-Type": "application/json",
161
+ },
162
+ body: JSON.stringify({
163
+ secret: share.secret,
164
+ }),
165
+ })
166
+ await Storage.remove(["session_share", sessionID])
167
+ }
168
+
169
+ async function fullSync(sessionID: string) {
170
+ log.info("full sync", { sessionID })
171
+ const session = await Session.get(sessionID)
172
+ const diffs = await Session.diff(sessionID)
173
+ const messages = await Array.fromAsync(MessageV2.stream(sessionID))
174
+ const models = await Promise.all(
175
+ messages
176
+ .filter((m) => m.info.role === "user")
177
+ .map((m) => (m.info as SDK.UserMessage).model)
178
+ .map((m) => Provider.getModel(m.providerID, m.modelID).then((m) => m)),
179
+ )
180
+ await sync(sessionID, [
181
+ {
182
+ type: "session",
183
+ data: session,
184
+ },
185
+ ...messages.map((x) => ({
186
+ type: "message" as const,
187
+ data: x.info,
188
+ })),
189
+ ...messages.flatMap((x) => x.parts.map((y) => ({ type: "part" as const, data: y }))),
190
+ {
191
+ type: "session_diff",
192
+ data: diffs,
193
+ },
194
+ {
195
+ type: "model",
196
+ data: models,
197
+ },
198
+ ])
199
+ }
200
+ }
@@ -0,0 +1,92 @@
1
+ import { Bus } from "../bus"
2
+ import { Installation } from "../installation"
3
+ import { Session } from "../session"
4
+ import { MessageV2 } from "../session/message-v2"
5
+ import { Log } from "../util/log"
6
+
7
+ export namespace Share {
8
+ const log = Log.create({ service: "share" })
9
+
10
+ let queue: Promise<void> = Promise.resolve()
11
+ const pending = new Map<string, any>()
12
+
13
+ export async function sync(key: string, content: any) {
14
+ if (disabled) return
15
+ const [root, ...splits] = key.split("/")
16
+ if (root !== "session") return
17
+ const [sub, sessionID] = splits
18
+ if (sub === "share") return
19
+ const share = await Session.getShare(sessionID).catch(() => {})
20
+ if (!share) return
21
+ const { secret } = share
22
+ pending.set(key, content)
23
+ queue = queue
24
+ .then(async () => {
25
+ const content = pending.get(key)
26
+ if (content === undefined) return
27
+ pending.delete(key)
28
+
29
+ return fetch(`${URL}/share_sync`, {
30
+ method: "POST",
31
+ body: JSON.stringify({
32
+ sessionID: sessionID,
33
+ secret,
34
+ key: key,
35
+ content,
36
+ }),
37
+ })
38
+ })
39
+ .then((x) => {
40
+ if (x) {
41
+ log.info("synced", {
42
+ key: key,
43
+ status: x.status,
44
+ })
45
+ }
46
+ })
47
+ }
48
+
49
+ export function init() {
50
+ Bus.subscribe(Session.Event.Updated, async (evt) => {
51
+ await sync("session/info/" + evt.properties.info.id, evt.properties.info)
52
+ })
53
+ Bus.subscribe(MessageV2.Event.Updated, async (evt) => {
54
+ await sync("session/message/" + evt.properties.info.sessionID + "/" + evt.properties.info.id, evt.properties.info)
55
+ })
56
+ Bus.subscribe(MessageV2.Event.PartUpdated, async (evt) => {
57
+ await sync(
58
+ "session/part/" +
59
+ evt.properties.part.sessionID +
60
+ "/" +
61
+ evt.properties.part.messageID +
62
+ "/" +
63
+ evt.properties.part.id,
64
+ evt.properties.part,
65
+ )
66
+ })
67
+ }
68
+
69
+ export const URL =
70
+ process.env["OPENCODE_API"] ??
71
+ (Installation.isPreview() || Installation.isLocal() ? "https://api.dev.opencode.ai" : "https://api.opencode.ai")
72
+
73
+ const disabled = process.env["OPENCODE_DISABLE_SHARE"] === "true" || process.env["OPENCODE_DISABLE_SHARE"] === "1"
74
+
75
+ export async function create(sessionID: string) {
76
+ if (disabled) return { url: "", secret: "" }
77
+ return fetch(`${URL}/share_create`, {
78
+ method: "POST",
79
+ body: JSON.stringify({ sessionID: sessionID }),
80
+ })
81
+ .then((x) => x.json())
82
+ .then((x) => x as { url: string; secret: string })
83
+ }
84
+
85
+ export async function remove(sessionID: string, secret: string) {
86
+ if (disabled) return {}
87
+ return fetch(`${URL}/share_delete`, {
88
+ method: "POST",
89
+ body: JSON.stringify({ sessionID, secret }),
90
+ }).then((x) => x.json())
91
+ }
92
+ }
@@ -0,0 +1,67 @@
1
+ import { Flag } from "@/flag/flag"
2
+ import { lazy } from "@/util/lazy"
3
+ import path from "path"
4
+ import { spawn, type ChildProcess } from "child_process"
5
+
6
+ const SIGKILL_TIMEOUT_MS = 200
7
+
8
+ export namespace Shell {
9
+ export async function killTree(proc: ChildProcess, opts?: { exited?: () => boolean }): Promise<void> {
10
+ const pid = proc.pid
11
+ if (!pid || opts?.exited?.()) return
12
+
13
+ if (process.platform === "win32") {
14
+ await new Promise<void>((resolve) => {
15
+ const killer = spawn("taskkill", ["/pid", String(pid), "/f", "/t"], { stdio: "ignore" })
16
+ killer.once("exit", () => resolve())
17
+ killer.once("error", () => resolve())
18
+ })
19
+ return
20
+ }
21
+
22
+ try {
23
+ process.kill(-pid, "SIGTERM")
24
+ await Bun.sleep(SIGKILL_TIMEOUT_MS)
25
+ if (!opts?.exited?.()) {
26
+ process.kill(-pid, "SIGKILL")
27
+ }
28
+ } catch (_e) {
29
+ proc.kill("SIGTERM")
30
+ await Bun.sleep(SIGKILL_TIMEOUT_MS)
31
+ if (!opts?.exited?.()) {
32
+ proc.kill("SIGKILL")
33
+ }
34
+ }
35
+ }
36
+ const BLACKLIST = new Set(["fish", "nu"])
37
+
38
+ function fallback() {
39
+ if (process.platform === "win32") {
40
+ if (Flag.OPENCODE_GIT_BASH_PATH) return Flag.OPENCODE_GIT_BASH_PATH
41
+ const git = Bun.which("git")
42
+ if (git) {
43
+ // git.exe is typically at: C:\Program Files\Git\cmd\git.exe
44
+ // bash.exe is at: C:\Program Files\Git\bin\bash.exe
45
+ const bash = path.join(git, "..", "..", "bin", "bash.exe")
46
+ if (Bun.file(bash).size) return bash
47
+ }
48
+ return process.env.COMSPEC || "cmd.exe"
49
+ }
50
+ if (process.platform === "darwin") return "/bin/zsh"
51
+ const bash = Bun.which("bash")
52
+ if (bash) return bash
53
+ return "/bin/sh"
54
+ }
55
+
56
+ export const preferred = lazy(() => {
57
+ const s = process.env.SHELL
58
+ if (s) return s
59
+ return fallback()
60
+ })
61
+
62
+ export const acceptable = lazy(() => {
63
+ const s = process.env.SHELL
64
+ if (s && !BLACKLIST.has(process.platform === "win32" ? path.win32.basename(s) : path.basename(s))) return s
65
+ return fallback()
66
+ })
67
+ }
@@ -0,0 +1,97 @@
1
+ import path from "path"
2
+ import { mkdir } from "fs/promises"
3
+ import { Log } from "../util/log"
4
+ import { Global } from "../global"
5
+
6
+ export namespace Discovery {
7
+ const log = Log.create({ service: "skill-discovery" })
8
+
9
+ type Index = {
10
+ skills: Array<{
11
+ name: string
12
+ description: string
13
+ files: string[]
14
+ }>
15
+ }
16
+
17
+ export function dir() {
18
+ return path.join(Global.Path.cache, "skills")
19
+ }
20
+
21
+ async function get(url: string, dest: string): Promise<boolean> {
22
+ if (await Bun.file(dest).exists()) return true
23
+ return fetch(url)
24
+ .then(async (response) => {
25
+ if (!response.ok) {
26
+ log.error("failed to download", { url, status: response.status })
27
+ return false
28
+ }
29
+ await Bun.write(dest, await response.text())
30
+ return true
31
+ })
32
+ .catch((err) => {
33
+ log.error("failed to download", { url, err })
34
+ return false
35
+ })
36
+ }
37
+
38
+ export async function pull(url: string): Promise<string[]> {
39
+ const result: string[] = []
40
+ const base = url.endsWith("/") ? url : `${url}/`
41
+ const index = new URL("index.json", base).href
42
+ const cache = dir()
43
+ const host = base.slice(0, -1)
44
+
45
+ log.info("fetching index", { url: index })
46
+ const data = await fetch(index)
47
+ .then(async (response) => {
48
+ if (!response.ok) {
49
+ log.error("failed to fetch index", { url: index, status: response.status })
50
+ return undefined
51
+ }
52
+ return response
53
+ .json()
54
+ .then((json) => json as Index)
55
+ .catch((err) => {
56
+ log.error("failed to parse index", { url: index, err })
57
+ return undefined
58
+ })
59
+ })
60
+ .catch((err) => {
61
+ log.error("failed to fetch index", { url: index, err })
62
+ return undefined
63
+ })
64
+
65
+ if (!data?.skills || !Array.isArray(data.skills)) {
66
+ log.warn("invalid index format", { url: index })
67
+ return result
68
+ }
69
+
70
+ const list = data.skills.filter((skill) => {
71
+ if (!skill?.name || !Array.isArray(skill.files)) {
72
+ log.warn("invalid skill entry", { url: index, skill })
73
+ return false
74
+ }
75
+ return true
76
+ })
77
+
78
+ await Promise.all(
79
+ list.map(async (skill) => {
80
+ const root = path.join(cache, skill.name)
81
+ await Promise.all(
82
+ skill.files.map(async (file) => {
83
+ const link = new URL(file, `${host}/${skill.name}/`).href
84
+ const dest = path.join(root, file)
85
+ await mkdir(path.dirname(dest), { recursive: true })
86
+ await get(link, dest)
87
+ }),
88
+ )
89
+
90
+ const md = path.join(root, "SKILL.md")
91
+ if (await Bun.file(md).exists()) result.push(root)
92
+ }),
93
+ )
94
+
95
+ return result
96
+ }
97
+ }
@@ -0,0 +1 @@
1
+ export * from "./skill"
@@ -0,0 +1,188 @@
1
+ import z from "zod"
2
+ import path from "path"
3
+ import os from "os"
4
+ import { Config } from "../config/config"
5
+ import { Instance } from "../project/instance"
6
+ import { NamedError } from "@opencode-ai/util/error"
7
+ import { ConfigMarkdown } from "../config/markdown"
8
+ import { Log } from "../util/log"
9
+ import { Global } from "@/global"
10
+ import { Filesystem } from "@/util/filesystem"
11
+ import { Flag } from "@/flag/flag"
12
+ import { Bus } from "@/bus"
13
+ import { Session } from "@/session"
14
+ import { Discovery } from "./discovery"
15
+
16
+ export namespace Skill {
17
+ const log = Log.create({ service: "skill" })
18
+ export const Info = z.object({
19
+ name: z.string(),
20
+ description: z.string(),
21
+ location: z.string(),
22
+ content: z.string(),
23
+ })
24
+ export type Info = z.infer<typeof Info>
25
+
26
+ export const InvalidError = NamedError.create(
27
+ "SkillInvalidError",
28
+ z.object({
29
+ path: z.string(),
30
+ message: z.string().optional(),
31
+ issues: z.custom<z.core.$ZodIssue[]>().optional(),
32
+ }),
33
+ )
34
+
35
+ export const NameMismatchError = NamedError.create(
36
+ "SkillNameMismatchError",
37
+ z.object({
38
+ path: z.string(),
39
+ expected: z.string(),
40
+ actual: z.string(),
41
+ }),
42
+ )
43
+
44
+ // External skill directories to search for (project-level and global)
45
+ // These follow the directory layout used by Claude Code and other agents.
46
+ const EXTERNAL_DIRS = [".claude", ".agents"]
47
+ const EXTERNAL_SKILL_GLOB = new Bun.Glob("skills/**/SKILL.md")
48
+
49
+ const OPENCODE_SKILL_GLOB = new Bun.Glob("{skill,skills}/**/SKILL.md")
50
+ const SKILL_GLOB = new Bun.Glob("**/SKILL.md")
51
+
52
+ export const state = Instance.state(async () => {
53
+ const skills: Record<string, Info> = {}
54
+ const dirs = new Set<string>()
55
+
56
+ const addSkill = async (match: string) => {
57
+ const md = await ConfigMarkdown.parse(match).catch((err) => {
58
+ const message = ConfigMarkdown.FrontmatterError.isInstance(err)
59
+ ? err.data.message
60
+ : `Failed to parse skill ${match}`
61
+ Bus.publish(Session.Event.Error, { error: new NamedError.Unknown({ message }).toObject() })
62
+ log.error("failed to load skill", { skill: match, err })
63
+ return undefined
64
+ })
65
+
66
+ if (!md) return
67
+
68
+ const parsed = Info.pick({ name: true, description: true }).safeParse(md.data)
69
+ if (!parsed.success) return
70
+
71
+ // Warn on duplicate skill names
72
+ if (skills[parsed.data.name]) {
73
+ log.warn("duplicate skill name", {
74
+ name: parsed.data.name,
75
+ existing: skills[parsed.data.name].location,
76
+ duplicate: match,
77
+ })
78
+ }
79
+
80
+ dirs.add(path.dirname(match))
81
+
82
+ skills[parsed.data.name] = {
83
+ name: parsed.data.name,
84
+ description: parsed.data.description,
85
+ location: match,
86
+ content: md.content,
87
+ }
88
+ }
89
+
90
+ const scanExternal = async (root: string, scope: "global" | "project") => {
91
+ return Array.fromAsync(
92
+ EXTERNAL_SKILL_GLOB.scan({
93
+ cwd: root,
94
+ absolute: true,
95
+ onlyFiles: true,
96
+ followSymlinks: true,
97
+ dot: true,
98
+ }),
99
+ )
100
+ .then((matches) => Promise.all(matches.map(addSkill)))
101
+ .catch((error) => {
102
+ log.error(`failed to scan ${scope} skills`, { dir: root, error })
103
+ })
104
+ }
105
+
106
+ // Scan external skill directories (.claude/skills/, .agents/skills/, etc.)
107
+ // Load global (home) first, then project-level (so project-level overwrites)
108
+ if (!Flag.OPENCODE_DISABLE_EXTERNAL_SKILLS) {
109
+ for (const dir of EXTERNAL_DIRS) {
110
+ const root = path.join(Global.Path.home, dir)
111
+ if (!(await Filesystem.isDir(root))) continue
112
+ await scanExternal(root, "global")
113
+ }
114
+
115
+ for await (const root of Filesystem.up({
116
+ targets: EXTERNAL_DIRS,
117
+ start: Instance.directory,
118
+ stop: Instance.worktree,
119
+ })) {
120
+ await scanExternal(root, "project")
121
+ }
122
+ }
123
+
124
+ // Scan .opencode/skill/ directories
125
+ for (const dir of await Config.directories()) {
126
+ for await (const match of OPENCODE_SKILL_GLOB.scan({
127
+ cwd: dir,
128
+ absolute: true,
129
+ onlyFiles: true,
130
+ followSymlinks: true,
131
+ })) {
132
+ await addSkill(match)
133
+ }
134
+ }
135
+
136
+ // Scan additional skill paths from config
137
+ const config = await Config.get()
138
+ for (const skillPath of config.skills?.paths ?? []) {
139
+ const expanded = skillPath.startsWith("~/") ? path.join(os.homedir(), skillPath.slice(2)) : skillPath
140
+ const resolved = path.isAbsolute(expanded) ? expanded : path.join(Instance.directory, expanded)
141
+ if (!(await Filesystem.isDir(resolved))) {
142
+ log.warn("skill path not found", { path: resolved })
143
+ continue
144
+ }
145
+ for await (const match of SKILL_GLOB.scan({
146
+ cwd: resolved,
147
+ absolute: true,
148
+ onlyFiles: true,
149
+ followSymlinks: true,
150
+ })) {
151
+ await addSkill(match)
152
+ }
153
+ }
154
+
155
+ // Download and load skills from URLs
156
+ for (const url of config.skills?.urls ?? []) {
157
+ const list = await Discovery.pull(url)
158
+ for (const dir of list) {
159
+ dirs.add(dir)
160
+ for await (const match of SKILL_GLOB.scan({
161
+ cwd: dir,
162
+ absolute: true,
163
+ onlyFiles: true,
164
+ followSymlinks: true,
165
+ })) {
166
+ await addSkill(match)
167
+ }
168
+ }
169
+ }
170
+
171
+ return {
172
+ skills,
173
+ dirs: Array.from(dirs),
174
+ }
175
+ })
176
+
177
+ export async function get(name: string) {
178
+ return state().then((x) => x.skills[name])
179
+ }
180
+
181
+ export async function all() {
182
+ return state().then((x) => Object.values(x.skills))
183
+ }
184
+
185
+ export async function dirs() {
186
+ return state().then((x) => x.dirs)
187
+ }
188
+ }