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,357 @@
1
+ import type { Argv } from "yargs"
2
+ import { UI } from "../ui"
3
+ import * as prompts from "@clack/prompts"
4
+ import { Installation } from "../../installation"
5
+ import { Global } from "../../global"
6
+ import { $ } from "bun"
7
+ import fs from "fs/promises"
8
+ import path from "path"
9
+ import os from "os"
10
+
11
+ interface UninstallArgs {
12
+ keepConfig: boolean
13
+ keepData: boolean
14
+ dryRun: boolean
15
+ force: boolean
16
+ }
17
+
18
+ interface RemovalTargets {
19
+ directories: Array<{ path: string; label: string; keep: boolean }>
20
+ shellConfig: string | null
21
+ binary: string | null
22
+ }
23
+
24
+ export const UninstallCommand = {
25
+ command: "uninstall",
26
+ describe: "uninstall opencode and remove all related files",
27
+ builder: (yargs: Argv) =>
28
+ yargs
29
+ .option("keep-config", {
30
+ alias: "c",
31
+ type: "boolean",
32
+ describe: "keep configuration files",
33
+ default: false,
34
+ })
35
+ .option("keep-data", {
36
+ alias: "d",
37
+ type: "boolean",
38
+ describe: "keep session data and snapshots",
39
+ default: false,
40
+ })
41
+ .option("dry-run", {
42
+ type: "boolean",
43
+ describe: "show what would be removed without removing",
44
+ default: false,
45
+ })
46
+ .option("force", {
47
+ alias: "f",
48
+ type: "boolean",
49
+ describe: "skip confirmation prompts",
50
+ default: false,
51
+ }),
52
+
53
+ handler: async (args: UninstallArgs) => {
54
+ UI.empty()
55
+ UI.println(UI.logo(" "))
56
+ UI.empty()
57
+ prompts.intro("Uninstall OpenCode")
58
+
59
+ const method = await Installation.method()
60
+ prompts.log.info(`Installation method: ${method}`)
61
+
62
+ const targets = await collectRemovalTargets(args, method)
63
+
64
+ await showRemovalSummary(targets, method)
65
+
66
+ if (!args.force && !args.dryRun) {
67
+ const confirm = await prompts.confirm({
68
+ message: "Are you sure you want to uninstall?",
69
+ initialValue: false,
70
+ })
71
+ if (!confirm || prompts.isCancel(confirm)) {
72
+ prompts.outro("Cancelled")
73
+ return
74
+ }
75
+ }
76
+
77
+ if (args.dryRun) {
78
+ prompts.log.warn("Dry run - no changes made")
79
+ prompts.outro("Done")
80
+ return
81
+ }
82
+
83
+ await executeUninstall(method, targets)
84
+
85
+ prompts.outro("Done")
86
+ },
87
+ }
88
+
89
+ async function collectRemovalTargets(args: UninstallArgs, method: Installation.Method): Promise<RemovalTargets> {
90
+ const directories: RemovalTargets["directories"] = [
91
+ { path: Global.Path.data, label: "Data", keep: args.keepData },
92
+ { path: Global.Path.cache, label: "Cache", keep: false },
93
+ { path: Global.Path.config, label: "Config", keep: args.keepConfig },
94
+ { path: Global.Path.state, label: "State", keep: false },
95
+ ]
96
+
97
+ const shellConfig = method === "curl" ? await getShellConfigFile() : null
98
+ const binary = method === "curl" ? process.execPath : null
99
+
100
+ return { directories, shellConfig, binary }
101
+ }
102
+
103
+ async function showRemovalSummary(targets: RemovalTargets, method: Installation.Method) {
104
+ prompts.log.message("The following will be removed:")
105
+
106
+ for (const dir of targets.directories) {
107
+ const exists = await fs
108
+ .access(dir.path)
109
+ .then(() => true)
110
+ .catch(() => false)
111
+ if (!exists) continue
112
+
113
+ const size = await getDirectorySize(dir.path)
114
+ const sizeStr = formatSize(size)
115
+ const status = dir.keep ? UI.Style.TEXT_DIM + "(keeping)" : ""
116
+ const prefix = dir.keep ? "○" : "✓"
117
+
118
+ prompts.log.info(` ${prefix} ${dir.label}: ${shortenPath(dir.path)} ${UI.Style.TEXT_DIM}(${sizeStr})${status}`)
119
+ }
120
+
121
+ if (targets.binary) {
122
+ prompts.log.info(` ✓ Binary: ${shortenPath(targets.binary)}`)
123
+ }
124
+
125
+ if (targets.shellConfig) {
126
+ prompts.log.info(` ✓ Shell PATH in ${shortenPath(targets.shellConfig)}`)
127
+ }
128
+
129
+ if (method !== "curl" && method !== "unknown") {
130
+ const cmds: Record<string, string> = {
131
+ npm: "npm uninstall -g eliseart.ai",
132
+ pnpm: "pnpm uninstall -g eliseart.ai",
133
+ bun: "bun remove -g eliseart.ai",
134
+ yarn: "yarn global remove eliseart.ai",
135
+ brew: "brew uninstall opencode",
136
+ choco: "choco uninstall opencode",
137
+ scoop: "scoop uninstall opencode",
138
+ }
139
+ prompts.log.info(` ✓ Package: ${cmds[method] || method}`)
140
+ }
141
+ }
142
+
143
+ async function executeUninstall(method: Installation.Method, targets: RemovalTargets) {
144
+ const spinner = prompts.spinner()
145
+ const errors: string[] = []
146
+
147
+ for (const dir of targets.directories) {
148
+ if (dir.keep) {
149
+ prompts.log.step(`Skipping ${dir.label} (--keep-${dir.label.toLowerCase()})`)
150
+ continue
151
+ }
152
+
153
+ const exists = await fs
154
+ .access(dir.path)
155
+ .then(() => true)
156
+ .catch(() => false)
157
+ if (!exists) continue
158
+
159
+ spinner.start(`Removing ${dir.label}...`)
160
+ const err = await fs.rm(dir.path, { recursive: true, force: true }).catch((e) => e)
161
+ if (err) {
162
+ spinner.stop(`Failed to remove ${dir.label}`, 1)
163
+ errors.push(`${dir.label}: ${err.message}`)
164
+ continue
165
+ }
166
+ spinner.stop(`Removed ${dir.label}`)
167
+ }
168
+
169
+ if (targets.shellConfig) {
170
+ spinner.start("Cleaning shell config...")
171
+ const err = await cleanShellConfig(targets.shellConfig).catch((e) => e)
172
+ if (err) {
173
+ spinner.stop("Failed to clean shell config", 1)
174
+ errors.push(`Shell config: ${err.message}`)
175
+ } else {
176
+ spinner.stop("Cleaned shell config")
177
+ }
178
+ }
179
+
180
+ if (method !== "curl" && method !== "unknown") {
181
+ const cmds: Record<string, string[]> = {
182
+ npm: ["npm", "uninstall", "-g", "eliseart.ai"],
183
+ pnpm: ["pnpm", "uninstall", "-g", "eliseart.ai"],
184
+ bun: ["bun", "remove", "-g", "eliseart.ai"],
185
+ yarn: ["yarn", "global", "remove", "eliseart.ai"],
186
+ brew: ["brew", "uninstall", "opencode"],
187
+ choco: ["choco", "uninstall", "opencode"],
188
+ scoop: ["scoop", "uninstall", "opencode"],
189
+ }
190
+
191
+ const cmd = cmds[method]
192
+ if (cmd) {
193
+ spinner.start(`Running ${cmd.join(" ")}...`)
194
+ const result =
195
+ method === "choco"
196
+ ? await $`echo Y | choco uninstall opencode -y -r`.quiet().nothrow()
197
+ : await $`${cmd}`.quiet().nothrow()
198
+ if (result.exitCode !== 0) {
199
+ spinner.stop(`Package manager uninstall failed: exit code ${result.exitCode}`, 1)
200
+ if (
201
+ method === "choco" &&
202
+ result.stdout.toString("utf8").includes("not running from an elevated command shell")
203
+ ) {
204
+ prompts.log.warn(`You may need to run '${cmd.join(" ")}' from an elevated command shell`)
205
+ } else {
206
+ prompts.log.warn(`You may need to run manually: ${cmd.join(" ")}`)
207
+ }
208
+ } else {
209
+ spinner.stop("Package removed")
210
+ }
211
+ }
212
+ }
213
+
214
+ if (method === "curl" && targets.binary) {
215
+ UI.empty()
216
+ prompts.log.message("To finish removing the binary, run:")
217
+ prompts.log.info(` rm "${targets.binary}"`)
218
+
219
+ const binDir = path.dirname(targets.binary)
220
+ if (binDir.includes(".opencode")) {
221
+ prompts.log.info(` rmdir "${binDir}" 2>/dev/null`)
222
+ }
223
+ }
224
+
225
+ if (errors.length > 0) {
226
+ UI.empty()
227
+ prompts.log.warn("Some operations failed:")
228
+ for (const err of errors) {
229
+ prompts.log.error(` ${err}`)
230
+ }
231
+ }
232
+
233
+ UI.empty()
234
+ prompts.log.success("Thank you for using OpenCode!")
235
+ }
236
+
237
+ async function getShellConfigFile(): Promise<string | null> {
238
+ const shell = path.basename(process.env.SHELL || "bash")
239
+ const home = os.homedir()
240
+ const xdgConfig = process.env.XDG_CONFIG_HOME || path.join(home, ".config")
241
+
242
+ const configFiles: Record<string, string[]> = {
243
+ fish: [path.join(xdgConfig, "fish", "config.fish")],
244
+ zsh: [
245
+ path.join(home, ".zshrc"),
246
+ path.join(home, ".zshenv"),
247
+ path.join(xdgConfig, "zsh", ".zshrc"),
248
+ path.join(xdgConfig, "zsh", ".zshenv"),
249
+ ],
250
+ bash: [
251
+ path.join(home, ".bashrc"),
252
+ path.join(home, ".bash_profile"),
253
+ path.join(home, ".profile"),
254
+ path.join(xdgConfig, "bash", ".bashrc"),
255
+ path.join(xdgConfig, "bash", ".bash_profile"),
256
+ ],
257
+ ash: [path.join(home, ".ashrc"), path.join(home, ".profile")],
258
+ sh: [path.join(home, ".profile")],
259
+ }
260
+
261
+ const candidates = configFiles[shell] || configFiles.bash
262
+
263
+ for (const file of candidates) {
264
+ const exists = await fs
265
+ .access(file)
266
+ .then(() => true)
267
+ .catch(() => false)
268
+ if (!exists) continue
269
+
270
+ const content = await Bun.file(file)
271
+ .text()
272
+ .catch(() => "")
273
+ if (content.includes("# opencode") || content.includes(".opencode/bin")) {
274
+ return file
275
+ }
276
+ }
277
+
278
+ return null
279
+ }
280
+
281
+ async function cleanShellConfig(file: string) {
282
+ const content = await Bun.file(file).text()
283
+ const lines = content.split("\n")
284
+
285
+ const filtered: string[] = []
286
+ let skip = false
287
+
288
+ for (const line of lines) {
289
+ const trimmed = line.trim()
290
+
291
+ if (trimmed === "# opencode") {
292
+ skip = true
293
+ continue
294
+ }
295
+
296
+ if (skip) {
297
+ skip = false
298
+ if (trimmed.includes(".opencode/bin") || trimmed.includes("fish_add_path")) {
299
+ continue
300
+ }
301
+ }
302
+
303
+ if (
304
+ (trimmed.startsWith("export PATH=") && trimmed.includes(".opencode/bin")) ||
305
+ (trimmed.startsWith("fish_add_path") && trimmed.includes(".opencode"))
306
+ ) {
307
+ continue
308
+ }
309
+
310
+ filtered.push(line)
311
+ }
312
+
313
+ while (filtered.length > 0 && filtered[filtered.length - 1].trim() === "") {
314
+ filtered.pop()
315
+ }
316
+
317
+ const output = filtered.join("\n") + "\n"
318
+ await Bun.write(file, output)
319
+ }
320
+
321
+ async function getDirectorySize(dir: string): Promise<number> {
322
+ let total = 0
323
+
324
+ const walk = async (current: string) => {
325
+ const entries = await fs.readdir(current, { withFileTypes: true }).catch(() => [])
326
+
327
+ for (const entry of entries) {
328
+ const full = path.join(current, entry.name)
329
+ if (entry.isDirectory()) {
330
+ await walk(full)
331
+ continue
332
+ }
333
+ if (entry.isFile()) {
334
+ const stat = await fs.stat(full).catch(() => null)
335
+ if (stat) total += stat.size
336
+ }
337
+ }
338
+ }
339
+
340
+ await walk(dir)
341
+ return total
342
+ }
343
+
344
+ function formatSize(bytes: number): string {
345
+ if (bytes < 1024) return `${bytes} B`
346
+ if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`
347
+ if (bytes < 1024 * 1024 * 1024) return `${(bytes / (1024 * 1024)).toFixed(1)} MB`
348
+ return `${(bytes / (1024 * 1024 * 1024)).toFixed(1)} GB`
349
+ }
350
+
351
+ function shortenPath(p: string): string {
352
+ const home = os.homedir()
353
+ if (p.startsWith(home)) {
354
+ return p.replace(home, "~")
355
+ }
356
+ return p
357
+ }
@@ -0,0 +1,73 @@
1
+ import type { Argv } from "yargs"
2
+ import { UI } from "../ui"
3
+ import * as prompts from "@clack/prompts"
4
+ import { Installation } from "../../installation"
5
+
6
+ export const UpgradeCommand = {
7
+ command: "upgrade [target]",
8
+ describe: "upgrade opencode to the latest or a specific version",
9
+ builder: (yargs: Argv) => {
10
+ return yargs
11
+ .positional("target", {
12
+ describe: "version to upgrade to, for ex '0.1.48' or 'v0.1.48'",
13
+ type: "string",
14
+ })
15
+ .option("method", {
16
+ alias: "m",
17
+ describe: "installation method to use",
18
+ type: "string",
19
+ choices: ["curl", "npm", "pnpm", "bun", "brew", "choco", "scoop"],
20
+ })
21
+ },
22
+ handler: async (args: { target?: string; method?: string }) => {
23
+ UI.empty()
24
+ UI.println(UI.logo(" "))
25
+ UI.empty()
26
+ prompts.intro("Upgrade")
27
+ const detectedMethod = await Installation.method()
28
+ const method = (args.method as Installation.Method) ?? detectedMethod
29
+ if (method === "unknown") {
30
+ prompts.log.error(`opencode is installed to ${process.execPath} and may be managed by a package manager`)
31
+ const install = await prompts.select({
32
+ message: "Install anyways?",
33
+ options: [
34
+ { label: "Yes", value: true },
35
+ { label: "No", value: false },
36
+ ],
37
+ initialValue: false,
38
+ })
39
+ if (!install) {
40
+ prompts.outro("Done")
41
+ return
42
+ }
43
+ }
44
+ prompts.log.info("Using method: " + method)
45
+ const target = args.target ? args.target.replace(/^v/, "") : await Installation.latest()
46
+
47
+ if (Installation.VERSION === target) {
48
+ prompts.log.warn(`opencode upgrade skipped: ${target} is already installed`)
49
+ prompts.outro("Done")
50
+ return
51
+ }
52
+
53
+ prompts.log.info(`From ${Installation.VERSION} → ${target}`)
54
+ const spinner = prompts.spinner()
55
+ spinner.start("Upgrading...")
56
+ const err = await Installation.upgrade(method, target).catch((err) => err)
57
+ if (err) {
58
+ spinner.stop("Upgrade failed", 1)
59
+ if (err instanceof Installation.UpgradeFailedError) {
60
+ // necessary because choco only allows install/upgrade in elevated terminals
61
+ if (method === "choco" && err.data.stderr.includes("not running from an elevated command shell")) {
62
+ prompts.log.error("Please run the terminal as Administrator and try again")
63
+ } else {
64
+ prompts.log.error(err.data.stderr)
65
+ }
66
+ } else if (err instanceof Error) prompts.log.error(err.message)
67
+ prompts.outro("Done")
68
+ return
69
+ }
70
+ spinner.stop("Upgrade complete")
71
+ prompts.outro("Done")
72
+ },
73
+ }
@@ -0,0 +1,81 @@
1
+ import { Server } from "../../server/server"
2
+ import { UI } from "../ui"
3
+ import { cmd } from "./cmd"
4
+ import { withNetworkOptions, resolveNetworkOptions } from "../network"
5
+ import { Flag } from "../../flag/flag"
6
+ import open from "open"
7
+ import { networkInterfaces } from "os"
8
+
9
+ function getNetworkIPs() {
10
+ const nets = networkInterfaces()
11
+ const results: string[] = []
12
+
13
+ for (const name of Object.keys(nets)) {
14
+ const net = nets[name]
15
+ if (!net) continue
16
+
17
+ for (const netInfo of net) {
18
+ // Skip internal and non-IPv4 addresses
19
+ if (netInfo.internal || netInfo.family !== "IPv4") continue
20
+
21
+ // Skip Docker bridge networks (typically 172.x.x.x)
22
+ if (netInfo.address.startsWith("172.")) continue
23
+
24
+ results.push(netInfo.address)
25
+ }
26
+ }
27
+
28
+ return results
29
+ }
30
+
31
+ export const WebCommand = cmd({
32
+ command: "web",
33
+ builder: (yargs) => withNetworkOptions(yargs),
34
+ describe: "start opencode server and open web interface",
35
+ handler: async (args) => {
36
+ if (!Flag.OPENCODE_SERVER_PASSWORD) {
37
+ UI.println(UI.Style.TEXT_WARNING_BOLD + "! " + "OPENCODE_SERVER_PASSWORD is not set; server is unsecured.")
38
+ }
39
+ const opts = await resolveNetworkOptions(args)
40
+ const server = Server.listen(opts)
41
+ UI.empty()
42
+ UI.println(UI.logo(" "))
43
+ UI.empty()
44
+
45
+ if (opts.hostname === "0.0.0.0") {
46
+ // Show localhost for local access
47
+ const localhostUrl = `http://localhost:${server.port}`
48
+ UI.println(UI.Style.TEXT_INFO_BOLD + " Local access: ", UI.Style.TEXT_NORMAL, localhostUrl)
49
+
50
+ // Show network IPs for remote access
51
+ const networkIPs = getNetworkIPs()
52
+ if (networkIPs.length > 0) {
53
+ for (const ip of networkIPs) {
54
+ UI.println(
55
+ UI.Style.TEXT_INFO_BOLD + " Network access: ",
56
+ UI.Style.TEXT_NORMAL,
57
+ `http://${ip}:${server.port}`,
58
+ )
59
+ }
60
+ }
61
+
62
+ if (opts.mdns) {
63
+ UI.println(
64
+ UI.Style.TEXT_INFO_BOLD + " mDNS: ",
65
+ UI.Style.TEXT_NORMAL,
66
+ `opencode.local:${server.port}`,
67
+ )
68
+ }
69
+
70
+ // Open localhost in browser
71
+ open(localhostUrl.toString()).catch(() => {})
72
+ } else {
73
+ const displayUrl = server.url.toString()
74
+ UI.println(UI.Style.TEXT_INFO_BOLD + " Web interface: ", UI.Style.TEXT_NORMAL, displayUrl)
75
+ open(displayUrl).catch(() => {})
76
+ }
77
+
78
+ await new Promise(() => {})
79
+ await server.stop()
80
+ },
81
+ })
@@ -0,0 +1,57 @@
1
+ import { ConfigMarkdown } from "@/config/markdown"
2
+ import { Config } from "../config/config"
3
+ import { MCP } from "../mcp"
4
+ import { Provider } from "../provider/provider"
5
+ import { UI } from "./ui"
6
+
7
+ export function FormatError(input: unknown) {
8
+ if (MCP.Failed.isInstance(input))
9
+ return `MCP server "${input.data.name}" failed. Note, opencode does not support MCP authentication yet.`
10
+ if (Provider.ModelNotFoundError.isInstance(input)) {
11
+ const { providerID, modelID, suggestions } = input.data
12
+ return [
13
+ `Model not found: ${providerID}/${modelID}`,
14
+ ...(Array.isArray(suggestions) && suggestions.length ? ["Did you mean: " + suggestions.join(", ")] : []),
15
+ `Try: \`opencode models\` to list available models`,
16
+ `Or check your config (opencode.json) provider/model names`,
17
+ ].join("\n")
18
+ }
19
+ if (Provider.InitError.isInstance(input)) {
20
+ return `Failed to initialize provider "${input.data.providerID}". Check credentials and configuration.`
21
+ }
22
+ if (Config.JsonError.isInstance(input)) {
23
+ return (
24
+ `Config file at ${input.data.path} is not valid JSON(C)` + (input.data.message ? `: ${input.data.message}` : "")
25
+ )
26
+ }
27
+ if (Config.ConfigDirectoryTypoError.isInstance(input)) {
28
+ return `Directory "${input.data.dir}" in ${input.data.path} is not valid. Rename the directory to "${input.data.suggestion}" or remove it. This is a common typo.`
29
+ }
30
+ if (ConfigMarkdown.FrontmatterError.isInstance(input)) {
31
+ return input.data.message
32
+ }
33
+ if (Config.InvalidError.isInstance(input))
34
+ return [
35
+ `Configuration is invalid${input.data.path && input.data.path !== "config" ? ` at ${input.data.path}` : ""}` +
36
+ (input.data.message ? `: ${input.data.message}` : ""),
37
+ ...(input.data.issues?.map((issue) => "↳ " + issue.message + " " + issue.path.join(".")) ?? []),
38
+ ].join("\n")
39
+
40
+ if (UI.CancelledError.isInstance(input)) return ""
41
+ }
42
+
43
+ export function FormatUnknownError(input: unknown): string {
44
+ if (input instanceof Error) {
45
+ return input.stack ?? `${input.name}: ${input.message}`
46
+ }
47
+
48
+ if (typeof input === "object" && input !== null) {
49
+ try {
50
+ return JSON.stringify(input, null, 2)
51
+ } catch {
52
+ return "Unexpected error (unserializable)"
53
+ }
54
+ }
55
+
56
+ return String(input)
57
+ }
@@ -0,0 +1,53 @@
1
+ import type { Argv, InferredOptionTypes } from "yargs"
2
+ import { Config } from "../config/config"
3
+
4
+ const options = {
5
+ port: {
6
+ type: "number" as const,
7
+ describe: "port to listen on",
8
+ default: 0,
9
+ },
10
+ hostname: {
11
+ type: "string" as const,
12
+ describe: "hostname to listen on",
13
+ default: "127.0.0.1",
14
+ },
15
+ mdns: {
16
+ type: "boolean" as const,
17
+ describe: "enable mDNS service discovery (defaults hostname to 0.0.0.0)",
18
+ default: false,
19
+ },
20
+ cors: {
21
+ type: "string" as const,
22
+ array: true,
23
+ describe: "additional domains to allow for CORS",
24
+ default: [] as string[],
25
+ },
26
+ }
27
+
28
+ export type NetworkOptions = InferredOptionTypes<typeof options>
29
+
30
+ export function withNetworkOptions<T>(yargs: Argv<T>) {
31
+ return yargs.options(options)
32
+ }
33
+
34
+ export async function resolveNetworkOptions(args: NetworkOptions) {
35
+ const config = await Config.global()
36
+ const portExplicitlySet = process.argv.includes("--port")
37
+ const hostnameExplicitlySet = process.argv.includes("--hostname")
38
+ const mdnsExplicitlySet = process.argv.includes("--mdns")
39
+ const corsExplicitlySet = process.argv.includes("--cors")
40
+
41
+ const mdns = mdnsExplicitlySet ? args.mdns : (config?.server?.mdns ?? args.mdns)
42
+ const port = portExplicitlySet ? args.port : (config?.server?.port ?? args.port)
43
+ const hostname = hostnameExplicitlySet
44
+ ? args.hostname
45
+ : mdns && !config?.server?.hostname
46
+ ? "0.0.0.0"
47
+ : (config?.server?.hostname ?? args.hostname)
48
+ const configCors = config?.server?.cors ?? []
49
+ const argsCors = Array.isArray(args.cors) ? args.cors : args.cors ? [args.cors] : []
50
+ const cors = [...configCors, ...argsCors]
51
+
52
+ return { hostname, port, mdns, cors }
53
+ }