@mty-coder/cli 0.1.1

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 (1169) hide show
  1. package/Dockerfile +18 -0
  2. package/README.md +15 -0
  3. package/bin/mty +180 -0
  4. package/bunfig.toml +7 -0
  5. package/drizzle.config.ts +10 -0
  6. package/git +0 -0
  7. package/migration/20260127222353_familiar_lady_ursula/migration.sql +90 -0
  8. package/migration/20260127222353_familiar_lady_ursula/snapshot.json +796 -0
  9. package/migration/20260211171708_add_project_commands/migration.sql +1 -0
  10. package/migration/20260211171708_add_project_commands/snapshot.json +806 -0
  11. package/migration/20260213144116_wakeful_the_professor/migration.sql +11 -0
  12. package/migration/20260213144116_wakeful_the_professor/snapshot.json +897 -0
  13. package/migration/20260225215848_workspace/migration.sql +7 -0
  14. package/migration/20260225215848_workspace/snapshot.json +959 -0
  15. package/migration/20260227213759_add_session_workspace_id/migration.sql +2 -0
  16. package/migration/20260227213759_add_session_workspace_id/snapshot.json +983 -0
  17. package/migration/20260228203230_blue_harpoon/migration.sql +17 -0
  18. package/migration/20260228203230_blue_harpoon/snapshot.json +1102 -0
  19. package/migration/20260303231226_add_workspace_fields/migration.sql +5 -0
  20. package/migration/20260303231226_add_workspace_fields/snapshot.json +1013 -0
  21. package/migration/20260309230000_move_org_to_state/migration.sql +3 -0
  22. package/migration/20260309230000_move_org_to_state/snapshot.json +1156 -0
  23. package/migration/20260312043431_session_message_cursor/migration.sql +4 -0
  24. package/migration/20260312043431_session_message_cursor/snapshot.json +1168 -0
  25. package/migration/20260323234822_events/migration.sql +13 -0
  26. package/migration/20260323234822_events/snapshot.json +1271 -0
  27. package/migration/20260410174513_workspace-name/migration.sql +16 -0
  28. package/migration/20260410174513_workspace-name/snapshot.json +1271 -0
  29. package/migration/20260413175956_chief_energizer/migration.sql +13 -0
  30. package/migration/20260413175956_chief_energizer/snapshot.json +1399 -0
  31. package/migration/20260422160000_context_inheritance/migration.sql +3 -0
  32. package/migration/20260422170000_task_registry/migration.sql +18 -0
  33. package/migration/20260423145421_remove_session_entry/migration.sql +4 -0
  34. package/migration/20260515000000_actor_rename/migration.sql +7 -0
  35. package/migration/20260515010000_memory_fts/migration.sql +33 -0
  36. package/migration/20260515020000_user_task/migration.sql +29 -0
  37. package/migration/20260519000000_last_checkpoint_message_id/migration.sql +1 -0
  38. package/migration/20260521000000_message_agent_id/migration.sql +2 -0
  39. package/migration/20260521000100_actor_registry_v6/migration.sql +25 -0
  40. package/migration/20260521010000_memory_fts_v6/migration.sql +33 -0
  41. package/migration/20260521020000_memory_fts_triggers/migration.sql +17 -0
  42. package/migration/20260526000000_agent_id_main/migration.sql +14 -0
  43. package/migration/20260527000000_actor_lifecycle/migration.sql +8 -0
  44. package/migration/20260527000100_inbox/migration.sql +12 -0
  45. package/migration/20260529000000_task_todo_redesign/migration.sql +16 -0
  46. package/migration/20260603000000_task_in_progress_owner/migration.sql +1 -0
  47. package/migration/20260603000000_workflow_run/migration.sql +17 -0
  48. package/migration/20260604000000_workflow_script_sha/migration.sql +1 -0
  49. package/migration/20260608000000_claude_import/migration.sql +7 -0
  50. package/migration/20260608010000_claude_import_message_ids/migration.sql +1 -0
  51. package/migration/20260609000000_history_fts/migration.sql +29 -0
  52. package/migration/20260609230000_workflow_agent_timeout/migration.sql +1 -0
  53. package/migration/20260612000000_external_import/migration.sql +16 -0
  54. package/package.json +201 -0
  55. package/parsers-config.ts +290 -0
  56. package/script/build.ts +303 -0
  57. package/script/check-migrations.ts +16 -0
  58. package/script/fix-node-pty.ts +28 -0
  59. package/script/generate.ts +23 -0
  60. package/script/postinstall.mjs +102 -0
  61. package/script/publish.ts +74 -0
  62. package/script/run-workspace-server +106 -0
  63. package/script/schema.ts +63 -0
  64. package/script/time.ts +6 -0
  65. package/script/trace-imports.ts +153 -0
  66. package/script/upgrade-opentui.ts +64 -0
  67. package/src/account/account.sql.ts +39 -0
  68. package/src/account/account.ts +456 -0
  69. package/src/account/repo.ts +166 -0
  70. package/src/account/schema.ts +99 -0
  71. package/src/account/url.ts +8 -0
  72. package/src/acp/README.md +174 -0
  73. package/src/acp/agent.ts +1783 -0
  74. package/src/acp/session.ts +116 -0
  75. package/src/acp/types.ts +24 -0
  76. package/src/actor/actor.sql.ts +38 -0
  77. package/src/actor/events.ts +67 -0
  78. package/src/actor/index.ts +2 -0
  79. package/src/actor/registry.ts +412 -0
  80. package/src/actor/return-header.ts +24 -0
  81. package/src/actor/schema.ts +47 -0
  82. package/src/actor/spawn-ref.ts +16 -0
  83. package/src/actor/spawn.ts +741 -0
  84. package/src/actor/turn.ts +49 -0
  85. package/src/actor/waiter.ts +166 -0
  86. package/src/agent/agent.ts +554 -0
  87. package/src/agent/config.ts +5 -0
  88. package/src/agent/generate.txt +75 -0
  89. package/src/agent/prompt/checkpoint-writer.txt +167 -0
  90. package/src/agent/prompt/compaction.txt +9 -0
  91. package/src/agent/prompt/distill.txt +226 -0
  92. package/src/agent/prompt/dream.txt +155 -0
  93. package/src/agent/prompt/explore.txt +18 -0
  94. package/src/agent/prompt/summary.txt +11 -0
  95. package/src/agent/prompt/title.txt +44 -0
  96. package/src/audio.d.ts +9 -0
  97. package/src/auth/index.ts +97 -0
  98. package/src/bus/bus-event.ts +33 -0
  99. package/src/bus/global.ts +12 -0
  100. package/src/bus/index.ts +193 -0
  101. package/src/cli/bootstrap.ts +33 -0
  102. package/src/cli/cmd/account.ts +258 -0
  103. package/src/cli/cmd/acp.ts +70 -0
  104. package/src/cli/cmd/agent.ts +248 -0
  105. package/src/cli/cmd/cmd.ts +7 -0
  106. package/src/cli/cmd/db.ts +120 -0
  107. package/src/cli/cmd/debug/agent.ts +192 -0
  108. package/src/cli/cmd/debug/config.ts +17 -0
  109. package/src/cli/cmd/debug/file.ts +100 -0
  110. package/src/cli/cmd/debug/index.ts +48 -0
  111. package/src/cli/cmd/debug/lsp.ts +61 -0
  112. package/src/cli/cmd/debug/ripgrep.ts +105 -0
  113. package/src/cli/cmd/debug/scrap.ts +16 -0
  114. package/src/cli/cmd/debug/skill.ts +23 -0
  115. package/src/cli/cmd/debug/snapshot.ts +53 -0
  116. package/src/cli/cmd/export.ts +306 -0
  117. package/src/cli/cmd/generate.ts +50 -0
  118. package/src/cli/cmd/github.ts +1647 -0
  119. package/src/cli/cmd/import.ts +208 -0
  120. package/src/cli/cmd/init.ts +309 -0
  121. package/src/cli/cmd/mcp.ts +812 -0
  122. package/src/cli/cmd/models.ts +88 -0
  123. package/src/cli/cmd/plug.ts +233 -0
  124. package/src/cli/cmd/pr.ts +138 -0
  125. package/src/cli/cmd/providers.ts +692 -0
  126. package/src/cli/cmd/run-completion.ts +77 -0
  127. package/src/cli/cmd/run.ts +694 -0
  128. package/src/cli/cmd/serve.ts +21 -0
  129. package/src/cli/cmd/session.ts +181 -0
  130. package/src/cli/cmd/stats.ts +413 -0
  131. package/src/cli/cmd/tui/app.tsx +1153 -0
  132. package/src/cli/cmd/tui/asset/TEN_VAD_LICENSE +12 -0
  133. package/src/cli/cmd/tui/asset/charge.wav +0 -0
  134. package/src/cli/cmd/tui/asset/pulse-a.wav +0 -0
  135. package/src/cli/cmd/tui/asset/pulse-b.wav +0 -0
  136. package/src/cli/cmd/tui/asset/pulse-c.wav +0 -0
  137. package/src/cli/cmd/tui/asset/ten_vad.wasm +0 -0
  138. package/src/cli/cmd/tui/asset/ten_vad_loader.js +30 -0
  139. package/src/cli/cmd/tui/attach.ts +84 -0
  140. package/src/cli/cmd/tui/component/background-image.tsx +150 -0
  141. package/src/cli/cmd/tui/component/bg-pulse.tsx +130 -0
  142. package/src/cli/cmd/tui/component/border.tsx +21 -0
  143. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  144. package/src/cli/cmd/tui/component/dialog-agreement.tsx +111 -0
  145. package/src/cli/cmd/tui/component/dialog-command.tsx +208 -0
  146. package/src/cli/cmd/tui/component/dialog-console-org.tsx +103 -0
  147. package/src/cli/cmd/tui/component/dialog-go-upsell.tsx +157 -0
  148. package/src/cli/cmd/tui/component/dialog-image-list.tsx +111 -0
  149. package/src/cli/cmd/tui/component/dialog-logo-design.tsx +37 -0
  150. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  151. package/src/cli/cmd/tui/component/dialog-model.tsx +253 -0
  152. package/src/cli/cmd/tui/component/dialog-mty-login.tsx +27 -0
  153. package/src/cli/cmd/tui/component/dialog-onboarding.tsx +97 -0
  154. package/src/cli/cmd/tui/component/dialog-provider.tsx +454 -0
  155. package/src/cli/cmd/tui/component/dialog-session-delete-failed.tsx +101 -0
  156. package/src/cli/cmd/tui/component/dialog-session-list.tsx +269 -0
  157. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  158. package/src/cli/cmd/tui/component/dialog-skill.tsx +42 -0
  159. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  160. package/src/cli/cmd/tui/component/dialog-status.tsx +170 -0
  161. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  162. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  163. package/src/cli/cmd/tui/component/dialog-variant.tsx +39 -0
  164. package/src/cli/cmd/tui/component/dialog-workflows.tsx +62 -0
  165. package/src/cli/cmd/tui/component/dialog-workspace-create.tsx +289 -0
  166. package/src/cli/cmd/tui/component/dialog-workspace-unavailable.tsx +81 -0
  167. package/src/cli/cmd/tui/component/dialog-worktree.tsx +90 -0
  168. package/src/cli/cmd/tui/component/error-component.tsx +92 -0
  169. package/src/cli/cmd/tui/component/logo.tsx +961 -0
  170. package/src/cli/cmd/tui/component/plugin-route-missing.tsx +14 -0
  171. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +684 -0
  172. package/src/cli/cmd/tui/component/prompt/cwd.ts +0 -0
  173. package/src/cli/cmd/tui/component/prompt/frecency.tsx +90 -0
  174. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  175. package/src/cli/cmd/tui/component/prompt/index.tsx +1874 -0
  176. package/src/cli/cmd/tui/component/prompt/part.ts +16 -0
  177. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  178. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  179. package/src/cli/cmd/tui/component/starry-background.tsx +305 -0
  180. package/src/cli/cmd/tui/component/startup-loading.tsx +67 -0
  181. package/src/cli/cmd/tui/component/task-item.tsx +63 -0
  182. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  183. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  184. package/src/cli/cmd/tui/config/cwd.ts +5 -0
  185. package/src/cli/cmd/tui/config/tui-migrate.ts +151 -0
  186. package/src/cli/cmd/tui/config/tui-schema.ts +38 -0
  187. package/src/cli/cmd/tui/config/tui.ts +219 -0
  188. package/src/cli/cmd/tui/context/args.tsx +16 -0
  189. package/src/cli/cmd/tui/context/directory.ts +15 -0
  190. package/src/cli/cmd/tui/context/event.ts +45 -0
  191. package/src/cli/cmd/tui/context/exit.tsx +65 -0
  192. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  193. package/src/cli/cmd/tui/context/keybind.tsx +105 -0
  194. package/src/cli/cmd/tui/context/kv.tsx +86 -0
  195. package/src/cli/cmd/tui/context/language.tsx +91 -0
  196. package/src/cli/cmd/tui/context/local.tsx +455 -0
  197. package/src/cli/cmd/tui/context/plugin-keybinds.ts +41 -0
  198. package/src/cli/cmd/tui/context/project.tsx +109 -0
  199. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  200. package/src/cli/cmd/tui/context/route.tsx +61 -0
  201. package/src/cli/cmd/tui/context/sdk.tsx +150 -0
  202. package/src/cli/cmd/tui/context/sync.tsx +828 -0
  203. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  204. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  205. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  206. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +230 -0
  207. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +230 -0
  208. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  209. package/src/cli/cmd/tui/context/theme/cobalt2.json +225 -0
  210. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  211. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  212. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  213. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  214. package/src/cli/cmd/tui/context/theme/frost.json +245 -0
  215. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  216. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  217. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  218. package/src/cli/cmd/tui/context/theme/lucent-orng.json +234 -0
  219. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  220. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  221. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  222. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  223. package/src/cli/cmd/tui/context/theme/mtycoder.json +245 -0
  224. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  225. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  226. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  227. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  228. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  229. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  230. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  231. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  232. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  233. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  234. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  235. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  236. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  237. package/src/cli/cmd/tui/context/theme.tsx +1300 -0
  238. package/src/cli/cmd/tui/context/thinking.ts +48 -0
  239. package/src/cli/cmd/tui/context/tui-config.tsx +9 -0
  240. package/src/cli/cmd/tui/event.ts +56 -0
  241. package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +93 -0
  242. package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +215 -0
  243. package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +55 -0
  244. package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +114 -0
  245. package/src/cli/cmd/tui/feature-plugins/sidebar/cwd.tsx +45 -0
  246. package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +62 -0
  247. package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +93 -0
  248. package/src/cli/cmd/tui/feature-plugins/sidebar/goal.tsx +84 -0
  249. package/src/cli/cmd/tui/feature-plugins/sidebar/instructions.tsx +54 -0
  250. package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +66 -0
  251. package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +98 -0
  252. package/src/cli/cmd/tui/feature-plugins/sidebar/task.tsx +95 -0
  253. package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +51 -0
  254. package/src/cli/cmd/tui/feature-plugins/sidebar/tps.ts +31 -0
  255. package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +274 -0
  256. package/src/cli/cmd/tui/i18n/en.ts +436 -0
  257. package/src/cli/cmd/tui/i18n/es.ts +478 -0
  258. package/src/cli/cmd/tui/i18n/fr.ts +485 -0
  259. package/src/cli/cmd/tui/i18n/ja.ts +437 -0
  260. package/src/cli/cmd/tui/i18n/locales.ts +82 -0
  261. package/src/cli/cmd/tui/i18n/ru.ts +497 -0
  262. package/src/cli/cmd/tui/i18n/zh.ts +429 -0
  263. package/src/cli/cmd/tui/i18n/zht.ts +405 -0
  264. package/src/cli/cmd/tui/layer.ts +6 -0
  265. package/src/cli/cmd/tui/plugin/api.tsx +402 -0
  266. package/src/cli/cmd/tui/plugin/index.ts +3 -0
  267. package/src/cli/cmd/tui/plugin/internal.ts +35 -0
  268. package/src/cli/cmd/tui/plugin/runtime.ts +1057 -0
  269. package/src/cli/cmd/tui/plugin/slots.tsx +60 -0
  270. package/src/cli/cmd/tui/routes/home.tsx +165 -0
  271. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +76 -0
  272. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +116 -0
  273. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +47 -0
  274. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  275. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  276. package/src/cli/cmd/tui/routes/session/index.tsx +2567 -0
  277. package/src/cli/cmd/tui/routes/session/permission.tsx +691 -0
  278. package/src/cli/cmd/tui/routes/session/question.tsx +488 -0
  279. package/src/cli/cmd/tui/routes/session/sidebar.tsx +97 -0
  280. package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +142 -0
  281. package/src/cli/cmd/tui/thread.ts +323 -0
  282. package/src/cli/cmd/tui/ui/dialog-alert.tsx +61 -0
  283. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +95 -0
  284. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +223 -0
  285. package/src/cli/cmd/tui/ui/dialog-help.tsx +99 -0
  286. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +123 -0
  287. package/src/cli/cmd/tui/ui/dialog-select.tsx +452 -0
  288. package/src/cli/cmd/tui/ui/dialog.tsx +207 -0
  289. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  290. package/src/cli/cmd/tui/ui/spinner.ts +378 -0
  291. package/src/cli/cmd/tui/ui/toast.tsx +102 -0
  292. package/src/cli/cmd/tui/util/clipboard.ts +203 -0
  293. package/src/cli/cmd/tui/util/editor.ts +35 -0
  294. package/src/cli/cmd/tui/util/image-protocol.ts +35 -0
  295. package/src/cli/cmd/tui/util/index.ts +6 -0
  296. package/src/cli/cmd/tui/util/model.ts +23 -0
  297. package/src/cli/cmd/tui/util/provider-origin.ts +7 -0
  298. package/src/cli/cmd/tui/util/revert-diff.ts +18 -0
  299. package/src/cli/cmd/tui/util/scroll.ts +23 -0
  300. package/src/cli/cmd/tui/util/selection.ts +23 -0
  301. package/src/cli/cmd/tui/util/signal.ts +41 -0
  302. package/src/cli/cmd/tui/util/sound.ts +154 -0
  303. package/src/cli/cmd/tui/util/system-locale.ts +209 -0
  304. package/src/cli/cmd/tui/util/terminal.ts +110 -0
  305. package/src/cli/cmd/tui/util/transcript.ts +112 -0
  306. package/src/cli/cmd/tui/util/vad.ts +229 -0
  307. package/src/cli/cmd/tui/util/voice.ts +450 -0
  308. package/src/cli/cmd/tui/win32.ts +130 -0
  309. package/src/cli/cmd/tui/worker.ts +104 -0
  310. package/src/cli/cmd/uninstall.ts +351 -0
  311. package/src/cli/cmd/upgrade.ts +79 -0
  312. package/src/cli/cmd/web.ts +82 -0
  313. package/src/cli/effect/prompt.ts +25 -0
  314. package/src/cli/error.ts +82 -0
  315. package/src/cli/heap.ts +59 -0
  316. package/src/cli/i18n.ts +15 -0
  317. package/src/cli/logo.ts +53 -0
  318. package/src/cli/network.ts +62 -0
  319. package/src/cli/ui.ts +133 -0
  320. package/src/cli/upgrade.ts +41 -0
  321. package/src/command/index.ts +325 -0
  322. package/src/command/template/initialize.txt +66 -0
  323. package/src/command/template/mty-backend.txt +164 -0
  324. package/src/command/template/mty-frontend-design.txt +173 -0
  325. package/src/command/template/mty-frontend.txt +164 -0
  326. package/src/command/template/mty-team.txt +239 -0
  327. package/src/command/template/review.txt +101 -0
  328. package/src/config/agent.ts +197 -0
  329. package/src/config/command.ts +69 -0
  330. package/src/config/config.ts +1036 -0
  331. package/src/config/console-state.ts +16 -0
  332. package/src/config/entry-name.ts +16 -0
  333. package/src/config/error.ts +21 -0
  334. package/src/config/formatter.ts +17 -0
  335. package/src/config/history.ts +21 -0
  336. package/src/config/index.ts +16 -0
  337. package/src/config/keybinds.ts +127 -0
  338. package/src/config/layout.ts +10 -0
  339. package/src/config/lsp.ts +45 -0
  340. package/src/config/managed.ts +70 -0
  341. package/src/config/markdown.ts +97 -0
  342. package/src/config/mcp.ts +172 -0
  343. package/src/config/model-id.ts +14 -0
  344. package/src/config/parse.ts +44 -0
  345. package/src/config/paths.ts +73 -0
  346. package/src/config/permission.ts +76 -0
  347. package/src/config/plugin.ts +88 -0
  348. package/src/config/provider.ts +118 -0
  349. package/src/config/server.ts +20 -0
  350. package/src/config/skills.ts +16 -0
  351. package/src/config/variable.ts +90 -0
  352. package/src/control-plane/adaptors/index.ts +52 -0
  353. package/src/control-plane/adaptors/worktree.ts +47 -0
  354. package/src/control-plane/dev/debug-workspace-plugin.ts +73 -0
  355. package/src/control-plane/schema.ts +19 -0
  356. package/src/control-plane/sse.ts +66 -0
  357. package/src/control-plane/types.ts +34 -0
  358. package/src/control-plane/util.ts +37 -0
  359. package/src/control-plane/workspace-context.ts +26 -0
  360. package/src/control-plane/workspace.sql.ts +17 -0
  361. package/src/control-plane/workspace.ts +615 -0
  362. package/src/effect/app-runtime.ts +146 -0
  363. package/src/effect/bootstrap-runtime.ts +33 -0
  364. package/src/effect/bridge.ts +48 -0
  365. package/src/effect/cross-spawn-spawner.ts +514 -0
  366. package/src/effect/index.ts +5 -0
  367. package/src/effect/instance-ref.ts +11 -0
  368. package/src/effect/instance-registry.ts +12 -0
  369. package/src/effect/instance-state.ts +81 -0
  370. package/src/effect/logger.ts +73 -0
  371. package/src/effect/memo-map.ts +3 -0
  372. package/src/effect/observability.ts +107 -0
  373. package/src/effect/run-service.ts +52 -0
  374. package/src/effect/runner.ts +210 -0
  375. package/src/effect/runtime.ts +19 -0
  376. package/src/env/index.ts +37 -0
  377. package/src/file/ignore.ts +81 -0
  378. package/src/file/index.ts +664 -0
  379. package/src/file/protected.ts +59 -0
  380. package/src/file/ripgrep.ts +485 -0
  381. package/src/file/watcher.ts +163 -0
  382. package/src/flag/flag.ts +167 -0
  383. package/src/format/formatter.ts +403 -0
  384. package/src/format/index.ts +203 -0
  385. package/src/git/index.ts +260 -0
  386. package/src/global/index.ts +54 -0
  387. package/src/history/backfill.ts +162 -0
  388. package/src/history/extract.ts +67 -0
  389. package/src/history/fts-query.ts +15 -0
  390. package/src/history/fts.sql.ts +20 -0
  391. package/src/history/index.ts +10 -0
  392. package/src/history/resolve.ts +65 -0
  393. package/src/history/service.ts +258 -0
  394. package/src/history/writer.ts +112 -0
  395. package/src/id/id.ts +87 -0
  396. package/src/ide/index.ts +73 -0
  397. package/src/inbox/inbox-ref.ts +38 -0
  398. package/src/inbox/inbox.sql.ts +26 -0
  399. package/src/inbox/inbox.ts +223 -0
  400. package/src/inbox/index.ts +3 -0
  401. package/src/inbox/render.ts +40 -0
  402. package/src/index.ts +264 -0
  403. package/src/installation/index.ts +362 -0
  404. package/src/installation/version.ts +8 -0
  405. package/src/lsp/client.ts +249 -0
  406. package/src/lsp/diagnostic.ts +29 -0
  407. package/src/lsp/index.ts +3 -0
  408. package/src/lsp/language.ts +120 -0
  409. package/src/lsp/launch.ts +21 -0
  410. package/src/lsp/lsp.ts +519 -0
  411. package/src/lsp/server.ts +1956 -0
  412. package/src/mcp/auth.ts +144 -0
  413. package/src/mcp/index.ts +944 -0
  414. package/src/mcp/oauth-callback.ts +232 -0
  415. package/src/mcp/oauth-provider.ts +214 -0
  416. package/src/memory/fts-query.ts +37 -0
  417. package/src/memory/fts.sql.ts +19 -0
  418. package/src/memory/index.ts +1 -0
  419. package/src/memory/paths.ts +116 -0
  420. package/src/memory/reconcile.ts +231 -0
  421. package/src/memory/service.ts +152 -0
  422. package/src/metrics/client.ts +40 -0
  423. package/src/metrics/event.ts +43 -0
  424. package/src/metrics/index.ts +5 -0
  425. package/src/metrics/installation.ts +18 -0
  426. package/src/metrics/subscriber.ts +58 -0
  427. package/src/metrics/util.ts +9 -0
  428. package/src/node.ts +6 -0
  429. package/src/npm/config.ts +0 -0
  430. package/src/npm/index.ts +293 -0
  431. package/src/npmcli-config.d.ts +43 -0
  432. package/src/patch/index.ts +680 -0
  433. package/src/permission/arity.ts +163 -0
  434. package/src/permission/evaluate.ts +15 -0
  435. package/src/permission/index.ts +378 -0
  436. package/src/permission/schema.ts +17 -0
  437. package/src/plugin/checkpoint-splitover.ts +60 -0
  438. package/src/plugin/cloudflare.ts +76 -0
  439. package/src/plugin/codex.ts +607 -0
  440. package/src/plugin/github-copilot/copilot.ts +368 -0
  441. package/src/plugin/github-copilot/models.ts +153 -0
  442. package/src/plugin/index.ts +584 -0
  443. package/src/plugin/install.ts +439 -0
  444. package/src/plugin/loader.ts +216 -0
  445. package/src/plugin/matcher.ts +33 -0
  446. package/src/plugin/meta.ts +188 -0
  447. package/src/plugin/mty.ts +9 -0
  448. package/src/plugin/shared.ts +323 -0
  449. package/src/plugin/subagent-progress-checker.ts +147 -0
  450. package/src/project/bootstrap.ts +59 -0
  451. package/src/project/index.ts +2 -0
  452. package/src/project/instance.ts +190 -0
  453. package/src/project/project-id.ts +48 -0
  454. package/src/project/project.sql.ts +16 -0
  455. package/src/project/project.ts +522 -0
  456. package/src/project/schema.ts +15 -0
  457. package/src/project/vcs.ts +227 -0
  458. package/src/project/workspace-trust.ts +67 -0
  459. package/src/provider/auth.ts +234 -0
  460. package/src/provider/error.ts +216 -0
  461. package/src/provider/index.ts +5 -0
  462. package/src/provider/models.ts +180 -0
  463. package/src/provider/provider.ts +1787 -0
  464. package/src/provider/schema.ts +36 -0
  465. package/src/provider/sdk/copilot/README.md +5 -0
  466. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +170 -0
  467. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  468. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +19 -0
  469. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  470. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +815 -0
  471. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  472. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  473. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +83 -0
  474. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  475. package/src/provider/sdk/copilot/index.ts +2 -0
  476. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  477. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +335 -0
  478. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  479. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  480. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  481. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +214 -0
  482. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1770 -0
  483. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +173 -0
  484. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  485. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +87 -0
  486. package/src/provider/sdk/copilot/responses/tool/file-search.ts +127 -0
  487. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +114 -0
  488. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +64 -0
  489. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +103 -0
  490. package/src/provider/sdk/copilot/responses/tool/web-search.ts +102 -0
  491. package/src/provider/transform.ts +1322 -0
  492. package/src/pty/index.ts +364 -0
  493. package/src/pty/pty.bun.ts +26 -0
  494. package/src/pty/pty.node.ts +27 -0
  495. package/src/pty/pty.ts +25 -0
  496. package/src/pty/schema.ts +17 -0
  497. package/src/question/index.ts +252 -0
  498. package/src/question/schema.ts +17 -0
  499. package/src/server/adapter.bun.ts +40 -0
  500. package/src/server/adapter.node.ts +66 -0
  501. package/src/server/adapter.ts +21 -0
  502. package/src/server/auth.ts +16 -0
  503. package/src/server/error.ts +53 -0
  504. package/src/server/event.ts +7 -0
  505. package/src/server/fence.ts +81 -0
  506. package/src/server/mdns.ts +60 -0
  507. package/src/server/middleware.ts +96 -0
  508. package/src/server/projectors.ts +28 -0
  509. package/src/server/proxy.ts +171 -0
  510. package/src/server/pty-ticket.ts +42 -0
  511. package/src/server/routes/control/index.ts +160 -0
  512. package/src/server/routes/control/workspace.ts +203 -0
  513. package/src/server/routes/global.ts +367 -0
  514. package/src/server/routes/instance/bash-interactive.ts +82 -0
  515. package/src/server/routes/instance/config.ts +89 -0
  516. package/src/server/routes/instance/event.ts +108 -0
  517. package/src/server/routes/instance/experimental.ts +408 -0
  518. package/src/server/routes/instance/file.ts +190 -0
  519. package/src/server/routes/instance/httpapi/config.ts +51 -0
  520. package/src/server/routes/instance/httpapi/permission.ts +72 -0
  521. package/src/server/routes/instance/httpapi/project.ts +62 -0
  522. package/src/server/routes/instance/httpapi/provider.ts +142 -0
  523. package/src/server/routes/instance/httpapi/question.ts +121 -0
  524. package/src/server/routes/instance/httpapi/server.ts +136 -0
  525. package/src/server/routes/instance/index.ts +301 -0
  526. package/src/server/routes/instance/mcp.ts +260 -0
  527. package/src/server/routes/instance/middleware.ts +35 -0
  528. package/src/server/routes/instance/permission.ts +73 -0
  529. package/src/server/routes/instance/project.ts +122 -0
  530. package/src/server/routes/instance/provider.ts +158 -0
  531. package/src/server/routes/instance/pty.ts +302 -0
  532. package/src/server/routes/instance/question.ts +162 -0
  533. package/src/server/routes/instance/session.ts +1296 -0
  534. package/src/server/routes/instance/sync.ts +143 -0
  535. package/src/server/routes/instance/trace.ts +59 -0
  536. package/src/server/routes/instance/tui.ts +384 -0
  537. package/src/server/routes/instance/workflows.ts +72 -0
  538. package/src/server/routes/ui.ts +37 -0
  539. package/src/server/server.ts +136 -0
  540. package/src/server/workspace.ts +122 -0
  541. package/src/session/auto-dream.ts +204 -0
  542. package/src/session/boundary.ts +77 -0
  543. package/src/session/budgeted-read.ts +118 -0
  544. package/src/session/checkpoint-align.ts +29 -0
  545. package/src/session/checkpoint-context.ts +36 -0
  546. package/src/session/checkpoint-paths.ts +86 -0
  547. package/src/session/checkpoint-progress-reconcile.ts +111 -0
  548. package/src/session/checkpoint-retry.ts +192 -0
  549. package/src/session/checkpoint-templates.ts +114 -0
  550. package/src/session/checkpoint-validator.ts +259 -0
  551. package/src/session/checkpoint.ts +1478 -0
  552. package/src/session/classify.ts +92 -0
  553. package/src/session/claude-import.ts +381 -0
  554. package/src/session/codex-import.ts +416 -0
  555. package/src/session/compaction.ts +543 -0
  556. package/src/session/external-import.sql.ts +18 -0
  557. package/src/session/external-import.ts +136 -0
  558. package/src/session/goal.ts +232 -0
  559. package/src/session/index.ts +1 -0
  560. package/src/session/instruction.ts +276 -0
  561. package/src/session/last-message-info.ts +32 -0
  562. package/src/session/llm-request-prefix.ts +82 -0
  563. package/src/session/llm.ts +735 -0
  564. package/src/session/max-mode.ts +397 -0
  565. package/src/session/message-v2.ts +1136 -0
  566. package/src/session/message.ts +191 -0
  567. package/src/session/opencode-import.ts +281 -0
  568. package/src/session/overflow.ts +62 -0
  569. package/src/session/prefix-capture-ref.ts +48 -0
  570. package/src/session/processor.ts +985 -0
  571. package/src/session/projectors.ts +137 -0
  572. package/src/session/prompt/anthropic.txt +154 -0
  573. package/src/session/prompt/beast.txt +155 -0
  574. package/src/session/prompt/build-switch.txt +5 -0
  575. package/src/session/prompt/codex.txt +79 -0
  576. package/src/session/prompt/compose.txt +115 -0
  577. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  578. package/src/session/prompt/default.txt +151 -0
  579. package/src/session/prompt/gemini.txt +155 -0
  580. package/src/session/prompt/gpt.txt +107 -0
  581. package/src/session/prompt/kimi.txt +95 -0
  582. package/src/session/prompt/max-steps.txt +16 -0
  583. package/src/session/prompt/text-loop-recovery.ts +40 -0
  584. package/src/session/prompt/trinity.txt +97 -0
  585. package/src/session/prompt-utils.ts +58 -0
  586. package/src/session/prompt.ts +3482 -0
  587. package/src/session/prune.ts +481 -0
  588. package/src/session/retry.ts +166 -0
  589. package/src/session/revert.ts +161 -0
  590. package/src/session/run-state.ts +135 -0
  591. package/src/session/schema.ts +36 -0
  592. package/src/session/session.sql.ts +110 -0
  593. package/src/session/session.ts +908 -0
  594. package/src/session/status.ts +89 -0
  595. package/src/session/summary.ts +163 -0
  596. package/src/session/system.ts +86 -0
  597. package/src/session/todo.ts +77 -0
  598. package/src/share/index.ts +2 -0
  599. package/src/share/session.ts +57 -0
  600. package/src/share/share-next.ts +381 -0
  601. package/src/share/share.sql.ts +13 -0
  602. package/src/shell/shell.ts +110 -0
  603. package/src/skill/compose/.bundle/ask/SKILL.md +58 -0
  604. package/src/skill/compose/.bundle/brainstorm/SKILL.md +220 -0
  605. package/src/skill/compose/.bundle/brainstorm/scripts/frame-template.html +214 -0
  606. package/src/skill/compose/.bundle/brainstorm/scripts/helper.js +88 -0
  607. package/src/skill/compose/.bundle/brainstorm/scripts/server.cjs +354 -0
  608. package/src/skill/compose/.bundle/brainstorm/scripts/start-server.sh +148 -0
  609. package/src/skill/compose/.bundle/brainstorm/scripts/stop-server.sh +56 -0
  610. package/src/skill/compose/.bundle/brainstorm/spec-document-reviewer-prompt.md +50 -0
  611. package/src/skill/compose/.bundle/brainstorm/visual-companion.md +287 -0
  612. package/src/skill/compose/.bundle/code-review/SKILL.md +68 -0
  613. package/src/skill/compose/.bundle/debug/CREATION-LOG.md +119 -0
  614. package/src/skill/compose/.bundle/debug/SKILL.md +297 -0
  615. package/src/skill/compose/.bundle/debug/condition-based-waiting-example.ts +158 -0
  616. package/src/skill/compose/.bundle/debug/condition-based-waiting.md +115 -0
  617. package/src/skill/compose/.bundle/debug/defense-in-depth.md +122 -0
  618. package/src/skill/compose/.bundle/debug/find-polluter.sh +63 -0
  619. package/src/skill/compose/.bundle/debug/root-cause-tracing.md +169 -0
  620. package/src/skill/compose/.bundle/debug/test-academic.md +14 -0
  621. package/src/skill/compose/.bundle/debug/test-pressure-1.md +58 -0
  622. package/src/skill/compose/.bundle/debug/test-pressure-2.md +68 -0
  623. package/src/skill/compose/.bundle/debug/test-pressure-3.md +69 -0
  624. package/src/skill/compose/.bundle/execute/SKILL.md +71 -0
  625. package/src/skill/compose/.bundle/feedback/SKILL.md +214 -0
  626. package/src/skill/compose/.bundle/frontend-design/LICENSE.txt +177 -0
  627. package/src/skill/compose/.bundle/frontend-design/SKILL.md +42 -0
  628. package/src/skill/compose/.bundle/mcp-builder/LICENSE.txt +202 -0
  629. package/src/skill/compose/.bundle/mcp-builder/SKILL.md +236 -0
  630. package/src/skill/compose/.bundle/mcp-builder/reference/evaluation.md +602 -0
  631. package/src/skill/compose/.bundle/mcp-builder/reference/mcp_best_practices.md +249 -0
  632. package/src/skill/compose/.bundle/mcp-builder/reference/node_mcp_server.md +970 -0
  633. package/src/skill/compose/.bundle/mcp-builder/reference/python_mcp_server.md +719 -0
  634. package/src/skill/compose/.bundle/mcp-builder/scripts/connections.py +151 -0
  635. package/src/skill/compose/.bundle/mcp-builder/scripts/evaluation.py +373 -0
  636. package/src/skill/compose/.bundle/mcp-builder/scripts/example_evaluation.xml +22 -0
  637. package/src/skill/compose/.bundle/mcp-builder/scripts/requirements.txt +2 -0
  638. package/src/skill/compose/.bundle/merge/SKILL.md +252 -0
  639. package/src/skill/compose/.bundle/new-skill/SKILL.md +656 -0
  640. package/src/skill/compose/.bundle/new-skill/anthropic-best-practices.md +1150 -0
  641. package/src/skill/compose/.bundle/new-skill/examples/CLAUDE_MD_TESTING.md +189 -0
  642. package/src/skill/compose/.bundle/new-skill/graphviz-conventions.dot +172 -0
  643. package/src/skill/compose/.bundle/new-skill/persuasion-principles.md +187 -0
  644. package/src/skill/compose/.bundle/new-skill/render-graphs.js +168 -0
  645. package/src/skill/compose/.bundle/new-skill/testing-skills-with-subagents.md +384 -0
  646. package/src/skill/compose/.bundle/parallel/SKILL.md +182 -0
  647. package/src/skill/compose/.bundle/plan/SKILL.md +161 -0
  648. package/src/skill/compose/.bundle/plan/plan-document-reviewer-prompt.md +50 -0
  649. package/src/skill/compose/.bundle/report/SKILL.md +180 -0
  650. package/src/skill/compose/.bundle/review/SKILL.md +104 -0
  651. package/src/skill/compose/.bundle/review/code-reviewer.md +171 -0
  652. package/src/skill/compose/.bundle/search-first/SKILL.md +161 -0
  653. package/src/skill/compose/.bundle/security-review/SKILL.md +495 -0
  654. package/src/skill/compose/.bundle/security-review/cloud-infrastructure-security.md +361 -0
  655. package/src/skill/compose/.bundle/self-extend/SKILL.md +131 -0
  656. package/src/skill/compose/.bundle/self-extend/reference/hook-api.md +242 -0
  657. package/src/skill/compose/.bundle/self-extend/reference/skill-api.md +114 -0
  658. package/src/skill/compose/.bundle/self-extend/reference/tool-api.md +115 -0
  659. package/src/skill/compose/.bundle/self-extend/reference/tui-api.md +258 -0
  660. package/src/skill/compose/.bundle/skill-status/SKILL.md +37 -0
  661. package/src/skill/compose/.bundle/strategic-compact/SKILL.md +103 -0
  662. package/src/skill/compose/.bundle/strategic-compact/suggest-compact.sh +54 -0
  663. package/src/skill/compose/.bundle/subagent/SKILL.md +344 -0
  664. package/src/skill/compose/.bundle/subagent/code-quality-reviewer-prompt.md +24 -0
  665. package/src/skill/compose/.bundle/subagent/implementer-prompt.md +126 -0
  666. package/src/skill/compose/.bundle/subagent/spec-reviewer-prompt.md +112 -0
  667. package/src/skill/compose/.bundle/tdd/SKILL.md +372 -0
  668. package/src/skill/compose/.bundle/tdd/testing-anti-patterns.md +299 -0
  669. package/src/skill/compose/.bundle/verification-loop/SKILL.md +126 -0
  670. package/src/skill/compose/.bundle/verify/SKILL.md +140 -0
  671. package/src/skill/compose/.bundle/worktree/SKILL.md +234 -0
  672. package/src/skill/compose/LICENSE-karpathy +28 -0
  673. package/src/skill/compose/LICENSE-superpowers +26 -0
  674. package/src/skill/compose/bundle.macro.ts +30 -0
  675. package/src/skill/compose/extract.ts +85 -0
  676. package/src/skill/discovery.ts +116 -0
  677. package/src/skill/index.ts +317 -0
  678. package/src/snapshot/index.ts +777 -0
  679. package/src/sql.d.ts +4 -0
  680. package/src/storage/db.bun.ts +8 -0
  681. package/src/storage/db.node.ts +8 -0
  682. package/src/storage/db.ts +172 -0
  683. package/src/storage/index.ts +26 -0
  684. package/src/storage/json-migration.ts +426 -0
  685. package/src/storage/read-sqlite.bun.ts +11 -0
  686. package/src/storage/read-sqlite.node.ts +13 -0
  687. package/src/storage/read-sqlite.ts +10 -0
  688. package/src/storage/schema.sql.ts +10 -0
  689. package/src/storage/schema.ts +7 -0
  690. package/src/storage/storage.ts +331 -0
  691. package/src/sync/README.md +179 -0
  692. package/src/sync/event.sql.ts +16 -0
  693. package/src/sync/index.ts +278 -0
  694. package/src/sync/schema.ts +14 -0
  695. package/src/task/events.ts +28 -0
  696. package/src/task/gate-state.ts +54 -0
  697. package/src/task/gate.ts +116 -0
  698. package/src/task/index.ts +1 -0
  699. package/src/task/registry.ts +394 -0
  700. package/src/task/schema.ts +43 -0
  701. package/src/task/task.sql.ts +50 -0
  702. package/src/team/events.ts +22 -0
  703. package/src/team/index.ts +113 -0
  704. package/src/team/schema.ts +31 -0
  705. package/src/temporary.ts +33 -0
  706. package/src/tool/actor.shell.txt +72 -0
  707. package/src/tool/actor.ts +808 -0
  708. package/src/tool/actor.txt +103 -0
  709. package/src/tool/apply_patch.ts +308 -0
  710. package/src/tool/apply_patch.txt +33 -0
  711. package/src/tool/bash-interactive.ts +183 -0
  712. package/src/tool/bash.ts +696 -0
  713. package/src/tool/bash.txt +123 -0
  714. package/src/tool/change-directory.ts +91 -0
  715. package/src/tool/codesearch.ts +63 -0
  716. package/src/tool/codesearch.txt +12 -0
  717. package/src/tool/edit.ts +702 -0
  718. package/src/tool/edit.txt +10 -0
  719. package/src/tool/external-directory.ts +132 -0
  720. package/src/tool/glob.ts +100 -0
  721. package/src/tool/glob.txt +6 -0
  722. package/src/tool/grep.ts +145 -0
  723. package/src/tool/grep.txt +8 -0
  724. package/src/tool/history.ts +146 -0
  725. package/src/tool/history.txt +17 -0
  726. package/src/tool/index.ts +4 -0
  727. package/src/tool/invalid.ts +20 -0
  728. package/src/tool/invocation-style.ts +17 -0
  729. package/src/tool/lsp.ts +91 -0
  730. package/src/tool/lsp.txt +19 -0
  731. package/src/tool/mcp-exa.ts +78 -0
  732. package/src/tool/memory-path-guard.ts +162 -0
  733. package/src/tool/memory.ts +81 -0
  734. package/src/tool/memory.txt +69 -0
  735. package/src/tool/multiedit.ts +61 -0
  736. package/src/tool/multiedit.txt +41 -0
  737. package/src/tool/plan-enter.txt +14 -0
  738. package/src/tool/plan-exit.txt +13 -0
  739. package/src/tool/plan.ts +90 -0
  740. package/src/tool/question.ts +67 -0
  741. package/src/tool/question.txt +10 -0
  742. package/src/tool/read.ts +327 -0
  743. package/src/tool/read.txt +14 -0
  744. package/src/tool/recoverable.ts +35 -0
  745. package/src/tool/registry.ts +423 -0
  746. package/src/tool/schema.ts +17 -0
  747. package/src/tool/session-cwd.ts +35 -0
  748. package/src/tool/shell-tokenize.ts +346 -0
  749. package/src/tool/shell-wrap.ts +190 -0
  750. package/src/tool/skill.ts +76 -0
  751. package/src/tool/skill.txt +5 -0
  752. package/src/tool/task.shell.txt +57 -0
  753. package/src/tool/task.ts +456 -0
  754. package/src/tool/task.txt +56 -0
  755. package/src/tool/tool.ts +166 -0
  756. package/src/tool/truncate.ts +201 -0
  757. package/src/tool/truncation-dir.ts +4 -0
  758. package/src/tool/webfetch.ts +199 -0
  759. package/src/tool/webfetch.txt +13 -0
  760. package/src/tool/websearch/index.ts +104 -0
  761. package/src/tool/websearch/mty.ts +14 -0
  762. package/src/tool/websearch/websearch.txt +14 -0
  763. package/src/tool/workflow.ts +164 -0
  764. package/src/tool/workflow.txt +25 -0
  765. package/src/tool/write.ts +97 -0
  766. package/src/tool/write.txt +9 -0
  767. package/src/util/abort.ts +35 -0
  768. package/src/util/archive.ts +15 -0
  769. package/src/util/color.ts +17 -0
  770. package/src/util/data-url.ts +9 -0
  771. package/src/util/defer.ts +10 -0
  772. package/src/util/effect-http-client.ts +11 -0
  773. package/src/util/effect-zod.ts +367 -0
  774. package/src/util/error.ts +78 -0
  775. package/src/util/filesystem.ts +243 -0
  776. package/src/util/fn.ts +21 -0
  777. package/src/util/format.ts +20 -0
  778. package/src/util/iife.ts +3 -0
  779. package/src/util/index.ts +12 -0
  780. package/src/util/keybind.ts +101 -0
  781. package/src/util/lazy.ts +18 -0
  782. package/src/util/local-context.ts +23 -0
  783. package/src/util/locale.ts +79 -0
  784. package/src/util/lock.ts +96 -0
  785. package/src/util/log.ts +228 -0
  786. package/src/util/media.ts +26 -0
  787. package/src/util/mty-process.ts +24 -0
  788. package/src/util/network.ts +9 -0
  789. package/src/util/process.ts +174 -0
  790. package/src/util/queue.ts +60 -0
  791. package/src/util/record.ts +3 -0
  792. package/src/util/rpc.ts +64 -0
  793. package/src/util/schema.ts +53 -0
  794. package/src/util/scrap.ts +10 -0
  795. package/src/util/signal.ts +12 -0
  796. package/src/util/timeout.ts +14 -0
  797. package/src/util/token.ts +5 -0
  798. package/src/util/update-schema.ts +13 -0
  799. package/src/util/which.ts +14 -0
  800. package/src/util/wildcard.ts +57 -0
  801. package/src/workflow/builtin/deep-research.js +391 -0
  802. package/src/workflow/builtin.ts +54 -0
  803. package/src/workflow/events.ts +72 -0
  804. package/src/workflow/meta.ts +335 -0
  805. package/src/workflow/persistence.ts +312 -0
  806. package/src/workflow/resolve.ts +45 -0
  807. package/src/workflow/runtime-ref.ts +18 -0
  808. package/src/workflow/runtime.ts +1234 -0
  809. package/src/workflow/sandbox.ts +286 -0
  810. package/src/workflow/workflow.sql.ts +31 -0
  811. package/src/workflow/workspace.ts +69 -0
  812. package/src/worktree/index.ts +614 -0
  813. package/sst-env.d.ts +10 -0
  814. package/test/AGENTS.md +133 -0
  815. package/test/account/repo.test.ts +352 -0
  816. package/test/account/service.test.ts +456 -0
  817. package/test/acp/agent-interface.test.ts +51 -0
  818. package/test/acp/event-subscription.test.ts +725 -0
  819. package/test/actor/cancel-cascade.test.ts +432 -0
  820. package/test/actor/no-completion-listener.test.ts +41 -0
  821. package/test/actor/poststop-progress-write-permission.repro.test.ts +414 -0
  822. package/test/actor/registry-render.test.ts +113 -0
  823. package/test/actor/registry-status.test.ts +111 -0
  824. package/test/actor/registry.test.ts +619 -0
  825. package/test/actor/return-header.test.ts +40 -0
  826. package/test/actor/spawn-lifecycle.test.ts +346 -0
  827. package/test/actor/spawn-no-deadlock.test.ts +340 -0
  828. package/test/actor/spawn-notification.test.ts +393 -0
  829. package/test/actor/spawn-task-autostart.test.ts +530 -0
  830. package/test/actor/spawn.test.ts +1072 -0
  831. package/test/actor/status-event-payload.test.ts +132 -0
  832. package/test/actor/terminology.test.ts +39 -0
  833. package/test/actor/turn.test.ts +125 -0
  834. package/test/actor/waiter.test.ts +246 -0
  835. package/test/agent/agent.test.ts +874 -0
  836. package/test/agent/allowlist.test.ts +45 -0
  837. package/test/auth/auth.test.ts +86 -0
  838. package/test/bus/bus-effect.test.ts +162 -0
  839. package/test/bus/bus-integration.test.ts +87 -0
  840. package/test/bus/bus.test.ts +219 -0
  841. package/test/cli/account.test.ts +26 -0
  842. package/test/cli/cmd/tui/prompt-part.test.ts +47 -0
  843. package/test/cli/error.test.ts +18 -0
  844. package/test/cli/github-action.test.ts +198 -0
  845. package/test/cli/github-remote.test.ts +80 -0
  846. package/test/cli/import.test.ts +54 -0
  847. package/test/cli/plugin-auth-picker.test.ts +120 -0
  848. package/test/cli/run-completion.test.ts +131 -0
  849. package/test/cli/tui/keybind-plugin.test.ts +90 -0
  850. package/test/cli/tui/plugin-add.test.ts +111 -0
  851. package/test/cli/tui/plugin-install.test.ts +87 -0
  852. package/test/cli/tui/plugin-lifecycle.test.ts +224 -0
  853. package/test/cli/tui/plugin-loader-entrypoint.test.ts +484 -0
  854. package/test/cli/tui/plugin-loader-pure.test.ts +71 -0
  855. package/test/cli/tui/plugin-loader.test.ts +816 -0
  856. package/test/cli/tui/plugin-toggle.test.ts +157 -0
  857. package/test/cli/tui/revert-diff.test.ts +35 -0
  858. package/test/cli/tui/route-agent-id.test.ts +26 -0
  859. package/test/cli/tui/sidebar-tps.test.ts +63 -0
  860. package/test/cli/tui/slot-replace.test.tsx +47 -0
  861. package/test/cli/tui/sync-bucket.test.ts +29 -0
  862. package/test/cli/tui/theme-store.test.ts +51 -0
  863. package/test/cli/tui/thread.test.ts +122 -0
  864. package/test/cli/tui/transcript.test.ts +426 -0
  865. package/test/cli/tui/use-event.test.tsx +175 -0
  866. package/test/cli/tui/voice.test.ts +443 -0
  867. package/test/command/deep-research-command.test.ts +16 -0
  868. package/test/config/agent-color.test.ts +77 -0
  869. package/test/config/checkpoint-fork.test.ts +21 -0
  870. package/test/config/config.test.ts +2577 -0
  871. package/test/config/fixtures/empty-frontmatter.md +4 -0
  872. package/test/config/fixtures/frontmatter.md +28 -0
  873. package/test/config/fixtures/markdown-header.md +11 -0
  874. package/test/config/fixtures/no-frontmatter.md +1 -0
  875. package/test/config/fixtures/weird-model-id.md +13 -0
  876. package/test/config/lsp.test.ts +87 -0
  877. package/test/config/markdown.test.ts +228 -0
  878. package/test/config/plugin.test.ts +0 -0
  879. package/test/config/tui.test.ts +627 -0
  880. package/test/control-plane/adaptors.test.ts +71 -0
  881. package/test/control-plane/sse.test.ts +56 -0
  882. package/test/effect/app-runtime-logger.test.ts +92 -0
  883. package/test/effect/cross-spawn-spawner.test.ts +411 -0
  884. package/test/effect/instance-state.test.ts +482 -0
  885. package/test/effect/observability.test.ts +46 -0
  886. package/test/effect/run-service.test.ts +46 -0
  887. package/test/effect/runner-warn-log.test.ts +111 -0
  888. package/test/effect/runner.test.ts +494 -0
  889. package/test/fake/provider.ts +90 -0
  890. package/test/file/fsmonitor.test.ts +68 -0
  891. package/test/file/ignore.test.ts +10 -0
  892. package/test/file/index.test.ts +956 -0
  893. package/test/file/path-traversal.test.ts +204 -0
  894. package/test/file/ripgrep.test.ts +214 -0
  895. package/test/file/watcher.test.ts +249 -0
  896. package/test/filesystem/filesystem.test.ts +319 -0
  897. package/test/fixture/db.ts +11 -0
  898. package/test/fixture/fixture.test.ts +58 -0
  899. package/test/fixture/fixture.ts +190 -0
  900. package/test/fixture/flock-worker.ts +72 -0
  901. package/test/fixture/lsp/fake-lsp-server.js +75 -0
  902. package/test/fixture/plug-worker.ts +93 -0
  903. package/test/fixture/plugin-meta-worker.ts +19 -0
  904. package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
  905. package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
  906. package/test/fixture/skills/cloudflare/SKILL.md +211 -0
  907. package/test/fixture/skills/index.json +6 -0
  908. package/test/fixture/tui-plugin.ts +328 -0
  909. package/test/fixture/tui-runtime.ts +31 -0
  910. package/test/format/format.test.ts +244 -0
  911. package/test/git/git.test.ts +128 -0
  912. package/test/global/fixture/global-paths-worker.ts +17 -0
  913. package/test/global/mtycoder-home.test.ts +143 -0
  914. package/test/history/backfill.test.ts +149 -0
  915. package/test/history/extract.test.ts +106 -0
  916. package/test/history/fts-query.test.ts +30 -0
  917. package/test/history/resolve.test.ts +130 -0
  918. package/test/history/service.test.ts +210 -0
  919. package/test/history/writer.test.ts +163 -0
  920. package/test/ide/ide.test.ts +82 -0
  921. package/test/inbox/drain-in-loop.test.ts +230 -0
  922. package/test/inbox/fork-agent-compat.test.ts +387 -0
  923. package/test/inbox/gc-on-init.test.ts +167 -0
  924. package/test/inbox/send-no-block.test.ts +120 -0
  925. package/test/inbox/sender-cancel-independence.test.ts +160 -0
  926. package/test/inbox/wake-matrix.test.ts +141 -0
  927. package/test/installation/installation.test.ts +255 -0
  928. package/test/keybind.test.ts +421 -0
  929. package/test/lib/effect.ts +53 -0
  930. package/test/lib/filesystem.ts +10 -0
  931. package/test/lib/llm-server.ts +770 -0
  932. package/test/lib/mock-llm.ts +159 -0
  933. package/test/lib/scripted-llm-server.ts +245 -0
  934. package/test/lsp/client.test.ts +98 -0
  935. package/test/lsp/index.test.ts +109 -0
  936. package/test/lsp/launch.test.ts +22 -0
  937. package/test/lsp/lifecycle.test.ts +184 -0
  938. package/test/mcp/headers.test.ts +178 -0
  939. package/test/mcp/lifecycle.test.ts +824 -0
  940. package/test/mcp/oauth-auto-connect.test.ts +281 -0
  941. package/test/mcp/oauth-browser.test.ts +268 -0
  942. package/test/mcp/oauth-callback.test.ts +34 -0
  943. package/test/memory/abort-leak-webfetch.ts +49 -0
  944. package/test/memory/abort-leak.test.ts +127 -0
  945. package/test/memory/cc-frontmatter.test.ts +85 -0
  946. package/test/memory/cc-paths.test.ts +60 -0
  947. package/test/memory/cc-reconcile.test.ts +239 -0
  948. package/test/memory/cc-search.test.ts +151 -0
  949. package/test/memory/fts-query.test.ts +48 -0
  950. package/test/memory/fts-rowid-stability.test.ts +271 -0
  951. package/test/memory/paths.test.ts +210 -0
  952. package/test/memory/reconcile.test.ts +115 -0
  953. package/test/memory/service.test.ts +169 -0
  954. package/test/npm.test.ts +18 -0
  955. package/test/patch/patch.test.ts +348 -0
  956. package/test/permission/abort.test.ts +116 -0
  957. package/test/permission/arity.test.ts +33 -0
  958. package/test/permission/disabled.test.ts +51 -0
  959. package/test/permission/next.test.ts +1080 -0
  960. package/test/permission/non-interactive.test.ts +55 -0
  961. package/test/permission-task.test.ts +326 -0
  962. package/test/plugin/actor-hooks.test.ts +1471 -0
  963. package/test/plugin/auth-override.test.ts +79 -0
  964. package/test/plugin/checkpoint-splitover.test.ts +434 -0
  965. package/test/plugin/cloudflare.test.ts +68 -0
  966. package/test/plugin/codex.test.ts +123 -0
  967. package/test/plugin/github-copilot-models.test.ts +163 -0
  968. package/test/plugin/install-concurrency.test.ts +140 -0
  969. package/test/plugin/install.test.ts +570 -0
  970. package/test/plugin/loader-shared.test.ts +1169 -0
  971. package/test/plugin/matcher.test.ts +97 -0
  972. package/test/plugin/meta.test.ts +137 -0
  973. package/test/plugin/shared.test.ts +88 -0
  974. package/test/plugin/subagent-progress-checker.test.ts +227 -0
  975. package/test/plugin/trigger.test.ts +116 -0
  976. package/test/plugin/workspace-adaptor.test.ts +109 -0
  977. package/test/preload.ts +102 -0
  978. package/test/project/migrate-global.test.ts +150 -0
  979. package/test/project/project-id.test.ts +64 -0
  980. package/test/project/project.test.ts +502 -0
  981. package/test/project/vcs.test.ts +286 -0
  982. package/test/project/worktree-remove.test.ts +126 -0
  983. package/test/project/worktree.test.ts +214 -0
  984. package/test/provider/amazon-bedrock.test.ts +462 -0
  985. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  986. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  987. package/test/provider/error.test.ts +160 -0
  988. package/test/provider/gitlab-duo.test.ts +413 -0
  989. package/test/provider/model-groups.test.ts +389 -0
  990. package/test/provider/provider-chunk-timeout.test.ts +23 -0
  991. package/test/provider/provider.test.ts +2648 -0
  992. package/test/provider/transform.test.ts +3379 -0
  993. package/test/pty/pty-output-isolation.test.ts +146 -0
  994. package/test/pty/pty-session.test.ts +102 -0
  995. package/test/pty/pty-shell.test.ts +69 -0
  996. package/test/question/question.test.ts +464 -0
  997. package/test/server/global-session-list.test.ts +105 -0
  998. package/test/server/project-init-git.test.ts +122 -0
  999. package/test/server/session-actions.test.ts +49 -0
  1000. package/test/server/session-list.test.ts +110 -0
  1001. package/test/server/session-messages.test.ts +220 -0
  1002. package/test/server/session-prompt-busy.test.ts +146 -0
  1003. package/test/server/session-select.test.ts +100 -0
  1004. package/test/server/session-task-route.test.ts +165 -0
  1005. package/test/server/summarize-route-main-slice.test.ts +99 -0
  1006. package/test/server/trace-attributes.test.ts +76 -0
  1007. package/test/server/workflows-route.test.ts +279 -0
  1008. package/test/session/bootstrap-skip-system.test.ts +121 -0
  1009. package/test/session/boundary.test.ts +33 -0
  1010. package/test/session/budgeted-read.test.ts +74 -0
  1011. package/test/session/checkpoint-align.test.ts +58 -0
  1012. package/test/session/checkpoint-boundary.test.ts +186 -0
  1013. package/test/session/checkpoint-child-session.test.ts +508 -0
  1014. package/test/session/checkpoint-context.test.ts +141 -0
  1015. package/test/session/checkpoint-drain.test.ts +188 -0
  1016. package/test/session/checkpoint-extract-titles.test.ts +58 -0
  1017. package/test/session/checkpoint-fork-mode.test.ts +576 -0
  1018. package/test/session/checkpoint-main-slice.test.ts +259 -0
  1019. package/test/session/checkpoint-paths.test.ts +78 -0
  1020. package/test/session/checkpoint-permission.test.ts +136 -0
  1021. package/test/session/checkpoint-progress-reconcile.test.ts +219 -0
  1022. package/test/session/checkpoint-rebuild-unify.test.ts +143 -0
  1023. package/test/session/checkpoint-rebuild-v3.test.ts +248 -0
  1024. package/test/session/checkpoint-render-verify.test.ts +512 -0
  1025. package/test/session/checkpoint-retry.test.ts +150 -0
  1026. package/test/session/checkpoint-splitover-integration.test.ts +533 -0
  1027. package/test/session/checkpoint-templates.test.ts +51 -0
  1028. package/test/session/checkpoint-thresholds.test.ts +120 -0
  1029. package/test/session/checkpoint-validator.test.ts +189 -0
  1030. package/test/session/classify-integration.test.ts +476 -0
  1031. package/test/session/classify.test.ts +335 -0
  1032. package/test/session/codex-import.test.ts +331 -0
  1033. package/test/session/compaction-agent-scope.test.ts +209 -0
  1034. package/test/session/context-inheritance.test.ts +46 -0
  1035. package/test/session/external-import.test.ts +17 -0
  1036. package/test/session/fork-prefix-invariant.test.ts +116 -0
  1037. package/test/session/goal.test.ts +106 -0
  1038. package/test/session/instruction.test.ts +387 -0
  1039. package/test/session/invalid-output-continuation.test.ts +150 -0
  1040. package/test/session/last-message-info.test.ts +47 -0
  1041. package/test/session/length-tool-safety.test.ts +121 -0
  1042. package/test/session/llm-request-prefix.test.ts +197 -0
  1043. package/test/session/llm-retry.test.ts +59 -0
  1044. package/test/session/llm-system-prompt.test.ts +479 -0
  1045. package/test/session/llm.test.ts +1272 -0
  1046. package/test/session/main-lifecycle.test.ts +51 -0
  1047. package/test/session/main-runloop-history-invariant.test.ts +182 -0
  1048. package/test/session/max-mode-econnreset.test.ts +229 -0
  1049. package/test/session/max-mode.test.ts +54 -0
  1050. package/test/session/message-v2-filter.test.ts +197 -0
  1051. package/test/session/message-v2.test.ts +1119 -0
  1052. package/test/session/messages-default-main.test.ts +105 -0
  1053. package/test/session/messages-pagination.test.ts +888 -0
  1054. package/test/session/overflow.test.ts +576 -0
  1055. package/test/session/processor-effect.test.ts +853 -0
  1056. package/test/session/prompt-effect.test.ts +1609 -0
  1057. package/test/session/prompt-rebuild-loop.test.ts +108 -0
  1058. package/test/session/prompt-rebuild-reset.test.ts +67 -0
  1059. package/test/session/prompt-sweep.test.ts +145 -0
  1060. package/test/session/prompt-task-gate.test.ts +127 -0
  1061. package/test/session/prompt.test.ts +703 -0
  1062. package/test/session/prune-main-slice.test.ts +272 -0
  1063. package/test/session/prune-skip-system.test.ts +346 -0
  1064. package/test/session/prune.test.ts +419 -0
  1065. package/test/session/rebuild-microcompact.test.ts +318 -0
  1066. package/test/session/recall-reminder.test.ts +37 -0
  1067. package/test/session/retry.test.ts +410 -0
  1068. package/test/session/revert-compact.test.ts +639 -0
  1069. package/test/session/run-state-tuple-key.test.ts +152 -0
  1070. package/test/session/session-create-registers-main.test.ts +70 -0
  1071. package/test/session/session.test.ts +181 -0
  1072. package/test/session/snapshot-tool-race.test.ts +301 -0
  1073. package/test/session/structured-output-integration.test.ts +264 -0
  1074. package/test/session/structured-output-retry.test.ts +127 -0
  1075. package/test/session/structured-output.test.ts +397 -0
  1076. package/test/session/summary-main-slice.test.ts +170 -0
  1077. package/test/session/system.test.ts +72 -0
  1078. package/test/session/text-loop-detection.test.ts +185 -0
  1079. package/test/session/text-loop-integration.test.ts +448 -0
  1080. package/test/share/share-next.test.ts +332 -0
  1081. package/test/shell/shell.test.ts +73 -0
  1082. package/test/skill/compose-review.test.ts +141 -0
  1083. package/test/skill/discovery.test.ts +116 -0
  1084. package/test/skill/skill.test.ts +465 -0
  1085. package/test/snapshot/snapshot.test.ts +1531 -0
  1086. package/test/storage/db.test.ts +16 -0
  1087. package/test/storage/json-migration.test.ts +831 -0
  1088. package/test/storage/storage.test.ts +293 -0
  1089. package/test/sync/index.test.ts +237 -0
  1090. package/test/task/gate-state.test.ts +66 -0
  1091. package/test/task/gate.test.ts +167 -0
  1092. package/test/task/registry.test.ts +171 -0
  1093. package/test/task/state-machine.test.ts +292 -0
  1094. package/test/team/migrate-to-inbox.test.ts +124 -0
  1095. package/test/team/team.test.ts +75 -0
  1096. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  1097. package/test/tool/actor-cancel.test.ts +206 -0
  1098. package/test/tool/actor-recover.test.ts +50 -0
  1099. package/test/tool/actor-send.test.ts +200 -0
  1100. package/test/tool/actor-status.test.ts +296 -0
  1101. package/test/tool/actor-wait.test.ts +193 -0
  1102. package/test/tool/actor.shell.test.ts +250 -0
  1103. package/test/tool/actor.test.ts +748 -0
  1104. package/test/tool/apply_patch.test.ts +626 -0
  1105. package/test/tool/bash.test.ts +1195 -0
  1106. package/test/tool/describe-workflow.test.ts +12 -0
  1107. package/test/tool/edit.test.ts +691 -0
  1108. package/test/tool/external-directory.test.ts +207 -0
  1109. package/test/tool/fixtures/large-image.png +0 -0
  1110. package/test/tool/fixtures/models-api.json +65179 -0
  1111. package/test/tool/glob.test.ts +81 -0
  1112. package/test/tool/grep.test.ts +114 -0
  1113. package/test/tool/history.test.ts +144 -0
  1114. package/test/tool/invocation-style.test.ts +30 -0
  1115. package/test/tool/memory-edit-ask-skip.test.ts +62 -0
  1116. package/test/tool/memory-path-guard.test.ts +594 -0
  1117. package/test/tool/memory.test.ts +71 -0
  1118. package/test/tool/question.test.ts +167 -0
  1119. package/test/tool/read.test.ts +483 -0
  1120. package/test/tool/recoverable.test.ts +36 -0
  1121. package/test/tool/registry-invocation-style.test.ts +121 -0
  1122. package/test/tool/registry.test.ts +164 -0
  1123. package/test/tool/shell-tokenize.test.ts +273 -0
  1124. package/test/tool/shell-wrap-missing-script.test.ts +128 -0
  1125. package/test/tool/shell-wrap.test.ts +257 -0
  1126. package/test/tool/skill.test.ts +99 -0
  1127. package/test/tool/task-recover.test.ts +36 -0
  1128. package/test/tool/task.shell.test.ts +234 -0
  1129. package/test/tool/task.test.ts +296 -0
  1130. package/test/tool/tool-def-shell-shape.test.ts +23 -0
  1131. package/test/tool/tool-define.test.ts +59 -0
  1132. package/test/tool/tool-validation-error.test.ts +25 -0
  1133. package/test/tool/truncation.test.ts +253 -0
  1134. package/test/tool/webfetch.test.ts +103 -0
  1135. package/test/tool/whitelist.test.ts +373 -0
  1136. package/test/tool/write.test.ts +244 -0
  1137. package/test/util/data-url.test.ts +14 -0
  1138. package/test/util/effect-zod.test.ts +869 -0
  1139. package/test/util/error.test.ts +38 -0
  1140. package/test/util/filesystem.test.ts +656 -0
  1141. package/test/util/format.test.ts +59 -0
  1142. package/test/util/glob.test.ts +164 -0
  1143. package/test/util/iife.test.ts +36 -0
  1144. package/test/util/lazy.test.ts +50 -0
  1145. package/test/util/lock.test.ts +72 -0
  1146. package/test/util/log.test.ts +69 -0
  1147. package/test/util/module.test.ts +59 -0
  1148. package/test/util/process.test.ts +128 -0
  1149. package/test/util/queue.test.ts +64 -0
  1150. package/test/util/timeout.test.ts +21 -0
  1151. package/test/util/which.test.ts +100 -0
  1152. package/test/util/wildcard.test.ts +90 -0
  1153. package/test/workflow/builtin.test.ts +22 -0
  1154. package/test/workflow/deep-research-cluster.test.ts +47 -0
  1155. package/test/workflow/lib.ts +243 -0
  1156. package/test/workflow/meta.test.ts +142 -0
  1157. package/test/workflow/model-routing.test.ts +68 -0
  1158. package/test/workflow/persistence.test.ts +229 -0
  1159. package/test/workflow/resolve.test.ts +37 -0
  1160. package/test/workflow/runtime-nested.test.ts +419 -0
  1161. package/test/workflow/runtime-worktree.test.ts +261 -0
  1162. package/test/workflow/runtime.test.ts +1078 -0
  1163. package/test/workflow/sandbox.test.ts +259 -0
  1164. package/test/workflow/tool.test.ts +473 -0
  1165. package/test/workflow/verify-wow.test.ts +144 -0
  1166. package/test/workflow/workspace.test.ts +88 -0
  1167. package/test/workspace/workspace-restore.test.ts +281 -0
  1168. package/test/worktree/index.test.ts +30 -0
  1169. package/tsconfig.json +24 -0
@@ -0,0 +1,2567 @@
1
+ import {
2
+ batch,
3
+ createContext,
4
+ createEffect,
5
+ createMemo,
6
+ createSignal,
7
+ For,
8
+ Match,
9
+ on,
10
+ onCleanup,
11
+ onMount,
12
+ Show,
13
+ Switch,
14
+ useContext,
15
+ } from "solid-js"
16
+ import { Dynamic } from "solid-js/web"
17
+ import path from "path"
18
+ import { useCurrentAgentID, useRoute, useRouteData } from "@tui/context/route"
19
+ import { useProject } from "@tui/context/project"
20
+ import { useSync } from "@tui/context/sync"
21
+ import { useEvent } from "@tui/context/event"
22
+ import { SplitBorder } from "@tui/component/border"
23
+ import { Spinner } from "@tui/component/spinner"
24
+ import { selectedForeground, useTheme } from "@tui/context/theme"
25
+ import { BoxRenderable, ScrollBoxRenderable, addDefaultParsers, TextAttributes, RGBA, MouseEvent } from "@opentui/core"
26
+ import { Prompt, type PromptRef } from "@tui/component/prompt"
27
+ import type {
28
+ AssistantMessage,
29
+ Part,
30
+ Provider,
31
+ ToolPart,
32
+ UserMessage,
33
+ TextPart,
34
+ ReasoningPart,
35
+ } from "@mty-coder/sdk/v2"
36
+ import { useLocal } from "@tui/context/local"
37
+ import { Locale } from "@/util"
38
+ import type { Tool } from "@/tool"
39
+ import type { ReadTool } from "@/tool/read"
40
+ import type { WriteTool } from "@/tool/write"
41
+ import { BashTool } from "@/tool/bash"
42
+ import type { GlobTool } from "@/tool/glob"
43
+ import type { GrepTool } from "@/tool/grep"
44
+ import type { EditTool } from "@/tool/edit"
45
+ import type { ApplyPatchTool } from "@/tool/apply_patch"
46
+ import type { WebFetchTool } from "@/tool/webfetch"
47
+ import type { CodeSearchTool } from "@/tool/codesearch"
48
+ import type { WebSearchTool } from "@/tool/websearch"
49
+ import type { ActorTool } from "@/tool/actor"
50
+ import type { TaskTool } from "@/tool/task"
51
+ import type { QuestionTool } from "@/tool/question"
52
+ import type { SkillTool } from "@/tool/skill"
53
+ import { useKeyboard, useRenderer, useTerminalDimensions, type JSX } from "@opentui/solid"
54
+ import { useSDK } from "@tui/context/sdk"
55
+ import { useCommandDialog } from "@tui/component/dialog-command"
56
+ import { useLanguage } from "@tui/context/language"
57
+ import type { DialogContext } from "@tui/ui/dialog"
58
+ import { useKeybind } from "@tui/context/keybind"
59
+ import { useDialog } from "../../ui/dialog"
60
+ import { DialogMessage } from "./dialog-message"
61
+ import type { PromptInfo } from "../../component/prompt/history"
62
+ import { DialogConfirm } from "@tui/ui/dialog-confirm"
63
+ import { DialogTimeline } from "./dialog-timeline"
64
+ import { DialogForkFromTimeline } from "./dialog-fork-from-timeline"
65
+ import { DialogSessionRename } from "../../component/dialog-session-rename"
66
+ import { Sidebar } from "./sidebar"
67
+ import { SubagentFooter } from "./subagent-footer.tsx"
68
+ import { DialogSubagent } from "./dialog-subagent.tsx"
69
+ import { Flag } from "@/flag/flag"
70
+ import { LANGUAGE_EXTENSIONS } from "@/lsp/language"
71
+ import parsers from "../../../../../../parsers-config.ts"
72
+ import * as Clipboard from "../../util/clipboard"
73
+ import { Toast, useToast } from "../../ui/toast"
74
+ import { useKV } from "../../context/kv.tsx"
75
+ import * as Editor from "../../util/editor"
76
+ import stripAnsi from "strip-ansi"
77
+ import { usePromptRef } from "../../context/prompt"
78
+ import { useExit } from "../../context/exit"
79
+ import { Filesystem } from "@/util"
80
+ import { Global } from "@/global"
81
+ import { PermissionPrompt } from "./permission"
82
+ import { QuestionPrompt } from "./question"
83
+ import { DialogExportOptions } from "../../ui/dialog-export-options"
84
+ import * as Model from "../../util/model"
85
+ import { formatTranscript } from "../../util/transcript"
86
+ import { UI } from "@/cli/ui.ts"
87
+ import { useTuiConfig } from "../../context/tui-config"
88
+ import { getScrollAcceleration } from "../../util/scroll"
89
+ import { nextThinkingMode, reasoningSummary, useThinkingMode, type ThinkingMode } from "../../context/thinking"
90
+ import { TuiPluginRuntime } from "../../plugin"
91
+ import { DialogGoUpsell } from "../../component/dialog-go-upsell"
92
+ import { SessionRetry } from "@/session/retry"
93
+ import { getRevertDiffFiles } from "../../util/revert-diff"
94
+
95
+ addDefaultParsers(parsers.parsers)
96
+
97
+ const GO_UPSELL_LAST_SEEN_AT = "go_upsell_last_seen_at"
98
+ const GO_UPSELL_DONT_SHOW = "go_upsell_dont_show"
99
+ const GO_UPSELL_WINDOW = 86_400_000 // 24 hrs
100
+
101
+ const context = createContext<{
102
+ width: number
103
+ sessionID: string
104
+ conceal: () => boolean
105
+ thinkingMode: () => ThinkingMode
106
+ showThinking: () => boolean
107
+ showTimestamps: () => boolean
108
+ showDetails: () => boolean
109
+ showGenericToolOutput: () => boolean
110
+ diffWrapMode: () => "word" | "none"
111
+ providers: () => ReadonlyMap<string, Provider>
112
+ sync: ReturnType<typeof useSync>
113
+ tui: ReturnType<typeof useTuiConfig>
114
+ }>()
115
+
116
+ function use() {
117
+ const ctx = useContext(context)
118
+ if (!ctx) throw new Error("useContext must be used within a Session component")
119
+ return ctx
120
+ }
121
+
122
+ export function Session() {
123
+ const route = useRouteData("session")
124
+ const fullRoute = useRoute()
125
+ const navigate = fullRoute.navigate
126
+ const sync = useSync()
127
+ const event = useEvent()
128
+ const project = useProject()
129
+ const tuiConfig = useTuiConfig()
130
+ const kv = useKV()
131
+ const { theme } = useTheme()
132
+ const promptRef = usePromptRef()
133
+ const session = createMemo(() => sync.session.get(route.sessionID))
134
+ const currentAgentID = useCurrentAgentID()
135
+ const actors = createMemo(() => sync.data.actor[route.sessionID] ?? [])
136
+ const messages = createMemo(() => sync.data.message[route.sessionID]?.[currentAgentID()] ?? [])
137
+ const permissions = createMemo(() => sync.data.permission[route.sessionID] ?? [])
138
+ const questions = createMemo(() => sync.data.question[route.sessionID] ?? [])
139
+ const visible = createMemo(
140
+ () =>
141
+ !session()?.parentID &&
142
+ currentAgentID() === "main" &&
143
+ permissions().length === 0 &&
144
+ questions().length === 0,
145
+ )
146
+ const disabled = createMemo(() => permissions().length > 0 || questions().length > 0)
147
+
148
+ const pending = createMemo(() => {
149
+ return messages().findLast((x) => x.role === "assistant" && !x.time.completed)?.id
150
+ })
151
+
152
+ const lastAssistant = createMemo(() => {
153
+ return messages().findLast((x) => x.role === "assistant")
154
+ })
155
+
156
+ const dimensions = useTerminalDimensions()
157
+ const [sidebar, setSidebar] = kv.signal<"auto" | "hide">("sidebar", "auto")
158
+ const [sidebarOpen, setSidebarOpen] = createSignal(false)
159
+ const [conceal, setConceal] = createSignal(true)
160
+ const thinking = useThinkingMode()
161
+ const thinkingMode = thinking.mode
162
+ const showThinking = createMemo(() => true)
163
+ const [timestamps, setTimestamps] = kv.signal<"hide" | "show">("timestamps", "hide")
164
+ const [showDetails, setShowDetails] = kv.signal("tool_details_visibility", true)
165
+ const [showAssistantMetadata, _setShowAssistantMetadata] = kv.signal("assistant_metadata_visibility", true)
166
+ const [showScrollbar, setShowScrollbar] = kv.signal("scrollbar_visible", false)
167
+ const [scrolling, setScrolling] = createSignal(false)
168
+ let scrollHideTimer: ReturnType<typeof setTimeout> | undefined
169
+ const scrollbarVisible = createMemo(() => showScrollbar() || scrolling())
170
+ const onWheel = (evt: MouseEvent) => {
171
+ if (evt.type !== "scroll") return
172
+ setScrolling(true)
173
+ if (scrollHideTimer) clearTimeout(scrollHideTimer)
174
+ scrollHideTimer = setTimeout(() => setScrolling(false), 1000)
175
+ }
176
+ onCleanup(() => {
177
+ if (scrollHideTimer) clearTimeout(scrollHideTimer)
178
+ })
179
+ const [diffWrapMode] = kv.signal<"word" | "none">("diff_wrap_mode", "word")
180
+ const [_animationsEnabled, _setAnimationsEnabled] = kv.signal("animations_enabled", true)
181
+ const [showGenericToolOutput, setShowGenericToolOutput] = kv.signal("generic_tool_output_visibility", false)
182
+
183
+ const wide = createMemo(() => dimensions().width > 120)
184
+ const sidebarVisible = createMemo(() => {
185
+ if (session()?.parentID) return false
186
+ if (currentAgentID() !== "main") return false
187
+ if (sidebarOpen()) return true
188
+ if (sidebar() === "auto" && wide()) return true
189
+ return false
190
+ })
191
+ const showTimestamps = createMemo(() => timestamps() === "show")
192
+ const contentWidth = createMemo(() => dimensions().width - (sidebarVisible() ? 42 : 0) - 4)
193
+ const providers = createMemo(() => Model.index(sync.data.provider))
194
+
195
+ const scrollAcceleration = createMemo(() => getScrollAcceleration(tuiConfig))
196
+ const toast = useToast()
197
+ const sdk = useSDK()
198
+
199
+ createEffect(async () => {
200
+ const previousWorkspace = project.workspace.current()
201
+ const result = await sdk.client.session.get({ sessionID: route.sessionID }, { throwOnError: true })
202
+ if (!result.data) {
203
+ toast.show({
204
+ message: `Session not found: ${route.sessionID}`,
205
+ variant: "error",
206
+ })
207
+ navigate({ type: "home" })
208
+ return
209
+ }
210
+
211
+ if (result.data.workspaceID !== previousWorkspace) {
212
+ project.workspace.set(result.data.workspaceID)
213
+
214
+ // Sync all the data for this workspace. Note that this
215
+ // workspace may not exist anymore which is why this is not
216
+ // fatal. If it doesn't we still want to show the session
217
+ // (which will be non-interactive)
218
+ try {
219
+ await sync.bootstrap({ fatal: false })
220
+ } catch (e) {}
221
+ }
222
+ await sync.session.sync(route.sessionID)
223
+ if (scroll) scroll.scrollBy(100_000)
224
+ })
225
+
226
+ let lastSwitch: string | undefined = undefined
227
+ event.on("message.part.updated", (evt) => {
228
+ const part = evt.properties.part
229
+ if (part.type !== "tool") return
230
+ if (part.sessionID !== route.sessionID) return
231
+ if (part.state.status !== "completed") return
232
+ if (part.id === lastSwitch) return
233
+
234
+ if (part.tool === "plan_exit" && part.state.metadata?.switched) {
235
+ local.agent.set("build")
236
+ lastSwitch = part.id
237
+ } else if (part.tool === "plan_enter") {
238
+ local.agent.set("plan")
239
+ lastSwitch = part.id
240
+ }
241
+ })
242
+
243
+ let seeded = false
244
+ let scroll: ScrollBoxRenderable
245
+ let prompt: PromptRef | undefined
246
+ const bind = (r: PromptRef | undefined) => {
247
+ prompt = r
248
+ promptRef.set(r)
249
+ if (seeded || !route.prompt || !r) return
250
+ seeded = true
251
+ r.set(route.prompt)
252
+ }
253
+ const keybind = useKeybind()
254
+ const dialog = useDialog()
255
+ const renderer = useRenderer()
256
+
257
+ event.on("session.status", (evt) => {
258
+ if (evt.properties.sessionID !== route.sessionID) return
259
+ if (evt.properties.status.type !== "retry") return
260
+ if (evt.properties.status.message !== SessionRetry.GO_UPSELL_MESSAGE) return
261
+ if (dialog.stack.length > 0) return
262
+
263
+ const seen = kv.get(GO_UPSELL_LAST_SEEN_AT)
264
+ if (typeof seen === "number" && Date.now() - seen < GO_UPSELL_WINDOW) return
265
+
266
+ if (kv.get(GO_UPSELL_DONT_SHOW)) return
267
+
268
+ void DialogGoUpsell.show(dialog).then((dontShowAgain) => {
269
+ if (dontShowAgain) kv.set(GO_UPSELL_DONT_SHOW, true)
270
+ kv.set(GO_UPSELL_LAST_SEEN_AT, Date.now())
271
+ })
272
+ })
273
+
274
+ // Allow exit when in child session (prompt is hidden)
275
+ const exit = useExit()
276
+
277
+ createEffect(() => {
278
+ const title = Locale.truncate(session()?.title ?? "", 50)
279
+ const pad = (text: string) => text.padEnd(10, " ")
280
+ const weak = (text: string) => UI.Style.TEXT_DIM + pad(text) + UI.Style.TEXT_NORMAL
281
+ const logo = UI.logo(" ").split(/\r?\n/)
282
+ return exit.message.set(
283
+ [
284
+ ...logo,
285
+ ``,
286
+ ` ${weak("Session")}${UI.Style.TEXT_NORMAL_BOLD}${title}${UI.Style.TEXT_NORMAL}`,
287
+ ` ${weak("Continue")}${UI.Style.TEXT_NORMAL_BOLD}mty -s ${session()?.id}${UI.Style.TEXT_NORMAL}`,
288
+ ``,
289
+ ].join("\n"),
290
+ )
291
+ })
292
+
293
+ useKeyboard((evt) => {
294
+ if (!session()?.parentID && currentAgentID() === "main") return
295
+ if (keybind.match("app_exit", evt)) {
296
+ const status = sync.data.session_status?.[route.sessionID]
297
+ if (status && status.type !== "idle") {
298
+ void sdk.client.session.abort({ sessionID: route.sessionID }).catch(() => {})
299
+ return
300
+ }
301
+ void exit()
302
+ }
303
+ })
304
+
305
+ // Helper: Find next visible message boundary in direction
306
+ const findNextVisibleMessage = (direction: "next" | "prev"): string | null => {
307
+ const children = scroll.getChildren()
308
+ const messagesList = messages()
309
+ const scrollTop = scroll.y
310
+
311
+ // Get visible messages sorted by position, filtering for valid non-synthetic, non-ignored content
312
+ const visibleMessages = children
313
+ .filter((c) => {
314
+ if (!c.id) return false
315
+ const message = messagesList.find((m) => m.id === c.id)
316
+ if (!message) return false
317
+
318
+ // Check if message has valid non-synthetic, non-ignored text parts
319
+ const parts = sync.data.part[message.id]
320
+ if (!parts || !Array.isArray(parts)) return false
321
+
322
+ return parts.some((part) => part && part.type === "text" && !part.synthetic && !part.ignored)
323
+ })
324
+ .sort((a, b) => a.y - b.y)
325
+
326
+ if (visibleMessages.length === 0) return null
327
+
328
+ if (direction === "next") {
329
+ // Find first message below current position
330
+ return visibleMessages.find((c) => c.y > scrollTop + 10)?.id ?? null
331
+ }
332
+ // Find last message above current position
333
+ return [...visibleMessages].reverse().find((c) => c.y < scrollTop - 10)?.id ?? null
334
+ }
335
+
336
+ // Helper: Scroll to message in direction or fallback to page scroll
337
+ const scrollToMessage = (direction: "next" | "prev", dialog: ReturnType<typeof useDialog>) => {
338
+ const targetID = findNextVisibleMessage(direction)
339
+
340
+ if (!targetID) {
341
+ scroll.scrollBy(direction === "next" ? scroll.height : -scroll.height)
342
+ dialog.clear()
343
+ return
344
+ }
345
+
346
+ const child = scroll.getChildren().find((c) => c.id === targetID)
347
+ if (child) scroll.scrollBy(child.y - scroll.y - 1)
348
+ dialog.clear()
349
+ }
350
+
351
+ function toBottom() {
352
+ setTimeout(() => {
353
+ if (!scroll || scroll.isDestroyed) return
354
+ scroll.scrollTo(scroll.scrollHeight)
355
+ }, 50)
356
+ }
357
+
358
+ const local = useLocal()
359
+
360
+ function moveFirstChild() {
361
+ const list = actors().filter((a) => a.mode === "subagent")
362
+ if (list.length === 0) {
363
+ dialog.replace(() => <DialogSubagent sessionID={route.sessionID} />)
364
+ return
365
+ }
366
+ if (fullRoute.data.type !== "session") return
367
+ navigate({ ...fullRoute.data, agentID: list[0].actor_id })
368
+ }
369
+
370
+ function moveChild(direction: 1 | -1) {
371
+ const list = actors().filter((a) => a.mode === "subagent")
372
+ if (list.length === 0) return
373
+ if (fullRoute.data.type !== "session") return
374
+ const cur = currentAgentID()
375
+ const idx = list.findIndex((a) => a.actor_id === cur)
376
+ const next =
377
+ idx === -1
378
+ ? direction === 1
379
+ ? 0
380
+ : list.length - 1
381
+ : (idx + direction + list.length) % list.length
382
+ navigate({ ...fullRoute.data, agentID: list[next].actor_id })
383
+ }
384
+
385
+ const command = useCommandDialog()
386
+ const t = useLanguage().t
387
+ command.register(() => [
388
+ {
389
+ title: t(session()?.share?.url ? "tui.command.session.share.copy_link" : "tui.command.session.share.title"),
390
+ value: "session.share",
391
+ suggested: route.type === "session",
392
+ keybind: "session_share",
393
+ category: "session",
394
+ enabled: sync.data.config.share !== "disabled",
395
+ slash: {
396
+ name: "share",
397
+ },
398
+ onSelect: async (dialog) => {
399
+ const copy = (url: string) =>
400
+ Clipboard.copy(url)
401
+ .then(() => toast.show({ message: "Share URL copied to clipboard!", variant: "success" }))
402
+ .catch(() => toast.show({ message: "Failed to copy URL to clipboard", variant: "error" }))
403
+ const url = session()?.share?.url
404
+ if (url) {
405
+ await copy(url)
406
+ dialog.clear()
407
+ return
408
+ }
409
+ if (!kv.get("share_consent", false)) {
410
+ const ok = await DialogConfirm.show(dialog, "Share Session", "Are you sure you want to share it?")
411
+ if (ok !== true) return
412
+ kv.set("share_consent", true)
413
+ }
414
+ await sdk.client.session
415
+ .share({
416
+ sessionID: route.sessionID,
417
+ })
418
+ .then((res) => copy(res.data!.share!.url))
419
+ .catch((error) => {
420
+ toast.show({
421
+ message: error instanceof Error ? error.message : "Failed to share session",
422
+ variant: "error",
423
+ })
424
+ })
425
+ dialog.clear()
426
+ },
427
+ },
428
+ {
429
+ title: t("tui.command.session.rename.title"),
430
+ value: "session.rename",
431
+ keybind: "session_rename",
432
+ category: "session",
433
+ slash: {
434
+ name: "rename",
435
+ },
436
+ onSelect: (dialog) => {
437
+ dialog.replace(() => <DialogSessionRename session={route.sessionID} />)
438
+ },
439
+ },
440
+ {
441
+ title: t("tui.command.session.timeline.title"),
442
+ value: "session.timeline",
443
+ keybind: "session_timeline",
444
+ category: "session",
445
+ slash: {
446
+ name: "timeline",
447
+ },
448
+ onSelect: (dialog) => {
449
+ dialog.replace(() => (
450
+ <DialogTimeline
451
+ onMove={(messageID) => {
452
+ const child = scroll.getChildren().find((child) => {
453
+ return child.id === messageID
454
+ })
455
+ if (child) scroll.scrollBy(child.y - scroll.y - 1)
456
+ }}
457
+ sessionID={route.sessionID}
458
+ setPrompt={(promptInfo) => prompt?.set(promptInfo)}
459
+ />
460
+ ))
461
+ },
462
+ },
463
+ {
464
+ title: t("tui.command.session.fork.title"),
465
+ value: "session.fork",
466
+ keybind: "session_fork",
467
+ category: "session",
468
+ slash: {
469
+ name: "fork",
470
+ },
471
+ onSelect: (dialog) => {
472
+ dialog.replace(() => (
473
+ <DialogForkFromTimeline
474
+ onMove={(messageID) => {
475
+ if (!messageID) return
476
+ const child = scroll.getChildren().find((child) => {
477
+ return child.id === messageID
478
+ })
479
+ if (child) scroll.scrollBy(child.y - scroll.y - 1)
480
+ }}
481
+ sessionID={route.sessionID}
482
+ />
483
+ ))
484
+ },
485
+ },
486
+ {
487
+ title: t("tui.command.session.compact.title"),
488
+ value: "session.compact",
489
+ keybind: "session_compact",
490
+ category: "session",
491
+ slash: {
492
+ name: "compact",
493
+ aliases: ["summarize"],
494
+ },
495
+ onSelect: (dialog) => {
496
+ const selectedModel = local.model.current()
497
+ if (!selectedModel) {
498
+ toast.show({
499
+ variant: "warning",
500
+ message: "Connect a provider to summarize this session",
501
+ duration: 3000,
502
+ })
503
+ return
504
+ }
505
+ void sdk.client.session.summarize({
506
+ sessionID: route.sessionID,
507
+ modelID: selectedModel.modelID,
508
+ providerID: selectedModel.providerID,
509
+ })
510
+ dialog.clear()
511
+ },
512
+ },
513
+ {
514
+ title: t("tui.command.session.unshare.title"),
515
+ value: "session.unshare",
516
+ keybind: "session_unshare",
517
+ category: "session",
518
+ enabled: !!session()?.share?.url,
519
+ slash: {
520
+ name: "unshare",
521
+ },
522
+ onSelect: async (dialog) => {
523
+ await sdk.client.session
524
+ .unshare({
525
+ sessionID: route.sessionID,
526
+ })
527
+ .then(() => toast.show({ message: "Session unshared successfully", variant: "success" }))
528
+ .catch((error) => {
529
+ toast.show({
530
+ message: error instanceof Error ? error.message : "Failed to unshare session",
531
+ variant: "error",
532
+ })
533
+ })
534
+ dialog.clear()
535
+ },
536
+ },
537
+ {
538
+ title: t("tui.command.session.undo.title"),
539
+ value: "session.undo",
540
+ keybind: "messages_undo",
541
+ category: "session",
542
+ slash: {
543
+ name: "undo",
544
+ },
545
+ onSelect: async (dialog) => {
546
+ const status = sync.data.session_status?.[route.sessionID]
547
+ if (status?.type !== "idle") await sdk.client.session.abort({ sessionID: route.sessionID }).catch(() => {})
548
+ const revert = session()?.revert?.messageID
549
+ const message = messages().findLast((x) => (!revert || x.id < revert) && x.role === "user")
550
+ if (!message) return
551
+ void sdk.client.session
552
+ .revert({
553
+ sessionID: route.sessionID,
554
+ messageID: message.id,
555
+ })
556
+ .then(() => {
557
+ toBottom()
558
+ })
559
+ const parts = sync.data.part[message.id]
560
+ prompt?.set(
561
+ parts.reduce(
562
+ (agg, part) => {
563
+ if (part.type === "text") {
564
+ if (!part.synthetic) agg.input += part.text
565
+ }
566
+ if (part.type === "file") agg.parts.push(part)
567
+ return agg
568
+ },
569
+ { input: "", parts: [] as PromptInfo["parts"] },
570
+ ),
571
+ )
572
+ dialog.clear()
573
+ },
574
+ },
575
+ {
576
+ title: t("tui.command.session.redo.title"),
577
+ value: "session.redo",
578
+ keybind: "messages_redo",
579
+ category: "session",
580
+ enabled: !!session()?.revert?.messageID,
581
+ slash: {
582
+ name: "redo",
583
+ },
584
+ onSelect: (dialog) => {
585
+ dialog.clear()
586
+ const messageID = session()?.revert?.messageID
587
+ if (!messageID) return
588
+ const message = messages().find((x) => x.role === "user" && x.id > messageID)
589
+ if (!message) {
590
+ void sdk.client.session.unrevert({
591
+ sessionID: route.sessionID,
592
+ })
593
+ prompt?.set({ input: "", parts: [] })
594
+ return
595
+ }
596
+ void sdk.client.session.revert({
597
+ sessionID: route.sessionID,
598
+ messageID: message.id,
599
+ })
600
+ },
601
+ },
602
+ {
603
+ title: t(sidebarVisible() ? "tui.command.session.sidebar.hide" : "tui.command.session.sidebar.show"),
604
+ value: "session.sidebar.toggle",
605
+ keybind: "sidebar_toggle",
606
+ category: "session",
607
+ onSelect: (dialog) => {
608
+ batch(() => {
609
+ const isVisible = sidebarVisible()
610
+ setSidebar(() => (isVisible ? "hide" : "auto"))
611
+ setSidebarOpen(!isVisible)
612
+ })
613
+ dialog.clear()
614
+ },
615
+ },
616
+ {
617
+ title: t(conceal() ? "tui.command.session.conceal.disable" : "tui.command.session.conceal.enable"),
618
+ value: "session.toggle.conceal",
619
+ keybind: "messages_toggle_conceal",
620
+ category: "session",
621
+ onSelect: (dialog) => {
622
+ setConceal((prev) => !prev)
623
+ dialog.clear()
624
+ },
625
+ },
626
+ {
627
+ title: t(showTimestamps() ? "tui.command.session.timestamps.hide" : "tui.command.session.timestamps.show"),
628
+ value: "session.toggle.timestamps",
629
+ category: "session",
630
+ slash: {
631
+ name: "timestamps",
632
+ aliases: ["toggle-timestamps"],
633
+ },
634
+ onSelect: (dialog) => {
635
+ setTimestamps((prev) => (prev === "show" ? "hide" : "show"))
636
+ dialog.clear()
637
+ },
638
+ },
639
+ {
640
+ title: t(
641
+ nextThinkingMode(thinkingMode()) === "hide"
642
+ ? "tui.command.session.thinking.collapse"
643
+ : "tui.command.session.thinking.expand",
644
+ ),
645
+ value: "session.toggle.thinking",
646
+ keybind: "display_thinking",
647
+ category: "session",
648
+ slash: {
649
+ name: "thinking",
650
+ aliases: ["toggle-thinking"],
651
+ },
652
+ onSelect: (dialog) => {
653
+ thinking.set(nextThinkingMode(thinkingMode()))
654
+ dialog.clear()
655
+ },
656
+ },
657
+ {
658
+ title: t(showDetails() ? "tui.command.session.tool_details.hide" : "tui.command.session.tool_details.show"),
659
+ value: "session.toggle.actions",
660
+ keybind: "tool_details",
661
+ category: "session",
662
+ onSelect: (dialog) => {
663
+ setShowDetails((prev) => !prev)
664
+ dialog.clear()
665
+ },
666
+ },
667
+ {
668
+ title: t("tui.command.session.scrollbar.toggle"),
669
+ value: "session.toggle.scrollbar",
670
+ keybind: "scrollbar_toggle",
671
+ category: "session",
672
+ onSelect: (dialog) => {
673
+ setShowScrollbar((prev) => !prev)
674
+ dialog.clear()
675
+ },
676
+ },
677
+ {
678
+ title: t(
679
+ showGenericToolOutput()
680
+ ? "tui.command.session.generic_tool_output.hide"
681
+ : "tui.command.session.generic_tool_output.show",
682
+ ),
683
+ value: "session.toggle.generic_tool_output",
684
+ category: "session",
685
+ onSelect: (dialog) => {
686
+ setShowGenericToolOutput((prev) => !prev)
687
+ dialog.clear()
688
+ },
689
+ },
690
+ {
691
+ title: t("tui.command.session.page_up.title"),
692
+ value: "session.page.up",
693
+ keybind: "messages_page_up",
694
+ category: "session",
695
+ hidden: true,
696
+ onSelect: (dialog) => {
697
+ scroll.scrollBy(-scroll.height / 2)
698
+ dialog.clear()
699
+ },
700
+ },
701
+ {
702
+ title: t("tui.command.session.page_down.title"),
703
+ value: "session.page.down",
704
+ keybind: "messages_page_down",
705
+ category: "session",
706
+ hidden: true,
707
+ onSelect: (dialog) => {
708
+ scroll.scrollBy(scroll.height / 2)
709
+ dialog.clear()
710
+ },
711
+ },
712
+ {
713
+ title: t("tui.command.session.line_up.title"),
714
+ value: "session.line.up",
715
+ keybind: "messages_line_up",
716
+ category: "session",
717
+ disabled: true,
718
+ onSelect: (dialog) => {
719
+ scroll.scrollBy(-1)
720
+ dialog.clear()
721
+ },
722
+ },
723
+ {
724
+ title: t("tui.command.session.line_down.title"),
725
+ value: "session.line.down",
726
+ keybind: "messages_line_down",
727
+ category: "session",
728
+ disabled: true,
729
+ onSelect: (dialog) => {
730
+ scroll.scrollBy(1)
731
+ dialog.clear()
732
+ },
733
+ },
734
+ {
735
+ title: t("tui.command.session.half_page_up.title"),
736
+ value: "session.half.page.up",
737
+ keybind: "messages_half_page_up",
738
+ category: "session",
739
+ hidden: true,
740
+ onSelect: (dialog) => {
741
+ scroll.scrollBy(-scroll.height / 4)
742
+ dialog.clear()
743
+ },
744
+ },
745
+ {
746
+ title: t("tui.command.session.half_page_down.title"),
747
+ value: "session.half.page.down",
748
+ keybind: "messages_half_page_down",
749
+ category: "session",
750
+ hidden: true,
751
+ onSelect: (dialog) => {
752
+ scroll.scrollBy(scroll.height / 4)
753
+ dialog.clear()
754
+ },
755
+ },
756
+ {
757
+ title: t("tui.command.session.first.title"),
758
+ value: "session.first",
759
+ keybind: "messages_first",
760
+ category: "session",
761
+ hidden: true,
762
+ onSelect: (dialog) => {
763
+ scroll.scrollTo(0)
764
+ dialog.clear()
765
+ },
766
+ },
767
+ {
768
+ title: t("tui.command.session.last.title"),
769
+ value: "session.last",
770
+ keybind: "messages_last",
771
+ category: "session",
772
+ hidden: true,
773
+ onSelect: (dialog) => {
774
+ scroll.scrollTo(scroll.scrollHeight)
775
+ dialog.clear()
776
+ },
777
+ },
778
+ {
779
+ title: t("tui.command.session.last_user.title"),
780
+ value: "session.messages_last_user",
781
+ keybind: "messages_last_user",
782
+ category: "session",
783
+ hidden: true,
784
+ onSelect: () => {
785
+ const msgs = messages()
786
+ if (!msgs || !msgs.length) return
787
+
788
+ // Find the most recent user message with non-ignored, non-synthetic text parts
789
+ for (let i = msgs.length - 1; i >= 0; i--) {
790
+ const message = msgs[i]
791
+ if (!message || message.role !== "user") continue
792
+
793
+ const parts = sync.data.part[message.id]
794
+ if (!parts || !Array.isArray(parts)) continue
795
+
796
+ const hasValidTextPart = parts.some(
797
+ (part) => part && part.type === "text" && !part.synthetic && !part.ignored,
798
+ )
799
+
800
+ if (hasValidTextPart) {
801
+ const child = scroll.getChildren().find((child) => {
802
+ return child.id === message.id
803
+ })
804
+ if (child) scroll.scrollBy(child.y - scroll.y - 1)
805
+ break
806
+ }
807
+ }
808
+ },
809
+ },
810
+ {
811
+ title: t("tui.command.session.message_next.title"),
812
+ value: "session.message.next",
813
+ keybind: "messages_next",
814
+ category: "session",
815
+ hidden: true,
816
+ onSelect: (dialog) => scrollToMessage("next", dialog),
817
+ },
818
+ {
819
+ title: t("tui.command.session.message_previous.title"),
820
+ value: "session.message.previous",
821
+ keybind: "messages_previous",
822
+ category: "session",
823
+ hidden: true,
824
+ onSelect: (dialog) => scrollToMessage("prev", dialog),
825
+ },
826
+ {
827
+ title: t("tui.command.messages.copy.title"),
828
+ value: "messages.copy",
829
+ keybind: "messages_copy",
830
+ category: "session",
831
+ onSelect: (dialog) => {
832
+ const revertID = session()?.revert?.messageID
833
+ const lastAssistantMessage = messages().findLast(
834
+ (msg) => msg.role === "assistant" && (!revertID || msg.id < revertID),
835
+ )
836
+ if (!lastAssistantMessage) {
837
+ toast.show({ message: "No assistant messages found", variant: "error" })
838
+ dialog.clear()
839
+ return
840
+ }
841
+
842
+ const parts = sync.data.part[lastAssistantMessage.id] ?? []
843
+ const textParts = parts.filter((part) => part.type === "text")
844
+ if (textParts.length === 0) {
845
+ toast.show({ message: "No text parts found in last assistant message", variant: "error" })
846
+ dialog.clear()
847
+ return
848
+ }
849
+
850
+ const text = textParts
851
+ .map((part) => part.text)
852
+ .join("\n")
853
+ .trim()
854
+ if (!text) {
855
+ toast.show({
856
+ message: "No text content found in last assistant message",
857
+ variant: "error",
858
+ })
859
+ dialog.clear()
860
+ return
861
+ }
862
+
863
+ Clipboard.copy(text)
864
+ .then(() => toast.show({ message: "Message copied to clipboard!", variant: "success" }))
865
+ .catch(() => toast.show({ message: "Failed to copy to clipboard", variant: "error" }))
866
+ dialog.clear()
867
+ },
868
+ },
869
+ {
870
+ title: t("tui.command.session.copy.title"),
871
+ value: "session.copy",
872
+ category: "session",
873
+ slash: {
874
+ name: "copy",
875
+ },
876
+ onSelect: async (dialog) => {
877
+ try {
878
+ const sessionData = session()
879
+ if (!sessionData) return
880
+ const sessionMessages = messages()
881
+ const transcript = formatTranscript(
882
+ sessionData,
883
+ sessionMessages.map((msg) => ({ info: msg, parts: sync.data.part[msg.id] ?? [] })),
884
+ {
885
+ thinking: showThinking(),
886
+ toolDetails: showDetails(),
887
+ assistantMetadata: showAssistantMetadata(),
888
+ providers: sync.data.provider,
889
+ },
890
+ )
891
+ await Clipboard.copy(transcript)
892
+ toast.show({ message: "Session transcript copied to clipboard!", variant: "success" })
893
+ } catch {
894
+ toast.show({ message: "Failed to copy session transcript", variant: "error" })
895
+ }
896
+ dialog.clear()
897
+ },
898
+ },
899
+ {
900
+ title: t("tui.command.session.export.title"),
901
+ value: "session.export",
902
+ keybind: "session_export",
903
+ category: "session",
904
+ slash: {
905
+ name: "export",
906
+ },
907
+ onSelect: async (dialog) => {
908
+ try {
909
+ const sessionData = session()
910
+ if (!sessionData) return
911
+ const sessionMessages = messages()
912
+
913
+ const defaultFilename = `session-${sessionData.id.slice(0, 8)}.md`
914
+
915
+ const options = await DialogExportOptions.show(
916
+ dialog,
917
+ defaultFilename,
918
+ showThinking(),
919
+ showDetails(),
920
+ showAssistantMetadata(),
921
+ false,
922
+ )
923
+
924
+ if (options === null) return
925
+
926
+ const transcript = formatTranscript(
927
+ sessionData,
928
+ sessionMessages.map((msg) => ({ info: msg, parts: sync.data.part[msg.id] ?? [] })),
929
+ {
930
+ thinking: options.thinking,
931
+ toolDetails: options.toolDetails,
932
+ assistantMetadata: options.assistantMetadata,
933
+ providers: sync.data.provider,
934
+ },
935
+ )
936
+
937
+ if (options.openWithoutSaving) {
938
+ // Just open in editor without saving
939
+ await Editor.open({ value: transcript, renderer })
940
+ } else {
941
+ const exportDir = process.cwd()
942
+ const filename = options.filename.trim()
943
+ const filepath = path.join(exportDir, filename)
944
+
945
+ await Filesystem.write(filepath, transcript)
946
+
947
+ // Open with EDITOR if available
948
+ const result = await Editor.open({ value: transcript, renderer })
949
+ if (result !== undefined) {
950
+ await Filesystem.write(filepath, result)
951
+ }
952
+
953
+ toast.show({ message: `Session exported to ${filename}`, variant: "success" })
954
+ }
955
+ } catch {
956
+ toast.show({ message: "Failed to export session", variant: "error" })
957
+ }
958
+ dialog.clear()
959
+ },
960
+ },
961
+ {
962
+ title: t("tui.command.session.child_first.title"),
963
+ value: "session.child.first",
964
+ keybind: "session_child_first",
965
+ category: "session",
966
+ hidden: true,
967
+ onSelect: (dialog) => {
968
+ moveFirstChild()
969
+ dialog.clear()
970
+ },
971
+ },
972
+ {
973
+ title: t("tui.command.session.parent.title"),
974
+ value: "session.parent",
975
+ keybind: "session_parent",
976
+ category: "session",
977
+ hidden: true,
978
+ enabled: currentAgentID() !== "main" || !!session()?.parentID,
979
+ onSelect: (dialog) => {
980
+ if (fullRoute.data.type === "session" && currentAgentID() !== "main") {
981
+ navigate({ ...fullRoute.data, agentID: undefined })
982
+ dialog.clear()
983
+ return
984
+ }
985
+ const parentID = session()?.parentID
986
+ if (parentID) {
987
+ navigate({
988
+ type: "session",
989
+ sessionID: parentID,
990
+ })
991
+ }
992
+ dialog.clear()
993
+ },
994
+ },
995
+ {
996
+ title: t("tui.command.session.child_next.title"),
997
+ value: "session.child.next",
998
+ keybind: "session_child_cycle",
999
+ category: "session",
1000
+ hidden: true,
1001
+ onSelect: (dialog) => {
1002
+ moveChild(1)
1003
+ dialog.clear()
1004
+ },
1005
+ },
1006
+ {
1007
+ title: t("tui.command.session.child_previous.title"),
1008
+ value: "session.child.previous",
1009
+ keybind: "session_child_cycle_reverse",
1010
+ category: "session",
1011
+ hidden: true,
1012
+ onSelect: (dialog) => {
1013
+ moveChild(-1)
1014
+ dialog.clear()
1015
+ },
1016
+ },
1017
+ ])
1018
+
1019
+ const revertInfo = createMemo(() => session()?.revert)
1020
+ const revertMessageID = createMemo(() => revertInfo()?.messageID)
1021
+
1022
+ const revertDiffFiles = createMemo(() => getRevertDiffFiles(revertInfo()?.diff ?? ""))
1023
+
1024
+ const revertRevertedMessages = createMemo(() => {
1025
+ const messageID = revertMessageID()
1026
+ if (!messageID) return []
1027
+ return messages().filter((x) => x.id >= messageID && x.role === "user")
1028
+ })
1029
+
1030
+ const revert = createMemo(() => {
1031
+ const info = revertInfo()
1032
+ if (!info) return
1033
+ if (!info.messageID) return
1034
+ return {
1035
+ messageID: info.messageID,
1036
+ reverted: revertRevertedMessages(),
1037
+ diff: info.diff,
1038
+ diffFiles: revertDiffFiles(),
1039
+ }
1040
+ })
1041
+
1042
+ // snap to bottom when session changes
1043
+ createEffect(on(() => route.sessionID, toBottom))
1044
+
1045
+ return (
1046
+ <context.Provider
1047
+ value={{
1048
+ get width() {
1049
+ return contentWidth()
1050
+ },
1051
+ sessionID: route.sessionID,
1052
+ conceal,
1053
+ thinkingMode,
1054
+ showThinking,
1055
+ showTimestamps,
1056
+ showDetails,
1057
+ showGenericToolOutput,
1058
+ diffWrapMode,
1059
+ providers,
1060
+ sync,
1061
+ tui: tuiConfig,
1062
+ }}
1063
+ >
1064
+ <box flexDirection="row">
1065
+ <box flexGrow={1} paddingBottom={1} paddingLeft={2} paddingRight={2} gap={1} onMouse={onWheel}>
1066
+ <Show when={session()}>
1067
+ <scrollbox
1068
+ ref={(r) => (scroll = r)}
1069
+ viewportOptions={{
1070
+ paddingRight: 1,
1071
+ }}
1072
+ verticalScrollbarOptions={{
1073
+ paddingLeft: 1,
1074
+ visible: true,
1075
+ trackOptions: {
1076
+ backgroundColor: scrollbarVisible() ? theme.backgroundElement : theme.background,
1077
+ foregroundColor: scrollbarVisible() ? theme.border : theme.background,
1078
+ },
1079
+ }}
1080
+ stickyScroll={true}
1081
+ stickyStart="bottom"
1082
+ flexGrow={1}
1083
+ scrollAcceleration={scrollAcceleration()}
1084
+ >
1085
+ <box height={1} />
1086
+ <For each={messages()}>
1087
+ {(message, index) => (
1088
+ <Switch>
1089
+ <Match when={message.id === revert()?.messageID}>
1090
+ {(function () {
1091
+ const command = useCommandDialog()
1092
+ const [hover, setHover] = createSignal(false)
1093
+ const dialog = useDialog()
1094
+
1095
+ const handleUnrevert = async () => {
1096
+ const confirmed = await DialogConfirm.show(
1097
+ dialog,
1098
+ "Confirm Redo",
1099
+ "Are you sure you want to restore the reverted messages?",
1100
+ )
1101
+ if (confirmed) {
1102
+ command.trigger("session.redo")
1103
+ }
1104
+ }
1105
+
1106
+ return (
1107
+ <box
1108
+ onMouseOver={() => setHover(true)}
1109
+ onMouseOut={() => setHover(false)}
1110
+ onMouseUp={handleUnrevert}
1111
+ marginTop={1}
1112
+ flexShrink={0}
1113
+ border={["left"]}
1114
+ customBorderChars={SplitBorder.customBorderChars}
1115
+ borderColor={theme.backgroundPanel}
1116
+ >
1117
+ <box
1118
+ paddingTop={1}
1119
+ paddingBottom={1}
1120
+ paddingLeft={2}
1121
+ backgroundColor={hover() ? theme.backgroundElement : theme.backgroundPanel}
1122
+ >
1123
+ <text fg={theme.textMuted}>{revert()!.reverted.length} message reverted</text>
1124
+ <text fg={theme.textMuted}>
1125
+ <span style={{ fg: theme.text }}>{keybind.print("messages_redo")}</span> or /redo to
1126
+ restore
1127
+ </text>
1128
+ <Show when={revert()!.diffFiles?.length}>
1129
+ <box marginTop={1}>
1130
+ <For each={revert()!.diffFiles}>
1131
+ {(file) => (
1132
+ <text fg={theme.text}>
1133
+ {file.filename}
1134
+ <Show when={file.additions > 0}>
1135
+ <span style={{ fg: theme.diffAdded }}> +{file.additions}</span>
1136
+ </Show>
1137
+ <Show when={file.deletions > 0}>
1138
+ <span style={{ fg: theme.diffRemoved }}> -{file.deletions}</span>
1139
+ </Show>
1140
+ </text>
1141
+ )}
1142
+ </For>
1143
+ </box>
1144
+ </Show>
1145
+ </box>
1146
+ </box>
1147
+ )
1148
+ })()}
1149
+ </Match>
1150
+ <Match when={revert()?.messageID && message.id >= revert()!.messageID}>
1151
+ <></>
1152
+ </Match>
1153
+ <Match when={message.role === "user"}>
1154
+ <UserMessage
1155
+ index={index()}
1156
+ onMouseUp={() => {
1157
+ if (renderer.getSelection()?.getSelectedText()) return
1158
+ dialog.replace(() => (
1159
+ <DialogMessage
1160
+ messageID={message.id}
1161
+ sessionID={route.sessionID}
1162
+ setPrompt={(promptInfo) => prompt?.set(promptInfo)}
1163
+ />
1164
+ ))
1165
+ }}
1166
+ message={message as UserMessage}
1167
+ parts={sync.data.part[message.id] ?? []}
1168
+ pending={pending()}
1169
+ />
1170
+ </Match>
1171
+ <Match when={message.role === "assistant"}>
1172
+ <AssistantMessage
1173
+ last={lastAssistant()?.id === message.id}
1174
+ message={message as AssistantMessage}
1175
+ parts={sync.data.part[message.id] ?? []}
1176
+ />
1177
+ </Match>
1178
+ </Switch>
1179
+ )}
1180
+ </For>
1181
+ </scrollbox>
1182
+ <box flexShrink={0}>
1183
+ <Show when={permissions().length > 0}>
1184
+ <PermissionPrompt request={permissions()[0]} />
1185
+ </Show>
1186
+ <Show when={permissions().length === 0 && questions().length > 0}>
1187
+ <QuestionPrompt request={questions()[0]} />
1188
+ </Show>
1189
+ <Show when={session()?.parentID || currentAgentID() !== "main"}>
1190
+ <SubagentFooter />
1191
+ </Show>
1192
+ <Show when={visible()}>
1193
+ <TuiPluginRuntime.Slot
1194
+ name="session_prompt"
1195
+ mode="replace"
1196
+ session_id={route.sessionID}
1197
+ visible={visible()}
1198
+ disabled={disabled()}
1199
+ on_submit={toBottom}
1200
+ ref={bind}
1201
+ >
1202
+ <Prompt
1203
+ visible={visible()}
1204
+ ref={bind}
1205
+ disabled={disabled()}
1206
+ onSubmit={() => {
1207
+ toBottom()
1208
+ }}
1209
+ sessionID={route.sessionID}
1210
+ right={<TuiPluginRuntime.Slot name="session_prompt_right" session_id={route.sessionID} />}
1211
+ />
1212
+ </TuiPluginRuntime.Slot>
1213
+ </Show>
1214
+ </box>
1215
+ </Show>
1216
+ <Toast />
1217
+ </box>
1218
+ <Show when={sidebarVisible()}>
1219
+ <Switch>
1220
+ <Match when={wide()}>
1221
+ <Sidebar sessionID={route.sessionID} />
1222
+ </Match>
1223
+ <Match when={!wide()}>
1224
+ <box
1225
+ position="absolute"
1226
+ top={0}
1227
+ left={0}
1228
+ right={0}
1229
+ bottom={0}
1230
+ alignItems="flex-end"
1231
+ backgroundColor={RGBA.fromInts(0, 0, 0, 70)}
1232
+ >
1233
+ <Sidebar sessionID={route.sessionID} />
1234
+ </box>
1235
+ </Match>
1236
+ </Switch>
1237
+ </Show>
1238
+ </box>
1239
+ </context.Provider>
1240
+ )
1241
+ }
1242
+
1243
+ const MIME_BADGE: Record<string, string> = {
1244
+ "text/plain": "txt",
1245
+ "image/png": "img",
1246
+ "image/jpeg": "img",
1247
+ "image/gif": "img",
1248
+ "image/webp": "img",
1249
+ "application/pdf": "pdf",
1250
+ "application/x-directory": "dir",
1251
+ }
1252
+
1253
+ function UserMessage(props: {
1254
+ message: UserMessage
1255
+ parts: Part[]
1256
+ onMouseUp: () => void
1257
+ index: number
1258
+ pending?: string
1259
+ }) {
1260
+ const ctx = use()
1261
+ const local = useLocal()
1262
+ const text = createMemo(() => props.parts.flatMap((x) => (x.type === "text" && !x.synthetic ? [x] : []))[0])
1263
+ const files = createMemo(() => props.parts.flatMap((x) => (x.type === "file" ? [x] : [])))
1264
+ const { theme } = useTheme()
1265
+ const [hover, setHover] = createSignal(false)
1266
+ const queued = createMemo(() => props.pending && props.message.id > props.pending)
1267
+ const color = createMemo(() => local.agent.color(props.message.agent))
1268
+ const queuedFg = createMemo(() => selectedForeground(theme, color()))
1269
+ const metadataVisible = createMemo(() => queued() || ctx.showTimestamps())
1270
+
1271
+ return (
1272
+ <>
1273
+ <Show when={text()}>
1274
+ <box
1275
+ id={props.message.id}
1276
+ border={["left"]}
1277
+ borderColor={color()}
1278
+ customBorderChars={SplitBorder.customBorderChars}
1279
+ marginTop={props.index === 0 ? 0 : 1}
1280
+ >
1281
+ <box
1282
+ onMouseOver={() => {
1283
+ setHover(true)
1284
+ }}
1285
+ onMouseOut={() => {
1286
+ setHover(false)
1287
+ }}
1288
+ onMouseUp={props.onMouseUp}
1289
+ paddingTop={1}
1290
+ paddingBottom={1}
1291
+ paddingLeft={2}
1292
+ backgroundColor={hover() ? theme.backgroundElement : theme.backgroundPanel}
1293
+ flexShrink={0}
1294
+ >
1295
+ <text fg={theme.text}>{text()?.text}</text>
1296
+ <Show when={files().length}>
1297
+ <box flexDirection="row" paddingBottom={metadataVisible() ? 1 : 0} paddingTop={1} gap={1} flexWrap="wrap">
1298
+ <For each={files()}>
1299
+ {(file) => {
1300
+ const bg = createMemo(() => {
1301
+ if (file.mime.startsWith("image/")) return theme.accent
1302
+ if (file.mime === "application/pdf") return theme.primary
1303
+ return theme.secondary
1304
+ })
1305
+ return (
1306
+ <text fg={theme.text}>
1307
+ <span style={{ bg: bg(), fg: theme.background }}> {MIME_BADGE[file.mime] ?? file.mime} </span>
1308
+ <span style={{ bg: theme.backgroundElement, fg: theme.textMuted }}> {file.filename} </span>
1309
+ </text>
1310
+ )
1311
+ }}
1312
+ </For>
1313
+ </box>
1314
+ </Show>
1315
+ <Show
1316
+ when={queued()}
1317
+ fallback={
1318
+ <Show when={ctx.showTimestamps()}>
1319
+ <text fg={theme.textMuted}>
1320
+ <span style={{ fg: theme.textMuted }}>
1321
+ {Locale.todayTimeOrDateTime(props.message.time.created)}
1322
+ </span>
1323
+ </text>
1324
+ </Show>
1325
+ }
1326
+ >
1327
+ <text fg={theme.textMuted}>
1328
+ <span style={{ bg: color(), fg: queuedFg(), bold: true }}> QUEUED </span>
1329
+ </text>
1330
+ </Show>
1331
+ </box>
1332
+ </box>
1333
+ </Show>
1334
+ </>
1335
+ )
1336
+ }
1337
+
1338
+ function AssistantMessage(props: { message: AssistantMessage; parts: Part[]; last: boolean }) {
1339
+ const ctx = use()
1340
+ const local = useLocal()
1341
+ const { theme } = useTheme()
1342
+ const sync = useSync()
1343
+ const toast = useToast()
1344
+ const renderer = useRenderer()
1345
+ const t = useLanguage().t
1346
+ const [copyHover, setCopyHover] = createSignal(false)
1347
+ const messages = createMemo(() => sync.data.message[props.message.sessionID]?.[props.message.agentID ?? "main"] ?? [])
1348
+ const model = createMemo(() => Model.name(ctx.providers(), props.message.providerID, props.message.modelID))
1349
+
1350
+ const final = createMemo(() => {
1351
+ return props.message.finish && props.message.finish !== "tool-calls"
1352
+ })
1353
+
1354
+ const duration = createMemo(() => {
1355
+ if (!final()) return 0
1356
+ if (!props.message.time.completed) return 0
1357
+ const user = messages().find((x) => x.role === "user" && x.id === props.message.parentID)
1358
+ if (!user || !user.time) return 0
1359
+ return props.message.time.completed - user.time.created
1360
+ })
1361
+
1362
+ const keybind = useKeybind()
1363
+
1364
+ const handleCopy = () => {
1365
+ if (renderer.getSelection()?.getSelectedText()) return
1366
+ const text = props.parts
1367
+ .filter((p) => p.type === "text")
1368
+ .map((p) => (p as TextPart).text)
1369
+ .join("\n")
1370
+ .trim()
1371
+ if (!text) return
1372
+ Clipboard.copy(text)
1373
+ .then(() => toast.show({ message: t("tui.toast.copied_to_clipboard"), variant: "success" }))
1374
+ .catch(() => toast.show({ message: "Failed to copy to clipboard", variant: "error" }))
1375
+ }
1376
+
1377
+ // Goal judge verdict for this specific turn, if the stop-condition judge
1378
+ // evaluated it. Rendered as a foldable per-turn marker so the user can trace
1379
+ // back which turn failed the check — without polluting the message stream.
1380
+ const verdict = createMemo(() => sync.data.session_goal?.[props.message.sessionID]?.verdicts?.[props.message.id])
1381
+ const [verdictOpen, setVerdictOpen] = createSignal(false)
1382
+ const verdictMark = createMemo(() => {
1383
+ const v = verdict()
1384
+ if (!v) return undefined
1385
+ if (v.error) return { icon: "!", fg: theme.textMuted, label: "Judge: error (stopped)" }
1386
+ if (v.ok) return { icon: "✓", fg: theme.success, label: "Judge: met" }
1387
+ if (v.impossible) return { icon: "⊘", fg: theme.error, label: "Judge: impossible" }
1388
+ return { icon: "⟳", fg: theme.warning, label: `Judge [round ${v.attempt}]: not met` }
1389
+ })
1390
+
1391
+ return (
1392
+ <>
1393
+ <For each={props.parts}>
1394
+ {(part, index) => {
1395
+ const component = createMemo(() => PART_MAPPING[part.type as keyof typeof PART_MAPPING])
1396
+ return (
1397
+ <Show when={component()}>
1398
+ <Dynamic
1399
+ last={index() === props.parts.length - 1}
1400
+ component={component()}
1401
+ part={part as any}
1402
+ message={props.message}
1403
+ />
1404
+ </Show>
1405
+ )
1406
+ }}
1407
+ </For>
1408
+ <Show when={props.parts.some((x) => x.type === "tool" && x.tool === "actor")}>
1409
+ <box paddingTop={1} paddingLeft={3}>
1410
+ <text fg={theme.text}>
1411
+ {keybind.print("session_child_first")}
1412
+ <span style={{ fg: theme.textMuted }}> view subagents</span>
1413
+ </text>
1414
+ </box>
1415
+ </Show>
1416
+ <Show when={props.message.error && props.message.error.name !== "MessageAbortedError"}>
1417
+ <ErrorBlock error={props.message.error!} />
1418
+ </Show>
1419
+ <Switch>
1420
+ <Match when={props.last || final() || props.message.error?.name === "MessageAbortedError"}>
1421
+ <box paddingLeft={3} flexDirection="row" justifyContent="space-between" marginTop={1}>
1422
+ <text>
1423
+ <span
1424
+ style={{
1425
+ fg:
1426
+ props.message.error?.name === "MessageAbortedError"
1427
+ ? theme.textMuted
1428
+ : local.agent.color(props.message.agent),
1429
+ }}
1430
+ >
1431
+ ▣{" "}
1432
+ </span>{" "}
1433
+ <span style={{ fg: theme.text }}>{Locale.titlecase(props.message.mode)}</span>
1434
+ <span style={{ fg: theme.textMuted }}> · {model()}</span>
1435
+ <Show when={duration()}>
1436
+ <span style={{ fg: theme.textMuted }}> · {Locale.duration(duration())}</span>
1437
+ </Show>
1438
+ <Show when={props.message.error?.name === "MessageAbortedError"}>
1439
+ <span style={{ fg: theme.textMuted }}> · interrupted</span>
1440
+ </Show>
1441
+ </text>
1442
+ <Show when={props.message.time.completed}>
1443
+ <box
1444
+ onMouseOver={() => setCopyHover(true)}
1445
+ onMouseOut={() => setCopyHover(false)}
1446
+ onMouseUp={handleCopy}
1447
+ >
1448
+ <text fg={copyHover() ? theme.text : theme.textMuted}>⎘ copy</text>
1449
+ </box>
1450
+ </Show>
1451
+ </box>
1452
+ </Match>
1453
+ </Switch>
1454
+ <Show when={verdictMark()}>
1455
+ {(mark) => (
1456
+ <box paddingLeft={3} onMouseUp={() => setVerdictOpen((x) => !x)}>
1457
+ <text>
1458
+ <span style={{ fg: theme.textMuted }}>{verdictOpen() ? "▼" : "▶"} </span>
1459
+ <span style={{ fg: mark().fg }}>
1460
+ {mark().icon} {mark().label}
1461
+ </span>
1462
+ </text>
1463
+ <Show when={verdictOpen()}>
1464
+ <box paddingLeft={2}>
1465
+ <text fg={theme.textMuted} wrapMode="word">
1466
+ {verdict()!.reason}
1467
+ </text>
1468
+ </box>
1469
+ </Show>
1470
+ </box>
1471
+ )}
1472
+ </Show>
1473
+ </>
1474
+ )
1475
+ }
1476
+
1477
+ const PART_MAPPING = {
1478
+ text: TextPart,
1479
+ tool: ToolPart,
1480
+ reasoning: ReasoningPart,
1481
+ }
1482
+
1483
+ type MessageError = NonNullable<AssistantMessage["error"]>
1484
+
1485
+ function errorBody(error: MessageError): string {
1486
+ if (error.name === "MessageOutputLengthError") return "Output length limit reached"
1487
+ return (error.data as { message?: string }).message ?? "Unknown error"
1488
+ }
1489
+
1490
+ function errorMeta(error: MessageError): string | undefined {
1491
+ if (error.name === "APIError") {
1492
+ const parts: string[] = []
1493
+ if (error.data.statusCode !== undefined) parts.push(`status ${error.data.statusCode}`)
1494
+ parts.push(error.data.isRetryable ? "retryable" : "non-retryable")
1495
+ return parts.join(" · ")
1496
+ }
1497
+ if (error.name === "ProviderAuthError") return `provider: ${error.data.providerID}`
1498
+ if (error.name === "StructuredOutputError") return `retries: ${error.data.retries}`
1499
+ return undefined
1500
+ }
1501
+
1502
+ function ErrorBlock(props: { error: MessageError }) {
1503
+ const { theme } = useTheme()
1504
+ const meta = createMemo(() => errorMeta(props.error))
1505
+ return (
1506
+ <box flexDirection="column" paddingLeft={3} marginTop={1}>
1507
+ <text fg={theme.error} wrapMode="word">
1508
+ <span style={{ fg: theme.error }}>✗ </span>
1509
+ {errorBody(props.error)}
1510
+ </text>
1511
+ <Show when={meta()}>
1512
+ <box paddingLeft={3}>
1513
+ <text fg={theme.textMuted} wrapMode="word">
1514
+ {meta()}
1515
+ </text>
1516
+ </box>
1517
+ </Show>
1518
+ </box>
1519
+ )
1520
+ }
1521
+
1522
+ function ReasoningPart(props: { last: boolean; part: ReasoningPart; message: AssistantMessage }) {
1523
+ const { theme, subtleSyntax } = useTheme()
1524
+ const ctx = use()
1525
+ const [expanded, setExpanded] = createSignal(false)
1526
+
1527
+ const content = createMemo(() => {
1528
+ return props.part.text.replace("[REDACTED]", "").trim()
1529
+ })
1530
+ const isDone = createMemo(() => props.part.time.end !== undefined)
1531
+ const inMinimal = createMemo(() => ctx.thinkingMode() === "hide")
1532
+ const duration = createMemo(() => {
1533
+ const end = props.part.time.end
1534
+ return end === undefined ? 0 : Math.max(0, end - props.part.time.start)
1535
+ })
1536
+ const summary = createMemo(() => reasoningSummary(content()))
1537
+
1538
+ const toggle = () => {
1539
+ if (!inMinimal()) return
1540
+ setExpanded((prev) => !prev)
1541
+ }
1542
+
1543
+ return (
1544
+ <Show when={content()}>
1545
+ <box id={"text-" + props.part.id} paddingLeft={3} marginTop={1} flexDirection="column" flexShrink={0}>
1546
+ <box onMouseUp={toggle}>
1547
+ <ReasoningHeader
1548
+ toggleable={inMinimal()}
1549
+ open={!inMinimal() || expanded()}
1550
+ done={isDone()}
1551
+ title={summary().title}
1552
+ duration={isDone() ? Locale.duration(duration()) : undefined}
1553
+ />
1554
+ </box>
1555
+ <Show when={(!inMinimal() || expanded()) && summary().body}>
1556
+ <box paddingLeft={inMinimal() ? 2 : 0} marginTop={1}>
1557
+ <code
1558
+ filetype="markdown"
1559
+ drawUnstyledText={false}
1560
+ streaming={true}
1561
+ syntaxStyle={subtleSyntax()}
1562
+ content={summary().body}
1563
+ conceal={ctx.conceal()}
1564
+ fg={theme.textMuted}
1565
+ />
1566
+ </box>
1567
+ </Show>
1568
+ </box>
1569
+ </Show>
1570
+ )
1571
+ }
1572
+
1573
+ function ReasoningHeader(props: {
1574
+ toggleable: boolean
1575
+ open: boolean
1576
+ done: boolean
1577
+ title: string | null
1578
+ duration?: string
1579
+ }) {
1580
+ const { theme } = useTheme()
1581
+ const fg = () =>
1582
+ props.open
1583
+ ? RGBA.fromValues(theme.warning.r, theme.warning.g, theme.warning.b, theme.thinkingOpacity)
1584
+ : theme.warning
1585
+
1586
+ return (
1587
+ <Switch>
1588
+ <Match when={!props.done}>
1589
+ <box flexDirection="row">
1590
+ <Spinner color={fg()}>{props.title ? "Thinking: " + props.title : "Thinking"}</Spinner>
1591
+ </box>
1592
+ </Match>
1593
+ <Match when={true}>
1594
+ <text fg={fg()} wrapMode="none">
1595
+ <Show when={props.toggleable}>
1596
+ <span>{props.open ? "- " : "+ "}</span>
1597
+ </Show>
1598
+ <span>Thought</span>
1599
+ <Show when={props.title || props.duration}>
1600
+ <span>: </span>
1601
+ </Show>
1602
+ <Show when={props.title}>
1603
+ <span>{props.title}</span>
1604
+ </Show>
1605
+ <Show when={props.duration}>
1606
+ <span>
1607
+ {props.title ? " · " : ""}
1608
+ {props.duration}
1609
+ </span>
1610
+ </Show>
1611
+ </text>
1612
+ </Match>
1613
+ </Switch>
1614
+ )
1615
+ }
1616
+
1617
+ function TextPart(props: { last: boolean; part: TextPart; message: AssistantMessage }) {
1618
+ const ctx = use()
1619
+ const { theme, syntax } = useTheme()
1620
+ return (
1621
+ <Show when={props.part.text.trim()}>
1622
+ <box id={"text-" + props.part.id} paddingLeft={3} marginTop={1} flexShrink={0}>
1623
+ <Switch>
1624
+ <Match when={Flag.MTYCODER_EXPERIMENTAL_MARKDOWN}>
1625
+ <markdown
1626
+ syntaxStyle={syntax()}
1627
+ streaming={true}
1628
+ content={props.part.text.trim()}
1629
+ conceal={ctx.conceal()}
1630
+ fg={theme.markdownText}
1631
+ bg={theme.background}
1632
+ />
1633
+ </Match>
1634
+ <Match when={!Flag.MTYCODER_EXPERIMENTAL_MARKDOWN}>
1635
+ <code
1636
+ filetype="markdown"
1637
+ drawUnstyledText={false}
1638
+ streaming={true}
1639
+ syntaxStyle={syntax()}
1640
+ content={props.part.text.trim()}
1641
+ conceal={ctx.conceal()}
1642
+ fg={theme.text}
1643
+ />
1644
+ </Match>
1645
+ </Switch>
1646
+ </box>
1647
+ </Show>
1648
+ )
1649
+ }
1650
+
1651
+ // Pending messages moved to individual tool pending functions
1652
+
1653
+ function ToolPart(props: { last: boolean; part: ToolPart; message: AssistantMessage }) {
1654
+ const ctx = use()
1655
+ const sync = useSync()
1656
+
1657
+ // Hide tool if showDetails is false and tool completed successfully
1658
+ const shouldHide = createMemo(() => {
1659
+ if (ctx.showDetails()) return false
1660
+ if (props.part.state.status !== "completed") return false
1661
+ return true
1662
+ })
1663
+
1664
+ const toolprops = {
1665
+ get metadata() {
1666
+ return props.part.state.status === "pending" ? {} : (props.part.state.metadata ?? {})
1667
+ },
1668
+ get input() {
1669
+ return props.part.state.input ?? {}
1670
+ },
1671
+ get output() {
1672
+ return props.part.state.status === "completed" ? props.part.state.output : undefined
1673
+ },
1674
+ get permission() {
1675
+ const permissions = sync.data.permission[props.message.sessionID] ?? []
1676
+ const permissionIndex = permissions.findIndex((x) => x.tool?.callID === props.part.callID)
1677
+ return permissions[permissionIndex]
1678
+ },
1679
+ get tool() {
1680
+ return props.part.tool
1681
+ },
1682
+ get part() {
1683
+ return props.part
1684
+ },
1685
+ }
1686
+
1687
+ return (
1688
+ <Show when={!shouldHide()}>
1689
+ <Switch>
1690
+ <Match when={props.part.tool === "bash"}>
1691
+ <Bash {...toolprops} />
1692
+ </Match>
1693
+ <Match when={props.part.tool === "glob"}>
1694
+ <Glob {...toolprops} />
1695
+ </Match>
1696
+ <Match when={props.part.tool === "read"}>
1697
+ <Read {...toolprops} />
1698
+ </Match>
1699
+ <Match when={props.part.tool === "grep"}>
1700
+ <Grep {...toolprops} />
1701
+ </Match>
1702
+ <Match when={props.part.tool === "webfetch"}>
1703
+ <WebFetch {...toolprops} />
1704
+ </Match>
1705
+ <Match when={props.part.tool === "codesearch"}>
1706
+ <CodeSearch {...toolprops} />
1707
+ </Match>
1708
+ <Match when={props.part.tool === "websearch"}>
1709
+ <WebSearch {...toolprops} />
1710
+ </Match>
1711
+ <Match when={props.part.tool === "write"}>
1712
+ <Write {...toolprops} />
1713
+ </Match>
1714
+ <Match when={props.part.tool === "edit"}>
1715
+ <Edit {...toolprops} />
1716
+ </Match>
1717
+ <Match when={props.part.tool === "actor"}>
1718
+ <Task {...toolprops} />
1719
+ </Match>
1720
+ <Match when={props.part.tool === "task"}>
1721
+ <WorkItemTask {...toolprops} />
1722
+ </Match>
1723
+ <Match when={props.part.tool === "apply_patch"}>
1724
+ <ApplyPatch {...toolprops} />
1725
+ </Match>
1726
+ <Match when={props.part.tool === "question"}>
1727
+ <Question {...toolprops} />
1728
+ </Match>
1729
+ <Match when={props.part.tool === "skill"}>
1730
+ <Skill {...toolprops} />
1731
+ </Match>
1732
+ <Match when={props.part.tool === "plan_exit"}>
1733
+ <PlanExit {...toolprops} />
1734
+ </Match>
1735
+ <Match when={true}>
1736
+ <GenericTool {...toolprops} />
1737
+ </Match>
1738
+ </Switch>
1739
+ </Show>
1740
+ )
1741
+ }
1742
+
1743
+ type ToolProps<T> = {
1744
+ input: Partial<Tool.InferParameters<T>>
1745
+ metadata: Partial<Tool.InferMetadata<T>>
1746
+ permission: Record<string, any>
1747
+ tool: string
1748
+ output?: string
1749
+ part: ToolPart
1750
+ }
1751
+ function PlanExit(props: ToolProps<any>) {
1752
+ const { theme } = useTheme()
1753
+ const dismissed = createMemo(
1754
+ () => props.part.state.status === "completed" && props.part.state.metadata?.switched === false,
1755
+ )
1756
+ const feedback = createMemo(() => (dismissed() ? props.metadata.feedback : undefined))
1757
+
1758
+ return (
1759
+ <>
1760
+ <InlineTool icon="⚙" pending="Asking..." complete={true} part={props.part} dismissed={dismissed()}>
1761
+ plan_exit
1762
+ </InlineTool>
1763
+ <Show when={feedback()}>
1764
+ <box paddingLeft={6}>
1765
+ <text fg={theme.textMuted}>{feedback()}</text>
1766
+ </box>
1767
+ </Show>
1768
+ </>
1769
+ )
1770
+ }
1771
+
1772
+ function GenericTool(props: ToolProps<any>) {
1773
+ const { theme } = useTheme()
1774
+ const ctx = use()
1775
+ const output = createMemo(() => props.output?.trim() ?? "")
1776
+ const [expanded, setExpanded] = createSignal(false)
1777
+ const lines = createMemo(() => output().split("\n"))
1778
+ const maxLines = 3
1779
+ const overflow = createMemo(() => lines().length > maxLines)
1780
+ const limited = createMemo(() => {
1781
+ if (expanded() || !overflow()) return output()
1782
+ return [...lines().slice(0, maxLines), "…"].join("\n")
1783
+ })
1784
+
1785
+ return (
1786
+ <Show
1787
+ when={props.output && ctx.showGenericToolOutput()}
1788
+ fallback={
1789
+ <InlineTool icon="⚙" pending="Writing command..." complete={true} part={props.part}>
1790
+ {props.tool} {input(props.input)}
1791
+ </InlineTool>
1792
+ }
1793
+ >
1794
+ <BlockTool
1795
+ title={`# ${props.tool} ${input(props.input)}`}
1796
+ part={props.part}
1797
+ onClick={overflow() ? () => setExpanded((prev) => !prev) : undefined}
1798
+ >
1799
+ <box gap={1}>
1800
+ <text fg={theme.text}>{limited()}</text>
1801
+ <Show when={overflow()}>
1802
+ <text fg={theme.textMuted}>{expanded() ? "Click to collapse" : "Click to expand"}</text>
1803
+ </Show>
1804
+ </box>
1805
+ </BlockTool>
1806
+ </Show>
1807
+ )
1808
+ }
1809
+
1810
+ // Inline renderer for the work-item `task` tool (distinct from <Task>, which
1811
+ // renders the subagent-spawning `actor` tool). Shows the operation as a concise
1812
+ // one-liner derived from the nested `{ operation: { action } }` discriminator,
1813
+ // so task create/start/done don't fall through to GenericTool's raw-JSON dump.
1814
+ function WorkItemTask(props: ToolProps<typeof TaskTool>) {
1815
+ const summary = createMemo(() => {
1816
+ const op = (props.input as { operation?: Record<string, any> }).operation
1817
+ if (!op || typeof op !== "object") return "task"
1818
+ const verb = typeof op.action === "string" ? op.action : "task"
1819
+ if (verb === "create") return op.summary ? `create "${op.summary}"` : "create"
1820
+ if (verb === "list") return op.status ? `list ${op.status}` : "list"
1821
+ if (op.id) return `${verb} ${op.id}`
1822
+ return verb
1823
+ })
1824
+ return (
1825
+ <InlineTool icon="#" pending="Updating tasks..." complete={true} part={props.part}>
1826
+ task {summary()}
1827
+ </InlineTool>
1828
+ )
1829
+ }
1830
+
1831
+ function CollapsibleError(props: { error: string; paddingLeft?: number }) {
1832
+ const { theme } = useTheme()
1833
+ const renderer = useRenderer()
1834
+ const [expanded, setExpanded] = createSignal(false)
1835
+
1836
+
1837
+ const lineCount = createMemo(() => props.error.split("\n").length)
1838
+
1839
+ return (
1840
+ <box
1841
+ paddingLeft={props.paddingLeft}
1842
+ onMouseUp={(evt) => {
1843
+ evt.stopPropagation()
1844
+ if (renderer.getSelection()?.getSelectedText()) return
1845
+ setExpanded((prev) => !prev)
1846
+ }}
1847
+ >
1848
+ <Show
1849
+ when={expanded()}
1850
+ fallback={
1851
+ <text fg={theme.error}>
1852
+ + Error ({lineCount()} {lineCount() === 1 ? "line" : "lines"})
1853
+ </text>
1854
+ }
1855
+ >
1856
+ <text fg={theme.error}>- Error</text>
1857
+ <box paddingLeft={2}>
1858
+ <text fg={theme.error}>{props.error}</text>
1859
+ </box>
1860
+ </Show>
1861
+ </box>
1862
+ )
1863
+ }
1864
+
1865
+ function InlineTool(props: {
1866
+ icon: string
1867
+ iconColor?: RGBA
1868
+ complete: any
1869
+ pending: string
1870
+ spinner?: boolean
1871
+ dismissed?: boolean
1872
+ children: JSX.Element
1873
+ part: ToolPart
1874
+ onClick?: () => void
1875
+ }) {
1876
+ const [margin, setMargin] = createSignal(0)
1877
+ const { theme } = useTheme()
1878
+ const ctx = use()
1879
+ const sync = useSync()
1880
+ const renderer = useRenderer()
1881
+ const [hover, setHover] = createSignal(false)
1882
+
1883
+ const permission = createMemo(() => {
1884
+ const callID = sync.data.permission[ctx.sessionID]?.at(0)?.tool?.callID
1885
+ if (!callID) return false
1886
+ return callID === props.part.callID
1887
+ })
1888
+
1889
+ const fg = createMemo(() => {
1890
+ if (permission()) return theme.warning
1891
+ if (hover() && props.onClick) return theme.text
1892
+ if (props.complete) return theme.textMuted
1893
+ return theme.text
1894
+ })
1895
+
1896
+ const error = createMemo(() => (props.part.state.status === "error" ? props.part.state.error : undefined))
1897
+
1898
+ const denied = createMemo(
1899
+ () =>
1900
+ error()?.includes("QuestionRejectedError") ||
1901
+ error()?.includes("rejected permission") ||
1902
+ error()?.includes("specified a rule") ||
1903
+ error()?.includes("user dismissed"),
1904
+ )
1905
+
1906
+ // Agent-recoverable failures (bad args, malformed call, unknown task/actor id)
1907
+ // are flagged on the error state. Render them muted (struck through, no red
1908
+ // block) like denials — the agent self-corrects; the user needn't be alarmed.
1909
+ const recoverable = createMemo(() => {
1910
+ const state = props.part.state
1911
+ return state.status === "error" && state.metadata?.recoverable === true
1912
+ })
1913
+
1914
+ return (
1915
+ <box
1916
+ marginTop={margin()}
1917
+ paddingLeft={3}
1918
+ onMouseOver={() => props.onClick && setHover(true)}
1919
+ onMouseOut={() => setHover(false)}
1920
+ onMouseUp={() => {
1921
+ if (renderer.getSelection()?.getSelectedText()) return
1922
+ props.onClick?.()
1923
+ }}
1924
+ renderBefore={function () {
1925
+ const el = this as BoxRenderable
1926
+ const parent = el.parent
1927
+ if (!parent) {
1928
+ return
1929
+ }
1930
+ if (el.height > 1) {
1931
+ setMargin(1)
1932
+ return
1933
+ }
1934
+ const children = parent.getChildren()
1935
+ const index = children.indexOf(el)
1936
+ const previous = children[index - 1]
1937
+ if (!previous) {
1938
+ setMargin(0)
1939
+ return
1940
+ }
1941
+ if (previous.height > 1 || previous.id.startsWith("text-")) {
1942
+ setMargin(1)
1943
+ return
1944
+ }
1945
+ }}
1946
+ >
1947
+ <Switch>
1948
+ <Match when={props.spinner}>
1949
+ <Spinner color={fg()} children={props.children} />
1950
+ </Match>
1951
+ <Match when={true}>
1952
+ <text paddingLeft={3} fg={fg()} attributes={denied() || recoverable() || props.dismissed ? TextAttributes.STRIKETHROUGH : undefined}>
1953
+ <Show fallback={<>~ {props.pending}</>} when={props.complete}>
1954
+ <span style={{ fg: props.iconColor }}>{props.icon}</span> {props.children}
1955
+ </Show>
1956
+ </text>
1957
+ </Match>
1958
+ </Switch>
1959
+ <Show when={error() && !denied() && !recoverable()}>
1960
+ <CollapsibleError error={error()!} paddingLeft={3} />
1961
+ </Show>
1962
+ </box>
1963
+ )
1964
+ }
1965
+
1966
+ function BlockTool(props: {
1967
+ title: string
1968
+ children: JSX.Element
1969
+ onClick?: () => void
1970
+ part?: ToolPart
1971
+ spinner?: boolean
1972
+ }) {
1973
+ const { theme } = useTheme()
1974
+ const renderer = useRenderer()
1975
+ const [hover, setHover] = createSignal(false)
1976
+ const error = createMemo(() => (props.part?.state.status === "error" ? props.part.state.error : undefined))
1977
+ return (
1978
+ <box
1979
+ border={["left"]}
1980
+ paddingTop={1}
1981
+ paddingBottom={1}
1982
+ paddingLeft={2}
1983
+ marginTop={1}
1984
+ gap={1}
1985
+ backgroundColor={hover() ? theme.backgroundMenu : theme.backgroundPanel}
1986
+ customBorderChars={SplitBorder.customBorderChars}
1987
+ borderColor={theme.background}
1988
+ onMouseOver={() => props.onClick && setHover(true)}
1989
+ onMouseOut={() => setHover(false)}
1990
+ onMouseUp={() => {
1991
+ if (renderer.getSelection()?.getSelectedText()) return
1992
+ props.onClick?.()
1993
+ }}
1994
+ >
1995
+ <Show
1996
+ when={props.spinner}
1997
+ fallback={
1998
+ <text paddingLeft={3} fg={theme.textMuted}>
1999
+ {props.title}
2000
+ </text>
2001
+ }
2002
+ >
2003
+ <Spinner color={theme.textMuted}>{props.title.replace(/^# /, "")}</Spinner>
2004
+ </Show>
2005
+ {props.children}
2006
+ <Show when={error()}>
2007
+ <CollapsibleError error={error()!} />
2008
+ </Show>
2009
+ </box>
2010
+ )
2011
+ }
2012
+
2013
+ const TOOL_COLLAPSE_MAX_LINES = 3
2014
+ const TOOL_COLLAPSE_MAX_LINE_LENGTH = 120
2015
+
2016
+ function displayLines(content: string) {
2017
+ if (!content) return []
2018
+ return content.replace(/\n$/, "").split("\n")
2019
+ }
2020
+
2021
+ function hasLongDisplayLine(content: string) {
2022
+ return displayLines(content).some((line) => line.length > TOOL_COLLAPSE_MAX_LINE_LENGTH)
2023
+ }
2024
+
2025
+ function Bash(props: ToolProps<typeof BashTool>) {
2026
+ const { theme } = useTheme()
2027
+ const sync = useSync()
2028
+ const isRunning = createMemo(() => props.part.state.status === "running")
2029
+ const output = createMemo(() => stripAnsi(props.metadata.output?.trim() ?? ""))
2030
+ const [expanded, setExpanded] = createSignal(false)
2031
+ const lines = createMemo(() => output().split("\n"))
2032
+ const overflow = createMemo(() => lines().length > 10)
2033
+ const limited = createMemo(() => {
2034
+ if (expanded() || !overflow()) return output()
2035
+ return [...lines().slice(0, 10), "…"].join("\n")
2036
+ })
2037
+
2038
+ const workdirDisplay = createMemo(() => {
2039
+ const workdir = props.input.workdir
2040
+ if (!workdir || workdir === ".") return undefined
2041
+
2042
+ const base = sync.path.directory
2043
+ if (!base) return undefined
2044
+
2045
+ const absolute = path.resolve(base, workdir)
2046
+ if (absolute === base) return undefined
2047
+
2048
+ const home = Global.Path.home
2049
+ if (!home) return absolute
2050
+
2051
+ const match = absolute === home || absolute.startsWith(home + path.sep)
2052
+ return match ? absolute.replace(home, "~") : absolute
2053
+ })
2054
+
2055
+ const title = createMemo(() => {
2056
+ const desc = props.input.description ?? "Shell"
2057
+ const wd = workdirDisplay()
2058
+ if (!wd) return `# ${desc}`
2059
+ if (desc.includes(wd)) return `# ${desc}`
2060
+ return `# ${desc} in ${wd}`
2061
+ })
2062
+
2063
+ return (
2064
+ <Switch>
2065
+ <Match when={props.metadata.output !== undefined}>
2066
+ <BlockTool
2067
+ title={title()}
2068
+ part={props.part}
2069
+ spinner={isRunning()}
2070
+ onClick={overflow() ? () => setExpanded((prev) => !prev) : undefined}
2071
+ >
2072
+ <box gap={1}>
2073
+ <text fg={theme.text}>$ {props.input.command}</text>
2074
+ <Show when={output()}>
2075
+ <text fg={theme.text}>{limited()}</text>
2076
+ </Show>
2077
+ <Show when={overflow()}>
2078
+ <text fg={theme.textMuted}>{expanded() ? "Click to collapse" : "Click to expand"}</text>
2079
+ </Show>
2080
+ </box>
2081
+ </BlockTool>
2082
+ </Match>
2083
+ <Match when={true}>
2084
+ <InlineTool icon="$" pending="Writing command..." complete={props.input.command} part={props.part}>
2085
+ {props.input.command}
2086
+ </InlineTool>
2087
+ </Match>
2088
+ </Switch>
2089
+ )
2090
+ }
2091
+
2092
+ function Write(props: ToolProps<typeof WriteTool>) {
2093
+ const { theme, syntax } = useTheme()
2094
+ const [expanded, setExpanded] = createSignal(false)
2095
+ const code = createMemo(() => {
2096
+ if (!props.input.content) return ""
2097
+ return props.input.content
2098
+ })
2099
+ const lineCount = createMemo(() => displayLines(code()).length)
2100
+ const collapsed = createMemo(() => lineCount() > TOOL_COLLAPSE_MAX_LINES || hasLongDisplayLine(code()))
2101
+
2102
+ return (
2103
+ <Switch>
2104
+ <Match when={props.metadata.diagnostics !== undefined}>
2105
+ <BlockTool
2106
+ title={"# Wrote " + normalizePath(props.input.filePath!)}
2107
+ part={props.part}
2108
+ onClick={collapsed() ? () => setExpanded((prev) => !prev) : undefined}
2109
+ >
2110
+ <Show
2111
+ when={!collapsed() || expanded()}
2112
+ fallback={
2113
+ <text fg={theme.textMuted}>
2114
+ Click to expand ({lineCount()} {lineCount() === 1 ? "line" : "lines"})
2115
+ </text>
2116
+ }
2117
+ >
2118
+ <line_number fg={theme.textMuted} minWidth={3} paddingRight={1}>
2119
+ <code
2120
+ conceal={false}
2121
+ fg={theme.text}
2122
+ filetype={filetype(props.input.filePath!)}
2123
+ syntaxStyle={syntax()}
2124
+ content={code()}
2125
+ />
2126
+ </line_number>
2127
+ <Show when={collapsed()}>
2128
+ <text fg={theme.textMuted}>Click to collapse</text>
2129
+ </Show>
2130
+ </Show>
2131
+ <Diagnostics diagnostics={props.metadata.diagnostics} filePath={props.input.filePath ?? ""} />
2132
+ </BlockTool>
2133
+ </Match>
2134
+ <Match when={true}>
2135
+ <InlineTool icon="←" pending="Preparing write..." complete={props.input.filePath} part={props.part}>
2136
+ Write {normalizePath(props.input.filePath!)}
2137
+ </InlineTool>
2138
+ </Match>
2139
+ </Switch>
2140
+ )
2141
+ }
2142
+
2143
+ function Glob(props: ToolProps<typeof GlobTool>) {
2144
+ return (
2145
+ <InlineTool icon="✱" pending="Finding files..." complete={props.input.pattern} part={props.part}>
2146
+ Glob "{props.input.pattern}" <Show when={props.input.path}>in {normalizePath(props.input.path)} </Show>
2147
+ <Show when={props.metadata.count}>
2148
+ ({props.metadata.count} {props.metadata.count === 1 ? "match" : "matches"})
2149
+ </Show>
2150
+ </InlineTool>
2151
+ )
2152
+ }
2153
+
2154
+ function Read(props: ToolProps<typeof ReadTool>) {
2155
+ const { theme } = useTheme()
2156
+ const isRunning = createMemo(() => props.part.state.status === "running")
2157
+ const loaded = createMemo(() => {
2158
+ if (props.part.state.status !== "completed") return []
2159
+ if (props.part.state.time.compacted) return []
2160
+ const value = props.metadata.loaded
2161
+ if (!value || !Array.isArray(value)) return []
2162
+ return value.filter((p): p is string => typeof p === "string")
2163
+ })
2164
+ return (
2165
+ <>
2166
+ <InlineTool
2167
+ icon="→"
2168
+ pending="Reading file..."
2169
+ complete={props.input.filePath}
2170
+ spinner={isRunning()}
2171
+ part={props.part}
2172
+ >
2173
+ Read {normalizePath(props.input.filePath!)} {input(props.input, ["filePath"])}
2174
+ </InlineTool>
2175
+ <For each={loaded()}>
2176
+ {(filepath) => (
2177
+ <box paddingLeft={3}>
2178
+ <text paddingLeft={3} fg={theme.textMuted}>
2179
+ ↳ Loaded {normalizePath(filepath)}
2180
+ </text>
2181
+ </box>
2182
+ )}
2183
+ </For>
2184
+ </>
2185
+ )
2186
+ }
2187
+
2188
+ function Grep(props: ToolProps<typeof GrepTool>) {
2189
+ return (
2190
+ <InlineTool icon="✱" pending="Searching content..." complete={props.input.pattern} part={props.part}>
2191
+ Grep "{props.input.pattern}" <Show when={props.input.path}>in {normalizePath(props.input.path)} </Show>
2192
+ <Show when={props.metadata.matches}>
2193
+ ({props.metadata.matches} {props.metadata.matches === 1 ? "match" : "matches"})
2194
+ </Show>
2195
+ </InlineTool>
2196
+ )
2197
+ }
2198
+
2199
+ function WebFetch(props: ToolProps<typeof WebFetchTool>) {
2200
+ return (
2201
+ <InlineTool icon="%" pending="Fetching from the web..." complete={props.input.url} part={props.part}>
2202
+ WebFetch {props.input.url}
2203
+ </InlineTool>
2204
+ )
2205
+ }
2206
+
2207
+ function CodeSearch(props: ToolProps<typeof CodeSearchTool>) {
2208
+ const metadata = props.metadata as { results?: number }
2209
+ return (
2210
+ <InlineTool icon="◇" pending="Searching code..." complete={props.input.query} part={props.part}>
2211
+ Exa Code Search "{props.input.query}" <Show when={metadata.results}>({metadata.results} results)</Show>
2212
+ </InlineTool>
2213
+ )
2214
+ }
2215
+
2216
+ function WebSearch(props: ToolProps<typeof WebSearchTool>) {
2217
+ const metadata = props.metadata as { numResults?: number }
2218
+ return (
2219
+ <InlineTool icon="◈" pending="Searching web..." complete={props.input.query} part={props.part}>
2220
+ Web Search "{props.input.query}" <Show when={metadata.numResults}>({metadata.numResults} results)</Show>
2221
+ </InlineTool>
2222
+ )
2223
+ }
2224
+
2225
+ function Task(props: ToolProps<typeof ActorTool>) {
2226
+ const route = useRoute()
2227
+ const sync = useSync()
2228
+
2229
+ const input = createMemo(() => {
2230
+ const raw = props.input as Partial<{ operation: { description: string; subagent_type: string } } & {
2231
+ description: string
2232
+ subagent_type: string
2233
+ }>
2234
+ return (raw?.operation ?? raw) as Partial<{ description: string; subagent_type: string }>
2235
+ })
2236
+
2237
+ const targetSession = createMemo(() => props.metadata.sessionId as string | undefined)
2238
+ const targetBucket = createMemo(() => (props.metadata.actorId as string | undefined) ?? "main")
2239
+
2240
+ createEffect(() => {
2241
+ const session = targetSession()
2242
+ const bucket = targetBucket()
2243
+ if (session && !sync.data.message[session]?.[bucket]?.length)
2244
+ void sync.session.sync(session)
2245
+ })
2246
+
2247
+ const messages = createMemo(() => sync.data.message[targetSession() ?? ""]?.[targetBucket()] ?? [])
2248
+
2249
+ const tools = createMemo(() => {
2250
+ return messages().flatMap((msg) =>
2251
+ (sync.data.part[msg.id] ?? [])
2252
+ .filter((part): part is ToolPart => part.type === "tool")
2253
+ .map((part) => ({ tool: part.tool, state: part.state })),
2254
+ )
2255
+ })
2256
+
2257
+ const current = createMemo(() =>
2258
+ tools().findLast((x) => (x.state.status === "running" || x.state.status === "completed") && x.state.title),
2259
+ )
2260
+
2261
+ const isRunning = createMemo(() => props.part.state.status === "running")
2262
+
2263
+ const duration = createMemo(() => {
2264
+ const first = messages().find((x) => x.role === "user")?.time.created
2265
+ const assistant = messages().findLast((x) => x.role === "assistant")?.time.completed
2266
+ if (!first || !assistant) return 0
2267
+ return assistant - first
2268
+ })
2269
+
2270
+ const content = createMemo(() => {
2271
+ if (!input().description) return ""
2272
+ let content = [`${Locale.titlecase(input().subagent_type ?? "General")} Task — ${input().description}`]
2273
+
2274
+ if (isRunning() && tools().length > 0) {
2275
+ // content[0] += ` · ${tools().length} toolcalls`
2276
+ if (current()) {
2277
+ const state = current()!.state
2278
+ const title = state.status === "running" || state.status === "completed" ? state.title : undefined
2279
+ content.push(`↳ ${Locale.titlecase(current()!.tool)} ${title}`)
2280
+ } else content.push(`↳ ${tools().length} toolcalls`)
2281
+ }
2282
+
2283
+ if (props.part.state.status === "completed") {
2284
+ content.push(`└ ${tools().length} toolcalls · ${Locale.duration(duration())}`)
2285
+ }
2286
+
2287
+ return content.join("\n")
2288
+ })
2289
+
2290
+ return (
2291
+ <InlineTool
2292
+ icon="│"
2293
+ spinner={isRunning()}
2294
+ complete={input().description}
2295
+ pending="Delegating..."
2296
+ part={props.part}
2297
+ onClick={() => {
2298
+ const targetSession = props.metadata.sessionId
2299
+ const targetActor = props.metadata.actorId as string | undefined
2300
+ if (!targetSession) return
2301
+ if (
2302
+ route.data.type === "session" &&
2303
+ targetSession === route.data.sessionID &&
2304
+ targetActor
2305
+ ) {
2306
+ // Subagent mode (shared sessionID): switch the agent slice in place.
2307
+ route.navigate({ ...route.data, agentID: targetActor })
2308
+ return
2309
+ }
2310
+ // Peer mode (different sessionID): navigate to peer's session, viewing its slice.
2311
+ route.navigate({ type: "session", sessionID: targetSession, agentID: targetActor })
2312
+ }}
2313
+ >
2314
+ {content()}
2315
+ </InlineTool>
2316
+ )
2317
+ }
2318
+
2319
+ function Edit(props: ToolProps<typeof EditTool>) {
2320
+ const ctx = use()
2321
+ const { theme, syntax } = useTheme()
2322
+
2323
+ const view = createMemo(() => {
2324
+ const diffStyle = ctx.tui.diff_style
2325
+ if (diffStyle === "stacked") return "unified"
2326
+ // Default to "auto" behavior
2327
+ return ctx.width > 120 ? "split" : "unified"
2328
+ })
2329
+
2330
+ const ft = createMemo(() => filetype(props.input.filePath))
2331
+
2332
+ const diffContent = createMemo(() => props.metadata.diff)
2333
+
2334
+ return (
2335
+ <Switch>
2336
+ <Match when={props.metadata.diff !== undefined}>
2337
+ <BlockTool title={"← Edit " + normalizePath(props.input.filePath!)} part={props.part}>
2338
+ <box paddingLeft={1}>
2339
+ <diff
2340
+ diff={diffContent()}
2341
+ view={view()}
2342
+ filetype={ft()}
2343
+ syntaxStyle={syntax()}
2344
+ showLineNumbers={true}
2345
+ width="100%"
2346
+ wrapMode={ctx.diffWrapMode()}
2347
+ fg={theme.text}
2348
+ addedBg={theme.diffAddedBg}
2349
+ removedBg={theme.diffRemovedBg}
2350
+ contextBg={theme.diffContextBg}
2351
+ addedSignColor={theme.diffHighlightAdded}
2352
+ removedSignColor={theme.diffHighlightRemoved}
2353
+ lineNumberFg={theme.diffLineNumber}
2354
+ lineNumberBg={theme.diffContextBg}
2355
+ addedLineNumberBg={theme.diffAddedLineNumberBg}
2356
+ removedLineNumberBg={theme.diffRemovedLineNumberBg}
2357
+ />
2358
+ </box>
2359
+ <Diagnostics diagnostics={props.metadata.diagnostics} filePath={props.input.filePath ?? ""} />
2360
+ </BlockTool>
2361
+ </Match>
2362
+ <Match when={true}>
2363
+ <InlineTool icon="←" pending="Preparing edit..." complete={props.input.filePath} part={props.part}>
2364
+ Edit {normalizePath(props.input.filePath!)} {input({ replaceAll: props.input.replaceAll })}
2365
+ </InlineTool>
2366
+ </Match>
2367
+ </Switch>
2368
+ )
2369
+ }
2370
+
2371
+ function ApplyPatch(props: ToolProps<typeof ApplyPatchTool>) {
2372
+ const ctx = use()
2373
+ const { theme, syntax } = useTheme()
2374
+ const [expanded, setExpanded] = createSignal<string[]>([])
2375
+
2376
+ const files = createMemo(() => props.metadata.files ?? [])
2377
+
2378
+ const view = createMemo(() => {
2379
+ const diffStyle = ctx.tui.diff_style
2380
+ if (diffStyle === "stacked") return "unified"
2381
+ return ctx.width > 120 ? "split" : "unified"
2382
+ })
2383
+
2384
+ function Diff(p: { diff: string; filePath: string }) {
2385
+ return (
2386
+ <box paddingLeft={1}>
2387
+ <diff
2388
+ diff={p.diff}
2389
+ view={view()}
2390
+ filetype={filetype(p.filePath)}
2391
+ syntaxStyle={syntax()}
2392
+ showLineNumbers={true}
2393
+ width="100%"
2394
+ wrapMode={ctx.diffWrapMode()}
2395
+ fg={theme.text}
2396
+ addedBg={theme.diffAddedBg}
2397
+ removedBg={theme.diffRemovedBg}
2398
+ contextBg={theme.diffContextBg}
2399
+ addedSignColor={theme.diffHighlightAdded}
2400
+ removedSignColor={theme.diffHighlightRemoved}
2401
+ lineNumberFg={theme.diffLineNumber}
2402
+ lineNumberBg={theme.diffContextBg}
2403
+ addedLineNumberBg={theme.diffAddedLineNumberBg}
2404
+ removedLineNumberBg={theme.diffRemovedLineNumberBg}
2405
+ />
2406
+ </box>
2407
+ )
2408
+ }
2409
+
2410
+ function title(file: { type: string; relativePath: string; filePath: string; deletions: number }) {
2411
+ if (file.type === "delete") return "# Deleted " + file.relativePath
2412
+ if (file.type === "add") return "# Created " + file.relativePath
2413
+ if (file.type === "move") return "# Moved " + normalizePath(file.filePath) + " → " + file.relativePath
2414
+ return "← Patched " + file.relativePath
2415
+ }
2416
+
2417
+ function toggle(filePath: string) {
2418
+ setExpanded((prev) => (prev.includes(filePath) ? prev.filter((item) => item !== filePath) : [...prev, filePath]))
2419
+ }
2420
+
2421
+ return (
2422
+ <Switch>
2423
+ <Match when={files().length > 0}>
2424
+ <For each={files()}>
2425
+ {(file) => {
2426
+ const open = createMemo(() => expanded().includes(file.filePath))
2427
+ const count = createMemo(() => file.additions + file.deletions)
2428
+ const collapsed = createMemo(() => count() > TOOL_COLLAPSE_MAX_LINES || hasLongDisplayLine(file.patch))
2429
+
2430
+ return (
2431
+ <BlockTool
2432
+ title={title(file)}
2433
+ part={props.part}
2434
+ onClick={file.type !== "delete" && collapsed() ? () => toggle(file.filePath) : undefined}
2435
+ >
2436
+ <Show
2437
+ when={file.type !== "delete"}
2438
+ fallback={
2439
+ <text fg={theme.diffRemoved}>
2440
+ -{file.deletions} line{file.deletions !== 1 ? "s" : ""}
2441
+ </text>
2442
+ }
2443
+ >
2444
+ <Show
2445
+ when={!collapsed() || open()}
2446
+ fallback={
2447
+ <text fg={theme.textMuted}>
2448
+ Click to expand ({count()} change{count() !== 1 ? "s" : ""})
2449
+ </text>
2450
+ }
2451
+ >
2452
+ <Diff diff={file.patch} filePath={file.filePath} />
2453
+ <Show when={collapsed()}>
2454
+ <text fg={theme.textMuted}>Click to collapse</text>
2455
+ </Show>
2456
+ </Show>
2457
+ <Diagnostics diagnostics={props.metadata.diagnostics} filePath={file.movePath ?? file.filePath} />
2458
+ </Show>
2459
+ </BlockTool>
2460
+ )
2461
+ }}
2462
+ </For>
2463
+ </Match>
2464
+ <Match when={true}>
2465
+ <InlineTool icon="%" pending="Preparing patch..." complete={false} part={props.part}>
2466
+ Patch
2467
+ </InlineTool>
2468
+ </Match>
2469
+ </Switch>
2470
+ )
2471
+ }
2472
+
2473
+ function Question(props: ToolProps<typeof QuestionTool>) {
2474
+ const { theme } = useTheme()
2475
+ const count = createMemo(() => props.input.questions?.length ?? 0)
2476
+
2477
+ function format(answer?: ReadonlyArray<string>) {
2478
+ if (!answer?.length) return "(no answer)"
2479
+ return answer.join(", ")
2480
+ }
2481
+
2482
+ return (
2483
+ <Switch>
2484
+ <Match when={props.metadata.answers}>
2485
+ <BlockTool title="# Questions" part={props.part}>
2486
+ <box gap={1}>
2487
+ <For each={props.input.questions ?? []}>
2488
+ {(q, i) => (
2489
+ <box flexDirection="column">
2490
+ <text fg={theme.textMuted}>{q.question}</text>
2491
+ <text fg={theme.text}>{format(props.metadata.answers?.[i()])}</text>
2492
+ </box>
2493
+ )}
2494
+ </For>
2495
+ </box>
2496
+ </BlockTool>
2497
+ </Match>
2498
+ <Match when={true}>
2499
+ <InlineTool icon="→" pending="Asking questions..." complete={count()} part={props.part}>
2500
+ Asked {count()} question{count() !== 1 ? "s" : ""}
2501
+ </InlineTool>
2502
+ </Match>
2503
+ </Switch>
2504
+ )
2505
+ }
2506
+
2507
+ function Skill(props: ToolProps<typeof SkillTool>) {
2508
+ return (
2509
+ <InlineTool icon="→" pending="Loading skill..." complete={props.input.name} part={props.part}>
2510
+ Skill "{props.input.name}"
2511
+ </InlineTool>
2512
+ )
2513
+ }
2514
+
2515
+ function Diagnostics(props: { diagnostics?: Record<string, Record<string, any>[]>; filePath: string }) {
2516
+ const { theme } = useTheme()
2517
+ const errors = createMemo(() => {
2518
+ const normalized = Filesystem.normalizePath(props.filePath)
2519
+ const arr = props.diagnostics?.[normalized] ?? []
2520
+ return arr.filter((x) => x.severity === 1).slice(0, 3)
2521
+ })
2522
+
2523
+ return (
2524
+ <Show when={errors().length}>
2525
+ <box>
2526
+ <For each={errors()}>
2527
+ {(diagnostic) => (
2528
+ <text fg={theme.error}>
2529
+ Error [{diagnostic.range.start.line + 1}:{diagnostic.range.start.character + 1}] {diagnostic.message}
2530
+ </text>
2531
+ )}
2532
+ </For>
2533
+ </box>
2534
+ </Show>
2535
+ )
2536
+ }
2537
+
2538
+ function normalizePath(input?: string) {
2539
+ if (!input) return ""
2540
+
2541
+ const cwd = process.cwd()
2542
+ const absolute = path.isAbsolute(input) ? input : path.resolve(cwd, input)
2543
+ const relative = path.relative(cwd, absolute)
2544
+
2545
+ if (!relative) return "."
2546
+ if (!relative.startsWith("..")) return relative
2547
+
2548
+ // outside cwd - use absolute
2549
+ return absolute
2550
+ }
2551
+
2552
+ function input(input: Record<string, any>, omit?: string[]): string {
2553
+ const primitives = Object.entries(input).filter(([key, value]) => {
2554
+ if (omit?.includes(key)) return false
2555
+ return typeof value === "string" || typeof value === "number" || typeof value === "boolean"
2556
+ })
2557
+ if (primitives.length === 0) return ""
2558
+ return `[${primitives.map(([key, value]) => `${key}=${value}`).join(", ")}]`
2559
+ }
2560
+
2561
+ function filetype(input?: string) {
2562
+ if (!input) return "none"
2563
+ const ext = path.extname(input)
2564
+ const language = LANGUAGE_EXTENSIONS[ext]
2565
+ if (["typescriptreact", "javascriptreact", "javascript"].includes(language)) return "typescript"
2566
+ return language
2567
+ }