oma-coding-agent 1.1.4

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 (1298) hide show
  1. package/CHANGELOG.md +12164 -0
  2. package/README.md +35 -0
  3. package/dist/cli.js +18266 -0
  4. package/examples/README.md +21 -0
  5. package/examples/custom-tools/README.md +104 -0
  6. package/examples/custom-tools/hello/index.ts +20 -0
  7. package/examples/extensions/README.md +142 -0
  8. package/examples/extensions/api-demo.ts +79 -0
  9. package/examples/extensions/chalk-logger.ts +25 -0
  10. package/examples/extensions/hello.ts +31 -0
  11. package/examples/extensions/pirate.ts +43 -0
  12. package/examples/extensions/plan-mode.ts +549 -0
  13. package/examples/extensions/reload-runtime.ts +38 -0
  14. package/examples/extensions/thinking-note.ts +13 -0
  15. package/examples/extensions/tools.ts +145 -0
  16. package/examples/extensions/with-deps/index.ts +36 -0
  17. package/examples/extensions/with-deps/package-lock.json +31 -0
  18. package/examples/extensions/with-deps/package.json +17 -0
  19. package/examples/hooks/README.md +56 -0
  20. package/examples/hooks/auto-commit-on-exit.ts +48 -0
  21. package/examples/hooks/confirm-destructive.ts +58 -0
  22. package/examples/hooks/custom-compaction.ts +115 -0
  23. package/examples/hooks/dirty-repo-guard.ts +51 -0
  24. package/examples/hooks/file-trigger.ts +40 -0
  25. package/examples/hooks/git-checkpoint.ts +52 -0
  26. package/examples/hooks/handoff.ts +149 -0
  27. package/examples/hooks/permission-gate.ts +33 -0
  28. package/examples/hooks/protected-paths.ts +29 -0
  29. package/examples/hooks/qna.ts +118 -0
  30. package/examples/hooks/status-line.ts +39 -0
  31. package/examples/sdk/01-minimal.ts +21 -0
  32. package/examples/sdk/02-custom-model.ts +49 -0
  33. package/examples/sdk/03-custom-prompt.ts +46 -0
  34. package/examples/sdk/04-skills.ts +43 -0
  35. package/examples/sdk/06-extensions.ts +82 -0
  36. package/examples/sdk/06-hooks.ts +61 -0
  37. package/examples/sdk/07-context-files.ts +35 -0
  38. package/examples/sdk/08-prompt-templates.ts +41 -0
  39. package/examples/sdk/08-slash-commands.ts +46 -0
  40. package/examples/sdk/09-api-keys-and-oauth.ts +54 -0
  41. package/examples/sdk/11-sessions.ts +47 -0
  42. package/examples/sdk/12-redis-sessions.ts +54 -0
  43. package/examples/sdk/13-sql-sessions.ts +61 -0
  44. package/examples/sdk/README.md +172 -0
  45. package/package.json +573 -0
  46. package/scripts/bench-guard.ts +71 -0
  47. package/scripts/build-binary.ts +108 -0
  48. package/scripts/bundle-dist.ts +110 -0
  49. package/scripts/embed-mupdf-wasm.ts +67 -0
  50. package/scripts/format-prompts.ts +68 -0
  51. package/scripts/generate-docs-index.ts +56 -0
  52. package/scripts/generate-share-viewer.ts +34 -0
  53. package/scripts/measure-prompt-tokens.ts +63 -0
  54. package/scripts/omp +42 -0
  55. package/scripts/omp.ts +19 -0
  56. package/src/advisor/__tests__/advisor.test.ts +915 -0
  57. package/src/advisor/advise-tool.ts +165 -0
  58. package/src/advisor/index.ts +4 -0
  59. package/src/advisor/runtime.ts +270 -0
  60. package/src/advisor/transcript-recorder.ts +136 -0
  61. package/src/advisor/watchdog.ts +83 -0
  62. package/src/async/index.ts +1 -0
  63. package/src/async/job-manager.ts +674 -0
  64. package/src/auto-thinking/classifier.ts +190 -0
  65. package/src/autolearn/controller.ts +139 -0
  66. package/src/autolearn/managed-skills.ts +255 -0
  67. package/src/autoresearch/command-resume.md +14 -0
  68. package/src/autoresearch/dashboard.ts +436 -0
  69. package/src/autoresearch/git.ts +319 -0
  70. package/src/autoresearch/helpers.ts +218 -0
  71. package/src/autoresearch/index.ts +536 -0
  72. package/src/autoresearch/prompt-setup.md +43 -0
  73. package/src/autoresearch/prompt.md +103 -0
  74. package/src/autoresearch/resume-message.md +10 -0
  75. package/src/autoresearch/state.ts +273 -0
  76. package/src/autoresearch/storage.ts +700 -0
  77. package/src/autoresearch/tools/init-experiment.ts +269 -0
  78. package/src/autoresearch/tools/log-experiment.ts +521 -0
  79. package/src/autoresearch/tools/run-experiment.ts +407 -0
  80. package/src/autoresearch/tools/update-notes.ts +109 -0
  81. package/src/autoresearch/types.ts +168 -0
  82. package/src/capability/context-file.ts +44 -0
  83. package/src/capability/extension-module.ts +34 -0
  84. package/src/capability/extension.ts +47 -0
  85. package/src/capability/fs.ts +117 -0
  86. package/src/capability/hook.ts +40 -0
  87. package/src/capability/index.ts +436 -0
  88. package/src/capability/instruction.ts +37 -0
  89. package/src/capability/mcp.ts +76 -0
  90. package/src/capability/prompt.ts +35 -0
  91. package/src/capability/rule-buckets.ts +66 -0
  92. package/src/capability/rule.ts +261 -0
  93. package/src/capability/settings.ts +34 -0
  94. package/src/capability/skill.ts +63 -0
  95. package/src/capability/slash-command.ts +40 -0
  96. package/src/capability/ssh.ts +41 -0
  97. package/src/capability/system-prompt.ts +34 -0
  98. package/src/capability/tool.ts +38 -0
  99. package/src/capability/types.ts +168 -0
  100. package/src/cli/agents-cli.ts +138 -0
  101. package/src/cli/args.ts +361 -0
  102. package/src/cli/auth-broker-cli.ts +893 -0
  103. package/src/cli/auth-gateway-cli.ts +608 -0
  104. package/src/cli/bench-cli.ts +552 -0
  105. package/src/cli/classify-install-target.ts +76 -0
  106. package/src/cli/claude-trace-cli.ts +795 -0
  107. package/src/cli/commands/init-xdg.ts +27 -0
  108. package/src/cli/completion-gen.ts +550 -0
  109. package/src/cli/config-cli.ts +418 -0
  110. package/src/cli/dry-balance-cli.ts +858 -0
  111. package/src/cli/extension-flags.ts +48 -0
  112. package/src/cli/file-processor.ts +133 -0
  113. package/src/cli/flag-tables.ts +280 -0
  114. package/src/cli/gallery-cli.ts +231 -0
  115. package/src/cli/gallery-fixtures/agentic.ts +407 -0
  116. package/src/cli/gallery-fixtures/codeintel.ts +187 -0
  117. package/src/cli/gallery-fixtures/edit.ts +194 -0
  118. package/src/cli/gallery-fixtures/fs.ts +220 -0
  119. package/src/cli/gallery-fixtures/index.ts +40 -0
  120. package/src/cli/gallery-fixtures/interaction.ts +49 -0
  121. package/src/cli/gallery-fixtures/memory.ts +81 -0
  122. package/src/cli/gallery-fixtures/misc.ts +250 -0
  123. package/src/cli/gallery-fixtures/search.ts +213 -0
  124. package/src/cli/gallery-fixtures/shell.ts +167 -0
  125. package/src/cli/gallery-fixtures/types.ts +57 -0
  126. package/src/cli/gallery-fixtures/web.ts +158 -0
  127. package/src/cli/gallery-screenshot.ts +279 -0
  128. package/src/cli/grep-cli.ts +160 -0
  129. package/src/cli/grievances-cli.ts +256 -0
  130. package/src/cli/initial-message.ts +58 -0
  131. package/src/cli/models-cli.ts +427 -0
  132. package/src/cli/plugin-cli.ts +996 -0
  133. package/src/cli/profile-alias.ts +338 -0
  134. package/src/cli/profile-bootstrap.ts +243 -0
  135. package/src/cli/read-cli.ts +57 -0
  136. package/src/cli/session-picker.ts +80 -0
  137. package/src/cli/setup-cli.ts +332 -0
  138. package/src/cli/setup-model-picker.ts +43 -0
  139. package/src/cli/shell-cli.ts +176 -0
  140. package/src/cli/ssh-cli.ts +179 -0
  141. package/src/cli/startup-cwd.ts +58 -0
  142. package/src/cli/stats-cli.ts +229 -0
  143. package/src/cli/tiny-models-cli.ts +127 -0
  144. package/src/cli/ttsr-cli.ts +995 -0
  145. package/src/cli/update-cli.ts +671 -0
  146. package/src/cli/usage-cli.ts +774 -0
  147. package/src/cli/web-search-cli.ts +132 -0
  148. package/src/cli/worktree-cli.ts +291 -0
  149. package/src/cli-commands.ts +85 -0
  150. package/src/cli.ts +326 -0
  151. package/src/collab/crypto.ts +63 -0
  152. package/src/collab/guest.ts +450 -0
  153. package/src/collab/host.ts +577 -0
  154. package/src/collab/protocol.ts +274 -0
  155. package/src/collab/relay-client.ts +216 -0
  156. package/src/commands/acp.ts +24 -0
  157. package/src/commands/agents.ts +57 -0
  158. package/src/commands/auth-broker.ts +99 -0
  159. package/src/commands/auth-gateway.ts +69 -0
  160. package/src/commands/bench.ts +42 -0
  161. package/src/commands/commit.ts +46 -0
  162. package/src/commands/complete.ts +66 -0
  163. package/src/commands/completions.ts +60 -0
  164. package/src/commands/config.ts +51 -0
  165. package/src/commands/dry-balance.ts +43 -0
  166. package/src/commands/gallery.ts +52 -0
  167. package/src/commands/grep.ts +48 -0
  168. package/src/commands/grievances.ts +51 -0
  169. package/src/commands/install.ts +107 -0
  170. package/src/commands/join.ts +39 -0
  171. package/src/commands/launch.ts +182 -0
  172. package/src/commands/models.ts +61 -0
  173. package/src/commands/plugin.ts +78 -0
  174. package/src/commands/read.ts +38 -0
  175. package/src/commands/say.ts +102 -0
  176. package/src/commands/setup.ts +67 -0
  177. package/src/commands/shell.ts +29 -0
  178. package/src/commands/ssh.ts +60 -0
  179. package/src/commands/stats.ts +29 -0
  180. package/src/commands/tiny-models.ts +36 -0
  181. package/src/commands/token.ts +108 -0
  182. package/src/commands/ttsr.ts +125 -0
  183. package/src/commands/update.ts +21 -0
  184. package/src/commands/usage.ts +43 -0
  185. package/src/commands/web-search.ts +42 -0
  186. package/src/commands/worktree.ts +56 -0
  187. package/src/commit/agentic/agent.ts +318 -0
  188. package/src/commit/agentic/fallback.ts +96 -0
  189. package/src/commit/agentic/index.ts +355 -0
  190. package/src/commit/agentic/prompts/analyze-file.md +22 -0
  191. package/src/commit/agentic/prompts/session-user.md +25 -0
  192. package/src/commit/agentic/prompts/split-confirm.md +1 -0
  193. package/src/commit/agentic/prompts/system.md +38 -0
  194. package/src/commit/agentic/state.ts +60 -0
  195. package/src/commit/agentic/tools/analyze-file.ts +149 -0
  196. package/src/commit/agentic/tools/git-file-diff.ts +191 -0
  197. package/src/commit/agentic/tools/git-hunk.ts +52 -0
  198. package/src/commit/agentic/tools/git-overview.ts +81 -0
  199. package/src/commit/agentic/tools/index.ts +54 -0
  200. package/src/commit/agentic/tools/propose-changelog.ts +147 -0
  201. package/src/commit/agentic/tools/propose-commit.ts +109 -0
  202. package/src/commit/agentic/tools/recent-commits.ts +81 -0
  203. package/src/commit/agentic/tools/schemas.ts +11 -0
  204. package/src/commit/agentic/tools/split-commit.ts +241 -0
  205. package/src/commit/agentic/topo-sort.ts +44 -0
  206. package/src/commit/agentic/trivial.ts +51 -0
  207. package/src/commit/agentic/validation.ts +183 -0
  208. package/src/commit/analysis/conventional.ts +64 -0
  209. package/src/commit/analysis/index.ts +4 -0
  210. package/src/commit/analysis/scope.ts +242 -0
  211. package/src/commit/analysis/summary.ts +107 -0
  212. package/src/commit/analysis/validation.ts +66 -0
  213. package/src/commit/changelog/detect.ts +40 -0
  214. package/src/commit/changelog/generate.ts +101 -0
  215. package/src/commit/changelog/index.ts +234 -0
  216. package/src/commit/changelog/parse.ts +44 -0
  217. package/src/commit/cli.ts +85 -0
  218. package/src/commit/git/diff.ts +148 -0
  219. package/src/commit/index.ts +5 -0
  220. package/src/commit/map-reduce/index.ts +69 -0
  221. package/src/commit/map-reduce/map-phase.ts +193 -0
  222. package/src/commit/map-reduce/reduce-phase.ts +49 -0
  223. package/src/commit/map-reduce/utils.ts +9 -0
  224. package/src/commit/message.ts +11 -0
  225. package/src/commit/model-selection.ts +89 -0
  226. package/src/commit/pipeline.ts +243 -0
  227. package/src/commit/prompts/analysis-system.md +148 -0
  228. package/src/commit/prompts/analysis-user.md +38 -0
  229. package/src/commit/prompts/changelog-system.md +50 -0
  230. package/src/commit/prompts/changelog-user.md +18 -0
  231. package/src/commit/prompts/file-observer-system.md +24 -0
  232. package/src/commit/prompts/file-observer-user.md +8 -0
  233. package/src/commit/prompts/reduce-system.md +50 -0
  234. package/src/commit/prompts/reduce-user.md +17 -0
  235. package/src/commit/prompts/summary-retry.md +3 -0
  236. package/src/commit/prompts/summary-system.md +38 -0
  237. package/src/commit/prompts/summary-user.md +13 -0
  238. package/src/commit/prompts/types-description.md +2 -0
  239. package/src/commit/shared-llm.ts +70 -0
  240. package/src/commit/types.ts +118 -0
  241. package/src/commit/utils/exclusions.ts +42 -0
  242. package/src/commit/utils.ts +58 -0
  243. package/src/config/api-key-resolver.ts +67 -0
  244. package/src/config/append-only-context-mode.ts +76 -0
  245. package/src/config/config-file.ts +315 -0
  246. package/src/config/file-lock.ts +164 -0
  247. package/src/config/keybindings.ts +634 -0
  248. package/src/config/mcp-schema.json +238 -0
  249. package/src/config/model-discovery.ts +589 -0
  250. package/src/config/model-registry.ts +2260 -0
  251. package/src/config/model-resolver.ts +1819 -0
  252. package/src/config/model-roles.ts +99 -0
  253. package/src/config/models-config-schema.ts +266 -0
  254. package/src/config/models-config.ts +131 -0
  255. package/src/config/prompt-templates.ts +185 -0
  256. package/src/config/resolve-config-value.ts +94 -0
  257. package/src/config/settings-schema.ts +4740 -0
  258. package/src/config/settings.ts +1243 -0
  259. package/src/config.ts +242 -0
  260. package/src/cursor.ts +340 -0
  261. package/src/dap/client.ts +760 -0
  262. package/src/dap/config.ts +189 -0
  263. package/src/dap/defaults.json +212 -0
  264. package/src/dap/index.ts +4 -0
  265. package/src/dap/session.ts +1441 -0
  266. package/src/dap/types.ts +610 -0
  267. package/src/debug/index.ts +559 -0
  268. package/src/debug/log-formatting.ts +58 -0
  269. package/src/debug/log-viewer.ts +908 -0
  270. package/src/debug/profiler.ts +162 -0
  271. package/src/debug/protocol-probe.ts +267 -0
  272. package/src/debug/raw-sse-buffer.ts +294 -0
  273. package/src/debug/raw-sse.ts +292 -0
  274. package/src/debug/remote-debugger.ts +151 -0
  275. package/src/debug/report-bundle.ts +375 -0
  276. package/src/debug/system-info.ts +111 -0
  277. package/src/debug/terminal-info.ts +124 -0
  278. package/src/discovery/agents-md.ts +67 -0
  279. package/src/discovery/agents.ts +230 -0
  280. package/src/discovery/at-imports.ts +273 -0
  281. package/src/discovery/builtin-defaults.ts +39 -0
  282. package/src/discovery/builtin-rules/index.ts +63 -0
  283. package/src/discovery/builtin-rules/low-end/no-hallucinated-apis.md +14 -0
  284. package/src/discovery/builtin-rules/low-end/no-hallucinated-paths.md +14 -0
  285. package/src/discovery/builtin-rules/low-end/no-premature-completion.md +14 -0
  286. package/src/discovery/builtin-rules/rs-box-leak.md +48 -0
  287. package/src/discovery/builtin-rules/rs-future-prelude.md +23 -0
  288. package/src/discovery/builtin-rules/rs-lazylock.md +51 -0
  289. package/src/discovery/builtin-rules/rs-match-ergonomics.md +67 -0
  290. package/src/discovery/builtin-rules/rs-parking-lot.md +44 -0
  291. package/src/discovery/builtin-rules/rs-result-type.md +19 -0
  292. package/src/discovery/builtin-rules/ts-bare-catch.md +38 -0
  293. package/src/discovery/builtin-rules/ts-import-type.md +42 -0
  294. package/src/discovery/builtin-rules/ts-no-any.md +65 -0
  295. package/src/discovery/builtin-rules/ts-no-deprecated-leftovers.md +44 -0
  296. package/src/discovery/builtin-rules/ts-no-dynamic-import.md +39 -0
  297. package/src/discovery/builtin-rules/ts-no-inline-cast-access.md +55 -0
  298. package/src/discovery/builtin-rules/ts-no-return-type.md +44 -0
  299. package/src/discovery/builtin-rules/ts-no-test-timers.md +55 -0
  300. package/src/discovery/builtin-rules/ts-no-tiny-functions.md +51 -0
  301. package/src/discovery/builtin-rules/ts-promise-with-resolvers.md +65 -0
  302. package/src/discovery/builtin-rules/ts-redundant-clear-guard.md +75 -0
  303. package/src/discovery/builtin-rules/ts-set-map.md +28 -0
  304. package/src/discovery/builtin.ts +934 -0
  305. package/src/discovery/claude-plugins.ts +386 -0
  306. package/src/discovery/claude.ts +584 -0
  307. package/src/discovery/cline.ts +83 -0
  308. package/src/discovery/codex.ts +522 -0
  309. package/src/discovery/cursor.ts +220 -0
  310. package/src/discovery/gemini.ts +383 -0
  311. package/src/discovery/github.ts +337 -0
  312. package/src/discovery/helpers.ts +1092 -0
  313. package/src/discovery/index.ts +81 -0
  314. package/src/discovery/mcp-json.ts +172 -0
  315. package/src/discovery/omp-extension-roots.ts +190 -0
  316. package/src/discovery/omp-plugins.ts +383 -0
  317. package/src/discovery/opencode.ts +398 -0
  318. package/src/discovery/plugin-dir-roots.ts +28 -0
  319. package/src/discovery/ssh.ts +153 -0
  320. package/src/discovery/substitute-plugin-root.ts +29 -0
  321. package/src/discovery/vscode.ts +105 -0
  322. package/src/discovery/windsurf.ts +147 -0
  323. package/src/edit/apply-patch/index.ts +87 -0
  324. package/src/edit/apply-patch/parser.ts +174 -0
  325. package/src/edit/diff.ts +999 -0
  326. package/src/edit/file-snapshot-store.ts +143 -0
  327. package/src/edit/hashline/block-resolver.ts +33 -0
  328. package/src/edit/hashline/diff.ts +290 -0
  329. package/src/edit/hashline/execute.ts +237 -0
  330. package/src/edit/hashline/filesystem.ts +130 -0
  331. package/src/edit/hashline/index.ts +5 -0
  332. package/src/edit/hashline/noop-loop-guard.ts +99 -0
  333. package/src/edit/hashline/params.ts +19 -0
  334. package/src/edit/index.ts +620 -0
  335. package/src/edit/modes/apply-patch.lark +19 -0
  336. package/src/edit/modes/apply-patch.ts +53 -0
  337. package/src/edit/modes/patch.ts +1888 -0
  338. package/src/edit/modes/replace.ts +1133 -0
  339. package/src/edit/normalize.ts +345 -0
  340. package/src/edit/notebook.ts +242 -0
  341. package/src/edit/read-file.ts +25 -0
  342. package/src/edit/renderer.ts +823 -0
  343. package/src/edit/streaming.ts +517 -0
  344. package/src/eval/__tests__/agent-bridge.test.ts +769 -0
  345. package/src/eval/__tests__/bridge-timeout.test.ts +64 -0
  346. package/src/eval/__tests__/budget-bridge.test.ts +69 -0
  347. package/src/eval/__tests__/completion-bridge.test.ts +412 -0
  348. package/src/eval/__tests__/helpers-local-roots.test.ts +58 -0
  349. package/src/eval/__tests__/idle-timeout.test.ts +80 -0
  350. package/src/eval/__tests__/js-context-manager.test.ts +291 -0
  351. package/src/eval/__tests__/kernel-spawn.test.ts +103 -0
  352. package/src/eval/__tests__/prelude-agent.test.ts +73 -0
  353. package/src/eval/agent-bridge.ts +319 -0
  354. package/src/eval/backend.ts +71 -0
  355. package/src/eval/bridge-timeout.ts +44 -0
  356. package/src/eval/budget-bridge.ts +48 -0
  357. package/src/eval/completion-bridge.ts +211 -0
  358. package/src/eval/concurrency-bridge.ts +34 -0
  359. package/src/eval/idle-timeout.ts +91 -0
  360. package/src/eval/index.ts +4 -0
  361. package/src/eval/js/context-manager.ts +621 -0
  362. package/src/eval/js/executor.ts +173 -0
  363. package/src/eval/js/index.ts +51 -0
  364. package/src/eval/js/shared/helpers.ts +283 -0
  365. package/src/eval/js/shared/indirect-eval.ts +30 -0
  366. package/src/eval/js/shared/local-module-loader.ts +342 -0
  367. package/src/eval/js/shared/prelude.ts +2 -0
  368. package/src/eval/js/shared/prelude.txt +307 -0
  369. package/src/eval/js/shared/rewrite-imports.ts +532 -0
  370. package/src/eval/js/shared/runtime.ts +580 -0
  371. package/src/eval/js/shared/types.ts +18 -0
  372. package/src/eval/js/tool-bridge.ts +163 -0
  373. package/src/eval/js/worker-core.ts +151 -0
  374. package/src/eval/js/worker-entry.ts +37 -0
  375. package/src/eval/js/worker-protocol.ts +47 -0
  376. package/src/eval/py/__tests__/prelude.test.ts +19 -0
  377. package/src/eval/py/display.ts +71 -0
  378. package/src/eval/py/executor.ts +742 -0
  379. package/src/eval/py/index.ts +68 -0
  380. package/src/eval/py/kernel.ts +748 -0
  381. package/src/eval/py/prelude.py +683 -0
  382. package/src/eval/py/prelude.ts +3 -0
  383. package/src/eval/py/runner.py +1177 -0
  384. package/src/eval/py/runtime.ts +276 -0
  385. package/src/eval/py/spawn-options.ts +126 -0
  386. package/src/eval/py/tool-bridge.ts +182 -0
  387. package/src/eval/session-id.ts +8 -0
  388. package/src/eval/types.ts +48 -0
  389. package/src/exa/index.ts +2 -0
  390. package/src/exa/mcp-client.ts +370 -0
  391. package/src/exa/types.ts +69 -0
  392. package/src/exec/bash-executor.ts +434 -0
  393. package/src/exec/exec.ts +53 -0
  394. package/src/exec/non-interactive-env.ts +119 -0
  395. package/src/export/custom-share.ts +65 -0
  396. package/src/export/html/index.ts +266 -0
  397. package/src/export/html/share-loader.js +102 -0
  398. package/src/export/html/template.css +1337 -0
  399. package/src/export/html/template.html +49 -0
  400. package/src/export/html/template.js +1626 -0
  401. package/src/export/html/tool-views.generated.js +37 -0
  402. package/src/export/html/vendor/highlight.min.js +1213 -0
  403. package/src/export/html/vendor/marked.min.js +6 -0
  404. package/src/export/share.ts +268 -0
  405. package/src/export/ttsr.ts +583 -0
  406. package/src/extensibility/custom-commands/bundled/ci-green/index.ts +54 -0
  407. package/src/extensibility/custom-commands/bundled/review/index.ts +698 -0
  408. package/src/extensibility/custom-commands/index.ts +2 -0
  409. package/src/extensibility/custom-commands/loader.ts +242 -0
  410. package/src/extensibility/custom-commands/types.ts +119 -0
  411. package/src/extensibility/custom-tools/index.ts +7 -0
  412. package/src/extensibility/custom-tools/loader.ts +268 -0
  413. package/src/extensibility/custom-tools/types.ts +277 -0
  414. package/src/extensibility/custom-tools/wrapper.ts +47 -0
  415. package/src/extensibility/extensions/compact-handler.ts +40 -0
  416. package/src/extensibility/extensions/get-commands-handler.ts +78 -0
  417. package/src/extensibility/extensions/index.ts +16 -0
  418. package/src/extensibility/extensions/loader.ts +587 -0
  419. package/src/extensibility/extensions/model-api.ts +41 -0
  420. package/src/extensibility/extensions/runner.ts +989 -0
  421. package/src/extensibility/extensions/types.ts +1394 -0
  422. package/src/extensibility/extensions/wrapper.ts +259 -0
  423. package/src/extensibility/hooks/index.ts +6 -0
  424. package/src/extensibility/hooks/loader.ts +262 -0
  425. package/src/extensibility/hooks/runner.ts +425 -0
  426. package/src/extensibility/hooks/tool-wrapper.ts +107 -0
  427. package/src/extensibility/hooks/types.ts +613 -0
  428. package/src/extensibility/legacy-pi-ai-shim.ts +61 -0
  429. package/src/extensibility/legacy-pi-coding-agent-shim.ts +128 -0
  430. package/src/extensibility/plugins/doctor.ts +65 -0
  431. package/src/extensibility/plugins/git-url.ts +367 -0
  432. package/src/extensibility/plugins/index.ts +9 -0
  433. package/src/extensibility/plugins/installer.ts +192 -0
  434. package/src/extensibility/plugins/legacy-pi-compat.ts +712 -0
  435. package/src/extensibility/plugins/loader.ts +458 -0
  436. package/src/extensibility/plugins/manager.ts +1026 -0
  437. package/src/extensibility/plugins/marketplace/cache.ts +136 -0
  438. package/src/extensibility/plugins/marketplace/fetcher.ts +315 -0
  439. package/src/extensibility/plugins/marketplace/index.ts +6 -0
  440. package/src/extensibility/plugins/marketplace/manager.ts +770 -0
  441. package/src/extensibility/plugins/marketplace/registry.ts +196 -0
  442. package/src/extensibility/plugins/marketplace/source-resolver.ts +147 -0
  443. package/src/extensibility/plugins/marketplace/types.ts +191 -0
  444. package/src/extensibility/plugins/marketplace-auto-update.ts +49 -0
  445. package/src/extensibility/plugins/parser.ts +105 -0
  446. package/src/extensibility/plugins/runtime-config.ts +9 -0
  447. package/src/extensibility/plugins/types.ts +194 -0
  448. package/src/extensibility/shared-events.ts +367 -0
  449. package/src/extensibility/skills.ts +408 -0
  450. package/src/extensibility/slash-commands.ts +131 -0
  451. package/src/extensibility/tool-proxy.ts +28 -0
  452. package/src/extensibility/typebox.ts +945 -0
  453. package/src/extensibility/utils.ts +44 -0
  454. package/src/goals/guided-setup.ts +142 -0
  455. package/src/goals/index.ts +3 -0
  456. package/src/goals/runtime.ts +521 -0
  457. package/src/goals/state.ts +37 -0
  458. package/src/goals/tools/goal-tool.ts +251 -0
  459. package/src/hindsight/backend.ts +354 -0
  460. package/src/hindsight/bank.ts +156 -0
  461. package/src/hindsight/client.ts +623 -0
  462. package/src/hindsight/config.ts +175 -0
  463. package/src/hindsight/content.ts +210 -0
  464. package/src/hindsight/index.ts +8 -0
  465. package/src/hindsight/mental-models.ts +429 -0
  466. package/src/hindsight/seeds.json +32 -0
  467. package/src/hindsight/state.ts +492 -0
  468. package/src/hindsight/transcript.ts +71 -0
  469. package/src/index.ts +66 -0
  470. package/src/internal-urls/agent-protocol.ts +146 -0
  471. package/src/internal-urls/artifact-protocol.ts +107 -0
  472. package/src/internal-urls/docs-index.generated.txt +2 -0
  473. package/src/internal-urls/docs-index.ts +102 -0
  474. package/src/internal-urls/history-protocol.ts +118 -0
  475. package/src/internal-urls/index.ts +25 -0
  476. package/src/internal-urls/issue-pr-protocol.ts +594 -0
  477. package/src/internal-urls/json-query.ts +126 -0
  478. package/src/internal-urls/local-protocol.ts +309 -0
  479. package/src/internal-urls/mcp-protocol.ts +151 -0
  480. package/src/internal-urls/memory-protocol.ts +169 -0
  481. package/src/internal-urls/omp-protocol.ts +94 -0
  482. package/src/internal-urls/parse.ts +72 -0
  483. package/src/internal-urls/registry-helpers.ts +25 -0
  484. package/src/internal-urls/router.ts +105 -0
  485. package/src/internal-urls/rule-protocol.ts +45 -0
  486. package/src/internal-urls/skill-protocol.ts +96 -0
  487. package/src/internal-urls/types.ts +152 -0
  488. package/src/internal-urls/vault-protocol.ts +936 -0
  489. package/src/irc/bus.ts +311 -0
  490. package/src/lib/xai-http.ts +124 -0
  491. package/src/lsp/client.ts +1217 -0
  492. package/src/lsp/clients/biome-client.ts +264 -0
  493. package/src/lsp/clients/index.ts +50 -0
  494. package/src/lsp/clients/lsp-linter-client.ts +85 -0
  495. package/src/lsp/clients/swiftlint-client.ts +120 -0
  496. package/src/lsp/config.ts +502 -0
  497. package/src/lsp/defaults.json +499 -0
  498. package/src/lsp/diagnostics-ledger.ts +51 -0
  499. package/src/lsp/edits.ts +267 -0
  500. package/src/lsp/format-options.ts +119 -0
  501. package/src/lsp/index.ts +2480 -0
  502. package/src/lsp/lspmux.ts +233 -0
  503. package/src/lsp/render.ts +668 -0
  504. package/src/lsp/startup-events.ts +13 -0
  505. package/src/lsp/types.ts +444 -0
  506. package/src/lsp/utils.ts +718 -0
  507. package/src/main.ts +1421 -0
  508. package/src/markit/NOTICE +32 -0
  509. package/src/markit/converters/docx.ts +56 -0
  510. package/src/markit/converters/epub.ts +136 -0
  511. package/src/markit/converters/mammoth.d.ts +24 -0
  512. package/src/markit/converters/pdf/columns.ts +103 -0
  513. package/src/markit/converters/pdf/extract.ts +574 -0
  514. package/src/markit/converters/pdf/grid.ts +780 -0
  515. package/src/markit/converters/pdf/headers.ts +106 -0
  516. package/src/markit/converters/pdf/index.ts +146 -0
  517. package/src/markit/converters/pdf/render.ts +501 -0
  518. package/src/markit/converters/pdf/types.ts +84 -0
  519. package/src/markit/converters/pptx.ts +325 -0
  520. package/src/markit/converters/xlsx.ts +173 -0
  521. package/src/markit/index.ts +2 -0
  522. package/src/markit/registry.ts +59 -0
  523. package/src/markit/types.ts +35 -0
  524. package/src/mcp/client.ts +509 -0
  525. package/src/mcp/config-writer.ts +229 -0
  526. package/src/mcp/config.ts +365 -0
  527. package/src/mcp/index.ts +29 -0
  528. package/src/mcp/json-rpc.ts +122 -0
  529. package/src/mcp/loader.ts +124 -0
  530. package/src/mcp/manager.ts +1326 -0
  531. package/src/mcp/oauth-credentials.ts +104 -0
  532. package/src/mcp/oauth-discovery.ts +467 -0
  533. package/src/mcp/oauth-flow.ts +555 -0
  534. package/src/mcp/render.ts +155 -0
  535. package/src/mcp/smithery-auth.ts +104 -0
  536. package/src/mcp/smithery-connect.ts +145 -0
  537. package/src/mcp/smithery-registry.ts +477 -0
  538. package/src/mcp/startup-events.ts +21 -0
  539. package/src/mcp/timeout.ts +59 -0
  540. package/src/mcp/tool-bridge.ts +429 -0
  541. package/src/mcp/tool-cache.ts +117 -0
  542. package/src/mcp/transports/http.ts +519 -0
  543. package/src/mcp/transports/index.ts +6 -0
  544. package/src/mcp/transports/stdio.ts +606 -0
  545. package/src/mcp/types.ts +427 -0
  546. package/src/memories/index.ts +1281 -0
  547. package/src/memories/storage.ts +578 -0
  548. package/src/memory-backend/index.ts +18 -0
  549. package/src/memory-backend/local-backend.ts +45 -0
  550. package/src/memory-backend/off-backend.ts +25 -0
  551. package/src/memory-backend/resolve.ts +25 -0
  552. package/src/memory-backend/runtime.ts +66 -0
  553. package/src/memory-backend/types.ts +166 -0
  554. package/src/mnemopi/backend.ts +612 -0
  555. package/src/mnemopi/config.ts +265 -0
  556. package/src/mnemopi/embed-client.ts +401 -0
  557. package/src/mnemopi/embed-protocol.ts +35 -0
  558. package/src/mnemopi/embed-worker.ts +113 -0
  559. package/src/mnemopi/index.ts +3 -0
  560. package/src/mnemopi/state.ts +657 -0
  561. package/src/modes/acp/acp-agent.ts +2362 -0
  562. package/src/modes/acp/acp-client-bridge.ts +154 -0
  563. package/src/modes/acp/acp-event-mapper.ts +933 -0
  564. package/src/modes/acp/acp-mode.ts +23 -0
  565. package/src/modes/acp/index.ts +2 -0
  566. package/src/modes/acp/terminal-auth.ts +37 -0
  567. package/src/modes/components/__tests__/skill-message.test.ts +92 -0
  568. package/src/modes/components/advisor-message.ts +99 -0
  569. package/src/modes/components/agent-dashboard.ts +1206 -0
  570. package/src/modes/components/agent-hub.ts +566 -0
  571. package/src/modes/components/agent-transcript-viewer.ts +461 -0
  572. package/src/modes/components/assistant-message.ts +612 -0
  573. package/src/modes/components/background-tan-message.ts +36 -0
  574. package/src/modes/components/bash-execution.ts +220 -0
  575. package/src/modes/components/bordered-loader.ts +41 -0
  576. package/src/modes/components/btw-panel.ts +112 -0
  577. package/src/modes/components/cache-invalidation-marker.ts +110 -0
  578. package/src/modes/components/chat-block.ts +111 -0
  579. package/src/modes/components/chat-transcript-builder.ts +476 -0
  580. package/src/modes/components/collab-prompt-message.ts +32 -0
  581. package/src/modes/components/compaction-summary-message.ts +215 -0
  582. package/src/modes/components/copy-selector.ts +206 -0
  583. package/src/modes/components/countdown-timer.ts +75 -0
  584. package/src/modes/components/custom-editor.test.ts +142 -0
  585. package/src/modes/components/custom-editor.ts +620 -0
  586. package/src/modes/components/custom-message.ts +67 -0
  587. package/src/modes/components/diff.ts +254 -0
  588. package/src/modes/components/dynamic-border.ts +34 -0
  589. package/src/modes/components/error-banner.ts +33 -0
  590. package/src/modes/components/eval-execution.ts +158 -0
  591. package/src/modes/components/execution-shared.ts +101 -0
  592. package/src/modes/components/extensions/extension-dashboard.ts +399 -0
  593. package/src/modes/components/extensions/extension-list.ts +502 -0
  594. package/src/modes/components/extensions/index.ts +9 -0
  595. package/src/modes/components/extensions/inspector-panel.ts +321 -0
  596. package/src/modes/components/extensions/state-manager.ts +627 -0
  597. package/src/modes/components/extensions/types.ts +186 -0
  598. package/src/modes/components/footer.ts +275 -0
  599. package/src/modes/components/history-search.ts +280 -0
  600. package/src/modes/components/hook-editor.ts +167 -0
  601. package/src/modes/components/hook-input.ts +87 -0
  602. package/src/modes/components/hook-message.ts +67 -0
  603. package/src/modes/components/hook-selector.ts +659 -0
  604. package/src/modes/components/index.ts +38 -0
  605. package/src/modes/components/keybinding-hints.ts +65 -0
  606. package/src/modes/components/late-diagnostics-message.ts +60 -0
  607. package/src/modes/components/login-dialog.ts +164 -0
  608. package/src/modes/components/logout-account-selector.ts +130 -0
  609. package/src/modes/components/mcp-add-wizard.ts +1360 -0
  610. package/src/modes/components/message-frame.ts +92 -0
  611. package/src/modes/components/model-selector.ts +1315 -0
  612. package/src/modes/components/oauth-selector.ts +457 -0
  613. package/src/modes/components/omfg-panel.ts +141 -0
  614. package/src/modes/components/overlay-box.ts +109 -0
  615. package/src/modes/components/plan-review-overlay.ts +847 -0
  616. package/src/modes/components/plan-toc.ts +138 -0
  617. package/src/modes/components/plugin-selector.ts +95 -0
  618. package/src/modes/components/plugin-settings.ts +739 -0
  619. package/src/modes/components/queue-mode-selector.ts +56 -0
  620. package/src/modes/components/read-tool-group.ts +676 -0
  621. package/src/modes/components/reset-usage-selector.ts +161 -0
  622. package/src/modes/components/segment-track.ts +89 -0
  623. package/src/modes/components/session-selector.ts +631 -0
  624. package/src/modes/components/settings-defs.ts +225 -0
  625. package/src/modes/components/settings-selector.ts +1095 -0
  626. package/src/modes/components/show-images-selector.ts +45 -0
  627. package/src/modes/components/skill-message.ts +110 -0
  628. package/src/modes/components/snapcompact-shape-preview-doc.md +18 -0
  629. package/src/modes/components/snapcompact-shape-preview.ts +192 -0
  630. package/src/modes/components/status-line/component.ts +1001 -0
  631. package/src/modes/components/status-line/context-thresholds.ts +78 -0
  632. package/src/modes/components/status-line/git-utils.ts +42 -0
  633. package/src/modes/components/status-line/index.ts +5 -0
  634. package/src/modes/components/status-line/presets.ts +106 -0
  635. package/src/modes/components/status-line/segments.ts +616 -0
  636. package/src/modes/components/status-line/separators.ts +55 -0
  637. package/src/modes/components/status-line/token-rate.ts +66 -0
  638. package/src/modes/components/status-line/types.ts +124 -0
  639. package/src/modes/components/theme-selector.ts +63 -0
  640. package/src/modes/components/thinking-selector.ts +52 -0
  641. package/src/modes/components/tiny-title-download-progress.ts +90 -0
  642. package/src/modes/components/tips.txt +24 -0
  643. package/src/modes/components/todo-reminder.ts +39 -0
  644. package/src/modes/components/tool-execution.ts +1165 -0
  645. package/src/modes/components/transcript-container.ts +806 -0
  646. package/src/modes/components/tree-selector.ts +994 -0
  647. package/src/modes/components/ttsr-notification.ts +123 -0
  648. package/src/modes/components/usage-row.ts +18 -0
  649. package/src/modes/components/user-message-selector.ts +227 -0
  650. package/src/modes/components/user-message.ts +68 -0
  651. package/src/modes/components/visual-truncate.ts +63 -0
  652. package/src/modes/components/welcome.ts +581 -0
  653. package/src/modes/controllers/btw-controller.ts +173 -0
  654. package/src/modes/controllers/command-controller-shared.ts +109 -0
  655. package/src/modes/controllers/command-controller.ts +1653 -0
  656. package/src/modes/controllers/event-controller.ts +1153 -0
  657. package/src/modes/controllers/extension-ui-controller.ts +893 -0
  658. package/src/modes/controllers/input-controller.ts +1627 -0
  659. package/src/modes/controllers/mcp-command-controller.ts +2162 -0
  660. package/src/modes/controllers/omfg-controller.ts +283 -0
  661. package/src/modes/controllers/omfg-rule.ts +647 -0
  662. package/src/modes/controllers/selector-controller.ts +1285 -0
  663. package/src/modes/controllers/session-focus-controller.ts +112 -0
  664. package/src/modes/controllers/ssh-command-controller.ts +384 -0
  665. package/src/modes/controllers/streaming-reveal.ts +295 -0
  666. package/src/modes/controllers/tan-command-controller.ts +190 -0
  667. package/src/modes/controllers/todo-command-controller.ts +485 -0
  668. package/src/modes/controllers/tool-args-reveal.ts +174 -0
  669. package/src/modes/data/emojis.json +1 -0
  670. package/src/modes/emoji-autocomplete.ts +285 -0
  671. package/src/modes/gradient-highlight.ts +99 -0
  672. package/src/modes/image-references.ts +137 -0
  673. package/src/modes/index.ts +17 -0
  674. package/src/modes/interactive-mode.ts +3940 -0
  675. package/src/modes/internal-url-autocomplete.ts +143 -0
  676. package/src/modes/loop-limit.ts +192 -0
  677. package/src/modes/magic-keywords.ts +42 -0
  678. package/src/modes/markdown-prose.ts +247 -0
  679. package/src/modes/oauth-manual-input.ts +69 -0
  680. package/src/modes/orchestrate.ts +42 -0
  681. package/src/modes/print-mode.ts +130 -0
  682. package/src/modes/prompt-action-autocomplete.ts +260 -0
  683. package/src/modes/rpc/host-tools.ts +186 -0
  684. package/src/modes/rpc/host-uris.ts +235 -0
  685. package/src/modes/rpc/rpc-client.ts +995 -0
  686. package/src/modes/rpc/rpc-mode.ts +1156 -0
  687. package/src/modes/rpc/rpc-subagents.ts +265 -0
  688. package/src/modes/rpc/rpc-types.ts +487 -0
  689. package/src/modes/runtime-init.ts +142 -0
  690. package/src/modes/session-observer-registry.ts +215 -0
  691. package/src/modes/setup-version.ts +11 -0
  692. package/src/modes/setup-wizard/index.ts +101 -0
  693. package/src/modes/setup-wizard/lazy.ts +16 -0
  694. package/src/modes/setup-wizard/scenes/glyph.ts +114 -0
  695. package/src/modes/setup-wizard/scenes/outro.ts +35 -0
  696. package/src/modes/setup-wizard/scenes/providers.ts +103 -0
  697. package/src/modes/setup-wizard/scenes/sign-in.ts +286 -0
  698. package/src/modes/setup-wizard/scenes/splash.ts +201 -0
  699. package/src/modes/setup-wizard/scenes/theme.ts +326 -0
  700. package/src/modes/setup-wizard/scenes/types.ts +57 -0
  701. package/src/modes/setup-wizard/scenes/web-search.ts +145 -0
  702. package/src/modes/setup-wizard/startup-splash.ts +107 -0
  703. package/src/modes/setup-wizard/wizard-overlay.ts +334 -0
  704. package/src/modes/shared.ts +49 -0
  705. package/src/modes/theme/dark.json +95 -0
  706. package/src/modes/theme/defaults/alabaster.json +93 -0
  707. package/src/modes/theme/defaults/amethyst.json +96 -0
  708. package/src/modes/theme/defaults/anthracite.json +93 -0
  709. package/src/modes/theme/defaults/basalt.json +91 -0
  710. package/src/modes/theme/defaults/birch.json +95 -0
  711. package/src/modes/theme/defaults/dark-abyss.json +91 -0
  712. package/src/modes/theme/defaults/dark-arctic.json +104 -0
  713. package/src/modes/theme/defaults/dark-aurora.json +95 -0
  714. package/src/modes/theme/defaults/dark-catppuccin.json +107 -0
  715. package/src/modes/theme/defaults/dark-cavern.json +91 -0
  716. package/src/modes/theme/defaults/dark-copper.json +95 -0
  717. package/src/modes/theme/defaults/dark-cosmos.json +90 -0
  718. package/src/modes/theme/defaults/dark-cyberpunk.json +102 -0
  719. package/src/modes/theme/defaults/dark-dracula.json +98 -0
  720. package/src/modes/theme/defaults/dark-eclipse.json +91 -0
  721. package/src/modes/theme/defaults/dark-ember.json +95 -0
  722. package/src/modes/theme/defaults/dark-equinox.json +90 -0
  723. package/src/modes/theme/defaults/dark-forest.json +96 -0
  724. package/src/modes/theme/defaults/dark-github.json +105 -0
  725. package/src/modes/theme/defaults/dark-gruvbox.json +112 -0
  726. package/src/modes/theme/defaults/dark-lavender.json +95 -0
  727. package/src/modes/theme/defaults/dark-lunar.json +89 -0
  728. package/src/modes/theme/defaults/dark-midnight.json +95 -0
  729. package/src/modes/theme/defaults/dark-monochrome.json +94 -0
  730. package/src/modes/theme/defaults/dark-monokai.json +98 -0
  731. package/src/modes/theme/defaults/dark-nebula.json +90 -0
  732. package/src/modes/theme/defaults/dark-nord.json +97 -0
  733. package/src/modes/theme/defaults/dark-ocean.json +101 -0
  734. package/src/modes/theme/defaults/dark-one.json +100 -0
  735. package/src/modes/theme/defaults/dark-poimandres.json +142 -0
  736. package/src/modes/theme/defaults/dark-rainforest.json +91 -0
  737. package/src/modes/theme/defaults/dark-reef.json +91 -0
  738. package/src/modes/theme/defaults/dark-retro.json +92 -0
  739. package/src/modes/theme/defaults/dark-rose-pine.json +96 -0
  740. package/src/modes/theme/defaults/dark-sakura.json +95 -0
  741. package/src/modes/theme/defaults/dark-slate.json +95 -0
  742. package/src/modes/theme/defaults/dark-solarized.json +97 -0
  743. package/src/modes/theme/defaults/dark-solstice.json +90 -0
  744. package/src/modes/theme/defaults/dark-starfall.json +91 -0
  745. package/src/modes/theme/defaults/dark-sunset.json +99 -0
  746. package/src/modes/theme/defaults/dark-swamp.json +90 -0
  747. package/src/modes/theme/defaults/dark-synthwave.json +103 -0
  748. package/src/modes/theme/defaults/dark-taiga.json +91 -0
  749. package/src/modes/theme/defaults/dark-terminal.json +95 -0
  750. package/src/modes/theme/defaults/dark-tokyo-night.json +101 -0
  751. package/src/modes/theme/defaults/dark-tundra.json +91 -0
  752. package/src/modes/theme/defaults/dark-twilight.json +91 -0
  753. package/src/modes/theme/defaults/dark-volcanic.json +91 -0
  754. package/src/modes/theme/defaults/graphite.json +92 -0
  755. package/src/modes/theme/defaults/index.ts +199 -0
  756. package/src/modes/theme/defaults/light-arctic.json +107 -0
  757. package/src/modes/theme/defaults/light-aurora-day.json +91 -0
  758. package/src/modes/theme/defaults/light-canyon.json +91 -0
  759. package/src/modes/theme/defaults/light-catppuccin.json +106 -0
  760. package/src/modes/theme/defaults/light-cirrus.json +90 -0
  761. package/src/modes/theme/defaults/light-coral.json +95 -0
  762. package/src/modes/theme/defaults/light-cyberpunk.json +96 -0
  763. package/src/modes/theme/defaults/light-dawn.json +90 -0
  764. package/src/modes/theme/defaults/light-dunes.json +91 -0
  765. package/src/modes/theme/defaults/light-eucalyptus.json +95 -0
  766. package/src/modes/theme/defaults/light-forest.json +100 -0
  767. package/src/modes/theme/defaults/light-frost.json +95 -0
  768. package/src/modes/theme/defaults/light-github.json +115 -0
  769. package/src/modes/theme/defaults/light-glacier.json +91 -0
  770. package/src/modes/theme/defaults/light-gruvbox.json +108 -0
  771. package/src/modes/theme/defaults/light-haze.json +90 -0
  772. package/src/modes/theme/defaults/light-honeycomb.json +95 -0
  773. package/src/modes/theme/defaults/light-lagoon.json +91 -0
  774. package/src/modes/theme/defaults/light-lavender.json +95 -0
  775. package/src/modes/theme/defaults/light-meadow.json +91 -0
  776. package/src/modes/theme/defaults/light-mint.json +95 -0
  777. package/src/modes/theme/defaults/light-monochrome.json +101 -0
  778. package/src/modes/theme/defaults/light-ocean.json +99 -0
  779. package/src/modes/theme/defaults/light-one.json +99 -0
  780. package/src/modes/theme/defaults/light-opal.json +91 -0
  781. package/src/modes/theme/defaults/light-orchard.json +91 -0
  782. package/src/modes/theme/defaults/light-paper.json +95 -0
  783. package/src/modes/theme/defaults/light-poimandres.json +142 -0
  784. package/src/modes/theme/defaults/light-prism.json +90 -0
  785. package/src/modes/theme/defaults/light-retro.json +98 -0
  786. package/src/modes/theme/defaults/light-sand.json +95 -0
  787. package/src/modes/theme/defaults/light-savanna.json +91 -0
  788. package/src/modes/theme/defaults/light-solarized.json +102 -0
  789. package/src/modes/theme/defaults/light-soleil.json +90 -0
  790. package/src/modes/theme/defaults/light-sunset.json +99 -0
  791. package/src/modes/theme/defaults/light-synthwave.json +98 -0
  792. package/src/modes/theme/defaults/light-tokyo-night.json +111 -0
  793. package/src/modes/theme/defaults/light-wetland.json +91 -0
  794. package/src/modes/theme/defaults/light-zenith.json +89 -0
  795. package/src/modes/theme/defaults/limestone.json +94 -0
  796. package/src/modes/theme/defaults/mahogany.json +97 -0
  797. package/src/modes/theme/defaults/marble.json +93 -0
  798. package/src/modes/theme/defaults/obsidian.json +91 -0
  799. package/src/modes/theme/defaults/onyx.json +91 -0
  800. package/src/modes/theme/defaults/pearl.json +93 -0
  801. package/src/modes/theme/defaults/porcelain.json +91 -0
  802. package/src/modes/theme/defaults/quartz.json +96 -0
  803. package/src/modes/theme/defaults/sandstone.json +95 -0
  804. package/src/modes/theme/defaults/titanium.json +90 -0
  805. package/src/modes/theme/light.json +93 -0
  806. package/src/modes/theme/mermaid-cache.ts +92 -0
  807. package/src/modes/theme/shimmer.ts +235 -0
  808. package/src/modes/theme/theme-schema.json +459 -0
  809. package/src/modes/theme/theme.ts +2915 -0
  810. package/src/modes/turn-budget.ts +31 -0
  811. package/src/modes/types.ts +406 -0
  812. package/src/modes/ultrathink.ts +41 -0
  813. package/src/modes/utils/context-usage.ts +432 -0
  814. package/src/modes/utils/copy-targets.ts +360 -0
  815. package/src/modes/utils/hotkeys-markdown.ts +62 -0
  816. package/src/modes/utils/keybinding-matchers.ts +51 -0
  817. package/src/modes/utils/tools-markdown.ts +27 -0
  818. package/src/modes/utils/ui-helpers.ts +886 -0
  819. package/src/modes/workflow.ts +42 -0
  820. package/src/plan-mode/approved-plan.ts +186 -0
  821. package/src/plan-mode/plan-handoff.ts +37 -0
  822. package/src/plan-mode/plan-protection.ts +31 -0
  823. package/src/plan-mode/state.ts +6 -0
  824. package/src/priority.json +45 -0
  825. package/src/prompts/advisor/advise-tool.md +3 -0
  826. package/src/prompts/advisor/system.md +113 -0
  827. package/src/prompts/agents/designer.md +74 -0
  828. package/src/prompts/agents/explore.md +58 -0
  829. package/src/prompts/agents/frontmatter.md +11 -0
  830. package/src/prompts/agents/init.md +33 -0
  831. package/src/prompts/agents/librarian.md +119 -0
  832. package/src/prompts/agents/oracle.md +54 -0
  833. package/src/prompts/agents/plan.md +48 -0
  834. package/src/prompts/agents/reviewer.md +139 -0
  835. package/src/prompts/agents/task.md +17 -0
  836. package/src/prompts/bench.md +12 -0
  837. package/src/prompts/ci-green-request.md +36 -0
  838. package/src/prompts/dry-balance-bench.md +8 -0
  839. package/src/prompts/goals/goal-budget-limit.md +16 -0
  840. package/src/prompts/goals/goal-continuation.md +28 -0
  841. package/src/prompts/goals/goal-mode-active.md +23 -0
  842. package/src/prompts/goals/guided-goal-interview.md +8 -0
  843. package/src/prompts/goals/guided-goal-system.md +12 -0
  844. package/src/prompts/low-end/system.md +47 -0
  845. package/src/prompts/memories/consolidation.md +30 -0
  846. package/src/prompts/memories/consolidation_system.md +4 -0
  847. package/src/prompts/memories/read-path.md +17 -0
  848. package/src/prompts/memories/stage_one_input.md +6 -0
  849. package/src/prompts/memories/stage_one_system.md +21 -0
  850. package/src/prompts/review-custom-request.md +22 -0
  851. package/src/prompts/review-headless-request.md +16 -0
  852. package/src/prompts/review-request.md +69 -0
  853. package/src/prompts/steering/user-interjection.md +9 -0
  854. package/src/prompts/system/agent-creation-architect.md +50 -0
  855. package/src/prompts/system/agent-creation-user.md +6 -0
  856. package/src/prompts/system/auto-continue.md +1 -0
  857. package/src/prompts/system/auto-thinking-difficulty-local.md +14 -0
  858. package/src/prompts/system/auto-thinking-difficulty.md +12 -0
  859. package/src/prompts/system/autolearn-guidance-learn.md +1 -0
  860. package/src/prompts/system/autolearn-guidance.md +7 -0
  861. package/src/prompts/system/autolearn-nudge.md +3 -0
  862. package/src/prompts/system/background-tan-dispatch.md +8 -0
  863. package/src/prompts/system/btw-user.md +8 -0
  864. package/src/prompts/system/commit-message-system.md +14 -0
  865. package/src/prompts/system/custom-system-prompt.md +64 -0
  866. package/src/prompts/system/eager-task.md +7 -0
  867. package/src/prompts/system/eager-todo.md +18 -0
  868. package/src/prompts/system/empty-stop-retry.md +4 -0
  869. package/src/prompts/system/irc-autoreply.md +6 -0
  870. package/src/prompts/system/irc-incoming.md +7 -0
  871. package/src/prompts/system/manual-continue.md +7 -0
  872. package/src/prompts/system/memory-consolidation-system.md +8 -0
  873. package/src/prompts/system/memory-extraction-system.md +26 -0
  874. package/src/prompts/system/omfg-user.md +50 -0
  875. package/src/prompts/system/orchestrate-notice.md +40 -0
  876. package/src/prompts/system/personalities/default.md +18 -0
  877. package/src/prompts/system/personalities/friendly.md +17 -0
  878. package/src/prompts/system/personalities/pragmatic.md +15 -0
  879. package/src/prompts/system/plan-mode-active.md +109 -0
  880. package/src/prompts/system/plan-mode-approved.md +25 -0
  881. package/src/prompts/system/plan-mode-compact-instructions.md +16 -0
  882. package/src/prompts/system/plan-mode-reference.md +11 -0
  883. package/src/prompts/system/plan-mode-subagent.md +33 -0
  884. package/src/prompts/system/plan-mode-tool-decision-reminder.md +9 -0
  885. package/src/prompts/system/project-prompt.md +52 -0
  886. package/src/prompts/system/snapcompact-context-frames-note.md +1 -0
  887. package/src/prompts/system/snapcompact-context-stub.md +1 -0
  888. package/src/prompts/system/snapcompact-system-frames-note.md +1 -0
  889. package/src/prompts/system/snapcompact-system-stub.md +1 -0
  890. package/src/prompts/system/snapcompact-toolresult-note.md +1 -0
  891. package/src/prompts/system/subagent-system-prompt.md +71 -0
  892. package/src/prompts/system/subagent-user-prompt.md +3 -0
  893. package/src/prompts/system/subagent-yield-reminder.md +12 -0
  894. package/src/prompts/system/system-prompt.md +251 -0
  895. package/src/prompts/system/tiny-title-system.md +8 -0
  896. package/src/prompts/system/title-marker-instruction.md +1 -0
  897. package/src/prompts/system/title-system-marker.md +16 -0
  898. package/src/prompts/system/title-system.md +16 -0
  899. package/src/prompts/system/ttsr-interrupt.md +7 -0
  900. package/src/prompts/system/ttsr-tool-reminder.md +5 -0
  901. package/src/prompts/system/ultrathink-notice.md +3 -0
  902. package/src/prompts/system/unexpected-stop-classifier.md +17 -0
  903. package/src/prompts/system/unexpected-stop-retry.md +4 -0
  904. package/src/prompts/system/web-search.md +25 -0
  905. package/src/prompts/system/workflow-notice.md +70 -0
  906. package/src/prompts/tools/apply-patch.md +65 -0
  907. package/src/prompts/tools/ask.md +22 -0
  908. package/src/prompts/tools/ast-edit.md +22 -0
  909. package/src/prompts/tools/ast-grep.md +25 -0
  910. package/src/prompts/tools/async-result.md +8 -0
  911. package/src/prompts/tools/bash.md +45 -0
  912. package/src/prompts/tools/browser.md +42 -0
  913. package/src/prompts/tools/checkpoint.md +15 -0
  914. package/src/prompts/tools/debug.md +17 -0
  915. package/src/prompts/tools/eval.md +70 -0
  916. package/src/prompts/tools/find.md +19 -0
  917. package/src/prompts/tools/github.md +17 -0
  918. package/src/prompts/tools/goal.md +11 -0
  919. package/src/prompts/tools/image-attachment-describe-system.md +8 -0
  920. package/src/prompts/tools/image-attachment-describe.md +10 -0
  921. package/src/prompts/tools/image-gen.md +7 -0
  922. package/src/prompts/tools/inspect-image-system.md +20 -0
  923. package/src/prompts/tools/inspect-image.md +22 -0
  924. package/src/prompts/tools/irc.md +33 -0
  925. package/src/prompts/tools/job.md +17 -0
  926. package/src/prompts/tools/learn.md +7 -0
  927. package/src/prompts/tools/lsp-late-diagnostic.md +8 -0
  928. package/src/prompts/tools/lsp.md +39 -0
  929. package/src/prompts/tools/manage-skill.md +9 -0
  930. package/src/prompts/tools/memory-edit.md +8 -0
  931. package/src/prompts/tools/patch.md +57 -0
  932. package/src/prompts/tools/read.md +76 -0
  933. package/src/prompts/tools/recall.md +5 -0
  934. package/src/prompts/tools/reflect.md +5 -0
  935. package/src/prompts/tools/replace.md +29 -0
  936. package/src/prompts/tools/resolve.md +4 -0
  937. package/src/prompts/tools/retain.md +6 -0
  938. package/src/prompts/tools/rewind.md +13 -0
  939. package/src/prompts/tools/search-tool-bm25.md +32 -0
  940. package/src/prompts/tools/search.md +22 -0
  941. package/src/prompts/tools/ssh.md +22 -0
  942. package/src/prompts/tools/task-summary.md +17 -0
  943. package/src/prompts/tools/task.md +91 -0
  944. package/src/prompts/tools/todo.md +39 -0
  945. package/src/prompts/tools/web-search.md +6 -0
  946. package/src/prompts/tools/write.md +14 -0
  947. package/src/registry/agent-lifecycle.ts +270 -0
  948. package/src/registry/agent-registry.ts +190 -0
  949. package/src/sdk.ts +2919 -0
  950. package/src/secrets/index.ts +123 -0
  951. package/src/secrets/obfuscator.ts +298 -0
  952. package/src/secrets/regex.ts +21 -0
  953. package/src/session/agent-session.ts +12539 -0
  954. package/src/session/agent-storage.ts +478 -0
  955. package/src/session/artifacts.ts +153 -0
  956. package/src/session/auth-broker-config.ts +92 -0
  957. package/src/session/auth-storage.ts +24 -0
  958. package/src/session/blob-store.ts +255 -0
  959. package/src/session/client-bridge.ts +85 -0
  960. package/src/session/codex-auto-reset.ts +202 -0
  961. package/src/session/compact-modes.ts +105 -0
  962. package/src/session/history-storage.ts +361 -0
  963. package/src/session/indexed-session-storage.ts +427 -0
  964. package/src/session/messages.ts +546 -0
  965. package/src/session/redis-session-storage.ts +170 -0
  966. package/src/session/session-context.ts +399 -0
  967. package/src/session/session-dump-format.ts +216 -0
  968. package/src/session/session-entries.ts +198 -0
  969. package/src/session/session-history-format.ts +308 -0
  970. package/src/session/session-listing.ts +588 -0
  971. package/src/session/session-loader.ts +93 -0
  972. package/src/session/session-manager.ts +1748 -0
  973. package/src/session/session-migrations.ts +78 -0
  974. package/src/session/session-paths.ts +193 -0
  975. package/src/session/session-persistence.ts +147 -0
  976. package/src/session/session-storage.ts +590 -0
  977. package/src/session/shake-types.ts +43 -0
  978. package/src/session/snapcompact-inline.ts +542 -0
  979. package/src/session/snapcompact-savings-journal.ts +113 -0
  980. package/src/session/sql-session-storage.ts +314 -0
  981. package/src/session/streaming-output.ts +1330 -0
  982. package/src/session/tool-choice-queue.ts +290 -0
  983. package/src/session/unexpected-stop-classifier.ts +129 -0
  984. package/src/session/yield-queue.ts +183 -0
  985. package/src/slash-commands/acp-builtins.ts +70 -0
  986. package/src/slash-commands/available-commands.ts +105 -0
  987. package/src/slash-commands/builtin-registry.ts +2332 -0
  988. package/src/slash-commands/helpers/active-oauth-account.ts +44 -0
  989. package/src/slash-commands/helpers/collab-qrcode.ts +28 -0
  990. package/src/slash-commands/helpers/context-report.ts +66 -0
  991. package/src/slash-commands/helpers/format.ts +46 -0
  992. package/src/slash-commands/helpers/logout.ts +88 -0
  993. package/src/slash-commands/helpers/marketplace-manager.ts +25 -0
  994. package/src/slash-commands/helpers/mcp.ts +532 -0
  995. package/src/slash-commands/helpers/parse.ts +85 -0
  996. package/src/slash-commands/helpers/reset-usage.ts +66 -0
  997. package/src/slash-commands/helpers/ssh.ts +195 -0
  998. package/src/slash-commands/helpers/stats-dashboard.ts +85 -0
  999. package/src/slash-commands/helpers/todo.ts +279 -0
  1000. package/src/slash-commands/helpers/usage-report.ts +128 -0
  1001. package/src/slash-commands/marketplace-install-parser.ts +99 -0
  1002. package/src/slash-commands/types.ts +135 -0
  1003. package/src/ssh/config-writer.ts +183 -0
  1004. package/src/ssh/connection-manager.ts +510 -0
  1005. package/src/ssh/ssh-executor.ts +189 -0
  1006. package/src/ssh/sshfs-mount.ts +140 -0
  1007. package/src/ssh/utils.ts +8 -0
  1008. package/src/startup-splash.ts +19 -0
  1009. package/src/stt/asr-client.ts +521 -0
  1010. package/src/stt/asr-protocol.ts +65 -0
  1011. package/src/stt/asr-worker.ts +790 -0
  1012. package/src/stt/downloader.ts +138 -0
  1013. package/src/stt/endpointer.ts +259 -0
  1014. package/src/stt/index.ts +7 -0
  1015. package/src/stt/models.ts +150 -0
  1016. package/src/stt/recorder.ts +538 -0
  1017. package/src/stt/stt-controller.ts +380 -0
  1018. package/src/stt/transcriber.ts +60 -0
  1019. package/src/stt/wav.ts +173 -0
  1020. package/src/system-prompt.ts +709 -0
  1021. package/src/task/agents.ts +166 -0
  1022. package/src/task/commands.ts +132 -0
  1023. package/src/task/discovery.ts +122 -0
  1024. package/src/task/executor.ts +2356 -0
  1025. package/src/task/index.ts +1580 -0
  1026. package/src/task/name-generator.ts +1577 -0
  1027. package/src/task/omp-command.ts +26 -0
  1028. package/src/task/output-manager.ts +93 -0
  1029. package/src/task/parallel.ts +116 -0
  1030. package/src/task/persisted-revive.ts +128 -0
  1031. package/src/task/render.ts +1558 -0
  1032. package/src/task/repair-args.ts +129 -0
  1033. package/src/task/subprocess-tool-registry.ts +88 -0
  1034. package/src/task/types.ts +401 -0
  1035. package/src/task/worktree.ts +514 -0
  1036. package/src/telemetry-export.ts +144 -0
  1037. package/src/thinking.ts +187 -0
  1038. package/src/tiny/device.ts +111 -0
  1039. package/src/tiny/dtype.ts +101 -0
  1040. package/src/tiny/models.ts +252 -0
  1041. package/src/tiny/text.ts +169 -0
  1042. package/src/tiny/title-client.ts +538 -0
  1043. package/src/tiny/title-protocol.ts +56 -0
  1044. package/src/tiny/worker.ts +491 -0
  1045. package/src/tool-discovery/mode.ts +24 -0
  1046. package/src/tool-discovery/tool-index.ts +271 -0
  1047. package/src/tools/__tests__/json-tree.test.ts +35 -0
  1048. package/src/tools/approval.ts +189 -0
  1049. package/src/tools/ask.ts +977 -0
  1050. package/src/tools/ast-edit.ts +700 -0
  1051. package/src/tools/ast-grep.ts +483 -0
  1052. package/src/tools/auto-generated-guard.ts +322 -0
  1053. package/src/tools/bash-command-fixup.ts +37 -0
  1054. package/src/tools/bash-interactive.ts +408 -0
  1055. package/src/tools/bash-interceptor.ts +67 -0
  1056. package/src/tools/bash-pty-selection.ts +14 -0
  1057. package/src/tools/bash-skill-urls.ts +248 -0
  1058. package/src/tools/bash.ts +1405 -0
  1059. package/src/tools/browser/attach.ts +194 -0
  1060. package/src/tools/browser/cmux/cmux-tab.ts +1264 -0
  1061. package/src/tools/browser/cmux/rpc.ts +156 -0
  1062. package/src/tools/browser/cmux/socket-client.ts +309 -0
  1063. package/src/tools/browser/launch.ts +673 -0
  1064. package/src/tools/browser/readable.ts +112 -0
  1065. package/src/tools/browser/registry.ts +241 -0
  1066. package/src/tools/browser/render.ts +221 -0
  1067. package/src/tools/browser/tab-protocol.ts +107 -0
  1068. package/src/tools/browser/tab-supervisor.ts +799 -0
  1069. package/src/tools/browser/tab-worker-entry.ts +29 -0
  1070. package/src/tools/browser/tab-worker.ts +1226 -0
  1071. package/src/tools/browser.ts +403 -0
  1072. package/src/tools/builtin-names.ts +34 -0
  1073. package/src/tools/checkpoint.ts +136 -0
  1074. package/src/tools/conflict-detect.ts +718 -0
  1075. package/src/tools/context.ts +39 -0
  1076. package/src/tools/debug.ts +1087 -0
  1077. package/src/tools/eval-backends.ts +27 -0
  1078. package/src/tools/eval-render.ts +762 -0
  1079. package/src/tools/eval.ts +600 -0
  1080. package/src/tools/fetch.ts +1902 -0
  1081. package/src/tools/file-recorder.ts +35 -0
  1082. package/src/tools/find.ts +629 -0
  1083. package/src/tools/fs-cache-invalidation.ts +28 -0
  1084. package/src/tools/gh-cache-invalidation.ts +255 -0
  1085. package/src/tools/gh-format.ts +12 -0
  1086. package/src/tools/gh-renderer.ts +481 -0
  1087. package/src/tools/gh.ts +3752 -0
  1088. package/src/tools/github-cache.ts +663 -0
  1089. package/src/tools/grouped-file-output.ts +210 -0
  1090. package/src/tools/image-gen.ts +1586 -0
  1091. package/src/tools/index.ts +649 -0
  1092. package/src/tools/inspect-image-renderer.ts +132 -0
  1093. package/src/tools/inspect-image.ts +260 -0
  1094. package/src/tools/irc.ts +788 -0
  1095. package/src/tools/job.ts +612 -0
  1096. package/src/tools/json-tree.ts +260 -0
  1097. package/src/tools/jtd-to-json-schema.ts +219 -0
  1098. package/src/tools/jtd-to-typescript.ts +136 -0
  1099. package/src/tools/jtd-utils.ts +102 -0
  1100. package/src/tools/learn.ts +141 -0
  1101. package/src/tools/list-limit.ts +40 -0
  1102. package/src/tools/manage-skill.ts +100 -0
  1103. package/src/tools/match-line-format.ts +20 -0
  1104. package/src/tools/memory-edit.ts +59 -0
  1105. package/src/tools/memory-recall.ts +102 -0
  1106. package/src/tools/memory-reflect.ts +88 -0
  1107. package/src/tools/memory-render.ts +202 -0
  1108. package/src/tools/memory-retain.ts +89 -0
  1109. package/src/tools/output-meta.ts +768 -0
  1110. package/src/tools/output-schema-validator.ts +132 -0
  1111. package/src/tools/path-utils.ts +1116 -0
  1112. package/src/tools/plan-mode-guard.ts +142 -0
  1113. package/src/tools/puppeteer/00_stealth_tampering.txt +63 -0
  1114. package/src/tools/puppeteer/01_stealth_activity.txt +20 -0
  1115. package/src/tools/puppeteer/02_stealth_hairline.txt +11 -0
  1116. package/src/tools/puppeteer/03_stealth_botd.txt +384 -0
  1117. package/src/tools/puppeteer/04_stealth_iframe.txt +81 -0
  1118. package/src/tools/puppeteer/05_stealth_webgl.txt +75 -0
  1119. package/src/tools/puppeteer/06_stealth_screen.txt +72 -0
  1120. package/src/tools/puppeteer/07_stealth_fonts.txt +97 -0
  1121. package/src/tools/puppeteer/08_stealth_audio.txt +51 -0
  1122. package/src/tools/puppeteer/09_stealth_locale.txt +46 -0
  1123. package/src/tools/puppeteer/10_stealth_plugins.txt +208 -0
  1124. package/src/tools/puppeteer/11_stealth_hardware.txt +8 -0
  1125. package/src/tools/puppeteer/12_stealth_codecs.txt +40 -0
  1126. package/src/tools/puppeteer/13_stealth_worker.txt +74 -0
  1127. package/src/tools/read.ts +3124 -0
  1128. package/src/tools/render-utils.ts +895 -0
  1129. package/src/tools/renderers.ts +86 -0
  1130. package/src/tools/report-tool-issue.ts +530 -0
  1131. package/src/tools/resolve.ts +302 -0
  1132. package/src/tools/review.ts +251 -0
  1133. package/src/tools/search-tool-bm25.ts +351 -0
  1134. package/src/tools/search.ts +1583 -0
  1135. package/src/tools/sqlite-reader.ts +828 -0
  1136. package/src/tools/ssh.ts +369 -0
  1137. package/src/tools/todo.ts +938 -0
  1138. package/src/tools/tool-errors.ts +62 -0
  1139. package/src/tools/tool-result.ts +102 -0
  1140. package/src/tools/tool-timeouts.ts +30 -0
  1141. package/src/tools/tts.ts +265 -0
  1142. package/src/tools/write.ts +1182 -0
  1143. package/src/tools/yield.ts +269 -0
  1144. package/src/tts/downloader.ts +64 -0
  1145. package/src/tts/index.ts +8 -0
  1146. package/src/tts/models.ts +137 -0
  1147. package/src/tts/player.ts +137 -0
  1148. package/src/tts/runtime.ts +21 -0
  1149. package/src/tts/streaming-player.ts +266 -0
  1150. package/src/tts/tts-client.ts +642 -0
  1151. package/src/tts/tts-protocol.ts +60 -0
  1152. package/src/tts/tts-worker.ts +505 -0
  1153. package/src/tts/vocalizer.ts +162 -0
  1154. package/src/tts/wav.ts +58 -0
  1155. package/src/tui/code-cell.ts +257 -0
  1156. package/src/tui/file-list.ts +55 -0
  1157. package/src/tui/hyperlink.ts +178 -0
  1158. package/src/tui/index.ts +13 -0
  1159. package/src/tui/output-block.ts +240 -0
  1160. package/src/tui/status-line.ts +54 -0
  1161. package/src/tui/tree-list.ts +133 -0
  1162. package/src/tui/types.ts +15 -0
  1163. package/src/tui/utils.ts +103 -0
  1164. package/src/tui/width-aware-text.ts +58 -0
  1165. package/src/utils/block-context.ts +312 -0
  1166. package/src/utils/changelog.ts +132 -0
  1167. package/src/utils/clipboard.ts +262 -0
  1168. package/src/utils/command-args.ts +76 -0
  1169. package/src/utils/commit-message-generator.ts +147 -0
  1170. package/src/utils/edit-mode.ts +41 -0
  1171. package/src/utils/enhanced-paste.ts +230 -0
  1172. package/src/utils/event-bus.ts +33 -0
  1173. package/src/utils/external-editor.ts +78 -0
  1174. package/src/utils/file-display-mode.ts +45 -0
  1175. package/src/utils/file-mentions.ts +284 -0
  1176. package/src/utils/git.ts +1838 -0
  1177. package/src/utils/image-loading.ts +231 -0
  1178. package/src/utils/image-resize.ts +309 -0
  1179. package/src/utils/image-vision-fallback.ts +197 -0
  1180. package/src/utils/ipc.ts +38 -0
  1181. package/src/utils/jj.ts +248 -0
  1182. package/src/utils/lang-from-path.ts +244 -0
  1183. package/src/utils/markit.ts +143 -0
  1184. package/src/utils/mupdf-wasm-embed.ts +12 -0
  1185. package/src/utils/open.ts +55 -0
  1186. package/src/utils/qrcode.ts +535 -0
  1187. package/src/utils/session-color.ts +142 -0
  1188. package/src/utils/shell-snapshot.ts +187 -0
  1189. package/src/utils/sixel.ts +69 -0
  1190. package/src/utils/thinking-display.ts +11 -0
  1191. package/src/utils/title-generator.ts +416 -0
  1192. package/src/utils/tool-choice.ts +49 -0
  1193. package/src/utils/tools-manager.ts +372 -0
  1194. package/src/utils/turndown.ts +83 -0
  1195. package/src/utils/zip.ts +1091 -0
  1196. package/src/web/kagi.ts +304 -0
  1197. package/src/web/parallel.ts +353 -0
  1198. package/src/web/scrapers/artifacthub.ts +207 -0
  1199. package/src/web/scrapers/arxiv.ts +83 -0
  1200. package/src/web/scrapers/aur.ts +162 -0
  1201. package/src/web/scrapers/biorxiv.ts +133 -0
  1202. package/src/web/scrapers/bluesky.ts +262 -0
  1203. package/src/web/scrapers/brew.ts +172 -0
  1204. package/src/web/scrapers/cheatsh.ts +68 -0
  1205. package/src/web/scrapers/chocolatey.ts +196 -0
  1206. package/src/web/scrapers/choosealicense.ts +95 -0
  1207. package/src/web/scrapers/cisa-kev.ts +87 -0
  1208. package/src/web/scrapers/clojars.ts +154 -0
  1209. package/src/web/scrapers/coingecko.ts +177 -0
  1210. package/src/web/scrapers/crates-io.ts +97 -0
  1211. package/src/web/scrapers/crossref.ts +136 -0
  1212. package/src/web/scrapers/devto.ts +147 -0
  1213. package/src/web/scrapers/discogs.ts +306 -0
  1214. package/src/web/scrapers/discourse.ts +197 -0
  1215. package/src/web/scrapers/dockerhub.ts +138 -0
  1216. package/src/web/scrapers/docs-rs.ts +652 -0
  1217. package/src/web/scrapers/fdroid.ts +134 -0
  1218. package/src/web/scrapers/firefox-addons.ts +191 -0
  1219. package/src/web/scrapers/flathub.ts +223 -0
  1220. package/src/web/scrapers/github-gist.ts +58 -0
  1221. package/src/web/scrapers/github.ts +800 -0
  1222. package/src/web/scrapers/gitlab.ts +401 -0
  1223. package/src/web/scrapers/go-pkg.ts +266 -0
  1224. package/src/web/scrapers/hackage.ts +140 -0
  1225. package/src/web/scrapers/hackernews.ts +189 -0
  1226. package/src/web/scrapers/hex.ts +105 -0
  1227. package/src/web/scrapers/huggingface.ts +321 -0
  1228. package/src/web/scrapers/iacr.ts +89 -0
  1229. package/src/web/scrapers/index.ts +252 -0
  1230. package/src/web/scrapers/jetbrains-marketplace.ts +159 -0
  1231. package/src/web/scrapers/lemmy.ts +203 -0
  1232. package/src/web/scrapers/lobsters.ts +175 -0
  1233. package/src/web/scrapers/mastodon.ts +292 -0
  1234. package/src/web/scrapers/maven.ts +138 -0
  1235. package/src/web/scrapers/mdn.ts +173 -0
  1236. package/src/web/scrapers/metacpan.ts +222 -0
  1237. package/src/web/scrapers/musicbrainz.ts +250 -0
  1238. package/src/web/scrapers/npm.ts +98 -0
  1239. package/src/web/scrapers/nuget.ts +183 -0
  1240. package/src/web/scrapers/nvd.ts +222 -0
  1241. package/src/web/scrapers/ollama.ts +239 -0
  1242. package/src/web/scrapers/open-vsx.ts +106 -0
  1243. package/src/web/scrapers/opencorporates.ts +292 -0
  1244. package/src/web/scrapers/openlibrary.ts +336 -0
  1245. package/src/web/scrapers/orcid.ts +286 -0
  1246. package/src/web/scrapers/osv.ts +176 -0
  1247. package/src/web/scrapers/packagist.ts +160 -0
  1248. package/src/web/scrapers/pub-dev.ts +143 -0
  1249. package/src/web/scrapers/pubmed.ts +211 -0
  1250. package/src/web/scrapers/pypi.ts +112 -0
  1251. package/src/web/scrapers/rawg.ts +110 -0
  1252. package/src/web/scrapers/readthedocs.ts +120 -0
  1253. package/src/web/scrapers/reddit.ts +95 -0
  1254. package/src/web/scrapers/repology.ts +251 -0
  1255. package/src/web/scrapers/rfc.ts +201 -0
  1256. package/src/web/scrapers/rubygems.ts +103 -0
  1257. package/src/web/scrapers/searchcode.ts +189 -0
  1258. package/src/web/scrapers/sec-edgar.ts +261 -0
  1259. package/src/web/scrapers/semantic-scholar.ts +171 -0
  1260. package/src/web/scrapers/snapcraft.ts +187 -0
  1261. package/src/web/scrapers/sourcegraph.ts +336 -0
  1262. package/src/web/scrapers/spdx.ts +108 -0
  1263. package/src/web/scrapers/spotify.ts +198 -0
  1264. package/src/web/scrapers/stackoverflow.ts +120 -0
  1265. package/src/web/scrapers/terraform.ts +277 -0
  1266. package/src/web/scrapers/tldr.ts +47 -0
  1267. package/src/web/scrapers/twitter.ts +94 -0
  1268. package/src/web/scrapers/types.ts +354 -0
  1269. package/src/web/scrapers/utils.ts +109 -0
  1270. package/src/web/scrapers/vimeo.ts +133 -0
  1271. package/src/web/scrapers/vscode-marketplace.ts +187 -0
  1272. package/src/web/scrapers/w3c.ts +156 -0
  1273. package/src/web/scrapers/wikidata.ts +344 -0
  1274. package/src/web/scrapers/wikipedia.ts +84 -0
  1275. package/src/web/scrapers/youtube.ts +325 -0
  1276. package/src/web/search/index.ts +317 -0
  1277. package/src/web/search/provider.ts +169 -0
  1278. package/src/web/search/providers/anthropic.ts +343 -0
  1279. package/src/web/search/providers/base.ts +90 -0
  1280. package/src/web/search/providers/brave.ts +152 -0
  1281. package/src/web/search/providers/codex.ts +593 -0
  1282. package/src/web/search/providers/exa.ts +400 -0
  1283. package/src/web/search/providers/gemini.ts +518 -0
  1284. package/src/web/search/providers/jina.ts +111 -0
  1285. package/src/web/search/providers/kagi.ts +86 -0
  1286. package/src/web/search/providers/kimi.ts +196 -0
  1287. package/src/web/search/providers/parallel.ts +225 -0
  1288. package/src/web/search/providers/perplexity-auth.ts +133 -0
  1289. package/src/web/search/providers/perplexity.ts +866 -0
  1290. package/src/web/search/providers/searxng.ts +325 -0
  1291. package/src/web/search/providers/synthetic.ts +114 -0
  1292. package/src/web/search/providers/tavily.ts +176 -0
  1293. package/src/web/search/providers/utils.ts +128 -0
  1294. package/src/web/search/providers/zai.ts +333 -0
  1295. package/src/web/search/render.ts +262 -0
  1296. package/src/web/search/types.ts +462 -0
  1297. package/src/web/search/utils.ts +17 -0
  1298. package/src/workspace-tree.ts +326 -0
@@ -0,0 +1,1558 @@
1
+ /**
2
+ * TUI rendering for task tool.
3
+ *
4
+ * Provides renderCall and renderResult functions for displaying
5
+ * task execution in the terminal UI.
6
+ */
7
+ import path from "node:path";
8
+ import type { Component } from "@oh-my-pi/pi-tui";
9
+ import { Container, Markdown, Text } from "@oh-my-pi/pi-tui";
10
+ import { formatNumber } from "@oh-my-pi/pi-utils";
11
+ import { settings } from "../config/settings";
12
+ import type { RenderResultOptions } from "../extensibility/custom-tools/types";
13
+ import { formatContextUsage } from "../modes/components/status-line/context-thresholds";
14
+ import { getMarkdownTheme, type Theme } from "../modes/theme/theme";
15
+ import {
16
+ formatBadge,
17
+ formatDuration,
18
+ formatExpandHint,
19
+ formatMoreItems,
20
+ formatStatusIcon,
21
+ replaceTabs,
22
+ type ToolUIStatus,
23
+ truncateToWidth,
24
+ } from "../tools/render-utils";
25
+ import {
26
+ type FindingPriority,
27
+ getPriorityInfo,
28
+ PRIORITY_LABELS,
29
+ parseReportFindingDetails,
30
+ type ReportFindingDetails,
31
+ type SubmitReviewDetails,
32
+ } from "../tools/review";
33
+ import { framedBlock, renderStatusLine } from "../tui";
34
+ import { repairDoubleEncodedJsonString } from "./repair-args";
35
+ import { subprocessToolRegistry } from "./subprocess-tool-registry";
36
+ import type { AgentProgress, SingleResult, TaskItem, TaskParams, TaskToolDetails } from "./types";
37
+
38
+ /** Render context threaded in from `ToolExecutionComponent.#buildRenderContext`. */
39
+ interface TaskRenderContext {
40
+ hasResult?: boolean;
41
+ /**
42
+ * The block left the transcript live region (detached spawn the transcript
43
+ * has moved past, or a sealed block): progress rows render static gray, so
44
+ * commit-eligible rows do not repaint after entering native scrollback.
45
+ */
46
+ frozen?: boolean;
47
+ }
48
+ type TaskRenderOptions = RenderResultOptions & { renderContext?: TaskRenderContext };
49
+
50
+ const MAX_NESTED_TASK_RENDER_DEPTH = 8;
51
+
52
+ function renderNestedCycleLine(theme: Theme): string {
53
+ return theme.fg("dim", "… nested task progress already shown");
54
+ }
55
+
56
+ /**
57
+ * Get status icon for agent state.
58
+ * For running status, uses animated spinner if spinnerFrame is provided.
59
+ * Maps AgentProgress status to styled icon format.
60
+ */
61
+ function getStatusIcon(status: AgentProgress["status"], theme: Theme, spinnerFrame?: number): string {
62
+ switch (status) {
63
+ case "pending":
64
+ return formatStatusIcon("pending", theme);
65
+ case "running":
66
+ return formatStatusIcon("running", theme, spinnerFrame);
67
+ case "completed":
68
+ return formatStatusIcon("success", theme);
69
+ case "failed":
70
+ return formatStatusIcon("error", theme);
71
+ case "aborted":
72
+ return formatStatusIcon("aborted", theme);
73
+ }
74
+ }
75
+
76
+ /**
77
+ * Append tool-count, context, and cost stats to a status line string.
78
+ */
79
+ function appendAgentStats(
80
+ line: string,
81
+ opts: {
82
+ toolCount?: number;
83
+ requests?: number;
84
+ tokens: number;
85
+ contextTokens?: number;
86
+ contextWindow?: number;
87
+ cost: number;
88
+ resolvedModel?: string;
89
+ showResolvedModelBadge?: boolean;
90
+ },
91
+ theme: Theme,
92
+ ): string {
93
+ if (opts.toolCount) {
94
+ line += `${theme.sep.dot}${theme.fg("dim", `${formatNumber(opts.toolCount)} ${theme.icon.extensionTool}`)}`;
95
+ }
96
+ if (opts.requests) {
97
+ line += `${theme.sep.dot}${theme.fg("dim", `${formatNumber(opts.requests)} req`)}`;
98
+ }
99
+ // Current per-turn context — match the status line's `<pct>%/<window>` gauge (e.g. `5.1%/1M`).
100
+ if (opts.contextTokens && opts.contextTokens > 0) {
101
+ const ctx =
102
+ opts.contextWindow && opts.contextWindow > 0
103
+ ? formatContextUsage((opts.contextTokens / opts.contextWindow) * 100, opts.contextWindow)
104
+ : `${formatNumber(opts.contextTokens)}`;
105
+ line += `${theme.sep.dot}${theme.fg("dim", ctx)}`;
106
+ }
107
+ if (opts.cost > 0) {
108
+ line += `${theme.sep.dot}${theme.fg("statusLineCost", `$${opts.cost.toFixed(2)}`)}`;
109
+ }
110
+ if (opts.resolvedModel && opts.showResolvedModelBadge) {
111
+ line += `${theme.sep.dot}${theme.fg("dim", truncateToWidth(replaceTabs(opts.resolvedModel), 30))}`;
112
+ }
113
+ return line;
114
+ }
115
+
116
+ function formatFindingSummary(findings: ReportFindingDetails[], theme: Theme): string {
117
+ if (findings.length === 0) return theme.fg("dim", "Findings: none");
118
+
119
+ const counts: { [P in FindingPriority]?: number } = {};
120
+ for (const finding of findings) {
121
+ counts[finding.priority] = (counts[finding.priority] ?? 0) + 1;
122
+ }
123
+
124
+ const parts: string[] = [];
125
+ for (const label of PRIORITY_LABELS) {
126
+ const { symbol, color } = getPriorityInfo(label);
127
+ const count = counts[label] ?? 0;
128
+ const text = theme.fg(color, `${label}:${count}`);
129
+ parts.push(theme.styledSymbol(symbol, color) ? `${theme.styledSymbol(symbol, color)} ${text}` : text);
130
+ }
131
+
132
+ return `${theme.fg("dim", "Findings:")} ${parts.join(theme.sep.dot)}`;
133
+ }
134
+
135
+ function normalizeReportFindings(value: unknown): ReportFindingDetails[] {
136
+ if (!Array.isArray(value)) return [];
137
+ const findings: ReportFindingDetails[] = [];
138
+ for (const item of value) {
139
+ const finding = parseReportFindingDetails(item);
140
+ if (finding) findings.push(finding);
141
+ }
142
+ return findings;
143
+ }
144
+
145
+ /**
146
+ * Normalize the `yield` slot of `extractedToolData` into an array of
147
+ * yield-detail records. The subprocess executor always populates this slot as
148
+ * `unknown[]` (see `executor.ts` `extractData` handler), but the renderer
149
+ * MUST also tolerate a stray single object — optional chaining short-circuits
150
+ * on `null`/`undefined` only, so calling `.map` on a plain object would throw
151
+ * `TypeError: completeData?.map is not a function` and crash the TUI.
152
+ * A single object is wrapped as a 1-element array so the review verdict still
153
+ * renders; non-object primitives drop out.
154
+ */
155
+ function normalizeYieldData(value: unknown): Array<{ data: unknown }> {
156
+ if (Array.isArray(value)) {
157
+ return value.filter((item): item is { data: unknown } => item !== null && typeof item === "object");
158
+ }
159
+ if (value !== null && typeof value === "object") {
160
+ return [value as { data: unknown }];
161
+ }
162
+ return [];
163
+ }
164
+
165
+ function formatJsonScalar(value: unknown, _theme: Theme): string {
166
+ if (value === null) return "null";
167
+ if (typeof value === "string") {
168
+ const trimmed = truncateToWidth(value, 70);
169
+ return `"${trimmed}"`;
170
+ }
171
+ if (typeof value === "number" || typeof value === "boolean") return String(value);
172
+ return "";
173
+ }
174
+
175
+ export function formatTaskId(id: string): string {
176
+ // Ids are name-based (e.g. "Anna", "Anna-2"); a "." separates nesting levels
177
+ // (e.g. "Anna.Bob"). Render the hierarchy with a ">" breadcrumb.
178
+ const segments = id.split(".");
179
+ return segments.length < 2 ? id : segments.join(">");
180
+ }
181
+
182
+ const MISSING_YIELD_WARNING_PREFIX = "SYSTEM WARNING: Subagent exited without calling yield tool";
183
+
184
+ function extractMissingYieldWarning(output: string): { warning?: string; rest: string } {
185
+ const lines = output.split("\n");
186
+ const firstLine = lines[0]?.trim() ?? "";
187
+ if (!firstLine.startsWith(MISSING_YIELD_WARNING_PREFIX)) {
188
+ return { rest: output };
189
+ }
190
+ const rest = lines
191
+ .slice(1)
192
+ .join("\n")
193
+ .replace(/^\s*\n+/, "");
194
+ return { warning: firstLine, rest };
195
+ }
196
+
197
+ function buildTreePrefix(ancestors: boolean[], theme: Theme): string {
198
+ return ancestors.map(hasNext => (hasNext ? `${theme.tree.vertical} ` : " ")).join("");
199
+ }
200
+
201
+ function renderJsonTreeLines(
202
+ value: unknown,
203
+ theme: Theme,
204
+ maxDepth: number,
205
+ maxLines: number,
206
+ ): { lines: string[]; truncated: boolean } {
207
+ const lines: string[] = [];
208
+ let truncated = false;
209
+
210
+ const iconObject = theme.styledSymbol("icon.folder", "muted");
211
+ const iconArray = theme.styledSymbol("icon.package", "muted");
212
+ const iconScalar = theme.styledSymbol("icon.file", "muted");
213
+
214
+ const pushLine = (line: string) => {
215
+ if (lines.length >= maxLines) {
216
+ truncated = true;
217
+ return false;
218
+ }
219
+ lines.push(line);
220
+ return true;
221
+ };
222
+
223
+ const renderNode = (val: unknown, key: string | undefined, ancestors: boolean[], isLast: boolean, depth: number) => {
224
+ if (lines.length >= maxLines) {
225
+ truncated = true;
226
+ return;
227
+ }
228
+
229
+ const connector = isLast ? theme.tree.last : theme.tree.branch;
230
+ const prefix = `${buildTreePrefix(ancestors, theme)}${theme.fg("dim", connector)} `;
231
+ const scalar = formatJsonScalar(val, theme);
232
+
233
+ if (scalar) {
234
+ const label = key ? theme.fg("muted", key) : theme.fg("muted", "value");
235
+ pushLine(`${prefix}${iconScalar} ${label}: ${theme.fg("dim", scalar)}`);
236
+ return;
237
+ }
238
+
239
+ if (Array.isArray(val)) {
240
+ const header = key ? theme.fg("muted", key) : theme.fg("muted", "array");
241
+ pushLine(`${prefix}${iconArray} ${header}`);
242
+ if (val.length === 0) {
243
+ pushLine(
244
+ `${buildTreePrefix([...ancestors, !isLast], theme)}${theme.fg("dim", theme.tree.last)} ${theme.fg(
245
+ "dim",
246
+ "[]",
247
+ )}`,
248
+ );
249
+ return;
250
+ }
251
+ if (depth >= maxDepth) {
252
+ pushLine(
253
+ `${buildTreePrefix([...ancestors, !isLast], theme)}${theme.fg("dim", theme.tree.last)} ${theme.fg(
254
+ "dim",
255
+ "…",
256
+ )}`,
257
+ );
258
+ return;
259
+ }
260
+ const nextAncestors = [...ancestors, !isLast];
261
+ for (let i = 0; i < val.length; i++) {
262
+ renderNode(val[i], `[${i}]`, nextAncestors, i === val.length - 1, depth + 1);
263
+ if (lines.length >= maxLines) {
264
+ truncated = true;
265
+ return;
266
+ }
267
+ }
268
+ return;
269
+ }
270
+
271
+ if (val && typeof val === "object") {
272
+ const header = key ? theme.fg("muted", key) : theme.fg("muted", "object");
273
+ pushLine(`${prefix}${iconObject} ${header}`);
274
+ const entries = Object.entries(val as Record<string, unknown>);
275
+ if (entries.length === 0) {
276
+ pushLine(
277
+ `${buildTreePrefix([...ancestors, !isLast], theme)}${theme.fg("dim", theme.tree.last)} ${theme.fg(
278
+ "dim",
279
+ "{}",
280
+ )}`,
281
+ );
282
+ return;
283
+ }
284
+ if (depth >= maxDepth) {
285
+ pushLine(
286
+ `${buildTreePrefix([...ancestors, !isLast], theme)}${theme.fg("dim", theme.tree.last)} ${theme.fg(
287
+ "dim",
288
+ "…",
289
+ )}`,
290
+ );
291
+ return;
292
+ }
293
+ const nextAncestors = [...ancestors, !isLast];
294
+ for (let i = 0; i < entries.length; i++) {
295
+ const [childKey, child] = entries[i];
296
+ renderNode(child, childKey, nextAncestors, i === entries.length - 1, depth + 1);
297
+ if (lines.length >= maxLines) {
298
+ truncated = true;
299
+ return;
300
+ }
301
+ }
302
+ return;
303
+ }
304
+
305
+ const label = key ? theme.fg("muted", key) : theme.fg("muted", "value");
306
+ pushLine(`${prefix}${iconScalar} ${label}: ${theme.fg("dim", String(val))}`);
307
+ };
308
+
309
+ const renderRoot = (val: unknown) => {
310
+ if (Array.isArray(val)) {
311
+ for (let i = 0; i < val.length; i++) {
312
+ renderNode(val[i], `[${i}]`, [], i === val.length - 1, 1);
313
+ if (lines.length >= maxLines) {
314
+ truncated = true;
315
+ return;
316
+ }
317
+ }
318
+ return;
319
+ }
320
+ if (val && typeof val === "object") {
321
+ const entries = Object.entries(val as Record<string, unknown>);
322
+ for (let i = 0; i < entries.length; i++) {
323
+ const [childKey, child] = entries[i];
324
+ renderNode(child, childKey, [], i === entries.length - 1, 1);
325
+ if (lines.length >= maxLines) {
326
+ truncated = true;
327
+ return;
328
+ }
329
+ }
330
+ return;
331
+ }
332
+ renderNode(val, undefined, [], true, 0);
333
+ };
334
+
335
+ renderRoot(value);
336
+
337
+ return { lines, truncated };
338
+ }
339
+
340
+ function renderOutputSection(
341
+ output: string,
342
+ continuePrefix: string,
343
+ expanded: boolean,
344
+ theme: Theme,
345
+ maxCollapsed = 3,
346
+ maxExpanded = 10,
347
+ warning?: string,
348
+ ): string[] {
349
+ const lines: string[] = [];
350
+ const trimmedOutput = output.trimEnd();
351
+ if (!trimmedOutput && !warning) return lines;
352
+
353
+ if (warning) {
354
+ lines.push(`${continuePrefix}${theme.fg("dim", "Output")}`);
355
+ lines.push(
356
+ `${continuePrefix} ${theme.fg("warning", theme.status.warning)} ${theme.fg(
357
+ "dim",
358
+ truncateToWidth(warning, 80),
359
+ )}`,
360
+ );
361
+
362
+ if (!trimmedOutput) {
363
+ return lines;
364
+ }
365
+
366
+ if (trimmedOutput.startsWith("{") || trimmedOutput.startsWith("[")) {
367
+ try {
368
+ const parsed = JSON.parse(trimmedOutput);
369
+
370
+ if (!expanded) {
371
+ lines.push(`${continuePrefix} ${theme.fg("dim", formatOutputInline(parsed, theme))}`);
372
+ return lines;
373
+ }
374
+
375
+ const tree = renderJsonTreeLines(parsed, theme, expanded ? 6 : 2, expanded ? 24 : 6);
376
+ if (tree.lines.length > 0) {
377
+ for (const line of tree.lines) {
378
+ lines.push(`${continuePrefix} ${line}`);
379
+ }
380
+ if (tree.truncated) {
381
+ lines.push(`${continuePrefix} ${theme.fg("dim", "…")}`);
382
+ }
383
+ return lines;
384
+ }
385
+ } catch {
386
+ // Fall back to raw output
387
+ }
388
+ }
389
+
390
+ const outputLines = output.trimEnd().split("\n");
391
+ const previewCount = expanded ? maxExpanded : maxCollapsed;
392
+ for (const line of outputLines.slice(0, previewCount)) {
393
+ lines.push(`${continuePrefix} ${theme.fg("dim", truncateToWidth(replaceTabs(line), 70))}`);
394
+ }
395
+
396
+ if (outputLines.length > previewCount) {
397
+ lines.push(
398
+ `${continuePrefix} ${theme.fg("dim", formatMoreItems(outputLines.length - previewCount, "line"))}`,
399
+ );
400
+ }
401
+
402
+ return lines;
403
+ }
404
+
405
+ if (trimmedOutput.startsWith("{") || trimmedOutput.startsWith("[")) {
406
+ try {
407
+ const parsed = JSON.parse(trimmedOutput);
408
+
409
+ // Collapsed: inline format like Args
410
+ if (!expanded) {
411
+ lines.push(`${continuePrefix}${theme.fg("dim", formatOutputInline(parsed, theme))}`);
412
+ return lines;
413
+ }
414
+
415
+ // Expanded: tree format
416
+ lines.push(`${continuePrefix}${theme.fg("dim", "Output")}`);
417
+ const tree = renderJsonTreeLines(parsed, theme, expanded ? 6 : 2, expanded ? 24 : 6);
418
+ if (tree.lines.length > 0) {
419
+ for (const line of tree.lines) {
420
+ lines.push(`${continuePrefix} ${line}`);
421
+ }
422
+ if (tree.truncated) {
423
+ lines.push(`${continuePrefix} ${theme.fg("dim", "…")}`);
424
+ }
425
+ return lines;
426
+ }
427
+ } catch {
428
+ // Fall back to raw output
429
+ }
430
+ }
431
+
432
+ lines.push(`${continuePrefix}${theme.fg("dim", "Output")}`);
433
+
434
+ const outputLines = output.trimEnd().split("\n");
435
+ const previewCount = expanded ? maxExpanded : maxCollapsed;
436
+ for (const line of outputLines.slice(0, previewCount)) {
437
+ lines.push(`${continuePrefix} ${theme.fg("dim", truncateToWidth(replaceTabs(line), 70))}`);
438
+ }
439
+
440
+ if (outputLines.length > previewCount) {
441
+ lines.push(`${continuePrefix} ${theme.fg("dim", formatMoreItems(outputLines.length - previewCount, "line"))}`);
442
+ }
443
+
444
+ return lines;
445
+ }
446
+
447
+ function renderTaskSection(
448
+ task: string,
449
+ continuePrefix: string,
450
+ expanded: boolean,
451
+ theme: Theme,
452
+ maxExpanded = 20,
453
+ ): string[] {
454
+ const lines: string[] = [];
455
+ const trimmed = task.trim();
456
+ if (!expanded || !trimmed) return lines;
457
+
458
+ lines.push(`${continuePrefix}${theme.fg("dim", "Task")}`);
459
+ const taskLines = trimmed.split("\n");
460
+ for (const line of taskLines.slice(0, maxExpanded)) {
461
+ lines.push(`${continuePrefix} ${theme.fg("dim", truncateToWidth(replaceTabs(line), 70))}`);
462
+ }
463
+ if (taskLines.length > maxExpanded) {
464
+ lines.push(`${continuePrefix} ${theme.fg("dim", formatMoreItems(taskLines.length - maxExpanded, "line"))}`);
465
+ }
466
+
467
+ return lines;
468
+ }
469
+
470
+ function formatScalarInline(value: unknown, maxLen: number, _theme: Theme): string {
471
+ if (value === null) return "null";
472
+ if (value === undefined) return "undefined";
473
+ if (typeof value === "boolean") return String(value);
474
+ if (typeof value === "number") return String(value);
475
+ if (typeof value === "string") {
476
+ const firstLine = value.split("\n")[0].trim();
477
+ if (firstLine.length === 0) return `"" (${value.split("\n").length} lines)`;
478
+ const preview = truncateToWidth(firstLine, maxLen);
479
+ if (value.includes("\n")) return `"${preview}…" (${value.split("\n").length} lines)`;
480
+ return `"${preview}"`;
481
+ }
482
+ if (Array.isArray(value)) return `[${value.length} items]`;
483
+ if (typeof value === "object") {
484
+ const keys = Object.keys(value);
485
+ return `{${keys.length} keys}`;
486
+ }
487
+ return String(value);
488
+ }
489
+
490
+ function formatOutputInline(data: unknown, theme: Theme, maxWidth = 80): string {
491
+ if (data === null || data === undefined) return "Output: none";
492
+
493
+ // For scalars, show directly
494
+ if (typeof data !== "object") {
495
+ return `Output: ${formatScalarInline(data, 60, theme)}`;
496
+ }
497
+
498
+ // For arrays, show count and first element preview
499
+ if (Array.isArray(data)) {
500
+ if (data.length === 0) return "Output: []";
501
+ const preview = formatScalarInline(data[0], 40, theme);
502
+ return `Output: [${data.length} items] ${preview}${data.length > 1 ? "…" : ""}`;
503
+ }
504
+
505
+ // For objects, show key=value pairs inline
506
+ const entries = Object.entries(data as Record<string, unknown>);
507
+ if (entries.length === 0) return "Output: {}";
508
+
509
+ const pairs: string[] = [];
510
+ let totalLen = "Output: ".length;
511
+
512
+ for (const [key, value] of entries) {
513
+ const valueStr = formatScalarInline(value, 24, theme);
514
+ const pairStr = `${key}=${valueStr}`;
515
+ const addLen = pairs.length > 0 ? pairStr.length + 2 : pairStr.length; // +2 for ", "
516
+
517
+ if (totalLen + addLen > maxWidth && pairs.length > 0) {
518
+ pairs.push("…");
519
+ break;
520
+ }
521
+
522
+ pairs.push(pairStr);
523
+ totalLen += addLen;
524
+ }
525
+
526
+ return `Output: ${pairs.join(", ")}`;
527
+ }
528
+
529
+ /**
530
+ * Render the call preview lines for the single spawned agent. The
531
+ * args stream in token by token, so every field access is defensive.
532
+ */
533
+ function renderTaskCallLines(args: Partial<TaskParams> | undefined, theme: Theme): string[] {
534
+ if (!args) return [];
535
+ const bullet = theme.fg("dim", "•");
536
+ const lines: string[] = [];
537
+
538
+ const rawId = typeof args.id === "string" ? args.id.trim() : "";
539
+ const idLabel = rawId ? formatTaskId(rawId) : "";
540
+ const desc = typeof args.description === "string" ? args.description.trim() : "";
541
+ if (idLabel || desc) {
542
+ let line = `${bullet} ${theme.fg("accent", theme.bold(idLabel || "agent"))}`;
543
+ if (desc) {
544
+ line += `: ${theme.fg("muted", truncateToWidth(replaceTabs(desc), 64))}`;
545
+ }
546
+ lines.push(line);
547
+ }
548
+ lines.push(...renderTaskItemLines(args.tasks, theme));
549
+ return lines;
550
+ }
551
+
552
+ /**
553
+ * Agent rows shown per collapsed task list; the rest fold into a single
554
+ * `… N more agents` summary line (expand uncaps).
555
+ */
556
+ const COLLAPSED_AGENT_LIMIT = 4;
557
+
558
+ /**
559
+ * Render the per-item list (`id` + ui `description`) for a batch call's
560
+ * streaming preview. The args stream in token by token, so the array grows
561
+ * over time and trailing entries may be partially parsed — every field access
562
+ * is defensive.
563
+ */
564
+ function renderTaskItemLines(tasks: TaskItem[] | undefined, theme: Theme): string[] {
565
+ if (!Array.isArray(tasks) || tasks.length === 0) return [];
566
+
567
+ const bullet = theme.fg("dim", "•");
568
+ const cap = Math.min(tasks.length, COLLAPSED_AGENT_LIMIT);
569
+ const lines: string[] = [];
570
+ for (let i = 0; i < cap; i++) {
571
+ const task = tasks[i] as Partial<TaskItem> | undefined;
572
+ const rawId = typeof task?.id === "string" ? task.id.trim() : "";
573
+ const idLabel = rawId ? formatTaskId(rawId) : `#${i + 1}`;
574
+ let line = `${bullet} ${theme.fg("accent", theme.bold(idLabel))}`;
575
+ const desc = typeof task?.description === "string" ? task.description.trim() : "";
576
+ if (desc) {
577
+ line += `: ${theme.fg("muted", truncateToWidth(replaceTabs(desc), 64))}`;
578
+ }
579
+ if (task?.isolated === true) {
580
+ line += theme.fg("dim", " [isolated]");
581
+ }
582
+ lines.push(line);
583
+ }
584
+ if (cap < tasks.length) {
585
+ lines.push(`${bullet} ${theme.fg("dim", formatMoreItems(tasks.length - cap, "agent"))}`);
586
+ }
587
+ return lines;
588
+ }
589
+
590
+ /** One renderable frame section: optional label, body rows, leading divider. */
591
+ type TaskRenderSection = { label?: string; lines: readonly string[]; separator?: boolean };
592
+ type AssignmentSectionRenderer = (width: number) => TaskRenderSection;
593
+
594
+ // Default output-block layout is: left border + one-cell content inset + right
595
+ // border. Render markdown at that inner width so the output block does not need
596
+ // to rewrap already-rendered assignment lines.
597
+ const ASSIGNMENT_FRAME_INSET = 3;
598
+
599
+ /**
600
+ * Build the assignment section (the markdown brief handed to the subagent).
601
+ * Rendered in both the streaming call preview and the result frame so the
602
+ * brief stays visible for the whole task lifecycle — not just until the first
603
+ * progress snapshot replaces the call view.
604
+ */
605
+ function createAssignmentSectionRenderer(
606
+ args: Partial<TaskParams> | undefined,
607
+ theme: Theme,
608
+ ): AssignmentSectionRenderer | undefined {
609
+ // `renderResult` receives the raw tool args (unlike `renderCall`, which is
610
+ // fed through `repairTaskParams`), so undo any per-field double-encoding
611
+ // here too. The repair is idempotent on already-clean text.
612
+ const assignment = repairDoubleEncodedJsonString(typeof args?.assignment === "string" ? args.assignment : "").trim();
613
+ if (!assignment) return undefined;
614
+ return createMarkdownSectionRenderer(assignment, theme);
615
+ }
616
+
617
+ /**
618
+ * Build the shared-context section (the `# Goal / # Constraints` background a
619
+ * batch call hands every subagent). Rendered like the assignment brief so the
620
+ * shared background stays visible for the whole task lifecycle.
621
+ */
622
+ function createContextSectionRenderer(
623
+ args: Partial<TaskParams> | undefined,
624
+ theme: Theme,
625
+ ): AssignmentSectionRenderer | undefined {
626
+ const context = repairDoubleEncodedJsonString(typeof args?.context === "string" ? args.context : "").trim();
627
+ if (!context) return undefined;
628
+ return createMarkdownSectionRenderer(context, theme);
629
+ }
630
+
631
+ function createMarkdownSectionRenderer(text: string, theme: Theme): AssignmentSectionRenderer {
632
+ const markdown = new Markdown(text, 0, 0, getMarkdownTheme(), {
633
+ color: line => theme.fg("muted", line),
634
+ });
635
+ return width => ({ lines: markdown.render(Math.max(1, width - ASSIGNMENT_FRAME_INSET)) });
636
+ }
637
+
638
+ /**
639
+ * Render the tool call arguments.
640
+ */
641
+ export function renderCall(args: TaskParams, options: TaskRenderOptions, theme: Theme): Component {
642
+ const showIsolated = "isolated" in args && args.isolated === true;
643
+ // Dispatch glyph from the first frame: spawning is non-blocking, so a
644
+ // pending/hourglass icon would misread the call as something the turn
645
+ // waits on.
646
+ const header = renderStatusLine(
647
+ { iconOverride: theme.styledSymbol("tool.task", "accent"), title: "Task", description: args.agent },
648
+ theme,
649
+ );
650
+ const assignmentSection = createAssignmentSectionRenderer(args, theme);
651
+ const contextSection = createContextSectionRenderer(args, theme);
652
+ return framedBlock(theme, width => {
653
+ const sections: Array<{ label?: string; lines: readonly string[]; separator?: boolean }> = [];
654
+
655
+ // The call preview only exists to surface the dispatched agent while the
656
+ // args stream in. Once a result snapshot exists, `renderResult` draws the
657
+ // same agent (and the assignment brief) itself, so showing it here would
658
+ // repeat what the result frame already shows.
659
+ if (!options.renderContext?.hasResult) {
660
+ // Mirror renderResult's layout — context, assignment, then the
661
+ // per-agent list — so the agent rows do not jump from above the
662
+ // brief to below it when the first progress snapshot replaces the
663
+ // call view. This also matches the schema's field order (`context`
664
+ // streams before `tasks`), so the streaming preview grows
665
+ // append-only instead of inserting agent rows above the
666
+ // already-rendered markdown and pushing it down on every item.
667
+ if (contextSection) sections.push(contextSection(width));
668
+ if (assignmentSection) sections.push(assignmentSection(width));
669
+ const callLines = renderTaskCallLines(args, theme);
670
+ // Guarded: an empty trailing section would still draw its divider.
671
+ if (callLines.length > 0) sections.push({ separator: true, lines: callLines });
672
+ }
673
+
674
+ return {
675
+ header,
676
+ headerMeta: showIsolated ? "isolated" : undefined,
677
+ sections,
678
+ state: "pending",
679
+ borderColor: "borderMuted",
680
+ width,
681
+ };
682
+ });
683
+ }
684
+
685
+ /**
686
+ * Render streaming progress for a single agent.
687
+ */
688
+ function renderAgentProgress(
689
+ progress: AgentProgress,
690
+ prefix: string,
691
+ continuePrefix: string,
692
+ expanded: boolean,
693
+ theme: Theme,
694
+ spinnerFrame?: number,
695
+ frozen = false,
696
+ seenNestedTasks?: WeakSet<object>,
697
+ nestedDepth = 0,
698
+ ): string[] {
699
+ const lines: string[] = [];
700
+
701
+ const icon = getStatusIcon(progress.status, theme, spinnerFrame);
702
+ const iconColor =
703
+ progress.status === "completed"
704
+ ? "success"
705
+ : progress.status === "failed" || progress.status === "aborted"
706
+ ? "error"
707
+ : "accent";
708
+
709
+ // Main status line: id: description [status] · stats · ⟨agent⟩
710
+ const description = progress.description?.trim();
711
+ const displayId = formatTaskId(progress.id);
712
+ const titlePart = description ? `${theme.bold(displayId)}: ${description}` : displayId;
713
+ const indent = prefix ? `${prefix} ` : "";
714
+ let statusLine: string;
715
+ if (progress.status === "running" || progress.status === "pending") {
716
+ // Live (or queued) agents use the same dot finished rows keep: detached
717
+ // async spawns can stay "pending" while real work is running, so a
718
+ // pending/hourglass or spinner glyph reads wrong in the transcript. Keep
719
+ // the row static; the Task tool header already carries the dispatch icon.
720
+ const dot = theme.styledSymbol("status.done", frozen ? "dim" : "accent");
721
+ const nameColor = frozen ? "dim" : "accent";
722
+ const name = theme.fg(nameColor, description ? theme.bold(displayId) : displayId);
723
+ statusLine = `${indent}${dot} ${name}`;
724
+ if (description) {
725
+ statusLine += `${theme.fg(nameColor, ":")} ${theme.fg(nameColor, description)}`;
726
+ }
727
+ } else if (progress.status === "completed") {
728
+ // Finished rows keep the dot but settle from accent to the plain
729
+ // foreground: completion reads as a color change, not a new glyph.
730
+ statusLine = `${indent}${theme.styledSymbol("status.done", "text")} ${theme.fg("text", titlePart)}`;
731
+ } else {
732
+ statusLine = `${indent}${theme.fg(iconColor, icon)} ${theme.fg("accent", titlePart)}`;
733
+ }
734
+
735
+ // Show retry-blocked badge so the parent immediately sees that a child
736
+ // is sleeping on a provider 429, not silently progressing. Wins over the
737
+ // generic running marker because "we're waiting on a quota window" is
738
+ // the operationally meaningful state.
739
+ if (progress.retryState && progress.status === "running") {
740
+ statusLine += ` ${formatBadge("retrying", "warning", theme)}`;
741
+ } else if (progress.retryFailure && (progress.status === "failed" || progress.status === "aborted")) {
742
+ statusLine += ` ${formatBadge("rate-limited", "error", theme)}`;
743
+ } else if (progress.status === "failed" || progress.status === "aborted") {
744
+ const statusLabel = progress.status === "failed" ? "failed" : "aborted";
745
+ statusLine += ` ${formatBadge(statusLabel, iconColor, theme)}`;
746
+ }
747
+
748
+ const showBadge = settings.get("task.showResolvedModelBadge");
749
+ if (progress.status === "running") {
750
+ if (!description) {
751
+ const taskPreview = truncateToWidth(progress.assignment ?? progress.task, 40);
752
+ statusLine += ` ${theme.fg("muted", taskPreview)}`;
753
+ }
754
+ statusLine = appendAgentStats(statusLine, { ...progress, showResolvedModelBadge: showBadge }, theme);
755
+ } else if (progress.status === "completed") {
756
+ statusLine = appendAgentStats(statusLine, { ...progress, showResolvedModelBadge: showBadge }, theme);
757
+ }
758
+
759
+ lines.push(statusLine);
760
+
761
+ lines.push(...renderTaskSection(progress.assignment ?? progress.task, continuePrefix, expanded, theme));
762
+
763
+ // Current tool (if running) or most recent completed tool
764
+ if (progress.status === "running") {
765
+ if (progress.currentTool) {
766
+ let toolLine = `${continuePrefix}${theme.tree.hook} ${theme.fg("muted", progress.currentTool)}`;
767
+ const toolDetail = progress.lastIntent ?? progress.currentToolArgs;
768
+ if (toolDetail) {
769
+ toolLine += `: ${theme.fg("dim", truncateToWidth(replaceTabs(toolDetail), 40))}`;
770
+ }
771
+ if (progress.currentToolStartMs) {
772
+ const elapsed = Date.now() - progress.currentToolStartMs;
773
+ if (elapsed > 5000) {
774
+ toolLine += `${theme.sep.dot}${theme.fg("warning", formatDuration(elapsed))}`;
775
+ }
776
+ }
777
+ lines.push(toolLine);
778
+ } else if (progress.recentTools.length > 0) {
779
+ // Show most recent completed tool when idle between tools
780
+ const recent = progress.recentTools[0];
781
+ let toolLine = `${continuePrefix}${theme.tree.hook} ${theme.fg("dim", recent.tool)}`;
782
+ const toolDetail = progress.lastIntent ?? recent.args;
783
+ if (toolDetail) {
784
+ toolLine += `: ${theme.fg("dim", truncateToWidth(replaceTabs(toolDetail), 40))}`;
785
+ }
786
+ lines.push(toolLine);
787
+ }
788
+ }
789
+
790
+ // Retry detail line: surface why the subagent is paused and roughly how
791
+ // long until the next attempt. Without this, the parent UI would just
792
+ // keep spinning while a child sleeps on a 3-hour provider rate-limit.
793
+ if (progress.retryState && progress.status === "running") {
794
+ const remainingMs = Math.max(0, progress.retryState.startedAtMs + progress.retryState.delayMs - Date.now());
795
+ const waitLabel = remainingMs > 0 ? `in ${formatDuration(remainingMs)}` : "now";
796
+ const summary =
797
+ `retrying ${progress.retryState.attempt}/${progress.retryState.maxAttempts} ${waitLabel}: ` +
798
+ truncateToWidth(replaceTabs(progress.retryState.errorMessage), 60);
799
+ lines.push(`${continuePrefix}${theme.tree.hook} ${theme.fg("warning", summary)}`);
800
+ } else if (progress.retryFailure && progress.status !== "running") {
801
+ const summary = `auto-retry gave up after ${progress.retryFailure.attempt} attempt${
802
+ progress.retryFailure.attempt === 1 ? "" : "s"
803
+ }: ${truncateToWidth(replaceTabs(progress.retryFailure.errorMessage), 80)}`;
804
+ lines.push(`${continuePrefix}${theme.tree.hook} ${theme.fg("error", summary)}`);
805
+ }
806
+
807
+ // Render extracted tool data inline (e.g., review findings)
808
+ if (progress.extractedToolData) {
809
+ // For completed tasks, check for review verdict from yield tool
810
+ if (progress.status === "completed") {
811
+ const completeData = normalizeYieldData(progress.extractedToolData.yield);
812
+ const reportFindingData = normalizeReportFindings(progress.extractedToolData.report_finding);
813
+ const reviewData = completeData
814
+ .map(c => c.data as SubmitReviewDetails)
815
+ .filter(d => d && typeof d === "object" && "overall_correctness" in d);
816
+ if (reviewData.length > 0) {
817
+ const summary = reviewData[reviewData.length - 1];
818
+ const findings = reportFindingData;
819
+ lines.push(...renderReviewResult(summary, findings, continuePrefix, expanded, theme));
820
+ return lines; // Review result handles its own rendering
821
+ }
822
+ }
823
+
824
+ for (const toolName in progress.extractedToolData) {
825
+ const dataArray = progress.extractedToolData[toolName];
826
+ // Handle report_finding with tree formatting
827
+ if (toolName === "report_finding") {
828
+ const findings = normalizeReportFindings(dataArray);
829
+ if (findings.length === 0) continue;
830
+ lines.push(`${continuePrefix}${formatFindingSummary(findings, theme)}`);
831
+ lines.push(...renderFindings(findings, continuePrefix, expanded, theme));
832
+ continue;
833
+ }
834
+
835
+ // Nested `task` data has its own dedicated tree renderer below that
836
+ // also merges in the in-flight snapshot — skip the generic inline
837
+ // path so we don't render twice.
838
+ if (toolName === "task") continue;
839
+
840
+ const handler = subprocessToolRegistry.getHandler(toolName);
841
+ if (handler?.renderInline) {
842
+ const displayCount = expanded ? (dataArray as unknown[]).length : 3;
843
+ const recentData = (dataArray as unknown[]).slice(-displayCount);
844
+ for (const data of recentData) {
845
+ const component = handler.renderInline(data, theme);
846
+ if (component instanceof Text) {
847
+ lines.push(`${continuePrefix}${component.getText()}`);
848
+ }
849
+ }
850
+ if ((dataArray as unknown[]).length > displayCount) {
851
+ lines.push(
852
+ `${continuePrefix}${theme.fg(
853
+ "dim",
854
+ formatMoreItems((dataArray as unknown[]).length - displayCount, "item"),
855
+ )}`,
856
+ );
857
+ }
858
+ }
859
+ }
860
+ }
861
+
862
+ // Nested `task` tree: completed sub-calls from `extractedToolData.task` plus
863
+ // the in-flight snapshot (if any). Surfacing this in the live view means
864
+ // the user sees deep-tree progress without waiting for this agent to finish
865
+ // its own turn.
866
+ const completedTaskCalls = (progress.extractedToolData?.task as TaskToolDetails[] | undefined) ?? [];
867
+ const inflight = progress.inflightTaskDetails;
868
+ if (completedTaskCalls.length > 0 || inflight) {
869
+ const snapshots = inflight ? [...completedTaskCalls, inflight] : completedTaskCalls;
870
+ const nestedLines = renderNestedTaskTree(
871
+ snapshots,
872
+ expanded,
873
+ theme,
874
+ spinnerFrame,
875
+ frozen,
876
+ seenNestedTasks,
877
+ nestedDepth,
878
+ );
879
+ for (const line of nestedLines) {
880
+ lines.push(`${continuePrefix}${line}`);
881
+ }
882
+ }
883
+
884
+ // Expanded view: recent output and tools
885
+ if (expanded && progress.status === "running") {
886
+ const output = progress.recentOutput.join("\n");
887
+ lines.push(...renderOutputSection(output, continuePrefix, true, theme, 2, 6));
888
+ }
889
+
890
+ return lines;
891
+ }
892
+
893
+ /**
894
+ * Render review result with combined verdict + findings in tree structure.
895
+ */
896
+ function renderReviewResult(
897
+ summary: SubmitReviewDetails,
898
+ findings: ReportFindingDetails[],
899
+ continuePrefix: string,
900
+ expanded: boolean,
901
+ theme: Theme,
902
+ ): string[] {
903
+ const lines: string[] = [];
904
+
905
+ // Verdict line
906
+ const verdictColor = summary.overall_correctness === "correct" ? "success" : "error";
907
+ const isCorrect = summary.overall_correctness === "correct";
908
+ const verdictIcon = isCorrect
909
+ ? theme.styledSymbol("status.done", "accent")
910
+ : theme.fg(verdictColor, theme.status.error);
911
+ lines.push(
912
+ `${continuePrefix} Patch is ${theme.fg(verdictColor, summary.overall_correctness)} ${verdictIcon} ${theme.fg(
913
+ "dim",
914
+ `(${(summary.confidence * 100).toFixed(0)}% confidence)`,
915
+ )}`,
916
+ );
917
+
918
+ // Explanation preview (first ~80 chars when collapsed, full when expanded)
919
+ if (summary.explanation) {
920
+ if (expanded) {
921
+ lines.push(`${continuePrefix}${theme.fg("dim", "Summary")}`);
922
+ const explanationLines = summary.explanation.split("\n");
923
+ for (const line of explanationLines) {
924
+ lines.push(`${continuePrefix} ${theme.fg("dim", replaceTabs(line))}`);
925
+ }
926
+ } else {
927
+ // Preview: first sentence or ~100 chars (flatten tabs/newlines first)
928
+ const flat = replaceTabs(summary.explanation).replace(/[\r\n]+/g, " ");
929
+ const firstSentence = flat.split(/[.!?]/)[0].trim();
930
+ const preview = truncateToWidth(`${firstSentence}.`, 100);
931
+ lines.push(`${continuePrefix}${theme.fg("dim", preview)}`);
932
+ }
933
+ }
934
+
935
+ // Findings summary + list
936
+ lines.push(`${continuePrefix}${formatFindingSummary(findings, theme)}`);
937
+
938
+ if (findings.length > 0) {
939
+ lines.push(...renderFindings(findings, continuePrefix, expanded, theme));
940
+ }
941
+
942
+ return lines;
943
+ }
944
+
945
+ /**
946
+ * Render review findings list.
947
+ */
948
+ function renderFindings(
949
+ findings: ReportFindingDetails[],
950
+ continuePrefix: string,
951
+ expanded: boolean,
952
+ theme: Theme,
953
+ ): string[] {
954
+ const lines: string[] = [];
955
+
956
+ // Sort by priority (lower = more severe) when collapsed to show most important first
957
+ const sortedFindings = expanded
958
+ ? findings
959
+ : [...findings].sort((a, b) => getPriorityInfo(a.priority).ord - getPriorityInfo(b.priority).ord);
960
+ const displayCount = expanded ? sortedFindings.length : Math.min(3, sortedFindings.length);
961
+
962
+ for (let i = 0; i < displayCount; i++) {
963
+ const finding = sortedFindings[i];
964
+ const isLastFinding = i === displayCount - 1 && (expanded || sortedFindings.length <= 3);
965
+ const findingPrefix = isLastFinding ? theme.tree.last : theme.tree.branch;
966
+ const findingContinue = isLastFinding ? " " : `${theme.tree.vertical} `;
967
+
968
+ const { color } = getPriorityInfo(finding.priority);
969
+ const rawTitle = finding.title?.replace(/^\[P\d\]\s*/, "") ?? "Untitled";
970
+ const titleText = replaceTabs(rawTitle).replace(/[\r\n]+/g, " ");
971
+ const loc = `${path.basename(finding.file_path || "<unknown>")}:${finding.line_start}`;
972
+
973
+ lines.push(
974
+ `${continuePrefix}${findingPrefix} ${theme.fg(color, `[${finding.priority}]`)} ${titleText} ${theme.fg("dim", loc)}`,
975
+ );
976
+
977
+ // Show body when expanded
978
+ if (expanded && finding.body) {
979
+ // Wrap body text
980
+ const bodyLines = finding.body.split("\n");
981
+ for (const bodyLine of bodyLines) {
982
+ lines.push(`${continuePrefix}${findingContinue}${theme.fg("dim", replaceTabs(bodyLine))}`);
983
+ }
984
+ }
985
+ }
986
+
987
+ if (!expanded && findings.length > 3) {
988
+ lines.push(`${continuePrefix}${theme.fg("dim", formatMoreItems(findings.length - 3, "finding"))}`);
989
+ }
990
+
991
+ return lines;
992
+ }
993
+
994
+ /**
995
+ * Render final result for a single agent.
996
+ */
997
+ function renderAgentResult(
998
+ result: SingleResult,
999
+ prefix: string,
1000
+ continuePrefix: string,
1001
+ expanded: boolean,
1002
+ theme: Theme,
1003
+ seenNestedTasks?: WeakSet<object>,
1004
+ nestedDepth = 0,
1005
+ ): string[] {
1006
+ const lines: string[] = [];
1007
+
1008
+ const { warning: missingCompleteWarning, rest: outputWithoutWarning } = extractMissingYieldWarning(result.output);
1009
+ const aborted = result.aborted ?? false;
1010
+ const mergeFailed = !aborted && result.exitCode === 0 && !!result.error;
1011
+ const success = !aborted && result.exitCode === 0 && !result.error;
1012
+ const needsWarning = Boolean(missingCompleteWarning) && success;
1013
+ const icon = aborted
1014
+ ? theme.status.aborted
1015
+ : needsWarning
1016
+ ? theme.status.warning
1017
+ : success
1018
+ ? theme.styledSymbol("status.done", "text")
1019
+ : theme.status.error;
1020
+ const iconColor = needsWarning ? "warning" : success ? "success" : mergeFailed ? "warning" : "error";
1021
+ const statusText = aborted
1022
+ ? "aborted"
1023
+ : needsWarning
1024
+ ? "warning"
1025
+ : success
1026
+ ? "done"
1027
+ : mergeFailed
1028
+ ? "merge failed"
1029
+ : "failed";
1030
+
1031
+ // Main status line: id: description [status] · stats · ⟨agent⟩
1032
+ const description = result.description?.trim();
1033
+ const displayId = formatTaskId(result.id);
1034
+ const titlePart = description ? `${theme.bold(displayId)}: ${description}` : displayId;
1035
+ let statusLine = `${prefix ? `${prefix} ` : ""}${theme.fg(iconColor, icon)} ${theme.fg(
1036
+ success && !needsWarning ? "text" : "accent",
1037
+ titlePart,
1038
+ )} ${formatBadge(statusText, iconColor, theme)}`;
1039
+ const showBadge = settings.get("task.showResolvedModelBadge");
1040
+ statusLine = appendAgentStats(
1041
+ statusLine,
1042
+ {
1043
+ tokens: result.tokens,
1044
+ requests: result.requests,
1045
+ contextTokens: result.contextTokens,
1046
+ contextWindow: result.contextWindow,
1047
+ cost: result.usage?.cost.total ?? 0,
1048
+ resolvedModel: result.resolvedModel,
1049
+ showResolvedModelBadge: showBadge,
1050
+ },
1051
+ theme,
1052
+ );
1053
+ statusLine += `${theme.sep.dot}${theme.fg("dim", formatDuration(result.durationMs))}`;
1054
+
1055
+ if (result.truncated) {
1056
+ statusLine += ` ${theme.fg("warning", "[truncated]")}`;
1057
+ }
1058
+
1059
+ lines.push(statusLine);
1060
+
1061
+ lines.push(...renderTaskSection(result.assignment ?? result.task, continuePrefix, expanded, theme));
1062
+
1063
+ if (aborted && result.abortReason) {
1064
+ lines.push(
1065
+ `${continuePrefix}${theme.fg("error", theme.status.aborted)} ${theme.fg("dim", truncateToWidth(replaceTabs(result.abortReason), 80))}`,
1066
+ );
1067
+ }
1068
+ // Check for review result (yield with review schema + report_finding)
1069
+ // Check for review result (yield with review schema + report_finding).
1070
+ // `normalizeYieldData` guards against a stray non-array `yield` slot —
1071
+ // optional chaining on `.map` only short-circuits on null/undefined and
1072
+ // would otherwise crash the renderer with `TypeError: completeData?.map
1073
+ // is not a function` when the slot is a plain object (see issue #1987).
1074
+ const completeData = normalizeYieldData(result.extractedToolData?.yield);
1075
+ const reportFindingData = normalizeReportFindings(result.extractedToolData?.report_finding);
1076
+
1077
+ // Extract review verdict from yield tool's data field if it matches SubmitReviewDetails
1078
+ const reviewData = completeData
1079
+ .map(c => c.data as SubmitReviewDetails)
1080
+ .filter(d => d && typeof d === "object" && "overall_correctness" in d);
1081
+ const submitReviewData = reviewData.length > 0 ? reviewData : undefined;
1082
+
1083
+ if (submitReviewData) {
1084
+ // Use combined review renderer
1085
+ const summary = submitReviewData[submitReviewData.length - 1];
1086
+ const findings = reportFindingData;
1087
+ lines.push(...renderReviewResult(summary, findings, continuePrefix, expanded, theme));
1088
+ return lines;
1089
+ }
1090
+ if (reportFindingData.length > 0) {
1091
+ const hasCompleteData = completeData.length > 0;
1092
+ const message = hasCompleteData
1093
+ ? "Review verdict missing expected fields"
1094
+ : "Review incomplete (yield not called)";
1095
+ lines.push(`${continuePrefix}${theme.fg("warning", theme.status.warning)} ${theme.fg("dim", message)}`);
1096
+ lines.push(`${continuePrefix}${formatFindingSummary(reportFindingData, theme)}`);
1097
+ lines.push(...renderFindings(reportFindingData, continuePrefix, expanded, theme));
1098
+ return lines;
1099
+ }
1100
+
1101
+ // Check for extracted tool data with custom renderers (skip review tools)
1102
+ let hasCustomRendering = false;
1103
+ const deferredToolLines: string[] = [];
1104
+ if (result.extractedToolData) {
1105
+ for (const [toolName, dataArray] of Object.entries(result.extractedToolData)) {
1106
+ // Skip review tools - handled above
1107
+ if (toolName === "yield" || toolName === "report_finding") continue;
1108
+
1109
+ const isTaskTool = toolName === "task";
1110
+ if (isTaskTool && (dataArray as unknown[]).length > 0) {
1111
+ for (const line of renderNestedTaskResults(
1112
+ dataArray as TaskToolDetails[],
1113
+ expanded,
1114
+ theme,
1115
+ seenNestedTasks,
1116
+ nestedDepth,
1117
+ )) {
1118
+ deferredToolLines.push(`${continuePrefix}${line}`);
1119
+ }
1120
+ continue;
1121
+ }
1122
+
1123
+ const handler = subprocessToolRegistry.getHandler(toolName);
1124
+ if (handler?.renderFinal && (dataArray as unknown[]).length > 0) {
1125
+ const component = handler.renderFinal(dataArray as unknown[], theme, expanded);
1126
+ const target = lines;
1127
+ if (!isTaskTool) {
1128
+ hasCustomRendering = true;
1129
+ target.push(`${continuePrefix}${theme.fg("dim", `Tool: ${toolName}`)}`);
1130
+ }
1131
+ if (component instanceof Text) {
1132
+ // Prefix each line with continuePrefix
1133
+ const text = component.getText();
1134
+ for (const line of text.split("\n")) {
1135
+ target.push(`${continuePrefix}${line}`);
1136
+ }
1137
+ } else if (component instanceof Container) {
1138
+ // For containers, render each child
1139
+ for (const child of (component as Container).children) {
1140
+ if (child instanceof Text) {
1141
+ target.push(`${continuePrefix}${child.getText()}`);
1142
+ }
1143
+ }
1144
+ }
1145
+ }
1146
+ }
1147
+ }
1148
+
1149
+ if (hasCustomRendering && missingCompleteWarning) {
1150
+ lines.push(
1151
+ `${continuePrefix}${theme.fg("warning", theme.status.warning)} ${theme.fg(
1152
+ "dim",
1153
+ truncateToWidth(missingCompleteWarning, 80),
1154
+ )}`,
1155
+ );
1156
+ }
1157
+
1158
+ // Fallback to output preview if no custom rendering
1159
+ if (!hasCustomRendering) {
1160
+ lines.push(
1161
+ ...renderOutputSection(outputWithoutWarning, continuePrefix, expanded, theme, 3, 12, missingCompleteWarning),
1162
+ );
1163
+ }
1164
+
1165
+ if (deferredToolLines.length > 0) {
1166
+ lines.push(...deferredToolLines);
1167
+ }
1168
+
1169
+ if (result.patchPath && !aborted && result.exitCode === 0) {
1170
+ lines.push(`${continuePrefix}${theme.fg("dim", `Patch: ${result.patchPath}`)}`);
1171
+ } else if (result.branchName && !aborted && result.exitCode === 0) {
1172
+ lines.push(`${continuePrefix}${theme.fg("dim", `Branch: ${result.branchName}`)}`);
1173
+ }
1174
+
1175
+ // Error message
1176
+ if (result.error && (!success || mergeFailed) && (!aborted || result.error !== result.abortReason)) {
1177
+ lines.push(
1178
+ `${continuePrefix}${theme.fg(mergeFailed ? "warning" : "error", truncateToWidth(replaceTabs(result.error), 70))}`,
1179
+ );
1180
+ }
1181
+
1182
+ return lines;
1183
+ }
1184
+
1185
+ /**
1186
+ * Order live progress entries so finished agents render first — sorted by
1187
+ * runtime ascending, matching {@link orderResultsForDisplay} — while
1188
+ * unfinished (pending/running) ones stay pinned at the bottom in dispatch
1189
+ * order. Because a finished agent's runtime is fixed, finalization renders
1190
+ * the same order and rows never reshuffle.
1191
+ */
1192
+ function orderProgressForDisplay(progress: readonly AgentProgress[]): AgentProgress[] {
1193
+ const finished: AgentProgress[] = [];
1194
+ const unfinished: AgentProgress[] = [];
1195
+ for (const p of progress) {
1196
+ (p.status === "pending" || p.status === "running" ? unfinished : finished).push(p);
1197
+ }
1198
+ finished.sort((a, b) => a.durationMs - b.durationMs || a.index - b.index);
1199
+ return finished.concat(unfinished);
1200
+ }
1201
+
1202
+ /**
1203
+ * Order finalized results by runtime ascending (tie-break: dispatch index) so
1204
+ * the finalized list matches the live-progress order produced by
1205
+ * {@link orderProgressForDisplay}.
1206
+ */
1207
+ function orderResultsForDisplay(results: readonly SingleResult[]): SingleResult[] {
1208
+ return [...results].sort((a, b) => a.durationMs - b.durationMs || a.index - b.index);
1209
+ }
1210
+
1211
+ /**
1212
+ * Summary line for progress rows folded away by the collapsed cap: per-status
1213
+ * counts plus the expand hint, e.g. `… 21 more agents (18 pending · 3 done)`.
1214
+ */
1215
+ function formatHiddenProgressLine(hidden: readonly AgentProgress[], theme: Theme): string {
1216
+ const counts: Record<AgentProgress["status"], number> = {
1217
+ pending: 0,
1218
+ running: 0,
1219
+ completed: 0,
1220
+ failed: 0,
1221
+ aborted: 0,
1222
+ };
1223
+ for (const p of hidden) counts[p.status]++;
1224
+ const parts: string[] = [];
1225
+ if (counts.completed > 0) parts.push(theme.fg("dim", `${counts.completed} done`));
1226
+ if (counts.running > 0) parts.push(theme.fg("dim", `${counts.running} running`));
1227
+ if (counts.pending > 0) parts.push(theme.fg("dim", `${counts.pending} pending`));
1228
+ if (counts.failed > 0) parts.push(theme.fg("error", `${counts.failed} failed`));
1229
+ if (counts.aborted > 0) parts.push(theme.fg("error", `${counts.aborted} aborted`));
1230
+ const breakdown =
1231
+ parts.length > 0
1232
+ ? `${theme.fg("dim", " (")}${parts.join(theme.fg("dim", theme.sep.dot))}${theme.fg("dim", ")")}`
1233
+ : "";
1234
+ const hint = formatExpandHint(theme, false, true);
1235
+ return `${theme.fg("dim", formatMoreItems(hidden.length, "agent"))}${breakdown}${hint ? ` ${hint}` : ""}`;
1236
+ }
1237
+
1238
+ /**
1239
+ * Pick the agent rows that stay visible when a finalized batch is collapsed:
1240
+ * problem rows (aborted/failed/merge-failed) claim slots first so they are
1241
+ * never folded away, then fastest finishers fill the remainder. The pick is
1242
+ * filtered out of the display order, so visible rows keep the expanded layout.
1243
+ */
1244
+ function selectCollapsedResults(ordered: readonly SingleResult[]): readonly SingleResult[] {
1245
+ if (ordered.length <= COLLAPSED_AGENT_LIMIT) return ordered;
1246
+ const picked = new Set<SingleResult>();
1247
+ for (const result of ordered) {
1248
+ if (picked.size >= COLLAPSED_AGENT_LIMIT) break;
1249
+ if (result.aborted || result.exitCode !== 0 || result.error) picked.add(result);
1250
+ }
1251
+ for (const result of ordered) {
1252
+ if (picked.size >= COLLAPSED_AGENT_LIMIT) break;
1253
+ picked.add(result);
1254
+ }
1255
+ return ordered.filter(result => picked.has(result));
1256
+ }
1257
+
1258
+ /**
1259
+ * Render the tool result.
1260
+ */
1261
+ export function renderResult(
1262
+ result: { content: Array<{ type: string; text?: string }>; details?: TaskToolDetails; isError?: boolean },
1263
+ options: TaskRenderOptions,
1264
+ theme: Theme,
1265
+ args?: TaskParams,
1266
+ ): Component {
1267
+ const fallbackText = result.content.find(c => c.type === "text")?.text ?? "";
1268
+ const details = result.details;
1269
+ const agentLabel = args?.agent?.trim() || undefined;
1270
+ const assignmentSection = createAssignmentSectionRenderer(args, theme);
1271
+ const contextSection = createContextSectionRenderer(args, theme);
1272
+
1273
+ if (!details) {
1274
+ const text = result.content.find(c => c.type === "text")?.text || "";
1275
+ const errored = result.isError === true;
1276
+ const header = errored
1277
+ ? renderStatusLine({ icon: "error", title: "Task", description: agentLabel }, theme)
1278
+ : renderStatusLine(
1279
+ {
1280
+ iconOverride: theme.styledSymbol("status.done", "accent"),
1281
+ title: "Task",
1282
+ description: agentLabel,
1283
+ },
1284
+ theme,
1285
+ );
1286
+ return framedBlock(theme, width => ({
1287
+ header,
1288
+ sections: [
1289
+ ...(contextSection ? [contextSection(width)] : []),
1290
+ ...(assignmentSection ? [assignmentSection(width)] : []),
1291
+ ...(text ? [{ separator: true, lines: [theme.fg("dim", truncateToWidth(text, width))] }] : []),
1292
+ ],
1293
+ state: errored ? "error" : "success",
1294
+ borderColor: errored ? "error" : "borderMuted",
1295
+ width,
1296
+ }));
1297
+ }
1298
+
1299
+ const hasResults = Boolean(details.results && details.results.length > 0);
1300
+ const aborted = hasResults && details.results.some(r => r.aborted);
1301
+ const failed = hasResults && details.results.some(r => !r.aborted && r.exitCode !== 0);
1302
+ const mergeFailed = hasResults && details.results.some(r => !r.aborted && r.exitCode === 0 && Boolean(r.error));
1303
+ const isError = aborted || failed;
1304
+ const agentCount = hasResults ? details.results.length : (details.progress?.length ?? 0);
1305
+ const icon: ToolUIStatus = options.isPartial ? "running" : isError ? "error" : mergeFailed ? "warning" : "success";
1306
+ // Surface the dispatched agent type (e.g. `Reviewer`) alongside the count
1307
+ // so the header reads `Task 1 agent: Reviewer`.
1308
+ const countLabel = agentCount > 0 ? `${agentCount} ${agentCount === 1 ? "agent" : "agents"}` : undefined;
1309
+ const metaLabel = countLabel ? (agentLabel ? `${countLabel}: ${agentLabel}` : countLabel) : agentLabel;
1310
+ const header = renderStatusLine(
1311
+ {
1312
+ icon: icon === "success" || icon === "running" ? undefined : icon,
1313
+ // While agents are in flight the header shows the dispatch glyph, not a
1314
+ // spinner: async spawns return immediately, so "running" means
1315
+ // "delegated to peers", not "this call is blocking the turn".
1316
+ iconOverride:
1317
+ icon === "running"
1318
+ ? theme.styledSymbol("tool.task", "accent")
1319
+ : icon === "success"
1320
+ ? theme.styledSymbol("status.done", "accent")
1321
+ : undefined,
1322
+ title: "Task",
1323
+ meta: metaLabel ? [metaLabel] : undefined,
1324
+ },
1325
+ theme,
1326
+ );
1327
+
1328
+ return framedBlock(theme, width => {
1329
+ const { expanded, isPartial, spinnerFrame } = options;
1330
+ const frozen = options.renderContext?.frozen === true;
1331
+ const lines: string[] = [];
1332
+
1333
+ const shouldRenderProgress =
1334
+ Boolean(details.progress && details.progress.length > 0) && (isPartial || details.results.length === 0);
1335
+ if (shouldRenderProgress && details.progress) {
1336
+ const ordered = orderProgressForDisplay(details.progress);
1337
+ // Collapsed view keeps the live edge: finished rows sort to the top of
1338
+ // the display order, so folding from the top keeps running/pending
1339
+ // agents (and their current-tool lines) visible while one summary line
1340
+ // stands in for everything above it.
1341
+ const visible = expanded ? ordered : ordered.slice(Math.max(0, ordered.length - COLLAPSED_AGENT_LIMIT));
1342
+ if (visible.length < ordered.length) {
1343
+ lines.push(formatHiddenProgressLine(ordered.slice(0, ordered.length - visible.length), theme));
1344
+ }
1345
+ for (const progress of visible) {
1346
+ lines.push(...renderAgentProgress(progress, "", " ", expanded, theme, spinnerFrame, frozen));
1347
+ }
1348
+ } else if (details.results && details.results.length > 0) {
1349
+ const ordered = orderResultsForDisplay(details.results);
1350
+ const visible = expanded ? ordered : selectCollapsedResults(ordered);
1351
+ for (const res of visible) {
1352
+ lines.push(...renderAgentResult(res, "", " ", expanded, theme));
1353
+ }
1354
+ if (visible.length < ordered.length) {
1355
+ const hint = formatExpandHint(theme, false, true);
1356
+ lines.push(
1357
+ `${theme.fg("dim", formatMoreItems(ordered.length - visible.length, "agent"))}${hint ? ` ${hint}` : ""}`,
1358
+ );
1359
+ }
1360
+
1361
+ const abortedCount = details.results.filter(r => r.aborted).length;
1362
+ const mergeFailedCount = details.results.filter(r => !r.aborted && r.exitCode === 0 && r.error).length;
1363
+ const successCount = details.results.filter(r => !r.aborted && r.exitCode === 0 && !r.error).length;
1364
+ const failCount = details.results.length - successCount - mergeFailedCount - abortedCount;
1365
+ const summaryParts: string[] = [];
1366
+ if (abortedCount > 0) summaryParts.push(theme.fg("error", `${abortedCount} aborted`));
1367
+ if (successCount > 0) summaryParts.push(theme.fg("success", `${successCount} succeeded`));
1368
+ if (mergeFailedCount > 0) summaryParts.push(theme.fg("warning", `${mergeFailedCount} merge failed`));
1369
+ if (failCount > 0) summaryParts.push(theme.fg("error", `${failCount} failed`));
1370
+ const totalRequests = details.results.reduce((sum, r) => sum + (r.requests ?? 0), 0);
1371
+ if (totalRequests > 0) summaryParts.push(theme.fg("dim", `${formatNumber(totalRequests)} req`));
1372
+ summaryParts.push(theme.fg("dim", formatDuration(details.totalDurationMs)));
1373
+ // Wrap the run summary in the theme's bracket glyphs (dim chrome, colored
1374
+ // counts) to match the bash tool's `[Wall: … | Exit: …]` footer.
1375
+ lines.push(
1376
+ theme.fg("dim", theme.format.bracketLeft) +
1377
+ summaryParts.join(theme.fg("dim", theme.sep.dot)) +
1378
+ theme.fg("dim", theme.format.bracketRight),
1379
+ );
1380
+ }
1381
+
1382
+ const state = isPartial ? "running" : isError ? "error" : mergeFailed ? "warning" : "success";
1383
+ const borderColor = isError ? "error" : "borderMuted";
1384
+
1385
+ if (lines.length === 0) {
1386
+ const text = fallbackText.trim() ? fallbackText : "No results";
1387
+ return {
1388
+ header,
1389
+ sections: [
1390
+ ...(contextSection ? [contextSection(width)] : []),
1391
+ ...(assignmentSection ? [assignmentSection(width)] : []),
1392
+ { separator: true, lines: [theme.fg("dim", truncateToWidth(text, width))] },
1393
+ ],
1394
+ state,
1395
+ borderColor,
1396
+ width,
1397
+ };
1398
+ }
1399
+
1400
+ if (fallbackText.trim()) {
1401
+ const summaryLines = fallbackText.split("\n");
1402
+ const markerIndex = summaryLines.findIndex(
1403
+ line =>
1404
+ line.includes("<system-notification>") ||
1405
+ line.startsWith("Applied patches:") ||
1406
+ line.startsWith("No changes to apply."),
1407
+ );
1408
+ if (markerIndex >= 0) {
1409
+ const extra = summaryLines.slice(markerIndex);
1410
+ for (const line of extra) {
1411
+ if (!line.trim()) continue;
1412
+ lines.push(theme.fg("dim", line));
1413
+ }
1414
+ }
1415
+ }
1416
+
1417
+ while (lines.length > 0 && lines[0].trim() === "") lines.shift();
1418
+ return {
1419
+ header,
1420
+ sections: [
1421
+ ...(contextSection ? [contextSection(width)] : []),
1422
+ ...(assignmentSection ? [assignmentSection(width)] : []),
1423
+ ...(lines.length > 0 ? [{ separator: true, lines }] : []),
1424
+ ],
1425
+ state,
1426
+ borderColor,
1427
+ width,
1428
+ };
1429
+ });
1430
+ }
1431
+
1432
+ function isTaskToolDetails(value: unknown): value is TaskToolDetails {
1433
+ return (
1434
+ Boolean(value) &&
1435
+ typeof value === "object" &&
1436
+ "results" in (value as TaskToolDetails) &&
1437
+ Array.isArray((value as TaskToolDetails).results)
1438
+ );
1439
+ }
1440
+
1441
+ // Nested subagent snapshots sit one or more levels below the frame border, so
1442
+ // they keep tree guides to convey depth (the parent prepends its own continue
1443
+ // prefix). Only the top-level agent list drops guides (the frame is its box).
1444
+ function nestedMarkers(isLast: boolean, theme: Theme): { prefix: string; continuePrefix: string } {
1445
+ return {
1446
+ prefix: isLast ? theme.fg("dim", theme.tree.last) : theme.fg("dim", theme.tree.branch),
1447
+ continuePrefix: isLast ? " " : `${theme.fg("dim", theme.tree.vertical)} `,
1448
+ };
1449
+ }
1450
+
1451
+ function renderNestedTaskResults(
1452
+ detailsList: TaskToolDetails[],
1453
+ expanded: boolean,
1454
+ theme: Theme,
1455
+ seen: WeakSet<object> = new WeakSet<object>(),
1456
+ depth = 0,
1457
+ ): string[] {
1458
+ const lines: string[] = [];
1459
+ for (const details of detailsList) {
1460
+ if (seen.has(details)) {
1461
+ lines.push(renderNestedCycleLine(theme));
1462
+ continue;
1463
+ }
1464
+ if (depth >= MAX_NESTED_TASK_RENDER_DEPTH) {
1465
+ lines.push(theme.fg("dim", "… nested task depth limit reached"));
1466
+ continue;
1467
+ }
1468
+ seen.add(details);
1469
+ if (!details.results || details.results.length === 0) {
1470
+ seen.delete(details);
1471
+ continue;
1472
+ }
1473
+ const ordered = orderResultsForDisplay(details.results);
1474
+ ordered.forEach((result, index) => {
1475
+ const { prefix, continuePrefix } = nestedMarkers(index === ordered.length - 1, theme);
1476
+ lines.push(...renderAgentResult(result, prefix, continuePrefix, expanded, theme, seen, depth + 1));
1477
+ });
1478
+ seen.delete(details);
1479
+ }
1480
+ return lines;
1481
+ }
1482
+
1483
+ /**
1484
+ * Render a list of `TaskToolDetails` snapshots — completed (`results[]`) or
1485
+ * in-flight (`progress[]`) — as an interleaved tree. Used by the live progress
1486
+ * view to surface nested subagent activity while this agent is still running.
1487
+ */
1488
+ function renderNestedTaskTree(
1489
+ detailsList: TaskToolDetails[],
1490
+ expanded: boolean,
1491
+ theme: Theme,
1492
+ spinnerFrame?: number,
1493
+ frozen = false,
1494
+ seen: WeakSet<object> = new WeakSet<object>(),
1495
+ depth = 0,
1496
+ ): string[] {
1497
+ const lines: string[] = [];
1498
+ for (const details of detailsList) {
1499
+ if (seen.has(details)) {
1500
+ lines.push(renderNestedCycleLine(theme));
1501
+ continue;
1502
+ }
1503
+ if (depth >= MAX_NESTED_TASK_RENDER_DEPTH) {
1504
+ lines.push(theme.fg("dim", "… nested task depth limit reached"));
1505
+ continue;
1506
+ }
1507
+ seen.add(details);
1508
+ const hasResults = Boolean(details.results && details.results.length > 0);
1509
+ if (hasResults) {
1510
+ const ordered = orderResultsForDisplay(details.results);
1511
+ ordered.forEach((result, index) => {
1512
+ const { prefix, continuePrefix } = nestedMarkers(index === ordered.length - 1, theme);
1513
+ lines.push(...renderAgentResult(result, prefix, continuePrefix, expanded, theme, seen, depth + 1));
1514
+ });
1515
+ seen.delete(details);
1516
+ continue;
1517
+ }
1518
+ const inflight = details.progress;
1519
+ if (inflight && inflight.length > 0) {
1520
+ const ordered = orderProgressForDisplay(inflight);
1521
+ ordered.forEach((prog, index) => {
1522
+ const { prefix, continuePrefix } = nestedMarkers(index === ordered.length - 1, theme);
1523
+ lines.push(
1524
+ ...renderAgentProgress(
1525
+ prog,
1526
+ prefix,
1527
+ continuePrefix,
1528
+ expanded,
1529
+ theme,
1530
+ spinnerFrame,
1531
+ frozen,
1532
+ seen,
1533
+ depth + 1,
1534
+ ),
1535
+ );
1536
+ });
1537
+ }
1538
+ seen.delete(details);
1539
+ }
1540
+ return lines;
1541
+ }
1542
+
1543
+ subprocessToolRegistry.register<TaskToolDetails>("task", {
1544
+ extractData: event => {
1545
+ const details = event.result?.details;
1546
+ return isTaskToolDetails(details) ? details : undefined;
1547
+ },
1548
+ renderFinal: (allData, theme, expanded) => {
1549
+ const lines = renderNestedTaskResults(allData, expanded, theme);
1550
+ return new Text(lines.join("\n"), 0, 0);
1551
+ },
1552
+ });
1553
+
1554
+ export const taskToolRenderer = {
1555
+ renderCall,
1556
+ renderResult,
1557
+ mergeCallAndResult: true,
1558
+ };