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,493 @@
1
+ import type { Hooks, PluginInput } from "@eliseart.ai/plugin"
2
+ import { Log } from "../util/log"
3
+ import { OAUTH_DUMMY_KEY } from "../auth"
4
+ import { ProviderTransform } from "../provider/transform"
5
+
6
+ const log = Log.create({ service: "plugin.codex" })
7
+
8
+ const CLIENT_ID = "app_EMoamEEZ73f0CkXaXp7hrann"
9
+ const ISSUER = "https://auth.openai.com"
10
+ const CODEX_API_ENDPOINT = "https://chatgpt.com/backend-api/codex/responses"
11
+ const OAUTH_PORT = 1455
12
+
13
+ interface PkceCodes {
14
+ verifier: string
15
+ challenge: string
16
+ }
17
+
18
+ async function generatePKCE(): Promise<PkceCodes> {
19
+ const verifier = generateRandomString(43)
20
+ const encoder = new TextEncoder()
21
+ const data = encoder.encode(verifier)
22
+ const hash = await crypto.subtle.digest("SHA-256", data)
23
+ const challenge = base64UrlEncode(hash)
24
+ return { verifier, challenge }
25
+ }
26
+
27
+ function generateRandomString(length: number): string {
28
+ const chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-._~"
29
+ const bytes = crypto.getRandomValues(new Uint8Array(length))
30
+ return Array.from(bytes)
31
+ .map((b) => chars[b % chars.length])
32
+ .join("")
33
+ }
34
+
35
+ function base64UrlEncode(buffer: ArrayBuffer): string {
36
+ const bytes = new Uint8Array(buffer)
37
+ const binary = String.fromCharCode(...bytes)
38
+ return btoa(binary).replace(/\+/g, "-").replace(/\//g, "_").replace(/=+$/, "")
39
+ }
40
+
41
+ function generateState(): string {
42
+ return base64UrlEncode(crypto.getRandomValues(new Uint8Array(32)).buffer)
43
+ }
44
+
45
+ export interface IdTokenClaims {
46
+ chatgpt_account_id?: string
47
+ organizations?: Array<{ id: string }>
48
+ email?: string
49
+ "https://api.openai.com/auth"?: {
50
+ chatgpt_account_id?: string
51
+ }
52
+ }
53
+
54
+ export function parseJwtClaims(token: string): IdTokenClaims | undefined {
55
+ const parts = token.split(".")
56
+ if (parts.length !== 3) return undefined
57
+ try {
58
+ return JSON.parse(Buffer.from(parts[1], "base64url").toString())
59
+ } catch {
60
+ return undefined
61
+ }
62
+ }
63
+
64
+ export function extractAccountIdFromClaims(claims: IdTokenClaims): string | undefined {
65
+ return (
66
+ claims.chatgpt_account_id ||
67
+ claims["https://api.openai.com/auth"]?.chatgpt_account_id ||
68
+ claims.organizations?.[0]?.id
69
+ )
70
+ }
71
+
72
+ export function extractAccountId(tokens: TokenResponse): string | undefined {
73
+ if (tokens.id_token) {
74
+ const claims = parseJwtClaims(tokens.id_token)
75
+ const accountId = claims && extractAccountIdFromClaims(claims)
76
+ if (accountId) return accountId
77
+ }
78
+ if (tokens.access_token) {
79
+ const claims = parseJwtClaims(tokens.access_token)
80
+ return claims ? extractAccountIdFromClaims(claims) : undefined
81
+ }
82
+ return undefined
83
+ }
84
+
85
+ function buildAuthorizeUrl(redirectUri: string, pkce: PkceCodes, state: string): string {
86
+ const params = new URLSearchParams({
87
+ response_type: "code",
88
+ client_id: CLIENT_ID,
89
+ redirect_uri: redirectUri,
90
+ scope: "openid profile email offline_access",
91
+ code_challenge: pkce.challenge,
92
+ code_challenge_method: "S256",
93
+ id_token_add_organizations: "true",
94
+ codex_cli_simplified_flow: "true",
95
+ state,
96
+ originator: "opencode",
97
+ })
98
+ return `${ISSUER}/oauth/authorize?${params.toString()}`
99
+ }
100
+
101
+ interface TokenResponse {
102
+ id_token: string
103
+ access_token: string
104
+ refresh_token: string
105
+ expires_in?: number
106
+ }
107
+
108
+ async function exchangeCodeForTokens(code: string, redirectUri: string, pkce: PkceCodes): Promise<TokenResponse> {
109
+ const response = await fetch(`${ISSUER}/oauth/token`, {
110
+ method: "POST",
111
+ headers: { "Content-Type": "application/x-www-form-urlencoded" },
112
+ body: new URLSearchParams({
113
+ grant_type: "authorization_code",
114
+ code,
115
+ redirect_uri: redirectUri,
116
+ client_id: CLIENT_ID,
117
+ code_verifier: pkce.verifier,
118
+ }).toString(),
119
+ })
120
+ if (!response.ok) {
121
+ throw new Error(`Token exchange failed: ${response.status}`)
122
+ }
123
+ return response.json()
124
+ }
125
+
126
+ async function refreshAccessToken(refreshToken: string): Promise<TokenResponse> {
127
+ const response = await fetch(`${ISSUER}/oauth/token`, {
128
+ method: "POST",
129
+ headers: { "Content-Type": "application/x-www-form-urlencoded" },
130
+ body: new URLSearchParams({
131
+ grant_type: "refresh_token",
132
+ refresh_token: refreshToken,
133
+ client_id: CLIENT_ID,
134
+ }).toString(),
135
+ })
136
+ if (!response.ok) {
137
+ throw new Error(`Token refresh failed: ${response.status}`)
138
+ }
139
+ return response.json()
140
+ }
141
+
142
+ const HTML_SUCCESS = `<!doctype html>
143
+ <html>
144
+ <head>
145
+ <title>OpenCode - Codex Authorization Successful</title>
146
+ <style>
147
+ body {
148
+ font-family:
149
+ system-ui,
150
+ -apple-system,
151
+ sans-serif;
152
+ display: flex;
153
+ justify-content: center;
154
+ align-items: center;
155
+ height: 100vh;
156
+ margin: 0;
157
+ background: #131010;
158
+ color: #f1ecec;
159
+ }
160
+ .container {
161
+ text-align: center;
162
+ padding: 2rem;
163
+ }
164
+ h1 {
165
+ color: #f1ecec;
166
+ margin-bottom: 1rem;
167
+ }
168
+ p {
169
+ color: #b7b1b1;
170
+ }
171
+ </style>
172
+ </head>
173
+ <body>
174
+ <div class="container">
175
+ <h1>Authorization Successful</h1>
176
+ <p>You can close this window and return to OpenCode.</p>
177
+ </div>
178
+ <script>
179
+ setTimeout(() => window.close(), 2000)
180
+ </script>
181
+ </body>
182
+ </html>`
183
+
184
+ const HTML_ERROR = (error: string) => `<!doctype html>
185
+ <html>
186
+ <head>
187
+ <title>OpenCode - Codex Authorization Failed</title>
188
+ <style>
189
+ body {
190
+ font-family:
191
+ system-ui,
192
+ -apple-system,
193
+ sans-serif;
194
+ display: flex;
195
+ justify-content: center;
196
+ align-items: center;
197
+ height: 100vh;
198
+ margin: 0;
199
+ background: #131010;
200
+ color: #f1ecec;
201
+ }
202
+ .container {
203
+ text-align: center;
204
+ padding: 2rem;
205
+ }
206
+ h1 {
207
+ color: #fc533a;
208
+ margin-bottom: 1rem;
209
+ }
210
+ p {
211
+ color: #b7b1b1;
212
+ }
213
+ .error {
214
+ color: #ff917b;
215
+ font-family: monospace;
216
+ margin-top: 1rem;
217
+ padding: 1rem;
218
+ background: #3c140d;
219
+ border-radius: 0.5rem;
220
+ }
221
+ </style>
222
+ </head>
223
+ <body>
224
+ <div class="container">
225
+ <h1>Authorization Failed</h1>
226
+ <p>An error occurred during authorization.</p>
227
+ <div class="error">${error}</div>
228
+ </div>
229
+ </body>
230
+ </html>`
231
+
232
+ interface PendingOAuth {
233
+ pkce: PkceCodes
234
+ state: string
235
+ resolve: (tokens: TokenResponse) => void
236
+ reject: (error: Error) => void
237
+ }
238
+
239
+ let oauthServer: ReturnType<typeof Bun.serve> | undefined
240
+ let pendingOAuth: PendingOAuth | undefined
241
+
242
+ async function startOAuthServer(): Promise<{ port: number; redirectUri: string }> {
243
+ if (oauthServer) {
244
+ return { port: OAUTH_PORT, redirectUri: `http://localhost:${OAUTH_PORT}/auth/callback` }
245
+ }
246
+
247
+ oauthServer = Bun.serve({
248
+ port: OAUTH_PORT,
249
+ fetch(req) {
250
+ const url = new URL(req.url)
251
+
252
+ if (url.pathname === "/auth/callback") {
253
+ const code = url.searchParams.get("code")
254
+ const state = url.searchParams.get("state")
255
+ const error = url.searchParams.get("error")
256
+ const errorDescription = url.searchParams.get("error_description")
257
+
258
+ if (error) {
259
+ const errorMsg = errorDescription || error
260
+ pendingOAuth?.reject(new Error(errorMsg))
261
+ pendingOAuth = undefined
262
+ return new Response(HTML_ERROR(errorMsg), {
263
+ headers: { "Content-Type": "text/html" },
264
+ })
265
+ }
266
+
267
+ if (!code) {
268
+ const errorMsg = "Missing authorization code"
269
+ pendingOAuth?.reject(new Error(errorMsg))
270
+ pendingOAuth = undefined
271
+ return new Response(HTML_ERROR(errorMsg), {
272
+ status: 400,
273
+ headers: { "Content-Type": "text/html" },
274
+ })
275
+ }
276
+
277
+ if (!pendingOAuth || state !== pendingOAuth.state) {
278
+ const errorMsg = "Invalid state - potential CSRF attack"
279
+ pendingOAuth?.reject(new Error(errorMsg))
280
+ pendingOAuth = undefined
281
+ return new Response(HTML_ERROR(errorMsg), {
282
+ status: 400,
283
+ headers: { "Content-Type": "text/html" },
284
+ })
285
+ }
286
+
287
+ const current = pendingOAuth
288
+ pendingOAuth = undefined
289
+
290
+ exchangeCodeForTokens(code, `http://localhost:${OAUTH_PORT}/auth/callback`, current.pkce)
291
+ .then((tokens) => current.resolve(tokens))
292
+ .catch((err) => current.reject(err))
293
+
294
+ return new Response(HTML_SUCCESS, {
295
+ headers: { "Content-Type": "text/html" },
296
+ })
297
+ }
298
+
299
+ if (url.pathname === "/cancel") {
300
+ pendingOAuth?.reject(new Error("Login cancelled"))
301
+ pendingOAuth = undefined
302
+ return new Response("Login cancelled", { status: 200 })
303
+ }
304
+
305
+ return new Response("Not found", { status: 404 })
306
+ },
307
+ })
308
+
309
+ log.info("codex oauth server started", { port: OAUTH_PORT })
310
+ return { port: OAUTH_PORT, redirectUri: `http://localhost:${OAUTH_PORT}/auth/callback` }
311
+ }
312
+
313
+ function stopOAuthServer() {
314
+ if (oauthServer) {
315
+ oauthServer.stop()
316
+ oauthServer = undefined
317
+ log.info("codex oauth server stopped")
318
+ }
319
+ }
320
+
321
+ function waitForOAuthCallback(pkce: PkceCodes, state: string): Promise<TokenResponse> {
322
+ return new Promise((resolve, reject) => {
323
+ const timeout = setTimeout(
324
+ () => {
325
+ if (pendingOAuth) {
326
+ pendingOAuth = undefined
327
+ reject(new Error("OAuth callback timeout - authorization took too long"))
328
+ }
329
+ },
330
+ 5 * 60 * 1000,
331
+ ) // 5 minute timeout
332
+
333
+ pendingOAuth = {
334
+ pkce,
335
+ state,
336
+ resolve: (tokens) => {
337
+ clearTimeout(timeout)
338
+ resolve(tokens)
339
+ },
340
+ reject: (error) => {
341
+ clearTimeout(timeout)
342
+ reject(error)
343
+ },
344
+ }
345
+ })
346
+ }
347
+
348
+ export async function CodexAuthPlugin(input: PluginInput): Promise<Hooks> {
349
+ return {
350
+ auth: {
351
+ provider: "openai",
352
+ async loader(getAuth, provider) {
353
+ const auth = await getAuth()
354
+ if (auth.type !== "oauth") return {}
355
+
356
+ // Filter models to only allowed Codex models for OAuth
357
+ const allowedModels = new Set(["gpt-5.1-codex-max", "gpt-5.1-codex-mini", "gpt-5.2", "gpt-5.2-codex"])
358
+ for (const modelId of Object.keys(provider.models)) {
359
+ if (!allowedModels.has(modelId)) {
360
+ delete provider.models[modelId]
361
+ }
362
+ }
363
+
364
+ // Zero out costs for Codex (included with ChatGPT subscription)
365
+ for (const model of Object.values(provider.models)) {
366
+ model.cost = {
367
+ input: 0,
368
+ output: 0,
369
+ cache: { read: 0, write: 0 },
370
+ }
371
+ }
372
+
373
+ return {
374
+ apiKey: OAUTH_DUMMY_KEY,
375
+ async fetch(requestInput: RequestInfo | URL, init?: RequestInit) {
376
+ // Remove dummy API key authorization header
377
+ if (init?.headers) {
378
+ if (init.headers instanceof Headers) {
379
+ init.headers.delete("authorization")
380
+ init.headers.delete("Authorization")
381
+ } else if (Array.isArray(init.headers)) {
382
+ init.headers = init.headers.filter(([key]) => key.toLowerCase() !== "authorization")
383
+ } else {
384
+ delete init.headers["authorization"]
385
+ delete init.headers["Authorization"]
386
+ }
387
+ }
388
+
389
+ const currentAuth = await getAuth()
390
+ if (currentAuth.type !== "oauth") return fetch(requestInput, init)
391
+
392
+ // Cast to include accountId field
393
+ const authWithAccount = currentAuth as typeof currentAuth & { accountId?: string }
394
+
395
+ // Check if token needs refresh
396
+ if (!currentAuth.access || currentAuth.expires < Date.now()) {
397
+ log.info("refreshing codex access token")
398
+ const tokens = await refreshAccessToken(currentAuth.refresh)
399
+ const newAccountId = extractAccountId(tokens) || authWithAccount.accountId
400
+ await input.client.auth.set({
401
+ path: { id: "codex" },
402
+ body: {
403
+ type: "oauth",
404
+ refresh: tokens.refresh_token,
405
+ access: tokens.access_token,
406
+ expires: Date.now() + (tokens.expires_in ?? 3600) * 1000,
407
+ ...(newAccountId && { accountId: newAccountId }),
408
+ },
409
+ })
410
+ currentAuth.access = tokens.access_token
411
+ authWithAccount.accountId = newAccountId
412
+ }
413
+
414
+ // Build headers
415
+ const headers = new Headers()
416
+ if (init?.headers) {
417
+ if (init.headers instanceof Headers) {
418
+ init.headers.forEach((value, key) => headers.set(key, value))
419
+ } else if (Array.isArray(init.headers)) {
420
+ for (const [key, value] of init.headers) {
421
+ if (value !== undefined) headers.set(key, String(value))
422
+ }
423
+ } else {
424
+ for (const [key, value] of Object.entries(init.headers)) {
425
+ if (value !== undefined) headers.set(key, String(value))
426
+ }
427
+ }
428
+ }
429
+
430
+ // Set authorization header with access token
431
+ headers.set("authorization", `Bearer ${currentAuth.access}`)
432
+
433
+ // Set ChatGPT-Account-Id header for organization subscriptions
434
+ if (authWithAccount.accountId) {
435
+ headers.set("ChatGPT-Account-Id", authWithAccount.accountId)
436
+ }
437
+
438
+ // Rewrite URL to Codex endpoint
439
+ const parsed =
440
+ requestInput instanceof URL
441
+ ? requestInput
442
+ : new URL(typeof requestInput === "string" ? requestInput : requestInput.url)
443
+ const url =
444
+ parsed.pathname.includes("/v1/responses") || parsed.pathname.includes("/chat/completions")
445
+ ? new URL(CODEX_API_ENDPOINT)
446
+ : parsed
447
+
448
+ return fetch(url, {
449
+ ...init,
450
+ headers,
451
+ })
452
+ },
453
+ }
454
+ },
455
+ methods: [
456
+ {
457
+ label: "ChatGPT Pro/Plus",
458
+ type: "oauth",
459
+ authorize: async () => {
460
+ const { redirectUri } = await startOAuthServer()
461
+ const pkce = await generatePKCE()
462
+ const state = generateState()
463
+ const authUrl = buildAuthorizeUrl(redirectUri, pkce, state)
464
+
465
+ const callbackPromise = waitForOAuthCallback(pkce, state)
466
+
467
+ return {
468
+ url: authUrl,
469
+ instructions: "Complete authorization in your browser. This window will close automatically.",
470
+ method: "auto" as const,
471
+ callback: async () => {
472
+ const tokens = await callbackPromise
473
+ stopOAuthServer()
474
+ const accountId = extractAccountId(tokens)
475
+ return {
476
+ type: "success" as const,
477
+ refresh: tokens.refresh_token,
478
+ access: tokens.access_token,
479
+ expires: Date.now() + (tokens.expires_in ?? 3600) * 1000,
480
+ accountId,
481
+ }
482
+ },
483
+ }
484
+ },
485
+ },
486
+ {
487
+ label: "Manually enter API Key",
488
+ type: "api",
489
+ },
490
+ ],
491
+ },
492
+ }
493
+ }