@wahack/pi-coding-agent 15.11.0
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 +10031 -0
- package/README.md +36 -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 +554 -0
- package/scripts/build-binary.ts +100 -0
- package/scripts/bundle-dist.ts +90 -0
- package/scripts/format-prompts.ts +68 -0
- package/scripts/generate-docs-index.ts +40 -0
- package/scripts/generate-template.ts +33 -0
- package/scripts/omp +42 -0
- package/scripts/omp.ts +19 -0
- package/src/async/index.ts +1 -0
- package/src/async/job-manager.ts +625 -0
- package/src/auto-thinking/classifier.ts +185 -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 +699 -0
- package/src/autoresearch/tools/init-experiment.ts +272 -0
- package/src/autoresearch/tools/log-experiment.ts +524 -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/bun-imports.d.ts +28 -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 +74 -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 +340 -0
- package/src/cli/auth-broker-cli.ts +895 -0
- package/src/cli/auth-gateway-cli.ts +611 -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 +856 -0
- package/src/cli/extension-flags.ts +48 -0
- package/src/cli/file-processor.ts +133 -0
- package/src/cli/gallery-cli.ts +230 -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/list-models.ts +194 -0
- package/src/cli/plugin-cli.ts +996 -0
- package/src/cli/read-cli.ts +57 -0
- package/src/cli/session-picker.ts +79 -0
- package/src/cli/setup-cli.ts +231 -0
- package/src/cli/shell-cli.ts +176 -0
- package/src/cli/ssh-cli.ts +179 -0
- package/src/cli/startup-cwd.ts +68 -0
- package/src/cli/stats-cli.ts +238 -0
- package/src/cli/tiny-models-cli.ts +127 -0
- package/src/cli/update-cli.ts +611 -0
- package/src/cli/usage-cli.ts +603 -0
- package/src/cli/web-search-cli.ts +132 -0
- package/src/cli/worktree-cli.ts +291 -0
- package/src/cli-commands.ts +79 -0
- package/src/cli.ts +200 -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/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/launch.ts +169 -0
- package/src/commands/plugin.ts +78 -0
- package/src/commands/read.ts +38 -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/update.ts +21 -0
- package/src/commands/usage.ts +35 -0
- package/src/commands/web-search.ts +42 -0
- package/src/commands/worktree.ts +56 -0
- package/src/commit/agentic/agent.ts +317 -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 +146 -0
- package/src/commit/agentic/tools/git-file-diff.ts +191 -0
- package/src/commit/agentic/tools/git-hunk.ts +50 -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 +144 -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 +23 -0
- package/src/commit/agentic/tools/split-commit.ts +245 -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 +105 -0
- package/src/commit/analysis/validation.ts +66 -0
- package/src/commit/changelog/detect.ts +40 -0
- package/src/commit/changelog/generate.ts +97 -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 +92 -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 +77 -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 +60 -0
- package/src/config/append-only-context-mode.ts +31 -0
- package/src/config/config-file.ts +317 -0
- package/src/config/file-lock.ts +164 -0
- package/src/config/keybindings.ts +628 -0
- package/src/config/mcp-schema.json +230 -0
- package/src/config/model-discovery.ts +554 -0
- package/src/config/model-registry.ts +2090 -0
- package/src/config/model-resolver.ts +1502 -0
- package/src/config/model-roles.ts +74 -0
- package/src/config/models-config-schema.ts +226 -0
- package/src/config/models-config.ts +129 -0
- package/src/config/prompt-templates.ts +185 -0
- package/src/config/resolve-config-value.ts +94 -0
- package/src/config/settings-schema.ts +3530 -0
- package/src/config/settings.ts +1178 -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 +515 -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 +273 -0
- package/src/debug/raw-sse.ts +292 -0
- package/src/debug/report-bundle.ts +374 -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 +54 -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 +56 -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-return-type.md +45 -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 +906 -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 +154 -0
- package/src/discovery/helpers.ts +1016 -0
- package/src/discovery/index.ts +81 -0
- package/src/discovery/mcp-json.ts +171 -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 +91 -0
- package/src/edit/hashline/block-resolver.ts +33 -0
- package/src/edit/hashline/diff.ts +290 -0
- package/src/edit/hashline/execute.ts +242 -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 +18 -0
- package/src/edit/index.ts +571 -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 +1891 -0
- package/src/edit/modes/replace.ts +1137 -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 +769 -0
- package/src/edit/streaming.ts +517 -0
- package/src/eval/__tests__/agent-bridge.test.ts +708 -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 +241 -0
- package/src/eval/__tests__/kernel-spawn.test.ts +103 -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 +207 -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 +502 -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 +246 -0
- package/src/eval/js/shared/rewrite-imports.ts +532 -0
- package/src/eval/js/shared/runtime.ts +352 -0
- package/src/eval/js/shared/types.ts +18 -0
- package/src/eval/js/tool-bridge.ts +162 -0
- package/src/eval/js/worker-core.ts +132 -0
- package/src/eval/js/worker-entry.ts +30 -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 +658 -0
- package/src/eval/py/prelude.ts +3 -0
- package/src/eval/py/runner.py +1133 -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 +419 -0
- package/src/exec/exec.ts +53 -0
- package/src/exec/non-interactive-env.ts +48 -0
- package/src/export/custom-share.ts +65 -0
- package/src/export/html/index.ts +164 -0
- package/src/export/html/template.css +1051 -0
- package/src/export/html/template.generated.ts +2 -0
- package/src/export/html/template.html +46 -0
- package/src/export/html/template.js +2271 -0
- package/src/export/html/template.macro.ts +25 -0
- package/src/export/html/vendor/highlight.min.js +1213 -0
- package/src/export/html/vendor/marked.min.js +6 -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 +489 -0
- package/src/extensibility/custom-commands/index.ts +2 -0
- package/src/extensibility/custom-commands/loader.ts +238 -0
- package/src/extensibility/custom-commands/types.ts +113 -0
- package/src/extensibility/custom-tools/index.ts +7 -0
- package/src/extensibility/custom-tools/loader.ts +269 -0
- package/src/extensibility/custom-tools/types.ts +270 -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 +572 -0
- package/src/extensibility/extensions/runner.ts +922 -0
- package/src/extensibility/extensions/types.ts +1322 -0
- package/src/extensibility/extensions/wrapper.ts +223 -0
- package/src/extensibility/hooks/index.ts +5 -0
- package/src/extensibility/hooks/loader.ts +257 -0
- package/src/extensibility/hooks/runner.ts +425 -0
- package/src/extensibility/hooks/tool-wrapper.ts +107 -0
- package/src/extensibility/hooks/types.ts +606 -0
- package/src/extensibility/legacy-pi-ai-shim.ts +24 -0
- package/src/extensibility/legacy-pi-coding-agent-shim.ts +15 -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 +682 -0
- package/src/extensibility/plugins/loader.ts +313 -0
- package/src/extensibility/plugins/manager.ts +827 -0
- package/src/extensibility/plugins/marketplace/cache.ts +136 -0
- package/src/extensibility/plugins/marketplace/fetcher.ts +317 -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/types.ts +194 -0
- package/src/extensibility/shared-events.ts +343 -0
- package/src/extensibility/skills.ts +312 -0
- package/src/extensibility/slash-commands.ts +227 -0
- package/src/extensibility/tool-proxy.ts +25 -0
- package/src/extensibility/typebox.ts +418 -0
- package/src/extensibility/utils.ts +44 -0
- package/src/goals/index.ts +3 -0
- package/src/goals/runtime.ts +528 -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 +598 -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 +488 -0
- package/src/hindsight/transcript.ts +71 -0
- package/src/index.ts +59 -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.ts +106 -0
- package/src/internal-urls/history-protocol.ts +113 -0
- package/src/internal-urls/index.ts +25 -0
- package/src/internal-urls/issue-pr-protocol.ts +584 -0
- package/src/internal-urls/json-query.ts +126 -0
- package/src/internal-urls/local-protocol.ts +287 -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 +93 -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 +292 -0
- package/src/lib/xai-http.ts +124 -0
- package/src/lsp/client.ts +1193 -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 +93 -0
- package/src/lsp/clients/swiftlint-client.ts +120 -0
- package/src/lsp/config.ts +502 -0
- package/src/lsp/defaults.json +493 -0
- package/src/lsp/diagnostics-ledger.ts +51 -0
- package/src/lsp/edits.ts +267 -0
- package/src/lsp/index.ts +2477 -0
- package/src/lsp/lspmux.ts +233 -0
- package/src/lsp/render.ts +694 -0
- package/src/lsp/startup-events.ts +13 -0
- package/src/lsp/types.ts +455 -0
- package/src/lsp/utils.ts +718 -0
- package/src/main.ts +1325 -0
- package/src/mcp/client.ts +484 -0
- package/src/mcp/config-writer.ts +225 -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 +1275 -0
- package/src/mcp/oauth-discovery.ts +442 -0
- package/src/mcp/oauth-flow.ts +442 -0
- package/src/mcp/render.ts +132 -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/timeout.ts +59 -0
- package/src/mcp/tool-bridge.ts +426 -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 +528 -0
- package/src/mcp/types.ts +423 -0
- package/src/memories/index.ts +1150 -0
- package/src/memories/storage.ts +577 -0
- package/src/memory-backend/index.ts +18 -0
- package/src/memory-backend/local-backend.ts +39 -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 +547 -0
- package/src/mnemopi/config.ts +160 -0
- package/src/mnemopi/index.ts +3 -0
- package/src/mnemopi/state.ts +584 -0
- package/src/modes/acp/acp-agent.ts +2407 -0
- package/src/modes/acp/acp-client-bridge.ts +154 -0
- package/src/modes/acp/acp-event-mapper.ts +929 -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/agent-dashboard.ts +1206 -0
- package/src/modes/components/agent-hub.ts +1071 -0
- package/src/modes/components/assistant-message.ts +307 -0
- package/src/modes/components/bash-execution.ts +220 -0
- package/src/modes/components/bordered-loader.ts +41 -0
- package/src/modes/components/branch-summary-message.ts +45 -0
- package/src/modes/components/btw-panel.ts +104 -0
- package/src/modes/components/chat-block.ts +111 -0
- package/src/modes/components/compaction-summary-message.ts +87 -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.ts +398 -0
- package/src/modes/components/custom-message.ts +63 -0
- package/src/modes/components/diff.ts +277 -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 +317 -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 +274 -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 +66 -0
- package/src/modes/components/hook-selector.ts +660 -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/mcp-add-wizard.ts +1340 -0
- package/src/modes/components/message-frame.ts +88 -0
- package/src/modes/components/model-selector.ts +1271 -0
- package/src/modes/components/oauth-selector.ts +368 -0
- package/src/modes/components/omfg-panel.ts +141 -0
- package/src/modes/components/overlay-box.ts +108 -0
- package/src/modes/components/plan-review-overlay.ts +820 -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 +722 -0
- package/src/modes/components/queue-mode-selector.ts +56 -0
- package/src/modes/components/read-tool-group.ts +670 -0
- package/src/modes/components/segment-track.ts +52 -0
- package/src/modes/components/session-selector.ts +625 -0
- package/src/modes/components/settings-defs.ts +189 -0
- package/src/modes/components/settings-selector.ts +651 -0
- package/src/modes/components/show-images-selector.ts +45 -0
- package/src/modes/components/skill-message.ts +89 -0
- package/src/modes/components/status-line/component.ts +869 -0
- package/src/modes/components/status-line/context-thresholds.ts +79 -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 +584 -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 +108 -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 +19 -0
- package/src/modes/components/todo-reminder.ts +38 -0
- package/src/modes/components/tool-execution.ts +1024 -0
- package/src/modes/components/transcript-container.ts +608 -0
- package/src/modes/components/tree-selector.ts +978 -0
- package/src/modes/components/ttsr-notification.ts +122 -0
- package/src/modes/components/user-message-selector.ts +227 -0
- package/src/modes/components/user-message.ts +66 -0
- package/src/modes/components/visual-truncate.ts +63 -0
- package/src/modes/components/welcome.ts +493 -0
- package/src/modes/controllers/btw-controller.ts +105 -0
- package/src/modes/controllers/command-controller-shared.ts +109 -0
- package/src/modes/controllers/command-controller.ts +1566 -0
- package/src/modes/controllers/event-controller.ts +1054 -0
- package/src/modes/controllers/extension-ui-controller.ts +886 -0
- package/src/modes/controllers/input-controller.ts +1073 -0
- package/src/modes/controllers/mcp-command-controller.ts +2017 -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 +1108 -0
- package/src/modes/controllers/ssh-command-controller.ts +384 -0
- package/src/modes/controllers/streaming-reveal.ts +279 -0
- package/src/modes/controllers/tan-command-controller.ts +173 -0
- package/src/modes/controllers/todo-command-controller.ts +485 -0
- package/src/modes/data/emojis.json +1 -0
- package/src/modes/emoji-autocomplete.ts +285 -0
- package/src/modes/gradient-highlight.ts +87 -0
- package/src/modes/image-references.ts +117 -0
- package/src/modes/index.ts +17 -0
- package/src/modes/interactive-mode.ts +3370 -0
- package/src/modes/internal-url-autocomplete.ts +143 -0
- package/src/modes/loop-limit.ts +140 -0
- package/src/modes/magic-keywords.ts +20 -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 +126 -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 +963 -0
- package/src/modes/rpc/rpc-mode.ts +947 -0
- package/src/modes/rpc/rpc-subagents.ts +265 -0
- package/src/modes/rpc/rpc-types.ts +458 -0
- package/src/modes/runtime-init.ts +116 -0
- package/src/modes/session-observer-registry.ts +146 -0
- package/src/modes/setup-version.ts +11 -0
- package/src/modes/setup-wizard/index.ts +99 -0
- package/src/modes/setup-wizard/lazy.ts +16 -0
- package/src/modes/setup-wizard/scenes/glyph.ts +96 -0
- package/src/modes/setup-wizard/scenes/outro.ts +35 -0
- package/src/modes/setup-wizard/scenes/providers.ts +69 -0
- package/src/modes/setup-wizard/scenes/sign-in.ts +205 -0
- package/src/modes/setup-wizard/scenes/splash.ts +201 -0
- package/src/modes/setup-wizard/scenes/theme.ts +299 -0
- package/src/modes/setup-wizard/scenes/types.ts +48 -0
- package/src/modes/setup-wizard/scenes/web-search.ts +129 -0
- package/src/modes/setup-wizard/wizard-overlay.ts +275 -0
- package/src/modes/shared.ts +47 -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 +29 -0
- package/src/modes/theme/shimmer.ts +235 -0
- package/src/modes/theme/theme-schema.json +459 -0
- package/src/modes/theme/theme.ts +2676 -0
- package/src/modes/turn-budget.ts +31 -0
- package/src/modes/types.ts +359 -0
- package/src/modes/ultrathink.ts +41 -0
- package/src/modes/utils/context-usage.ts +339 -0
- package/src/modes/utils/copy-targets.ts +360 -0
- package/src/modes/utils/hotkeys-markdown.ts +61 -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 +801 -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 +41 -0
- package/src/prompts/agents/designer.md +66 -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 +55 -0
- package/src/prompts/agents/plan.md +48 -0
- package/src/prompts/agents/reviewer.md +140 -0
- package/src/prompts/agents/task.md +16 -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/memories/consolidation.md +30 -0
- package/src/prompts/memories/read-path.md +11 -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 +10 -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/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-todo.md +13 -0
- package/src/prompts/system/empty-stop-retry.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/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/subagent-system-prompt.md +64 -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 +258 -0
- package/src/prompts/system/tiny-title-system.md +8 -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/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 +30 -0
- package/src/prompts/tools/ast-edit.md +39 -0
- package/src/prompts/tools/ast-grep.md +42 -0
- package/src/prompts/tools/async-result.md +8 -0
- package/src/prompts/tools/bash.md +46 -0
- package/src/prompts/tools/browser.md +73 -0
- package/src/prompts/tools/checkpoint.md +16 -0
- package/src/prompts/tools/debug.md +34 -0
- package/src/prompts/tools/eval.md +92 -0
- package/src/prompts/tools/find.md +36 -0
- package/src/prompts/tools/github.md +21 -0
- package/src/prompts/tools/goal.md +18 -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 +32 -0
- package/src/prompts/tools/irc.md +59 -0
- package/src/prompts/tools/job.md +19 -0
- package/src/prompts/tools/lsp-late-diagnostic.md +8 -0
- package/src/prompts/tools/lsp.md +42 -0
- package/src/prompts/tools/memory-edit.md +8 -0
- package/src/prompts/tools/patch.md +70 -0
- package/src/prompts/tools/read.md +84 -0
- package/src/prompts/tools/recall.md +5 -0
- package/src/prompts/tools/reflect.md +5 -0
- package/src/prompts/tools/render-mermaid.md +9 -0
- package/src/prompts/tools/replace.md +30 -0
- package/src/prompts/tools/resolve.md +9 -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 +24 -0
- package/src/prompts/tools/ssh.md +31 -0
- package/src/prompts/tools/task-summary.md +17 -0
- package/src/prompts/tools/task.md +88 -0
- package/src/prompts/tools/todo.md +62 -0
- package/src/prompts/tools/web-search.md +10 -0
- package/src/prompts/tools/write.md +14 -0
- package/src/registry/agent-lifecycle.ts +218 -0
- package/src/registry/agent-registry.ts +151 -0
- package/src/sdk.ts +2558 -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 +10121 -0
- package/src/session/agent-storage.ts +455 -0
- package/src/session/artifacts.ts +135 -0
- package/src/session/auth-broker-config.ts +131 -0
- package/src/session/auth-storage.ts +29 -0
- package/src/session/blob-store.ts +255 -0
- package/src/session/client-bridge.ts +85 -0
- package/src/session/history-storage.ts +348 -0
- package/src/session/indexed-session-storage.ts +430 -0
- package/src/session/messages.ts +541 -0
- package/src/session/redis-session-storage.ts +170 -0
- package/src/session/session-dump-format.ts +209 -0
- package/src/session/session-history-format.ts +246 -0
- package/src/session/session-manager.ts +3676 -0
- package/src/session/session-storage.ts +529 -0
- package/src/session/shake-types.ts +43 -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 +213 -0
- package/src/session/yield-queue.ts +173 -0
- package/src/slash-commands/acp-builtins.ts +70 -0
- package/src/slash-commands/builtin-registry.ts +1798 -0
- package/src/slash-commands/helpers/context-report.ts +39 -0
- package/src/slash-commands/helpers/format.ts +46 -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/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 +95 -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 +509 -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/stt/downloader.ts +71 -0
- package/src/stt/index.ts +3 -0
- package/src/stt/recorder.ts +351 -0
- package/src/stt/setup.ts +52 -0
- package/src/stt/stt-controller.ts +160 -0
- package/src/stt/transcribe.py +70 -0
- package/src/stt/transcriber.ts +91 -0
- package/src/stubs/natives/index.ts +814 -0
- package/src/stubs/natives/package.json +7 -0
- package/src/stubs/tui/index.ts +282 -0
- package/src/stubs/tui/package.json +7 -0
- package/src/system-prompt.ts +611 -0
- package/src/task/agents.ts +167 -0
- package/src/task/commands.ts +132 -0
- package/src/task/discovery.ts +122 -0
- package/src/task/executor.ts +2133 -0
- package/src/task/index.ts +1419 -0
- package/src/task/name-generator.ts +1577 -0
- package/src/task/omp-command.ts +26 -0
- package/src/task/output-manager.ts +88 -0
- package/src/task/parallel.ts +116 -0
- package/src/task/render.ts +1381 -0
- package/src/task/repair-args.ts +129 -0
- package/src/task/subprocess-tool-registry.ts +88 -0
- package/src/task/types.ts +336 -0
- package/src/task/worktree.ts +514 -0
- package/src/telemetry-export.ts +144 -0
- package/src/thinking.ts +167 -0
- package/src/tiny/compiled-runtime.ts +179 -0
- package/src/tiny/device.ts +111 -0
- package/src/tiny/dtype.ts +101 -0
- package/src/tiny/models.ts +242 -0
- package/src/tiny/text.ts +165 -0
- package/src/tiny/title-client.ts +543 -0
- package/src/tiny/title-protocol.ts +56 -0
- package/src/tiny/worker.ts +568 -0
- package/src/tool-discovery/mode.ts +24 -0
- package/src/tool-discovery/tool-index.ts +256 -0
- package/src/tools/approval.ts +189 -0
- package/src/tools/archive-reader.ts +721 -0
- package/src/tools/ask.ts +928 -0
- package/src/tools/ast-edit.ts +642 -0
- package/src/tools/ast-grep.ts +452 -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 +1386 -0
- package/src/tools/browser/attach.ts +175 -0
- package/src/tools/browser/launch.ts +660 -0
- package/src/tools/browser/readable.ts +112 -0
- package/src/tools/browser/registry.ts +197 -0
- package/src/tools/browser/render.ts +216 -0
- package/src/tools/browser/tab-protocol.ts +105 -0
- package/src/tools/browser/tab-supervisor.ts +628 -0
- package/src/tools/browser/tab-worker-entry.ts +21 -0
- package/src/tools/browser/tab-worker.ts +1226 -0
- package/src/tools/browser.ts +343 -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 +1067 -0
- package/src/tools/eval-backends.ts +27 -0
- package/src/tools/eval-render.ts +752 -0
- package/src/tools/eval.ts +577 -0
- package/src/tools/fetch.ts +1926 -0
- package/src/tools/file-recorder.ts +35 -0
- package/src/tools/find.ts +609 -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 +3720 -0
- package/src/tools/github-cache.ts +637 -0
- package/src/tools/grouped-file-output.ts +210 -0
- package/src/tools/image-gen.ts +1517 -0
- package/src/tools/index.ts +599 -0
- package/src/tools/inspect-image-renderer.ts +132 -0
- package/src/tools/inspect-image.ts +174 -0
- package/src/tools/irc.ts +723 -0
- package/src/tools/job.ts +557 -0
- package/src/tools/json-tree.ts +243 -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/list-limit.ts +40 -0
- package/src/tools/match-line-format.ts +20 -0
- package/src/tools/memory-edit.ts +59 -0
- package/src/tools/memory-recall.ts +100 -0
- package/src/tools/memory-reflect.ts +88 -0
- package/src/tools/memory-render.ts +202 -0
- package/src/tools/memory-retain.ts +91 -0
- package/src/tools/output-meta.ts +754 -0
- package/src/tools/output-schema-validator.ts +132 -0
- package/src/tools/path-utils.ts +1054 -0
- package/src/tools/plan-mode-guard.ts +108 -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 +206 -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 +2929 -0
- package/src/tools/render-mermaid.ts +69 -0
- package/src/tools/render-utils.ts +838 -0
- package/src/tools/renderers.ts +77 -0
- package/src/tools/report-tool-issue.ts +534 -0
- package/src/tools/resolve.ts +276 -0
- package/src/tools/review.ts +253 -0
- package/src/tools/search-tool-bm25.ts +351 -0
- package/src/tools/search.ts +1580 -0
- package/src/tools/sqlite-reader.ts +828 -0
- package/src/tools/ssh.ts +349 -0
- package/src/tools/todo.ts +982 -0
- package/src/tools/tool-errors.ts +62 -0
- package/src/tools/tool-result.ts +94 -0
- package/src/tools/tool-timeouts.ts +30 -0
- package/src/tools/tts.ts +133 -0
- package/src/tools/write.ts +1217 -0
- package/src/tools/yield.ts +269 -0
- package/src/tui/code-cell.ts +216 -0
- package/src/tui/file-list.ts +55 -0
- package/src/tui/hyperlink.ts +175 -0
- package/src/tui/index.ts +12 -0
- package/src/tui/output-block.ts +240 -0
- package/src/tui/status-line.ts +54 -0
- package/src/tui/tree-list.ts +84 -0
- package/src/tui/types.ts +15 -0
- package/src/tui/utils.ts +103 -0
- package/src/utils/block-context.ts +312 -0
- package/src/utils/changelog.ts +132 -0
- package/src/utils/clipboard.ts +193 -0
- package/src/utils/command-args.ts +76 -0
- package/src/utils/commit-message-generator.ts +151 -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 +65 -0
- package/src/utils/file-display-mode.ts +45 -0
- package/src/utils/file-mentions.ts +281 -0
- package/src/utils/git.ts +1833 -0
- package/src/utils/image-loading.ts +132 -0
- package/src/utils/image-resize.ts +309 -0
- package/src/utils/jj.ts +248 -0
- package/src/utils/lang-from-path.ts +239 -0
- package/src/utils/markit.ts +89 -0
- package/src/utils/open.ts +55 -0
- package/src/utils/session-color.ts +68 -0
- package/src/utils/shell-snapshot.ts +187 -0
- package/src/utils/sixel.ts +69 -0
- package/src/utils/title-generator.ts +373 -0
- package/src/utils/tool-choice.ts +33 -0
- package/src/utils/tools-manager.ts +363 -0
- package/src/web/kagi.ts +305 -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 +653 -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 +704 -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 +397 -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 +292 -0
- package/src/web/search/provider.ts +157 -0
- package/src/web/search/providers/anthropic.ts +318 -0
- package/src/web/search/providers/base.ts +89 -0
- package/src/web/search/providers/brave.ts +152 -0
- package/src/web/search/providers/codex.ts +591 -0
- package/src/web/search/providers/exa.ts +400 -0
- package/src/web/search/providers/gemini.ts +460 -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.ts +730 -0
- package/src/web/search/providers/searxng.ts +313 -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 +482 -0
- package/src/web/search/utils.ts +17 -0
- package/src/workspace-tree.ts +286 -0
|
@@ -0,0 +1,1566 @@
|
|
|
1
|
+
import * as fs from "node:fs/promises";
|
|
2
|
+
import * as os from "node:os";
|
|
3
|
+
import * as path from "node:path";
|
|
4
|
+
import { CompactionCancelledError, type CompactionOutcome } from "@oh-my-pi/pi-agent-core/compaction";
|
|
5
|
+
import {
|
|
6
|
+
getEnvApiKey,
|
|
7
|
+
getProviderDetails,
|
|
8
|
+
type ProviderDetails,
|
|
9
|
+
type UsageLimit,
|
|
10
|
+
type UsageReport,
|
|
11
|
+
} from "@oh-my-pi/pi-ai";
|
|
12
|
+
import { Loader, Markdown, padding, Spacer, Text, visibleWidth } from './stubs/tui/index.ts';
|
|
13
|
+
import { formatDuration, Snowflake } from "@oh-my-pi/pi-utils";
|
|
14
|
+
import { $ } from "bun";
|
|
15
|
+
import { shouldEnableAppendOnlyContext } from "../../config/append-only-context-mode";
|
|
16
|
+
import type { CompactOptions } from "../../extensibility/extensions/types";
|
|
17
|
+
import {
|
|
18
|
+
diffMentalModelContent,
|
|
19
|
+
type HindsightApi,
|
|
20
|
+
type HindsightSessionState,
|
|
21
|
+
loadHindsightConfig,
|
|
22
|
+
reloadMentalModelsForSession,
|
|
23
|
+
resolveSeedsForScope,
|
|
24
|
+
seedAlreadyExists,
|
|
25
|
+
summarizeMentalModel,
|
|
26
|
+
} from "../../hindsight";
|
|
27
|
+
import { resolveMemoryBackend } from "../../memory-backend";
|
|
28
|
+
import { BashExecutionComponent } from "../../modes/components/bash-execution";
|
|
29
|
+
import { BorderedLoader } from "../../modes/components/bordered-loader";
|
|
30
|
+
import { DynamicBorder } from "../../modes/components/dynamic-border";
|
|
31
|
+
import { EvalExecutionComponent } from "../../modes/components/eval-execution";
|
|
32
|
+
import { TranscriptBlock } from "../../modes/components/transcript-container";
|
|
33
|
+
import { getMarkdownTheme, getSymbolTheme, theme } from "../../modes/theme/theme";
|
|
34
|
+
import type { InteractiveModeContext } from "../../modes/types";
|
|
35
|
+
import { computeContextBreakdown, renderContextUsage } from "../../modes/utils/context-usage";
|
|
36
|
+
import { buildHotkeysMarkdown } from "../../modes/utils/hotkeys-markdown";
|
|
37
|
+
import { buildToolsMarkdown } from "../../modes/utils/tools-markdown";
|
|
38
|
+
import type { AsyncJobSnapshotItem } from "../../session/agent-session";
|
|
39
|
+
import type { AuthStorage } from "../../session/auth-storage";
|
|
40
|
+
import type { NewSessionOptions } from "../../session/session-manager";
|
|
41
|
+
import { formatShakeSummary, type ShakeMode, type ShakeResult } from "../../session/shake-types";
|
|
42
|
+
import { outputMeta } from "../../tools/output-meta";
|
|
43
|
+
import { resolveToCwd, stripOuterDoubleQuotes } from "../../tools/path-utils";
|
|
44
|
+
import { replaceTabs } from "../../tools/render-utils";
|
|
45
|
+
import { getChangelogPath, parseChangelog } from "../../utils/changelog";
|
|
46
|
+
import { copyToClipboard } from "../../utils/clipboard";
|
|
47
|
+
import { openPath } from "../../utils/open";
|
|
48
|
+
import { setSessionTerminalTitle } from "../../utils/title-generator";
|
|
49
|
+
|
|
50
|
+
function showMarkdownPanel(ctx: InteractiveModeContext, title: string, markdown: string): void {
|
|
51
|
+
const block = new TranscriptBlock();
|
|
52
|
+
block.addChild(new DynamicBorder());
|
|
53
|
+
block.addChild(new Text(theme.bold(theme.fg("accent", title)), 1, 0));
|
|
54
|
+
block.addChild(new Spacer(1));
|
|
55
|
+
block.addChild(new Markdown(markdown.trim(), 1, 1, getMarkdownTheme()));
|
|
56
|
+
block.addChild(new DynamicBorder());
|
|
57
|
+
ctx.present(block);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
export class CommandController {
|
|
61
|
+
constructor(private readonly ctx: InteractiveModeContext) {}
|
|
62
|
+
|
|
63
|
+
openInBrowser(urlOrPath: string): void {
|
|
64
|
+
openPath(urlOrPath);
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
async handleExportCommand(text: string): Promise<void> {
|
|
68
|
+
const parts = text.split(/\s+/);
|
|
69
|
+
const arg = parts.length > 1 ? parts[1] : undefined;
|
|
70
|
+
|
|
71
|
+
if (arg === "--copy" || arg === "clipboard" || arg === "copy") {
|
|
72
|
+
this.ctx.showWarning("Use /dump to copy the session to clipboard.");
|
|
73
|
+
return;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
try {
|
|
77
|
+
const filePath = await this.ctx.session.exportToHtml(arg);
|
|
78
|
+
this.ctx.showStatus(`Session exported to: ${filePath}`);
|
|
79
|
+
this.openInBrowser(filePath);
|
|
80
|
+
} catch (error: unknown) {
|
|
81
|
+
this.ctx.showError(`Failed to export session: ${error instanceof Error ? error.message : "Unknown error"}`);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
handleDumpCommand() {
|
|
86
|
+
try {
|
|
87
|
+
const formatted = this.ctx.session.formatSessionAsText();
|
|
88
|
+
if (!formatted) {
|
|
89
|
+
this.ctx.showError("No messages to dump yet.");
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
copyToClipboard(formatted);
|
|
93
|
+
this.ctx.showStatus("Session copied to clipboard");
|
|
94
|
+
} catch (error: unknown) {
|
|
95
|
+
this.ctx.showError(`Failed to copy session: ${error instanceof Error ? error.message : "Unknown error"}`);
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
async handleDebugTranscriptCommand(): Promise<void> {
|
|
100
|
+
try {
|
|
101
|
+
const width = Math.max(1, this.ctx.ui.terminal.columns);
|
|
102
|
+
const renderedLines = this.ctx.chatContainer.render(width).map(line => replaceTabs(Bun.stripANSI(line)));
|
|
103
|
+
const rendered = renderedLines.join("\n").trimEnd();
|
|
104
|
+
if (!rendered) {
|
|
105
|
+
this.ctx.showError("No messages to dump yet.");
|
|
106
|
+
return;
|
|
107
|
+
}
|
|
108
|
+
const tmpPath = path.join(os.tmpdir(), `${Snowflake.next()}-tmp.txt`);
|
|
109
|
+
await Bun.write(tmpPath, `${rendered}\n`);
|
|
110
|
+
this.ctx.showStatus(`Debug transcript written to:\n${tmpPath}`);
|
|
111
|
+
} catch (error: unknown) {
|
|
112
|
+
this.ctx.showError(
|
|
113
|
+
`Failed to write debug transcript: ${error instanceof Error ? error.message : "Unknown error"}`,
|
|
114
|
+
);
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
async handleShareCommand(): Promise<void> {
|
|
119
|
+
const tmpFile = path.join(os.tmpdir(), `${Snowflake.next()}.html`);
|
|
120
|
+
const cleanupTempFile = async () => {
|
|
121
|
+
try {
|
|
122
|
+
await fs.rm(tmpFile, { force: true });
|
|
123
|
+
} catch {
|
|
124
|
+
// Ignore cleanup errors
|
|
125
|
+
}
|
|
126
|
+
};
|
|
127
|
+
try {
|
|
128
|
+
await this.ctx.session.exportToHtml(tmpFile);
|
|
129
|
+
} catch (error: unknown) {
|
|
130
|
+
this.ctx.showError(`Failed to export session: ${error instanceof Error ? error.message : "Unknown error"}`);
|
|
131
|
+
return;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
try {
|
|
135
|
+
const { loadCustomShare } = await import("../../export/custom-share");
|
|
136
|
+
const customShare = await loadCustomShare();
|
|
137
|
+
if (customShare) {
|
|
138
|
+
const loader = new BorderedLoader(this.ctx.ui, theme, "Sharing...");
|
|
139
|
+
this.ctx.editorContainer.clear();
|
|
140
|
+
this.ctx.editorContainer.addChild(loader);
|
|
141
|
+
this.ctx.ui.setFocus(loader);
|
|
142
|
+
this.ctx.ui.requestRender();
|
|
143
|
+
|
|
144
|
+
const restoreEditor = async () => {
|
|
145
|
+
loader.dispose();
|
|
146
|
+
this.ctx.editorContainer.clear();
|
|
147
|
+
this.ctx.editorContainer.addChild(this.ctx.editor);
|
|
148
|
+
this.ctx.ui.setFocus(this.ctx.editor);
|
|
149
|
+
await cleanupTempFile();
|
|
150
|
+
};
|
|
151
|
+
|
|
152
|
+
try {
|
|
153
|
+
const result = await customShare.fn(tmpFile);
|
|
154
|
+
await restoreEditor();
|
|
155
|
+
|
|
156
|
+
if (typeof result === "string") {
|
|
157
|
+
this.ctx.showStatus(`Share URL: ${result}`);
|
|
158
|
+
this.openInBrowser(result);
|
|
159
|
+
} else if (result) {
|
|
160
|
+
const parts: string[] = [];
|
|
161
|
+
if (result.url) parts.push(`Share URL: ${result.url}`);
|
|
162
|
+
if (result.message) parts.push(result.message);
|
|
163
|
+
if (parts.length > 0) this.ctx.showStatus(parts.join("\n"));
|
|
164
|
+
if (result.url) this.openInBrowser(result.url);
|
|
165
|
+
} else {
|
|
166
|
+
this.ctx.showStatus("Session shared");
|
|
167
|
+
}
|
|
168
|
+
return;
|
|
169
|
+
} catch (err) {
|
|
170
|
+
await restoreEditor();
|
|
171
|
+
this.ctx.showError(`Custom share failed: ${err instanceof Error ? err.message : String(err)}`);
|
|
172
|
+
return;
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
} catch (err) {
|
|
176
|
+
await cleanupTempFile();
|
|
177
|
+
this.ctx.showError(err instanceof Error ? err.message : String(err));
|
|
178
|
+
return;
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
try {
|
|
182
|
+
const authResult = await $`gh auth status`.quiet().nothrow();
|
|
183
|
+
if (authResult.exitCode !== 0) {
|
|
184
|
+
await cleanupTempFile();
|
|
185
|
+
this.ctx.showError("GitHub CLI is not logged in. Run 'gh auth login' first.");
|
|
186
|
+
return;
|
|
187
|
+
}
|
|
188
|
+
} catch {
|
|
189
|
+
await cleanupTempFile();
|
|
190
|
+
this.ctx.showError("GitHub CLI (gh) is not installed. Install it from https://cli.github.com/");
|
|
191
|
+
return;
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
const loader = new BorderedLoader(this.ctx.ui, theme, "Creating gist...");
|
|
195
|
+
this.ctx.editorContainer.clear();
|
|
196
|
+
this.ctx.editorContainer.addChild(loader);
|
|
197
|
+
this.ctx.ui.setFocus(loader);
|
|
198
|
+
this.ctx.ui.requestRender();
|
|
199
|
+
|
|
200
|
+
const restoreEditor = async () => {
|
|
201
|
+
loader.dispose();
|
|
202
|
+
this.ctx.editorContainer.clear();
|
|
203
|
+
this.ctx.editorContainer.addChild(this.ctx.editor);
|
|
204
|
+
this.ctx.ui.setFocus(this.ctx.editor);
|
|
205
|
+
await cleanupTempFile();
|
|
206
|
+
};
|
|
207
|
+
|
|
208
|
+
loader.onAbort = () => {
|
|
209
|
+
void restoreEditor();
|
|
210
|
+
this.ctx.showStatus("Share cancelled");
|
|
211
|
+
};
|
|
212
|
+
|
|
213
|
+
try {
|
|
214
|
+
const result = await $`gh gist create --public=false ${tmpFile}`.quiet().nothrow();
|
|
215
|
+
if (loader.signal.aborted) return;
|
|
216
|
+
|
|
217
|
+
await restoreEditor();
|
|
218
|
+
|
|
219
|
+
if (result.exitCode !== 0) {
|
|
220
|
+
const errorMsg = result.stderr.toString("utf-8").trim() || "Unknown error";
|
|
221
|
+
this.ctx.showError(`Failed to create gist: ${errorMsg}`);
|
|
222
|
+
return;
|
|
223
|
+
}
|
|
224
|
+
|
|
225
|
+
const gistUrl = result.stdout.toString("utf-8").trim();
|
|
226
|
+
const gistId = gistUrl.split("/").pop();
|
|
227
|
+
if (!gistId) {
|
|
228
|
+
this.ctx.showError("Failed to parse gist ID from gh output");
|
|
229
|
+
return;
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
const previewUrl = `https://gistpreview.github.io/?${gistId}`;
|
|
233
|
+
this.ctx.showStatus(`Share URL: ${previewUrl}\nGist: ${gistUrl}`);
|
|
234
|
+
this.openInBrowser(previewUrl);
|
|
235
|
+
} catch (error: unknown) {
|
|
236
|
+
if (!loader.signal.aborted) {
|
|
237
|
+
await restoreEditor();
|
|
238
|
+
this.ctx.showError(`Failed to create gist: ${error instanceof Error ? error.message : "Unknown error"}`);
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
async handleSessionCommand(): Promise<void> {
|
|
244
|
+
const stats = this.ctx.session.getSessionStats();
|
|
245
|
+
const premiumRequests =
|
|
246
|
+
"premiumRequests" in stats && typeof stats.premiumRequests === "number"
|
|
247
|
+
? stats.premiumRequests
|
|
248
|
+
: this.ctx.session.sessionManager.getUsageStatistics().premiumRequests;
|
|
249
|
+
const normalizedPremiumRequests = Math.round((premiumRequests + Number.EPSILON) * 100) / 100;
|
|
250
|
+
|
|
251
|
+
let info = `${theme.bold("Session Info")}\n\n`;
|
|
252
|
+
info += `${theme.fg("dim", "File:")} ${stats.sessionFile ?? "In-memory"}\n`;
|
|
253
|
+
info += `${theme.fg("dim", "ID:")} ${stats.sessionId}\n\n`;
|
|
254
|
+
info += `\n${theme.bold("Provider")}\n`;
|
|
255
|
+
const model = this.ctx.session.model;
|
|
256
|
+
if (!model) {
|
|
257
|
+
info += `${theme.fg("dim", "No model selected")}\n`;
|
|
258
|
+
} else {
|
|
259
|
+
const authMode = resolveProviderAuthMode(this.ctx.session.modelRegistry.authStorage, model.provider);
|
|
260
|
+
const openaiWebsocketSetting = this.ctx.settings.get("providers.openaiWebsockets") ?? "auto";
|
|
261
|
+
const preferOpenAICodexWebsockets =
|
|
262
|
+
openaiWebsocketSetting === "on" ? true : openaiWebsocketSetting === "off" ? false : undefined;
|
|
263
|
+
const credentialSource = this.ctx.session.modelRegistry.authStorage.describeCredentialSource(
|
|
264
|
+
model.provider,
|
|
265
|
+
stats.sessionId,
|
|
266
|
+
);
|
|
267
|
+
const providerDetails = getProviderDetails({
|
|
268
|
+
model,
|
|
269
|
+
sessionId: stats.sessionId,
|
|
270
|
+
authMode,
|
|
271
|
+
credentialSource,
|
|
272
|
+
preferWebsockets: preferOpenAICodexWebsockets,
|
|
273
|
+
providerSessionState: this.ctx.session.providerSessionState,
|
|
274
|
+
});
|
|
275
|
+
info += renderProviderSection(providerDetails, theme);
|
|
276
|
+
}
|
|
277
|
+
info += `\n`;
|
|
278
|
+
info += `${theme.bold("Messages")}\n`;
|
|
279
|
+
info += `${theme.fg("dim", "User:")} ${stats.userMessages}\n`;
|
|
280
|
+
info += `${theme.fg("dim", "Assistant:")} ${stats.assistantMessages}\n`;
|
|
281
|
+
info += `${theme.fg("dim", "Tool Calls:")} ${stats.toolCalls}\n`;
|
|
282
|
+
info += `${theme.fg("dim", "Tool Results:")} ${stats.toolResults}\n`;
|
|
283
|
+
info += `${theme.fg("dim", "Total:")} ${stats.totalMessages}\n\n`;
|
|
284
|
+
// Append-only context
|
|
285
|
+
{
|
|
286
|
+
const setting = this.ctx.settings.get("provider.appendOnlyContext") ?? "auto";
|
|
287
|
+
const model = this.ctx.session.model;
|
|
288
|
+
const mode = shouldEnableAppendOnlyContext(setting, model);
|
|
289
|
+
const activeLabel = mode ? theme.fg("success", "active") : theme.fg("dim", "inactive");
|
|
290
|
+
const settingLabel = setting === "auto" ? `${setting} (${model?.provider ?? "?"})` : setting;
|
|
291
|
+
info += `${theme.fg("dim", "Append-Only:")} ${activeLabel} (setting: ${settingLabel})\n`;
|
|
292
|
+
}
|
|
293
|
+
info += `${theme.bold("Tokens")}\n`;
|
|
294
|
+
info += `${theme.fg("dim", "Input:")} ${stats.tokens.input.toLocaleString()}\n`;
|
|
295
|
+
info += `${theme.fg("dim", "Output:")} ${stats.tokens.output.toLocaleString()}\n`;
|
|
296
|
+
if (stats.tokens.cacheRead > 0) {
|
|
297
|
+
info += `${theme.fg("dim", "Cache Read:")} ${stats.tokens.cacheRead.toLocaleString()}\n`;
|
|
298
|
+
}
|
|
299
|
+
if (stats.tokens.cacheWrite > 0) {
|
|
300
|
+
info += `${theme.fg("dim", "Cache Write:")} ${stats.tokens.cacheWrite.toLocaleString()}\n`;
|
|
301
|
+
}
|
|
302
|
+
info += `${theme.fg("dim", "Total:")} ${stats.tokens.total.toLocaleString()}\n`;
|
|
303
|
+
|
|
304
|
+
if (stats.cost > 0 || normalizedPremiumRequests > 0) {
|
|
305
|
+
info += `\n${theme.bold("Cost")}\n`;
|
|
306
|
+
if (stats.cost > 0) {
|
|
307
|
+
info += `${theme.fg("dim", "Total:")} ${stats.cost.toFixed(4)}\n`;
|
|
308
|
+
}
|
|
309
|
+
if (normalizedPremiumRequests > 0) {
|
|
310
|
+
info += `${theme.fg("dim", "Premium Requests:")} ${normalizedPremiumRequests.toLocaleString()}\n`;
|
|
311
|
+
}
|
|
312
|
+
}
|
|
313
|
+
|
|
314
|
+
if (this.ctx.lspServers && this.ctx.lspServers.length > 0) {
|
|
315
|
+
info += `\n${theme.bold("LSP Servers")}\n`;
|
|
316
|
+
for (const server of this.ctx.lspServers) {
|
|
317
|
+
const statusColor =
|
|
318
|
+
server.status === "ready"
|
|
319
|
+
? "success"
|
|
320
|
+
: server.status === "available"
|
|
321
|
+
? "dim"
|
|
322
|
+
: server.status === "connecting"
|
|
323
|
+
? "warning"
|
|
324
|
+
: "error";
|
|
325
|
+
const statusText =
|
|
326
|
+
server.status === "error" && server.error ? `${server.status}: ${server.error}` : server.status;
|
|
327
|
+
info += `${theme.fg("dim", `${server.name}:`)} ${theme.fg(statusColor, statusText)} ${theme.fg("dim", `(${server.fileTypes.join(", ")})`)}\n`;
|
|
328
|
+
}
|
|
329
|
+
}
|
|
330
|
+
|
|
331
|
+
if (this.ctx.mcpManager) {
|
|
332
|
+
const mcpServers = this.ctx.mcpManager.getConnectedServers();
|
|
333
|
+
info += `\n${theme.bold("MCP Servers")}\n`;
|
|
334
|
+
if (mcpServers.length === 0) {
|
|
335
|
+
info += `${theme.fg("dim", "None connected")}\n`;
|
|
336
|
+
} else {
|
|
337
|
+
for (const name of mcpServers) {
|
|
338
|
+
const conn = this.ctx.mcpManager.getConnection(name);
|
|
339
|
+
const toolCount = conn?.tools?.length ?? 0;
|
|
340
|
+
info += `${theme.fg("dim", `${name}:`)} ${theme.fg("success", "connected")} ${theme.fg("dim", `(${toolCount} tools)`)}\n`;
|
|
341
|
+
}
|
|
342
|
+
}
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
this.ctx.present([new Spacer(1), new Text(info, 1, 0)]);
|
|
346
|
+
}
|
|
347
|
+
|
|
348
|
+
async handleJobsCommand(): Promise<void> {
|
|
349
|
+
const snapshot = this.ctx.session.getAsyncJobSnapshot({ recentLimit: 5 });
|
|
350
|
+
if (!snapshot) {
|
|
351
|
+
this.ctx.showWarning("Async background jobs are unavailable in this session.");
|
|
352
|
+
return;
|
|
353
|
+
}
|
|
354
|
+
|
|
355
|
+
const now = Date.now();
|
|
356
|
+
const lineWidth = Math.max(24, (this.ctx.ui.terminal.columns ?? 100) - 24);
|
|
357
|
+
let info = `${theme.bold("Background Jobs")}\n\n`;
|
|
358
|
+
info += `${theme.fg("dim", "Running:")} ${snapshot.running.length}\n`;
|
|
359
|
+
|
|
360
|
+
if (snapshot.running.length === 0 && snapshot.recent.length === 0) {
|
|
361
|
+
info += `\n${theme.fg("dim", "No async jobs yet.")}\n`;
|
|
362
|
+
this.ctx.present([new Spacer(1), new Text(info, 1, 0)]);
|
|
363
|
+
return;
|
|
364
|
+
}
|
|
365
|
+
|
|
366
|
+
if (snapshot.running.length > 0) {
|
|
367
|
+
info += `\n${theme.bold("Running Jobs")}\n`;
|
|
368
|
+
for (const job of snapshot.running) {
|
|
369
|
+
info += `${renderJobLine(job, now)}\n`;
|
|
370
|
+
info += ` ${theme.fg("dim", truncateJobLabel(job.label, lineWidth))}\n`;
|
|
371
|
+
}
|
|
372
|
+
}
|
|
373
|
+
|
|
374
|
+
if (snapshot.recent.length > 0) {
|
|
375
|
+
info += `\n${theme.bold("Recent Jobs")}\n`;
|
|
376
|
+
for (const job of snapshot.recent) {
|
|
377
|
+
info += `${renderJobLine(job, now)}\n`;
|
|
378
|
+
info += ` ${theme.fg("dim", truncateJobLabel(job.label, lineWidth))}\n`;
|
|
379
|
+
}
|
|
380
|
+
}
|
|
381
|
+
|
|
382
|
+
this.ctx.present([new Spacer(1), new Text(info.trimEnd(), 1, 0)]);
|
|
383
|
+
}
|
|
384
|
+
|
|
385
|
+
async handleUsageCommand(reports?: UsageReport[] | null): Promise<void> {
|
|
386
|
+
let usageReports = reports ?? null;
|
|
387
|
+
if (!usageReports) {
|
|
388
|
+
const provider = this.ctx.session as { fetchUsageReports?: () => Promise<UsageReport[] | null> };
|
|
389
|
+
if (!provider.fetchUsageReports) {
|
|
390
|
+
this.ctx.showWarning("Usage reporting is not configured for this session.");
|
|
391
|
+
return;
|
|
392
|
+
}
|
|
393
|
+
try {
|
|
394
|
+
usageReports = await provider.fetchUsageReports();
|
|
395
|
+
} catch (error) {
|
|
396
|
+
this.ctx.showError(`Failed to fetch usage data: ${error instanceof Error ? error.message : String(error)}`);
|
|
397
|
+
return;
|
|
398
|
+
}
|
|
399
|
+
}
|
|
400
|
+
|
|
401
|
+
if (!usageReports || usageReports.length === 0) {
|
|
402
|
+
this.ctx.showWarning("No usage data available.");
|
|
403
|
+
return;
|
|
404
|
+
}
|
|
405
|
+
|
|
406
|
+
const availableWidth = Math.max(40, (this.ctx.ui.terminal.columns ?? 100) - 2);
|
|
407
|
+
const output = renderUsageReports(usageReports, theme, Date.now(), availableWidth);
|
|
408
|
+
this.ctx.present([new Spacer(1), new Text(output, 1, 0)]);
|
|
409
|
+
}
|
|
410
|
+
|
|
411
|
+
async handleChangelogCommand(showFull = false): Promise<void> {
|
|
412
|
+
const changelogPath = getChangelogPath();
|
|
413
|
+
const allEntries = await parseChangelog(changelogPath);
|
|
414
|
+
// Default to showing only the latest 3 versions unless --full is specified
|
|
415
|
+
// allEntries comes from parseChangelog with newest first, reverse to show oldest->newest
|
|
416
|
+
const entriesToShow = showFull ? allEntries : allEntries.slice(0, 3);
|
|
417
|
+
const changelogMarkdown =
|
|
418
|
+
entriesToShow.length > 0
|
|
419
|
+
? [...entriesToShow]
|
|
420
|
+
.reverse()
|
|
421
|
+
.map(e => e.content)
|
|
422
|
+
.join("\n\n")
|
|
423
|
+
: "No changelog entries found.";
|
|
424
|
+
const title = showFull ? "Full Changelog" : "Recent Changes";
|
|
425
|
+
const hint = showFull
|
|
426
|
+
? ""
|
|
427
|
+
: `\n\n${theme.fg("dim", "Use")} ${theme.bold("/changelog full")} ${theme.fg("dim", "to view the complete changelog.")}`;
|
|
428
|
+
|
|
429
|
+
const block = new TranscriptBlock();
|
|
430
|
+
block.addChild(new DynamicBorder());
|
|
431
|
+
block.addChild(new Text(theme.bold(theme.fg("accent", title)), 1, 0));
|
|
432
|
+
block.addChild(new Spacer(1));
|
|
433
|
+
block.addChild(new Markdown(changelogMarkdown + hint, 1, 1, getMarkdownTheme()));
|
|
434
|
+
block.addChild(new DynamicBorder());
|
|
435
|
+
this.ctx.present(block);
|
|
436
|
+
}
|
|
437
|
+
|
|
438
|
+
handleHotkeysCommand(): void {
|
|
439
|
+
const hotkeys = buildHotkeysMarkdown({ keybindings: this.ctx.keybindings });
|
|
440
|
+
showMarkdownPanel(this.ctx, "Keyboard Shortcuts", hotkeys);
|
|
441
|
+
}
|
|
442
|
+
|
|
443
|
+
handleToolsCommand(): void {
|
|
444
|
+
const tools = buildToolsMarkdown({ tools: this.ctx.session.agent.state.tools });
|
|
445
|
+
showMarkdownPanel(this.ctx, "Available Tools", tools);
|
|
446
|
+
}
|
|
447
|
+
|
|
448
|
+
handleContextCommand(): void {
|
|
449
|
+
const breakdown = computeContextBreakdown(this.ctx.session);
|
|
450
|
+
if (breakdown.contextWindow <= 0) {
|
|
451
|
+
this.ctx.showWarning("Context usage is unavailable: no model is selected for this session.");
|
|
452
|
+
return;
|
|
453
|
+
}
|
|
454
|
+
const output = renderContextUsage(breakdown, theme);
|
|
455
|
+
const block = new TranscriptBlock();
|
|
456
|
+
block.addChild(new DynamicBorder());
|
|
457
|
+
block.addChild(new Text(theme.bold(theme.fg("accent", "Context Usage")), 1, 0));
|
|
458
|
+
block.addChild(new Spacer(1));
|
|
459
|
+
block.addChild(new Text(output, 1, 0));
|
|
460
|
+
block.addChild(new DynamicBorder());
|
|
461
|
+
this.ctx.present(block);
|
|
462
|
+
}
|
|
463
|
+
|
|
464
|
+
async handleMemoryCommand(text: string): Promise<void> {
|
|
465
|
+
const argumentText = text.slice(7).trim();
|
|
466
|
+
const action = argumentText.split(/\s+/, 1)[0]?.toLowerCase() || "view";
|
|
467
|
+
const agentDir = this.ctx.settings.getAgentDir();
|
|
468
|
+
const backend = await resolveMemoryBackend(this.ctx.settings);
|
|
469
|
+
|
|
470
|
+
if (action === "view") {
|
|
471
|
+
const payload = await backend.buildDeveloperInstructions(agentDir, this.ctx.settings, this.ctx.session);
|
|
472
|
+
if (!payload) {
|
|
473
|
+
this.ctx.showWarning("Memory payload is empty (memory backend off, disabled, or no memory available).");
|
|
474
|
+
return;
|
|
475
|
+
}
|
|
476
|
+
const block = new TranscriptBlock();
|
|
477
|
+
block.addChild(new DynamicBorder());
|
|
478
|
+
block.addChild(new Text(theme.bold(theme.fg("accent", "Memory Injection Payload")), 1, 0));
|
|
479
|
+
block.addChild(new Spacer(1));
|
|
480
|
+
block.addChild(new Markdown(payload, 1, 1, getMarkdownTheme()));
|
|
481
|
+
block.addChild(new DynamicBorder());
|
|
482
|
+
this.ctx.present(block);
|
|
483
|
+
return;
|
|
484
|
+
}
|
|
485
|
+
|
|
486
|
+
if (action === "reset" || action === "clear") {
|
|
487
|
+
try {
|
|
488
|
+
await backend.clear(agentDir, this.ctx.sessionManager.getCwd(), this.ctx.session);
|
|
489
|
+
await this.ctx.session.refreshBaseSystemPrompt();
|
|
490
|
+
this.ctx.showStatus("Memory data cleared and system prompt refreshed.");
|
|
491
|
+
} catch (error) {
|
|
492
|
+
this.ctx.showError(`Memory clear failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
493
|
+
}
|
|
494
|
+
return;
|
|
495
|
+
}
|
|
496
|
+
|
|
497
|
+
if (action === "enqueue" || action === "rebuild") {
|
|
498
|
+
try {
|
|
499
|
+
await backend.enqueue(agentDir, this.ctx.sessionManager.getCwd(), this.ctx.session);
|
|
500
|
+
this.ctx.showStatus("Memory consolidation enqueued.");
|
|
501
|
+
} catch (error) {
|
|
502
|
+
this.ctx.showError(`Memory enqueue failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
503
|
+
}
|
|
504
|
+
return;
|
|
505
|
+
}
|
|
506
|
+
|
|
507
|
+
if (action === "stats" || action === "diagnose") {
|
|
508
|
+
const hook = action === "stats" ? backend.stats : backend.diagnose;
|
|
509
|
+
try {
|
|
510
|
+
const payload = await hook?.(agentDir, this.ctx.sessionManager.getCwd(), this.ctx.session);
|
|
511
|
+
if (!payload) {
|
|
512
|
+
this.ctx.showWarning(`Memory ${action} is not available for the ${backend.id} backend.`);
|
|
513
|
+
return;
|
|
514
|
+
}
|
|
515
|
+
showMarkdownPanel(this.ctx, `Memory ${action === "stats" ? "Stats" : "Diagnostics"}`, payload);
|
|
516
|
+
} catch (error) {
|
|
517
|
+
this.ctx.showError(`Memory ${action} failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
518
|
+
}
|
|
519
|
+
return;
|
|
520
|
+
}
|
|
521
|
+
|
|
522
|
+
if (action === "mm") {
|
|
523
|
+
await this.#handleMentalModelsSubcommand(argumentText);
|
|
524
|
+
return;
|
|
525
|
+
}
|
|
526
|
+
|
|
527
|
+
this.ctx.showError("Usage: /memory <view|stats|diagnose|clear|reset|enqueue|rebuild|mm ...>");
|
|
528
|
+
}
|
|
529
|
+
|
|
530
|
+
async #handleMentalModelsSubcommand(argumentText: string): Promise<void> {
|
|
531
|
+
// Parse: "mm <verb> [arg]"
|
|
532
|
+
const parts = argumentText.split(/\s+/).slice(1);
|
|
533
|
+
const verb = parts[0]?.toLowerCase() ?? "list";
|
|
534
|
+
const arg = parts[1];
|
|
535
|
+
|
|
536
|
+
const state = this.ctx.session.getHindsightSessionState();
|
|
537
|
+
const primary = state && !state.aliasOf ? state : undefined;
|
|
538
|
+
if (!primary) {
|
|
539
|
+
this.ctx.showError("Hindsight backend is not active for this session.");
|
|
540
|
+
return;
|
|
541
|
+
}
|
|
542
|
+
if (!primary.config.mentalModelsEnabled) {
|
|
543
|
+
this.ctx.showError("Mental models are disabled (hindsight.mentalModelsEnabled = false).");
|
|
544
|
+
return;
|
|
545
|
+
}
|
|
546
|
+
|
|
547
|
+
switch (verb) {
|
|
548
|
+
case "list":
|
|
549
|
+
await this.#mmList(primary);
|
|
550
|
+
return;
|
|
551
|
+
case "show":
|
|
552
|
+
if (!arg) return this.ctx.showError("Usage: /memory mm show <id>");
|
|
553
|
+
await this.#mmShow(primary, arg);
|
|
554
|
+
return;
|
|
555
|
+
case "refresh":
|
|
556
|
+
await this.#mmRefresh(primary, arg);
|
|
557
|
+
return;
|
|
558
|
+
case "history":
|
|
559
|
+
if (!arg) return this.ctx.showError("Usage: /memory mm history <id>");
|
|
560
|
+
await this.#mmHistory(primary, arg);
|
|
561
|
+
return;
|
|
562
|
+
case "seed":
|
|
563
|
+
await this.#mmSeed(primary);
|
|
564
|
+
return;
|
|
565
|
+
case "reload":
|
|
566
|
+
await this.#mmReload(primary);
|
|
567
|
+
return;
|
|
568
|
+
case "delete":
|
|
569
|
+
case "remove":
|
|
570
|
+
if (!arg) return this.ctx.showError("Usage: /memory mm delete <id>");
|
|
571
|
+
await this.#mmDelete(primary, arg);
|
|
572
|
+
return;
|
|
573
|
+
default:
|
|
574
|
+
this.ctx.showError("Usage: /memory mm <list|show|refresh|history|seed|reload|delete>");
|
|
575
|
+
}
|
|
576
|
+
}
|
|
577
|
+
|
|
578
|
+
async #mmList(state: HindsightSessionState): Promise<void> {
|
|
579
|
+
const client: HindsightApi = state.client;
|
|
580
|
+
try {
|
|
581
|
+
const response = await client.listMentalModels(state.bankId, { detail: "metadata" });
|
|
582
|
+
const items = response.items ?? [];
|
|
583
|
+
if (items.length === 0) {
|
|
584
|
+
this.ctx.showStatus(`No mental models on bank ${state.bankId}.`);
|
|
585
|
+
return;
|
|
586
|
+
}
|
|
587
|
+
const lines = items
|
|
588
|
+
.slice()
|
|
589
|
+
.sort((a, b) => a.id.localeCompare(b.id))
|
|
590
|
+
.map(summarizeMentalModel);
|
|
591
|
+
showMarkdownPanel(this.ctx, `Mental Models — ${state.bankId}`, lines.join("\n"));
|
|
592
|
+
} catch (error) {
|
|
593
|
+
this.ctx.showError(`mm list failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
594
|
+
}
|
|
595
|
+
}
|
|
596
|
+
|
|
597
|
+
async #mmShow(state: HindsightSessionState, id: string): Promise<void> {
|
|
598
|
+
try {
|
|
599
|
+
const model = await state.client.getMentalModel(state.bankId, id, { detail: "content" });
|
|
600
|
+
if (!model) {
|
|
601
|
+
this.ctx.showError(`Mental model not found: ${id}`);
|
|
602
|
+
return;
|
|
603
|
+
}
|
|
604
|
+
const tags = model.tags && model.tags.length > 0 ? `\n_tags: ${model.tags.join(", ")}_` : "";
|
|
605
|
+
const refreshed = model.last_refreshed_at ? `\n_last refreshed: ${model.last_refreshed_at}_` : "";
|
|
606
|
+
const sourceQuery = model.source_query ? `\n\n**Source query:** ${model.source_query}` : "";
|
|
607
|
+
const content = (model.content ?? "_(empty — background reflect may still be running)_").trim();
|
|
608
|
+
showMarkdownPanel(
|
|
609
|
+
this.ctx,
|
|
610
|
+
model.name,
|
|
611
|
+
`**id:** \`${model.id}\`${tags}${refreshed}${sourceQuery}\n\n${content}`,
|
|
612
|
+
);
|
|
613
|
+
} catch (error) {
|
|
614
|
+
this.ctx.showError(`mm show failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
615
|
+
}
|
|
616
|
+
}
|
|
617
|
+
|
|
618
|
+
async #mmRefresh(state: HindsightSessionState, id: string | undefined): Promise<void> {
|
|
619
|
+
try {
|
|
620
|
+
if (id) {
|
|
621
|
+
// Single-model refresh is explicit operator intent: bypass the
|
|
622
|
+
// auto-refresh filter so curated/manual models can still be
|
|
623
|
+
// refreshed on demand.
|
|
624
|
+
await state.client.refreshMentalModel(state.bankId, id);
|
|
625
|
+
this.ctx.showStatus(`Refresh queued for mental model ${id}.`);
|
|
626
|
+
} else {
|
|
627
|
+
// Bulk refresh: only touch models that opted into automatic
|
|
628
|
+
// refresh via `trigger.refresh_after_consolidation`. Curated
|
|
629
|
+
// models are reviewed before publishing and must not be
|
|
630
|
+
// silently regenerated by a bank-wide refresh sweep. Reading
|
|
631
|
+
// `detail: "content"` here is required because the trigger
|
|
632
|
+
// field is excluded from `detail: "metadata"`.
|
|
633
|
+
const list = await state.client.listMentalModels(state.bankId, { detail: "content" });
|
|
634
|
+
const items = list.items ?? [];
|
|
635
|
+
if (items.length === 0) {
|
|
636
|
+
this.ctx.showStatus(`No mental models on bank ${state.bankId}.`);
|
|
637
|
+
return;
|
|
638
|
+
}
|
|
639
|
+
const targets = items.filter(m => m.trigger?.refresh_after_consolidation === true);
|
|
640
|
+
const skipped = items.length - targets.length;
|
|
641
|
+
if (targets.length === 0) {
|
|
642
|
+
this.ctx.showStatus(
|
|
643
|
+
`No mental models opted into auto-refresh; ${skipped} curated model(s) left untouched. Pass an explicit id to refresh one of them.`,
|
|
644
|
+
);
|
|
645
|
+
return;
|
|
646
|
+
}
|
|
647
|
+
let queued = 0;
|
|
648
|
+
for (const item of targets) {
|
|
649
|
+
try {
|
|
650
|
+
await state.client.refreshMentalModel(state.bankId, item.id);
|
|
651
|
+
queued++;
|
|
652
|
+
} catch (error) {
|
|
653
|
+
this.ctx.showWarning(
|
|
654
|
+
`Refresh failed for ${item.id}: ${error instanceof Error ? error.message : String(error)}`,
|
|
655
|
+
);
|
|
656
|
+
}
|
|
657
|
+
}
|
|
658
|
+
const skippedSuffix = skipped > 0 ? `; skipped ${skipped} curated model(s)` : "";
|
|
659
|
+
this.ctx.showStatus(
|
|
660
|
+
`Refresh queued for ${queued}/${targets.length} auto-refresh model(s)${skippedSuffix}.`,
|
|
661
|
+
);
|
|
662
|
+
}
|
|
663
|
+
// Reload the cache after a brief grace so the new content (if the refresh
|
|
664
|
+
// completes synchronously on the server) flows into the system prompt.
|
|
665
|
+
await Bun.sleep(500);
|
|
666
|
+
await reloadMentalModelsForSession(state.session);
|
|
667
|
+
} catch (error) {
|
|
668
|
+
this.ctx.showError(`mm refresh failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
669
|
+
}
|
|
670
|
+
}
|
|
671
|
+
|
|
672
|
+
async #mmHistory(state: HindsightSessionState, id: string): Promise<void> {
|
|
673
|
+
try {
|
|
674
|
+
const [model, history] = await Promise.all([
|
|
675
|
+
state.client.getMentalModel(state.bankId, id, { detail: "content" }),
|
|
676
|
+
state.client.getMentalModelHistory(state.bankId, id),
|
|
677
|
+
]);
|
|
678
|
+
if (!model) {
|
|
679
|
+
this.ctx.showError(`Mental model not found: ${id}`);
|
|
680
|
+
return;
|
|
681
|
+
}
|
|
682
|
+
if (history.length === 0) {
|
|
683
|
+
this.ctx.showStatus(`No history recorded for ${id}.`);
|
|
684
|
+
return;
|
|
685
|
+
}
|
|
686
|
+
// History is most-recent first. Each entry stores the content BEFORE that
|
|
687
|
+
// change. To diff "what changed at entry N", compare entry N's
|
|
688
|
+
// previous_content (= state before that change) with entry N-1's
|
|
689
|
+
// previous_content (= state after that change, which was state before
|
|
690
|
+
// the next change). For the most recent change, compare against the
|
|
691
|
+
// model's CURRENT content.
|
|
692
|
+
const sections: string[] = [];
|
|
693
|
+
for (let i = 0; i < history.length; i++) {
|
|
694
|
+
const before = history[i].previous_content ?? "";
|
|
695
|
+
const after = i === 0 ? (model.content ?? "") : (history[i - 1].previous_content ?? "");
|
|
696
|
+
const diff = diffMentalModelContent(before, after);
|
|
697
|
+
sections.push(`### ${history[i].changed_at}\n\n\`\`\`diff\n${diff}\n\`\`\``);
|
|
698
|
+
}
|
|
699
|
+
showMarkdownPanel(this.ctx, `History — ${model.name}`, sections.join("\n\n"));
|
|
700
|
+
} catch (error) {
|
|
701
|
+
this.ctx.showError(`mm history failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
702
|
+
}
|
|
703
|
+
}
|
|
704
|
+
|
|
705
|
+
async #mmSeed(state: HindsightSessionState): Promise<void> {
|
|
706
|
+
try {
|
|
707
|
+
const config = loadHindsightConfig(this.ctx.settings);
|
|
708
|
+
const seeds = resolveSeedsForScope(
|
|
709
|
+
{
|
|
710
|
+
bankId: state.bankId,
|
|
711
|
+
retainTags: state.retainTags,
|
|
712
|
+
recallTags: state.recallTags,
|
|
713
|
+
recallTagsMatch: state.recallTagsMatch,
|
|
714
|
+
},
|
|
715
|
+
config.scoping,
|
|
716
|
+
);
|
|
717
|
+
if (seeds.length === 0) {
|
|
718
|
+
this.ctx.showStatus(`No built-in seeds apply to scoping=${config.scoping}.`);
|
|
719
|
+
return;
|
|
720
|
+
}
|
|
721
|
+
const list = await state.client.listMentalModels(state.bankId, { detail: "metadata" });
|
|
722
|
+
const existing = list.items ?? [];
|
|
723
|
+
let created = 0;
|
|
724
|
+
let skipped = 0;
|
|
725
|
+
for (const seed of seeds) {
|
|
726
|
+
if (seedAlreadyExists(seed, existing)) {
|
|
727
|
+
skipped++;
|
|
728
|
+
continue;
|
|
729
|
+
}
|
|
730
|
+
try {
|
|
731
|
+
await state.client.createMentalModel(state.bankId, seed.name, seed.sourceQuery, {
|
|
732
|
+
id: seed.id,
|
|
733
|
+
tags: seed.tags.length > 0 ? seed.tags : undefined,
|
|
734
|
+
maxTokens: seed.maxTokens,
|
|
735
|
+
trigger: seed.trigger,
|
|
736
|
+
});
|
|
737
|
+
created++;
|
|
738
|
+
} catch (error) {
|
|
739
|
+
this.ctx.showWarning(
|
|
740
|
+
`Seed failed for ${seed.id}: ${error instanceof Error ? error.message : String(error)}`,
|
|
741
|
+
);
|
|
742
|
+
}
|
|
743
|
+
}
|
|
744
|
+
this.ctx.showStatus(`Seeded ${created} new mental model(s); ${skipped} already present.`);
|
|
745
|
+
} catch (error) {
|
|
746
|
+
this.ctx.showError(`mm seed failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
747
|
+
}
|
|
748
|
+
}
|
|
749
|
+
|
|
750
|
+
async #mmReload(state: HindsightSessionState): Promise<void> {
|
|
751
|
+
const ok = await reloadMentalModelsForSession(state.session);
|
|
752
|
+
if (ok) {
|
|
753
|
+
this.ctx.showStatus("Mental-model cache reloaded.");
|
|
754
|
+
} else {
|
|
755
|
+
this.ctx.showError("Reload failed (Hindsight backend not active or mental models disabled).");
|
|
756
|
+
}
|
|
757
|
+
}
|
|
758
|
+
|
|
759
|
+
async #mmDelete(state: HindsightSessionState, id: string): Promise<void> {
|
|
760
|
+
try {
|
|
761
|
+
const removed = await state.client.deleteMentalModel(state.bankId, id);
|
|
762
|
+
if (!removed) {
|
|
763
|
+
this.ctx.showError(`Mental model not found: ${id}`);
|
|
764
|
+
return;
|
|
765
|
+
}
|
|
766
|
+
// Drop the cached snippet so the closing tag does not silently keep
|
|
767
|
+
// stale content in the system prompt until the next agent_end TTL.
|
|
768
|
+
await reloadMentalModelsForSession(state.session);
|
|
769
|
+
this.ctx.showStatus(`Deleted mental model ${id} from bank ${state.bankId}.`);
|
|
770
|
+
} catch (error) {
|
|
771
|
+
this.ctx.showError(`mm delete failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
772
|
+
}
|
|
773
|
+
}
|
|
774
|
+
|
|
775
|
+
async #runNewSessionFlow(options?: NewSessionOptions, label: string = "New session started"): Promise<void> {
|
|
776
|
+
if (this.ctx.loadingAnimation) {
|
|
777
|
+
this.ctx.loadingAnimation.stop();
|
|
778
|
+
this.ctx.loadingAnimation = undefined;
|
|
779
|
+
}
|
|
780
|
+
this.ctx.statusContainer.clear();
|
|
781
|
+
|
|
782
|
+
if (this.ctx.session.isCompacting) {
|
|
783
|
+
this.ctx.session.abortCompaction();
|
|
784
|
+
while (this.ctx.session.isCompacting) {
|
|
785
|
+
await Bun.sleep(10);
|
|
786
|
+
}
|
|
787
|
+
}
|
|
788
|
+
if (!(await this.ctx.session.newSession(options))) return;
|
|
789
|
+
this.ctx.resetObserverRegistry();
|
|
790
|
+
setSessionTerminalTitle(this.ctx.sessionManager.getSessionName(), this.ctx.sessionManager.getCwd());
|
|
791
|
+
|
|
792
|
+
this.ctx.statusLine.invalidate();
|
|
793
|
+
this.ctx.statusLine.setSessionStartTime(Date.now());
|
|
794
|
+
this.ctx.updateEditorTopBorder();
|
|
795
|
+
this.ctx.updateEditorBorderColor();
|
|
796
|
+
this.ctx.chatContainer.clear();
|
|
797
|
+
this.ctx.pendingMessagesContainer.clear();
|
|
798
|
+
this.ctx.compactionQueuedMessages = [];
|
|
799
|
+
this.ctx.streamingComponent = undefined;
|
|
800
|
+
this.ctx.streamingMessage = undefined;
|
|
801
|
+
this.ctx.pendingTools.clear();
|
|
802
|
+
|
|
803
|
+
this.ctx.present([new Spacer(1), new Text(`${theme.fg("accent", `${theme.status.success} ${label}`)}`, 1, 1)]);
|
|
804
|
+
await this.ctx.reloadTodos();
|
|
805
|
+
this.ctx.ui.requestRender(true, { clearScrollback: true });
|
|
806
|
+
}
|
|
807
|
+
|
|
808
|
+
async handleClearCommand(): Promise<void> {
|
|
809
|
+
await this.#runNewSessionFlow();
|
|
810
|
+
}
|
|
811
|
+
|
|
812
|
+
async handleFreshCommand(): Promise<void> {
|
|
813
|
+
const result = this.ctx.session.freshSession();
|
|
814
|
+
if (!result) {
|
|
815
|
+
this.ctx.showWarning("Wait for the current response to finish or abort it before refreshing provider state.");
|
|
816
|
+
return;
|
|
817
|
+
}
|
|
818
|
+
const stateLabel = result.closedProviderSessions === 1 ? "provider state" : "provider states";
|
|
819
|
+
this.ctx.statusLine.invalidate();
|
|
820
|
+
this.ctx.updateEditorTopBorder();
|
|
821
|
+
this.ctx.showStatus(`Fresh provider session started (${result.closedProviderSessions} ${stateLabel} pruned).`);
|
|
822
|
+
}
|
|
823
|
+
|
|
824
|
+
async handleDropCommand(): Promise<void> {
|
|
825
|
+
if (!this.ctx.sessionManager.getSessionFile()) {
|
|
826
|
+
this.ctx.showError("Nothing to drop (in-memory session)");
|
|
827
|
+
return;
|
|
828
|
+
}
|
|
829
|
+
await this.#runNewSessionFlow({ drop: true }, "Session dropped");
|
|
830
|
+
}
|
|
831
|
+
|
|
832
|
+
async handleForkCommand(): Promise<void> {
|
|
833
|
+
if (this.ctx.session.isStreaming) {
|
|
834
|
+
this.ctx.showWarning("Wait for the current response to finish or abort it before forking.");
|
|
835
|
+
return;
|
|
836
|
+
}
|
|
837
|
+
if (this.ctx.loadingAnimation) {
|
|
838
|
+
this.ctx.loadingAnimation.stop();
|
|
839
|
+
this.ctx.loadingAnimation = undefined;
|
|
840
|
+
}
|
|
841
|
+
this.ctx.statusContainer.clear();
|
|
842
|
+
|
|
843
|
+
const success = await this.ctx.session.fork();
|
|
844
|
+
if (!success) {
|
|
845
|
+
this.ctx.showError("Fork failed (session not persisted or cancelled)");
|
|
846
|
+
return;
|
|
847
|
+
}
|
|
848
|
+
|
|
849
|
+
this.ctx.statusLine.invalidate();
|
|
850
|
+
this.ctx.updateEditorTopBorder();
|
|
851
|
+
|
|
852
|
+
const sessionFile = this.ctx.session.sessionFile;
|
|
853
|
+
const shortPath = sessionFile ? sessionFile.split("/").pop() : "new session";
|
|
854
|
+
this.ctx.present([
|
|
855
|
+
new Spacer(1),
|
|
856
|
+
new Text(`${theme.fg("accent", `${theme.status.success} Session forked to ${shortPath}`)}`, 1, 1),
|
|
857
|
+
]);
|
|
858
|
+
}
|
|
859
|
+
|
|
860
|
+
async handleMoveCommand(targetPath: string): Promise<void> {
|
|
861
|
+
if (this.ctx.session.isStreaming) {
|
|
862
|
+
this.ctx.showWarning("Wait for the current response to finish or abort it before moving.");
|
|
863
|
+
return;
|
|
864
|
+
}
|
|
865
|
+
|
|
866
|
+
const unquoted = stripOuterDoubleQuotes(targetPath);
|
|
867
|
+
if (!unquoted) {
|
|
868
|
+
this.ctx.showError("Usage: /move <path>");
|
|
869
|
+
return;
|
|
870
|
+
}
|
|
871
|
+
|
|
872
|
+
const cwd = this.ctx.sessionManager.getCwd();
|
|
873
|
+
const resolvedPath = resolveToCwd(unquoted, cwd);
|
|
874
|
+
|
|
875
|
+
try {
|
|
876
|
+
const stat = await fs.stat(resolvedPath);
|
|
877
|
+
if (!stat.isDirectory()) {
|
|
878
|
+
this.ctx.showError(`Not a directory: ${resolvedPath}`);
|
|
879
|
+
return;
|
|
880
|
+
}
|
|
881
|
+
} catch {
|
|
882
|
+
this.ctx.showError(`Directory does not exist: ${resolvedPath}`);
|
|
883
|
+
return;
|
|
884
|
+
}
|
|
885
|
+
|
|
886
|
+
try {
|
|
887
|
+
await this.ctx.sessionManager.flush();
|
|
888
|
+
await this.ctx.sessionManager.moveTo(resolvedPath);
|
|
889
|
+
await this.ctx.applyCwdChange(resolvedPath);
|
|
890
|
+
|
|
891
|
+
this.ctx.present([
|
|
892
|
+
new Spacer(1),
|
|
893
|
+
new Text(`${theme.fg("accent", `${theme.status.success} Session moved to ${resolvedPath}`)}`, 1, 1),
|
|
894
|
+
]);
|
|
895
|
+
} catch (err) {
|
|
896
|
+
this.ctx.showError(`Move failed: ${err instanceof Error ? err.message : String(err)}`);
|
|
897
|
+
}
|
|
898
|
+
}
|
|
899
|
+
|
|
900
|
+
async handleRenameCommand(title: string): Promise<void> {
|
|
901
|
+
try {
|
|
902
|
+
const stored = await this.ctx.sessionManager.setSessionName(title, "user");
|
|
903
|
+
if (!stored) {
|
|
904
|
+
this.ctx.showError("Session name cannot be empty.");
|
|
905
|
+
return;
|
|
906
|
+
}
|
|
907
|
+
const name = this.ctx.sessionManager.getSessionName()!;
|
|
908
|
+
setSessionTerminalTitle(name, this.ctx.sessionManager.getCwd());
|
|
909
|
+
this.ctx.statusLine.invalidate();
|
|
910
|
+
this.ctx.updateEditorBorderColor();
|
|
911
|
+
this.ctx.showStatus(`Session renamed to "${name}".`);
|
|
912
|
+
} catch (err) {
|
|
913
|
+
this.ctx.showError(`Rename failed: ${err instanceof Error ? err.message : String(err)}`);
|
|
914
|
+
}
|
|
915
|
+
}
|
|
916
|
+
|
|
917
|
+
async handleBashCommand(command: string, excludeFromContext = false): Promise<void> {
|
|
918
|
+
const isDeferred = this.ctx.session.isStreaming;
|
|
919
|
+
this.ctx.bashComponent = new BashExecutionComponent(command, this.ctx.ui, excludeFromContext);
|
|
920
|
+
|
|
921
|
+
if (isDeferred) {
|
|
922
|
+
this.ctx.pendingMessagesContainer.addChild(this.ctx.bashComponent);
|
|
923
|
+
this.ctx.pendingBashComponents.push(this.ctx.bashComponent);
|
|
924
|
+
} else {
|
|
925
|
+
this.ctx.present(this.ctx.bashComponent);
|
|
926
|
+
}
|
|
927
|
+
this.ctx.ui.requestRender();
|
|
928
|
+
|
|
929
|
+
try {
|
|
930
|
+
const result = await this.ctx.session.executeBash(
|
|
931
|
+
command,
|
|
932
|
+
chunk => {
|
|
933
|
+
if (this.ctx.bashComponent) {
|
|
934
|
+
this.ctx.bashComponent.appendOutput(chunk);
|
|
935
|
+
}
|
|
936
|
+
},
|
|
937
|
+
{ excludeFromContext, useUserShell: true },
|
|
938
|
+
);
|
|
939
|
+
|
|
940
|
+
if (this.ctx.bashComponent) {
|
|
941
|
+
const meta = outputMeta().truncationFromSummary(result, { direction: "tail" }).get();
|
|
942
|
+
this.ctx.bashComponent.setComplete(result.exitCode, result.cancelled, {
|
|
943
|
+
output: result.output,
|
|
944
|
+
truncation: meta?.truncation,
|
|
945
|
+
});
|
|
946
|
+
}
|
|
947
|
+
} catch (error) {
|
|
948
|
+
if (this.ctx.bashComponent) {
|
|
949
|
+
this.ctx.bashComponent.setComplete(undefined, false);
|
|
950
|
+
}
|
|
951
|
+
this.ctx.showError(`Bash command failed: ${error instanceof Error ? error.message : "Unknown error"}`);
|
|
952
|
+
}
|
|
953
|
+
|
|
954
|
+
this.ctx.bashComponent = undefined;
|
|
955
|
+
this.ctx.ui.requestRender();
|
|
956
|
+
}
|
|
957
|
+
|
|
958
|
+
async handlePythonCommand(code: string, excludeFromContext = false): Promise<void> {
|
|
959
|
+
const isDeferred = this.ctx.session.isStreaming;
|
|
960
|
+
this.ctx.pythonComponent = new EvalExecutionComponent(code, this.ctx.ui, excludeFromContext);
|
|
961
|
+
|
|
962
|
+
if (isDeferred) {
|
|
963
|
+
this.ctx.pendingMessagesContainer.addChild(this.ctx.pythonComponent);
|
|
964
|
+
this.ctx.pendingPythonComponents.push(this.ctx.pythonComponent);
|
|
965
|
+
} else {
|
|
966
|
+
this.ctx.present(this.ctx.pythonComponent);
|
|
967
|
+
}
|
|
968
|
+
this.ctx.ui.requestRender();
|
|
969
|
+
|
|
970
|
+
try {
|
|
971
|
+
const result = await this.ctx.session.executePython(
|
|
972
|
+
code,
|
|
973
|
+
chunk => {
|
|
974
|
+
if (this.ctx.pythonComponent) {
|
|
975
|
+
this.ctx.pythonComponent.appendOutput(chunk);
|
|
976
|
+
}
|
|
977
|
+
},
|
|
978
|
+
{ excludeFromContext },
|
|
979
|
+
);
|
|
980
|
+
|
|
981
|
+
if (this.ctx.pythonComponent) {
|
|
982
|
+
const meta = outputMeta().truncationFromSummary(result, { direction: "tail" }).get();
|
|
983
|
+
this.ctx.pythonComponent.setComplete(result.exitCode, result.cancelled, {
|
|
984
|
+
output: result.output,
|
|
985
|
+
truncation: meta?.truncation,
|
|
986
|
+
});
|
|
987
|
+
}
|
|
988
|
+
} catch (error) {
|
|
989
|
+
if (this.ctx.pythonComponent) {
|
|
990
|
+
this.ctx.pythonComponent.setComplete(undefined, false);
|
|
991
|
+
}
|
|
992
|
+
this.ctx.showError(`Python execution failed: ${error instanceof Error ? error.message : "Unknown error"}`);
|
|
993
|
+
}
|
|
994
|
+
|
|
995
|
+
this.ctx.pythonComponent = undefined;
|
|
996
|
+
this.ctx.ui.requestRender();
|
|
997
|
+
}
|
|
998
|
+
|
|
999
|
+
async handleCompactCommand(customInstructions?: string): Promise<CompactionOutcome> {
|
|
1000
|
+
const entries = this.ctx.sessionManager.getEntries();
|
|
1001
|
+
const messageCount = entries.filter(e => e.type === "message").length;
|
|
1002
|
+
|
|
1003
|
+
if (messageCount < 2) {
|
|
1004
|
+
this.ctx.showWarning("Nothing to compact (no messages yet)");
|
|
1005
|
+
return "ok";
|
|
1006
|
+
}
|
|
1007
|
+
|
|
1008
|
+
return this.executeCompaction(customInstructions, false);
|
|
1009
|
+
}
|
|
1010
|
+
|
|
1011
|
+
/**
|
|
1012
|
+
* TUI handler for `/shake`. `elide` drops heavy structural content and
|
|
1013
|
+
* `images` strips image blocks. Rebuilds the chat and reports counts.
|
|
1014
|
+
*/
|
|
1015
|
+
async handleShakeCommand(mode: ShakeMode): Promise<void> {
|
|
1016
|
+
let result: ShakeResult;
|
|
1017
|
+
try {
|
|
1018
|
+
result = await this.ctx.session.shake(mode);
|
|
1019
|
+
} catch (error) {
|
|
1020
|
+
this.ctx.showError(`Shake failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
1021
|
+
return;
|
|
1022
|
+
}
|
|
1023
|
+
|
|
1024
|
+
const dropped = result.toolResultsDropped + result.blocksDropped + (result.imagesDropped ?? 0);
|
|
1025
|
+
if (dropped === 0) {
|
|
1026
|
+
this.ctx.showStatus("Nothing to shake.");
|
|
1027
|
+
return;
|
|
1028
|
+
}
|
|
1029
|
+
this.ctx.rebuildChatFromMessages();
|
|
1030
|
+
this.ctx.statusLine.invalidate();
|
|
1031
|
+
this.ctx.updateEditorTopBorder();
|
|
1032
|
+
this.ctx.showStatus(formatShakeSummary(result));
|
|
1033
|
+
}
|
|
1034
|
+
|
|
1035
|
+
async handleSkillCommand(skillPath: string, args: string): Promise<void> {
|
|
1036
|
+
try {
|
|
1037
|
+
const content = await Bun.file(skillPath).text();
|
|
1038
|
+
const body = content.replace(/^---\n[\s\S]*?\n---\n/, "").trim();
|
|
1039
|
+
const metaLines = [`Skill: ${skillPath}`];
|
|
1040
|
+
if (args) {
|
|
1041
|
+
metaLines.push(`User: ${args}`);
|
|
1042
|
+
}
|
|
1043
|
+
const message = `${body}\n\n---\n\n${metaLines.join("\n")}`;
|
|
1044
|
+
await this.ctx.session.prompt(message);
|
|
1045
|
+
} catch (err) {
|
|
1046
|
+
this.ctx.showError(`Failed to load skill: ${err instanceof Error ? err.message : String(err)}`);
|
|
1047
|
+
}
|
|
1048
|
+
}
|
|
1049
|
+
|
|
1050
|
+
async executeCompaction(
|
|
1051
|
+
customInstructionsOrOptions?: string | CompactOptions,
|
|
1052
|
+
isAuto = false,
|
|
1053
|
+
): Promise<CompactionOutcome> {
|
|
1054
|
+
if (this.ctx.loadingAnimation) {
|
|
1055
|
+
this.ctx.loadingAnimation.stop();
|
|
1056
|
+
this.ctx.loadingAnimation = undefined;
|
|
1057
|
+
}
|
|
1058
|
+
this.ctx.statusContainer.clear();
|
|
1059
|
+
|
|
1060
|
+
const originalOnEscape = this.ctx.editor.onEscape;
|
|
1061
|
+
this.ctx.editor.onEscape = () => {
|
|
1062
|
+
this.ctx.session.abortCompaction();
|
|
1063
|
+
};
|
|
1064
|
+
|
|
1065
|
+
this.ctx.chatContainer.addChild(new Spacer(1));
|
|
1066
|
+
const label = isAuto ? "Auto-compacting context... (esc to cancel)" : "Compacting context... (esc to cancel)";
|
|
1067
|
+
const compactingLoader = new Loader(
|
|
1068
|
+
this.ctx.ui,
|
|
1069
|
+
spinner => theme.fg("accent", spinner),
|
|
1070
|
+
text => theme.fg("muted", text),
|
|
1071
|
+
label,
|
|
1072
|
+
getSymbolTheme().spinnerFrames,
|
|
1073
|
+
);
|
|
1074
|
+
this.ctx.statusContainer.addChild(compactingLoader);
|
|
1075
|
+
this.ctx.ui.requestRender();
|
|
1076
|
+
|
|
1077
|
+
let outcome: CompactionOutcome = "ok";
|
|
1078
|
+
try {
|
|
1079
|
+
const instructions = typeof customInstructionsOrOptions === "string" ? customInstructionsOrOptions : undefined;
|
|
1080
|
+
const options =
|
|
1081
|
+
customInstructionsOrOptions && typeof customInstructionsOrOptions === "object"
|
|
1082
|
+
? customInstructionsOrOptions
|
|
1083
|
+
: undefined;
|
|
1084
|
+
await this.ctx.session.compact(instructions, options);
|
|
1085
|
+
|
|
1086
|
+
this.ctx.rebuildChatFromMessages();
|
|
1087
|
+
|
|
1088
|
+
this.ctx.statusLine.invalidate();
|
|
1089
|
+
this.ctx.updateEditorTopBorder();
|
|
1090
|
+
} catch (error) {
|
|
1091
|
+
if (error instanceof CompactionCancelledError) {
|
|
1092
|
+
outcome = "cancelled";
|
|
1093
|
+
this.ctx.showError("Compaction cancelled");
|
|
1094
|
+
} else {
|
|
1095
|
+
outcome = "failed";
|
|
1096
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
1097
|
+
this.ctx.showError(`Compaction failed: ${message}`);
|
|
1098
|
+
}
|
|
1099
|
+
} finally {
|
|
1100
|
+
compactingLoader.stop();
|
|
1101
|
+
this.ctx.statusContainer.clear();
|
|
1102
|
+
this.ctx.editor.onEscape = originalOnEscape;
|
|
1103
|
+
}
|
|
1104
|
+
await this.ctx.flushCompactionQueue({ willRetry: false });
|
|
1105
|
+
return outcome;
|
|
1106
|
+
}
|
|
1107
|
+
|
|
1108
|
+
async handleHandoffCommand(customInstructions?: string): Promise<void> {
|
|
1109
|
+
const entries = this.ctx.sessionManager.getEntries();
|
|
1110
|
+
const messageCount = entries.filter(e => e.type === "message").length;
|
|
1111
|
+
|
|
1112
|
+
if (messageCount < 2) {
|
|
1113
|
+
this.ctx.showWarning("Nothing to hand off (no messages yet)");
|
|
1114
|
+
return;
|
|
1115
|
+
}
|
|
1116
|
+
|
|
1117
|
+
if (this.ctx.loadingAnimation) {
|
|
1118
|
+
this.ctx.loadingAnimation.stop();
|
|
1119
|
+
this.ctx.loadingAnimation = undefined;
|
|
1120
|
+
}
|
|
1121
|
+
this.ctx.statusContainer.clear();
|
|
1122
|
+
|
|
1123
|
+
const originalOnEscape = this.ctx.editor.onEscape;
|
|
1124
|
+
this.ctx.editor.onEscape = () => {
|
|
1125
|
+
this.ctx.session.abortHandoff();
|
|
1126
|
+
};
|
|
1127
|
+
|
|
1128
|
+
const handoffLoader = new Loader(
|
|
1129
|
+
this.ctx.ui,
|
|
1130
|
+
spinner => theme.fg("accent", spinner),
|
|
1131
|
+
text => theme.fg("muted", text),
|
|
1132
|
+
"Generating handoff… (esc to cancel)",
|
|
1133
|
+
getSymbolTheme().spinnerFrames,
|
|
1134
|
+
);
|
|
1135
|
+
this.ctx.statusContainer.addChild(handoffLoader);
|
|
1136
|
+
this.ctx.ui.requestRender();
|
|
1137
|
+
|
|
1138
|
+
try {
|
|
1139
|
+
// Handoff generation runs as a oneshot request; the new session is shown after it completes.
|
|
1140
|
+
const result = await this.ctx.session.handoff(customInstructions);
|
|
1141
|
+
|
|
1142
|
+
if (!result) {
|
|
1143
|
+
this.ctx.showError("Handoff cancelled");
|
|
1144
|
+
return;
|
|
1145
|
+
}
|
|
1146
|
+
|
|
1147
|
+
// Rebuild chat from the new session (which now contains the handoff document)
|
|
1148
|
+
this.ctx.rebuildChatFromMessages();
|
|
1149
|
+
|
|
1150
|
+
this.ctx.statusLine.invalidate();
|
|
1151
|
+
this.ctx.updateEditorTopBorder();
|
|
1152
|
+
this.ctx.updateEditorBorderColor();
|
|
1153
|
+
await this.ctx.reloadTodos();
|
|
1154
|
+
|
|
1155
|
+
this.ctx.present([
|
|
1156
|
+
new Spacer(1),
|
|
1157
|
+
new Text(`${theme.fg("accent", `${theme.status.success} New session started with handoff context`)}`, 1, 1),
|
|
1158
|
+
]);
|
|
1159
|
+
if (result.savedPath) {
|
|
1160
|
+
this.ctx.showStatus(`Handoff document saved to: ${result.savedPath}`);
|
|
1161
|
+
}
|
|
1162
|
+
} catch (error) {
|
|
1163
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
1164
|
+
if (message === "Handoff cancelled" || (error instanceof Error && error.name === "AbortError")) {
|
|
1165
|
+
this.ctx.showError("Handoff cancelled");
|
|
1166
|
+
} else {
|
|
1167
|
+
this.ctx.showError(`Handoff failed: ${message}`);
|
|
1168
|
+
}
|
|
1169
|
+
} finally {
|
|
1170
|
+
handoffLoader.stop();
|
|
1171
|
+
this.ctx.statusContainer.clear();
|
|
1172
|
+
this.ctx.editor.onEscape = originalOnEscape;
|
|
1173
|
+
}
|
|
1174
|
+
this.ctx.ui.requestRender();
|
|
1175
|
+
}
|
|
1176
|
+
}
|
|
1177
|
+
|
|
1178
|
+
const BAR_WIDTH_MAX = 24;
|
|
1179
|
+
const BAR_WIDTH_MIN = 4;
|
|
1180
|
+
|
|
1181
|
+
function renderJobLine(job: AsyncJobSnapshotItem, now: number): string {
|
|
1182
|
+
const duration = formatDuration(Math.max(0, now - job.startTime));
|
|
1183
|
+
const status = formatJobStatus(job.status);
|
|
1184
|
+
return `${theme.fg("dim", job.id)} ${theme.fg("dim", `[${job.type}]`)} ${status} ${theme.fg("dim", `(${duration})`)}`;
|
|
1185
|
+
}
|
|
1186
|
+
|
|
1187
|
+
function formatJobStatus(status: AsyncJobSnapshotItem["status"]): string {
|
|
1188
|
+
if (status === "running") return theme.fg("warning", "running");
|
|
1189
|
+
if (status === "completed") return theme.fg("success", "completed");
|
|
1190
|
+
if (status === "cancelled") return theme.fg("dim", "cancelled");
|
|
1191
|
+
return theme.fg("error", "failed");
|
|
1192
|
+
}
|
|
1193
|
+
|
|
1194
|
+
function truncateJobLabel(label: string, maxWidth: number): string {
|
|
1195
|
+
if (visibleWidth(label) <= maxWidth) return label;
|
|
1196
|
+
if (maxWidth <= 1) return "…";
|
|
1197
|
+
|
|
1198
|
+
let out = "";
|
|
1199
|
+
for (const char of label) {
|
|
1200
|
+
const next = `${out}${char}`;
|
|
1201
|
+
if (visibleWidth(`${next}…`) > maxWidth) break;
|
|
1202
|
+
out = next;
|
|
1203
|
+
}
|
|
1204
|
+
|
|
1205
|
+
return `${out}…`;
|
|
1206
|
+
}
|
|
1207
|
+
|
|
1208
|
+
function formatProviderName(provider: string): string {
|
|
1209
|
+
return provider
|
|
1210
|
+
.split(/[-_]/g)
|
|
1211
|
+
.map(part => (part ? part[0].toUpperCase() + part.slice(1) : ""))
|
|
1212
|
+
.join(" ");
|
|
1213
|
+
}
|
|
1214
|
+
|
|
1215
|
+
function formatNumber(value: number, maxFractionDigits = 1): string {
|
|
1216
|
+
return new Intl.NumberFormat("en-US", { maximumFractionDigits: maxFractionDigits }).format(value);
|
|
1217
|
+
}
|
|
1218
|
+
|
|
1219
|
+
function resolveProviderAuthMode(authStorage: AuthStorage, provider: string): string {
|
|
1220
|
+
if (authStorage.hasOAuth(provider)) {
|
|
1221
|
+
return "oauth";
|
|
1222
|
+
}
|
|
1223
|
+
if (authStorage.has(provider)) {
|
|
1224
|
+
return "api key";
|
|
1225
|
+
}
|
|
1226
|
+
if (getEnvApiKey(provider)) {
|
|
1227
|
+
return "env api key";
|
|
1228
|
+
}
|
|
1229
|
+
if (authStorage.hasAuth(provider)) {
|
|
1230
|
+
return "runtime/fallback";
|
|
1231
|
+
}
|
|
1232
|
+
return "unknown";
|
|
1233
|
+
}
|
|
1234
|
+
|
|
1235
|
+
export function renderProviderSection(details: ProviderDetails, uiTheme: Pick<typeof theme, "fg">): string {
|
|
1236
|
+
const lines: string[] = [];
|
|
1237
|
+
lines.push(`${uiTheme.fg("dim", "Name:")} ${details.provider}`);
|
|
1238
|
+
for (const field of details.fields) {
|
|
1239
|
+
lines.push(`${uiTheme.fg("dim", `${field.label}:`)} ${field.value}`);
|
|
1240
|
+
}
|
|
1241
|
+
return `${lines.join("\n")}\n`;
|
|
1242
|
+
}
|
|
1243
|
+
|
|
1244
|
+
function resolveFraction(limit: UsageLimit): number | undefined {
|
|
1245
|
+
const amount = limit.amount;
|
|
1246
|
+
if (amount.usedFraction !== undefined) return amount.usedFraction;
|
|
1247
|
+
if (amount.used !== undefined && amount.limit !== undefined && amount.limit > 0) {
|
|
1248
|
+
return amount.used / amount.limit;
|
|
1249
|
+
}
|
|
1250
|
+
if (amount.unit === "percent" && amount.used !== undefined) {
|
|
1251
|
+
return amount.used / 100;
|
|
1252
|
+
}
|
|
1253
|
+
return undefined;
|
|
1254
|
+
}
|
|
1255
|
+
|
|
1256
|
+
function resolveProviderUsageTotal(reports: UsageReport[]): number {
|
|
1257
|
+
return reports
|
|
1258
|
+
.flatMap(report => report.limits)
|
|
1259
|
+
.map(limit => resolveFraction(limit) ?? 0)
|
|
1260
|
+
.reduce((sum, value) => sum + value, 0);
|
|
1261
|
+
}
|
|
1262
|
+
|
|
1263
|
+
function formatLimitTitle(limit: UsageLimit): string {
|
|
1264
|
+
const tier = limit.scope.tier;
|
|
1265
|
+
if (tier && !limit.label.toLowerCase().includes(tier.toLowerCase())) {
|
|
1266
|
+
return `${limit.label} (${tier})`;
|
|
1267
|
+
}
|
|
1268
|
+
return limit.label;
|
|
1269
|
+
}
|
|
1270
|
+
|
|
1271
|
+
function formatWindowSuffix(label: string, windowLabel: string, uiTheme: typeof theme): string {
|
|
1272
|
+
const normalizedLabel = label.toLowerCase();
|
|
1273
|
+
const normalizedWindow = windowLabel.toLowerCase();
|
|
1274
|
+
if (normalizedWindow === "quota window") return "";
|
|
1275
|
+
if (normalizedLabel.includes(normalizedWindow)) return "";
|
|
1276
|
+
return uiTheme.fg("dim", `(${windowLabel})`);
|
|
1277
|
+
}
|
|
1278
|
+
|
|
1279
|
+
function formatAccountLabel(limit: UsageLimit, report: UsageReport, index: number): string {
|
|
1280
|
+
const email = (report.metadata?.email as string | undefined) ?? limit.scope.accountId;
|
|
1281
|
+
if (email) return email;
|
|
1282
|
+
const accountId = (report.metadata?.accountId as string | undefined) ?? limit.scope.accountId;
|
|
1283
|
+
if (accountId) return accountId;
|
|
1284
|
+
const projectId = (report.metadata?.projectId as string | undefined) ?? limit.scope.projectId;
|
|
1285
|
+
if (projectId) return projectId;
|
|
1286
|
+
return `account ${index + 1}`;
|
|
1287
|
+
}
|
|
1288
|
+
|
|
1289
|
+
function formatUnlimitedReportLabel(report: UsageReport, index: number): string {
|
|
1290
|
+
const email = report.metadata?.email as string | undefined;
|
|
1291
|
+
if (email) return email;
|
|
1292
|
+
const accountId = report.metadata?.accountId as string | undefined;
|
|
1293
|
+
if (accountId) return accountId;
|
|
1294
|
+
const projectId = report.metadata?.projectId as string | undefined;
|
|
1295
|
+
if (projectId) return projectId;
|
|
1296
|
+
return `account ${index + 1}`;
|
|
1297
|
+
}
|
|
1298
|
+
|
|
1299
|
+
function formatResetShort(limit: UsageLimit, nowMs: number): string | undefined {
|
|
1300
|
+
const resetsAt = limit.window?.resetsAt;
|
|
1301
|
+
if (resetsAt === undefined) return undefined;
|
|
1302
|
+
// Codex returns the prior window's reset_at until a new request opens a fresh window —
|
|
1303
|
+
// rendering a negative delta is meaningless, so drop the suffix in that case.
|
|
1304
|
+
if (resetsAt <= nowMs) return undefined;
|
|
1305
|
+
return formatDuration(resetsAt - nowMs);
|
|
1306
|
+
}
|
|
1307
|
+
|
|
1308
|
+
function formatAccountHeaderRow(
|
|
1309
|
+
limits: UsageLimit[],
|
|
1310
|
+
reports: UsageReport[],
|
|
1311
|
+
nowMs: number,
|
|
1312
|
+
columnWidth: number,
|
|
1313
|
+
uiTheme: typeof theme,
|
|
1314
|
+
): string[] {
|
|
1315
|
+
const parts = limits.map((limit, index) => {
|
|
1316
|
+
const reset = formatResetShort(limit, nowMs);
|
|
1317
|
+
return {
|
|
1318
|
+
label: formatAccountLabel(limit, reports[index], index),
|
|
1319
|
+
suffix: reset ? `(${reset})` : "",
|
|
1320
|
+
};
|
|
1321
|
+
});
|
|
1322
|
+
const maxSuffixWidth = parts.reduce((max, p) => Math.max(max, visibleWidth(p.suffix)), 0);
|
|
1323
|
+
const gap = maxSuffixWidth > 0 ? 1 : 0;
|
|
1324
|
+
const prefixBudget = columnWidth - maxSuffixWidth - gap;
|
|
1325
|
+
|
|
1326
|
+
// If suffix can't share the cell with at least `x…`, fall back to whole-label truncation.
|
|
1327
|
+
if (prefixBudget < 2) {
|
|
1328
|
+
return parts.map(p => {
|
|
1329
|
+
const full = p.suffix ? `${p.label} ${p.suffix}` : p.label;
|
|
1330
|
+
return padColumn(truncateJobLabel(full, columnWidth), columnWidth);
|
|
1331
|
+
});
|
|
1332
|
+
}
|
|
1333
|
+
|
|
1334
|
+
return parts.map(p => {
|
|
1335
|
+
const prefix = truncateJobLabel(p.label, prefixBudget);
|
|
1336
|
+
const prefixCell = prefix + " ".repeat(prefixBudget - visibleWidth(prefix));
|
|
1337
|
+
if (!p.suffix) return prefixCell + " ".repeat(maxSuffixWidth + gap);
|
|
1338
|
+
const suffixPad = " ".repeat(maxSuffixWidth - visibleWidth(p.suffix));
|
|
1339
|
+
return `${prefixCell} ${suffixPad}${uiTheme.fg("dim", p.suffix)}`;
|
|
1340
|
+
});
|
|
1341
|
+
}
|
|
1342
|
+
|
|
1343
|
+
function padColumn(text: string, width: number): string {
|
|
1344
|
+
const visible = visibleWidth(text);
|
|
1345
|
+
if (visible >= width) return text;
|
|
1346
|
+
return `${text}${padding(width - visible)}`;
|
|
1347
|
+
}
|
|
1348
|
+
|
|
1349
|
+
function resolveAggregateStatus(limits: UsageLimit[]): UsageLimit["status"] {
|
|
1350
|
+
const hasOk = limits.some(limit => limit.status === "ok");
|
|
1351
|
+
const hasWarning = limits.some(limit => limit.status === "warning");
|
|
1352
|
+
const hasExhausted = limits.some(limit => limit.status === "exhausted");
|
|
1353
|
+
if (!hasOk && !hasWarning && !hasExhausted) return "unknown";
|
|
1354
|
+
if (hasOk) {
|
|
1355
|
+
return hasWarning || hasExhausted ? "warning" : "ok";
|
|
1356
|
+
}
|
|
1357
|
+
if (hasWarning) return "warning";
|
|
1358
|
+
return "exhausted";
|
|
1359
|
+
}
|
|
1360
|
+
|
|
1361
|
+
function formatAggregateAmount(limits: UsageLimit[]): string {
|
|
1362
|
+
const fractions = limits
|
|
1363
|
+
.map(limit => resolveFraction(limit))
|
|
1364
|
+
.filter((value): value is number => value !== undefined);
|
|
1365
|
+
if (fractions.length === limits.length && fractions.length > 0) {
|
|
1366
|
+
const sum = fractions.reduce((total, value) => total + value, 0);
|
|
1367
|
+
const avgRemaining = Math.max(0, ((limits.length - sum) / limits.length) * 100);
|
|
1368
|
+
return `${formatNumber(avgRemaining)}% free`;
|
|
1369
|
+
}
|
|
1370
|
+
|
|
1371
|
+
const amounts = limits
|
|
1372
|
+
.map(limit => limit.amount)
|
|
1373
|
+
.filter(amount => amount.used !== undefined && amount.limit !== undefined && amount.limit > 0);
|
|
1374
|
+
if (amounts.length === limits.length && amounts.length > 0) {
|
|
1375
|
+
const totalUsed = amounts.reduce((sum, amount) => sum + (amount.used ?? 0), 0);
|
|
1376
|
+
const totalLimit = amounts.reduce((sum, amount) => sum + (amount.limit ?? 0), 0);
|
|
1377
|
+
const remainingPct = totalLimit > 0 ? Math.max(0, 100 - (totalUsed / totalLimit) * 100) : 0;
|
|
1378
|
+
return `${formatNumber(remainingPct)}% free`;
|
|
1379
|
+
}
|
|
1380
|
+
|
|
1381
|
+
// Count unique accounts from limit scopes — not limits.length.
|
|
1382
|
+
const uniqueAccountIds = new Set(
|
|
1383
|
+
limits.map(limit => limit.scope.accountId).filter((id): id is string => typeof id === "string" && id.length > 0),
|
|
1384
|
+
);
|
|
1385
|
+
if (uniqueAccountIds.size > 0) return `${uniqueAccountIds.size} ${uniqueAccountIds.size === 1 ? "acct" : "accts"}`;
|
|
1386
|
+
// No account IDs available — keep the pre-existing fallback so providers
|
|
1387
|
+
// that don't populate scope.accountId still show a summary.
|
|
1388
|
+
return `${limits.length} accts`;
|
|
1389
|
+
}
|
|
1390
|
+
|
|
1391
|
+
function resolveResetRange(limits: UsageLimit[], nowMs: number): string | null {
|
|
1392
|
+
const absolute = limits
|
|
1393
|
+
.map(limit => limit.window?.resetsAt)
|
|
1394
|
+
.filter((value): value is number => value !== undefined && Number.isFinite(value) && value > nowMs);
|
|
1395
|
+
if (absolute.length === 0) return null;
|
|
1396
|
+
const offsets = absolute.map(value => value - nowMs);
|
|
1397
|
+
const minReset = Math.min(...offsets);
|
|
1398
|
+
const maxReset = Math.max(...offsets);
|
|
1399
|
+
if (maxReset - minReset > 60_000) {
|
|
1400
|
+
return `resets in ${formatDuration(minReset)}–${formatDuration(maxReset)}`;
|
|
1401
|
+
}
|
|
1402
|
+
return `resets in ${formatDuration(minReset)}`;
|
|
1403
|
+
}
|
|
1404
|
+
|
|
1405
|
+
function resolveStatusIcon(status: UsageLimit["status"], uiTheme: typeof theme): string {
|
|
1406
|
+
if (status === "exhausted") return uiTheme.fg("error", uiTheme.status.error);
|
|
1407
|
+
if (status === "warning") return uiTheme.fg("warning", uiTheme.status.warning);
|
|
1408
|
+
if (status === "ok") return uiTheme.fg("success", uiTheme.status.success);
|
|
1409
|
+
return uiTheme.fg("dim", uiTheme.status.pending);
|
|
1410
|
+
}
|
|
1411
|
+
|
|
1412
|
+
function resolveStatusColor(status: UsageLimit["status"]): "success" | "warning" | "error" | "dim" {
|
|
1413
|
+
if (status === "exhausted") return "error";
|
|
1414
|
+
if (status === "warning") return "warning";
|
|
1415
|
+
if (status === "ok") return "success";
|
|
1416
|
+
return "dim";
|
|
1417
|
+
}
|
|
1418
|
+
|
|
1419
|
+
function renderUsageBar(limit: UsageLimit, uiTheme: typeof theme, barWidth: number): string {
|
|
1420
|
+
const fraction = resolveFraction(limit);
|
|
1421
|
+
if (fraction === undefined) {
|
|
1422
|
+
return uiTheme.fg("dim", "·".repeat(barWidth));
|
|
1423
|
+
}
|
|
1424
|
+
const clamped = Math.min(Math.max(fraction, 0), 1);
|
|
1425
|
+
const exact = clamped * barWidth;
|
|
1426
|
+
const fullCells = Math.floor(exact);
|
|
1427
|
+
const remainder = exact - fullCells;
|
|
1428
|
+
let partial = "";
|
|
1429
|
+
if (remainder >= 2 / 3) partial = "▓";
|
|
1430
|
+
else if (remainder >= 1 / 3) partial = "▒";
|
|
1431
|
+
const leading = "█".repeat(fullCells) + partial;
|
|
1432
|
+
const empty = "░".repeat(Math.max(0, barWidth - fullCells - (partial ? 1 : 0)));
|
|
1433
|
+
const color = resolveStatusColor(limit.status);
|
|
1434
|
+
return `${uiTheme.fg(color, leading)}${uiTheme.fg("dim", empty)}`;
|
|
1435
|
+
}
|
|
1436
|
+
|
|
1437
|
+
/**
|
|
1438
|
+
* Pick a per-column width so n bars + a trailing amount string fit in `available` columns.
|
|
1439
|
+
* Falls back to the minimum when the terminal is too narrow rather than wrapping.
|
|
1440
|
+
*/
|
|
1441
|
+
function resolveColumnWidth(count: number, available: number, trailing: number): number {
|
|
1442
|
+
if (count <= 0) return BAR_WIDTH_MAX;
|
|
1443
|
+
const indent = 2;
|
|
1444
|
+
const gaps = count - 1;
|
|
1445
|
+
const spaceForBars = available - indent - gaps - (trailing > 0 ? trailing + 1 : 0);
|
|
1446
|
+
const ideal = Math.floor(spaceForBars / count);
|
|
1447
|
+
const min = BAR_WIDTH_MIN;
|
|
1448
|
+
const max = BAR_WIDTH_MAX;
|
|
1449
|
+
if (ideal < min) return min;
|
|
1450
|
+
if (ideal > max) return max;
|
|
1451
|
+
return ideal;
|
|
1452
|
+
}
|
|
1453
|
+
|
|
1454
|
+
function renderUsageReports(
|
|
1455
|
+
reports: UsageReport[],
|
|
1456
|
+
uiTheme: typeof theme,
|
|
1457
|
+
nowMs: number,
|
|
1458
|
+
availableWidth: number,
|
|
1459
|
+
): string {
|
|
1460
|
+
const lines: string[] = [];
|
|
1461
|
+
const latestFetchedAt = Math.max(...reports.map(report => report.fetchedAt ?? 0));
|
|
1462
|
+
const headerSuffix = latestFetchedAt ? ` (${formatDuration(nowMs - latestFetchedAt)} ago)` : "";
|
|
1463
|
+
lines.push(uiTheme.bold(uiTheme.fg("accent", `Usage${headerSuffix}`)));
|
|
1464
|
+
const grouped = new Map<string, UsageReport[]>();
|
|
1465
|
+
for (const report of reports) {
|
|
1466
|
+
const list = grouped.get(report.provider) ?? [];
|
|
1467
|
+
list.push(report);
|
|
1468
|
+
grouped.set(report.provider, list);
|
|
1469
|
+
}
|
|
1470
|
+
const providerEntries = Array.from(grouped.entries())
|
|
1471
|
+
.map(([provider, providerReports]) => ({
|
|
1472
|
+
provider,
|
|
1473
|
+
providerReports,
|
|
1474
|
+
totalUsage: resolveProviderUsageTotal(providerReports),
|
|
1475
|
+
}))
|
|
1476
|
+
.sort((a, b) => {
|
|
1477
|
+
if (a.totalUsage !== b.totalUsage) return a.totalUsage - b.totalUsage;
|
|
1478
|
+
return a.provider.localeCompare(b.provider);
|
|
1479
|
+
});
|
|
1480
|
+
|
|
1481
|
+
for (const { provider, providerReports } of providerEntries) {
|
|
1482
|
+
lines.push("");
|
|
1483
|
+
const providerName = formatProviderName(provider);
|
|
1484
|
+
|
|
1485
|
+
const limitGroups = new Map<
|
|
1486
|
+
string,
|
|
1487
|
+
{ label: string; windowLabel: string; limits: UsageLimit[]; reports: UsageReport[] }
|
|
1488
|
+
>();
|
|
1489
|
+
for (const report of providerReports) {
|
|
1490
|
+
for (const limit of report.limits) {
|
|
1491
|
+
const windowId = limit.window?.id ?? limit.scope.windowId ?? "default";
|
|
1492
|
+
const key = `${formatLimitTitle(limit)}|${windowId}`;
|
|
1493
|
+
const windowLabel = limit.window?.label ?? windowId;
|
|
1494
|
+
const entry = limitGroups.get(key) ?? {
|
|
1495
|
+
label: formatLimitTitle(limit),
|
|
1496
|
+
windowLabel,
|
|
1497
|
+
limits: [],
|
|
1498
|
+
reports: [],
|
|
1499
|
+
};
|
|
1500
|
+
entry.limits.push(limit);
|
|
1501
|
+
entry.reports.push(report);
|
|
1502
|
+
limitGroups.set(key, entry);
|
|
1503
|
+
}
|
|
1504
|
+
}
|
|
1505
|
+
|
|
1506
|
+
lines.push(uiTheme.bold(uiTheme.fg("accent", providerName)));
|
|
1507
|
+
|
|
1508
|
+
const renderableGroups = Array.from(limitGroups.values()).map(group => {
|
|
1509
|
+
const entries = group.limits.map((limit, index) => ({
|
|
1510
|
+
limit,
|
|
1511
|
+
report: group.reports[index],
|
|
1512
|
+
fraction: resolveFraction(limit),
|
|
1513
|
+
index,
|
|
1514
|
+
}));
|
|
1515
|
+
entries.sort((a, b) => {
|
|
1516
|
+
const aFraction = a.fraction ?? -1;
|
|
1517
|
+
const bFraction = b.fraction ?? -1;
|
|
1518
|
+
if (aFraction !== bFraction) return bFraction - aFraction;
|
|
1519
|
+
return a.index - b.index;
|
|
1520
|
+
});
|
|
1521
|
+
const sortedLimits = entries.map(entry => entry.limit);
|
|
1522
|
+
const sortedReports = entries.map(entry => entry.report);
|
|
1523
|
+
return { group, sortedLimits, sortedReports, amountText: formatAggregateAmount(sortedLimits) };
|
|
1524
|
+
});
|
|
1525
|
+
|
|
1526
|
+
const sectionCount = renderableGroups.reduce((max, g) => Math.max(max, g.sortedLimits.length), 0);
|
|
1527
|
+
const sectionTrailing = renderableGroups.reduce((max, g) => Math.max(max, visibleWidth(g.amountText)), 0);
|
|
1528
|
+
const sectionColumnWidth = resolveColumnWidth(sectionCount, availableWidth, sectionTrailing);
|
|
1529
|
+
|
|
1530
|
+
for (const { group, sortedLimits, sortedReports, amountText } of renderableGroups) {
|
|
1531
|
+
const status = resolveAggregateStatus(sortedLimits);
|
|
1532
|
+
const statusIcon = resolveStatusIcon(status, uiTheme);
|
|
1533
|
+
|
|
1534
|
+
const windowSuffix = formatWindowSuffix(group.label, group.windowLabel, uiTheme);
|
|
1535
|
+
lines.push(`${statusIcon} ${uiTheme.bold(group.label)} ${windowSuffix}`.trim());
|
|
1536
|
+
const accountLabels = formatAccountHeaderRow(sortedLimits, sortedReports, nowMs, sectionColumnWidth, uiTheme);
|
|
1537
|
+
lines.push(` ${accountLabels.join(" ")}`.trimEnd());
|
|
1538
|
+
const bars = sortedLimits.map(limit =>
|
|
1539
|
+
padColumn(renderUsageBar(limit, uiTheme, sectionColumnWidth), sectionColumnWidth),
|
|
1540
|
+
);
|
|
1541
|
+
lines.push(` ${bars.join(" ")} ${amountText}`.trimEnd());
|
|
1542
|
+
const resetText = sortedLimits.length <= 1 ? resolveResetRange(sortedLimits, nowMs) : null;
|
|
1543
|
+
if (resetText) {
|
|
1544
|
+
lines.push(` ${uiTheme.fg("dim", resetText)}`.trimEnd());
|
|
1545
|
+
}
|
|
1546
|
+
const notes = sortedLimits.flatMap(limit => limit.notes ?? []);
|
|
1547
|
+
if (notes.length > 0) {
|
|
1548
|
+
lines.push(` ${uiTheme.fg("dim", notes.join(" • "))}`.trimEnd());
|
|
1549
|
+
}
|
|
1550
|
+
}
|
|
1551
|
+
|
|
1552
|
+
// Render accounts with no rate limits (e.g. business/enterprise plans).
|
|
1553
|
+
const unlimitedReports = providerReports.filter(report => report.limits.length === 0);
|
|
1554
|
+
for (const report of unlimitedReports) {
|
|
1555
|
+
const label = formatUnlimitedReportLabel(report, 0);
|
|
1556
|
+
const tier = report.metadata?.planType as string | undefined;
|
|
1557
|
+
const tierSuffix = tier ? ` ${uiTheme.fg("dim", `(${tier})`)}` : "";
|
|
1558
|
+
lines.push(
|
|
1559
|
+
`${uiTheme.fg("success", uiTheme.status.success)} ${label}${tierSuffix} ${uiTheme.fg("dim", "-- no limits")}`,
|
|
1560
|
+
);
|
|
1561
|
+
}
|
|
1562
|
+
// No per-provider footer; global header shows last check.
|
|
1563
|
+
}
|
|
1564
|
+
|
|
1565
|
+
return lines.join("\n");
|
|
1566
|
+
}
|