artificial-code 1.17.13

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 (749) hide show
  1. package/AGENTS.md +131 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/ac +199 -0
  5. package/bunfig.toml +7 -0
  6. package/git +0 -0
  7. package/migration/20260511173437_session-metadata/migration.sql +1 -0
  8. package/migration/20260511173437_session-metadata/snapshot.json +1500 -0
  9. package/package.json +156 -0
  10. package/parsers-config.ts +1 -0
  11. package/script/bench-search.ts +94 -0
  12. package/script/bench-test-suite.ts +52 -0
  13. package/script/build.ts +243 -0
  14. package/script/generate.ts +14 -0
  15. package/script/httpapi-exercise.ts +1 -0
  16. package/script/postinstall.mjs +189 -0
  17. package/script/profile-test-files.ts +42 -0
  18. package/script/publish.ts +213 -0
  19. package/script/run-workspace-server +106 -0
  20. package/script/schema.ts +77 -0
  21. package/script/time.ts +6 -0
  22. package/script/trace-imports.ts +153 -0
  23. package/specs/effect/error-boundaries-plan.md +235 -0
  24. package/specs/effect/errors.md +207 -0
  25. package/specs/effect/facades.md +218 -0
  26. package/specs/effect/guide.md +247 -0
  27. package/specs/effect/instance-context.md +13 -0
  28. package/specs/effect/loose-ends.md +30 -0
  29. package/specs/effect/migration.md +62 -0
  30. package/specs/effect/routes.md +61 -0
  31. package/specs/effect/schema.md +88 -0
  32. package/specs/effect/server-package.md +58 -0
  33. package/specs/effect/todo.md +241 -0
  34. package/specs/effect/tools.md +88 -0
  35. package/specs/openapi-translation-cleanup.md +204 -0
  36. package/specs/tui-plugins.md +544 -0
  37. package/specs/v2/api.ts +67 -0
  38. package/specs/v2/message-shape.md +136 -0
  39. package/specs/v2/notifications.md +13 -0
  40. package/specs/v2/tui-command-shim.md +67 -0
  41. package/src/account/account.ts +461 -0
  42. package/src/account/repo.ts +171 -0
  43. package/src/account/schema.ts +99 -0
  44. package/src/account/url.ts +8 -0
  45. package/src/acp/agent.ts +95 -0
  46. package/src/acp/config-option.ts +203 -0
  47. package/src/acp/content.ts +269 -0
  48. package/src/acp/directory.ts +212 -0
  49. package/src/acp/error.ts +97 -0
  50. package/src/acp/event.ts +342 -0
  51. package/src/acp/permission.ts +254 -0
  52. package/src/acp/profile.ts +42 -0
  53. package/src/acp/service.ts +1097 -0
  54. package/src/acp/session.ts +232 -0
  55. package/src/acp/tool.ts +364 -0
  56. package/src/acp/usage.ts +239 -0
  57. package/src/agent/agent.ts +453 -0
  58. package/src/agent/generate.txt +75 -0
  59. package/src/agent/prompt/compaction.txt +9 -0
  60. package/src/agent/prompt/explore.txt +18 -0
  61. package/src/agent/prompt/summary.txt +11 -0
  62. package/src/agent/prompt/title.txt +44 -0
  63. package/src/agent/subagent-permissions.ts +27 -0
  64. package/src/audio.d.ts +14 -0
  65. package/src/auth/index.ts +97 -0
  66. package/src/background/job.ts +37 -0
  67. package/src/bus/global.ts +22 -0
  68. package/src/cli/bootstrap.ts +11 -0
  69. package/src/cli/cmd/account.ts +264 -0
  70. package/src/cli/cmd/acp.ts +73 -0
  71. package/src/cli/cmd/agent.ts +259 -0
  72. package/src/cli/cmd/attach.ts +148 -0
  73. package/src/cli/cmd/cmd.ts +7 -0
  74. package/src/cli/cmd/db.ts +62 -0
  75. package/src/cli/cmd/debug/agent.handler.ts +193 -0
  76. package/src/cli/cmd/debug/agent.ts +27 -0
  77. package/src/cli/cmd/debug/config.ts +14 -0
  78. package/src/cli/cmd/debug/file.ts +73 -0
  79. package/src/cli/cmd/debug/index.ts +87 -0
  80. package/src/cli/cmd/debug/lsp.ts +50 -0
  81. package/src/cli/cmd/debug/ripgrep.ts +79 -0
  82. package/src/cli/cmd/debug/scrap.ts +16 -0
  83. package/src/cli/cmd/debug/skill.ts +15 -0
  84. package/src/cli/cmd/debug/snapshot.ts +50 -0
  85. package/src/cli/cmd/debug/startup.ts +11 -0
  86. package/src/cli/cmd/debug/v2.ts +42 -0
  87. package/src/cli/cmd/export.ts +292 -0
  88. package/src/cli/cmd/generate.ts +54 -0
  89. package/src/cli/cmd/github.handler.ts +1593 -0
  90. package/src/cli/cmd/github.shared.ts +30 -0
  91. package/src/cli/cmd/github.ts +42 -0
  92. package/src/cli/cmd/import.ts +224 -0
  93. package/src/cli/cmd/mcp.ts +840 -0
  94. package/src/cli/cmd/models.ts +66 -0
  95. package/src/cli/cmd/plug.ts +230 -0
  96. package/src/cli/cmd/pr.ts +115 -0
  97. package/src/cli/cmd/prompt-display.ts +1 -0
  98. package/src/cli/cmd/providers.ts +534 -0
  99. package/src/cli/cmd/run/demo.ts +1274 -0
  100. package/src/cli/cmd/run/entry.body.ts +205 -0
  101. package/src/cli/cmd/run/footer.command.tsx +1064 -0
  102. package/src/cli/cmd/run/footer.menu.tsx +351 -0
  103. package/src/cli/cmd/run/footer.permission.tsx +472 -0
  104. package/src/cli/cmd/run/footer.prompt.tsx +1306 -0
  105. package/src/cli/cmd/run/footer.question.tsx +573 -0
  106. package/src/cli/cmd/run/footer.subagent.tsx +173 -0
  107. package/src/cli/cmd/run/footer.ts +1129 -0
  108. package/src/cli/cmd/run/footer.view.tsx +943 -0
  109. package/src/cli/cmd/run/footer.width.ts +27 -0
  110. package/src/cli/cmd/run/permission.shared.ts +256 -0
  111. package/src/cli/cmd/run/prompt.editor.ts +157 -0
  112. package/src/cli/cmd/run/prompt.shared.ts +153 -0
  113. package/src/cli/cmd/run/question.shared.ts +340 -0
  114. package/src/cli/cmd/run/runtime.boot.ts +202 -0
  115. package/src/cli/cmd/run/runtime.lifecycle.ts +406 -0
  116. package/src/cli/cmd/run/runtime.queue.ts +349 -0
  117. package/src/cli/cmd/run/runtime.shared.ts +17 -0
  118. package/src/cli/cmd/run/runtime.stdin.ts +37 -0
  119. package/src/cli/cmd/run/runtime.ts +814 -0
  120. package/src/cli/cmd/run/scrollback.shared.ts +92 -0
  121. package/src/cli/cmd/run/scrollback.surface.ts +431 -0
  122. package/src/cli/cmd/run/scrollback.writer.tsx +352 -0
  123. package/src/cli/cmd/run/session-data.ts +1113 -0
  124. package/src/cli/cmd/run/session-replay.ts +374 -0
  125. package/src/cli/cmd/run/session.shared.ts +196 -0
  126. package/src/cli/cmd/run/splash.ts +280 -0
  127. package/src/cli/cmd/run/stream.transport.ts +1462 -0
  128. package/src/cli/cmd/run/stream.ts +175 -0
  129. package/src/cli/cmd/run/subagent-data.ts +876 -0
  130. package/src/cli/cmd/run/theme.ts +690 -0
  131. package/src/cli/cmd/run/tool.ts +1486 -0
  132. package/src/cli/cmd/run/trace.ts +94 -0
  133. package/src/cli/cmd/run/turn-summary.ts +47 -0
  134. package/src/cli/cmd/run/types.ts +350 -0
  135. package/src/cli/cmd/run/variant.shared.ts +216 -0
  136. package/src/cli/cmd/run.ts +1011 -0
  137. package/src/cli/cmd/serve.ts +24 -0
  138. package/src/cli/cmd/session.ts +147 -0
  139. package/src/cli/cmd/stats.ts +393 -0
  140. package/src/cli/cmd/tui.ts +305 -0
  141. package/src/cli/cmd/uninstall.ts +353 -0
  142. package/src/cli/cmd/upgrade.ts +74 -0
  143. package/src/cli/cmd/web.ts +84 -0
  144. package/src/cli/effect/prompt.ts +37 -0
  145. package/src/cli/effect-cmd.ts +96 -0
  146. package/src/cli/error.ts +130 -0
  147. package/src/cli/heap.ts +45 -0
  148. package/src/cli/logo.ts +1 -0
  149. package/src/cli/network.ts +80 -0
  150. package/src/cli/tui/layer.ts +8 -0
  151. package/src/cli/tui/validate-session.ts +29 -0
  152. package/src/cli/tui/worker.ts +80 -0
  153. package/src/cli/ui.ts +132 -0
  154. package/src/cli/upgrade.ts +53 -0
  155. package/src/command/index.ts +177 -0
  156. package/src/command/template/initialize.txt +66 -0
  157. package/src/command/template/review.txt +101 -0
  158. package/src/config/agent.ts +59 -0
  159. package/src/config/command.ts +39 -0
  160. package/src/config/config.ts +680 -0
  161. package/src/config/entry-name.ts +19 -0
  162. package/src/config/managed.ts +69 -0
  163. package/src/config/markdown.ts +36 -0
  164. package/src/config/parse.ts +79 -0
  165. package/src/config/paths.ts +45 -0
  166. package/src/config/plugin.ts +79 -0
  167. package/src/config/tui-cwd.ts +5 -0
  168. package/src/config/tui-host-attention.ts +21 -0
  169. package/src/config/tui-migrate.ts +132 -0
  170. package/src/config/tui.ts +276 -0
  171. package/src/config/variable.ts +91 -0
  172. package/src/control-plane/adapters/index.ts +41 -0
  173. package/src/control-plane/adapters/worktree.ts +96 -0
  174. package/src/control-plane/dev/README.md +19 -0
  175. package/src/control-plane/dev/debug-workspace-plugin.ts +73 -0
  176. package/src/control-plane/types.ts +59 -0
  177. package/src/control-plane/util.ts +39 -0
  178. package/src/control-plane/workspace-adapter-runtime.ts +51 -0
  179. package/src/control-plane/workspace-context.ts +26 -0
  180. package/src/control-plane/workspace.ts +964 -0
  181. package/src/effect/app-node-builder-v1.ts +12 -0
  182. package/src/effect/app-runtime.ts +135 -0
  183. package/src/effect/bootstrap-runtime.ts +19 -0
  184. package/src/effect/bridge.ts +84 -0
  185. package/src/effect/config-service.ts +67 -0
  186. package/src/effect/instance-ref.ts +11 -0
  187. package/src/effect/instance-registry.ts +12 -0
  188. package/src/effect/instance-state.ts +69 -0
  189. package/src/effect/promise.ts +17 -0
  190. package/src/effect/run-service.ts +47 -0
  191. package/src/effect/runner.ts +217 -0
  192. package/src/effect/runtime-flags.ts +77 -0
  193. package/src/env/index.ts +41 -0
  194. package/src/event-manifest.ts +3 -0
  195. package/src/event-v2-bridge.ts +71 -0
  196. package/src/format/formatter.ts +404 -0
  197. package/src/format/index.ts +203 -0
  198. package/src/git/index.ts +348 -0
  199. package/src/id/id.ts +80 -0
  200. package/src/ide/index.ts +54 -0
  201. package/src/image/image.ts +172 -0
  202. package/src/index.ts +142 -0
  203. package/src/installation/index.ts +336 -0
  204. package/src/lsp/client.ts +650 -0
  205. package/src/lsp/diagnostic.ts +29 -0
  206. package/src/lsp/language.ts +121 -0
  207. package/src/lsp/launch.ts +21 -0
  208. package/src/lsp/lsp.ts +507 -0
  209. package/src/lsp/server.ts +1983 -0
  210. package/src/markdown.d.ts +4 -0
  211. package/src/mcp/auth.ts +163 -0
  212. package/src/mcp/catalog.ts +170 -0
  213. package/src/mcp/index.ts +1012 -0
  214. package/src/mcp/oauth-callback.ts +194 -0
  215. package/src/mcp/oauth-provider.ts +259 -0
  216. package/src/node.ts +4 -0
  217. package/src/patch/index.ts +686 -0
  218. package/src/permission/arity.ts +163 -0
  219. package/src/permission/evaluate.ts +1 -0
  220. package/src/permission/index.ts +218 -0
  221. package/src/plugin/azure.ts +26 -0
  222. package/src/plugin/cloudflare.ts +76 -0
  223. package/src/plugin/digitalocean.ts +325 -0
  224. package/src/plugin/github-copilot/copilot.ts +414 -0
  225. package/src/plugin/github-copilot/models.ts +246 -0
  226. package/src/plugin/index.ts +314 -0
  227. package/src/plugin/install.ts +439 -0
  228. package/src/plugin/loader.ts +237 -0
  229. package/src/plugin/meta.ts +188 -0
  230. package/src/plugin/openai/README.md +31 -0
  231. package/src/plugin/openai/codex.ts +556 -0
  232. package/src/plugin/openai/ws-pool.ts +270 -0
  233. package/src/plugin/openai/ws.ts +381 -0
  234. package/src/plugin/pty-environment.ts +24 -0
  235. package/src/plugin/shared.ts +323 -0
  236. package/src/plugin/snowflake-cortex.ts +507 -0
  237. package/src/plugin/tui/internal.ts +10 -0
  238. package/src/plugin/tui/runtime.ts +1131 -0
  239. package/src/plugin/xai.ts +626 -0
  240. package/src/project/bootstrap-service.ts +9 -0
  241. package/src/project/bootstrap.ts +58 -0
  242. package/src/project/instance-context.ts +24 -0
  243. package/src/project/instance-runtime.ts +16 -0
  244. package/src/project/instance-store.ts +213 -0
  245. package/src/project/project.ts +483 -0
  246. package/src/project/vcs.ts +423 -0
  247. package/src/provider/auth.ts +229 -0
  248. package/src/provider/error.ts +188 -0
  249. package/src/provider/model-status.ts +8 -0
  250. package/src/provider/provider.ts +1977 -0
  251. package/src/provider/transform.ts +1561 -0
  252. package/src/question/index.ts +161 -0
  253. package/src/question/schema.ts +4 -0
  254. package/src/server/auth.ts +48 -0
  255. package/src/server/event.ts +10 -0
  256. package/src/server/global-lifecycle.ts +28 -0
  257. package/src/server/init-projectors.ts +3 -0
  258. package/src/server/mdns.ts +47 -0
  259. package/src/server/projectors.ts +1 -0
  260. package/src/server/proxy-util.ts +48 -0
  261. package/src/server/routes/instance/httpapi/AGENTS.md +39 -0
  262. package/src/server/routes/instance/httpapi/api.ts +97 -0
  263. package/src/server/routes/instance/httpapi/errors.ts +193 -0
  264. package/src/server/routes/instance/httpapi/groups/config.ts +65 -0
  265. package/src/server/routes/instance/httpapi/groups/control-plane.ts +35 -0
  266. package/src/server/routes/instance/httpapi/groups/control.ts +76 -0
  267. package/src/server/routes/instance/httpapi/groups/event.ts +29 -0
  268. package/src/server/routes/instance/httpapi/groups/experimental.ts +275 -0
  269. package/src/server/routes/instance/httpapi/groups/file.ts +185 -0
  270. package/src/server/routes/instance/httpapi/groups/global.ts +136 -0
  271. package/src/server/routes/instance/httpapi/groups/instance.ts +206 -0
  272. package/src/server/routes/instance/httpapi/groups/mcp.ts +156 -0
  273. package/src/server/routes/instance/httpapi/groups/metadata.ts +18 -0
  274. package/src/server/routes/instance/httpapi/groups/permission.ts +61 -0
  275. package/src/server/routes/instance/httpapi/groups/project-copy.ts +32 -0
  276. package/src/server/routes/instance/httpapi/groups/project.ts +93 -0
  277. package/src/server/routes/instance/httpapi/groups/provider.ts +101 -0
  278. package/src/server/routes/instance/httpapi/groups/pty.ts +172 -0
  279. package/src/server/routes/instance/httpapi/groups/query.ts +12 -0
  280. package/src/server/routes/instance/httpapi/groups/question.ts +74 -0
  281. package/src/server/routes/instance/httpapi/groups/session.ts +462 -0
  282. package/src/server/routes/instance/httpapi/groups/sync.ts +113 -0
  283. package/src/server/routes/instance/httpapi/groups/tui.ts +208 -0
  284. package/src/server/routes/instance/httpapi/groups/workspace.ts +141 -0
  285. package/src/server/routes/instance/httpapi/handlers/config.ts +34 -0
  286. package/src/server/routes/instance/httpapi/handlers/control-plane.ts +37 -0
  287. package/src/server/routes/instance/httpapi/handlers/control.ts +43 -0
  288. package/src/server/routes/instance/httpapi/handlers/event.ts +99 -0
  289. package/src/server/routes/instance/httpapi/handlers/experimental.ts +193 -0
  290. package/src/server/routes/instance/httpapi/handlers/file.ts +139 -0
  291. package/src/server/routes/instance/httpapi/handlers/global.ts +156 -0
  292. package/src/server/routes/instance/httpapi/handlers/instance.ts +110 -0
  293. package/src/server/routes/instance/httpapi/handlers/mcp.ts +111 -0
  294. package/src/server/routes/instance/httpapi/handlers/permission.ts +41 -0
  295. package/src/server/routes/instance/httpapi/handlers/project-copy.ts +83 -0
  296. package/src/server/routes/instance/httpapi/handlers/project.ts +63 -0
  297. package/src/server/routes/instance/httpapi/handlers/provider.ts +113 -0
  298. package/src/server/routes/instance/httpapi/handlers/pty.ts +273 -0
  299. package/src/server/routes/instance/httpapi/handlers/question.ts +54 -0
  300. package/src/server/routes/instance/httpapi/handlers/session-errors.ts +21 -0
  301. package/src/server/routes/instance/httpapi/handlers/session.ts +442 -0
  302. package/src/server/routes/instance/httpapi/handlers/sync.ts +89 -0
  303. package/src/server/routes/instance/httpapi/handlers/tui.ts +131 -0
  304. package/src/server/routes/instance/httpapi/handlers/workspace.ts +102 -0
  305. package/src/server/routes/instance/httpapi/lifecycle.ts +54 -0
  306. package/src/server/routes/instance/httpapi/middleware/authorization.ts +150 -0
  307. package/src/server/routes/instance/httpapi/middleware/compression.ts +64 -0
  308. package/src/server/routes/instance/httpapi/middleware/cors-vary.ts +29 -0
  309. package/src/server/routes/instance/httpapi/middleware/error.ts +43 -0
  310. package/src/server/routes/instance/httpapi/middleware/fence.ts +25 -0
  311. package/src/server/routes/instance/httpapi/middleware/instance-context.ts +43 -0
  312. package/src/server/routes/instance/httpapi/middleware/proxy.ts +108 -0
  313. package/src/server/routes/instance/httpapi/middleware/schema-error.ts +41 -0
  314. package/src/server/routes/instance/httpapi/middleware/workspace-routing.ts +250 -0
  315. package/src/server/routes/instance/httpapi/public.ts +537 -0
  316. package/src/server/routes/instance/httpapi/server.ts +322 -0
  317. package/src/server/routes/instance/httpapi/websocket-tracker.ts +60 -0
  318. package/src/server/server.ts +226 -0
  319. package/src/server/shared/fence.ts +60 -0
  320. package/src/server/shared/pty-ticket.ts +15 -0
  321. package/src/server/shared/public-ui.ts +12 -0
  322. package/src/server/shared/tui-control.ts +28 -0
  323. package/src/server/shared/ui.ts +108 -0
  324. package/src/server/shared/workspace-routing.ts +38 -0
  325. package/src/server/tui-event.ts +1 -0
  326. package/src/session/compaction.ts +562 -0
  327. package/src/session/instruction.ts +237 -0
  328. package/src/session/llm/AGENTS.md +90 -0
  329. package/src/session/llm/ai-sdk.ts +288 -0
  330. package/src/session/llm/native-request.ts +196 -0
  331. package/src/session/llm/native-runtime.ts +195 -0
  332. package/src/session/llm/request.ts +226 -0
  333. package/src/session/llm.ts +404 -0
  334. package/src/session/message-error.ts +14 -0
  335. package/src/session/message-v2.ts +734 -0
  336. package/src/session/message.ts +148 -0
  337. package/src/session/overflow.ts +34 -0
  338. package/src/session/processor.ts +716 -0
  339. package/src/session/prompt/anthropic.txt +105 -0
  340. package/src/session/prompt/beast.txt +147 -0
  341. package/src/session/prompt/build-switch.txt +5 -0
  342. package/src/session/prompt/codex.txt +79 -0
  343. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  344. package/src/session/prompt/default.txt +95 -0
  345. package/src/session/prompt/gemini.txt +155 -0
  346. package/src/session/prompt/gpt.txt +107 -0
  347. package/src/session/prompt/kimi.txt +95 -0
  348. package/src/session/prompt/plan-mode.txt +70 -0
  349. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  350. package/src/session/prompt/plan.txt +26 -0
  351. package/src/session/prompt/trinity.txt +97 -0
  352. package/src/session/prompt.ts +1630 -0
  353. package/src/session/reminders.ts +92 -0
  354. package/src/session/retry.ts +201 -0
  355. package/src/session/revert.ts +146 -0
  356. package/src/session/run-state.ts +151 -0
  357. package/src/session/schema.ts +26 -0
  358. package/src/session/session.ts +1018 -0
  359. package/src/session/status.ts +56 -0
  360. package/src/session/summary.ts +160 -0
  361. package/src/session/system.ts +143 -0
  362. package/src/session/todo.ts +74 -0
  363. package/src/session/tools.ts +583 -0
  364. package/src/share/session.ts +58 -0
  365. package/src/share/share-next.ts +371 -0
  366. package/src/skill/discovery.ts +140 -0
  367. package/src/skill/index.ts +354 -0
  368. package/src/snapshot/index.ts +807 -0
  369. package/src/sql.d.ts +4 -0
  370. package/src/storage/schema.ts +5 -0
  371. package/src/storage/storage.ts +327 -0
  372. package/src/sync/README.md +179 -0
  373. package/src/sync/schema.ts +11 -0
  374. package/src/temporary.ts +31 -0
  375. package/src/tool/apply_patch.ts +313 -0
  376. package/src/tool/apply_patch.txt +33 -0
  377. package/src/tool/edit.ts +737 -0
  378. package/src/tool/edit.txt +10 -0
  379. package/src/tool/external-directory.ts +49 -0
  380. package/src/tool/glob.ts +76 -0
  381. package/src/tool/glob.txt +6 -0
  382. package/src/tool/grep.ts +112 -0
  383. package/src/tool/grep.txt +8 -0
  384. package/src/tool/invalid.ts +21 -0
  385. package/src/tool/json-schema.ts +164 -0
  386. package/src/tool/lsp.ts +113 -0
  387. package/src/tool/lsp.txt +24 -0
  388. package/src/tool/mcp-websearch.ts +96 -0
  389. package/src/tool/plan-enter.txt +14 -0
  390. package/src/tool/plan-exit.txt +13 -0
  391. package/src/tool/plan.ts +79 -0
  392. package/src/tool/question.ts +44 -0
  393. package/src/tool/question.txt +10 -0
  394. package/src/tool/read.ts +386 -0
  395. package/src/tool/read.txt +14 -0
  396. package/src/tool/registry.ts +420 -0
  397. package/src/tool/schema.ts +14 -0
  398. package/src/tool/shell/id.ts +19 -0
  399. package/src/tool/shell/prompt.ts +293 -0
  400. package/src/tool/shell/shell.txt +21 -0
  401. package/src/tool/shell.ts +645 -0
  402. package/src/tool/skill.ts +70 -0
  403. package/src/tool/skill.txt +5 -0
  404. package/src/tool/task.ts +346 -0
  405. package/src/tool/task.txt +19 -0
  406. package/src/tool/todo.ts +46 -0
  407. package/src/tool/todowrite.txt +44 -0
  408. package/src/tool/tool.ts +183 -0
  409. package/src/tool/truncate.ts +156 -0
  410. package/src/tool/truncation-dir.ts +4 -0
  411. package/src/tool/webfetch.ts +192 -0
  412. package/src/tool/webfetch.txt +13 -0
  413. package/src/tool/websearch.ts +143 -0
  414. package/src/tool/websearch.txt +14 -0
  415. package/src/tool/write.ts +104 -0
  416. package/src/tool/write.txt +8 -0
  417. package/src/util/archive.ts +17 -0
  418. package/src/util/bom.ts +27 -0
  419. package/src/util/data-url.ts +9 -0
  420. package/src/util/defer.ts +10 -0
  421. package/src/util/effect-http-client.ts +11 -0
  422. package/src/util/error.ts +1 -0
  423. package/src/util/filesystem.ts +251 -0
  424. package/src/util/html.ts +8 -0
  425. package/src/util/iife.ts +3 -0
  426. package/src/util/lazy.ts +20 -0
  427. package/src/util/local-context.ts +25 -0
  428. package/src/util/locale.ts +2 -0
  429. package/src/util/media.ts +26 -0
  430. package/src/util/process.ts +177 -0
  431. package/src/util/proxy-env.ts +72 -0
  432. package/src/util/queue.ts +32 -0
  433. package/src/util/record.ts +1 -0
  434. package/src/util/repository.ts +232 -0
  435. package/src/util/rpc.ts +66 -0
  436. package/src/util/signal.ts +12 -0
  437. package/src/util/timeout.ts +13 -0
  438. package/src/util/token.ts +1 -0
  439. package/src/util/wildcard.ts +59 -0
  440. package/src/worktree/index.ts +623 -0
  441. package/test/AGENTS.md +204 -0
  442. package/test/EFFECT_TEST_MIGRATION.md +169 -0
  443. package/test/account/repo.test.ts +355 -0
  444. package/test/account/service.test.ts +456 -0
  445. package/test/acp/config-option.test.ts +229 -0
  446. package/test/acp/content.test.ts +235 -0
  447. package/test/acp/directory.test.ts +188 -0
  448. package/test/acp/error.test.ts +67 -0
  449. package/test/acp/event.test.ts +751 -0
  450. package/test/acp/permission.test.ts +401 -0
  451. package/test/acp/service-session.test.ts +1248 -0
  452. package/test/acp/session.test.ts +201 -0
  453. package/test/acp/tool.test.ts +298 -0
  454. package/test/acp/usage.test.ts +318 -0
  455. package/test/agent/agent.test.ts +755 -0
  456. package/test/agent/plan-mode-subagent-bypass.test.ts +160 -0
  457. package/test/agent/plugin-agent-regression.test.ts +51 -0
  458. package/test/auth/auth.test.ts +75 -0
  459. package/test/background/job.test.ts +244 -0
  460. package/test/cli/account.test.ts +30 -0
  461. package/test/cli/acp/acp-test-client.ts +97 -0
  462. package/test/cli/acp/config-options.test.ts +103 -0
  463. package/test/cli/acp/helpers.ts +96 -0
  464. package/test/cli/acp/initialize-auth.test.ts +61 -0
  465. package/test/cli/acp/lifecycle.test.ts +118 -0
  466. package/test/cli/acp/prompt-content.test.ts +97 -0
  467. package/test/cli/acp/skills.test.ts +38 -0
  468. package/test/cli/cmd/tui/attention.test.ts +484 -0
  469. package/test/cli/effect-cmd-instance-als.test.ts +40 -0
  470. package/test/cli/error.test.ts +95 -0
  471. package/test/cli/github-action.test.ts +199 -0
  472. package/test/cli/github-remote.test.ts +90 -0
  473. package/test/cli/help/__snapshots__/help-snapshots.test.ts.snap +623 -0
  474. package/test/cli/help/help-snapshots.test.ts +140 -0
  475. package/test/cli/import.test.ts +54 -0
  476. package/test/cli/mcp-add.test.ts +74 -0
  477. package/test/cli/plugin-auth-picker.test.ts +120 -0
  478. package/test/cli/run/entry.body.test.ts +536 -0
  479. package/test/cli/run/footer.menu.test.ts +43 -0
  480. package/test/cli/run/footer.view.test.tsx +1375 -0
  481. package/test/cli/run/footer.width.test.ts +35 -0
  482. package/test/cli/run/permission.shared.test.ts +144 -0
  483. package/test/cli/run/prompt.editor.test.ts +101 -0
  484. package/test/cli/run/prompt.shared.test.ts +101 -0
  485. package/test/cli/run/question.shared.test.ts +115 -0
  486. package/test/cli/run/run-process.test.ts +331 -0
  487. package/test/cli/run/runtime.boot.test.ts +283 -0
  488. package/test/cli/run/runtime.queue.test.ts +481 -0
  489. package/test/cli/run/runtime.stdin.test.ts +71 -0
  490. package/test/cli/run/runtime.test.ts +238 -0
  491. package/test/cli/run/scrollback.surface.test.ts +1092 -0
  492. package/test/cli/run/session-data.test.ts +593 -0
  493. package/test/cli/run/session-replay.test.ts +691 -0
  494. package/test/cli/run/session.shared.test.ts +247 -0
  495. package/test/cli/run/stream.test.ts +56 -0
  496. package/test/cli/run/stream.transport.test.ts +2363 -0
  497. package/test/cli/run/subagent-data.test.ts +547 -0
  498. package/test/cli/run/theme.test.ts +177 -0
  499. package/test/cli/run/variant.shared.test.ts +218 -0
  500. package/test/cli/serve/serve-process.test.ts +61 -0
  501. package/test/cli/smokes/read-only.test.ts +115 -0
  502. package/test/cli/tui/attach.test.ts +11 -0
  503. package/test/cli/tui/editor-context-zed.test.ts +379 -0
  504. package/test/cli/tui/editor-context.test.tsx +297 -0
  505. package/test/cli/tui/plugin-add.test.ts +110 -0
  506. package/test/cli/tui/plugin-install.test.ts +87 -0
  507. package/test/cli/tui/plugin-lifecycle.test.ts +224 -0
  508. package/test/cli/tui/plugin-loader-entrypoint.test.ts +485 -0
  509. package/test/cli/tui/plugin-loader-pure.test.ts +72 -0
  510. package/test/cli/tui/plugin-loader.test.ts +1332 -0
  511. package/test/cli/tui/plugin-toggle.test.ts +264 -0
  512. package/test/cli/tui/thread.test.ts +95 -0
  513. package/test/config/agent-color.test.ts +47 -0
  514. package/test/config/config.test.ts +2027 -0
  515. package/test/config/entry-name.test.ts +57 -0
  516. package/test/config/fixtures/empty-frontmatter.md +4 -0
  517. package/test/config/fixtures/frontmatter.md +28 -0
  518. package/test/config/fixtures/markdown-header.md +11 -0
  519. package/test/config/fixtures/no-frontmatter.md +1 -0
  520. package/test/config/fixtures/weird-model-id.md +13 -0
  521. package/test/config/lsp.test.ts +69 -0
  522. package/test/config/markdown.test.ts +228 -0
  523. package/test/config/plugin.test.ts +0 -0
  524. package/test/config/tui.test.ts +894 -0
  525. package/test/control-plane/adapters.test.ts +71 -0
  526. package/test/control-plane/workspace.test.ts +1701 -0
  527. package/test/effect/app-runtime-logger.test.ts +101 -0
  528. package/test/effect/config-service.test.ts +65 -0
  529. package/test/effect/instance-state.test.ts +392 -0
  530. package/test/effect/run-service.test.ts +89 -0
  531. package/test/effect/runner.test.ts +514 -0
  532. package/test/effect/runtime-flags.test.ts +374 -0
  533. package/test/event-manifest.test.ts +24 -0
  534. package/test/fake/account.ts +9 -0
  535. package/test/fake/auth.ts +8 -0
  536. package/test/fake/npm.ts +8 -0
  537. package/test/fake/provider.ts +82 -0
  538. package/test/fake/skill.ts +8 -0
  539. package/test/filesystem/filesystem.test.ts +318 -0
  540. package/test/fixture/agent-plugin.constants.ts +6 -0
  541. package/test/fixture/agent-plugin.ts +12 -0
  542. package/test/fixture/config.ts +23 -0
  543. package/test/fixture/db.ts +11 -0
  544. package/test/fixture/fixture.test.ts +26 -0
  545. package/test/fixture/fixture.ts +228 -0
  546. package/test/fixture/flag.ts +20 -0
  547. package/test/fixture/flock-worker.ts +72 -0
  548. package/test/fixture/lsp/fake-lsp-server.js +249 -0
  549. package/test/fixture/mcp-session-recovery.ts +50 -0
  550. package/test/fixture/plug-worker.ts +93 -0
  551. package/test/fixture/plugin-meta-worker.ts +19 -0
  552. package/test/fixture/plugin.ts +10 -0
  553. package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
  554. package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
  555. package/test/fixture/skills/cloudflare/SKILL.md +211 -0
  556. package/test/fixture/skills/index.json +6 -0
  557. package/test/fixture/tui-environment.tsx +32 -0
  558. package/test/fixture/tui-plugin.ts +355 -0
  559. package/test/fixture/tui-runtime.ts +56 -0
  560. package/test/fixture/tui-sdk.ts +82 -0
  561. package/test/fixture/workspace.ts +34 -0
  562. package/test/fixtures/recordings/session/native-anthropic-tool-loop.json +49 -0
  563. package/test/fixtures/recordings/session/native-openai-oauth-tool-loop.json +45 -0
  564. package/test/fixtures/recordings/session/native-zen-tool-loop.json +49 -0
  565. package/test/format/format.test.ts +235 -0
  566. package/test/git/git.test.ts +179 -0
  567. package/test/ide/ide.test.ts +82 -0
  568. package/test/image/fixtures/picture-5mb-base64.png +0 -0
  569. package/test/image/image.test.ts +121 -0
  570. package/test/installation/installation.test.ts +240 -0
  571. package/test/lib/cli-process.ts +535 -0
  572. package/test/lib/effect.ts +177 -0
  573. package/test/lib/filesystem.ts +10 -0
  574. package/test/lib/llm-server.ts +779 -0
  575. package/test/lib/snapshot.ts +73 -0
  576. package/test/lib/test-provider.ts +37 -0
  577. package/test/lib/websocket.ts +46 -0
  578. package/test/lsp/client.test.ts +488 -0
  579. package/test/lsp/index.test.ts +231 -0
  580. package/test/lsp/jdtls-root.test.ts +459 -0
  581. package/test/lsp/launch.test.ts +22 -0
  582. package/test/lsp/lifecycle.test.ts +160 -0
  583. package/test/mcp/auth.test.ts +78 -0
  584. package/test/mcp/catalog.test.ts +47 -0
  585. package/test/mcp/headers.test.ts +127 -0
  586. package/test/mcp/lifecycle.test.ts +1304 -0
  587. package/test/mcp/oauth-auto-connect.test.ts +367 -0
  588. package/test/mcp/oauth-browser.test.ts +243 -0
  589. package/test/mcp/oauth-callback.test.ts +114 -0
  590. package/test/mcp/oauth-provider.test.ts +102 -0
  591. package/test/mcp/session-recovery.test.ts +27 -0
  592. package/test/patch/patch.test.ts +384 -0
  593. package/test/permission/arity.test.ts +33 -0
  594. package/test/permission/next.test.ts +1174 -0
  595. package/test/permission-task.test.ts +319 -0
  596. package/test/plugin/auth-override.test.ts +101 -0
  597. package/test/plugin/cloudflare.test.ts +68 -0
  598. package/test/plugin/codex.test.ts +256 -0
  599. package/test/plugin/github-copilot-models.test.ts +332 -0
  600. package/test/plugin/install-concurrency.test.ts +140 -0
  601. package/test/plugin/install.test.ts +570 -0
  602. package/test/plugin/loader-shared.test.ts +1306 -0
  603. package/test/plugin/meta.test.ts +137 -0
  604. package/test/plugin/openai-rollout.test.ts +17 -0
  605. package/test/plugin/openai-ws.test.ts +884 -0
  606. package/test/plugin/shared.test.ts +88 -0
  607. package/test/plugin/snowflake-cortex.test.ts +278 -0
  608. package/test/plugin/trigger.test.ts +108 -0
  609. package/test/plugin/workspace-adapter.test.ts +111 -0
  610. package/test/plugin/xai.test.ts +620 -0
  611. package/test/preload.ts +92 -0
  612. package/test/project/instance-bootstrap.test.ts +115 -0
  613. package/test/project/instance.test.ts +248 -0
  614. package/test/project/migrate-global.test.ts +168 -0
  615. package/test/project/project-directory.test.ts +202 -0
  616. package/test/project/project.test.ts +808 -0
  617. package/test/project/vcs.test.ts +335 -0
  618. package/test/project/worktree-remove.test.ts +128 -0
  619. package/test/project/worktree.test.ts +324 -0
  620. package/test/provider/amazon-bedrock.test.ts +361 -0
  621. package/test/provider/cf-ai-gateway-e2e.test.ts +132 -0
  622. package/test/provider/digitalocean.test.ts +124 -0
  623. package/test/provider/gitlab-duo.test.ts +412 -0
  624. package/test/provider/header-timeout.test.ts +234 -0
  625. package/test/provider/model-status.test.ts +61 -0
  626. package/test/provider/provider.test.ts +1912 -0
  627. package/test/provider/transform.test.ts +4742 -0
  628. package/test/question/question.test.ts +459 -0
  629. package/test/server/AGENTS.md +15 -0
  630. package/test/server/auth.test.ts +59 -0
  631. package/test/server/global-bus.ts +31 -0
  632. package/test/server/global-session-list.test.ts +108 -0
  633. package/test/server/httpapi-authorization.test.ts +174 -0
  634. package/test/server/httpapi-compression.test.ts +151 -0
  635. package/test/server/httpapi-config.test.ts +110 -0
  636. package/test/server/httpapi-control-plane.test.ts +63 -0
  637. package/test/server/httpapi-cors-vary.test.ts +63 -0
  638. package/test/server/httpapi-cors.test.ts +122 -0
  639. package/test/server/httpapi-error-middleware.test.ts +101 -0
  640. package/test/server/httpapi-event.test.ts +94 -0
  641. package/test/server/httpapi-exercise/assertions.ts +64 -0
  642. package/test/server/httpapi-exercise/backend.ts +144 -0
  643. package/test/server/httpapi-exercise/dsl.ts +210 -0
  644. package/test/server/httpapi-exercise/environment.ts +40 -0
  645. package/test/server/httpapi-exercise/index.ts +1814 -0
  646. package/test/server/httpapi-exercise/report.ts +66 -0
  647. package/test/server/httpapi-exercise/routing.ts +96 -0
  648. package/test/server/httpapi-exercise/runner.ts +267 -0
  649. package/test/server/httpapi-exercise/runtime.ts +52 -0
  650. package/test/server/httpapi-exercise/types.ts +127 -0
  651. package/test/server/httpapi-experimental.test.ts +298 -0
  652. package/test/server/httpapi-file.test.ts +83 -0
  653. package/test/server/httpapi-global.test.ts +66 -0
  654. package/test/server/httpapi-instance-context.test.ts +337 -0
  655. package/test/server/httpapi-instance-route-auth.test.ts +81 -0
  656. package/test/server/httpapi-instance.test.ts +265 -0
  657. package/test/server/httpapi-layer.ts +33 -0
  658. package/test/server/httpapi-listen.test.ts +465 -0
  659. package/test/server/httpapi-mcp-oauth.test.ts +73 -0
  660. package/test/server/httpapi-mcp.test.ts +223 -0
  661. package/test/server/httpapi-mdns.test.ts +79 -0
  662. package/test/server/httpapi-promptasync-context.test.ts +212 -0
  663. package/test/server/httpapi-provider.test.ts +401 -0
  664. package/test/server/httpapi-pty.test.ts +299 -0
  665. package/test/server/httpapi-public-openapi.test.ts +350 -0
  666. package/test/server/httpapi-query-schema-drift.test.ts +330 -0
  667. package/test/server/httpapi-reference.test.ts +63 -0
  668. package/test/server/httpapi-schema-error-body.test.ts +166 -0
  669. package/test/server/httpapi-sdk.test.ts +913 -0
  670. package/test/server/httpapi-session.test.ts +1090 -0
  671. package/test/server/httpapi-sync.test.ts +149 -0
  672. package/test/server/httpapi-ui.test.ts +456 -0
  673. package/test/server/httpapi-v2-location.test.ts +126 -0
  674. package/test/server/httpapi-v2-pty.test.ts +250 -0
  675. package/test/server/httpapi-workspace-routing.test.ts +552 -0
  676. package/test/server/httpapi-workspace.test.ts +506 -0
  677. package/test/server/negative-tokens-regression.test.ts +84 -0
  678. package/test/server/project-copy.test.ts +127 -0
  679. package/test/server/project-init-git.test.ts +118 -0
  680. package/test/server/proxy-util.test.ts +113 -0
  681. package/test/server/sdk-error-shape.test.ts +81 -0
  682. package/test/server/sdk-v1-smoke.test.ts +57 -0
  683. package/test/server/session-actions.test.ts +110 -0
  684. package/test/server/session-diff-missing-patch.test.ts +97 -0
  685. package/test/server/session-list.test.ts +302 -0
  686. package/test/server/session-messages.test.ts +180 -0
  687. package/test/server/session-select.test.ts +67 -0
  688. package/test/server/workspace-proxy.test.ts +181 -0
  689. package/test/server/workspace-routing.test.ts +94 -0
  690. package/test/server/worktree-endpoint-repro.test.ts +307 -0
  691. package/test/session/compaction.test.ts +1819 -0
  692. package/test/session/instruction.test.ts +264 -0
  693. package/test/session/llm-native-recorded.test.ts +413 -0
  694. package/test/session/llm-native.test.ts +761 -0
  695. package/test/session/llm.test.ts +2003 -0
  696. package/test/session/message-v2.test.ts +1661 -0
  697. package/test/session/messages-pagination.test.ts +1057 -0
  698. package/test/session/processor-effect.test.ts +1067 -0
  699. package/test/session/prompt.test.ts +2402 -0
  700. package/test/session/retry.test.ts +440 -0
  701. package/test/session/revert-compact.test.ts +638 -0
  702. package/test/session/schema-decoding.test.ts +313 -0
  703. package/test/session/session-schema.test.ts +78 -0
  704. package/test/session/session.test.ts +253 -0
  705. package/test/session/snapshot-tool-race.test.ts +189 -0
  706. package/test/session/structured-output-integration.test.ts +235 -0
  707. package/test/session/structured-output.test.ts +387 -0
  708. package/test/session/system.test.ts +142 -0
  709. package/test/share/share-next.test.ts +324 -0
  710. package/test/skill/discovery.test.ts +186 -0
  711. package/test/skill/skill.test.ts +585 -0
  712. package/test/snapshot/snapshot.test.ts +1218 -0
  713. package/test/storage/storage.test.ts +297 -0
  714. package/test/tool/__snapshots__/parameters.test.ts.snap +466 -0
  715. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  716. package/test/tool/apply_patch.test.ts +529 -0
  717. package/test/tool/edit.test.ts +574 -0
  718. package/test/tool/external-directory.test.ts +156 -0
  719. package/test/tool/fixtures/large-image.png +0 -0
  720. package/test/tool/fixtures/models-api.json +117299 -0
  721. package/test/tool/glob.test.ts +132 -0
  722. package/test/tool/grep.test.ts +221 -0
  723. package/test/tool/lsp.test.ts +184 -0
  724. package/test/tool/parameters.test.ts +290 -0
  725. package/test/tool/question.test.ts +133 -0
  726. package/test/tool/read.test.ts +608 -0
  727. package/test/tool/registry.test.ts +493 -0
  728. package/test/tool/shell.test.ts +1199 -0
  729. package/test/tool/skill.test.ts +134 -0
  730. package/test/tool/task.test.ts +905 -0
  731. package/test/tool/tool-define.test.ts +154 -0
  732. package/test/tool/truncation.test.ts +264 -0
  733. package/test/tool/webfetch.test.ts +119 -0
  734. package/test/tool/websearch.test.ts +99 -0
  735. package/test/tool/write.test.ts +279 -0
  736. package/test/util/data-url.test.ts +14 -0
  737. package/test/util/error.test.ts +16 -0
  738. package/test/util/filesystem.test.ts +656 -0
  739. package/test/util/glob.test.ts +164 -0
  740. package/test/util/html.test.ts +15 -0
  741. package/test/util/iife.test.ts +36 -0
  742. package/test/util/lazy.test.ts +50 -0
  743. package/test/util/module.test.ts +59 -0
  744. package/test/util/process.test.ts +128 -0
  745. package/test/util/repository.test.ts +93 -0
  746. package/test/util/timeout.test.ts +21 -0
  747. package/test/util/wildcard.test.ts +90 -0
  748. package/test/v2/session-message-updater.test.ts +269 -0
  749. package/tsconfig.json +16 -0
@@ -0,0 +1,1630 @@
1
+ import { LayerNode } from "@ac/core/effect/layer-node"
2
+ import { PermissionV1 } from "@ac/core/v1/permission"
3
+ import path from "path"
4
+ import { SessionV1 } from "@ac/core/v1/session"
5
+ import os from "os"
6
+ import { SessionID, MessageID, PartID } from "./schema"
7
+ import { MessageV2 } from "./message-v2"
8
+ import { SessionRevert } from "./revert"
9
+ import { Session } from "./session"
10
+ import { Agent } from "../agent/agent"
11
+ import { Provider } from "@/provider/provider"
12
+
13
+ import { type Tool as AITool, tool, jsonSchema } from "ai"
14
+ import type { JSONSchema7 } from "@ai-sdk/provider"
15
+ import { SessionCompaction } from "./compaction"
16
+ import { SystemPrompt } from "./system"
17
+ import { Instruction } from "./instruction"
18
+ import { Plugin } from "../plugin"
19
+ import { MAX_STEPS_PROMPT } from "@ac/core/session/runner/max-steps"
20
+ import { ToolRegistry } from "@/tool/registry"
21
+ import { MCP } from "../mcp"
22
+ import { LSP } from "@/lsp/lsp"
23
+ import { ulid } from "ulid"
24
+ import { ChildProcess, ChildProcessSpawner } from "effect/unstable/process"
25
+ import { CrossSpawnSpawner } from "@ac/core/cross-spawn-spawner"
26
+ import * as Stream from "effect/Stream"
27
+ import { Command } from "../command"
28
+ import { pathToFileURL, fileURLToPath } from "url"
29
+ import { Config } from "@/config/config"
30
+ import { ConfigMarkdown } from "@/config/markdown"
31
+ import { SessionSummary } from "./summary"
32
+ import { NamedError } from "@ac/core/util/error"
33
+ import { SessionProcessor } from "./processor"
34
+ import { Tool } from "@/tool/tool"
35
+ import { Permission } from "@/permission"
36
+ import { SessionStatus } from "./status"
37
+ import { LLM } from "./llm"
38
+ import { Shell } from "@ac/core/shell"
39
+ import { ShellID } from "@/tool/shell/id"
40
+ import { FSUtil } from "@ac/core/fs-util"
41
+ import { Truncate } from "@/tool/truncate"
42
+ import { Image } from "@/image/image"
43
+ import { decodeDataUrl } from "@/util/data-url"
44
+ import { Process } from "@/util/process"
45
+ import { Cause, Effect, Exit, Latch, Layer, Option, Scope, Context, Schema, Types } from "effect"
46
+ import { InstanceState } from "@/effect/instance-state"
47
+ import { TaskTool, type TaskPromptOps } from "@/tool/task"
48
+ import { SessionRunState } from "./run-state"
49
+ import { RuntimeFlags } from "@/effect/runtime-flags"
50
+ import { EventV2Bridge } from "@/event-v2-bridge"
51
+ import { Database } from "@ac/core/database/database"
52
+ import { ModelV2 } from "@ac/core/model"
53
+ import { ProviderV2 } from "@ac/core/provider"
54
+ import { eq } from "drizzle-orm"
55
+ import { SessionTable } from "@ac/core/session/sql"
56
+ import { SessionReminders } from "./reminders"
57
+ import { SessionTools } from "./tools"
58
+ import { LLMEvent } from "@ac/llm"
59
+
60
+ // @ts-ignore
61
+ globalThis.AI_SDK_LOG_WARNINGS = false
62
+
63
+ const decodeMessageInfo = Schema.decodeUnknownExit(SessionV1.Info)
64
+ const decodeMessagePart = Schema.decodeUnknownExit(SessionV1.Part)
65
+ const MAX_MCP_RESOURCE_BLOB_BYTES = 10 * 1024 * 1024
66
+ const SUPPORTED_MCP_RESOURCE_ATTACHMENT_MIMES = new Set([
67
+ "application/pdf",
68
+ "image/gif",
69
+ "image/jpeg",
70
+ "image/png",
71
+ "image/webp",
72
+ ])
73
+
74
+ const STRUCTURED_OUTPUT_DESCRIPTION = `Use this tool to return your final response in the requested structured format.
75
+
76
+ IMPORTANT:
77
+ - You MUST call this tool exactly once at the end of your response
78
+ - The input must be valid JSON matching the required schema
79
+ - Complete all necessary research and tool calls BEFORE calling this tool
80
+ - This tool provides your final answer - no further actions are taken after calling it`
81
+
82
+ const STRUCTURED_OUTPUT_SYSTEM_PROMPT = `IMPORTANT: The user has requested structured output. You MUST use the StructuredOutput tool to provide your final response. Do NOT respond with plain text - you MUST call the StructuredOutput tool with your answer formatted according to the schema.`
83
+
84
+ function mcpResourceBase64Size(value: string) {
85
+ const trimmed = value.replace(/\s/g, "")
86
+ const padding = trimmed.endsWith("==") ? 2 : trimmed.endsWith("=") ? 1 : 0
87
+ return Math.max(0, Math.floor((trimmed.length * 3) / 4) - padding)
88
+ }
89
+
90
+ function formatMcpResourceBytes(value: number) {
91
+ if (value < 1024) return `${value} B`
92
+ if (value < 1024 * 1024) return `${Math.ceil(value / 1024)} KB`
93
+ return `${Math.ceil(value / (1024 * 1024))} MB`
94
+ }
95
+
96
+ function isOrphanedInterruptedTool(part: SessionV1.ToolPart) {
97
+ // cleanup() marks abandoned tool_use blocks this way after retries/aborts.
98
+ // They are not pending work and must not trigger an assistant-prefill request.
99
+ return part.state.status === "error" && part.state.metadata?.interrupted === true
100
+ }
101
+
102
+ export interface Interface {
103
+ readonly cancel: (sessionID: SessionID) => Effect.Effect<void>
104
+ readonly prompt: (input: PromptInput) => Effect.Effect<SessionV1.WithParts, Image.Error>
105
+ readonly loop: (input: LoopInput) => Effect.Effect<SessionV1.WithParts>
106
+ readonly shell: (input: ShellInput) => Effect.Effect<SessionV1.WithParts, Session.BusyError>
107
+ readonly command: (input: CommandInput) => Effect.Effect<SessionV1.WithParts, Image.Error>
108
+ readonly resolvePromptParts: (template: string) => Effect.Effect<PromptInput["parts"]>
109
+ }
110
+
111
+ export class Service extends Context.Service<Service, Interface>()("@ac/SessionPrompt") {}
112
+
113
+ const layer = Layer.effect(
114
+ Service,
115
+ Effect.gen(function* () {
116
+ const status = yield* SessionStatus.Service
117
+ const sessions = yield* Session.Service
118
+ const agents = yield* Agent.Service
119
+ const provider = yield* Provider.Service
120
+ const processor = yield* SessionProcessor.Service
121
+ const compaction = yield* SessionCompaction.Service
122
+ const plugin = yield* Plugin.Service
123
+ const commands = yield* Command.Service
124
+ const config = yield* Config.Service
125
+ const permission = yield* Permission.Service
126
+ const fsys = yield* FSUtil.Service
127
+ const mcp = yield* MCP.Service
128
+ const lsp = yield* LSP.Service
129
+ const registry = yield* ToolRegistry.Service
130
+ const truncate = yield* Truncate.Service
131
+ const image = yield* Image.Service
132
+ const spawner = yield* ChildProcessSpawner.ChildProcessSpawner
133
+ const scope = yield* Scope.Scope
134
+ const instruction = yield* Instruction.Service
135
+ const state = yield* SessionRunState.Service
136
+ const revert = yield* SessionRevert.Service
137
+ const summary = yield* SessionSummary.Service
138
+ const sys = yield* SystemPrompt.Service
139
+ const llm = yield* LLM.Service
140
+ const events = yield* EventV2Bridge.Service
141
+ const flags = yield* RuntimeFlags.Service
142
+ const database = yield* Database.Service
143
+ const { db } = database
144
+ const ops = Effect.fn("SessionPrompt.ops")(function* () {
145
+ return {
146
+ cancel: (sessionID: SessionID) => cancel(sessionID),
147
+ resolvePromptParts: (template: string) => resolvePromptParts(template),
148
+ prompt: (input: PromptInput) => prompt(input).pipe(Effect.catch(Effect.die)),
149
+ } satisfies TaskPromptOps
150
+ })
151
+
152
+ const cancel = Effect.fn("SessionPrompt.cancel")(function* (sessionID: SessionID) {
153
+ yield* Effect.logInfo("cancel", { "session.id": sessionID })
154
+ yield* state.cancel(sessionID)
155
+ })
156
+
157
+ const resolvePromptParts = Effect.fn("SessionPrompt.resolvePromptParts")(function* (template: string) {
158
+ const ctx = yield* InstanceState.context
159
+ const parts: Types.DeepMutable<PromptInput["parts"]> = [{ type: "text", text: template }]
160
+ const files = ConfigMarkdown.files(template)
161
+ const seen = new Set<string>()
162
+ yield* Effect.forEach(
163
+ files,
164
+ Effect.fnUntraced(function* (match) {
165
+ const name = match[1]
166
+ if (!name) return
167
+ if (seen.has(name)) return
168
+ seen.add(name)
169
+
170
+ const filepath = name.startsWith("~/")
171
+ ? path.join(os.homedir(), name.slice(2))
172
+ : path.resolve(ctx.worktree, name)
173
+
174
+ const info = yield* fsys.stat(filepath).pipe(Effect.option)
175
+ if (Option.isNone(info)) {
176
+ const found = yield* agents.get(name)
177
+ if (found) parts.push({ type: "agent", name: found.name })
178
+ return
179
+ }
180
+ const stat = info.value
181
+ parts.push({
182
+ type: "file",
183
+ url: pathToFileURL(filepath).href,
184
+ filename: name,
185
+ mime: stat.type === "Directory" ? "application/x-directory" : "text/plain",
186
+ })
187
+ }),
188
+ { concurrency: "unbounded", discard: true },
189
+ )
190
+ return parts
191
+ })
192
+
193
+ const title = Effect.fn("SessionPrompt.ensureTitle")(function* (input: {
194
+ session: Session.Info
195
+ history: SessionV1.WithParts[]
196
+ providerID: ProviderV2.ID
197
+ modelID: ModelV2.ID
198
+ }) {
199
+ if (input.session.parentID) return
200
+ if (!Session.isDefaultTitle(input.session.title)) return
201
+
202
+ const real = (m: SessionV1.WithParts) =>
203
+ m.info.role === "user" && !m.parts.every((p) => "synthetic" in p && p.synthetic)
204
+ const idx = input.history.findIndex(real)
205
+ if (idx === -1) return
206
+ if (input.history.filter(real).length !== 1) return
207
+
208
+ const context = input.history.slice(0, idx + 1)
209
+ const firstUser = context[idx]
210
+ if (!firstUser || firstUser.info.role !== "user") return
211
+ const firstInfo = firstUser.info
212
+
213
+ const subtasks = firstUser.parts.filter((p): p is SessionV1.SubtaskPart => p.type === "subtask")
214
+ const onlySubtasks = subtasks.length > 0 && firstUser.parts.every((p) => p.type === "subtask")
215
+
216
+ const ag = yield* agents.get("title")
217
+ if (!ag) return
218
+ const mdl = ag.model
219
+ ? yield* provider.getModel(ag.model.providerID, ag.model.modelID)
220
+ : ((yield* provider.getSmallModel(input.providerID)) ??
221
+ (yield* provider.getModel(input.providerID, input.modelID)))
222
+ const msgs = onlySubtasks
223
+ ? [{ role: "user" as const, content: subtasks.map((p) => p.prompt).join("\n") }]
224
+ : yield* MessageV2.toModelMessagesEffect(context, mdl)
225
+ const text = yield* llm
226
+ .stream({
227
+ agent: ag,
228
+ user: firstInfo,
229
+ system: [],
230
+ small: true,
231
+ tools: {},
232
+ model: mdl,
233
+ sessionID: input.session.id,
234
+ retries: 2,
235
+ messages: [{ role: "user", content: "Generate a title for this conversation:\n" }, ...msgs],
236
+ })
237
+ .pipe(
238
+ Stream.filter(LLMEvent.is.textDelta),
239
+ Stream.map((e) => e.text),
240
+ Stream.mkString,
241
+ Effect.orDie,
242
+ )
243
+ const cleaned = text
244
+ .replace(/<think>[\s\S]*?<\/think>\s*/g, "")
245
+ .split("\n")
246
+ .map((line) => line.trim())
247
+ .find((line) => line.length > 0)
248
+ if (!cleaned) return
249
+ const t = cleaned.length > 100 ? cleaned.substring(0, 97) + "..." : cleaned
250
+ yield* sessions
251
+ .setTitle({ sessionID: input.session.id, title: t })
252
+ .pipe(Effect.catchCause((cause) => Effect.logError("failed to generate title", { error: Cause.squash(cause) })))
253
+ })
254
+
255
+ const handleSubtask = Effect.fn("SessionPrompt.handleSubtask")(function* (input: {
256
+ task: SessionV1.SubtaskPart
257
+ model: Provider.Model
258
+ lastUser: SessionV1.User
259
+ sessionID: SessionID
260
+ session: Session.Info
261
+ msgs: SessionV1.WithParts[]
262
+ }) {
263
+ const { task, model, lastUser, sessionID, session, msgs } = input
264
+ const ctx = yield* InstanceState.context
265
+ const promptOps = yield* ops()
266
+ const { task: taskTool } = yield* registry.named()
267
+ const taskModel = task.model ? yield* getModel(task.model.providerID, task.model.modelID, sessionID) : model
268
+ const assistantMessage: SessionV1.Assistant = yield* sessions.updateMessage({
269
+ id: MessageID.ascending(),
270
+ role: "assistant",
271
+ parentID: lastUser.id,
272
+ sessionID,
273
+ mode: task.agent,
274
+ agent: task.agent,
275
+ variant: lastUser.model.variant,
276
+ path: { cwd: ctx.directory, root: ctx.worktree },
277
+ cost: 0,
278
+ tokens: { input: 0, output: 0, reasoning: 0, cache: { read: 0, write: 0 } },
279
+ modelID: taskModel.id,
280
+ providerID: taskModel.providerID,
281
+ time: { created: Date.now() },
282
+ })
283
+ let part: SessionV1.ToolPart = yield* sessions.updatePart({
284
+ id: PartID.ascending(),
285
+ messageID: assistantMessage.id,
286
+ sessionID: assistantMessage.sessionID,
287
+ type: "tool",
288
+ callID: ulid(),
289
+ tool: TaskTool.id,
290
+ state: {
291
+ status: "running",
292
+ input: {
293
+ prompt: task.prompt,
294
+ description: task.description,
295
+ subagent_type: task.agent,
296
+ command: task.command,
297
+ },
298
+ time: { start: Date.now() },
299
+ },
300
+ })
301
+ const taskArgs = {
302
+ prompt: task.prompt,
303
+ description: task.description,
304
+ subagent_type: task.agent,
305
+ command: task.command,
306
+ }
307
+ yield* plugin.trigger(
308
+ "tool.execute.before",
309
+ { tool: TaskTool.id, sessionID, callID: part.id },
310
+ { args: taskArgs },
311
+ )
312
+
313
+ const taskAgent = yield* agents.get(task.agent)
314
+ if (!taskAgent) {
315
+ const available = (yield* agents.list()).filter((a) => !a.hidden).map((a) => a.name)
316
+ const hint = available.length ? ` Available agents: ${available.join(", ")}` : ""
317
+ const error = new NamedError.Unknown({ message: `Agent not found: "${task.agent}".${hint}` })
318
+ yield* events.publish(Session.Event.Error, { sessionID, error: error.toObject() })
319
+ throw error
320
+ }
321
+
322
+ let error: Error | undefined
323
+ const taskAbort = new AbortController()
324
+ const result = yield* taskTool
325
+ .execute(taskArgs, {
326
+ agent: task.agent,
327
+ messageID: assistantMessage.id,
328
+ sessionID,
329
+ abort: taskAbort.signal,
330
+ callID: part.callID,
331
+ extra: { bypassAgentCheck: true, promptOps },
332
+ messages: msgs,
333
+ metadata: (val: { title?: string; metadata?: Record<string, any> }) =>
334
+ Effect.gen(function* () {
335
+ part = yield* sessions.updatePart({
336
+ ...part,
337
+ type: "tool",
338
+ state: { ...part.state, ...val },
339
+ } satisfies SessionV1.ToolPart)
340
+ }),
341
+ ask: (req: any) =>
342
+ permission
343
+ .ask({
344
+ ...req,
345
+ sessionID,
346
+ ruleset: Permission.merge(taskAgent.permission, session.permission ?? []),
347
+ })
348
+ .pipe(Effect.orDie),
349
+ })
350
+ .pipe(
351
+ Effect.catchCause((cause) => {
352
+ const defect = Cause.squash(cause)
353
+ error = defect instanceof Error ? defect : new Error(String(defect))
354
+ return Effect.logError("subtask execution failed", {
355
+ error,
356
+ agent: task.agent,
357
+ description: task.description,
358
+ })
359
+ }),
360
+ Effect.onInterrupt(() =>
361
+ Effect.gen(function* () {
362
+ taskAbort.abort()
363
+ assistantMessage.finish = "tool-calls"
364
+ assistantMessage.time.completed = Date.now()
365
+ yield* sessions.updateMessage(assistantMessage)
366
+ if (part.state.status === "running") {
367
+ yield* sessions.updatePart({
368
+ ...part,
369
+ state: {
370
+ status: "error",
371
+ error: "Cancelled",
372
+ time: { start: part.state.time.start, end: Date.now() },
373
+ metadata: part.state.metadata,
374
+ input: part.state.input,
375
+ },
376
+ } satisfies SessionV1.ToolPart)
377
+ }
378
+ }),
379
+ ),
380
+ )
381
+
382
+ const attachments = result?.attachments?.map((attachment) => ({
383
+ ...attachment,
384
+ id: PartID.ascending(),
385
+ sessionID,
386
+ messageID: assistantMessage.id,
387
+ }))
388
+
389
+ yield* plugin.trigger(
390
+ "tool.execute.after",
391
+ { tool: TaskTool.id, sessionID, callID: part.id, args: taskArgs },
392
+ result,
393
+ )
394
+
395
+ assistantMessage.finish = "tool-calls"
396
+ assistantMessage.time.completed = Date.now()
397
+ yield* sessions.updateMessage(assistantMessage)
398
+
399
+ if (result && part.state.status === "running") {
400
+ yield* sessions.updatePart({
401
+ ...part,
402
+ state: {
403
+ status: "completed",
404
+ input: part.state.input,
405
+ title: result.title,
406
+ metadata: result.metadata,
407
+ output: result.output,
408
+ attachments,
409
+ time: { ...part.state.time, end: Date.now() },
410
+ },
411
+ } satisfies SessionV1.ToolPart)
412
+ }
413
+
414
+ if (!result) {
415
+ yield* sessions.updatePart({
416
+ ...part,
417
+ state: {
418
+ status: "error",
419
+ error: error ? `Tool execution failed: ${error.message}` : "Tool execution failed",
420
+ time: {
421
+ start: part.state.status === "running" ? part.state.time.start : Date.now(),
422
+ end: Date.now(),
423
+ },
424
+ metadata: part.state.status === "pending" ? undefined : part.state.metadata,
425
+ input: part.state.input,
426
+ },
427
+ } satisfies SessionV1.ToolPart)
428
+ }
429
+
430
+ if (!task.command) return
431
+
432
+ const summaryUserMsg: SessionV1.User = {
433
+ id: MessageID.ascending(),
434
+ sessionID,
435
+ role: "user",
436
+ time: { created: Date.now() },
437
+ agent: lastUser.agent,
438
+ model: lastUser.model,
439
+ }
440
+ yield* sessions.updateMessage(summaryUserMsg)
441
+ yield* sessions.updatePart({
442
+ id: PartID.ascending(),
443
+ messageID: summaryUserMsg.id,
444
+ sessionID,
445
+ type: "text",
446
+ text: "Summarize the task tool output above and continue with your task.",
447
+ synthetic: true,
448
+ } satisfies SessionV1.TextPart)
449
+ })
450
+
451
+ const shellImpl = Effect.fn("SessionPrompt.shellImpl")(function* (input: ShellInput, ready?: Latch.Latch) {
452
+ return yield* Effect.uninterruptibleMask((restore) =>
453
+ Effect.gen(function* () {
454
+ const markReady = ready ? ready.open.pipe(Effect.asVoid) : Effect.void
455
+ const { msg, part, cwd } = yield* Effect.gen(function* () {
456
+ const ctx = yield* InstanceState.context
457
+ const session = yield* sessions.get(input.sessionID).pipe(Effect.orDie)
458
+ if (session.revert) {
459
+ yield* revert.cleanup(session)
460
+ }
461
+ const agent = yield* agents.get(input.agent)
462
+ if (!agent) {
463
+ const available = (yield* agents.list()).filter((a) => !a.hidden).map((a) => a.name)
464
+ const hint = available.length ? ` Available agents: ${available.join(", ")}` : ""
465
+ const error = new NamedError.Unknown({ message: `Agent not found: "${input.agent}".${hint}` })
466
+ yield* events.publish(Session.Event.Error, { sessionID: input.sessionID, error: error.toObject() })
467
+ throw error
468
+ }
469
+ const model = input.model ?? agent.model ?? (yield* currentModel(input.sessionID))
470
+ const userMsg: SessionV1.User = {
471
+ id: input.messageID ?? MessageID.ascending(),
472
+ sessionID: input.sessionID,
473
+ time: { created: Date.now() },
474
+ role: "user",
475
+ agent: input.agent,
476
+ model: { providerID: model.providerID, modelID: model.modelID },
477
+ }
478
+ yield* sessions.updateMessage(userMsg)
479
+ const userPart: SessionV1.Part = {
480
+ type: "text",
481
+ id: PartID.ascending(),
482
+ messageID: userMsg.id,
483
+ sessionID: input.sessionID,
484
+ text: "The following tool was executed by the user",
485
+ synthetic: true,
486
+ }
487
+ yield* sessions.updatePart(userPart)
488
+
489
+ const msg: SessionV1.Assistant = {
490
+ id: MessageID.ascending(),
491
+ sessionID: input.sessionID,
492
+ parentID: userMsg.id,
493
+ mode: input.agent,
494
+ agent: input.agent,
495
+ cost: 0,
496
+ path: { cwd: ctx.directory, root: ctx.worktree },
497
+ time: { created: Date.now() },
498
+ role: "assistant",
499
+ tokens: { input: 0, output: 0, reasoning: 0, cache: { read: 0, write: 0 } },
500
+ modelID: model.modelID,
501
+ providerID: model.providerID,
502
+ }
503
+ yield* sessions.updateMessage(msg)
504
+ const started = Date.now()
505
+ const part: SessionV1.ToolPart = {
506
+ type: "tool",
507
+ id: PartID.ascending(),
508
+ messageID: msg.id,
509
+ sessionID: input.sessionID,
510
+ tool: ShellID.ToolID,
511
+ callID: ulid(),
512
+ state: {
513
+ status: "running",
514
+ time: { start: started },
515
+ input: { command: input.command },
516
+ },
517
+ }
518
+ yield* sessions.updatePart(part)
519
+ return { msg, part, cwd: ctx.directory }
520
+ }).pipe(Effect.ensuring(markReady))
521
+
522
+ const cfg = yield* config.get()
523
+ const sh = Shell.preferred(cfg.shell)
524
+ const args = Shell.args(sh, input.command, cwd)
525
+ let output = ""
526
+ let aborted = false
527
+
528
+ const finish = Effect.uninterruptible(
529
+ Effect.gen(function* () {
530
+ if (aborted) {
531
+ output += "\n\n" + ["<metadata>", "User aborted the command", "</metadata>"].join("\n")
532
+ }
533
+ const completed = Date.now()
534
+ if (!msg.time.completed) {
535
+ msg.time.completed = completed
536
+ yield* sessions.updateMessage(msg)
537
+ }
538
+ if (part.state.status === "running") {
539
+ part.state = {
540
+ status: "completed",
541
+ time: { ...part.state.time, end: completed },
542
+ input: part.state.input,
543
+ title: "",
544
+ metadata: { output },
545
+ output,
546
+ }
547
+ yield* sessions.updatePart(part)
548
+ }
549
+ }),
550
+ )
551
+
552
+ const exit = yield* restore(
553
+ Effect.gen(function* () {
554
+ const shellEnv = yield* plugin.trigger(
555
+ "shell.env",
556
+ { cwd, sessionID: input.sessionID, callID: part.callID },
557
+ { env: {} },
558
+ )
559
+ const cmd = ChildProcess.make(sh, args, {
560
+ cwd,
561
+ extendEnv: true,
562
+ env: { ...shellEnv.env, TERM: "dumb" },
563
+ stdin: "ignore",
564
+ forceKillAfter: "3 seconds",
565
+ })
566
+ const handle = yield* spawner.spawn(cmd)
567
+ yield* Stream.runForEach(Stream.decodeText(handle.all), (chunk) =>
568
+ Effect.gen(function* () {
569
+ output += chunk
570
+ if (part.state.status === "running") {
571
+ part.state.metadata = { output }
572
+ yield* sessions.updatePart(part)
573
+ }
574
+ }),
575
+ )
576
+ yield* handle.exitCode
577
+ }).pipe(Effect.scoped, Effect.orDie),
578
+ ).pipe(Effect.exit)
579
+
580
+ if (Exit.isFailure(exit) && Cause.hasInterrupts(exit.cause) && !Cause.hasDies(exit.cause)) {
581
+ aborted = true
582
+ }
583
+ yield* finish
584
+
585
+ if (Exit.isFailure(exit) && !aborted && !Cause.hasInterruptsOnly(exit.cause)) {
586
+ return yield* Effect.failCause(exit.cause)
587
+ }
588
+
589
+ return { info: msg, parts: [part] }
590
+ }),
591
+ )
592
+ })
593
+
594
+ const getModel = Effect.fn("SessionPrompt.getModel")(function* (
595
+ providerID: ProviderV2.ID,
596
+ modelID: ModelV2.ID,
597
+ sessionID: SessionID,
598
+ ) {
599
+ const exit = yield* provider.getModel(providerID, modelID).pipe(Effect.exit)
600
+ if (Exit.isSuccess(exit)) return exit.value
601
+ const err = Cause.squash(exit.cause)
602
+ if (Provider.ModelNotFoundError.isInstance(err)) {
603
+ const hint = err.suggestions?.length ? ` Did you mean: ${err.suggestions.join(", ")}?` : ""
604
+ yield* events.publish(Session.Event.Error, {
605
+ sessionID,
606
+ error: new NamedError.Unknown({
607
+ message: `Model not found: ${err.providerID}/${err.modelID}.${hint}`,
608
+ }).toObject(),
609
+ })
610
+ }
611
+ return yield* Effect.die(err)
612
+ })
613
+
614
+ const currentModel = Effect.fnUntraced(function* (sessionID: SessionID) {
615
+ const current = yield* db
616
+ .select({ model: SessionTable.model })
617
+ .from(SessionTable)
618
+ .where(eq(SessionTable.id, sessionID))
619
+ .get()
620
+ .pipe(Effect.orDie)
621
+ if (current?.model) {
622
+ return {
623
+ providerID: ProviderV2.ID.make(current.model.providerID),
624
+ modelID: ModelV2.ID.make(current.model.id),
625
+ ...(current.model.variant && current.model.variant !== "default" ? { variant: current.model.variant } : {}),
626
+ }
627
+ }
628
+ const match = yield* sessions
629
+ .findMessage(sessionID, (m) => m.info.role === "user" && !!m.info.model)
630
+ .pipe(Effect.orDie)
631
+ if (Option.isSome(match) && match.value.info.role === "user") return match.value.info.model
632
+ return yield* provider.defaultModel().pipe(Effect.orDie)
633
+ })
634
+
635
+ const createUserMessage = Effect.fn("SessionPrompt.createUserMessage")(function* (input: PromptInput) {
636
+ const agentName = input.agent
637
+ const ag = agentName ? yield* agents.get(agentName) : yield* agents.defaultInfo()
638
+ if (!ag) {
639
+ const available = (yield* agents.list()).filter((a) => !a.hidden).map((a) => a.name)
640
+ const hint = available.length ? ` Available agents: ${available.join(", ")}` : ""
641
+ const error = new NamedError.Unknown({ message: `Agent not found: "${agentName}".${hint}` })
642
+ yield* events.publish(Session.Event.Error, { sessionID: input.sessionID, error: error.toObject() })
643
+ throw error
644
+ }
645
+
646
+ const model = input.model ?? ag.model ?? (yield* currentModel(input.sessionID))
647
+ const same = ag.model && model.providerID === ag.model.providerID && model.modelID === ag.model.modelID
648
+ const full =
649
+ !input.variant && ag.variant && same
650
+ ? yield* provider
651
+ .getModel(model.providerID, model.modelID)
652
+ .pipe(Effect.catchIf(Provider.ModelNotFoundError.isInstance, () => Effect.succeed(undefined)))
653
+ : undefined
654
+ const variant = input.variant ?? (ag.variant && full?.variants?.[ag.variant] ? ag.variant : undefined)
655
+
656
+ const info: SessionV1.User = {
657
+ id: input.messageID ?? MessageID.ascending(),
658
+ role: "user",
659
+ sessionID: input.sessionID,
660
+ time: { created: Date.now() },
661
+ tools: input.tools,
662
+ agent: ag.name,
663
+ model: {
664
+ providerID: model.providerID,
665
+ modelID: model.modelID,
666
+ variant,
667
+ },
668
+ system: input.system,
669
+ format: input.format,
670
+ }
671
+
672
+ const current = yield* sessions.get(input.sessionID).pipe(Effect.orDie)
673
+ if (
674
+ current.agent !== info.agent ||
675
+ current.model?.providerID !== info.model.providerID ||
676
+ current.model?.id !== info.model.modelID ||
677
+ (current.model?.variant === "default" ? undefined : current.model?.variant) !== info.model.variant
678
+ ) {
679
+ yield* sessions.setAgentModel({
680
+ sessionID: input.sessionID,
681
+ agent: info.agent,
682
+ model: {
683
+ id: info.model.modelID,
684
+ providerID: info.model.providerID,
685
+ variant: info.model.variant ?? "default",
686
+ },
687
+ time: info.time.created,
688
+ })
689
+ }
690
+
691
+ yield* Effect.addFinalizer(() => instruction.clear(info.id))
692
+
693
+ type Draft<T> = T extends SessionV1.Part ? Omit<T, "id"> & { id?: string } : never
694
+ const assign = (part: Draft<SessionV1.Part>): SessionV1.Part => ({
695
+ ...part,
696
+ id: part.id ? PartID.make(part.id) : PartID.ascending(),
697
+ })
698
+
699
+ const resolvePart: (part: PromptInput["parts"][number]) => Effect.Effect<Draft<SessionV1.Part>[]> = Effect.fn(
700
+ "SessionPrompt.resolveUserPart",
701
+ )(function* (part) {
702
+ if (part.type === "file") {
703
+ if (part.source?.type === "resource") {
704
+ const { clientName, uri } = part.source
705
+ yield* Effect.logInfo("mcp resource", { clientName, uri, mime: part.mime })
706
+ const pieces: Draft<SessionV1.Part>[] = [
707
+ {
708
+ messageID: info.id,
709
+ sessionID: input.sessionID,
710
+ type: "text",
711
+ synthetic: true,
712
+ text: `Reading MCP resource: ${part.filename} (${uri})`,
713
+ },
714
+ ]
715
+ const exit = yield* mcp.readResource(clientName, uri).pipe(Effect.exit)
716
+ if (Exit.isSuccess(exit)) {
717
+ const content = exit.value
718
+ if (!content) throw new Error(`Resource not found: ${clientName}/${uri}`)
719
+ const items = Array.isArray(content.contents) ? content.contents : [content.contents]
720
+ for (const c of items) {
721
+ if (!c || typeof c !== "object") continue
722
+ if ("text" in c && typeof c.text === "string" && c.text) {
723
+ pieces.push({
724
+ messageID: info.id,
725
+ sessionID: input.sessionID,
726
+ type: "text",
727
+ synthetic: true,
728
+ text: c.text,
729
+ })
730
+ } else if ("blob" in c && typeof c.blob === "string" && c.blob) {
731
+ const mime = "mimeType" in c && typeof c.mimeType === "string" ? c.mimeType : part.mime
732
+ const filename = "uri" in c && typeof c.uri === "string" ? c.uri : part.filename
733
+ const size = mcpResourceBase64Size(c.blob)
734
+ if (!SUPPORTED_MCP_RESOURCE_ATTACHMENT_MIMES.has(mime)) {
735
+ pieces.push({
736
+ messageID: info.id,
737
+ sessionID: input.sessionID,
738
+ type: "text",
739
+ synthetic: true,
740
+ text: `[Binary MCP resource omitted: ${filename ?? uri} (${mime}, ${formatMcpResourceBytes(size)}) is not a supported attachment type]`,
741
+ })
742
+ continue
743
+ }
744
+ if (size > MAX_MCP_RESOURCE_BLOB_BYTES) {
745
+ pieces.push({
746
+ messageID: info.id,
747
+ sessionID: input.sessionID,
748
+ type: "text",
749
+ synthetic: true,
750
+ text: `[Binary MCP resource omitted: ${filename ?? uri} (${mime}, ${formatMcpResourceBytes(size)}) exceeds ${formatMcpResourceBytes(MAX_MCP_RESOURCE_BLOB_BYTES)}]`,
751
+ })
752
+ continue
753
+ }
754
+ pieces.push({
755
+ messageID: info.id,
756
+ sessionID: input.sessionID,
757
+ type: "text",
758
+ synthetic: true,
759
+ text: `[Binary MCP resource attached: ${filename ?? uri} (${mime})]`,
760
+ })
761
+ pieces.push({
762
+ messageID: info.id,
763
+ sessionID: input.sessionID,
764
+ type: "file",
765
+ mime,
766
+ filename,
767
+ url: `data:${mime};base64,${c.blob}`,
768
+ })
769
+ }
770
+ }
771
+ } else {
772
+ const error = Cause.squash(exit.cause)
773
+ yield* Effect.logError("failed to read MCP resource", { error, clientName, uri })
774
+ const message = error instanceof Error ? error.message : String(error)
775
+ pieces.push({
776
+ messageID: info.id,
777
+ sessionID: input.sessionID,
778
+ type: "text",
779
+ synthetic: true,
780
+ text: `Failed to read MCP resource ${part.filename}: ${message}`,
781
+ })
782
+ }
783
+ return pieces
784
+ }
785
+ const url = new URL(part.url)
786
+ switch (url.protocol) {
787
+ case "data:":
788
+ if (part.mime === "text/plain") {
789
+ return [
790
+ {
791
+ messageID: info.id,
792
+ sessionID: input.sessionID,
793
+ type: "text",
794
+ synthetic: true,
795
+ text: `Called the Read tool with the following input: ${JSON.stringify({ filePath: part.filename })}`,
796
+ },
797
+ {
798
+ messageID: info.id,
799
+ sessionID: input.sessionID,
800
+ type: "text",
801
+ synthetic: true,
802
+ text: decodeDataUrl(part.url),
803
+ },
804
+ { ...part, messageID: info.id, sessionID: input.sessionID },
805
+ ]
806
+ }
807
+ break
808
+ case "file:": {
809
+ yield* Effect.logInfo("file", { mime: part.mime })
810
+ const filepath = fileURLToPath(part.url)
811
+ const mime = (yield* fsys.isDir(filepath)) ? "application/x-directory" : part.mime
812
+
813
+ const { read } = yield* registry.named()
814
+ const execRead = (args: Parameters<typeof read.execute>[0], extra?: Tool.Context["extra"]) => {
815
+ const controller = new AbortController()
816
+ return read
817
+ .execute(args, {
818
+ sessionID: input.sessionID,
819
+ abort: controller.signal,
820
+ agent: input.agent!,
821
+ messageID: info.id,
822
+ extra: { bypassCwdCheck: true, ...extra },
823
+ messages: [],
824
+ metadata: () => Effect.void,
825
+ ask: () => Effect.void,
826
+ })
827
+ .pipe(Effect.onInterrupt(() => Effect.sync(() => controller.abort())))
828
+ }
829
+
830
+ if (mime === "text/plain") {
831
+ let offset: number | undefined
832
+ let limit: number | undefined
833
+ const range = { start: url.searchParams.get("start"), end: url.searchParams.get("end") }
834
+ if (range.start != null) {
835
+ const filePathURI = part.url.split("?")[0]
836
+ let start = parseInt(range.start)
837
+ let end = range.end ? parseInt(range.end) : undefined
838
+ if (start === end) {
839
+ const symbols = yield* lsp.documentSymbol(filePathURI).pipe(Effect.catch(() => Effect.succeed([])))
840
+ for (const symbol of symbols) {
841
+ let r: LSP.Range | undefined
842
+ if ("range" in symbol) r = symbol.range
843
+ else if ("location" in symbol) r = symbol.location.range
844
+ if (r?.start?.line && r?.start?.line === start) {
845
+ start = r.start.line
846
+ end = r?.end?.line ?? start
847
+ break
848
+ }
849
+ }
850
+ }
851
+ offset = Math.max(start, 1)
852
+ if (end) limit = end - (offset - 1)
853
+ }
854
+ const args = { filePath: filepath, offset, limit }
855
+ const pieces: Draft<SessionV1.Part>[] = [
856
+ {
857
+ messageID: info.id,
858
+ sessionID: input.sessionID,
859
+ type: "text",
860
+ synthetic: true,
861
+ text: `Called the Read tool with the following input: ${JSON.stringify(args)}`,
862
+ },
863
+ ]
864
+ const exit = yield* provider.getModel(info.model.providerID, info.model.modelID).pipe(
865
+ Effect.flatMap((mdl) => execRead(args, { model: mdl })),
866
+ Effect.exit,
867
+ )
868
+ if (Exit.isSuccess(exit)) {
869
+ const result = exit.value
870
+ pieces.push({
871
+ messageID: info.id,
872
+ sessionID: input.sessionID,
873
+ type: "text",
874
+ synthetic: true,
875
+ text: result.output,
876
+ })
877
+ if (result.attachments?.length) {
878
+ pieces.push(
879
+ ...result.attachments.map((a) => ({
880
+ ...a,
881
+ synthetic: true,
882
+ filename: a.filename ?? part.filename,
883
+ messageID: info.id,
884
+ sessionID: input.sessionID,
885
+ })),
886
+ )
887
+ } else {
888
+ pieces.push({ ...part, mime, messageID: info.id, sessionID: input.sessionID })
889
+ }
890
+ } else {
891
+ const error = Cause.squash(exit.cause)
892
+ yield* Effect.logError("failed to read file", { error, filepath })
893
+ const message = error instanceof Error ? error.message : String(error)
894
+ yield* events.publish(Session.Event.Error, {
895
+ sessionID: input.sessionID,
896
+ error: new NamedError.Unknown({ message }).toObject(),
897
+ })
898
+ pieces.push({
899
+ messageID: info.id,
900
+ sessionID: input.sessionID,
901
+ type: "text",
902
+ synthetic: true,
903
+ text: `Read tool failed to read ${filepath} with the following error: ${message}`,
904
+ })
905
+ }
906
+ return pieces
907
+ }
908
+
909
+ if (mime === "application/x-directory") {
910
+ const args = { filePath: filepath }
911
+ const exit = yield* execRead(args).pipe(Effect.exit)
912
+ if (Exit.isFailure(exit)) {
913
+ const error = Cause.squash(exit.cause)
914
+ yield* Effect.logError("failed to read directory", { error, filepath })
915
+ const message = error instanceof Error ? error.message : String(error)
916
+ yield* events.publish(Session.Event.Error, {
917
+ sessionID: input.sessionID,
918
+ error: new NamedError.Unknown({ message }).toObject(),
919
+ })
920
+ return [
921
+ {
922
+ messageID: info.id,
923
+ sessionID: input.sessionID,
924
+ type: "text",
925
+ synthetic: true,
926
+ text: `Read tool failed to read ${filepath} with the following error: ${message}`,
927
+ },
928
+ ]
929
+ }
930
+ return [
931
+ {
932
+ messageID: info.id,
933
+ sessionID: input.sessionID,
934
+ type: "text",
935
+ synthetic: true,
936
+ text: `Called the Read tool with the following input: ${JSON.stringify(args)}`,
937
+ },
938
+ {
939
+ messageID: info.id,
940
+ sessionID: input.sessionID,
941
+ type: "text",
942
+ synthetic: true,
943
+ text: exit.value.output,
944
+ },
945
+ { ...part, mime, messageID: info.id, sessionID: input.sessionID },
946
+ ]
947
+ }
948
+
949
+ return [
950
+ {
951
+ messageID: info.id,
952
+ sessionID: input.sessionID,
953
+ type: "text",
954
+ synthetic: true,
955
+ text: `Called the Read tool with the following input: {"filePath":"${filepath}"}`,
956
+ },
957
+ {
958
+ id: part.id,
959
+ messageID: info.id,
960
+ sessionID: input.sessionID,
961
+ type: "file",
962
+ url:
963
+ `data:${mime};base64,` +
964
+ Buffer.from(yield* fsys.readFile(filepath).pipe(Effect.catch(Effect.die))).toString("base64"),
965
+ mime,
966
+ filename: part.filename!,
967
+ source: part.source,
968
+ },
969
+ ]
970
+ }
971
+ }
972
+ }
973
+
974
+ if (part.type === "agent") {
975
+ const perm = Permission.evaluate("task", part.name, ag.permission)
976
+ const hint = perm.action === "deny" ? " . Invoked by user; guaranteed to exist." : ""
977
+ return [
978
+ { ...part, messageID: info.id, sessionID: input.sessionID },
979
+ {
980
+ messageID: info.id,
981
+ sessionID: input.sessionID,
982
+ type: "text",
983
+ synthetic: true,
984
+ text:
985
+ " Use the above message and context to generate a prompt and call the task tool with subagent: " +
986
+ part.name +
987
+ hint,
988
+ },
989
+ ]
990
+ }
991
+
992
+ return [{ ...part, messageID: info.id, sessionID: input.sessionID }]
993
+ })
994
+
995
+ const resolvedParts = yield* Effect.forEach(input.parts, resolvePart, { concurrency: "unbounded" }).pipe(
996
+ Effect.map((x) => x.flat().map(assign)),
997
+ )
998
+
999
+ yield* plugin.trigger(
1000
+ "chat.message",
1001
+ {
1002
+ sessionID: input.sessionID,
1003
+ agent: input.agent,
1004
+ model: input.model,
1005
+ messageID: input.messageID,
1006
+ variant: input.variant,
1007
+ },
1008
+ { message: info, parts: resolvedParts },
1009
+ )
1010
+
1011
+ const parts = yield* Effect.forEach(resolvedParts, (part) =>
1012
+ part.type === "file" && part.mime.startsWith("image/")
1013
+ ? image.normalize(part).pipe(
1014
+ Effect.catchIf(
1015
+ (error) => error instanceof Image.ResizerUnavailableError,
1016
+ () => Effect.succeed(part),
1017
+ ),
1018
+ )
1019
+ : Effect.succeed(part),
1020
+ )
1021
+
1022
+ const parsed = decodeMessageInfo(info, { errors: "all", propertyOrder: "original" })
1023
+ if (Exit.isFailure(parsed)) {
1024
+ yield* Effect.logError("invalid user message before save", {
1025
+ sessionID: input.sessionID,
1026
+ messageID: info.id,
1027
+ agent: info.agent,
1028
+ model: info.model,
1029
+ cause: Cause.pretty(parsed.cause),
1030
+ })
1031
+ }
1032
+ for (const [index, part] of parts.entries()) {
1033
+ const p = decodeMessagePart(part, { errors: "all", propertyOrder: "original" })
1034
+ if (Exit.isSuccess(p)) continue
1035
+ yield* Effect.logError("invalid user part before save", {
1036
+ sessionID: input.sessionID,
1037
+ messageID: info.id,
1038
+ partID: part.id,
1039
+ partType: part.type,
1040
+ index,
1041
+ cause: Cause.pretty(p.cause),
1042
+ part,
1043
+ })
1044
+ }
1045
+
1046
+ yield* sessions.updateMessage(info)
1047
+ for (const part of parts) yield* sessions.updatePart(part)
1048
+
1049
+ return { info, parts }
1050
+ }, Effect.scoped)
1051
+
1052
+ const prompt: (input: PromptInput) => Effect.Effect<SessionV1.WithParts, Image.Error> = Effect.fn(
1053
+ "SessionPrompt.prompt",
1054
+ )(function* (input: PromptInput) {
1055
+ const session = yield* sessions.get(input.sessionID).pipe(Effect.orDie)
1056
+ yield* revert.cleanup(session)
1057
+ const message = yield* createUserMessage(input)
1058
+ yield* sessions.touch(input.sessionID)
1059
+
1060
+ const permissions: PermissionV1.Rule[] = []
1061
+ for (const [t, enabled] of Object.entries(input.tools ?? {})) {
1062
+ permissions.push({ permission: t, action: enabled ? "allow" : "deny", pattern: "*" })
1063
+ }
1064
+ if (permissions.length > 0) {
1065
+ session.permission = permissions
1066
+ yield* sessions.setPermission({ sessionID: session.id, permission: permissions })
1067
+ }
1068
+
1069
+ if (input.noReply === true) return message
1070
+ return yield* loop({ sessionID: input.sessionID })
1071
+ })
1072
+
1073
+ const lastAssistant = Effect.fnUntraced(function* (sessionID: SessionID) {
1074
+ const match = yield* sessions.findMessage(sessionID, (m) => m.info.role !== "user").pipe(Effect.orDie)
1075
+ if (Option.isSome(match)) return match.value
1076
+ const msgs = yield* sessions.messages({ sessionID, limit: 1 }).pipe(Effect.orDie)
1077
+ if (msgs.length > 0) return msgs[0]
1078
+ throw new Error("Impossible")
1079
+ })
1080
+
1081
+ const runLoop: (sessionID: SessionID) => Effect.Effect<SessionV1.WithParts> = Effect.fn("SessionPrompt.run")(
1082
+ function* (sessionID: SessionID) {
1083
+ const ctx = yield* InstanceState.context
1084
+ let structured: unknown
1085
+ let step = 0
1086
+ const session = yield* sessions.get(sessionID).pipe(Effect.orDie)
1087
+
1088
+ while (true) {
1089
+ yield* status.set(sessionID, { type: "busy" })
1090
+ yield* Effect.logInfo("loop", { "session.id": sessionID, step })
1091
+
1092
+ let msgs = yield* MessageV2.filterCompactedEffect(sessionID).pipe(
1093
+ Effect.provideService(Database.Service, database),
1094
+ )
1095
+
1096
+ const { user: lastUser, assistant: lastAssistant, finished: lastFinished, tasks } = MessageV2.latest(msgs)
1097
+
1098
+ if (!lastUser) throw new Error("No user message found in stream. This should never happen.")
1099
+
1100
+ const lastAssistantMsg = msgs.findLast(
1101
+ (msg) => msg.info.role === "assistant" && msg.info.id === lastAssistant?.id,
1102
+ )
1103
+ // Some providers return "stop" even when the assistant message contains
1104
+ // tool calls. Keep the loop running so tool results can be sent back to
1105
+ // the model, but ignore cleanup-marked interrupted orphans.
1106
+ const hasToolCalls =
1107
+ lastAssistantMsg?.parts.some(
1108
+ (part) => part.type === "tool" && !part.metadata?.providerExecuted && !isOrphanedInterruptedTool(part),
1109
+ ) ?? false
1110
+
1111
+ if (
1112
+ lastAssistant?.finish &&
1113
+ !["tool-calls"].includes(lastAssistant.finish) &&
1114
+ !hasToolCalls &&
1115
+ lastUser.id < lastAssistant.id
1116
+ ) {
1117
+ const orphan = lastAssistantMsg?.parts.find(
1118
+ (part): part is SessionV1.ToolPart => part.type === "tool" && isOrphanedInterruptedTool(part),
1119
+ )
1120
+ if (orphan) {
1121
+ yield* Effect.logWarning("loop exit with orphaned interrupted tool", {
1122
+ "session.id": sessionID,
1123
+ messageID: lastAssistant.id,
1124
+ tool: orphan.tool,
1125
+ callID: orphan.callID,
1126
+ })
1127
+ }
1128
+ yield* Effect.logInfo("exiting loop", { "session.id": sessionID })
1129
+ break
1130
+ }
1131
+
1132
+ step++
1133
+ if (step === 1)
1134
+ yield* title({
1135
+ session,
1136
+ modelID: lastUser.model.modelID,
1137
+ providerID: lastUser.model.providerID,
1138
+ history: msgs,
1139
+ }).pipe(Effect.ignore, Effect.forkIn(scope))
1140
+
1141
+ const model = yield* getModel(lastUser.model.providerID, lastUser.model.modelID, sessionID)
1142
+ const task = tasks.pop()
1143
+
1144
+ if (task?.type === "subtask") {
1145
+ yield* handleSubtask({ task, model, lastUser, sessionID, session, msgs })
1146
+ continue
1147
+ }
1148
+
1149
+ if (task?.type === "compaction") {
1150
+ const result = yield* compaction.process({
1151
+ messages: msgs,
1152
+ parentID: lastUser.id,
1153
+ sessionID,
1154
+ auto: task.auto,
1155
+ overflow: task.overflow,
1156
+ })
1157
+ if (result === "stop") break
1158
+ continue
1159
+ }
1160
+
1161
+ if (
1162
+ lastFinished &&
1163
+ lastFinished.summary !== true &&
1164
+ (yield* compaction.isOverflow({ tokens: lastFinished.tokens, model }))
1165
+ ) {
1166
+ yield* compaction.create({ sessionID, agent: lastUser.agent, model: lastUser.model, auto: true })
1167
+ continue
1168
+ }
1169
+
1170
+ const agent = yield* agents.get(lastUser.agent)
1171
+ if (!agent) {
1172
+ const available = (yield* agents.list()).filter((a) => !a.hidden).map((a) => a.name)
1173
+ const hint = available.length ? ` Available agents: ${available.join(", ")}` : ""
1174
+ const error = new NamedError.Unknown({ message: `Agent not found: "${lastUser.agent}".${hint}` })
1175
+ yield* events.publish(Session.Event.Error, { sessionID, error: error.toObject() })
1176
+ throw error
1177
+ }
1178
+ const maxSteps = agent.steps ?? Infinity
1179
+ const isLastStep = step >= maxSteps
1180
+ msgs = yield* SessionReminders.apply({ messages: msgs, agent, session }).pipe(
1181
+ Effect.provideService(RuntimeFlags.Service, flags),
1182
+ Effect.provideService(FSUtil.Service, fsys),
1183
+ Effect.provideService(Session.Service, sessions),
1184
+ )
1185
+
1186
+ const msg: SessionV1.Assistant = {
1187
+ id: MessageID.ascending(),
1188
+ parentID: lastUser.id,
1189
+ role: "assistant",
1190
+ mode: agent.name,
1191
+ agent: agent.name,
1192
+ variant: lastUser.model.variant,
1193
+ path: { cwd: ctx.directory, root: ctx.worktree },
1194
+ cost: 0,
1195
+ tokens: { input: 0, output: 0, reasoning: 0, cache: { read: 0, write: 0 } },
1196
+ modelID: model.id,
1197
+ providerID: model.providerID,
1198
+ time: { created: Date.now() },
1199
+ sessionID,
1200
+ }
1201
+ yield* sessions.updateMessage(msg)
1202
+
1203
+ const finalizeInterruptedAssistant = Effect.gen(function* () {
1204
+ if (msg.time.completed) return
1205
+ msg.error ??= MessageV2.fromError(new DOMException("Aborted", "AbortError"), {
1206
+ providerID: msg.providerID,
1207
+ aborted: true,
1208
+ })
1209
+ msg.time.completed = Date.now()
1210
+ yield* sessions.updateMessage(msg)
1211
+ })
1212
+
1213
+ const handle = yield* processor
1214
+ .create({
1215
+ assistantMessage: msg,
1216
+ sessionID,
1217
+ model,
1218
+ })
1219
+ .pipe(Effect.onInterrupt(() => finalizeInterruptedAssistant))
1220
+
1221
+ const outcome: "break" | "continue" = yield* Effect.gen(function* () {
1222
+ const lastUserMsg = msgs.findLast((m) => m.info.role === "user")
1223
+ const bypassAgentCheck = lastUserMsg?.parts.some((p) => p.type === "agent") ?? false
1224
+ const promptOps = yield* ops()
1225
+
1226
+ const tools = yield* SessionTools.resolve({
1227
+ agent,
1228
+ session,
1229
+ model,
1230
+ processor: handle,
1231
+ bypassAgentCheck,
1232
+ messages: msgs,
1233
+ promptOps,
1234
+ }).pipe(
1235
+ Effect.provideService(Plugin.Service, plugin),
1236
+ Effect.provideService(Permission.Service, permission),
1237
+ Effect.provideService(ToolRegistry.Service, registry),
1238
+ Effect.provideService(MCP.Service, mcp),
1239
+ Effect.provideService(Truncate.Service, truncate),
1240
+ )
1241
+
1242
+ if (lastUser.format?.type === "json_schema") {
1243
+ tools["StructuredOutput"] = createStructuredOutputTool({
1244
+ schema: lastUser.format.schema,
1245
+ onSuccess(output) {
1246
+ structured = output
1247
+ },
1248
+ })
1249
+ }
1250
+
1251
+ if (step === 1)
1252
+ yield* summary.summarize({ sessionID, messageID: lastUser.id }).pipe(Effect.ignore, Effect.forkIn(scope))
1253
+
1254
+ yield* plugin.trigger("experimental.chat.messages.transform", {}, { messages: msgs })
1255
+
1256
+ const [skills, env, instructions, mcpInstructions, modelMsgs] = yield* Effect.all([
1257
+ sys.skills(agent),
1258
+ sys.environment(model),
1259
+ instruction.system().pipe(Effect.orDie),
1260
+ sys.mcp(agent, session.permission),
1261
+ MessageV2.toModelMessagesEffect(msgs, model),
1262
+ ])
1263
+ const system = [
1264
+ ...env,
1265
+ ...instructions,
1266
+ ...(mcpInstructions ? [mcpInstructions] : []),
1267
+ ...(skills ? [skills] : []),
1268
+ ]
1269
+ const format = lastUser.format ?? { type: "text" as const }
1270
+ if (format.type === "json_schema") system.push(STRUCTURED_OUTPUT_SYSTEM_PROMPT)
1271
+ const result = yield* handle.process({
1272
+ user: lastUser,
1273
+ agent,
1274
+ permission: session.permission,
1275
+ sessionID,
1276
+ parentSessionID: session.parentID,
1277
+ system,
1278
+ messages: [
1279
+ ...modelMsgs,
1280
+ ...(isLastStep ? [{ role: "assistant" as const, content: MAX_STEPS_PROMPT }] : []),
1281
+ ],
1282
+ tools,
1283
+ model,
1284
+ toolChoice: format.type === "json_schema" ? "required" : undefined,
1285
+ })
1286
+
1287
+ if (structured !== undefined) {
1288
+ handle.message.structured = structured
1289
+ handle.message.finish = handle.message.finish ?? "stop"
1290
+ yield* sessions.updateMessage(handle.message)
1291
+ return "break" as const
1292
+ }
1293
+
1294
+ const finished = handle.message.finish && !["tool-calls", "unknown"].includes(handle.message.finish)
1295
+ if (finished && !handle.message.error) {
1296
+ // Surface any content-filter finish (e.g. Anthropic stop_reason:
1297
+ // refusal) as an error. These turns may have produced no visible
1298
+ // output at all — previously the session went idle silently — or
1299
+ // partial text that was cut off by the provider's filter.
1300
+ if (handle.message.finish === "content-filter") {
1301
+ handle.message.error = new SessionV1.ContentFilterError({
1302
+ message: "The response was blocked by the provider's content filter",
1303
+ }).toObject()
1304
+ yield* sessions.updateMessage(handle.message)
1305
+ yield* events.publish(Session.Event.Error, { sessionID, error: handle.message.error })
1306
+ return "break" as const
1307
+ }
1308
+ if (format.type === "json_schema") {
1309
+ handle.message.error = new SessionV1.StructuredOutputError({
1310
+ message: "Model did not produce structured output",
1311
+ retries: 0,
1312
+ }).toObject()
1313
+ yield* sessions.updateMessage(handle.message)
1314
+ return "break" as const
1315
+ }
1316
+ }
1317
+
1318
+ if (result === "stop") return "break" as const
1319
+ if (result === "compact") {
1320
+ yield* compaction.create({
1321
+ sessionID,
1322
+ agent: lastUser.agent,
1323
+ model: lastUser.model,
1324
+ auto: true,
1325
+ overflow: !handle.message.finish,
1326
+ })
1327
+ }
1328
+ return "continue" as const
1329
+ }).pipe(
1330
+ Effect.ensuring(instruction.clear(handle.message.id)),
1331
+ Effect.onInterrupt(() => finalizeInterruptedAssistant),
1332
+ )
1333
+ if (outcome === "break") break
1334
+ continue
1335
+ }
1336
+
1337
+ yield* compaction.prune({ sessionID }).pipe(Effect.ignore, Effect.forkIn(scope))
1338
+ return yield* lastAssistant(sessionID)
1339
+ },
1340
+ )
1341
+
1342
+ const loop: (input: LoopInput) => Effect.Effect<SessionV1.WithParts> = Effect.fn("SessionPrompt.loop")(function* (
1343
+ input: LoopInput,
1344
+ ) {
1345
+ return yield* state.ensureRunning(input.sessionID, lastAssistant(input.sessionID), runLoop(input.sessionID))
1346
+ })
1347
+
1348
+ const shell: (input: ShellInput) => Effect.Effect<SessionV1.WithParts, Session.BusyError> = Effect.fn(
1349
+ "SessionPrompt.shell",
1350
+ )(function* (input: ShellInput) {
1351
+ const ready = yield* Latch.make()
1352
+ return yield* state.startShell(input.sessionID, lastAssistant(input.sessionID), shellImpl(input, ready), ready)
1353
+ })
1354
+
1355
+ const command = Effect.fn("SessionPrompt.command")(function* (input: CommandInput) {
1356
+ yield* Effect.logInfo("command", {
1357
+ "session.id": input.sessionID,
1358
+ command: input.command,
1359
+ agent: input.agent,
1360
+ })
1361
+ const cmd = yield* commands.get(input.command)
1362
+ if (!cmd) {
1363
+ const available = (yield* commands.list()).map((c) => c.name)
1364
+ const hint = available.length ? ` Available commands: ${available.join(", ")}` : ""
1365
+ const error = new NamedError.Unknown({ message: `Command not found: "${input.command}".${hint}` })
1366
+ yield* events.publish(Session.Event.Error, { sessionID: input.sessionID, error: error.toObject() })
1367
+ throw error
1368
+ }
1369
+ const agentName = cmd.agent ?? input.agent
1370
+
1371
+ const raw = input.arguments.match(argsRegex) ?? []
1372
+ const args = raw.map((arg) => arg.replace(quoteTrimRegex, ""))
1373
+ const templateCommand = yield* Effect.promise(async () => cmd.template)
1374
+
1375
+ const placeholders = templateCommand.match(placeholderRegex) ?? []
1376
+ let last = 0
1377
+ for (const item of placeholders) {
1378
+ const value = Number(item.slice(1))
1379
+ if (value > last) last = value
1380
+ }
1381
+
1382
+ const withArgs = templateCommand.replaceAll(placeholderRegex, (_, index) => {
1383
+ const position = Number(index)
1384
+ const argIndex = position - 1
1385
+ if (argIndex >= args.length) return ""
1386
+ if (position === last) return args.slice(argIndex).join(" ")
1387
+ return args[argIndex]
1388
+ })
1389
+ const usesArgumentsPlaceholder = templateCommand.includes("$ARGUMENTS")
1390
+ let template = withArgs.replaceAll("$ARGUMENTS", input.arguments)
1391
+
1392
+ if (placeholders.length === 0 && !usesArgumentsPlaceholder && input.arguments.trim()) {
1393
+ template = template + "\n\n" + input.arguments
1394
+ }
1395
+
1396
+ const shellMatches = ConfigMarkdown.shell(template)
1397
+ if (shellMatches.length > 0) {
1398
+ const cfg = yield* config.get()
1399
+ const sh = Shell.preferred(cfg.shell)
1400
+ const results = yield* Effect.promise(() =>
1401
+ Promise.all(
1402
+ shellMatches.map(async ([, cmd]) => (await Process.text([cmd], { shell: sh, nothrow: true })).text),
1403
+ ),
1404
+ )
1405
+ let index = 0
1406
+ template = template.replace(bashRegex, () => results[index++])
1407
+ }
1408
+ template = template.trim()
1409
+
1410
+ const taskModel = yield* Effect.gen(function* () {
1411
+ if (cmd.model) return Provider.parseModel(cmd.model)
1412
+ if (cmd.agent) {
1413
+ const cmdAgent = yield* agents.get(cmd.agent)
1414
+ if (cmdAgent?.model) return cmdAgent.model
1415
+ }
1416
+ if (input.model) return Provider.parseModel(input.model)
1417
+ return yield* currentModel(input.sessionID)
1418
+ })
1419
+
1420
+ yield* getModel(taskModel.providerID, taskModel.modelID, input.sessionID)
1421
+
1422
+ const agent = agentName ? yield* agents.get(agentName) : yield* agents.defaultInfo()
1423
+ if (!agent) {
1424
+ const available = (yield* agents.list()).filter((a) => !a.hidden).map((a) => a.name)
1425
+ const hint = available.length ? ` Available agents: ${available.join(", ")}` : ""
1426
+ const error = new NamedError.Unknown({ message: `Agent not found: "${agentName}".${hint}` })
1427
+ yield* events.publish(Session.Event.Error, { sessionID: input.sessionID, error: error.toObject() })
1428
+ throw error
1429
+ }
1430
+
1431
+ const templateParts = yield* resolvePromptParts(template)
1432
+ const inputFiles = new Set(
1433
+ input.parts?.filter((part) => new URL(part.url).protocol === "file:").map((part) => fileURLToPath(part.url)),
1434
+ )
1435
+ const uniqueTemplateParts = templateParts.filter(
1436
+ (part) => part.type !== "file" || !inputFiles.has(fileURLToPath(part.url)),
1437
+ )
1438
+ const isSubtask = (agent.mode === "subagent" && cmd.subtask !== false) || cmd.subtask === true
1439
+ const parts = isSubtask
1440
+ ? [
1441
+ {
1442
+ type: "subtask" as const,
1443
+ agent: agent.name,
1444
+ description: cmd.description ?? "",
1445
+ command: input.command,
1446
+ model: { providerID: taskModel.providerID, modelID: taskModel.modelID },
1447
+ prompt: templateParts.find((y) => y.type === "text")?.text ?? "",
1448
+ },
1449
+ ]
1450
+ : [...uniqueTemplateParts, ...(input.parts ?? [])]
1451
+
1452
+ const userAgent = isSubtask ? (input.agent ?? (yield* agents.defaultInfo()).name) : agent.name
1453
+ const userModel = isSubtask
1454
+ ? input.model
1455
+ ? Provider.parseModel(input.model)
1456
+ : yield* currentModel(input.sessionID)
1457
+ : taskModel
1458
+
1459
+ yield* plugin.trigger(
1460
+ "command.execute.before",
1461
+ { command: input.command, sessionID: input.sessionID, arguments: input.arguments },
1462
+ { parts },
1463
+ )
1464
+
1465
+ const result = yield* prompt({
1466
+ sessionID: input.sessionID,
1467
+ messageID: input.messageID,
1468
+ model: userModel,
1469
+ agent: userAgent,
1470
+ parts,
1471
+ variant: input.variant,
1472
+ })
1473
+ yield* events.publish(Command.Event.Executed, {
1474
+ name: input.command,
1475
+ sessionID: input.sessionID,
1476
+ arguments: input.arguments,
1477
+ messageID: result.info.id,
1478
+ })
1479
+ return result
1480
+ })
1481
+
1482
+ return Service.of({
1483
+ cancel,
1484
+ prompt,
1485
+ loop,
1486
+ shell,
1487
+ command,
1488
+ resolvePromptParts,
1489
+ })
1490
+ }),
1491
+ )
1492
+
1493
+ const ModelRef = Schema.Struct({
1494
+ providerID: ProviderV2.ID,
1495
+ modelID: ModelV2.ID,
1496
+ })
1497
+
1498
+ export const PromptInput = Schema.Struct({
1499
+ sessionID: SessionID,
1500
+ messageID: Schema.optional(MessageID),
1501
+ model: Schema.optional(ModelRef),
1502
+ agent: Schema.optional(Schema.String),
1503
+ noReply: Schema.optional(Schema.Boolean),
1504
+ tools: Schema.optional(Schema.Record(Schema.String, Schema.Boolean)).annotate({
1505
+ description:
1506
+ "@deprecated tools and permissions have been merged, you can set permissions on the session itself now",
1507
+ }),
1508
+ format: Schema.optional(SessionV1.Format),
1509
+ system: Schema.optional(Schema.String),
1510
+ variant: Schema.optional(Schema.String),
1511
+ parts: Schema.Array(
1512
+ Schema.Union([
1513
+ SessionV1.TextPartInput,
1514
+ SessionV1.FilePartInput,
1515
+ SessionV1.AgentPartInput,
1516
+ SessionV1.SubtaskPartInput,
1517
+ ]).annotate({ discriminator: "type" }),
1518
+ ),
1519
+ })
1520
+ export type PromptInput = Schema.Schema.Type<typeof PromptInput>
1521
+
1522
+ export class LoopInput extends Schema.Class<LoopInput>("SessionPrompt.LoopInput")({
1523
+ sessionID: SessionID,
1524
+ }) {}
1525
+
1526
+ export const ShellInput = Schema.Struct({
1527
+ sessionID: SessionID,
1528
+ messageID: Schema.optional(MessageID),
1529
+ agent: Schema.String,
1530
+ model: Schema.optional(ModelRef),
1531
+ command: Schema.String,
1532
+ })
1533
+ export type ShellInput = Schema.Schema.Type<typeof ShellInput>
1534
+
1535
+ export const CommandInput = Schema.Struct({
1536
+ messageID: Schema.optional(MessageID),
1537
+ sessionID: SessionID,
1538
+ agent: Schema.optional(Schema.String),
1539
+ model: Schema.optional(Schema.String),
1540
+ arguments: Schema.String,
1541
+ command: Schema.String,
1542
+ variant: Schema.optional(Schema.String),
1543
+ // Inlined (no identifier annotation) to keep the original SDK output — the
1544
+ // PromptInput call site below references FilePartInput by ref via the
1545
+ // Schema export in message-v2.ts.
1546
+ parts: Schema.optional(
1547
+ Schema.Array(
1548
+ Schema.Union([
1549
+ Schema.Struct({
1550
+ id: Schema.optional(PartID),
1551
+ type: Schema.Literal("file"),
1552
+ mime: Schema.String,
1553
+ filename: Schema.optional(Schema.String),
1554
+ url: Schema.String,
1555
+ source: Schema.optional(SessionV1.FilePartSource),
1556
+ }),
1557
+ ]).annotate({ discriminator: "type" }),
1558
+ ),
1559
+ ),
1560
+ })
1561
+ export type CommandInput = Schema.Schema.Type<typeof CommandInput>
1562
+
1563
+ /** @internal Exported for testing */
1564
+ export function createStructuredOutputTool(input: {
1565
+ schema: Record<string, any>
1566
+ onSuccess: (output: unknown) => void
1567
+ }): AITool {
1568
+ // Remove $schema property if present (not needed for tool input)
1569
+ const { $schema: _, ...toolSchema } = input.schema
1570
+
1571
+ return tool({
1572
+ description: STRUCTURED_OUTPUT_DESCRIPTION,
1573
+ inputSchema: jsonSchema(toolSchema as JSONSchema7),
1574
+ async execute(args) {
1575
+ // AI SDK validates args against inputSchema before calling execute()
1576
+ input.onSuccess(args)
1577
+ return {
1578
+ output: "Structured output captured successfully.",
1579
+ title: "Structured Output",
1580
+ metadata: { valid: true },
1581
+ }
1582
+ },
1583
+ toModelOutput({ output }) {
1584
+ return {
1585
+ type: "text",
1586
+ value: output.output,
1587
+ }
1588
+ },
1589
+ })
1590
+ }
1591
+ const bashRegex = /!`([^`]+)`/g
1592
+ // Match [Image N] as single token, quoted strings, or non-space sequences
1593
+ const argsRegex = /(?:\[Image\s+\d+\]|"[^"]*"|'[^']*'|[^\s"']+)/gi
1594
+ const placeholderRegex = /\$(\d+)/g
1595
+ const quoteTrimRegex = /^["']|["']$/g
1596
+
1597
+ export const node = LayerNode.make({
1598
+ service: Service,
1599
+ layer: layer,
1600
+ deps: [
1601
+ SessionStatus.node,
1602
+ Session.node,
1603
+ Agent.node,
1604
+ Provider.node,
1605
+ SessionProcessor.node,
1606
+ SessionCompaction.node,
1607
+ Plugin.node,
1608
+ Command.node,
1609
+ Config.node,
1610
+ Permission.node,
1611
+ FSUtil.node,
1612
+ MCP.node,
1613
+ LSP.node,
1614
+ ToolRegistry.node,
1615
+ Truncate.node,
1616
+ Image.node,
1617
+ CrossSpawnSpawner.node,
1618
+ Instruction.node,
1619
+ SessionRunState.node,
1620
+ SessionRevert.node,
1621
+ SessionSummary.node,
1622
+ SystemPrompt.node,
1623
+ LLM.node,
1624
+ EventV2Bridge.node,
1625
+ RuntimeFlags.node,
1626
+ Database.node,
1627
+ ],
1628
+ })
1629
+
1630
+ export * as SessionPrompt from "./prompt"