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,935 @@
1
+ import { Hono } from "hono"
2
+ import { stream } from "hono/streaming"
3
+ import { describeRoute, validator, resolver } from "hono-openapi"
4
+ import z from "zod"
5
+ import { Session } from "../../session"
6
+ import { MessageV2 } from "../../session/message-v2"
7
+ import { SessionPrompt } from "../../session/prompt"
8
+ import { SessionCompaction } from "../../session/compaction"
9
+ import { SessionRevert } from "../../session/revert"
10
+ import { SessionStatus } from "@/session/status"
11
+ import { SessionSummary } from "@/session/summary"
12
+ import { Todo } from "../../session/todo"
13
+ import { Agent } from "../../agent/agent"
14
+ import { Snapshot } from "@/snapshot"
15
+ import { Log } from "../../util/log"
16
+ import { PermissionNext } from "@/permission/next"
17
+ import { errors } from "../error"
18
+ import { lazy } from "../../util/lazy"
19
+
20
+ const log = Log.create({ service: "server" })
21
+
22
+ export const SessionRoutes = lazy(() =>
23
+ new Hono()
24
+ .get(
25
+ "/",
26
+ describeRoute({
27
+ summary: "List sessions",
28
+ description: "Get a list of all OpenCode sessions, sorted by most recently updated.",
29
+ operationId: "session.list",
30
+ responses: {
31
+ 200: {
32
+ description: "List of sessions",
33
+ content: {
34
+ "application/json": {
35
+ schema: resolver(Session.Info.array()),
36
+ },
37
+ },
38
+ },
39
+ },
40
+ }),
41
+ validator(
42
+ "query",
43
+ z.object({
44
+ directory: z.string().optional().meta({ description: "Filter sessions by project directory" }),
45
+ roots: z.coerce.boolean().optional().meta({ description: "Only return root sessions (no parentID)" }),
46
+ start: z.coerce
47
+ .number()
48
+ .optional()
49
+ .meta({ description: "Filter sessions updated on or after this timestamp (milliseconds since epoch)" }),
50
+ search: z.string().optional().meta({ description: "Filter sessions by title (case-insensitive)" }),
51
+ limit: z.coerce.number().optional().meta({ description: "Maximum number of sessions to return" }),
52
+ }),
53
+ ),
54
+ async (c) => {
55
+ const query = c.req.valid("query")
56
+ const term = query.search?.toLowerCase()
57
+ const sessions: Session.Info[] = []
58
+ for await (const session of Session.list()) {
59
+ if (query.directory !== undefined && session.directory !== query.directory) continue
60
+ if (query.roots && session.parentID) continue
61
+ if (query.start !== undefined && session.time.updated < query.start) continue
62
+ if (term !== undefined && !session.title.toLowerCase().includes(term)) continue
63
+ sessions.push(session)
64
+ if (query.limit !== undefined && sessions.length >= query.limit) break
65
+ }
66
+ return c.json(sessions)
67
+ },
68
+ )
69
+ .get(
70
+ "/status",
71
+ describeRoute({
72
+ summary: "Get session status",
73
+ description: "Retrieve the current status of all sessions, including active, idle, and completed states.",
74
+ operationId: "session.status",
75
+ responses: {
76
+ 200: {
77
+ description: "Get session status",
78
+ content: {
79
+ "application/json": {
80
+ schema: resolver(z.record(z.string(), SessionStatus.Info)),
81
+ },
82
+ },
83
+ },
84
+ ...errors(400),
85
+ },
86
+ }),
87
+ async (c) => {
88
+ const result = SessionStatus.list()
89
+ return c.json(result)
90
+ },
91
+ )
92
+ .get(
93
+ "/:sessionID",
94
+ describeRoute({
95
+ summary: "Get session",
96
+ description: "Retrieve detailed information about a specific OpenCode session.",
97
+ tags: ["Session"],
98
+ operationId: "session.get",
99
+ responses: {
100
+ 200: {
101
+ description: "Get session",
102
+ content: {
103
+ "application/json": {
104
+ schema: resolver(Session.Info),
105
+ },
106
+ },
107
+ },
108
+ ...errors(400, 404),
109
+ },
110
+ }),
111
+ validator(
112
+ "param",
113
+ z.object({
114
+ sessionID: Session.get.schema,
115
+ }),
116
+ ),
117
+ async (c) => {
118
+ const sessionID = c.req.valid("param").sessionID
119
+ log.info("SEARCH", { url: c.req.url })
120
+ const session = await Session.get(sessionID)
121
+ return c.json(session)
122
+ },
123
+ )
124
+ .get(
125
+ "/:sessionID/children",
126
+ describeRoute({
127
+ summary: "Get session children",
128
+ tags: ["Session"],
129
+ description: "Retrieve all child sessions that were forked from the specified parent session.",
130
+ operationId: "session.children",
131
+ responses: {
132
+ 200: {
133
+ description: "List of children",
134
+ content: {
135
+ "application/json": {
136
+ schema: resolver(Session.Info.array()),
137
+ },
138
+ },
139
+ },
140
+ ...errors(400, 404),
141
+ },
142
+ }),
143
+ validator(
144
+ "param",
145
+ z.object({
146
+ sessionID: Session.children.schema,
147
+ }),
148
+ ),
149
+ async (c) => {
150
+ const sessionID = c.req.valid("param").sessionID
151
+ const session = await Session.children(sessionID)
152
+ return c.json(session)
153
+ },
154
+ )
155
+ .get(
156
+ "/:sessionID/todo",
157
+ describeRoute({
158
+ summary: "Get session todos",
159
+ description: "Retrieve the todo list associated with a specific session, showing tasks and action items.",
160
+ operationId: "session.todo",
161
+ responses: {
162
+ 200: {
163
+ description: "Todo list",
164
+ content: {
165
+ "application/json": {
166
+ schema: resolver(Todo.Info.array()),
167
+ },
168
+ },
169
+ },
170
+ ...errors(400, 404),
171
+ },
172
+ }),
173
+ validator(
174
+ "param",
175
+ z.object({
176
+ sessionID: z.string().meta({ description: "Session ID" }),
177
+ }),
178
+ ),
179
+ async (c) => {
180
+ const sessionID = c.req.valid("param").sessionID
181
+ const todos = await Todo.get(sessionID)
182
+ return c.json(todos)
183
+ },
184
+ )
185
+ .post(
186
+ "/",
187
+ describeRoute({
188
+ summary: "Create session",
189
+ description: "Create a new OpenCode session for interacting with AI assistants and managing conversations.",
190
+ operationId: "session.create",
191
+ responses: {
192
+ ...errors(400),
193
+ 200: {
194
+ description: "Successfully created session",
195
+ content: {
196
+ "application/json": {
197
+ schema: resolver(Session.Info),
198
+ },
199
+ },
200
+ },
201
+ },
202
+ }),
203
+ validator("json", Session.create.schema.optional()),
204
+ async (c) => {
205
+ const body = c.req.valid("json") ?? {}
206
+ const session = await Session.create(body)
207
+ return c.json(session)
208
+ },
209
+ )
210
+ .delete(
211
+ "/:sessionID",
212
+ describeRoute({
213
+ summary: "Delete session",
214
+ description: "Delete a session and permanently remove all associated data, including messages and history.",
215
+ operationId: "session.delete",
216
+ responses: {
217
+ 200: {
218
+ description: "Successfully deleted session",
219
+ content: {
220
+ "application/json": {
221
+ schema: resolver(z.boolean()),
222
+ },
223
+ },
224
+ },
225
+ ...errors(400, 404),
226
+ },
227
+ }),
228
+ validator(
229
+ "param",
230
+ z.object({
231
+ sessionID: Session.remove.schema,
232
+ }),
233
+ ),
234
+ async (c) => {
235
+ const sessionID = c.req.valid("param").sessionID
236
+ await Session.remove(sessionID)
237
+ return c.json(true)
238
+ },
239
+ )
240
+ .patch(
241
+ "/:sessionID",
242
+ describeRoute({
243
+ summary: "Update session",
244
+ description: "Update properties of an existing session, such as title or other metadata.",
245
+ operationId: "session.update",
246
+ responses: {
247
+ 200: {
248
+ description: "Successfully updated session",
249
+ content: {
250
+ "application/json": {
251
+ schema: resolver(Session.Info),
252
+ },
253
+ },
254
+ },
255
+ ...errors(400, 404),
256
+ },
257
+ }),
258
+ validator(
259
+ "param",
260
+ z.object({
261
+ sessionID: z.string(),
262
+ }),
263
+ ),
264
+ validator(
265
+ "json",
266
+ z.object({
267
+ title: z.string().optional(),
268
+ time: z
269
+ .object({
270
+ archived: z.number().optional(),
271
+ })
272
+ .optional(),
273
+ }),
274
+ ),
275
+ async (c) => {
276
+ const sessionID = c.req.valid("param").sessionID
277
+ const updates = c.req.valid("json")
278
+
279
+ const updatedSession = await Session.update(sessionID, (session) => {
280
+ if (updates.title !== undefined) {
281
+ session.title = updates.title
282
+ }
283
+ if (updates.time?.archived !== undefined) session.time.archived = updates.time.archived
284
+ })
285
+
286
+ return c.json(updatedSession)
287
+ },
288
+ )
289
+ .post(
290
+ "/:sessionID/init",
291
+ describeRoute({
292
+ summary: "Initialize session",
293
+ description:
294
+ "Analyze the current application and create an AGENTS.md file with project-specific agent configurations.",
295
+ operationId: "session.init",
296
+ responses: {
297
+ 200: {
298
+ description: "200",
299
+ content: {
300
+ "application/json": {
301
+ schema: resolver(z.boolean()),
302
+ },
303
+ },
304
+ },
305
+ ...errors(400, 404),
306
+ },
307
+ }),
308
+ validator(
309
+ "param",
310
+ z.object({
311
+ sessionID: z.string().meta({ description: "Session ID" }),
312
+ }),
313
+ ),
314
+ validator("json", Session.initialize.schema.omit({ sessionID: true })),
315
+ async (c) => {
316
+ const sessionID = c.req.valid("param").sessionID
317
+ const body = c.req.valid("json")
318
+ await Session.initialize({ ...body, sessionID })
319
+ return c.json(true)
320
+ },
321
+ )
322
+ .post(
323
+ "/:sessionID/fork",
324
+ describeRoute({
325
+ summary: "Fork session",
326
+ description: "Create a new session by forking an existing session at a specific message point.",
327
+ operationId: "session.fork",
328
+ responses: {
329
+ 200: {
330
+ description: "200",
331
+ content: {
332
+ "application/json": {
333
+ schema: resolver(Session.Info),
334
+ },
335
+ },
336
+ },
337
+ },
338
+ }),
339
+ validator(
340
+ "param",
341
+ z.object({
342
+ sessionID: Session.fork.schema.shape.sessionID,
343
+ }),
344
+ ),
345
+ validator("json", Session.fork.schema.omit({ sessionID: true })),
346
+ async (c) => {
347
+ const sessionID = c.req.valid("param").sessionID
348
+ const body = c.req.valid("json")
349
+ const result = await Session.fork({ ...body, sessionID })
350
+ return c.json(result)
351
+ },
352
+ )
353
+ .post(
354
+ "/:sessionID/abort",
355
+ describeRoute({
356
+ summary: "Abort session",
357
+ description: "Abort an active session and stop any ongoing AI processing or command execution.",
358
+ operationId: "session.abort",
359
+ responses: {
360
+ 200: {
361
+ description: "Aborted session",
362
+ content: {
363
+ "application/json": {
364
+ schema: resolver(z.boolean()),
365
+ },
366
+ },
367
+ },
368
+ ...errors(400, 404),
369
+ },
370
+ }),
371
+ validator(
372
+ "param",
373
+ z.object({
374
+ sessionID: z.string(),
375
+ }),
376
+ ),
377
+ async (c) => {
378
+ SessionPrompt.cancel(c.req.valid("param").sessionID)
379
+ return c.json(true)
380
+ },
381
+ )
382
+ .post(
383
+ "/:sessionID/share",
384
+ describeRoute({
385
+ summary: "Share session",
386
+ description: "Create a shareable link for a session, allowing others to view the conversation.",
387
+ operationId: "session.share",
388
+ responses: {
389
+ 200: {
390
+ description: "Successfully shared session",
391
+ content: {
392
+ "application/json": {
393
+ schema: resolver(Session.Info),
394
+ },
395
+ },
396
+ },
397
+ ...errors(400, 404),
398
+ },
399
+ }),
400
+ validator(
401
+ "param",
402
+ z.object({
403
+ sessionID: z.string(),
404
+ }),
405
+ ),
406
+ async (c) => {
407
+ const sessionID = c.req.valid("param").sessionID
408
+ await Session.share(sessionID)
409
+ const session = await Session.get(sessionID)
410
+ return c.json(session)
411
+ },
412
+ )
413
+ .get(
414
+ "/:sessionID/diff",
415
+ describeRoute({
416
+ summary: "Get message diff",
417
+ description: "Get the file changes (diff) that resulted from a specific user message in the session.",
418
+ operationId: "session.diff",
419
+ responses: {
420
+ 200: {
421
+ description: "Successfully retrieved diff",
422
+ content: {
423
+ "application/json": {
424
+ schema: resolver(Snapshot.FileDiff.array()),
425
+ },
426
+ },
427
+ },
428
+ },
429
+ }),
430
+ validator(
431
+ "param",
432
+ z.object({
433
+ sessionID: SessionSummary.diff.schema.shape.sessionID,
434
+ }),
435
+ ),
436
+ validator(
437
+ "query",
438
+ z.object({
439
+ messageID: SessionSummary.diff.schema.shape.messageID,
440
+ }),
441
+ ),
442
+ async (c) => {
443
+ const query = c.req.valid("query")
444
+ const params = c.req.valid("param")
445
+ const result = await SessionSummary.diff({
446
+ sessionID: params.sessionID,
447
+ messageID: query.messageID,
448
+ })
449
+ return c.json(result)
450
+ },
451
+ )
452
+ .delete(
453
+ "/:sessionID/share",
454
+ describeRoute({
455
+ summary: "Unshare session",
456
+ description: "Remove the shareable link for a session, making it private again.",
457
+ operationId: "session.unshare",
458
+ responses: {
459
+ 200: {
460
+ description: "Successfully unshared session",
461
+ content: {
462
+ "application/json": {
463
+ schema: resolver(Session.Info),
464
+ },
465
+ },
466
+ },
467
+ ...errors(400, 404),
468
+ },
469
+ }),
470
+ validator(
471
+ "param",
472
+ z.object({
473
+ sessionID: Session.unshare.schema,
474
+ }),
475
+ ),
476
+ async (c) => {
477
+ const sessionID = c.req.valid("param").sessionID
478
+ await Session.unshare(sessionID)
479
+ const session = await Session.get(sessionID)
480
+ return c.json(session)
481
+ },
482
+ )
483
+ .post(
484
+ "/:sessionID/summarize",
485
+ describeRoute({
486
+ summary: "Summarize session",
487
+ description: "Generate a concise summary of the session using AI compaction to preserve key information.",
488
+ operationId: "session.summarize",
489
+ responses: {
490
+ 200: {
491
+ description: "Summarized session",
492
+ content: {
493
+ "application/json": {
494
+ schema: resolver(z.boolean()),
495
+ },
496
+ },
497
+ },
498
+ ...errors(400, 404),
499
+ },
500
+ }),
501
+ validator(
502
+ "param",
503
+ z.object({
504
+ sessionID: z.string().meta({ description: "Session ID" }),
505
+ }),
506
+ ),
507
+ validator(
508
+ "json",
509
+ z.object({
510
+ providerID: z.string(),
511
+ modelID: z.string(),
512
+ auto: z.boolean().optional().default(false),
513
+ }),
514
+ ),
515
+ async (c) => {
516
+ const sessionID = c.req.valid("param").sessionID
517
+ const body = c.req.valid("json")
518
+ const session = await Session.get(sessionID)
519
+ await SessionRevert.cleanup(session)
520
+ const msgs = await Session.messages({ sessionID })
521
+ let currentAgent = await Agent.defaultAgent()
522
+ for (let i = msgs.length - 1; i >= 0; i--) {
523
+ const info = msgs[i].info
524
+ if (info.role === "user") {
525
+ currentAgent = info.agent || (await Agent.defaultAgent())
526
+ break
527
+ }
528
+ }
529
+ await SessionCompaction.create({
530
+ sessionID,
531
+ agent: currentAgent,
532
+ model: {
533
+ providerID: body.providerID,
534
+ modelID: body.modelID,
535
+ },
536
+ auto: body.auto,
537
+ })
538
+ await SessionPrompt.loop(sessionID)
539
+ return c.json(true)
540
+ },
541
+ )
542
+ .get(
543
+ "/:sessionID/message",
544
+ describeRoute({
545
+ summary: "Get session messages",
546
+ description: "Retrieve all messages in a session, including user prompts and AI responses.",
547
+ operationId: "session.messages",
548
+ responses: {
549
+ 200: {
550
+ description: "List of messages",
551
+ content: {
552
+ "application/json": {
553
+ schema: resolver(MessageV2.WithParts.array()),
554
+ },
555
+ },
556
+ },
557
+ ...errors(400, 404),
558
+ },
559
+ }),
560
+ validator(
561
+ "param",
562
+ z.object({
563
+ sessionID: z.string().meta({ description: "Session ID" }),
564
+ }),
565
+ ),
566
+ validator(
567
+ "query",
568
+ z.object({
569
+ limit: z.coerce.number().optional(),
570
+ }),
571
+ ),
572
+ async (c) => {
573
+ const query = c.req.valid("query")
574
+ const messages = await Session.messages({
575
+ sessionID: c.req.valid("param").sessionID,
576
+ limit: query.limit,
577
+ })
578
+ return c.json(messages)
579
+ },
580
+ )
581
+ .get(
582
+ "/:sessionID/message/:messageID",
583
+ describeRoute({
584
+ summary: "Get message",
585
+ description: "Retrieve a specific message from a session by its message ID.",
586
+ operationId: "session.message",
587
+ responses: {
588
+ 200: {
589
+ description: "Message",
590
+ content: {
591
+ "application/json": {
592
+ schema: resolver(
593
+ z.object({
594
+ info: MessageV2.Info,
595
+ parts: MessageV2.Part.array(),
596
+ }),
597
+ ),
598
+ },
599
+ },
600
+ },
601
+ ...errors(400, 404),
602
+ },
603
+ }),
604
+ validator(
605
+ "param",
606
+ z.object({
607
+ sessionID: z.string().meta({ description: "Session ID" }),
608
+ messageID: z.string().meta({ description: "Message ID" }),
609
+ }),
610
+ ),
611
+ async (c) => {
612
+ const params = c.req.valid("param")
613
+ const message = await MessageV2.get({
614
+ sessionID: params.sessionID,
615
+ messageID: params.messageID,
616
+ })
617
+ return c.json(message)
618
+ },
619
+ )
620
+ .delete(
621
+ "/:sessionID/message/:messageID/part/:partID",
622
+ describeRoute({
623
+ description: "Delete a part from a message",
624
+ operationId: "part.delete",
625
+ responses: {
626
+ 200: {
627
+ description: "Successfully deleted part",
628
+ content: {
629
+ "application/json": {
630
+ schema: resolver(z.boolean()),
631
+ },
632
+ },
633
+ },
634
+ ...errors(400, 404),
635
+ },
636
+ }),
637
+ validator(
638
+ "param",
639
+ z.object({
640
+ sessionID: z.string().meta({ description: "Session ID" }),
641
+ messageID: z.string().meta({ description: "Message ID" }),
642
+ partID: z.string().meta({ description: "Part ID" }),
643
+ }),
644
+ ),
645
+ async (c) => {
646
+ const params = c.req.valid("param")
647
+ await Session.removePart({
648
+ sessionID: params.sessionID,
649
+ messageID: params.messageID,
650
+ partID: params.partID,
651
+ })
652
+ return c.json(true)
653
+ },
654
+ )
655
+ .patch(
656
+ "/:sessionID/message/:messageID/part/:partID",
657
+ describeRoute({
658
+ description: "Update a part in a message",
659
+ operationId: "part.update",
660
+ responses: {
661
+ 200: {
662
+ description: "Successfully updated part",
663
+ content: {
664
+ "application/json": {
665
+ schema: resolver(MessageV2.Part),
666
+ },
667
+ },
668
+ },
669
+ ...errors(400, 404),
670
+ },
671
+ }),
672
+ validator(
673
+ "param",
674
+ z.object({
675
+ sessionID: z.string().meta({ description: "Session ID" }),
676
+ messageID: z.string().meta({ description: "Message ID" }),
677
+ partID: z.string().meta({ description: "Part ID" }),
678
+ }),
679
+ ),
680
+ validator("json", MessageV2.Part),
681
+ async (c) => {
682
+ const params = c.req.valid("param")
683
+ const body = c.req.valid("json")
684
+ if (body.id !== params.partID || body.messageID !== params.messageID || body.sessionID !== params.sessionID) {
685
+ throw new Error(
686
+ `Part mismatch: body.id='${body.id}' vs partID='${params.partID}', body.messageID='${body.messageID}' vs messageID='${params.messageID}', body.sessionID='${body.sessionID}' vs sessionID='${params.sessionID}'`,
687
+ )
688
+ }
689
+ const part = await Session.updatePart(body)
690
+ return c.json(part)
691
+ },
692
+ )
693
+ .post(
694
+ "/:sessionID/message",
695
+ describeRoute({
696
+ summary: "Send message",
697
+ description: "Create and send a new message to a session, streaming the AI response.",
698
+ operationId: "session.prompt",
699
+ responses: {
700
+ 200: {
701
+ description: "Created message",
702
+ content: {
703
+ "application/json": {
704
+ schema: resolver(
705
+ z.object({
706
+ info: MessageV2.Assistant,
707
+ parts: MessageV2.Part.array(),
708
+ }),
709
+ ),
710
+ },
711
+ },
712
+ },
713
+ ...errors(400, 404),
714
+ },
715
+ }),
716
+ validator(
717
+ "param",
718
+ z.object({
719
+ sessionID: z.string().meta({ description: "Session ID" }),
720
+ }),
721
+ ),
722
+ validator("json", SessionPrompt.PromptInput.omit({ sessionID: true })),
723
+ async (c) => {
724
+ c.status(200)
725
+ c.header("Content-Type", "application/json")
726
+ return stream(c, async (stream) => {
727
+ const sessionID = c.req.valid("param").sessionID
728
+ const body = c.req.valid("json")
729
+ const msg = await SessionPrompt.prompt({ ...body, sessionID })
730
+ stream.write(JSON.stringify(msg))
731
+ })
732
+ },
733
+ )
734
+ .post(
735
+ "/:sessionID/prompt_async",
736
+ describeRoute({
737
+ summary: "Send async message",
738
+ description:
739
+ "Create and send a new message to a session asynchronously, starting the session if needed and returning immediately.",
740
+ operationId: "session.prompt_async",
741
+ responses: {
742
+ 204: {
743
+ description: "Prompt accepted",
744
+ },
745
+ ...errors(400, 404),
746
+ },
747
+ }),
748
+ validator(
749
+ "param",
750
+ z.object({
751
+ sessionID: z.string().meta({ description: "Session ID" }),
752
+ }),
753
+ ),
754
+ validator("json", SessionPrompt.PromptInput.omit({ sessionID: true })),
755
+ async (c) => {
756
+ c.status(204)
757
+ c.header("Content-Type", "application/json")
758
+ return stream(c, async () => {
759
+ const sessionID = c.req.valid("param").sessionID
760
+ const body = c.req.valid("json")
761
+ SessionPrompt.prompt({ ...body, sessionID })
762
+ })
763
+ },
764
+ )
765
+ .post(
766
+ "/:sessionID/command",
767
+ describeRoute({
768
+ summary: "Send command",
769
+ description: "Send a new command to a session for execution by the AI assistant.",
770
+ operationId: "session.command",
771
+ responses: {
772
+ 200: {
773
+ description: "Created message",
774
+ content: {
775
+ "application/json": {
776
+ schema: resolver(
777
+ z.object({
778
+ info: MessageV2.Assistant,
779
+ parts: MessageV2.Part.array(),
780
+ }),
781
+ ),
782
+ },
783
+ },
784
+ },
785
+ ...errors(400, 404),
786
+ },
787
+ }),
788
+ validator(
789
+ "param",
790
+ z.object({
791
+ sessionID: z.string().meta({ description: "Session ID" }),
792
+ }),
793
+ ),
794
+ validator("json", SessionPrompt.CommandInput.omit({ sessionID: true })),
795
+ async (c) => {
796
+ const sessionID = c.req.valid("param").sessionID
797
+ const body = c.req.valid("json")
798
+ const msg = await SessionPrompt.command({ ...body, sessionID })
799
+ return c.json(msg)
800
+ },
801
+ )
802
+ .post(
803
+ "/:sessionID/shell",
804
+ describeRoute({
805
+ summary: "Run shell command",
806
+ description: "Execute a shell command within the session context and return the AI's response.",
807
+ operationId: "session.shell",
808
+ responses: {
809
+ 200: {
810
+ description: "Created message",
811
+ content: {
812
+ "application/json": {
813
+ schema: resolver(MessageV2.Assistant),
814
+ },
815
+ },
816
+ },
817
+ ...errors(400, 404),
818
+ },
819
+ }),
820
+ validator(
821
+ "param",
822
+ z.object({
823
+ sessionID: z.string().meta({ description: "Session ID" }),
824
+ }),
825
+ ),
826
+ validator("json", SessionPrompt.ShellInput.omit({ sessionID: true })),
827
+ async (c) => {
828
+ const sessionID = c.req.valid("param").sessionID
829
+ const body = c.req.valid("json")
830
+ const msg = await SessionPrompt.shell({ ...body, sessionID })
831
+ return c.json(msg)
832
+ },
833
+ )
834
+ .post(
835
+ "/:sessionID/revert",
836
+ describeRoute({
837
+ summary: "Revert message",
838
+ description: "Revert a specific message in a session, undoing its effects and restoring the previous state.",
839
+ operationId: "session.revert",
840
+ responses: {
841
+ 200: {
842
+ description: "Updated session",
843
+ content: {
844
+ "application/json": {
845
+ schema: resolver(Session.Info),
846
+ },
847
+ },
848
+ },
849
+ ...errors(400, 404),
850
+ },
851
+ }),
852
+ validator(
853
+ "param",
854
+ z.object({
855
+ sessionID: z.string(),
856
+ }),
857
+ ),
858
+ validator("json", SessionRevert.RevertInput.omit({ sessionID: true })),
859
+ async (c) => {
860
+ const sessionID = c.req.valid("param").sessionID
861
+ log.info("revert", c.req.valid("json"))
862
+ const session = await SessionRevert.revert({
863
+ sessionID,
864
+ ...c.req.valid("json"),
865
+ })
866
+ return c.json(session)
867
+ },
868
+ )
869
+ .post(
870
+ "/:sessionID/unrevert",
871
+ describeRoute({
872
+ summary: "Restore reverted messages",
873
+ description: "Restore all previously reverted messages in a session.",
874
+ operationId: "session.unrevert",
875
+ responses: {
876
+ 200: {
877
+ description: "Updated session",
878
+ content: {
879
+ "application/json": {
880
+ schema: resolver(Session.Info),
881
+ },
882
+ },
883
+ },
884
+ ...errors(400, 404),
885
+ },
886
+ }),
887
+ validator(
888
+ "param",
889
+ z.object({
890
+ sessionID: z.string(),
891
+ }),
892
+ ),
893
+ async (c) => {
894
+ const sessionID = c.req.valid("param").sessionID
895
+ const session = await SessionRevert.unrevert({ sessionID })
896
+ return c.json(session)
897
+ },
898
+ )
899
+ .post(
900
+ "/:sessionID/permissions/:permissionID",
901
+ describeRoute({
902
+ summary: "Respond to permission",
903
+ deprecated: true,
904
+ description: "Approve or deny a permission request from the AI assistant.",
905
+ operationId: "permission.respond",
906
+ responses: {
907
+ 200: {
908
+ description: "Permission processed successfully",
909
+ content: {
910
+ "application/json": {
911
+ schema: resolver(z.boolean()),
912
+ },
913
+ },
914
+ },
915
+ ...errors(400, 404),
916
+ },
917
+ }),
918
+ validator(
919
+ "param",
920
+ z.object({
921
+ sessionID: z.string(),
922
+ permissionID: z.string(),
923
+ }),
924
+ ),
925
+ validator("json", z.object({ response: PermissionNext.Reply })),
926
+ async (c) => {
927
+ const params = c.req.valid("param")
928
+ PermissionNext.reply({
929
+ requestID: params.permissionID,
930
+ reply: c.req.valid("json").response,
931
+ })
932
+ return c.json(true)
933
+ },
934
+ ),
935
+ )