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,90 @@
1
+ import type { NamedError } from "@eliseart.ai/util/error"
2
+ import { MessageV2 } from "./message-v2"
3
+
4
+ export namespace SessionRetry {
5
+ export const RETRY_INITIAL_DELAY = 2000
6
+ export const RETRY_BACKOFF_FACTOR = 2
7
+ export const RETRY_MAX_DELAY_NO_HEADERS = 30_000 // 30 seconds
8
+ export const RETRY_MAX_DELAY = 2_147_483_647 // max 32-bit signed integer for setTimeout
9
+
10
+ export async function sleep(ms: number, signal: AbortSignal): Promise<void> {
11
+ return new Promise((resolve, reject) => {
12
+ const abortHandler = () => {
13
+ clearTimeout(timeout)
14
+ reject(new DOMException("Aborted", "AbortError"))
15
+ }
16
+ const timeout = setTimeout(
17
+ () => {
18
+ signal.removeEventListener("abort", abortHandler)
19
+ resolve()
20
+ },
21
+ Math.min(ms, RETRY_MAX_DELAY),
22
+ )
23
+ signal.addEventListener("abort", abortHandler, { once: true })
24
+ })
25
+ }
26
+
27
+ export function delay(attempt: number, error?: MessageV2.APIError) {
28
+ if (error) {
29
+ const headers = error.data.responseHeaders
30
+ if (headers) {
31
+ const retryAfterMs = headers["retry-after-ms"]
32
+ if (retryAfterMs) {
33
+ const parsedMs = Number.parseFloat(retryAfterMs)
34
+ if (!Number.isNaN(parsedMs)) {
35
+ return parsedMs
36
+ }
37
+ }
38
+
39
+ const retryAfter = headers["retry-after"]
40
+ if (retryAfter) {
41
+ const parsedSeconds = Number.parseFloat(retryAfter)
42
+ if (!Number.isNaN(parsedSeconds)) {
43
+ // convert seconds to milliseconds
44
+ return Math.ceil(parsedSeconds * 1000)
45
+ }
46
+ // Try parsing as HTTP date format
47
+ const parsed = Date.parse(retryAfter) - Date.now()
48
+ if (!Number.isNaN(parsed) && parsed > 0) {
49
+ return Math.ceil(parsed)
50
+ }
51
+ }
52
+
53
+ return RETRY_INITIAL_DELAY * Math.pow(RETRY_BACKOFF_FACTOR, attempt - 1)
54
+ }
55
+ }
56
+
57
+ return Math.min(RETRY_INITIAL_DELAY * Math.pow(RETRY_BACKOFF_FACTOR, attempt - 1), RETRY_MAX_DELAY_NO_HEADERS)
58
+ }
59
+
60
+ export function retryable(error: ReturnType<NamedError["toObject"]>) {
61
+ if (MessageV2.APIError.isInstance(error)) {
62
+ if (!error.data.isRetryable) return undefined
63
+ return error.data.message.includes("Overloaded") ? "Provider is overloaded" : error.data.message
64
+ }
65
+
66
+ if (typeof error.data?.message === "string") {
67
+ try {
68
+ const json = JSON.parse(error.data.message)
69
+ if (json.type === "error" && json.error?.type === "too_many_requests") {
70
+ return "Too Many Requests"
71
+ }
72
+ if (json.code.includes("exhausted") || json.code.includes("unavailable")) {
73
+ return "Provider is overloaded"
74
+ }
75
+ if (json.type === "error" && json.error?.code?.includes("rate_limit")) {
76
+ return "Rate Limited"
77
+ }
78
+ if (
79
+ json.error?.message?.includes("no_kv_space") ||
80
+ (json.type === "error" && json.error?.type === "server_error") ||
81
+ !!json.error
82
+ ) {
83
+ return "Provider Server Error"
84
+ }
85
+ } catch {}
86
+ }
87
+
88
+ return undefined
89
+ }
90
+ }
@@ -0,0 +1,108 @@
1
+ import z from "zod"
2
+ import { Identifier } from "../id/id"
3
+ import { Snapshot } from "../snapshot"
4
+ import { MessageV2 } from "./message-v2"
5
+ import { Session } from "."
6
+ import { Log } from "../util/log"
7
+ import { splitWhen } from "remeda"
8
+ import { Storage } from "../storage/storage"
9
+ import { Bus } from "../bus"
10
+ import { SessionPrompt } from "./prompt"
11
+
12
+ export namespace SessionRevert {
13
+ const log = Log.create({ service: "session.revert" })
14
+
15
+ export const RevertInput = z.object({
16
+ sessionID: Identifier.schema("session"),
17
+ messageID: Identifier.schema("message"),
18
+ partID: Identifier.schema("part").optional(),
19
+ })
20
+ export type RevertInput = z.infer<typeof RevertInput>
21
+
22
+ export async function revert(input: RevertInput) {
23
+ SessionPrompt.assertNotBusy(input.sessionID)
24
+ const all = await Session.messages({ sessionID: input.sessionID })
25
+ let lastUser: MessageV2.User | undefined
26
+ const session = await Session.get(input.sessionID)
27
+
28
+ let revert: Session.Info["revert"]
29
+ const patches: Snapshot.Patch[] = []
30
+ for (const msg of all) {
31
+ if (msg.info.role === "user") lastUser = msg.info
32
+ const remaining = []
33
+ for (const part of msg.parts) {
34
+ if (revert) {
35
+ if (part.type === "patch") {
36
+ patches.push(part)
37
+ }
38
+ continue
39
+ }
40
+
41
+ if (!revert) {
42
+ if ((msg.info.id === input.messageID && !input.partID) || part.id === input.partID) {
43
+ // if no useful parts left in message, same as reverting whole message
44
+ const partID = remaining.some((item) => ["text", "tool"].includes(item.type)) ? input.partID : undefined
45
+ revert = {
46
+ messageID: !partID && lastUser ? lastUser.id : msg.info.id,
47
+ partID,
48
+ }
49
+ }
50
+ remaining.push(part)
51
+ }
52
+ }
53
+ }
54
+
55
+ if (revert) {
56
+ const session = await Session.get(input.sessionID)
57
+ revert.snapshot = session.revert?.snapshot ?? (await Snapshot.track())
58
+ await Snapshot.revert(patches)
59
+ if (revert.snapshot) revert.diff = await Snapshot.diff(revert.snapshot)
60
+ return Session.update(input.sessionID, (draft) => {
61
+ draft.revert = revert
62
+ })
63
+ }
64
+ return session
65
+ }
66
+
67
+ export async function unrevert(input: { sessionID: string }) {
68
+ log.info("unreverting", input)
69
+ SessionPrompt.assertNotBusy(input.sessionID)
70
+ const session = await Session.get(input.sessionID)
71
+ if (!session.revert) return session
72
+ if (session.revert.snapshot) await Snapshot.restore(session.revert.snapshot)
73
+ const next = await Session.update(input.sessionID, (draft) => {
74
+ draft.revert = undefined
75
+ })
76
+ return next
77
+ }
78
+
79
+ export async function cleanup(session: Session.Info) {
80
+ if (!session.revert) return
81
+ const sessionID = session.id
82
+ let msgs = await Session.messages({ sessionID })
83
+ const messageID = session.revert.messageID
84
+ const [preserve, remove] = splitWhen(msgs, (x) => x.info.id === messageID)
85
+ msgs = preserve
86
+ for (const msg of remove) {
87
+ await Storage.remove(["message", sessionID, msg.info.id])
88
+ await Bus.publish(MessageV2.Event.Removed, { sessionID: sessionID, messageID: msg.info.id })
89
+ }
90
+ const last = preserve.at(-1)
91
+ if (session.revert.partID && last) {
92
+ const partID = session.revert.partID
93
+ const [preserveParts, removeParts] = splitWhen(last.parts, (x) => x.id === partID)
94
+ last.parts = preserveParts
95
+ for (const part of removeParts) {
96
+ await Storage.remove(["part", last.info.id, part.id])
97
+ await Bus.publish(MessageV2.Event.PartRemoved, {
98
+ sessionID: sessionID,
99
+ messageID: last.info.id,
100
+ partID: part.id,
101
+ })
102
+ }
103
+ }
104
+ await Session.update(sessionID, (draft) => {
105
+ draft.revert = undefined
106
+ })
107
+ }
108
+ }
@@ -0,0 +1,76 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { Bus } from "@/bus"
3
+ import { Instance } from "@/project/instance"
4
+ import z from "zod"
5
+
6
+ export namespace SessionStatus {
7
+ export const Info = z
8
+ .union([
9
+ z.object({
10
+ type: z.literal("idle"),
11
+ }),
12
+ z.object({
13
+ type: z.literal("retry"),
14
+ attempt: z.number(),
15
+ message: z.string(),
16
+ next: z.number(),
17
+ }),
18
+ z.object({
19
+ type: z.literal("busy"),
20
+ }),
21
+ ])
22
+ .meta({
23
+ ref: "SessionStatus",
24
+ })
25
+ export type Info = z.infer<typeof Info>
26
+
27
+ export const Event = {
28
+ Status: BusEvent.define(
29
+ "session.status",
30
+ z.object({
31
+ sessionID: z.string(),
32
+ status: Info,
33
+ }),
34
+ ),
35
+ // deprecated
36
+ Idle: BusEvent.define(
37
+ "session.idle",
38
+ z.object({
39
+ sessionID: z.string(),
40
+ }),
41
+ ),
42
+ }
43
+
44
+ const state = Instance.state(() => {
45
+ const data: Record<string, Info> = {}
46
+ return data
47
+ })
48
+
49
+ export function get(sessionID: string) {
50
+ return (
51
+ state()[sessionID] ?? {
52
+ type: "idle",
53
+ }
54
+ )
55
+ }
56
+
57
+ export function list() {
58
+ return state()
59
+ }
60
+
61
+ export function set(sessionID: string, status: Info) {
62
+ Bus.publish(Event.Status, {
63
+ sessionID,
64
+ status,
65
+ })
66
+ if (status.type === "idle") {
67
+ // deprecated
68
+ Bus.publish(Event.Idle, {
69
+ sessionID,
70
+ })
71
+ delete state()[sessionID]
72
+ return
73
+ }
74
+ state()[sessionID] = status
75
+ }
76
+ }
@@ -0,0 +1,150 @@
1
+ import { Provider } from "@/provider/provider"
2
+
3
+ import { fn } from "@/util/fn"
4
+ import z from "zod"
5
+ import { Session } from "."
6
+
7
+ import { MessageV2 } from "./message-v2"
8
+ import { Identifier } from "@/id/id"
9
+ import { Snapshot } from "@/snapshot"
10
+
11
+ import { Log } from "@/util/log"
12
+ import path from "path"
13
+ import { Instance } from "@/project/instance"
14
+ import { Storage } from "@/storage/storage"
15
+ import { Bus } from "@/bus"
16
+
17
+ import { LLM } from "./llm"
18
+ import { Agent } from "@/agent/agent"
19
+
20
+ export namespace SessionSummary {
21
+ const log = Log.create({ service: "session.summary" })
22
+
23
+ export const summarize = fn(
24
+ z.object({
25
+ sessionID: z.string(),
26
+ messageID: z.string(),
27
+ }),
28
+ async (input) => {
29
+ const all = await Session.messages({ sessionID: input.sessionID })
30
+ await Promise.all([
31
+ summarizeSession({ sessionID: input.sessionID, messages: all }),
32
+ summarizeMessage({ messageID: input.messageID, messages: all }),
33
+ ])
34
+ },
35
+ )
36
+
37
+ async function summarizeSession(input: { sessionID: string; messages: MessageV2.WithParts[] }) {
38
+ const files = new Set(
39
+ input.messages
40
+ .flatMap((x) => x.parts)
41
+ .filter((x) => x.type === "patch")
42
+ .flatMap((x) => x.files)
43
+ .map((x) => path.relative(Instance.worktree, x)),
44
+ )
45
+ const diffs = await computeDiff({ messages: input.messages }).then((x) =>
46
+ x.filter((x) => {
47
+ return files.has(x.file)
48
+ }),
49
+ )
50
+ await Session.update(input.sessionID, (draft) => {
51
+ draft.summary = {
52
+ additions: diffs.reduce((sum, x) => sum + x.additions, 0),
53
+ deletions: diffs.reduce((sum, x) => sum + x.deletions, 0),
54
+ files: diffs.length,
55
+ }
56
+ })
57
+ await Storage.write(["session_diff", input.sessionID], diffs)
58
+ Bus.publish(Session.Event.Diff, {
59
+ sessionID: input.sessionID,
60
+ diff: diffs,
61
+ })
62
+ }
63
+
64
+ async function summarizeMessage(input: { messageID: string; messages: MessageV2.WithParts[] }) {
65
+ const messages = input.messages.filter(
66
+ (m) => m.info.id === input.messageID || (m.info.role === "assistant" && m.info.parentID === input.messageID),
67
+ )
68
+ const msgWithParts = messages.find((m) => m.info.id === input.messageID)!
69
+ const userMsg = msgWithParts.info as MessageV2.User
70
+ const diffs = await computeDiff({ messages })
71
+ userMsg.summary = {
72
+ ...userMsg.summary,
73
+ diffs,
74
+ }
75
+ await Session.updateMessage(userMsg)
76
+
77
+ const textPart = msgWithParts.parts.find((p) => p.type === "text" && !p.synthetic) as MessageV2.TextPart
78
+ if (textPart && !userMsg.summary?.title) {
79
+ const agent = await Agent.get("title")
80
+ if (!agent) return
81
+ const stream = await LLM.stream({
82
+ agent,
83
+ user: userMsg,
84
+ tools: {},
85
+ model: agent.model
86
+ ? await Provider.getModel(agent.model.providerID, agent.model.modelID)
87
+ : ((await Provider.getSmallModel(userMsg.model.providerID)) ??
88
+ (await Provider.getModel(userMsg.model.providerID, userMsg.model.modelID))),
89
+ small: true,
90
+ messages: [
91
+ {
92
+ role: "user" as const,
93
+ content: `
94
+ The following is the text to summarize:
95
+ <text>
96
+ ${textPart?.text ?? ""}
97
+ </text>
98
+ `,
99
+ },
100
+ ],
101
+ abort: new AbortController().signal,
102
+ sessionID: userMsg.sessionID,
103
+ system: [],
104
+ retries: 3,
105
+ })
106
+ const result = await stream.text
107
+ log.info("title", { title: result })
108
+ userMsg.summary.title = result
109
+ await Session.updateMessage(userMsg)
110
+ }
111
+ }
112
+
113
+ export const diff = fn(
114
+ z.object({
115
+ sessionID: Identifier.schema("session"),
116
+ messageID: Identifier.schema("message").optional(),
117
+ }),
118
+ async (input) => {
119
+ return Storage.read<Snapshot.FileDiff[]>(["session_diff", input.sessionID]).catch(() => [])
120
+ },
121
+ )
122
+
123
+ async function computeDiff(input: { messages: MessageV2.WithParts[] }) {
124
+ let from: string | undefined
125
+ let to: string | undefined
126
+
127
+ // scan assistant messages to find earliest from and latest to
128
+ // snapshot
129
+ for (const item of input.messages) {
130
+ if (!from) {
131
+ for (const part of item.parts) {
132
+ if (part.type === "step-start" && part.snapshot) {
133
+ from = part.snapshot
134
+ break
135
+ }
136
+ }
137
+ }
138
+
139
+ for (const part of item.parts) {
140
+ if (part.type === "step-finish" && part.snapshot) {
141
+ to = part.snapshot
142
+ break
143
+ }
144
+ }
145
+ }
146
+
147
+ if (from && to) return Snapshot.diffFull(from, to)
148
+ return []
149
+ }
150
+ }
@@ -0,0 +1,152 @@
1
+ import { Ripgrep } from "../file/ripgrep"
2
+ import { Global } from "../global"
3
+ import { Filesystem } from "../util/filesystem"
4
+ import { Config } from "../config/config"
5
+
6
+ import { Instance } from "../project/instance"
7
+ import path from "path"
8
+ import os from "os"
9
+
10
+ import PROMPT_ANTHROPIC from "./prompt/anthropic.txt"
11
+ import PROMPT_ANTHROPIC_WITHOUT_TODO from "./prompt/qwen.txt"
12
+ import PROMPT_BEAST from "./prompt/beast.txt"
13
+ import PROMPT_GEMINI from "./prompt/gemini.txt"
14
+ import PROMPT_ANTHROPIC_SPOOF from "./prompt/anthropic_spoof.txt"
15
+
16
+ import PROMPT_CODEX from "./prompt/codex_header.txt"
17
+ import type { Provider } from "@/provider/provider"
18
+ import { Flag } from "@/flag/flag"
19
+
20
+ export namespace SystemPrompt {
21
+ export function header(providerID: string) {
22
+ if (providerID.includes("anthropic")) return [PROMPT_ANTHROPIC_SPOOF.trim()]
23
+ return []
24
+ }
25
+
26
+ export function instructions() {
27
+ return PROMPT_CODEX.trim()
28
+ }
29
+
30
+ export function provider(model: Provider.Model) {
31
+ if (model.api.id.includes("gpt-5")) return [PROMPT_CODEX]
32
+ if (model.api.id.includes("gpt-") || model.api.id.includes("o1") || model.api.id.includes("o3"))
33
+ return [PROMPT_BEAST]
34
+ if (model.api.id.includes("gemini-")) return [PROMPT_GEMINI]
35
+ if (model.api.id.includes("claude")) return [PROMPT_ANTHROPIC]
36
+ return [PROMPT_ANTHROPIC_WITHOUT_TODO]
37
+ }
38
+
39
+ export async function environment() {
40
+ const project = Instance.project
41
+ const cfg = await Config.get()
42
+ const supabaseInfo: string[] = []
43
+
44
+ if (cfg.mcp) {
45
+ for (const [name, mcp] of Object.entries(cfg.mcp)) {
46
+ if (typeof mcp === "object" && mcp !== null && "type" in mcp && mcp.type === "local") {
47
+ const refIndex = mcp.command.indexOf("--project-ref")
48
+ if (refIndex !== -1 && refIndex + 1 < mcp.command.length) {
49
+ const ref = mcp.command[refIndex + 1]
50
+ supabaseInfo.push(` Supabase Project connected via MCP server "${name}": ${ref}`)
51
+ }
52
+ }
53
+ }
54
+ }
55
+
56
+ return [
57
+ [
58
+ `Here is some useful information about the environment you are running in:`,
59
+ `<env>`,
60
+ ` Working directory: ${Instance.directory}`,
61
+ ` Is directory a git repo: ${project.vcs === "git" ? "yes" : "no"}`,
62
+ ` Platform: ${process.platform}`,
63
+ ` Today's date: ${new Date().toDateString()}`,
64
+ ...(supabaseInfo.length > 0 ? ["", " Connected Supabase projects:", ...supabaseInfo] : []),
65
+ `</env>`,
66
+ `<files>`,
67
+ ` ${project.vcs === "git" && false
68
+ ? await Ripgrep.tree({
69
+ cwd: Instance.directory,
70
+ limit: 200,
71
+ })
72
+ : ""
73
+ }`,
74
+ `</files>`,
75
+ ].join("\n"),
76
+ ]
77
+ }
78
+
79
+ const LOCAL_RULE_FILES = [
80
+ "AGENTS.md",
81
+ "CLAUDE.md",
82
+ "CONTEXT.md", // deprecated
83
+ ]
84
+ const GLOBAL_RULE_FILES = [path.join(Global.Path.config, "AGENTS.md")]
85
+ if (!Flag.EASC_DISABLE_CLAUDE_CODE_PROMPT) {
86
+ GLOBAL_RULE_FILES.push(path.join(os.homedir(), ".claude", "CLAUDE.md"))
87
+ }
88
+
89
+ if (Flag.EASC_CONFIG_DIR) {
90
+ GLOBAL_RULE_FILES.push(path.join(Flag.EASC_CONFIG_DIR, "AGENTS.md"))
91
+ }
92
+
93
+ export async function custom() {
94
+ const config = await Config.get()
95
+ const paths = new Set<string>()
96
+
97
+ for (const localRuleFile of LOCAL_RULE_FILES) {
98
+ const matches = await Filesystem.findUp(localRuleFile, Instance.directory, Instance.worktree)
99
+ if (matches.length > 0) {
100
+ matches.forEach((path) => paths.add(path))
101
+ break
102
+ }
103
+ }
104
+
105
+ for (const globalRuleFile of GLOBAL_RULE_FILES) {
106
+ if (await Bun.file(globalRuleFile).exists()) {
107
+ paths.add(globalRuleFile)
108
+ break
109
+ }
110
+ }
111
+
112
+ const urls: string[] = []
113
+ if (config.instructions) {
114
+ for (let instruction of config.instructions) {
115
+ if (instruction.startsWith("https://") || instruction.startsWith("http://")) {
116
+ urls.push(instruction)
117
+ continue
118
+ }
119
+ if (instruction.startsWith("~/")) {
120
+ instruction = path.join(os.homedir(), instruction.slice(2))
121
+ }
122
+ let matches: string[] = []
123
+ if (path.isAbsolute(instruction)) {
124
+ matches = await Array.fromAsync(
125
+ new Bun.Glob(path.basename(instruction)).scan({
126
+ cwd: path.dirname(instruction),
127
+ absolute: true,
128
+ onlyFiles: true,
129
+ }),
130
+ ).catch(() => [])
131
+ } else {
132
+ matches = await Filesystem.globUp(instruction, Instance.directory, Instance.worktree).catch(() => [])
133
+ }
134
+ matches.forEach((path) => paths.add(path))
135
+ }
136
+ }
137
+
138
+ const foundFiles = Array.from(paths).map((p) =>
139
+ Bun.file(p)
140
+ .text()
141
+ .catch(() => "")
142
+ .then((x) => "Instructions from: " + p + "\n" + x),
143
+ )
144
+ const foundUrls = urls.map((url) =>
145
+ fetch(url, { signal: AbortSignal.timeout(5000) })
146
+ .then((res) => (res.ok ? res.text() : ""))
147
+ .catch(() => "")
148
+ .then((x) => (x ? "Instructions from: " + url + "\n" + x : "")),
149
+ )
150
+ return Promise.all([...foundFiles, ...foundUrls]).then((result) => result.filter(Boolean))
151
+ }
152
+ }
@@ -0,0 +1,37 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { Bus } from "@/bus"
3
+ import z from "zod"
4
+ import { Storage } from "../storage/storage"
5
+
6
+ export namespace Todo {
7
+ export const Info = z
8
+ .object({
9
+ content: z.string().describe("Brief description of the task"),
10
+ status: z.string().describe("Current status of the task: pending, in_progress, completed, cancelled"),
11
+ priority: z.string().describe("Priority level of the task: high, medium, low"),
12
+ id: z.string().describe("Unique identifier for the todo item"),
13
+ })
14
+ .meta({ ref: "Todo" })
15
+ export type Info = z.infer<typeof Info>
16
+
17
+ export const Event = {
18
+ Updated: BusEvent.define(
19
+ "todo.updated",
20
+ z.object({
21
+ sessionID: z.string(),
22
+ todos: z.array(Info),
23
+ }),
24
+ ),
25
+ }
26
+
27
+ export async function update(input: { sessionID: string; todos: Info[] }) {
28
+ await Storage.write(["todo", input.sessionID], input.todos)
29
+ Bus.publish(Event.Updated, input)
30
+ }
31
+
32
+ export async function get(sessionID: string) {
33
+ return Storage.read<Info[]>(["todo", sessionID])
34
+ .then((x) => x || [])
35
+ .catch(() => [])
36
+ }
37
+ }