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,225 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { Bus } from "@/bus"
3
+ import { Session } from "."
4
+ import { Identifier } from "../id/id"
5
+ import { Instance } from "../project/instance"
6
+ import { Provider } from "../provider/provider"
7
+ import { MessageV2 } from "./message-v2"
8
+ import z from "zod"
9
+ import { SessionPrompt } from "./prompt"
10
+ import { Token } from "../util/token"
11
+ import { Log } from "../util/log"
12
+ import { SessionProcessor } from "./processor"
13
+ import { fn } from "@/util/fn"
14
+ import { Agent } from "@/agent/agent"
15
+ import { Plugin } from "@/plugin"
16
+ import { Config } from "@/config/config"
17
+
18
+ export namespace SessionCompaction {
19
+ const log = Log.create({ service: "session.compaction" })
20
+
21
+ export const Event = {
22
+ Compacted: BusEvent.define(
23
+ "session.compacted",
24
+ z.object({
25
+ sessionID: z.string(),
26
+ }),
27
+ ),
28
+ }
29
+
30
+ export async function isOverflow(input: { tokens: MessageV2.Assistant["tokens"]; model: Provider.Model }) {
31
+ const config = await Config.get()
32
+ if (config.compaction?.auto === false) return false
33
+ const context = input.model.limit.context
34
+ if (context === 0) return false
35
+ const count = input.tokens.input + input.tokens.cache.read + input.tokens.output
36
+ const output = Math.min(input.model.limit.output, SessionPrompt.OUTPUT_TOKEN_MAX) || SessionPrompt.OUTPUT_TOKEN_MAX
37
+ const usable = input.model.limit.input || context - output
38
+ return count > usable
39
+ }
40
+
41
+ export const PRUNE_MINIMUM = 20_000
42
+ export const PRUNE_PROTECT = 40_000
43
+
44
+ const PRUNE_PROTECTED_TOOLS = ["skill"]
45
+
46
+ // goes backwards through parts until there are 40_000 tokens worth of tool
47
+ // calls. then erases output of previous tool calls. idea is to throw away old
48
+ // tool calls that are no longer relevant.
49
+ export async function prune(input: { sessionID: string }) {
50
+ const config = await Config.get()
51
+ if (config.compaction?.prune === false) return
52
+ log.info("pruning")
53
+ const msgs = await Session.messages({ sessionID: input.sessionID })
54
+ let total = 0
55
+ let pruned = 0
56
+ const toPrune = []
57
+ let turns = 0
58
+
59
+ loop: for (let msgIndex = msgs.length - 1; msgIndex >= 0; msgIndex--) {
60
+ const msg = msgs[msgIndex]
61
+ if (msg.info.role === "user") turns++
62
+ if (turns < 2) continue
63
+ if (msg.info.role === "assistant" && msg.info.summary) break loop
64
+ for (let partIndex = msg.parts.length - 1; partIndex >= 0; partIndex--) {
65
+ const part = msg.parts[partIndex]
66
+ if (part.type === "tool")
67
+ if (part.state.status === "completed") {
68
+ if (PRUNE_PROTECTED_TOOLS.includes(part.tool)) continue
69
+
70
+ if (part.state.time.compacted) break loop
71
+ const estimate = Token.estimate(part.state.output)
72
+ total += estimate
73
+ if (total > PRUNE_PROTECT) {
74
+ pruned += estimate
75
+ toPrune.push(part)
76
+ }
77
+ }
78
+ }
79
+ }
80
+ log.info("found", { pruned, total })
81
+ if (pruned > PRUNE_MINIMUM) {
82
+ for (const part of toPrune) {
83
+ if (part.state.status === "completed") {
84
+ part.state.time.compacted = Date.now()
85
+ await Session.updatePart(part)
86
+ }
87
+ }
88
+ log.info("pruned", { count: toPrune.length })
89
+ }
90
+ }
91
+
92
+ export async function process(input: {
93
+ parentID: string
94
+ messages: MessageV2.WithParts[]
95
+ sessionID: string
96
+ abort: AbortSignal
97
+ auto: boolean
98
+ }) {
99
+ const userMessage = input.messages.findLast((m) => m.info.id === input.parentID)!.info as MessageV2.User
100
+ const agent = await Agent.get("compaction")
101
+ const model = agent.model
102
+ ? await Provider.getModel(agent.model.providerID, agent.model.modelID)
103
+ : await Provider.getModel(userMessage.model.providerID, userMessage.model.modelID)
104
+ const msg = (await Session.updateMessage({
105
+ id: Identifier.ascending("message"),
106
+ role: "assistant",
107
+ parentID: input.parentID,
108
+ sessionID: input.sessionID,
109
+ mode: "compaction",
110
+ agent: "compaction",
111
+ summary: true,
112
+ path: {
113
+ cwd: Instance.directory,
114
+ root: Instance.worktree,
115
+ },
116
+ cost: 0,
117
+ tokens: {
118
+ output: 0,
119
+ input: 0,
120
+ reasoning: 0,
121
+ cache: { read: 0, write: 0 },
122
+ },
123
+ modelID: model.id,
124
+ providerID: model.providerID,
125
+ time: {
126
+ created: Date.now(),
127
+ },
128
+ })) as MessageV2.Assistant
129
+ const processor = SessionProcessor.create({
130
+ assistantMessage: msg,
131
+ sessionID: input.sessionID,
132
+ model,
133
+ abort: input.abort,
134
+ })
135
+ // Allow plugins to inject context or replace compaction prompt
136
+ const compacting = await Plugin.trigger(
137
+ "experimental.session.compacting",
138
+ { sessionID: input.sessionID },
139
+ { context: [], prompt: undefined },
140
+ )
141
+ const defaultPrompt =
142
+ "Provide a detailed prompt for continuing our conversation above. Focus on information that would be helpful for continuing the conversation, including what we did, what we're doing, which files we're working on, and what we're going to do next considering new session will not have access to our conversation."
143
+ const promptText = compacting.prompt ?? [defaultPrompt, ...compacting.context].join("\n\n")
144
+ const result = await processor.process({
145
+ user: userMessage,
146
+ agent,
147
+ abort: input.abort,
148
+ sessionID: input.sessionID,
149
+ tools: {},
150
+ system: [],
151
+ messages: [
152
+ ...MessageV2.toModelMessage(input.messages),
153
+ {
154
+ role: "user",
155
+ content: [
156
+ {
157
+ type: "text",
158
+ text: promptText,
159
+ },
160
+ ],
161
+ },
162
+ ],
163
+ model,
164
+ })
165
+
166
+ if (result === "continue" && input.auto) {
167
+ const continueMsg = await Session.updateMessage({
168
+ id: Identifier.ascending("message"),
169
+ role: "user",
170
+ sessionID: input.sessionID,
171
+ time: {
172
+ created: Date.now(),
173
+ },
174
+ agent: userMessage.agent,
175
+ model: userMessage.model,
176
+ })
177
+ await Session.updatePart({
178
+ id: Identifier.ascending("part"),
179
+ messageID: continueMsg.id,
180
+ sessionID: input.sessionID,
181
+ type: "text",
182
+ synthetic: true,
183
+ text: "Continue if you have next steps",
184
+ time: {
185
+ start: Date.now(),
186
+ end: Date.now(),
187
+ },
188
+ })
189
+ }
190
+ if (processor.message.error) return "stop"
191
+ Bus.publish(Event.Compacted, { sessionID: input.sessionID })
192
+ return "continue"
193
+ }
194
+
195
+ export const create = fn(
196
+ z.object({
197
+ sessionID: Identifier.schema("session"),
198
+ agent: z.string(),
199
+ model: z.object({
200
+ providerID: z.string(),
201
+ modelID: z.string(),
202
+ }),
203
+ auto: z.boolean(),
204
+ }),
205
+ async (input) => {
206
+ const msg = await Session.updateMessage({
207
+ id: Identifier.ascending("message"),
208
+ role: "user",
209
+ model: input.model,
210
+ sessionID: input.sessionID,
211
+ agent: input.agent,
212
+ time: {
213
+ created: Date.now(),
214
+ },
215
+ })
216
+ await Session.updatePart({
217
+ id: Identifier.ascending("part"),
218
+ messageID: msg.id,
219
+ sessionID: msg.sessionID,
220
+ type: "compaction",
221
+ auto: input.auto,
222
+ })
223
+ },
224
+ )
225
+ }
@@ -0,0 +1,488 @@
1
+ import { Slug } from "@eliseart.ai/util/slug"
2
+ import path from "path"
3
+ import { BusEvent } from "@/bus/bus-event"
4
+ import { Bus } from "@/bus"
5
+ import { Decimal } from "decimal.js"
6
+ import z from "zod"
7
+ import { type LanguageModelUsage, type ProviderMetadata } from "ai"
8
+ import { Config } from "../config/config"
9
+ import { Flag } from "../flag/flag"
10
+ import { Identifier } from "../id/id"
11
+ import { Installation } from "../installation"
12
+
13
+ import { Storage } from "../storage/storage"
14
+ import { Log } from "../util/log"
15
+ import { MessageV2 } from "./message-v2"
16
+ import { Instance } from "../project/instance"
17
+ import { SessionPrompt } from "./prompt"
18
+ import { fn } from "@/util/fn"
19
+ import { Command } from "../command"
20
+ import { Snapshot } from "@/snapshot"
21
+
22
+ import type { Provider } from "@/provider/provider"
23
+ import { PermissionNext } from "@/permission/next"
24
+ import { Global } from "@/global"
25
+
26
+ export namespace Session {
27
+ const log = Log.create({ service: "session" })
28
+
29
+ const parentTitlePrefix = "New session - "
30
+ const childTitlePrefix = "Child session - "
31
+
32
+ function createDefaultTitle(isChild = false) {
33
+ return (isChild ? childTitlePrefix : parentTitlePrefix) + new Date().toISOString()
34
+ }
35
+
36
+ export function isDefaultTitle(title: string) {
37
+ return new RegExp(
38
+ `^(${parentTitlePrefix}|${childTitlePrefix})\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}\\.\\d{3}Z$`,
39
+ ).test(title)
40
+ }
41
+
42
+ export const Info = z
43
+ .object({
44
+ id: Identifier.schema("session"),
45
+ slug: z.string(),
46
+ projectID: z.string(),
47
+ directory: z.string(),
48
+ parentID: Identifier.schema("session").optional(),
49
+ summary: z
50
+ .object({
51
+ additions: z.number(),
52
+ deletions: z.number(),
53
+ files: z.number(),
54
+ diffs: Snapshot.FileDiff.array().optional(),
55
+ })
56
+ .optional(),
57
+ share: z
58
+ .object({
59
+ url: z.string(),
60
+ })
61
+ .optional(),
62
+ title: z.string(),
63
+ version: z.string(),
64
+ time: z.object({
65
+ created: z.number(),
66
+ updated: z.number(),
67
+ compacting: z.number().optional(),
68
+ archived: z.number().optional(),
69
+ }),
70
+ permission: PermissionNext.Ruleset.optional(),
71
+ revert: z
72
+ .object({
73
+ messageID: z.string(),
74
+ partID: z.string().optional(),
75
+ snapshot: z.string().optional(),
76
+ diff: z.string().optional(),
77
+ })
78
+ .optional(),
79
+ })
80
+ .meta({
81
+ ref: "Session",
82
+ })
83
+ export type Info = z.output<typeof Info>
84
+
85
+ export const ShareInfo = z
86
+ .object({
87
+ secret: z.string(),
88
+ url: z.string(),
89
+ })
90
+ .meta({
91
+ ref: "SessionShare",
92
+ })
93
+ export type ShareInfo = z.output<typeof ShareInfo>
94
+
95
+ export const Event = {
96
+ Created: BusEvent.define(
97
+ "session.created",
98
+ z.object({
99
+ info: Info,
100
+ }),
101
+ ),
102
+ Updated: BusEvent.define(
103
+ "session.updated",
104
+ z.object({
105
+ info: Info,
106
+ }),
107
+ ),
108
+ Deleted: BusEvent.define(
109
+ "session.deleted",
110
+ z.object({
111
+ info: Info,
112
+ }),
113
+ ),
114
+ Diff: BusEvent.define(
115
+ "session.diff",
116
+ z.object({
117
+ sessionID: z.string(),
118
+ diff: Snapshot.FileDiff.array(),
119
+ }),
120
+ ),
121
+ Error: BusEvent.define(
122
+ "session.error",
123
+ z.object({
124
+ sessionID: z.string().optional(),
125
+ error: MessageV2.Assistant.shape.error,
126
+ }),
127
+ ),
128
+ }
129
+
130
+ export const create = fn(
131
+ z
132
+ .object({
133
+ parentID: Identifier.schema("session").optional(),
134
+ title: z.string().optional(),
135
+ permission: Info.shape.permission,
136
+ })
137
+ .optional(),
138
+ async (input) => {
139
+ return createNext({
140
+ parentID: input?.parentID,
141
+ directory: Instance.directory,
142
+ title: input?.title,
143
+ permission: input?.permission,
144
+ })
145
+ },
146
+ )
147
+
148
+ export const fork = fn(
149
+ z.object({
150
+ sessionID: Identifier.schema("session"),
151
+ messageID: Identifier.schema("message").optional(),
152
+ }),
153
+ async (input) => {
154
+ const session = await createNext({
155
+ directory: Instance.directory,
156
+ })
157
+ const msgs = await messages({ sessionID: input.sessionID })
158
+ const idMap = new Map<string, string>()
159
+
160
+ for (const msg of msgs) {
161
+ if (input.messageID && msg.info.id >= input.messageID) break
162
+ const newID = Identifier.ascending("message")
163
+ idMap.set(msg.info.id, newID)
164
+
165
+ const parentID = msg.info.role === "assistant" && msg.info.parentID ? idMap.get(msg.info.parentID) : undefined
166
+ const cloned = await updateMessage({
167
+ ...msg.info,
168
+ sessionID: session.id,
169
+ id: newID,
170
+ ...(parentID && { parentID }),
171
+ })
172
+
173
+ for (const part of msg.parts) {
174
+ await updatePart({
175
+ ...part,
176
+ id: Identifier.ascending("part"),
177
+ messageID: cloned.id,
178
+ sessionID: session.id,
179
+ })
180
+ }
181
+ }
182
+ return session
183
+ },
184
+ )
185
+
186
+ export const touch = fn(Identifier.schema("session"), async (sessionID) => {
187
+ await update(sessionID, (draft) => {
188
+ draft.time.updated = Date.now()
189
+ })
190
+ })
191
+
192
+ export async function createNext(input: {
193
+ id?: string
194
+ title?: string
195
+ parentID?: string
196
+ directory: string
197
+ permission?: PermissionNext.Ruleset
198
+ }) {
199
+ const result: Info = {
200
+ id: Identifier.descending("session", input.id),
201
+ slug: Slug.create(),
202
+ version: Installation.VERSION,
203
+ projectID: Instance.project.id,
204
+ directory: input.directory,
205
+ parentID: input.parentID,
206
+ title: input.title ?? createDefaultTitle(!!input.parentID),
207
+ permission: input.permission,
208
+ time: {
209
+ created: Date.now(),
210
+ updated: Date.now(),
211
+ },
212
+ }
213
+ log.info("created", result)
214
+ await Storage.write(["session", Instance.project.id, result.id], result)
215
+ Bus.publish(Event.Created, {
216
+ info: result,
217
+ })
218
+ const cfg = await Config.get()
219
+ if (!result.parentID && (Flag.OPENCODE_AUTO_SHARE || cfg.share === "auto"))
220
+ share(result.id)
221
+ .then((share) => {
222
+ update(result.id, (draft) => {
223
+ draft.share = share
224
+ })
225
+ })
226
+ .catch(() => {
227
+ // Silently ignore sharing errors during session creation
228
+ })
229
+ Bus.publish(Event.Updated, {
230
+ info: result,
231
+ })
232
+ return result
233
+ }
234
+
235
+ export function plan(input: { slug: string; time: { created: number } }) {
236
+ const base = Instance.project.vcs
237
+ ? path.join(Instance.worktree, ".opencode", "plans")
238
+ : path.join(Global.Path.data, "plans")
239
+ return path.join(base, [input.time.created, input.slug].join("-") + ".md")
240
+ }
241
+
242
+ export const get = fn(Identifier.schema("session"), async (id) => {
243
+ const read = await Storage.read<Info>(["session", Instance.project.id, id])
244
+ return read as Info
245
+ })
246
+
247
+ export const getShare = fn(Identifier.schema("session"), async (id) => {
248
+ return Storage.read<ShareInfo>(["share", id])
249
+ })
250
+
251
+ export const share = fn(Identifier.schema("session"), async (id) => {
252
+ const cfg = await Config.get()
253
+ if (cfg.share === "disabled") {
254
+ throw new Error("Sharing is disabled in configuration")
255
+ }
256
+ const { ShareNext } = await import("@/share/share-next")
257
+ const share = await ShareNext.create(id)
258
+ await update(id, (draft) => {
259
+ draft.share = {
260
+ url: share.url,
261
+ }
262
+ })
263
+ return share
264
+ })
265
+
266
+ export const unshare = fn(Identifier.schema("session"), async (id) => {
267
+ // Use ShareNext to remove the share (same as share function uses ShareNext to create)
268
+ const { ShareNext } = await import("@/share/share-next")
269
+ await ShareNext.remove(id)
270
+ await update(id, (draft) => {
271
+ draft.share = undefined
272
+ })
273
+ })
274
+
275
+ export async function update(id: string, editor: (session: Info) => void) {
276
+ const project = Instance.project
277
+ const result = await Storage.update<Info>(["session", project.id, id], (draft) => {
278
+ editor(draft)
279
+ draft.time.updated = Date.now()
280
+ })
281
+ Bus.publish(Event.Updated, {
282
+ info: result,
283
+ })
284
+ return result
285
+ }
286
+
287
+ export const diff = fn(Identifier.schema("session"), async (sessionID) => {
288
+ const diffs = await Storage.read<Snapshot.FileDiff[]>(["session_diff", sessionID])
289
+ return diffs ?? []
290
+ })
291
+
292
+ export const messages = fn(
293
+ z.object({
294
+ sessionID: Identifier.schema("session"),
295
+ limit: z.number().optional(),
296
+ }),
297
+ async (input) => {
298
+ const result = [] as MessageV2.WithParts[]
299
+ for await (const msg of MessageV2.stream(input.sessionID)) {
300
+ if (input.limit && result.length >= input.limit) break
301
+ result.push(msg)
302
+ }
303
+ result.reverse()
304
+ return result
305
+ },
306
+ )
307
+
308
+ export async function* list() {
309
+ const project = Instance.project
310
+ for (const item of await Storage.list(["session", project.id])) {
311
+ yield Storage.read<Info>(item)
312
+ }
313
+ }
314
+
315
+ export const children = fn(Identifier.schema("session"), async (parentID) => {
316
+ const project = Instance.project
317
+ const result = [] as Session.Info[]
318
+ for (const item of await Storage.list(["session", project.id])) {
319
+ const session = await Storage.read<Info>(item)
320
+ if (session.parentID !== parentID) continue
321
+ result.push(session)
322
+ }
323
+ return result
324
+ })
325
+
326
+ export const remove = fn(Identifier.schema("session"), async (sessionID) => {
327
+ const project = Instance.project
328
+ try {
329
+ const session = await get(sessionID)
330
+ for (const child of await children(sessionID)) {
331
+ await remove(child.id)
332
+ }
333
+ await unshare(sessionID).catch(() => {})
334
+ for (const msg of await Storage.list(["message", sessionID])) {
335
+ for (const part of await Storage.list(["part", msg.at(-1)!])) {
336
+ await Storage.remove(part)
337
+ }
338
+ await Storage.remove(msg)
339
+ }
340
+ await Storage.remove(["session", project.id, sessionID])
341
+ Bus.publish(Event.Deleted, {
342
+ info: session,
343
+ })
344
+ } catch (e) {
345
+ log.error(e)
346
+ }
347
+ })
348
+
349
+ export const updateMessage = fn(MessageV2.Info, async (msg) => {
350
+ await Storage.write(["message", msg.sessionID, msg.id], msg)
351
+ Bus.publish(MessageV2.Event.Updated, {
352
+ info: msg,
353
+ })
354
+ return msg
355
+ })
356
+
357
+ export const removeMessage = fn(
358
+ z.object({
359
+ sessionID: Identifier.schema("session"),
360
+ messageID: Identifier.schema("message"),
361
+ }),
362
+ async (input) => {
363
+ await Storage.remove(["message", input.sessionID, input.messageID])
364
+ Bus.publish(MessageV2.Event.Removed, {
365
+ sessionID: input.sessionID,
366
+ messageID: input.messageID,
367
+ })
368
+ return input.messageID
369
+ },
370
+ )
371
+
372
+ export const removePart = fn(
373
+ z.object({
374
+ sessionID: Identifier.schema("session"),
375
+ messageID: Identifier.schema("message"),
376
+ partID: Identifier.schema("part"),
377
+ }),
378
+ async (input) => {
379
+ await Storage.remove(["part", input.messageID, input.partID])
380
+ Bus.publish(MessageV2.Event.PartRemoved, {
381
+ sessionID: input.sessionID,
382
+ messageID: input.messageID,
383
+ partID: input.partID,
384
+ })
385
+ return input.partID
386
+ },
387
+ )
388
+
389
+ const UpdatePartInput = z.union([
390
+ MessageV2.Part,
391
+ z.object({
392
+ part: MessageV2.TextPart,
393
+ delta: z.string(),
394
+ }),
395
+ z.object({
396
+ part: MessageV2.ReasoningPart,
397
+ delta: z.string(),
398
+ }),
399
+ ])
400
+
401
+ export const updatePart = fn(UpdatePartInput, async (input) => {
402
+ const part = "delta" in input ? input.part : input
403
+ const delta = "delta" in input ? input.delta : undefined
404
+ await Storage.write(["part", part.messageID, part.id], part)
405
+ Bus.publish(MessageV2.Event.PartUpdated, {
406
+ part,
407
+ delta,
408
+ })
409
+ return part
410
+ })
411
+
412
+ export const getUsage = fn(
413
+ z.object({
414
+ model: z.custom<Provider.Model>(),
415
+ usage: z.custom<LanguageModelUsage>(),
416
+ metadata: z.custom<ProviderMetadata>().optional(),
417
+ }),
418
+ (input) => {
419
+ const cachedInputTokens = input.usage.cachedInputTokens ?? 0
420
+ const excludesCachedTokens = !!(input.metadata?.["anthropic"] || input.metadata?.["bedrock"])
421
+ const adjustedInputTokens = excludesCachedTokens
422
+ ? (input.usage.inputTokens ?? 0)
423
+ : (input.usage.inputTokens ?? 0) - cachedInputTokens
424
+ const safe = (value: number) => {
425
+ if (!Number.isFinite(value)) return 0
426
+ return value
427
+ }
428
+
429
+ const tokens = {
430
+ input: safe(adjustedInputTokens),
431
+ output: safe(input.usage.outputTokens ?? 0),
432
+ reasoning: safe(input.usage?.reasoningTokens ?? 0),
433
+ cache: {
434
+ write: safe(
435
+ (input.metadata?.["anthropic"]?.["cacheCreationInputTokens"] ??
436
+ // @ts-expect-error
437
+ input.metadata?.["bedrock"]?.["usage"]?.["cacheWriteInputTokens"] ??
438
+ 0) as number,
439
+ ),
440
+ read: safe(cachedInputTokens),
441
+ },
442
+ }
443
+
444
+ const costInfo =
445
+ input.model.cost?.experimentalOver200K && tokens.input + tokens.cache.read > 200_000
446
+ ? input.model.cost.experimentalOver200K
447
+ : input.model.cost
448
+ return {
449
+ cost: safe(
450
+ new Decimal(0)
451
+ .add(new Decimal(tokens.input).mul(costInfo?.input ?? 0).div(1_000_000))
452
+ .add(new Decimal(tokens.output).mul(costInfo?.output ?? 0).div(1_000_000))
453
+ .add(new Decimal(tokens.cache.read).mul(costInfo?.cache?.read ?? 0).div(1_000_000))
454
+ .add(new Decimal(tokens.cache.write).mul(costInfo?.cache?.write ?? 0).div(1_000_000))
455
+ // TODO: update models.dev to have better pricing model, for now:
456
+ // charge reasoning tokens at the same rate as output tokens
457
+ .add(new Decimal(tokens.reasoning).mul(costInfo?.output ?? 0).div(1_000_000))
458
+ .toNumber(),
459
+ ),
460
+ tokens,
461
+ }
462
+ },
463
+ )
464
+
465
+ export class BusyError extends Error {
466
+ constructor(public readonly sessionID: string) {
467
+ super(`Session ${sessionID} is busy`)
468
+ }
469
+ }
470
+
471
+ export const initialize = fn(
472
+ z.object({
473
+ sessionID: Identifier.schema("session"),
474
+ modelID: z.string(),
475
+ providerID: z.string(),
476
+ messageID: Identifier.schema("message"),
477
+ }),
478
+ async (input) => {
479
+ await SessionPrompt.command({
480
+ sessionID: input.sessionID,
481
+ messageID: input.messageID,
482
+ model: input.providerID + "/" + input.modelID,
483
+ command: Command.Default.INIT,
484
+ arguments: "",
485
+ })
486
+ },
487
+ )
488
+ }