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.
- package/CHANGELOG.md +12164 -0
- package/README.md +35 -0
- package/dist/cli.js +18266 -0
- package/examples/README.md +21 -0
- package/examples/custom-tools/README.md +104 -0
- package/examples/custom-tools/hello/index.ts +20 -0
- package/examples/extensions/README.md +142 -0
- package/examples/extensions/api-demo.ts +79 -0
- package/examples/extensions/chalk-logger.ts +25 -0
- package/examples/extensions/hello.ts +31 -0
- package/examples/extensions/pirate.ts +43 -0
- package/examples/extensions/plan-mode.ts +549 -0
- package/examples/extensions/reload-runtime.ts +38 -0
- package/examples/extensions/thinking-note.ts +13 -0
- package/examples/extensions/tools.ts +145 -0
- package/examples/extensions/with-deps/index.ts +36 -0
- package/examples/extensions/with-deps/package-lock.json +31 -0
- package/examples/extensions/with-deps/package.json +17 -0
- package/examples/hooks/README.md +56 -0
- package/examples/hooks/auto-commit-on-exit.ts +48 -0
- package/examples/hooks/confirm-destructive.ts +58 -0
- package/examples/hooks/custom-compaction.ts +115 -0
- package/examples/hooks/dirty-repo-guard.ts +51 -0
- package/examples/hooks/file-trigger.ts +40 -0
- package/examples/hooks/git-checkpoint.ts +52 -0
- package/examples/hooks/handoff.ts +149 -0
- package/examples/hooks/permission-gate.ts +33 -0
- package/examples/hooks/protected-paths.ts +29 -0
- package/examples/hooks/qna.ts +118 -0
- package/examples/hooks/status-line.ts +39 -0
- package/examples/sdk/01-minimal.ts +21 -0
- package/examples/sdk/02-custom-model.ts +49 -0
- package/examples/sdk/03-custom-prompt.ts +46 -0
- package/examples/sdk/04-skills.ts +43 -0
- package/examples/sdk/06-extensions.ts +82 -0
- package/examples/sdk/06-hooks.ts +61 -0
- package/examples/sdk/07-context-files.ts +35 -0
- package/examples/sdk/08-prompt-templates.ts +41 -0
- package/examples/sdk/08-slash-commands.ts +46 -0
- package/examples/sdk/09-api-keys-and-oauth.ts +54 -0
- package/examples/sdk/11-sessions.ts +47 -0
- package/examples/sdk/12-redis-sessions.ts +54 -0
- package/examples/sdk/13-sql-sessions.ts +61 -0
- package/examples/sdk/README.md +172 -0
- package/package.json +573 -0
- package/scripts/bench-guard.ts +71 -0
- package/scripts/build-binary.ts +108 -0
- package/scripts/bundle-dist.ts +110 -0
- package/scripts/embed-mupdf-wasm.ts +67 -0
- package/scripts/format-prompts.ts +68 -0
- package/scripts/generate-docs-index.ts +56 -0
- package/scripts/generate-share-viewer.ts +34 -0
- package/scripts/measure-prompt-tokens.ts +63 -0
- package/scripts/omp +42 -0
- package/scripts/omp.ts +19 -0
- package/src/advisor/__tests__/advisor.test.ts +915 -0
- package/src/advisor/advise-tool.ts +165 -0
- package/src/advisor/index.ts +4 -0
- package/src/advisor/runtime.ts +270 -0
- package/src/advisor/transcript-recorder.ts +136 -0
- package/src/advisor/watchdog.ts +83 -0
- package/src/async/index.ts +1 -0
- package/src/async/job-manager.ts +674 -0
- package/src/auto-thinking/classifier.ts +190 -0
- package/src/autolearn/controller.ts +139 -0
- package/src/autolearn/managed-skills.ts +255 -0
- package/src/autoresearch/command-resume.md +14 -0
- package/src/autoresearch/dashboard.ts +436 -0
- package/src/autoresearch/git.ts +319 -0
- package/src/autoresearch/helpers.ts +218 -0
- package/src/autoresearch/index.ts +536 -0
- package/src/autoresearch/prompt-setup.md +43 -0
- package/src/autoresearch/prompt.md +103 -0
- package/src/autoresearch/resume-message.md +10 -0
- package/src/autoresearch/state.ts +273 -0
- package/src/autoresearch/storage.ts +700 -0
- package/src/autoresearch/tools/init-experiment.ts +269 -0
- package/src/autoresearch/tools/log-experiment.ts +521 -0
- package/src/autoresearch/tools/run-experiment.ts +407 -0
- package/src/autoresearch/tools/update-notes.ts +109 -0
- package/src/autoresearch/types.ts +168 -0
- package/src/capability/context-file.ts +44 -0
- package/src/capability/extension-module.ts +34 -0
- package/src/capability/extension.ts +47 -0
- package/src/capability/fs.ts +117 -0
- package/src/capability/hook.ts +40 -0
- package/src/capability/index.ts +436 -0
- package/src/capability/instruction.ts +37 -0
- package/src/capability/mcp.ts +76 -0
- package/src/capability/prompt.ts +35 -0
- package/src/capability/rule-buckets.ts +66 -0
- package/src/capability/rule.ts +261 -0
- package/src/capability/settings.ts +34 -0
- package/src/capability/skill.ts +63 -0
- package/src/capability/slash-command.ts +40 -0
- package/src/capability/ssh.ts +41 -0
- package/src/capability/system-prompt.ts +34 -0
- package/src/capability/tool.ts +38 -0
- package/src/capability/types.ts +168 -0
- package/src/cli/agents-cli.ts +138 -0
- package/src/cli/args.ts +361 -0
- package/src/cli/auth-broker-cli.ts +893 -0
- package/src/cli/auth-gateway-cli.ts +608 -0
- package/src/cli/bench-cli.ts +552 -0
- package/src/cli/classify-install-target.ts +76 -0
- package/src/cli/claude-trace-cli.ts +795 -0
- package/src/cli/commands/init-xdg.ts +27 -0
- package/src/cli/completion-gen.ts +550 -0
- package/src/cli/config-cli.ts +418 -0
- package/src/cli/dry-balance-cli.ts +858 -0
- package/src/cli/extension-flags.ts +48 -0
- package/src/cli/file-processor.ts +133 -0
- package/src/cli/flag-tables.ts +280 -0
- package/src/cli/gallery-cli.ts +231 -0
- package/src/cli/gallery-fixtures/agentic.ts +407 -0
- package/src/cli/gallery-fixtures/codeintel.ts +187 -0
- package/src/cli/gallery-fixtures/edit.ts +194 -0
- package/src/cli/gallery-fixtures/fs.ts +220 -0
- package/src/cli/gallery-fixtures/index.ts +40 -0
- package/src/cli/gallery-fixtures/interaction.ts +49 -0
- package/src/cli/gallery-fixtures/memory.ts +81 -0
- package/src/cli/gallery-fixtures/misc.ts +250 -0
- package/src/cli/gallery-fixtures/search.ts +213 -0
- package/src/cli/gallery-fixtures/shell.ts +167 -0
- package/src/cli/gallery-fixtures/types.ts +57 -0
- package/src/cli/gallery-fixtures/web.ts +158 -0
- package/src/cli/gallery-screenshot.ts +279 -0
- package/src/cli/grep-cli.ts +160 -0
- package/src/cli/grievances-cli.ts +256 -0
- package/src/cli/initial-message.ts +58 -0
- package/src/cli/models-cli.ts +427 -0
- package/src/cli/plugin-cli.ts +996 -0
- package/src/cli/profile-alias.ts +338 -0
- package/src/cli/profile-bootstrap.ts +243 -0
- package/src/cli/read-cli.ts +57 -0
- package/src/cli/session-picker.ts +80 -0
- package/src/cli/setup-cli.ts +332 -0
- package/src/cli/setup-model-picker.ts +43 -0
- package/src/cli/shell-cli.ts +176 -0
- package/src/cli/ssh-cli.ts +179 -0
- package/src/cli/startup-cwd.ts +58 -0
- package/src/cli/stats-cli.ts +229 -0
- package/src/cli/tiny-models-cli.ts +127 -0
- package/src/cli/ttsr-cli.ts +995 -0
- package/src/cli/update-cli.ts +671 -0
- package/src/cli/usage-cli.ts +774 -0
- package/src/cli/web-search-cli.ts +132 -0
- package/src/cli/worktree-cli.ts +291 -0
- package/src/cli-commands.ts +85 -0
- package/src/cli.ts +326 -0
- package/src/collab/crypto.ts +63 -0
- package/src/collab/guest.ts +450 -0
- package/src/collab/host.ts +577 -0
- package/src/collab/protocol.ts +274 -0
- package/src/collab/relay-client.ts +216 -0
- package/src/commands/acp.ts +24 -0
- package/src/commands/agents.ts +57 -0
- package/src/commands/auth-broker.ts +99 -0
- package/src/commands/auth-gateway.ts +69 -0
- package/src/commands/bench.ts +42 -0
- package/src/commands/commit.ts +46 -0
- package/src/commands/complete.ts +66 -0
- package/src/commands/completions.ts +60 -0
- package/src/commands/config.ts +51 -0
- package/src/commands/dry-balance.ts +43 -0
- package/src/commands/gallery.ts +52 -0
- package/src/commands/grep.ts +48 -0
- package/src/commands/grievances.ts +51 -0
- package/src/commands/install.ts +107 -0
- package/src/commands/join.ts +39 -0
- package/src/commands/launch.ts +182 -0
- package/src/commands/models.ts +61 -0
- package/src/commands/plugin.ts +78 -0
- package/src/commands/read.ts +38 -0
- package/src/commands/say.ts +102 -0
- package/src/commands/setup.ts +67 -0
- package/src/commands/shell.ts +29 -0
- package/src/commands/ssh.ts +60 -0
- package/src/commands/stats.ts +29 -0
- package/src/commands/tiny-models.ts +36 -0
- package/src/commands/token.ts +108 -0
- package/src/commands/ttsr.ts +125 -0
- package/src/commands/update.ts +21 -0
- package/src/commands/usage.ts +43 -0
- package/src/commands/web-search.ts +42 -0
- package/src/commands/worktree.ts +56 -0
- package/src/commit/agentic/agent.ts +318 -0
- package/src/commit/agentic/fallback.ts +96 -0
- package/src/commit/agentic/index.ts +355 -0
- package/src/commit/agentic/prompts/analyze-file.md +22 -0
- package/src/commit/agentic/prompts/session-user.md +25 -0
- package/src/commit/agentic/prompts/split-confirm.md +1 -0
- package/src/commit/agentic/prompts/system.md +38 -0
- package/src/commit/agentic/state.ts +60 -0
- package/src/commit/agentic/tools/analyze-file.ts +149 -0
- package/src/commit/agentic/tools/git-file-diff.ts +191 -0
- package/src/commit/agentic/tools/git-hunk.ts +52 -0
- package/src/commit/agentic/tools/git-overview.ts +81 -0
- package/src/commit/agentic/tools/index.ts +54 -0
- package/src/commit/agentic/tools/propose-changelog.ts +147 -0
- package/src/commit/agentic/tools/propose-commit.ts +109 -0
- package/src/commit/agentic/tools/recent-commits.ts +81 -0
- package/src/commit/agentic/tools/schemas.ts +11 -0
- package/src/commit/agentic/tools/split-commit.ts +241 -0
- package/src/commit/agentic/topo-sort.ts +44 -0
- package/src/commit/agentic/trivial.ts +51 -0
- package/src/commit/agentic/validation.ts +183 -0
- package/src/commit/analysis/conventional.ts +64 -0
- package/src/commit/analysis/index.ts +4 -0
- package/src/commit/analysis/scope.ts +242 -0
- package/src/commit/analysis/summary.ts +107 -0
- package/src/commit/analysis/validation.ts +66 -0
- package/src/commit/changelog/detect.ts +40 -0
- package/src/commit/changelog/generate.ts +101 -0
- package/src/commit/changelog/index.ts +234 -0
- package/src/commit/changelog/parse.ts +44 -0
- package/src/commit/cli.ts +85 -0
- package/src/commit/git/diff.ts +148 -0
- package/src/commit/index.ts +5 -0
- package/src/commit/map-reduce/index.ts +69 -0
- package/src/commit/map-reduce/map-phase.ts +193 -0
- package/src/commit/map-reduce/reduce-phase.ts +49 -0
- package/src/commit/map-reduce/utils.ts +9 -0
- package/src/commit/message.ts +11 -0
- package/src/commit/model-selection.ts +89 -0
- package/src/commit/pipeline.ts +243 -0
- package/src/commit/prompts/analysis-system.md +148 -0
- package/src/commit/prompts/analysis-user.md +38 -0
- package/src/commit/prompts/changelog-system.md +50 -0
- package/src/commit/prompts/changelog-user.md +18 -0
- package/src/commit/prompts/file-observer-system.md +24 -0
- package/src/commit/prompts/file-observer-user.md +8 -0
- package/src/commit/prompts/reduce-system.md +50 -0
- package/src/commit/prompts/reduce-user.md +17 -0
- package/src/commit/prompts/summary-retry.md +3 -0
- package/src/commit/prompts/summary-system.md +38 -0
- package/src/commit/prompts/summary-user.md +13 -0
- package/src/commit/prompts/types-description.md +2 -0
- package/src/commit/shared-llm.ts +70 -0
- package/src/commit/types.ts +118 -0
- package/src/commit/utils/exclusions.ts +42 -0
- package/src/commit/utils.ts +58 -0
- package/src/config/api-key-resolver.ts +67 -0
- package/src/config/append-only-context-mode.ts +76 -0
- package/src/config/config-file.ts +315 -0
- package/src/config/file-lock.ts +164 -0
- package/src/config/keybindings.ts +634 -0
- package/src/config/mcp-schema.json +238 -0
- package/src/config/model-discovery.ts +589 -0
- package/src/config/model-registry.ts +2260 -0
- package/src/config/model-resolver.ts +1819 -0
- package/src/config/model-roles.ts +99 -0
- package/src/config/models-config-schema.ts +266 -0
- package/src/config/models-config.ts +131 -0
- package/src/config/prompt-templates.ts +185 -0
- package/src/config/resolve-config-value.ts +94 -0
- package/src/config/settings-schema.ts +4740 -0
- package/src/config/settings.ts +1243 -0
- package/src/config.ts +242 -0
- package/src/cursor.ts +340 -0
- package/src/dap/client.ts +760 -0
- package/src/dap/config.ts +189 -0
- package/src/dap/defaults.json +212 -0
- package/src/dap/index.ts +4 -0
- package/src/dap/session.ts +1441 -0
- package/src/dap/types.ts +610 -0
- package/src/debug/index.ts +559 -0
- package/src/debug/log-formatting.ts +58 -0
- package/src/debug/log-viewer.ts +908 -0
- package/src/debug/profiler.ts +162 -0
- package/src/debug/protocol-probe.ts +267 -0
- package/src/debug/raw-sse-buffer.ts +294 -0
- package/src/debug/raw-sse.ts +292 -0
- package/src/debug/remote-debugger.ts +151 -0
- package/src/debug/report-bundle.ts +375 -0
- package/src/debug/system-info.ts +111 -0
- package/src/debug/terminal-info.ts +124 -0
- package/src/discovery/agents-md.ts +67 -0
- package/src/discovery/agents.ts +230 -0
- package/src/discovery/at-imports.ts +273 -0
- package/src/discovery/builtin-defaults.ts +39 -0
- package/src/discovery/builtin-rules/index.ts +63 -0
- package/src/discovery/builtin-rules/low-end/no-hallucinated-apis.md +14 -0
- package/src/discovery/builtin-rules/low-end/no-hallucinated-paths.md +14 -0
- package/src/discovery/builtin-rules/low-end/no-premature-completion.md +14 -0
- package/src/discovery/builtin-rules/rs-box-leak.md +48 -0
- package/src/discovery/builtin-rules/rs-future-prelude.md +23 -0
- package/src/discovery/builtin-rules/rs-lazylock.md +51 -0
- package/src/discovery/builtin-rules/rs-match-ergonomics.md +67 -0
- package/src/discovery/builtin-rules/rs-parking-lot.md +44 -0
- package/src/discovery/builtin-rules/rs-result-type.md +19 -0
- package/src/discovery/builtin-rules/ts-bare-catch.md +38 -0
- package/src/discovery/builtin-rules/ts-import-type.md +42 -0
- package/src/discovery/builtin-rules/ts-no-any.md +65 -0
- package/src/discovery/builtin-rules/ts-no-deprecated-leftovers.md +44 -0
- package/src/discovery/builtin-rules/ts-no-dynamic-import.md +39 -0
- package/src/discovery/builtin-rules/ts-no-inline-cast-access.md +55 -0
- package/src/discovery/builtin-rules/ts-no-return-type.md +44 -0
- package/src/discovery/builtin-rules/ts-no-test-timers.md +55 -0
- package/src/discovery/builtin-rules/ts-no-tiny-functions.md +51 -0
- package/src/discovery/builtin-rules/ts-promise-with-resolvers.md +65 -0
- package/src/discovery/builtin-rules/ts-redundant-clear-guard.md +75 -0
- package/src/discovery/builtin-rules/ts-set-map.md +28 -0
- package/src/discovery/builtin.ts +934 -0
- package/src/discovery/claude-plugins.ts +386 -0
- package/src/discovery/claude.ts +584 -0
- package/src/discovery/cline.ts +83 -0
- package/src/discovery/codex.ts +522 -0
- package/src/discovery/cursor.ts +220 -0
- package/src/discovery/gemini.ts +383 -0
- package/src/discovery/github.ts +337 -0
- package/src/discovery/helpers.ts +1092 -0
- package/src/discovery/index.ts +81 -0
- package/src/discovery/mcp-json.ts +172 -0
- package/src/discovery/omp-extension-roots.ts +190 -0
- package/src/discovery/omp-plugins.ts +383 -0
- package/src/discovery/opencode.ts +398 -0
- package/src/discovery/plugin-dir-roots.ts +28 -0
- package/src/discovery/ssh.ts +153 -0
- package/src/discovery/substitute-plugin-root.ts +29 -0
- package/src/discovery/vscode.ts +105 -0
- package/src/discovery/windsurf.ts +147 -0
- package/src/edit/apply-patch/index.ts +87 -0
- package/src/edit/apply-patch/parser.ts +174 -0
- package/src/edit/diff.ts +999 -0
- package/src/edit/file-snapshot-store.ts +143 -0
- package/src/edit/hashline/block-resolver.ts +33 -0
- package/src/edit/hashline/diff.ts +290 -0
- package/src/edit/hashline/execute.ts +237 -0
- package/src/edit/hashline/filesystem.ts +130 -0
- package/src/edit/hashline/index.ts +5 -0
- package/src/edit/hashline/noop-loop-guard.ts +99 -0
- package/src/edit/hashline/params.ts +19 -0
- package/src/edit/index.ts +620 -0
- package/src/edit/modes/apply-patch.lark +19 -0
- package/src/edit/modes/apply-patch.ts +53 -0
- package/src/edit/modes/patch.ts +1888 -0
- package/src/edit/modes/replace.ts +1133 -0
- package/src/edit/normalize.ts +345 -0
- package/src/edit/notebook.ts +242 -0
- package/src/edit/read-file.ts +25 -0
- package/src/edit/renderer.ts +823 -0
- package/src/edit/streaming.ts +517 -0
- package/src/eval/__tests__/agent-bridge.test.ts +769 -0
- package/src/eval/__tests__/bridge-timeout.test.ts +64 -0
- package/src/eval/__tests__/budget-bridge.test.ts +69 -0
- package/src/eval/__tests__/completion-bridge.test.ts +412 -0
- package/src/eval/__tests__/helpers-local-roots.test.ts +58 -0
- package/src/eval/__tests__/idle-timeout.test.ts +80 -0
- package/src/eval/__tests__/js-context-manager.test.ts +291 -0
- package/src/eval/__tests__/kernel-spawn.test.ts +103 -0
- package/src/eval/__tests__/prelude-agent.test.ts +73 -0
- package/src/eval/agent-bridge.ts +319 -0
- package/src/eval/backend.ts +71 -0
- package/src/eval/bridge-timeout.ts +44 -0
- package/src/eval/budget-bridge.ts +48 -0
- package/src/eval/completion-bridge.ts +211 -0
- package/src/eval/concurrency-bridge.ts +34 -0
- package/src/eval/idle-timeout.ts +91 -0
- package/src/eval/index.ts +4 -0
- package/src/eval/js/context-manager.ts +621 -0
- package/src/eval/js/executor.ts +173 -0
- package/src/eval/js/index.ts +51 -0
- package/src/eval/js/shared/helpers.ts +283 -0
- package/src/eval/js/shared/indirect-eval.ts +30 -0
- package/src/eval/js/shared/local-module-loader.ts +342 -0
- package/src/eval/js/shared/prelude.ts +2 -0
- package/src/eval/js/shared/prelude.txt +307 -0
- package/src/eval/js/shared/rewrite-imports.ts +532 -0
- package/src/eval/js/shared/runtime.ts +580 -0
- package/src/eval/js/shared/types.ts +18 -0
- package/src/eval/js/tool-bridge.ts +163 -0
- package/src/eval/js/worker-core.ts +151 -0
- package/src/eval/js/worker-entry.ts +37 -0
- package/src/eval/js/worker-protocol.ts +47 -0
- package/src/eval/py/__tests__/prelude.test.ts +19 -0
- package/src/eval/py/display.ts +71 -0
- package/src/eval/py/executor.ts +742 -0
- package/src/eval/py/index.ts +68 -0
- package/src/eval/py/kernel.ts +748 -0
- package/src/eval/py/prelude.py +683 -0
- package/src/eval/py/prelude.ts +3 -0
- package/src/eval/py/runner.py +1177 -0
- package/src/eval/py/runtime.ts +276 -0
- package/src/eval/py/spawn-options.ts +126 -0
- package/src/eval/py/tool-bridge.ts +182 -0
- package/src/eval/session-id.ts +8 -0
- package/src/eval/types.ts +48 -0
- package/src/exa/index.ts +2 -0
- package/src/exa/mcp-client.ts +370 -0
- package/src/exa/types.ts +69 -0
- package/src/exec/bash-executor.ts +434 -0
- package/src/exec/exec.ts +53 -0
- package/src/exec/non-interactive-env.ts +119 -0
- package/src/export/custom-share.ts +65 -0
- package/src/export/html/index.ts +266 -0
- package/src/export/html/share-loader.js +102 -0
- package/src/export/html/template.css +1337 -0
- package/src/export/html/template.html +49 -0
- package/src/export/html/template.js +1626 -0
- package/src/export/html/tool-views.generated.js +37 -0
- package/src/export/html/vendor/highlight.min.js +1213 -0
- package/src/export/html/vendor/marked.min.js +6 -0
- package/src/export/share.ts +268 -0
- package/src/export/ttsr.ts +583 -0
- package/src/extensibility/custom-commands/bundled/ci-green/index.ts +54 -0
- package/src/extensibility/custom-commands/bundled/review/index.ts +698 -0
- package/src/extensibility/custom-commands/index.ts +2 -0
- package/src/extensibility/custom-commands/loader.ts +242 -0
- package/src/extensibility/custom-commands/types.ts +119 -0
- package/src/extensibility/custom-tools/index.ts +7 -0
- package/src/extensibility/custom-tools/loader.ts +268 -0
- package/src/extensibility/custom-tools/types.ts +277 -0
- package/src/extensibility/custom-tools/wrapper.ts +47 -0
- package/src/extensibility/extensions/compact-handler.ts +40 -0
- package/src/extensibility/extensions/get-commands-handler.ts +78 -0
- package/src/extensibility/extensions/index.ts +16 -0
- package/src/extensibility/extensions/loader.ts +587 -0
- package/src/extensibility/extensions/model-api.ts +41 -0
- package/src/extensibility/extensions/runner.ts +989 -0
- package/src/extensibility/extensions/types.ts +1394 -0
- package/src/extensibility/extensions/wrapper.ts +259 -0
- package/src/extensibility/hooks/index.ts +6 -0
- package/src/extensibility/hooks/loader.ts +262 -0
- package/src/extensibility/hooks/runner.ts +425 -0
- package/src/extensibility/hooks/tool-wrapper.ts +107 -0
- package/src/extensibility/hooks/types.ts +613 -0
- package/src/extensibility/legacy-pi-ai-shim.ts +61 -0
- package/src/extensibility/legacy-pi-coding-agent-shim.ts +128 -0
- package/src/extensibility/plugins/doctor.ts +65 -0
- package/src/extensibility/plugins/git-url.ts +367 -0
- package/src/extensibility/plugins/index.ts +9 -0
- package/src/extensibility/plugins/installer.ts +192 -0
- package/src/extensibility/plugins/legacy-pi-compat.ts +712 -0
- package/src/extensibility/plugins/loader.ts +458 -0
- package/src/extensibility/plugins/manager.ts +1026 -0
- package/src/extensibility/plugins/marketplace/cache.ts +136 -0
- package/src/extensibility/plugins/marketplace/fetcher.ts +315 -0
- package/src/extensibility/plugins/marketplace/index.ts +6 -0
- package/src/extensibility/plugins/marketplace/manager.ts +770 -0
- package/src/extensibility/plugins/marketplace/registry.ts +196 -0
- package/src/extensibility/plugins/marketplace/source-resolver.ts +147 -0
- package/src/extensibility/plugins/marketplace/types.ts +191 -0
- package/src/extensibility/plugins/marketplace-auto-update.ts +49 -0
- package/src/extensibility/plugins/parser.ts +105 -0
- package/src/extensibility/plugins/runtime-config.ts +9 -0
- package/src/extensibility/plugins/types.ts +194 -0
- package/src/extensibility/shared-events.ts +367 -0
- package/src/extensibility/skills.ts +408 -0
- package/src/extensibility/slash-commands.ts +131 -0
- package/src/extensibility/tool-proxy.ts +28 -0
- package/src/extensibility/typebox.ts +945 -0
- package/src/extensibility/utils.ts +44 -0
- package/src/goals/guided-setup.ts +142 -0
- package/src/goals/index.ts +3 -0
- package/src/goals/runtime.ts +521 -0
- package/src/goals/state.ts +37 -0
- package/src/goals/tools/goal-tool.ts +251 -0
- package/src/hindsight/backend.ts +354 -0
- package/src/hindsight/bank.ts +156 -0
- package/src/hindsight/client.ts +623 -0
- package/src/hindsight/config.ts +175 -0
- package/src/hindsight/content.ts +210 -0
- package/src/hindsight/index.ts +8 -0
- package/src/hindsight/mental-models.ts +429 -0
- package/src/hindsight/seeds.json +32 -0
- package/src/hindsight/state.ts +492 -0
- package/src/hindsight/transcript.ts +71 -0
- package/src/index.ts +66 -0
- package/src/internal-urls/agent-protocol.ts +146 -0
- package/src/internal-urls/artifact-protocol.ts +107 -0
- package/src/internal-urls/docs-index.generated.txt +2 -0
- package/src/internal-urls/docs-index.ts +102 -0
- package/src/internal-urls/history-protocol.ts +118 -0
- package/src/internal-urls/index.ts +25 -0
- package/src/internal-urls/issue-pr-protocol.ts +594 -0
- package/src/internal-urls/json-query.ts +126 -0
- package/src/internal-urls/local-protocol.ts +309 -0
- package/src/internal-urls/mcp-protocol.ts +151 -0
- package/src/internal-urls/memory-protocol.ts +169 -0
- package/src/internal-urls/omp-protocol.ts +94 -0
- package/src/internal-urls/parse.ts +72 -0
- package/src/internal-urls/registry-helpers.ts +25 -0
- package/src/internal-urls/router.ts +105 -0
- package/src/internal-urls/rule-protocol.ts +45 -0
- package/src/internal-urls/skill-protocol.ts +96 -0
- package/src/internal-urls/types.ts +152 -0
- package/src/internal-urls/vault-protocol.ts +936 -0
- package/src/irc/bus.ts +311 -0
- package/src/lib/xai-http.ts +124 -0
- package/src/lsp/client.ts +1217 -0
- package/src/lsp/clients/biome-client.ts +264 -0
- package/src/lsp/clients/index.ts +50 -0
- package/src/lsp/clients/lsp-linter-client.ts +85 -0
- package/src/lsp/clients/swiftlint-client.ts +120 -0
- package/src/lsp/config.ts +502 -0
- package/src/lsp/defaults.json +499 -0
- package/src/lsp/diagnostics-ledger.ts +51 -0
- package/src/lsp/edits.ts +267 -0
- package/src/lsp/format-options.ts +119 -0
- package/src/lsp/index.ts +2480 -0
- package/src/lsp/lspmux.ts +233 -0
- package/src/lsp/render.ts +668 -0
- package/src/lsp/startup-events.ts +13 -0
- package/src/lsp/types.ts +444 -0
- package/src/lsp/utils.ts +718 -0
- package/src/main.ts +1421 -0
- package/src/markit/NOTICE +32 -0
- package/src/markit/converters/docx.ts +56 -0
- package/src/markit/converters/epub.ts +136 -0
- package/src/markit/converters/mammoth.d.ts +24 -0
- package/src/markit/converters/pdf/columns.ts +103 -0
- package/src/markit/converters/pdf/extract.ts +574 -0
- package/src/markit/converters/pdf/grid.ts +780 -0
- package/src/markit/converters/pdf/headers.ts +106 -0
- package/src/markit/converters/pdf/index.ts +146 -0
- package/src/markit/converters/pdf/render.ts +501 -0
- package/src/markit/converters/pdf/types.ts +84 -0
- package/src/markit/converters/pptx.ts +325 -0
- package/src/markit/converters/xlsx.ts +173 -0
- package/src/markit/index.ts +2 -0
- package/src/markit/registry.ts +59 -0
- package/src/markit/types.ts +35 -0
- package/src/mcp/client.ts +509 -0
- package/src/mcp/config-writer.ts +229 -0
- package/src/mcp/config.ts +365 -0
- package/src/mcp/index.ts +29 -0
- package/src/mcp/json-rpc.ts +122 -0
- package/src/mcp/loader.ts +124 -0
- package/src/mcp/manager.ts +1326 -0
- package/src/mcp/oauth-credentials.ts +104 -0
- package/src/mcp/oauth-discovery.ts +467 -0
- package/src/mcp/oauth-flow.ts +555 -0
- package/src/mcp/render.ts +155 -0
- package/src/mcp/smithery-auth.ts +104 -0
- package/src/mcp/smithery-connect.ts +145 -0
- package/src/mcp/smithery-registry.ts +477 -0
- package/src/mcp/startup-events.ts +21 -0
- package/src/mcp/timeout.ts +59 -0
- package/src/mcp/tool-bridge.ts +429 -0
- package/src/mcp/tool-cache.ts +117 -0
- package/src/mcp/transports/http.ts +519 -0
- package/src/mcp/transports/index.ts +6 -0
- package/src/mcp/transports/stdio.ts +606 -0
- package/src/mcp/types.ts +427 -0
- package/src/memories/index.ts +1281 -0
- package/src/memories/storage.ts +578 -0
- package/src/memory-backend/index.ts +18 -0
- package/src/memory-backend/local-backend.ts +45 -0
- package/src/memory-backend/off-backend.ts +25 -0
- package/src/memory-backend/resolve.ts +25 -0
- package/src/memory-backend/runtime.ts +66 -0
- package/src/memory-backend/types.ts +166 -0
- package/src/mnemopi/backend.ts +612 -0
- package/src/mnemopi/config.ts +265 -0
- package/src/mnemopi/embed-client.ts +401 -0
- package/src/mnemopi/embed-protocol.ts +35 -0
- package/src/mnemopi/embed-worker.ts +113 -0
- package/src/mnemopi/index.ts +3 -0
- package/src/mnemopi/state.ts +657 -0
- package/src/modes/acp/acp-agent.ts +2362 -0
- package/src/modes/acp/acp-client-bridge.ts +154 -0
- package/src/modes/acp/acp-event-mapper.ts +933 -0
- package/src/modes/acp/acp-mode.ts +23 -0
- package/src/modes/acp/index.ts +2 -0
- package/src/modes/acp/terminal-auth.ts +37 -0
- package/src/modes/components/__tests__/skill-message.test.ts +92 -0
- package/src/modes/components/advisor-message.ts +99 -0
- package/src/modes/components/agent-dashboard.ts +1206 -0
- package/src/modes/components/agent-hub.ts +566 -0
- package/src/modes/components/agent-transcript-viewer.ts +461 -0
- package/src/modes/components/assistant-message.ts +612 -0
- package/src/modes/components/background-tan-message.ts +36 -0
- package/src/modes/components/bash-execution.ts +220 -0
- package/src/modes/components/bordered-loader.ts +41 -0
- package/src/modes/components/btw-panel.ts +112 -0
- package/src/modes/components/cache-invalidation-marker.ts +110 -0
- package/src/modes/components/chat-block.ts +111 -0
- package/src/modes/components/chat-transcript-builder.ts +476 -0
- package/src/modes/components/collab-prompt-message.ts +32 -0
- package/src/modes/components/compaction-summary-message.ts +215 -0
- package/src/modes/components/copy-selector.ts +206 -0
- package/src/modes/components/countdown-timer.ts +75 -0
- package/src/modes/components/custom-editor.test.ts +142 -0
- package/src/modes/components/custom-editor.ts +620 -0
- package/src/modes/components/custom-message.ts +67 -0
- package/src/modes/components/diff.ts +254 -0
- package/src/modes/components/dynamic-border.ts +34 -0
- package/src/modes/components/error-banner.ts +33 -0
- package/src/modes/components/eval-execution.ts +158 -0
- package/src/modes/components/execution-shared.ts +101 -0
- package/src/modes/components/extensions/extension-dashboard.ts +399 -0
- package/src/modes/components/extensions/extension-list.ts +502 -0
- package/src/modes/components/extensions/index.ts +9 -0
- package/src/modes/components/extensions/inspector-panel.ts +321 -0
- package/src/modes/components/extensions/state-manager.ts +627 -0
- package/src/modes/components/extensions/types.ts +186 -0
- package/src/modes/components/footer.ts +275 -0
- package/src/modes/components/history-search.ts +280 -0
- package/src/modes/components/hook-editor.ts +167 -0
- package/src/modes/components/hook-input.ts +87 -0
- package/src/modes/components/hook-message.ts +67 -0
- package/src/modes/components/hook-selector.ts +659 -0
- package/src/modes/components/index.ts +38 -0
- package/src/modes/components/keybinding-hints.ts +65 -0
- package/src/modes/components/late-diagnostics-message.ts +60 -0
- package/src/modes/components/login-dialog.ts +164 -0
- package/src/modes/components/logout-account-selector.ts +130 -0
- package/src/modes/components/mcp-add-wizard.ts +1360 -0
- package/src/modes/components/message-frame.ts +92 -0
- package/src/modes/components/model-selector.ts +1315 -0
- package/src/modes/components/oauth-selector.ts +457 -0
- package/src/modes/components/omfg-panel.ts +141 -0
- package/src/modes/components/overlay-box.ts +109 -0
- package/src/modes/components/plan-review-overlay.ts +847 -0
- package/src/modes/components/plan-toc.ts +138 -0
- package/src/modes/components/plugin-selector.ts +95 -0
- package/src/modes/components/plugin-settings.ts +739 -0
- package/src/modes/components/queue-mode-selector.ts +56 -0
- package/src/modes/components/read-tool-group.ts +676 -0
- package/src/modes/components/reset-usage-selector.ts +161 -0
- package/src/modes/components/segment-track.ts +89 -0
- package/src/modes/components/session-selector.ts +631 -0
- package/src/modes/components/settings-defs.ts +225 -0
- package/src/modes/components/settings-selector.ts +1095 -0
- package/src/modes/components/show-images-selector.ts +45 -0
- package/src/modes/components/skill-message.ts +110 -0
- package/src/modes/components/snapcompact-shape-preview-doc.md +18 -0
- package/src/modes/components/snapcompact-shape-preview.ts +192 -0
- package/src/modes/components/status-line/component.ts +1001 -0
- package/src/modes/components/status-line/context-thresholds.ts +78 -0
- package/src/modes/components/status-line/git-utils.ts +42 -0
- package/src/modes/components/status-line/index.ts +5 -0
- package/src/modes/components/status-line/presets.ts +106 -0
- package/src/modes/components/status-line/segments.ts +616 -0
- package/src/modes/components/status-line/separators.ts +55 -0
- package/src/modes/components/status-line/token-rate.ts +66 -0
- package/src/modes/components/status-line/types.ts +124 -0
- package/src/modes/components/theme-selector.ts +63 -0
- package/src/modes/components/thinking-selector.ts +52 -0
- package/src/modes/components/tiny-title-download-progress.ts +90 -0
- package/src/modes/components/tips.txt +24 -0
- package/src/modes/components/todo-reminder.ts +39 -0
- package/src/modes/components/tool-execution.ts +1165 -0
- package/src/modes/components/transcript-container.ts +806 -0
- package/src/modes/components/tree-selector.ts +994 -0
- package/src/modes/components/ttsr-notification.ts +123 -0
- package/src/modes/components/usage-row.ts +18 -0
- package/src/modes/components/user-message-selector.ts +227 -0
- package/src/modes/components/user-message.ts +68 -0
- package/src/modes/components/visual-truncate.ts +63 -0
- package/src/modes/components/welcome.ts +581 -0
- package/src/modes/controllers/btw-controller.ts +173 -0
- package/src/modes/controllers/command-controller-shared.ts +109 -0
- package/src/modes/controllers/command-controller.ts +1653 -0
- package/src/modes/controllers/event-controller.ts +1153 -0
- package/src/modes/controllers/extension-ui-controller.ts +893 -0
- package/src/modes/controllers/input-controller.ts +1627 -0
- package/src/modes/controllers/mcp-command-controller.ts +2162 -0
- package/src/modes/controllers/omfg-controller.ts +283 -0
- package/src/modes/controllers/omfg-rule.ts +647 -0
- package/src/modes/controllers/selector-controller.ts +1285 -0
- package/src/modes/controllers/session-focus-controller.ts +112 -0
- package/src/modes/controllers/ssh-command-controller.ts +384 -0
- package/src/modes/controllers/streaming-reveal.ts +295 -0
- package/src/modes/controllers/tan-command-controller.ts +190 -0
- package/src/modes/controllers/todo-command-controller.ts +485 -0
- package/src/modes/controllers/tool-args-reveal.ts +174 -0
- package/src/modes/data/emojis.json +1 -0
- package/src/modes/emoji-autocomplete.ts +285 -0
- package/src/modes/gradient-highlight.ts +99 -0
- package/src/modes/image-references.ts +137 -0
- package/src/modes/index.ts +17 -0
- package/src/modes/interactive-mode.ts +3940 -0
- package/src/modes/internal-url-autocomplete.ts +143 -0
- package/src/modes/loop-limit.ts +192 -0
- package/src/modes/magic-keywords.ts +42 -0
- package/src/modes/markdown-prose.ts +247 -0
- package/src/modes/oauth-manual-input.ts +69 -0
- package/src/modes/orchestrate.ts +42 -0
- package/src/modes/print-mode.ts +130 -0
- package/src/modes/prompt-action-autocomplete.ts +260 -0
- package/src/modes/rpc/host-tools.ts +186 -0
- package/src/modes/rpc/host-uris.ts +235 -0
- package/src/modes/rpc/rpc-client.ts +995 -0
- package/src/modes/rpc/rpc-mode.ts +1156 -0
- package/src/modes/rpc/rpc-subagents.ts +265 -0
- package/src/modes/rpc/rpc-types.ts +487 -0
- package/src/modes/runtime-init.ts +142 -0
- package/src/modes/session-observer-registry.ts +215 -0
- package/src/modes/setup-version.ts +11 -0
- package/src/modes/setup-wizard/index.ts +101 -0
- package/src/modes/setup-wizard/lazy.ts +16 -0
- package/src/modes/setup-wizard/scenes/glyph.ts +114 -0
- package/src/modes/setup-wizard/scenes/outro.ts +35 -0
- package/src/modes/setup-wizard/scenes/providers.ts +103 -0
- package/src/modes/setup-wizard/scenes/sign-in.ts +286 -0
- package/src/modes/setup-wizard/scenes/splash.ts +201 -0
- package/src/modes/setup-wizard/scenes/theme.ts +326 -0
- package/src/modes/setup-wizard/scenes/types.ts +57 -0
- package/src/modes/setup-wizard/scenes/web-search.ts +145 -0
- package/src/modes/setup-wizard/startup-splash.ts +107 -0
- package/src/modes/setup-wizard/wizard-overlay.ts +334 -0
- package/src/modes/shared.ts +49 -0
- package/src/modes/theme/dark.json +95 -0
- package/src/modes/theme/defaults/alabaster.json +93 -0
- package/src/modes/theme/defaults/amethyst.json +96 -0
- package/src/modes/theme/defaults/anthracite.json +93 -0
- package/src/modes/theme/defaults/basalt.json +91 -0
- package/src/modes/theme/defaults/birch.json +95 -0
- package/src/modes/theme/defaults/dark-abyss.json +91 -0
- package/src/modes/theme/defaults/dark-arctic.json +104 -0
- package/src/modes/theme/defaults/dark-aurora.json +95 -0
- package/src/modes/theme/defaults/dark-catppuccin.json +107 -0
- package/src/modes/theme/defaults/dark-cavern.json +91 -0
- package/src/modes/theme/defaults/dark-copper.json +95 -0
- package/src/modes/theme/defaults/dark-cosmos.json +90 -0
- package/src/modes/theme/defaults/dark-cyberpunk.json +102 -0
- package/src/modes/theme/defaults/dark-dracula.json +98 -0
- package/src/modes/theme/defaults/dark-eclipse.json +91 -0
- package/src/modes/theme/defaults/dark-ember.json +95 -0
- package/src/modes/theme/defaults/dark-equinox.json +90 -0
- package/src/modes/theme/defaults/dark-forest.json +96 -0
- package/src/modes/theme/defaults/dark-github.json +105 -0
- package/src/modes/theme/defaults/dark-gruvbox.json +112 -0
- package/src/modes/theme/defaults/dark-lavender.json +95 -0
- package/src/modes/theme/defaults/dark-lunar.json +89 -0
- package/src/modes/theme/defaults/dark-midnight.json +95 -0
- package/src/modes/theme/defaults/dark-monochrome.json +94 -0
- package/src/modes/theme/defaults/dark-monokai.json +98 -0
- package/src/modes/theme/defaults/dark-nebula.json +90 -0
- package/src/modes/theme/defaults/dark-nord.json +97 -0
- package/src/modes/theme/defaults/dark-ocean.json +101 -0
- package/src/modes/theme/defaults/dark-one.json +100 -0
- package/src/modes/theme/defaults/dark-poimandres.json +142 -0
- package/src/modes/theme/defaults/dark-rainforest.json +91 -0
- package/src/modes/theme/defaults/dark-reef.json +91 -0
- package/src/modes/theme/defaults/dark-retro.json +92 -0
- package/src/modes/theme/defaults/dark-rose-pine.json +96 -0
- package/src/modes/theme/defaults/dark-sakura.json +95 -0
- package/src/modes/theme/defaults/dark-slate.json +95 -0
- package/src/modes/theme/defaults/dark-solarized.json +97 -0
- package/src/modes/theme/defaults/dark-solstice.json +90 -0
- package/src/modes/theme/defaults/dark-starfall.json +91 -0
- package/src/modes/theme/defaults/dark-sunset.json +99 -0
- package/src/modes/theme/defaults/dark-swamp.json +90 -0
- package/src/modes/theme/defaults/dark-synthwave.json +103 -0
- package/src/modes/theme/defaults/dark-taiga.json +91 -0
- package/src/modes/theme/defaults/dark-terminal.json +95 -0
- package/src/modes/theme/defaults/dark-tokyo-night.json +101 -0
- package/src/modes/theme/defaults/dark-tundra.json +91 -0
- package/src/modes/theme/defaults/dark-twilight.json +91 -0
- package/src/modes/theme/defaults/dark-volcanic.json +91 -0
- package/src/modes/theme/defaults/graphite.json +92 -0
- package/src/modes/theme/defaults/index.ts +199 -0
- package/src/modes/theme/defaults/light-arctic.json +107 -0
- package/src/modes/theme/defaults/light-aurora-day.json +91 -0
- package/src/modes/theme/defaults/light-canyon.json +91 -0
- package/src/modes/theme/defaults/light-catppuccin.json +106 -0
- package/src/modes/theme/defaults/light-cirrus.json +90 -0
- package/src/modes/theme/defaults/light-coral.json +95 -0
- package/src/modes/theme/defaults/light-cyberpunk.json +96 -0
- package/src/modes/theme/defaults/light-dawn.json +90 -0
- package/src/modes/theme/defaults/light-dunes.json +91 -0
- package/src/modes/theme/defaults/light-eucalyptus.json +95 -0
- package/src/modes/theme/defaults/light-forest.json +100 -0
- package/src/modes/theme/defaults/light-frost.json +95 -0
- package/src/modes/theme/defaults/light-github.json +115 -0
- package/src/modes/theme/defaults/light-glacier.json +91 -0
- package/src/modes/theme/defaults/light-gruvbox.json +108 -0
- package/src/modes/theme/defaults/light-haze.json +90 -0
- package/src/modes/theme/defaults/light-honeycomb.json +95 -0
- package/src/modes/theme/defaults/light-lagoon.json +91 -0
- package/src/modes/theme/defaults/light-lavender.json +95 -0
- package/src/modes/theme/defaults/light-meadow.json +91 -0
- package/src/modes/theme/defaults/light-mint.json +95 -0
- package/src/modes/theme/defaults/light-monochrome.json +101 -0
- package/src/modes/theme/defaults/light-ocean.json +99 -0
- package/src/modes/theme/defaults/light-one.json +99 -0
- package/src/modes/theme/defaults/light-opal.json +91 -0
- package/src/modes/theme/defaults/light-orchard.json +91 -0
- package/src/modes/theme/defaults/light-paper.json +95 -0
- package/src/modes/theme/defaults/light-poimandres.json +142 -0
- package/src/modes/theme/defaults/light-prism.json +90 -0
- package/src/modes/theme/defaults/light-retro.json +98 -0
- package/src/modes/theme/defaults/light-sand.json +95 -0
- package/src/modes/theme/defaults/light-savanna.json +91 -0
- package/src/modes/theme/defaults/light-solarized.json +102 -0
- package/src/modes/theme/defaults/light-soleil.json +90 -0
- package/src/modes/theme/defaults/light-sunset.json +99 -0
- package/src/modes/theme/defaults/light-synthwave.json +98 -0
- package/src/modes/theme/defaults/light-tokyo-night.json +111 -0
- package/src/modes/theme/defaults/light-wetland.json +91 -0
- package/src/modes/theme/defaults/light-zenith.json +89 -0
- package/src/modes/theme/defaults/limestone.json +94 -0
- package/src/modes/theme/defaults/mahogany.json +97 -0
- package/src/modes/theme/defaults/marble.json +93 -0
- package/src/modes/theme/defaults/obsidian.json +91 -0
- package/src/modes/theme/defaults/onyx.json +91 -0
- package/src/modes/theme/defaults/pearl.json +93 -0
- package/src/modes/theme/defaults/porcelain.json +91 -0
- package/src/modes/theme/defaults/quartz.json +96 -0
- package/src/modes/theme/defaults/sandstone.json +95 -0
- package/src/modes/theme/defaults/titanium.json +90 -0
- package/src/modes/theme/light.json +93 -0
- package/src/modes/theme/mermaid-cache.ts +92 -0
- package/src/modes/theme/shimmer.ts +235 -0
- package/src/modes/theme/theme-schema.json +459 -0
- package/src/modes/theme/theme.ts +2915 -0
- package/src/modes/turn-budget.ts +31 -0
- package/src/modes/types.ts +406 -0
- package/src/modes/ultrathink.ts +41 -0
- package/src/modes/utils/context-usage.ts +432 -0
- package/src/modes/utils/copy-targets.ts +360 -0
- package/src/modes/utils/hotkeys-markdown.ts +62 -0
- package/src/modes/utils/keybinding-matchers.ts +51 -0
- package/src/modes/utils/tools-markdown.ts +27 -0
- package/src/modes/utils/ui-helpers.ts +886 -0
- package/src/modes/workflow.ts +42 -0
- package/src/plan-mode/approved-plan.ts +186 -0
- package/src/plan-mode/plan-handoff.ts +37 -0
- package/src/plan-mode/plan-protection.ts +31 -0
- package/src/plan-mode/state.ts +6 -0
- package/src/priority.json +45 -0
- package/src/prompts/advisor/advise-tool.md +3 -0
- package/src/prompts/advisor/system.md +113 -0
- package/src/prompts/agents/designer.md +74 -0
- package/src/prompts/agents/explore.md +58 -0
- package/src/prompts/agents/frontmatter.md +11 -0
- package/src/prompts/agents/init.md +33 -0
- package/src/prompts/agents/librarian.md +119 -0
- package/src/prompts/agents/oracle.md +54 -0
- package/src/prompts/agents/plan.md +48 -0
- package/src/prompts/agents/reviewer.md +139 -0
- package/src/prompts/agents/task.md +17 -0
- package/src/prompts/bench.md +12 -0
- package/src/prompts/ci-green-request.md +36 -0
- package/src/prompts/dry-balance-bench.md +8 -0
- package/src/prompts/goals/goal-budget-limit.md +16 -0
- package/src/prompts/goals/goal-continuation.md +28 -0
- package/src/prompts/goals/goal-mode-active.md +23 -0
- package/src/prompts/goals/guided-goal-interview.md +8 -0
- package/src/prompts/goals/guided-goal-system.md +12 -0
- package/src/prompts/low-end/system.md +47 -0
- package/src/prompts/memories/consolidation.md +30 -0
- package/src/prompts/memories/consolidation_system.md +4 -0
- package/src/prompts/memories/read-path.md +17 -0
- package/src/prompts/memories/stage_one_input.md +6 -0
- package/src/prompts/memories/stage_one_system.md +21 -0
- package/src/prompts/review-custom-request.md +22 -0
- package/src/prompts/review-headless-request.md +16 -0
- package/src/prompts/review-request.md +69 -0
- package/src/prompts/steering/user-interjection.md +9 -0
- package/src/prompts/system/agent-creation-architect.md +50 -0
- package/src/prompts/system/agent-creation-user.md +6 -0
- package/src/prompts/system/auto-continue.md +1 -0
- package/src/prompts/system/auto-thinking-difficulty-local.md +14 -0
- package/src/prompts/system/auto-thinking-difficulty.md +12 -0
- package/src/prompts/system/autolearn-guidance-learn.md +1 -0
- package/src/prompts/system/autolearn-guidance.md +7 -0
- package/src/prompts/system/autolearn-nudge.md +3 -0
- package/src/prompts/system/background-tan-dispatch.md +8 -0
- package/src/prompts/system/btw-user.md +8 -0
- package/src/prompts/system/commit-message-system.md +14 -0
- package/src/prompts/system/custom-system-prompt.md +64 -0
- package/src/prompts/system/eager-task.md +7 -0
- package/src/prompts/system/eager-todo.md +18 -0
- package/src/prompts/system/empty-stop-retry.md +4 -0
- package/src/prompts/system/irc-autoreply.md +6 -0
- package/src/prompts/system/irc-incoming.md +7 -0
- package/src/prompts/system/manual-continue.md +7 -0
- package/src/prompts/system/memory-consolidation-system.md +8 -0
- package/src/prompts/system/memory-extraction-system.md +26 -0
- package/src/prompts/system/omfg-user.md +50 -0
- package/src/prompts/system/orchestrate-notice.md +40 -0
- package/src/prompts/system/personalities/default.md +18 -0
- package/src/prompts/system/personalities/friendly.md +17 -0
- package/src/prompts/system/personalities/pragmatic.md +15 -0
- package/src/prompts/system/plan-mode-active.md +109 -0
- package/src/prompts/system/plan-mode-approved.md +25 -0
- package/src/prompts/system/plan-mode-compact-instructions.md +16 -0
- package/src/prompts/system/plan-mode-reference.md +11 -0
- package/src/prompts/system/plan-mode-subagent.md +33 -0
- package/src/prompts/system/plan-mode-tool-decision-reminder.md +9 -0
- package/src/prompts/system/project-prompt.md +52 -0
- package/src/prompts/system/snapcompact-context-frames-note.md +1 -0
- package/src/prompts/system/snapcompact-context-stub.md +1 -0
- package/src/prompts/system/snapcompact-system-frames-note.md +1 -0
- package/src/prompts/system/snapcompact-system-stub.md +1 -0
- package/src/prompts/system/snapcompact-toolresult-note.md +1 -0
- package/src/prompts/system/subagent-system-prompt.md +71 -0
- package/src/prompts/system/subagent-user-prompt.md +3 -0
- package/src/prompts/system/subagent-yield-reminder.md +12 -0
- package/src/prompts/system/system-prompt.md +251 -0
- package/src/prompts/system/tiny-title-system.md +8 -0
- package/src/prompts/system/title-marker-instruction.md +1 -0
- package/src/prompts/system/title-system-marker.md +16 -0
- package/src/prompts/system/title-system.md +16 -0
- package/src/prompts/system/ttsr-interrupt.md +7 -0
- package/src/prompts/system/ttsr-tool-reminder.md +5 -0
- package/src/prompts/system/ultrathink-notice.md +3 -0
- package/src/prompts/system/unexpected-stop-classifier.md +17 -0
- package/src/prompts/system/unexpected-stop-retry.md +4 -0
- package/src/prompts/system/web-search.md +25 -0
- package/src/prompts/system/workflow-notice.md +70 -0
- package/src/prompts/tools/apply-patch.md +65 -0
- package/src/prompts/tools/ask.md +22 -0
- package/src/prompts/tools/ast-edit.md +22 -0
- package/src/prompts/tools/ast-grep.md +25 -0
- package/src/prompts/tools/async-result.md +8 -0
- package/src/prompts/tools/bash.md +45 -0
- package/src/prompts/tools/browser.md +42 -0
- package/src/prompts/tools/checkpoint.md +15 -0
- package/src/prompts/tools/debug.md +17 -0
- package/src/prompts/tools/eval.md +70 -0
- package/src/prompts/tools/find.md +19 -0
- package/src/prompts/tools/github.md +17 -0
- package/src/prompts/tools/goal.md +11 -0
- package/src/prompts/tools/image-attachment-describe-system.md +8 -0
- package/src/prompts/tools/image-attachment-describe.md +10 -0
- package/src/prompts/tools/image-gen.md +7 -0
- package/src/prompts/tools/inspect-image-system.md +20 -0
- package/src/prompts/tools/inspect-image.md +22 -0
- package/src/prompts/tools/irc.md +33 -0
- package/src/prompts/tools/job.md +17 -0
- package/src/prompts/tools/learn.md +7 -0
- package/src/prompts/tools/lsp-late-diagnostic.md +8 -0
- package/src/prompts/tools/lsp.md +39 -0
- package/src/prompts/tools/manage-skill.md +9 -0
- package/src/prompts/tools/memory-edit.md +8 -0
- package/src/prompts/tools/patch.md +57 -0
- package/src/prompts/tools/read.md +76 -0
- package/src/prompts/tools/recall.md +5 -0
- package/src/prompts/tools/reflect.md +5 -0
- package/src/prompts/tools/replace.md +29 -0
- package/src/prompts/tools/resolve.md +4 -0
- package/src/prompts/tools/retain.md +6 -0
- package/src/prompts/tools/rewind.md +13 -0
- package/src/prompts/tools/search-tool-bm25.md +32 -0
- package/src/prompts/tools/search.md +22 -0
- package/src/prompts/tools/ssh.md +22 -0
- package/src/prompts/tools/task-summary.md +17 -0
- package/src/prompts/tools/task.md +91 -0
- package/src/prompts/tools/todo.md +39 -0
- package/src/prompts/tools/web-search.md +6 -0
- package/src/prompts/tools/write.md +14 -0
- package/src/registry/agent-lifecycle.ts +270 -0
- package/src/registry/agent-registry.ts +190 -0
- package/src/sdk.ts +2919 -0
- package/src/secrets/index.ts +123 -0
- package/src/secrets/obfuscator.ts +298 -0
- package/src/secrets/regex.ts +21 -0
- package/src/session/agent-session.ts +12539 -0
- package/src/session/agent-storage.ts +478 -0
- package/src/session/artifacts.ts +153 -0
- package/src/session/auth-broker-config.ts +92 -0
- package/src/session/auth-storage.ts +24 -0
- package/src/session/blob-store.ts +255 -0
- package/src/session/client-bridge.ts +85 -0
- package/src/session/codex-auto-reset.ts +202 -0
- package/src/session/compact-modes.ts +105 -0
- package/src/session/history-storage.ts +361 -0
- package/src/session/indexed-session-storage.ts +427 -0
- package/src/session/messages.ts +546 -0
- package/src/session/redis-session-storage.ts +170 -0
- package/src/session/session-context.ts +399 -0
- package/src/session/session-dump-format.ts +216 -0
- package/src/session/session-entries.ts +198 -0
- package/src/session/session-history-format.ts +308 -0
- package/src/session/session-listing.ts +588 -0
- package/src/session/session-loader.ts +93 -0
- package/src/session/session-manager.ts +1748 -0
- package/src/session/session-migrations.ts +78 -0
- package/src/session/session-paths.ts +193 -0
- package/src/session/session-persistence.ts +147 -0
- package/src/session/session-storage.ts +590 -0
- package/src/session/shake-types.ts +43 -0
- package/src/session/snapcompact-inline.ts +542 -0
- package/src/session/snapcompact-savings-journal.ts +113 -0
- package/src/session/sql-session-storage.ts +314 -0
- package/src/session/streaming-output.ts +1330 -0
- package/src/session/tool-choice-queue.ts +290 -0
- package/src/session/unexpected-stop-classifier.ts +129 -0
- package/src/session/yield-queue.ts +183 -0
- package/src/slash-commands/acp-builtins.ts +70 -0
- package/src/slash-commands/available-commands.ts +105 -0
- package/src/slash-commands/builtin-registry.ts +2332 -0
- package/src/slash-commands/helpers/active-oauth-account.ts +44 -0
- package/src/slash-commands/helpers/collab-qrcode.ts +28 -0
- package/src/slash-commands/helpers/context-report.ts +66 -0
- package/src/slash-commands/helpers/format.ts +46 -0
- package/src/slash-commands/helpers/logout.ts +88 -0
- package/src/slash-commands/helpers/marketplace-manager.ts +25 -0
- package/src/slash-commands/helpers/mcp.ts +532 -0
- package/src/slash-commands/helpers/parse.ts +85 -0
- package/src/slash-commands/helpers/reset-usage.ts +66 -0
- package/src/slash-commands/helpers/ssh.ts +195 -0
- package/src/slash-commands/helpers/stats-dashboard.ts +85 -0
- package/src/slash-commands/helpers/todo.ts +279 -0
- package/src/slash-commands/helpers/usage-report.ts +128 -0
- package/src/slash-commands/marketplace-install-parser.ts +99 -0
- package/src/slash-commands/types.ts +135 -0
- package/src/ssh/config-writer.ts +183 -0
- package/src/ssh/connection-manager.ts +510 -0
- package/src/ssh/ssh-executor.ts +189 -0
- package/src/ssh/sshfs-mount.ts +140 -0
- package/src/ssh/utils.ts +8 -0
- package/src/startup-splash.ts +19 -0
- package/src/stt/asr-client.ts +521 -0
- package/src/stt/asr-protocol.ts +65 -0
- package/src/stt/asr-worker.ts +790 -0
- package/src/stt/downloader.ts +138 -0
- package/src/stt/endpointer.ts +259 -0
- package/src/stt/index.ts +7 -0
- package/src/stt/models.ts +150 -0
- package/src/stt/recorder.ts +538 -0
- package/src/stt/stt-controller.ts +380 -0
- package/src/stt/transcriber.ts +60 -0
- package/src/stt/wav.ts +173 -0
- package/src/system-prompt.ts +709 -0
- package/src/task/agents.ts +166 -0
- package/src/task/commands.ts +132 -0
- package/src/task/discovery.ts +122 -0
- package/src/task/executor.ts +2356 -0
- package/src/task/index.ts +1580 -0
- package/src/task/name-generator.ts +1577 -0
- package/src/task/omp-command.ts +26 -0
- package/src/task/output-manager.ts +93 -0
- package/src/task/parallel.ts +116 -0
- package/src/task/persisted-revive.ts +128 -0
- package/src/task/render.ts +1558 -0
- package/src/task/repair-args.ts +129 -0
- package/src/task/subprocess-tool-registry.ts +88 -0
- package/src/task/types.ts +401 -0
- package/src/task/worktree.ts +514 -0
- package/src/telemetry-export.ts +144 -0
- package/src/thinking.ts +187 -0
- package/src/tiny/device.ts +111 -0
- package/src/tiny/dtype.ts +101 -0
- package/src/tiny/models.ts +252 -0
- package/src/tiny/text.ts +169 -0
- package/src/tiny/title-client.ts +538 -0
- package/src/tiny/title-protocol.ts +56 -0
- package/src/tiny/worker.ts +491 -0
- package/src/tool-discovery/mode.ts +24 -0
- package/src/tool-discovery/tool-index.ts +271 -0
- package/src/tools/__tests__/json-tree.test.ts +35 -0
- package/src/tools/approval.ts +189 -0
- package/src/tools/ask.ts +977 -0
- package/src/tools/ast-edit.ts +700 -0
- package/src/tools/ast-grep.ts +483 -0
- package/src/tools/auto-generated-guard.ts +322 -0
- package/src/tools/bash-command-fixup.ts +37 -0
- package/src/tools/bash-interactive.ts +408 -0
- package/src/tools/bash-interceptor.ts +67 -0
- package/src/tools/bash-pty-selection.ts +14 -0
- package/src/tools/bash-skill-urls.ts +248 -0
- package/src/tools/bash.ts +1405 -0
- package/src/tools/browser/attach.ts +194 -0
- package/src/tools/browser/cmux/cmux-tab.ts +1264 -0
- package/src/tools/browser/cmux/rpc.ts +156 -0
- package/src/tools/browser/cmux/socket-client.ts +309 -0
- package/src/tools/browser/launch.ts +673 -0
- package/src/tools/browser/readable.ts +112 -0
- package/src/tools/browser/registry.ts +241 -0
- package/src/tools/browser/render.ts +221 -0
- package/src/tools/browser/tab-protocol.ts +107 -0
- package/src/tools/browser/tab-supervisor.ts +799 -0
- package/src/tools/browser/tab-worker-entry.ts +29 -0
- package/src/tools/browser/tab-worker.ts +1226 -0
- package/src/tools/browser.ts +403 -0
- package/src/tools/builtin-names.ts +34 -0
- package/src/tools/checkpoint.ts +136 -0
- package/src/tools/conflict-detect.ts +718 -0
- package/src/tools/context.ts +39 -0
- package/src/tools/debug.ts +1087 -0
- package/src/tools/eval-backends.ts +27 -0
- package/src/tools/eval-render.ts +762 -0
- package/src/tools/eval.ts +600 -0
- package/src/tools/fetch.ts +1902 -0
- package/src/tools/file-recorder.ts +35 -0
- package/src/tools/find.ts +629 -0
- package/src/tools/fs-cache-invalidation.ts +28 -0
- package/src/tools/gh-cache-invalidation.ts +255 -0
- package/src/tools/gh-format.ts +12 -0
- package/src/tools/gh-renderer.ts +481 -0
- package/src/tools/gh.ts +3752 -0
- package/src/tools/github-cache.ts +663 -0
- package/src/tools/grouped-file-output.ts +210 -0
- package/src/tools/image-gen.ts +1586 -0
- package/src/tools/index.ts +649 -0
- package/src/tools/inspect-image-renderer.ts +132 -0
- package/src/tools/inspect-image.ts +260 -0
- package/src/tools/irc.ts +788 -0
- package/src/tools/job.ts +612 -0
- package/src/tools/json-tree.ts +260 -0
- package/src/tools/jtd-to-json-schema.ts +219 -0
- package/src/tools/jtd-to-typescript.ts +136 -0
- package/src/tools/jtd-utils.ts +102 -0
- package/src/tools/learn.ts +141 -0
- package/src/tools/list-limit.ts +40 -0
- package/src/tools/manage-skill.ts +100 -0
- package/src/tools/match-line-format.ts +20 -0
- package/src/tools/memory-edit.ts +59 -0
- package/src/tools/memory-recall.ts +102 -0
- package/src/tools/memory-reflect.ts +88 -0
- package/src/tools/memory-render.ts +202 -0
- package/src/tools/memory-retain.ts +89 -0
- package/src/tools/output-meta.ts +768 -0
- package/src/tools/output-schema-validator.ts +132 -0
- package/src/tools/path-utils.ts +1116 -0
- package/src/tools/plan-mode-guard.ts +142 -0
- package/src/tools/puppeteer/00_stealth_tampering.txt +63 -0
- package/src/tools/puppeteer/01_stealth_activity.txt +20 -0
- package/src/tools/puppeteer/02_stealth_hairline.txt +11 -0
- package/src/tools/puppeteer/03_stealth_botd.txt +384 -0
- package/src/tools/puppeteer/04_stealth_iframe.txt +81 -0
- package/src/tools/puppeteer/05_stealth_webgl.txt +75 -0
- package/src/tools/puppeteer/06_stealth_screen.txt +72 -0
- package/src/tools/puppeteer/07_stealth_fonts.txt +97 -0
- package/src/tools/puppeteer/08_stealth_audio.txt +51 -0
- package/src/tools/puppeteer/09_stealth_locale.txt +46 -0
- package/src/tools/puppeteer/10_stealth_plugins.txt +208 -0
- package/src/tools/puppeteer/11_stealth_hardware.txt +8 -0
- package/src/tools/puppeteer/12_stealth_codecs.txt +40 -0
- package/src/tools/puppeteer/13_stealth_worker.txt +74 -0
- package/src/tools/read.ts +3124 -0
- package/src/tools/render-utils.ts +895 -0
- package/src/tools/renderers.ts +86 -0
- package/src/tools/report-tool-issue.ts +530 -0
- package/src/tools/resolve.ts +302 -0
- package/src/tools/review.ts +251 -0
- package/src/tools/search-tool-bm25.ts +351 -0
- package/src/tools/search.ts +1583 -0
- package/src/tools/sqlite-reader.ts +828 -0
- package/src/tools/ssh.ts +369 -0
- package/src/tools/todo.ts +938 -0
- package/src/tools/tool-errors.ts +62 -0
- package/src/tools/tool-result.ts +102 -0
- package/src/tools/tool-timeouts.ts +30 -0
- package/src/tools/tts.ts +265 -0
- package/src/tools/write.ts +1182 -0
- package/src/tools/yield.ts +269 -0
- package/src/tts/downloader.ts +64 -0
- package/src/tts/index.ts +8 -0
- package/src/tts/models.ts +137 -0
- package/src/tts/player.ts +137 -0
- package/src/tts/runtime.ts +21 -0
- package/src/tts/streaming-player.ts +266 -0
- package/src/tts/tts-client.ts +642 -0
- package/src/tts/tts-protocol.ts +60 -0
- package/src/tts/tts-worker.ts +505 -0
- package/src/tts/vocalizer.ts +162 -0
- package/src/tts/wav.ts +58 -0
- package/src/tui/code-cell.ts +257 -0
- package/src/tui/file-list.ts +55 -0
- package/src/tui/hyperlink.ts +178 -0
- package/src/tui/index.ts +13 -0
- package/src/tui/output-block.ts +240 -0
- package/src/tui/status-line.ts +54 -0
- package/src/tui/tree-list.ts +133 -0
- package/src/tui/types.ts +15 -0
- package/src/tui/utils.ts +103 -0
- package/src/tui/width-aware-text.ts +58 -0
- package/src/utils/block-context.ts +312 -0
- package/src/utils/changelog.ts +132 -0
- package/src/utils/clipboard.ts +262 -0
- package/src/utils/command-args.ts +76 -0
- package/src/utils/commit-message-generator.ts +147 -0
- package/src/utils/edit-mode.ts +41 -0
- package/src/utils/enhanced-paste.ts +230 -0
- package/src/utils/event-bus.ts +33 -0
- package/src/utils/external-editor.ts +78 -0
- package/src/utils/file-display-mode.ts +45 -0
- package/src/utils/file-mentions.ts +284 -0
- package/src/utils/git.ts +1838 -0
- package/src/utils/image-loading.ts +231 -0
- package/src/utils/image-resize.ts +309 -0
- package/src/utils/image-vision-fallback.ts +197 -0
- package/src/utils/ipc.ts +38 -0
- package/src/utils/jj.ts +248 -0
- package/src/utils/lang-from-path.ts +244 -0
- package/src/utils/markit.ts +143 -0
- package/src/utils/mupdf-wasm-embed.ts +12 -0
- package/src/utils/open.ts +55 -0
- package/src/utils/qrcode.ts +535 -0
- package/src/utils/session-color.ts +142 -0
- package/src/utils/shell-snapshot.ts +187 -0
- package/src/utils/sixel.ts +69 -0
- package/src/utils/thinking-display.ts +11 -0
- package/src/utils/title-generator.ts +416 -0
- package/src/utils/tool-choice.ts +49 -0
- package/src/utils/tools-manager.ts +372 -0
- package/src/utils/turndown.ts +83 -0
- package/src/utils/zip.ts +1091 -0
- package/src/web/kagi.ts +304 -0
- package/src/web/parallel.ts +353 -0
- package/src/web/scrapers/artifacthub.ts +207 -0
- package/src/web/scrapers/arxiv.ts +83 -0
- package/src/web/scrapers/aur.ts +162 -0
- package/src/web/scrapers/biorxiv.ts +133 -0
- package/src/web/scrapers/bluesky.ts +262 -0
- package/src/web/scrapers/brew.ts +172 -0
- package/src/web/scrapers/cheatsh.ts +68 -0
- package/src/web/scrapers/chocolatey.ts +196 -0
- package/src/web/scrapers/choosealicense.ts +95 -0
- package/src/web/scrapers/cisa-kev.ts +87 -0
- package/src/web/scrapers/clojars.ts +154 -0
- package/src/web/scrapers/coingecko.ts +177 -0
- package/src/web/scrapers/crates-io.ts +97 -0
- package/src/web/scrapers/crossref.ts +136 -0
- package/src/web/scrapers/devto.ts +147 -0
- package/src/web/scrapers/discogs.ts +306 -0
- package/src/web/scrapers/discourse.ts +197 -0
- package/src/web/scrapers/dockerhub.ts +138 -0
- package/src/web/scrapers/docs-rs.ts +652 -0
- package/src/web/scrapers/fdroid.ts +134 -0
- package/src/web/scrapers/firefox-addons.ts +191 -0
- package/src/web/scrapers/flathub.ts +223 -0
- package/src/web/scrapers/github-gist.ts +58 -0
- package/src/web/scrapers/github.ts +800 -0
- package/src/web/scrapers/gitlab.ts +401 -0
- package/src/web/scrapers/go-pkg.ts +266 -0
- package/src/web/scrapers/hackage.ts +140 -0
- package/src/web/scrapers/hackernews.ts +189 -0
- package/src/web/scrapers/hex.ts +105 -0
- package/src/web/scrapers/huggingface.ts +321 -0
- package/src/web/scrapers/iacr.ts +89 -0
- package/src/web/scrapers/index.ts +252 -0
- package/src/web/scrapers/jetbrains-marketplace.ts +159 -0
- package/src/web/scrapers/lemmy.ts +203 -0
- package/src/web/scrapers/lobsters.ts +175 -0
- package/src/web/scrapers/mastodon.ts +292 -0
- package/src/web/scrapers/maven.ts +138 -0
- package/src/web/scrapers/mdn.ts +173 -0
- package/src/web/scrapers/metacpan.ts +222 -0
- package/src/web/scrapers/musicbrainz.ts +250 -0
- package/src/web/scrapers/npm.ts +98 -0
- package/src/web/scrapers/nuget.ts +183 -0
- package/src/web/scrapers/nvd.ts +222 -0
- package/src/web/scrapers/ollama.ts +239 -0
- package/src/web/scrapers/open-vsx.ts +106 -0
- package/src/web/scrapers/opencorporates.ts +292 -0
- package/src/web/scrapers/openlibrary.ts +336 -0
- package/src/web/scrapers/orcid.ts +286 -0
- package/src/web/scrapers/osv.ts +176 -0
- package/src/web/scrapers/packagist.ts +160 -0
- package/src/web/scrapers/pub-dev.ts +143 -0
- package/src/web/scrapers/pubmed.ts +211 -0
- package/src/web/scrapers/pypi.ts +112 -0
- package/src/web/scrapers/rawg.ts +110 -0
- package/src/web/scrapers/readthedocs.ts +120 -0
- package/src/web/scrapers/reddit.ts +95 -0
- package/src/web/scrapers/repology.ts +251 -0
- package/src/web/scrapers/rfc.ts +201 -0
- package/src/web/scrapers/rubygems.ts +103 -0
- package/src/web/scrapers/searchcode.ts +189 -0
- package/src/web/scrapers/sec-edgar.ts +261 -0
- package/src/web/scrapers/semantic-scholar.ts +171 -0
- package/src/web/scrapers/snapcraft.ts +187 -0
- package/src/web/scrapers/sourcegraph.ts +336 -0
- package/src/web/scrapers/spdx.ts +108 -0
- package/src/web/scrapers/spotify.ts +198 -0
- package/src/web/scrapers/stackoverflow.ts +120 -0
- package/src/web/scrapers/terraform.ts +277 -0
- package/src/web/scrapers/tldr.ts +47 -0
- package/src/web/scrapers/twitter.ts +94 -0
- package/src/web/scrapers/types.ts +354 -0
- package/src/web/scrapers/utils.ts +109 -0
- package/src/web/scrapers/vimeo.ts +133 -0
- package/src/web/scrapers/vscode-marketplace.ts +187 -0
- package/src/web/scrapers/w3c.ts +156 -0
- package/src/web/scrapers/wikidata.ts +344 -0
- package/src/web/scrapers/wikipedia.ts +84 -0
- package/src/web/scrapers/youtube.ts +325 -0
- package/src/web/search/index.ts +317 -0
- package/src/web/search/provider.ts +169 -0
- package/src/web/search/providers/anthropic.ts +343 -0
- package/src/web/search/providers/base.ts +90 -0
- package/src/web/search/providers/brave.ts +152 -0
- package/src/web/search/providers/codex.ts +593 -0
- package/src/web/search/providers/exa.ts +400 -0
- package/src/web/search/providers/gemini.ts +518 -0
- package/src/web/search/providers/jina.ts +111 -0
- package/src/web/search/providers/kagi.ts +86 -0
- package/src/web/search/providers/kimi.ts +196 -0
- package/src/web/search/providers/parallel.ts +225 -0
- package/src/web/search/providers/perplexity-auth.ts +133 -0
- package/src/web/search/providers/perplexity.ts +866 -0
- package/src/web/search/providers/searxng.ts +325 -0
- package/src/web/search/providers/synthetic.ts +114 -0
- package/src/web/search/providers/tavily.ts +176 -0
- package/src/web/search/providers/utils.ts +128 -0
- package/src/web/search/providers/zai.ts +333 -0
- package/src/web/search/render.ts +262 -0
- package/src/web/search/types.ts +462 -0
- package/src/web/search/utils.ts +17 -0
- 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
|
+
};
|