easc-cli 1.1.28

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 (404) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/opencode +108 -0
  5. package/bunfig.toml +7 -0
  6. package/package.json +132 -0
  7. package/parsers-config.ts +253 -0
  8. package/script/build.ts +172 -0
  9. package/script/deploy.ts +64 -0
  10. package/script/postinstall.mjs +125 -0
  11. package/script/publish-registries.ts +187 -0
  12. package/script/publish.ts +70 -0
  13. package/script/schema.ts +47 -0
  14. package/script/seed-e2e.ts +50 -0
  15. package/src/acp/README.md +164 -0
  16. package/src/acp/agent.ts +1285 -0
  17. package/src/acp/session.ts +105 -0
  18. package/src/acp/types.ts +22 -0
  19. package/src/agent/agent.ts +332 -0
  20. package/src/agent/generate.txt +75 -0
  21. package/src/agent/prompt/compaction.txt +12 -0
  22. package/src/agent/prompt/explore.txt +18 -0
  23. package/src/agent/prompt/summary.txt +11 -0
  24. package/src/agent/prompt/title.txt +43 -0
  25. package/src/auth/eliseart.ts +76 -0
  26. package/src/auth/index.ts +73 -0
  27. package/src/bun/index.ts +134 -0
  28. package/src/bus/bus-event.ts +43 -0
  29. package/src/bus/global.ts +10 -0
  30. package/src/bus/index.ts +105 -0
  31. package/src/cli/bootstrap.ts +17 -0
  32. package/src/cli/cmd/account.ts +81 -0
  33. package/src/cli/cmd/acp.ts +69 -0
  34. package/src/cli/cmd/agent.ts +257 -0
  35. package/src/cli/cmd/auth.ts +427 -0
  36. package/src/cli/cmd/cmd.ts +7 -0
  37. package/src/cli/cmd/debug/agent.ts +166 -0
  38. package/src/cli/cmd/debug/config.ts +16 -0
  39. package/src/cli/cmd/debug/file.ts +97 -0
  40. package/src/cli/cmd/debug/index.ts +48 -0
  41. package/src/cli/cmd/debug/lsp.ts +52 -0
  42. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  43. package/src/cli/cmd/debug/scrap.ts +16 -0
  44. package/src/cli/cmd/debug/skill.ts +16 -0
  45. package/src/cli/cmd/debug/snapshot.ts +52 -0
  46. package/src/cli/cmd/export.ts +88 -0
  47. package/src/cli/cmd/generate.ts +38 -0
  48. package/src/cli/cmd/github.ts +1548 -0
  49. package/src/cli/cmd/import.ts +98 -0
  50. package/src/cli/cmd/mcp.ts +827 -0
  51. package/src/cli/cmd/models.ts +77 -0
  52. package/src/cli/cmd/pr.ts +112 -0
  53. package/src/cli/cmd/run.ts +407 -0
  54. package/src/cli/cmd/serve.ts +20 -0
  55. package/src/cli/cmd/session.ts +135 -0
  56. package/src/cli/cmd/stats.ts +402 -0
  57. package/src/cli/cmd/tui/app.tsx +774 -0
  58. package/src/cli/cmd/tui/attach.ts +31 -0
  59. package/src/cli/cmd/tui/component/border.tsx +21 -0
  60. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  61. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  62. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  63. package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
  64. package/src/cli/cmd/tui/component/dialog-provider.tsx +256 -0
  65. package/src/cli/cmd/tui/component/dialog-session-list.tsx +114 -0
  66. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  67. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  68. package/src/cli/cmd/tui/component/dialog-status.tsx +164 -0
  69. package/src/cli/cmd/tui/component/dialog-supabase.tsx +102 -0
  70. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  71. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  72. package/src/cli/cmd/tui/component/logo.tsx +88 -0
  73. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +653 -0
  74. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  75. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  76. package/src/cli/cmd/tui/component/prompt/index.tsx +1182 -0
  77. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  78. package/src/cli/cmd/tui/component/spinner.tsx +16 -0
  79. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  80. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  81. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  82. package/src/cli/cmd/tui/context/args.tsx +14 -0
  83. package/src/cli/cmd/tui/context/directory.ts +13 -0
  84. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  85. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  86. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  87. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  88. package/src/cli/cmd/tui/context/local.tsx +402 -0
  89. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  90. package/src/cli/cmd/tui/context/route.tsx +46 -0
  91. package/src/cli/cmd/tui/context/sdk.tsx +94 -0
  92. package/src/cli/cmd/tui/context/sync.tsx +445 -0
  93. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  94. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  95. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  96. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  97. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  98. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  99. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  100. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  101. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  102. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  103. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  104. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  105. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  106. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  107. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  108. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  109. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  110. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  111. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  112. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  113. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  114. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  115. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  116. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  117. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  118. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  119. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  120. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  121. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  122. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  123. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  124. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  125. package/src/cli/cmd/tui/context/theme.tsx +1152 -0
  126. package/src/cli/cmd/tui/event.ts +48 -0
  127. package/src/cli/cmd/tui/routes/home.tsx +140 -0
  128. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  129. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  130. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  131. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  132. package/src/cli/cmd/tui/routes/session/dialog-tool.tsx +63 -0
  133. package/src/cli/cmd/tui/routes/session/footer.tsx +129 -0
  134. package/src/cli/cmd/tui/routes/session/header.tsx +136 -0
  135. package/src/cli/cmd/tui/routes/session/index.tsx +2132 -0
  136. package/src/cli/cmd/tui/routes/session/permission.tsx +495 -0
  137. package/src/cli/cmd/tui/routes/session/question.tsx +435 -0
  138. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  139. package/src/cli/cmd/tui/thread.ts +165 -0
  140. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  141. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  142. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +204 -0
  143. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  144. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  145. package/src/cli/cmd/tui/ui/dialog-select.tsx +376 -0
  146. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  147. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  148. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  149. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  150. package/src/cli/cmd/tui/util/clipboard.ts +160 -0
  151. package/src/cli/cmd/tui/util/editor.ts +32 -0
  152. package/src/cli/cmd/tui/util/signal.ts +7 -0
  153. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  154. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  155. package/src/cli/cmd/tui/worker.ts +152 -0
  156. package/src/cli/cmd/uninstall.ts +357 -0
  157. package/src/cli/cmd/upgrade.ts +73 -0
  158. package/src/cli/cmd/web.ts +81 -0
  159. package/src/cli/error.ts +57 -0
  160. package/src/cli/network.ts +53 -0
  161. package/src/cli/ui.ts +84 -0
  162. package/src/cli/upgrade.ts +25 -0
  163. package/src/command/index.ts +131 -0
  164. package/src/command/template/initialize.txt +10 -0
  165. package/src/command/template/review.txt +99 -0
  166. package/src/config/config.ts +1361 -0
  167. package/src/config/markdown.ts +93 -0
  168. package/src/env/index.ts +26 -0
  169. package/src/file/ignore.ts +83 -0
  170. package/src/file/index.ts +411 -0
  171. package/src/file/ripgrep.ts +407 -0
  172. package/src/file/time.ts +64 -0
  173. package/src/file/watcher.ts +127 -0
  174. package/src/flag/flag.ts +54 -0
  175. package/src/format/formatter.ts +342 -0
  176. package/src/format/index.ts +137 -0
  177. package/src/global/index.ts +55 -0
  178. package/src/id/id.ts +83 -0
  179. package/src/ide/index.ts +76 -0
  180. package/src/index.ts +162 -0
  181. package/src/installation/index.ts +246 -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 +135 -0
  187. package/src/mcp/index.ts +931 -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 +269 -0
  194. package/src/plugin/codex.ts +493 -0
  195. package/src/plugin/copilot.ts +269 -0
  196. package/src/plugin/index.ts +135 -0
  197. package/src/project/bootstrap.ts +35 -0
  198. package/src/project/instance.ts +91 -0
  199. package/src/project/project.ts +339 -0
  200. package/src/project/state.ts +66 -0
  201. package/src/project/vcs.ts +76 -0
  202. package/src/provider/auth.ts +147 -0
  203. package/src/provider/models-macro.ts +11 -0
  204. package/src/provider/models.ts +112 -0
  205. package/src/provider/provider.ts +1391 -0
  206. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  207. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  208. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  209. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  210. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  211. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  212. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  213. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  214. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1732 -0
  215. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  216. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  217. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  218. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  219. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  220. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  221. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  222. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  223. package/src/provider/transform.ts +733 -0
  224. package/src/pty/index.ts +232 -0
  225. package/src/question/index.ts +171 -0
  226. package/src/scheduler/index.ts +61 -0
  227. package/src/server/error.ts +36 -0
  228. package/src/server/event.ts +7 -0
  229. package/src/server/mdns.ts +59 -0
  230. package/src/server/routes/config.ts +92 -0
  231. package/src/server/routes/experimental.ts +208 -0
  232. package/src/server/routes/file.ts +197 -0
  233. package/src/server/routes/global.ts +135 -0
  234. package/src/server/routes/mcp.ts +361 -0
  235. package/src/server/routes/permission.ts +68 -0
  236. package/src/server/routes/project.ts +82 -0
  237. package/src/server/routes/provider.ts +165 -0
  238. package/src/server/routes/pty.ts +169 -0
  239. package/src/server/routes/question.ts +98 -0
  240. package/src/server/routes/session.ts +935 -0
  241. package/src/server/routes/tui.ts +379 -0
  242. package/src/server/server.ts +573 -0
  243. package/src/session/compaction.ts +225 -0
  244. package/src/session/index.ts +488 -0
  245. package/src/session/llm.ts +279 -0
  246. package/src/session/message-v2.ts +702 -0
  247. package/src/session/message.ts +189 -0
  248. package/src/session/processor.ts +406 -0
  249. package/src/session/prompt/anthropic-20250930.txt +166 -0
  250. package/src/session/prompt/anthropic.txt +105 -0
  251. package/src/session/prompt/anthropic_spoof.txt +1 -0
  252. package/src/session/prompt/beast.txt +147 -0
  253. package/src/session/prompt/build-switch.txt +5 -0
  254. package/src/session/prompt/codex_header.txt +79 -0
  255. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  256. package/src/session/prompt/gemini.txt +155 -0
  257. package/src/session/prompt/max-steps.txt +16 -0
  258. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  259. package/src/session/prompt/plan.txt +26 -0
  260. package/src/session/prompt/qwen.txt +109 -0
  261. package/src/session/prompt.ts +1820 -0
  262. package/src/session/retry.ts +90 -0
  263. package/src/session/revert.ts +108 -0
  264. package/src/session/status.ts +76 -0
  265. package/src/session/summary.ts +150 -0
  266. package/src/session/system.ts +152 -0
  267. package/src/session/todo.ts +37 -0
  268. package/src/share/share-next.ts +200 -0
  269. package/src/share/share.ts +92 -0
  270. package/src/shell/shell.ts +67 -0
  271. package/src/skill/index.ts +1 -0
  272. package/src/skill/skill.ts +136 -0
  273. package/src/snapshot/index.ts +236 -0
  274. package/src/storage/storage.ts +227 -0
  275. package/src/tool/apply_patch.ts +269 -0
  276. package/src/tool/apply_patch.txt +33 -0
  277. package/src/tool/bash.ts +259 -0
  278. package/src/tool/bash.txt +115 -0
  279. package/src/tool/batch.ts +175 -0
  280. package/src/tool/batch.txt +24 -0
  281. package/src/tool/codesearch.ts +132 -0
  282. package/src/tool/codesearch.txt +12 -0
  283. package/src/tool/edit.ts +645 -0
  284. package/src/tool/edit.txt +10 -0
  285. package/src/tool/external-directory.ts +32 -0
  286. package/src/tool/glob.ts +77 -0
  287. package/src/tool/glob.txt +6 -0
  288. package/src/tool/grep.ts +154 -0
  289. package/src/tool/grep.txt +8 -0
  290. package/src/tool/invalid.ts +17 -0
  291. package/src/tool/ls.ts +121 -0
  292. package/src/tool/ls.txt +1 -0
  293. package/src/tool/lsp.ts +96 -0
  294. package/src/tool/lsp.txt +19 -0
  295. package/src/tool/multiedit.ts +46 -0
  296. package/src/tool/multiedit.txt +41 -0
  297. package/src/tool/plan-enter.txt +14 -0
  298. package/src/tool/plan-exit.txt +13 -0
  299. package/src/tool/plan.ts +130 -0
  300. package/src/tool/question.ts +33 -0
  301. package/src/tool/question.txt +10 -0
  302. package/src/tool/read.ts +202 -0
  303. package/src/tool/read.txt +12 -0
  304. package/src/tool/registry.ts +163 -0
  305. package/src/tool/skill.ts +75 -0
  306. package/src/tool/task.ts +188 -0
  307. package/src/tool/task.txt +60 -0
  308. package/src/tool/todo.ts +53 -0
  309. package/src/tool/todoread.txt +14 -0
  310. package/src/tool/todowrite.txt +167 -0
  311. package/src/tool/tool.ts +88 -0
  312. package/src/tool/truncation.ts +106 -0
  313. package/src/tool/webfetch.ts +182 -0
  314. package/src/tool/webfetch.txt +13 -0
  315. package/src/tool/websearch.ts +150 -0
  316. package/src/tool/websearch.txt +14 -0
  317. package/src/tool/write.ts +80 -0
  318. package/src/tool/write.txt +8 -0
  319. package/src/util/archive.ts +16 -0
  320. package/src/util/color.ts +19 -0
  321. package/src/util/context.ts +25 -0
  322. package/src/util/defer.ts +12 -0
  323. package/src/util/eventloop.ts +20 -0
  324. package/src/util/filesystem.ts +93 -0
  325. package/src/util/fn.ts +11 -0
  326. package/src/util/format.ts +20 -0
  327. package/src/util/iife.ts +3 -0
  328. package/src/util/keybind.ts +103 -0
  329. package/src/util/lazy.ts +18 -0
  330. package/src/util/locale.ts +81 -0
  331. package/src/util/lock.ts +98 -0
  332. package/src/util/log.ts +180 -0
  333. package/src/util/queue.ts +32 -0
  334. package/src/util/rpc.ts +66 -0
  335. package/src/util/scrap.ts +10 -0
  336. package/src/util/signal.ts +12 -0
  337. package/src/util/timeout.ts +14 -0
  338. package/src/util/token.ts +7 -0
  339. package/src/util/wildcard.ts +56 -0
  340. package/src/worktree/index.ts +424 -0
  341. package/sst-env.d.ts +9 -0
  342. package/test/acp/event-subscription.test.ts +436 -0
  343. package/test/agent/agent.test.ts +638 -0
  344. package/test/bun.test.ts +53 -0
  345. package/test/cli/github-action.test.ts +129 -0
  346. package/test/cli/github-remote.test.ts +80 -0
  347. package/test/cli/tui/transcript.test.ts +297 -0
  348. package/test/config/agent-color.test.ts +66 -0
  349. package/test/config/config.test.ts +1414 -0
  350. package/test/config/fixtures/empty-frontmatter.md +4 -0
  351. package/test/config/fixtures/frontmatter.md +28 -0
  352. package/test/config/fixtures/no-frontmatter.md +1 -0
  353. package/test/config/markdown.test.ts +192 -0
  354. package/test/file/ignore.test.ts +10 -0
  355. package/test/file/path-traversal.test.ts +198 -0
  356. package/test/fixture/fixture.ts +45 -0
  357. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  358. package/test/ide/ide.test.ts +82 -0
  359. package/test/keybind.test.ts +421 -0
  360. package/test/lsp/client.test.ts +95 -0
  361. package/test/mcp/headers.test.ts +153 -0
  362. package/test/mcp/oauth-browser.test.ts +261 -0
  363. package/test/patch/patch.test.ts +348 -0
  364. package/test/permission/arity.test.ts +33 -0
  365. package/test/permission/next.test.ts +652 -0
  366. package/test/permission-task.test.ts +319 -0
  367. package/test/plugin/codex.test.ts +123 -0
  368. package/test/preload.ts +65 -0
  369. package/test/project/project.test.ts +120 -0
  370. package/test/provider/amazon-bedrock.test.ts +268 -0
  371. package/test/provider/gitlab-duo.test.ts +286 -0
  372. package/test/provider/provider.test.ts +2149 -0
  373. package/test/provider/transform.test.ts +1596 -0
  374. package/test/question/question.test.ts +300 -0
  375. package/test/scheduler.test.ts +73 -0
  376. package/test/server/session-list.test.ts +39 -0
  377. package/test/server/session-select.test.ts +78 -0
  378. package/test/session/compaction.test.ts +293 -0
  379. package/test/session/llm.test.ts +90 -0
  380. package/test/session/message-v2.test.ts +662 -0
  381. package/test/session/retry.test.ts +131 -0
  382. package/test/session/revert-compact.test.ts +285 -0
  383. package/test/session/session.test.ts +71 -0
  384. package/test/skill/skill.test.ts +185 -0
  385. package/test/snapshot/snapshot.test.ts +939 -0
  386. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  387. package/test/tool/apply_patch.test.ts +499 -0
  388. package/test/tool/bash.test.ts +320 -0
  389. package/test/tool/external-directory.test.ts +126 -0
  390. package/test/tool/fixtures/large-image.png +0 -0
  391. package/test/tool/fixtures/models-api.json +33453 -0
  392. package/test/tool/grep.test.ts +109 -0
  393. package/test/tool/question.test.ts +105 -0
  394. package/test/tool/read.test.ts +332 -0
  395. package/test/tool/registry.test.ts +76 -0
  396. package/test/tool/truncation.test.ts +159 -0
  397. package/test/util/filesystem.test.ts +39 -0
  398. package/test/util/format.test.ts +59 -0
  399. package/test/util/iife.test.ts +36 -0
  400. package/test/util/lazy.test.ts +50 -0
  401. package/test/util/lock.test.ts +72 -0
  402. package/test/util/timeout.test.ts +21 -0
  403. package/test/util/wildcard.test.ts +75 -0
  404. package/tsconfig.json +16 -0
@@ -0,0 +1,257 @@
1
+ import { cmd } from "./cmd"
2
+ import * as prompts from "@clack/prompts"
3
+ import { UI } from "../ui"
4
+ import { Global } from "../../global"
5
+ import { Agent } from "../../agent/agent"
6
+ import { Provider } from "../../provider/provider"
7
+ import path from "path"
8
+ import fs from "fs/promises"
9
+ import matter from "gray-matter"
10
+ import { Instance } from "../../project/instance"
11
+ import { EOL } from "os"
12
+ import type { Argv } from "yargs"
13
+
14
+ type AgentMode = "all" | "primary" | "subagent"
15
+
16
+ const AVAILABLE_TOOLS = [
17
+ "bash",
18
+ "read",
19
+ "write",
20
+ "edit",
21
+ "list",
22
+ "glob",
23
+ "grep",
24
+ "webfetch",
25
+ "task",
26
+ "todowrite",
27
+ "todoread",
28
+ ]
29
+
30
+ const AgentCreateCommand = cmd({
31
+ command: "create",
32
+ describe: "create a new agent",
33
+ builder: (yargs: Argv) =>
34
+ yargs
35
+ .option("path", {
36
+ type: "string",
37
+ describe: "directory path to generate the agent file",
38
+ })
39
+ .option("description", {
40
+ type: "string",
41
+ describe: "what the agent should do",
42
+ })
43
+ .option("mode", {
44
+ type: "string",
45
+ describe: "agent mode",
46
+ choices: ["all", "primary", "subagent"] as const,
47
+ })
48
+ .option("tools", {
49
+ type: "string",
50
+ describe: `comma-separated list of tools to enable (default: all). Available: "${AVAILABLE_TOOLS.join(", ")}"`,
51
+ })
52
+ .option("model", {
53
+ type: "string",
54
+ alias: ["m"],
55
+ describe: "model to use in the format of provider/model",
56
+ }),
57
+ async handler(args) {
58
+ await Instance.provide({
59
+ directory: process.cwd(),
60
+ async fn() {
61
+ const cliPath = args.path
62
+ const cliDescription = args.description
63
+ const cliMode = args.mode as AgentMode | undefined
64
+ const cliTools = args.tools
65
+
66
+ const isFullyNonInteractive = cliPath && cliDescription && cliMode && cliTools !== undefined
67
+
68
+ if (!isFullyNonInteractive) {
69
+ UI.empty()
70
+ prompts.intro("Create agent")
71
+ }
72
+
73
+ const project = Instance.project
74
+
75
+ // Determine scope/path
76
+ let targetPath: string
77
+ if (cliPath) {
78
+ targetPath = path.join(cliPath, "agent")
79
+ } else {
80
+ let scope: "global" | "project" = "global"
81
+ if (project.vcs === "git") {
82
+ const scopeResult = await prompts.select({
83
+ message: "Location",
84
+ options: [
85
+ {
86
+ label: "Current project",
87
+ value: "project" as const,
88
+ hint: Instance.worktree,
89
+ },
90
+ {
91
+ label: "Global",
92
+ value: "global" as const,
93
+ hint: Global.Path.config,
94
+ },
95
+ ],
96
+ })
97
+ if (prompts.isCancel(scopeResult)) throw new UI.CancelledError()
98
+ scope = scopeResult
99
+ }
100
+ targetPath = path.join(
101
+ scope === "global" ? Global.Path.config : path.join(Instance.worktree, ".easc"),
102
+ "agent",
103
+ )
104
+ }
105
+
106
+ // Get description
107
+ let description: string
108
+ if (cliDescription) {
109
+ description = cliDescription
110
+ } else {
111
+ const query = await prompts.text({
112
+ message: "Description",
113
+ placeholder: "What should this agent do?",
114
+ validate: (x) => (x && x.length > 0 ? undefined : "Required"),
115
+ })
116
+ if (prompts.isCancel(query)) throw new UI.CancelledError()
117
+ description = query
118
+ }
119
+
120
+ // Generate agent
121
+ const spinner = prompts.spinner()
122
+ spinner.start("Generating agent configuration...")
123
+ const model = args.model ? Provider.parseModel(args.model) : undefined
124
+ const generated = await Agent.generate({ description, model }).catch((error) => {
125
+ spinner.stop(`LLM failed to generate agent: ${error.message}`, 1)
126
+ if (isFullyNonInteractive) process.exit(1)
127
+ throw new UI.CancelledError()
128
+ })
129
+ spinner.stop(`Agent ${generated.identifier} generated`)
130
+
131
+ // Select tools
132
+ let selectedTools: string[]
133
+ if (cliTools !== undefined) {
134
+ selectedTools = cliTools ? cliTools.split(",").map((t) => t.trim()) : AVAILABLE_TOOLS
135
+ } else {
136
+ const result = await prompts.multiselect({
137
+ message: "Select tools to enable (Space to toggle)",
138
+ options: AVAILABLE_TOOLS.map((tool) => ({
139
+ label: tool,
140
+ value: tool,
141
+ })),
142
+ initialValues: AVAILABLE_TOOLS,
143
+ })
144
+ if (prompts.isCancel(result)) throw new UI.CancelledError()
145
+ selectedTools = result
146
+ }
147
+
148
+ // Get mode
149
+ let mode: AgentMode
150
+ if (cliMode) {
151
+ mode = cliMode
152
+ } else {
153
+ const modeResult = await prompts.select({
154
+ message: "Agent mode",
155
+ options: [
156
+ {
157
+ label: "All",
158
+ value: "all" as const,
159
+ hint: "Can function in both primary and subagent roles",
160
+ },
161
+ {
162
+ label: "Primary",
163
+ value: "primary" as const,
164
+ hint: "Acts as a primary/main agent",
165
+ },
166
+ {
167
+ label: "Subagent",
168
+ value: "subagent" as const,
169
+ hint: "Can be used as a subagent by other agents",
170
+ },
171
+ ],
172
+ initialValue: "all" as const,
173
+ })
174
+ if (prompts.isCancel(modeResult)) throw new UI.CancelledError()
175
+ mode = modeResult
176
+ }
177
+
178
+ // Build tools config
179
+ const tools: Record<string, boolean> = {}
180
+ for (const tool of AVAILABLE_TOOLS) {
181
+ if (!selectedTools.includes(tool)) {
182
+ tools[tool] = false
183
+ }
184
+ }
185
+
186
+ // Build frontmatter
187
+ const frontmatter: {
188
+ description: string
189
+ mode: AgentMode
190
+ tools?: Record<string, boolean>
191
+ } = {
192
+ description: generated.whenToUse,
193
+ mode,
194
+ }
195
+ if (Object.keys(tools).length > 0) {
196
+ frontmatter.tools = tools
197
+ }
198
+
199
+ // Write file
200
+ const content = matter.stringify(generated.systemPrompt, frontmatter)
201
+ const filePath = path.join(targetPath, `${generated.identifier}.md`)
202
+
203
+ await fs.mkdir(targetPath, { recursive: true })
204
+
205
+ const file = Bun.file(filePath)
206
+ if (await file.exists()) {
207
+ if (isFullyNonInteractive) {
208
+ console.error(`Error: Agent file already exists: ${filePath}`)
209
+ process.exit(1)
210
+ }
211
+ prompts.log.error(`Agent file already exists: ${filePath}`)
212
+ throw new UI.CancelledError()
213
+ }
214
+
215
+ await Bun.write(filePath, content)
216
+
217
+ if (isFullyNonInteractive) {
218
+ console.log(filePath)
219
+ } else {
220
+ prompts.log.success(`Agent created: ${filePath}`)
221
+ prompts.outro("Done")
222
+ }
223
+ },
224
+ })
225
+ },
226
+ })
227
+
228
+ const AgentListCommand = cmd({
229
+ command: "list",
230
+ describe: "list all available agents",
231
+ async handler() {
232
+ await Instance.provide({
233
+ directory: process.cwd(),
234
+ async fn() {
235
+ const agents = await Agent.list()
236
+ const sortedAgents = agents.sort((a, b) => {
237
+ if (a.native !== b.native) {
238
+ return a.native ? -1 : 1
239
+ }
240
+ return a.name.localeCompare(b.name)
241
+ })
242
+
243
+ for (const agent of sortedAgents) {
244
+ process.stdout.write(`${agent.name} (${agent.mode})` + EOL)
245
+ process.stdout.write(` ${JSON.stringify(agent.permission, null, 2)}` + EOL)
246
+ }
247
+ },
248
+ })
249
+ },
250
+ })
251
+
252
+ export const AgentCommand = cmd({
253
+ command: "agent",
254
+ describe: "manage agents",
255
+ builder: (yargs) => yargs.command(AgentCreateCommand).command(AgentListCommand).demandCommand(),
256
+ async handler() { },
257
+ })
@@ -0,0 +1,427 @@
1
+ import { Auth } from "../../auth"
2
+ import { cmd } from "./cmd"
3
+ import * as prompts from "@clack/prompts"
4
+ import { UI } from "../ui"
5
+ import { ModelsDev } from "../../provider/models"
6
+ import { map, pipe, sortBy, values } from "remeda"
7
+ import path from "path"
8
+ import os from "os"
9
+ import { Config } from "../../config/config"
10
+ import { Global } from "../../global"
11
+ import { Plugin } from "../../plugin"
12
+ import { Instance } from "../../project/instance"
13
+ import type { Hooks } from "@eliseart.ai/plugin"
14
+ import { EliseArtAuth } from "../../auth/eliseart"
15
+
16
+ type PluginAuth = NonNullable<Hooks["auth"]>
17
+
18
+ /**
19
+ * Handle plugin-based authentication flow.
20
+ * Returns true if auth was handled, false if it should fall through to default handling.
21
+ */
22
+ async function handlePluginAuth(plugin: { auth: PluginAuth }, provider: string): Promise<boolean> {
23
+ let index = 0
24
+ if (plugin.auth.methods.length > 1) {
25
+ const method = await prompts.select({
26
+ message: "Login method",
27
+ options: [
28
+ ...plugin.auth.methods.map((x, index) => ({
29
+ label: x.label,
30
+ value: index.toString(),
31
+ })),
32
+ ],
33
+ })
34
+ if (prompts.isCancel(method)) throw new UI.CancelledError()
35
+ index = parseInt(method)
36
+ }
37
+ const method = plugin.auth.methods[index]
38
+
39
+ // Handle prompts for all auth types
40
+ await Bun.sleep(10)
41
+ const inputs: Record<string, string> = {}
42
+ if (method.prompts) {
43
+ for (const prompt of method.prompts) {
44
+ if (prompt.condition && !prompt.condition(inputs)) {
45
+ continue
46
+ }
47
+ if (prompt.type === "select") {
48
+ const value = await prompts.select({
49
+ message: prompt.message,
50
+ options: prompt.options,
51
+ })
52
+ if (prompts.isCancel(value)) throw new UI.CancelledError()
53
+ inputs[prompt.key] = value
54
+ } else {
55
+ const value = await prompts.text({
56
+ message: prompt.message,
57
+ placeholder: prompt.placeholder,
58
+ validate: prompt.validate ? (v) => prompt.validate!(v ?? "") : undefined,
59
+ })
60
+ if (prompts.isCancel(value)) throw new UI.CancelledError()
61
+ inputs[prompt.key] = value
62
+ }
63
+ }
64
+ }
65
+
66
+ if (method.type === "oauth") {
67
+ const authorize = await method.authorize(inputs)
68
+
69
+ if (authorize.url) {
70
+ prompts.log.info("Go to: " + authorize.url)
71
+ }
72
+
73
+ if (authorize.method === "auto") {
74
+ if (authorize.instructions) {
75
+ prompts.log.info(authorize.instructions)
76
+ }
77
+ const spinner = prompts.spinner()
78
+ spinner.start("Waiting for authorization...")
79
+ const result = await authorize.callback()
80
+ if (result.type === "failed") {
81
+ spinner.stop("Failed to authorize", 1)
82
+ }
83
+ if (result.type === "success") {
84
+ const saveProvider = result.provider ?? provider
85
+ if ("refresh" in result) {
86
+ const { type: _, provider: __, refresh, access, expires, ...extraFields } = result
87
+ await Auth.set(saveProvider, {
88
+ type: "oauth",
89
+ refresh,
90
+ access,
91
+ expires,
92
+ ...extraFields,
93
+ })
94
+ }
95
+ if ("key" in result) {
96
+ await Auth.set(saveProvider, {
97
+ type: "api",
98
+ key: result.key,
99
+ })
100
+ }
101
+ spinner.stop("Login successful")
102
+ }
103
+ }
104
+
105
+ if (authorize.method === "code") {
106
+ const code = await prompts.text({
107
+ message: "Paste the authorization code here: ",
108
+ validate: (x) => (x && x.length > 0 ? undefined : "Required"),
109
+ })
110
+ if (prompts.isCancel(code)) throw new UI.CancelledError()
111
+ const result = await authorize.callback(code)
112
+ if (result.type === "failed") {
113
+ prompts.log.error("Failed to authorize")
114
+ }
115
+ if (result.type === "success") {
116
+ const saveProvider = result.provider ?? provider
117
+ if ("refresh" in result) {
118
+ const { type: _, provider: __, refresh, access, expires, ...extraFields } = result
119
+ await Auth.set(saveProvider, {
120
+ type: "oauth",
121
+ refresh,
122
+ access,
123
+ expires,
124
+ ...extraFields,
125
+ })
126
+ }
127
+ if ("key" in result) {
128
+ await Auth.set(saveProvider, {
129
+ type: "api",
130
+ key: result.key,
131
+ })
132
+ }
133
+ prompts.log.success("Login successful")
134
+ }
135
+ }
136
+
137
+ prompts.outro("Done")
138
+ return true
139
+ }
140
+
141
+ if (method.type === "api") {
142
+ if (method.authorize) {
143
+ const result = await method.authorize(inputs)
144
+ if (result.type === "failed") {
145
+ prompts.log.error("Failed to authorize")
146
+ }
147
+ if (result.type === "success") {
148
+ const saveProvider = result.provider ?? provider
149
+ await Auth.set(saveProvider, {
150
+ type: "api",
151
+ key: result.key,
152
+ })
153
+ prompts.log.success("Login successful")
154
+ }
155
+ prompts.outro("Done")
156
+ return true
157
+ }
158
+ }
159
+
160
+ return false
161
+ }
162
+
163
+ export const AuthCommand = cmd({
164
+ command: "auth",
165
+ describe: "manage credentials",
166
+ builder: (yargs) =>
167
+ yargs.command(AuthLoginCommand).command(AuthLogoutCommand).command(AuthListCommand).demandCommand(),
168
+ async handler() { },
169
+ })
170
+
171
+ export const AuthListCommand = cmd({
172
+ command: "list",
173
+ aliases: ["ls"],
174
+ describe: "list providers",
175
+ async handler() {
176
+ UI.empty()
177
+ const authPath = path.join(Global.Path.data, "auth.json")
178
+ const homedir = os.homedir()
179
+ const displayPath = authPath.startsWith(homedir) ? authPath.replace(homedir, "~") : authPath
180
+ prompts.intro(`Credentials ${UI.Style.TEXT_DIM}${displayPath}`)
181
+ const results = Object.entries(await Auth.all())
182
+ const database = await ModelsDev.get()
183
+
184
+ for (const [providerID, result] of results) {
185
+ const name = database[providerID]?.name || providerID
186
+ prompts.log.info(`${name} ${UI.Style.TEXT_DIM}${result.type}`)
187
+ }
188
+
189
+ prompts.outro(`${results.length} credentials`)
190
+
191
+ // Environment variables section
192
+ const activeEnvVars: Array<{ provider: string; envVar: string }> = []
193
+
194
+ for (const [providerID, provider] of Object.entries(database)) {
195
+ for (const envVar of provider.env) {
196
+ if (process.env[envVar]) {
197
+ activeEnvVars.push({
198
+ provider: provider.name || providerID,
199
+ envVar,
200
+ })
201
+ }
202
+ }
203
+ }
204
+
205
+ if (activeEnvVars.length > 0) {
206
+ UI.empty()
207
+ prompts.intro("Environment")
208
+
209
+ for (const { provider, envVar } of activeEnvVars) {
210
+ prompts.log.info(`${provider} ${UI.Style.TEXT_DIM}${envVar}`)
211
+ }
212
+
213
+ prompts.outro(`${activeEnvVars.length} environment variable` + (activeEnvVars.length === 1 ? "" : "s"))
214
+ }
215
+ },
216
+ })
217
+
218
+ export const AuthLoginCommand = cmd({
219
+ command: "login [url]",
220
+ describe: "log in to a provider",
221
+ builder: (yargs) =>
222
+ yargs.positional("url", {
223
+ describe: "opencode auth provider",
224
+ type: "string",
225
+ }),
226
+ async handler(args) {
227
+ await Instance.provide({
228
+ directory: process.cwd(),
229
+ async fn() {
230
+ UI.empty()
231
+ prompts.intro("Add credential")
232
+ if (args.url && (args.url.startsWith("http://") || args.url.startsWith("https://"))) {
233
+ const wellknown = await fetch(`${args.url}/.well-known/opencode`).then((x) => x.json() as any)
234
+ prompts.log.info(`Running \`${wellknown.auth.command.join(" ")}\``)
235
+ const proc = Bun.spawn({
236
+ cmd: wellknown.auth.command,
237
+ stdout: "pipe",
238
+ })
239
+ const exit = await proc.exited
240
+ if (exit !== 0) {
241
+ prompts.log.error("Failed")
242
+ prompts.outro("Done")
243
+ return
244
+ }
245
+ const token = await new Response(proc.stdout).text()
246
+ await Auth.set(args.url, {
247
+ type: "wellknown",
248
+ key: wellknown.auth.env,
249
+ token: token.trim(),
250
+ })
251
+ prompts.log.success("Logged into " + args.url)
252
+ prompts.outro("Done")
253
+ return
254
+ }
255
+ await ModelsDev.refresh().catch(() => { })
256
+
257
+ const config = await Config.get()
258
+
259
+ const disabled = new Set(config.disabled_providers ?? [])
260
+ const enabled = config.enabled_providers ? new Set(config.enabled_providers) : undefined
261
+
262
+ const providers = await ModelsDev.get().then((x) => {
263
+ const filtered: Record<string, (typeof x)[string]> = {}
264
+ for (const [key, value] of Object.entries(x)) {
265
+ if ((enabled ? enabled.has(key) : true) && !disabled.has(key)) {
266
+ filtered[key] = value
267
+ }
268
+ }
269
+ return filtered
270
+ })
271
+
272
+ const priority: Record<string, number> = {
273
+ eliseart: 0,
274
+ opencode: 1,
275
+ anthropic: 2,
276
+ "github-copilot": 3,
277
+ openai: 4,
278
+ google: 5,
279
+ openrouter: 6,
280
+ vercel: 7,
281
+ }
282
+ let providerInput: string | symbol | undefined = args.url
283
+ if (!providerInput) {
284
+ providerInput = await prompts.autocomplete({
285
+ message: "Select provider",
286
+ maxItems: 8,
287
+ options: [
288
+ ...pipe(
289
+ providers,
290
+ values(),
291
+ sortBy(
292
+ (x) => priority[x.id] ?? 99,
293
+ (x) => x.name ?? x.id,
294
+ ),
295
+ map((x) => ({
296
+ label: x.name,
297
+ value: x.id,
298
+ hint: {
299
+ opencode: "recommended",
300
+ anthropic: "Claude Max or API key",
301
+ openai: "ChatGPT Plus/Pro or API key",
302
+ }[x.id],
303
+ })),
304
+ ),
305
+ {
306
+ value: "other",
307
+ label: "Other",
308
+ },
309
+ ],
310
+ })
311
+ }
312
+
313
+ if (prompts.isCancel(providerInput)) throw new UI.CancelledError()
314
+ let provider = providerInput as string
315
+
316
+ if (provider === "eliseart") {
317
+ prompts.log.info("To login, visit: https://elise-art.web.app/cli-auth")
318
+ const token = await prompts.text({
319
+ message: "Paste your access token",
320
+ validate: (x) => (x && x.length > 20 ? undefined : "Invalid token"),
321
+ })
322
+ if (prompts.isCancel(token)) throw new UI.CancelledError()
323
+
324
+ const spinner = prompts.spinner()
325
+ spinner.start("Verifying token...")
326
+ try {
327
+ await EliseArtAuth.loginWithToken(token as string)
328
+ spinner.stop("Login successful")
329
+ } catch (e: any) {
330
+ spinner.stop("Login failed: " + e.message, 1)
331
+ }
332
+ prompts.outro("Done")
333
+ return
334
+ }
335
+
336
+ const plugin = await Plugin.list().then((x) => x.find((x) => x.auth?.provider === provider))
337
+ if (plugin && plugin.auth) {
338
+ const handled = await handlePluginAuth({ auth: plugin.auth }, provider)
339
+ if (handled) return
340
+ }
341
+
342
+ if (provider === "other") {
343
+ const result = await prompts.text({
344
+ message: "Enter provider id",
345
+ validate: (x) => (x && x.match(/^[0-9a-z-]+$/) ? undefined : "a-z, 0-9 and hyphens only"),
346
+ })
347
+ if (prompts.isCancel(result)) throw new UI.CancelledError()
348
+ provider = result as string
349
+ provider = provider.replace(/^@ai-sdk\//, "")
350
+ if (prompts.isCancel(provider)) throw new UI.CancelledError()
351
+
352
+ // Check if a plugin provides auth for this custom provider
353
+ const customPlugin = await Plugin.list().then((x) => x.find((x) => x.auth?.provider === provider))
354
+ if (customPlugin && customPlugin.auth) {
355
+ const handled = await handlePluginAuth({ auth: customPlugin.auth }, provider)
356
+ if (handled) return
357
+ }
358
+
359
+ prompts.log.warn(
360
+ `This only stores a credential for ${provider} - you will need configure it in opencode.json, check the docs for examples.`,
361
+ )
362
+ }
363
+
364
+ if (provider === "amazon-bedrock") {
365
+ prompts.log.info(
366
+ "Amazon Bedrock authentication priority:\n" +
367
+ " 1. Bearer token (AWS_BEARER_TOKEN_BEDROCK or /connect)\n" +
368
+ " 2. AWS credential chain (profile, access keys, IAM roles, EKS IRSA)\n\n" +
369
+ "Configure via opencode.json options (profile, region, endpoint) or\n" +
370
+ "AWS environment variables (AWS_PROFILE, AWS_REGION, AWS_ACCESS_KEY_ID, AWS_WEB_IDENTITY_TOKEN_FILE).",
371
+ )
372
+ }
373
+
374
+ if (provider === "opencode") {
375
+ prompts.log.info("Create an api key at https://eliseart.ai/auth")
376
+ }
377
+
378
+ if (provider === "vercel") {
379
+ prompts.log.info("You can create an api key at https://vercel.link/ai-gateway-token")
380
+ }
381
+
382
+ if (["cloudflare", "cloudflare-ai-gateway"].includes(provider)) {
383
+ prompts.log.info(
384
+ "Cloudflare AI Gateway can be configured with CLOUDFLARE_GATEWAY_ID, CLOUDFLARE_ACCOUNT_ID, and CLOUDFLARE_API_TOKEN environment variables. Read more: https://eliseart.ai/docs/providers/#cloudflare-ai-gateway",
385
+ )
386
+ }
387
+
388
+ const key = await prompts.password({
389
+ message: "Enter your API key",
390
+ validate: (x) => (x && x.length > 0 ? undefined : "Required"),
391
+ })
392
+ if (prompts.isCancel(key)) throw new UI.CancelledError()
393
+ await Auth.set(provider, {
394
+ type: "api",
395
+ key,
396
+ })
397
+
398
+ prompts.outro("Done")
399
+ },
400
+ })
401
+ },
402
+ })
403
+
404
+ export const AuthLogoutCommand = cmd({
405
+ command: "logout",
406
+ describe: "log out from a configured provider",
407
+ async handler() {
408
+ UI.empty()
409
+ const credentials = await Auth.all().then((x) => Object.entries(x))
410
+ prompts.intro("Remove credential")
411
+ if (credentials.length === 0) {
412
+ prompts.log.error("No credentials found")
413
+ return
414
+ }
415
+ const database = await ModelsDev.get()
416
+ const providerID = await prompts.select({
417
+ message: "Select provider",
418
+ options: credentials.map(([key, value]) => ({
419
+ label: (database[key]?.name || key) + UI.Style.TEXT_DIM + " (" + value.type + ")",
420
+ value: key,
421
+ })),
422
+ })
423
+ if (prompts.isCancel(providerID)) throw new UI.CancelledError()
424
+ await Auth.remove(providerID)
425
+ prompts.outro("Logout successful")
426
+ },
427
+ })
@@ -0,0 +1,7 @@
1
+ import type { CommandModule } from "yargs"
2
+
3
+ type WithDoubleDash<T> = T & { "--"?: string[] }
4
+
5
+ export function cmd<T, U>(input: CommandModule<T, WithDoubleDash<U>>) {
6
+ return input
7
+ }