@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
package/src/utils/git.ts
ADDED
|
@@ -0,0 +1,1833 @@
|
|
|
1
|
+
import * as fs from "node:fs";
|
|
2
|
+
import * as os from "node:os";
|
|
3
|
+
import * as path from "node:path";
|
|
4
|
+
import { $which, hasFsCode, isEnoent, Snowflake } from "@oh-my-pi/pi-utils";
|
|
5
|
+
import {
|
|
6
|
+
parseDiffHunks as parseCommitDiffHunks,
|
|
7
|
+
parseFileDiffs,
|
|
8
|
+
parseFileHunks,
|
|
9
|
+
parseNumstat,
|
|
10
|
+
} from "../commit/git/diff";
|
|
11
|
+
import type { FileDiff, FileHunks, NumstatEntry } from "../commit/types";
|
|
12
|
+
import { ToolAbortError, ToolError, throwIfAborted } from "../tools/tool-errors";
|
|
13
|
+
|
|
14
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
15
|
+
// Types
|
|
16
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
17
|
+
|
|
18
|
+
export interface GitCommandResult {
|
|
19
|
+
exitCode: number;
|
|
20
|
+
stdout: string;
|
|
21
|
+
stderr: string;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
export interface GitRepository {
|
|
25
|
+
commonDir: string;
|
|
26
|
+
gitDir: string;
|
|
27
|
+
gitEntryPath: string;
|
|
28
|
+
headPath: string;
|
|
29
|
+
repoRoot: string;
|
|
30
|
+
isReftable?: boolean;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
export interface GitStatusSummary {
|
|
34
|
+
staged: number;
|
|
35
|
+
unstaged: number;
|
|
36
|
+
untracked: number;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
export type HunkSelection = {
|
|
40
|
+
path: string;
|
|
41
|
+
hunks: { type: "all" } | { type: "indices"; indices: number[] } | { type: "lines"; start: number; end: number };
|
|
42
|
+
};
|
|
43
|
+
|
|
44
|
+
export interface StageHunksOptions {
|
|
45
|
+
readonly diffCached?: boolean;
|
|
46
|
+
readonly rawDiff?: string;
|
|
47
|
+
readonly signal?: AbortSignal;
|
|
48
|
+
}
|
|
49
|
+
export interface HunkSelectionValidationError {
|
|
50
|
+
readonly path: string;
|
|
51
|
+
readonly message: string;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
export interface DiffOptions {
|
|
55
|
+
readonly allowFailure?: boolean;
|
|
56
|
+
readonly base?: string;
|
|
57
|
+
readonly binary?: boolean;
|
|
58
|
+
readonly cached?: boolean;
|
|
59
|
+
readonly env?: Record<string, string | undefined>;
|
|
60
|
+
readonly files?: readonly string[];
|
|
61
|
+
readonly head?: string;
|
|
62
|
+
readonly nameOnly?: boolean;
|
|
63
|
+
readonly noIndex?: { left: string; right: string };
|
|
64
|
+
readonly numstat?: boolean;
|
|
65
|
+
readonly signal?: AbortSignal;
|
|
66
|
+
readonly stat?: boolean;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
export interface StatusOptions {
|
|
70
|
+
readonly pathspecs?: readonly string[];
|
|
71
|
+
readonly porcelainV1?: boolean;
|
|
72
|
+
readonly signal?: AbortSignal;
|
|
73
|
+
readonly untrackedFiles?: "all" | "no" | "normal";
|
|
74
|
+
readonly z?: boolean;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
export interface CommitOptions {
|
|
78
|
+
readonly allowEmpty?: boolean;
|
|
79
|
+
readonly files?: readonly string[];
|
|
80
|
+
readonly signal?: AbortSignal;
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
export interface PushOptions {
|
|
84
|
+
readonly forceWithLease?: boolean;
|
|
85
|
+
readonly refspec?: string;
|
|
86
|
+
readonly remote?: string;
|
|
87
|
+
readonly signal?: AbortSignal;
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
export interface PatchOptions {
|
|
91
|
+
readonly cached?: boolean;
|
|
92
|
+
readonly check?: boolean;
|
|
93
|
+
readonly env?: Record<string, string | undefined>;
|
|
94
|
+
readonly signal?: AbortSignal;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
export interface RestoreOptions {
|
|
98
|
+
readonly files?: readonly string[];
|
|
99
|
+
readonly signal?: AbortSignal;
|
|
100
|
+
readonly source?: string;
|
|
101
|
+
readonly staged?: boolean;
|
|
102
|
+
readonly worktree?: boolean;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
export interface CloneOptions {
|
|
106
|
+
readonly ref?: string;
|
|
107
|
+
readonly sha?: string;
|
|
108
|
+
readonly signal?: AbortSignal;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
interface GitHeadBase extends GitRepository {
|
|
112
|
+
headContent: string;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
export interface GitRefHead extends GitHeadBase {
|
|
116
|
+
branchName: string | null;
|
|
117
|
+
commit: string | null;
|
|
118
|
+
kind: "ref";
|
|
119
|
+
ref: string;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
export interface GitDetachedHead extends GitHeadBase {
|
|
123
|
+
commit: string | null;
|
|
124
|
+
kind: "detached";
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
export type GitHeadState = GitRefHead | GitDetachedHead;
|
|
128
|
+
|
|
129
|
+
export interface GitWorktreeEntry {
|
|
130
|
+
branch?: string;
|
|
131
|
+
detached: boolean;
|
|
132
|
+
head?: string;
|
|
133
|
+
path: string;
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
137
|
+
// Error
|
|
138
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
139
|
+
|
|
140
|
+
export class GitCommandError extends Error {
|
|
141
|
+
readonly args: readonly string[];
|
|
142
|
+
readonly result: GitCommandResult;
|
|
143
|
+
|
|
144
|
+
constructor(args: readonly string[], result: GitCommandResult) {
|
|
145
|
+
super(formatCommandFailure(args, result));
|
|
146
|
+
this.name = "GitCommandError";
|
|
147
|
+
this.args = [...args];
|
|
148
|
+
this.result = result;
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
153
|
+
// Internal: Core execution
|
|
154
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
155
|
+
|
|
156
|
+
const NO_OPTIONAL_LOCKS = "--no-optional-locks";
|
|
157
|
+
const HEAD_REF_PREFIX = "ref:";
|
|
158
|
+
const LOCAL_BRANCH_PREFIX = "refs/heads/";
|
|
159
|
+
const DEFAULT_BRANCH_REFS = ["refs/remotes/origin/HEAD", "refs/remotes/upstream/HEAD"] as const;
|
|
160
|
+
const SHORT_LIVED_GIT_CONFIG: readonly (readonly [key: string, value: string])[] = [
|
|
161
|
+
["core.fsmonitor", "false"],
|
|
162
|
+
["core.untrackedCache", "false"],
|
|
163
|
+
];
|
|
164
|
+
const REMOTE_ALREADY_EXISTS = /remote .* already exists/i;
|
|
165
|
+
|
|
166
|
+
interface CommandOptions {
|
|
167
|
+
readonly env?: Record<string, string | undefined>;
|
|
168
|
+
readonly readOnly?: boolean;
|
|
169
|
+
readonly signal?: AbortSignal;
|
|
170
|
+
readonly stdin?: string | Uint8Array | ArrayBuffer | SharedArrayBuffer;
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
function normalizeStdin(input: CommandOptions["stdin"]): "ignore" | Uint8Array {
|
|
174
|
+
if (input === undefined) return "ignore";
|
|
175
|
+
if (typeof input === "string") return new TextEncoder().encode(input);
|
|
176
|
+
if (input instanceof Uint8Array) return input;
|
|
177
|
+
return new Uint8Array(input);
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
function ensureAvailable(): void {
|
|
181
|
+
if (!$which("git")) {
|
|
182
|
+
throw new Error("git is not installed.");
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
function formatCommandFailure(
|
|
187
|
+
args: readonly string[],
|
|
188
|
+
result: Pick<GitCommandResult, "exitCode" | "stdout" | "stderr">,
|
|
189
|
+
): string {
|
|
190
|
+
const stderr = result.stderr.trim();
|
|
191
|
+
if (stderr) return stderr;
|
|
192
|
+
const stdout = result.stdout.trim();
|
|
193
|
+
if (stdout) return stdout;
|
|
194
|
+
return `git ${args.join(" ")} failed with exit code ${result.exitCode}`;
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
async function git(cwd: string, args: readonly string[], options: CommandOptions = {}): Promise<GitCommandResult> {
|
|
198
|
+
const commandArgs = withShortLivedGitConfig(options.readOnly ? withNoOptionalLocks(args) : [...args]);
|
|
199
|
+
const child = Bun.spawn(["git", ...commandArgs], {
|
|
200
|
+
cwd,
|
|
201
|
+
env: options.env ? { ...process.env, GIT_OPTIONAL_LOCKS: "0", ...options.env } : undefined,
|
|
202
|
+
signal: options.signal,
|
|
203
|
+
stdin: normalizeStdin(options.stdin),
|
|
204
|
+
stdout: "pipe",
|
|
205
|
+
stderr: "pipe",
|
|
206
|
+
windowsHide: true,
|
|
207
|
+
});
|
|
208
|
+
|
|
209
|
+
if (!child.stdout || !child.stderr) {
|
|
210
|
+
throw new Error("Failed to capture git command output.");
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
const [stdout, stderr, exitCode] = await Promise.all([
|
|
214
|
+
new Response(child.stdout).text(),
|
|
215
|
+
new Response(child.stderr).text(),
|
|
216
|
+
child.exited,
|
|
217
|
+
]);
|
|
218
|
+
|
|
219
|
+
return { exitCode: exitCode ?? 0, stdout, stderr };
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
function withNoOptionalLocks(args: readonly string[]): string[] {
|
|
223
|
+
if (args.includes(NO_OPTIONAL_LOCKS)) return [...args];
|
|
224
|
+
return [NO_OPTIONAL_LOCKS, ...args];
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
function withShortLivedGitConfig(args: readonly string[]): string[] {
|
|
228
|
+
const prefix: string[] = [];
|
|
229
|
+
for (const [key, value] of SHORT_LIVED_GIT_CONFIG) {
|
|
230
|
+
if (hasGitConfig(args, key, value)) continue;
|
|
231
|
+
prefix.push("-c", `${key}=${value}`);
|
|
232
|
+
}
|
|
233
|
+
return [...prefix, ...args];
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
function hasGitConfig(args: readonly string[], key: string, value: string): boolean {
|
|
237
|
+
const expected = `${key}=${value}`;
|
|
238
|
+
for (let index = 0; index < args.length - 1; index += 1) {
|
|
239
|
+
if (args[index] === "-c" && args[index + 1] === expected) {
|
|
240
|
+
return true;
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
return false;
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
async function runChecked(
|
|
247
|
+
cwd: string,
|
|
248
|
+
args: readonly string[],
|
|
249
|
+
options: CommandOptions = {},
|
|
250
|
+
): Promise<GitCommandResult> {
|
|
251
|
+
ensureAvailable();
|
|
252
|
+
const result = await git(cwd, args, options);
|
|
253
|
+
if (result.exitCode !== 0) {
|
|
254
|
+
throw new GitCommandError(args, result);
|
|
255
|
+
}
|
|
256
|
+
return result;
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
async function runEffect(cwd: string, args: readonly string[], options: CommandOptions = {}): Promise<void> {
|
|
260
|
+
await runChecked(cwd, args, options);
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
async function runText(cwd: string, args: readonly string[], options: CommandOptions = {}): Promise<string> {
|
|
264
|
+
return (await runChecked(cwd, args, options)).stdout;
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
async function tryText(
|
|
268
|
+
cwd: string,
|
|
269
|
+
args: readonly string[],
|
|
270
|
+
options: CommandOptions = {},
|
|
271
|
+
): Promise<string | undefined> {
|
|
272
|
+
ensureAvailable();
|
|
273
|
+
const result = await git(cwd, args, options);
|
|
274
|
+
if (result.exitCode !== 0) return undefined;
|
|
275
|
+
return result.stdout;
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
279
|
+
// Internal: per-repo write serialization
|
|
280
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
281
|
+
|
|
282
|
+
// Git uses lock files (`.git/config.lock`, commit-graph chain locks,
|
|
283
|
+
// `packed-refs.lock`, …) for many of its mutating operations. Each is created
|
|
284
|
+
// O_EXCL with no waiter, so concurrent in-process git invocations against the
|
|
285
|
+
// same repository fail immediately rather than block. Worktrees share the
|
|
286
|
+
// primary repo's `.git` directory, so racing across worktrees has the same
|
|
287
|
+
// failure mode. We give callers a single per-repo serialization point keyed by
|
|
288
|
+
// the primary repo root: any block that mutates repo state should hold this
|
|
289
|
+
// lock so unrelated callers cannot collide on git's internal locks.
|
|
290
|
+
const repoWriteChain = new Map<string, Promise<unknown>>();
|
|
291
|
+
|
|
292
|
+
/**
|
|
293
|
+
* Serialize an async block that mutates a git repository against other
|
|
294
|
+
* in-process callers operating on the same repository. The lock is keyed by
|
|
295
|
+
* the primary repo root so worktrees of the same repo share a single queue.
|
|
296
|
+
* Failures in one block do not poison the queue for the next caller.
|
|
297
|
+
*
|
|
298
|
+
* Not reentrant: do NOT nest acquisitions for the same repo. Helpers in this
|
|
299
|
+
* module never auto-acquire — callers wrap the critical section themselves.
|
|
300
|
+
*/
|
|
301
|
+
export async function withRepoLock<T>(cwd: string, fn: () => Promise<T>, signal?: AbortSignal): Promise<T> {
|
|
302
|
+
const key = (await repo.primaryRoot(cwd, signal)) ?? cwd;
|
|
303
|
+
const prior = repoWriteChain.get(key);
|
|
304
|
+
const run = (async () => {
|
|
305
|
+
if (prior) {
|
|
306
|
+
try {
|
|
307
|
+
await prior;
|
|
308
|
+
} catch {
|
|
309
|
+
// A prior caller failing must not block us from running.
|
|
310
|
+
}
|
|
311
|
+
}
|
|
312
|
+
throwIfAborted(signal);
|
|
313
|
+
return fn();
|
|
314
|
+
})();
|
|
315
|
+
repoWriteChain.set(key, run);
|
|
316
|
+
try {
|
|
317
|
+
return await run;
|
|
318
|
+
} finally {
|
|
319
|
+
if (repoWriteChain.get(key) === run) repoWriteChain.delete(key);
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
|
|
323
|
+
function splitLines(text: string): string[] {
|
|
324
|
+
return text
|
|
325
|
+
.split("\n")
|
|
326
|
+
.map(line => line.trim())
|
|
327
|
+
.filter(Boolean);
|
|
328
|
+
}
|
|
329
|
+
|
|
330
|
+
function trimScalar(text: string | undefined): string | undefined {
|
|
331
|
+
const trimmed = text?.trim();
|
|
332
|
+
return trimmed || undefined;
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
336
|
+
// Internal: Argument builders
|
|
337
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
338
|
+
|
|
339
|
+
function buildDiffArgs(options: DiffOptions): string[] {
|
|
340
|
+
const args = ["diff"];
|
|
341
|
+
if (options.binary) args.push("--binary");
|
|
342
|
+
if (options.cached) args.push("--cached");
|
|
343
|
+
if (options.nameOnly) args.push("--name-only");
|
|
344
|
+
if (options.stat) args.push("--stat");
|
|
345
|
+
if (options.numstat) args.push("--numstat");
|
|
346
|
+
if (options.noIndex) {
|
|
347
|
+
args.push("--no-index", options.noIndex.left, options.noIndex.right);
|
|
348
|
+
return args;
|
|
349
|
+
}
|
|
350
|
+
if (options.base) {
|
|
351
|
+
args.push(options.base);
|
|
352
|
+
if (options.head) args.push(options.head);
|
|
353
|
+
}
|
|
354
|
+
if (options.files?.length) args.push("--", ...options.files);
|
|
355
|
+
return args;
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
function buildApplyArgs(patchPath: string, options: PatchOptions): string[] {
|
|
359
|
+
const args = ["apply"];
|
|
360
|
+
if (options.check) args.push("--check");
|
|
361
|
+
if (options.cached) args.push("--cached");
|
|
362
|
+
args.push("--binary", patchPath);
|
|
363
|
+
return args;
|
|
364
|
+
}
|
|
365
|
+
|
|
366
|
+
async function writeTempPatch(content: string): Promise<string> {
|
|
367
|
+
const tempPath = path.join(os.tmpdir(), `omp-git-patch-${Snowflake.next()}.patch`);
|
|
368
|
+
await Bun.write(tempPath, content);
|
|
369
|
+
return tempPath;
|
|
370
|
+
}
|
|
371
|
+
|
|
372
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
373
|
+
// Internal: Repository resolution
|
|
374
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
375
|
+
|
|
376
|
+
type EntryType = "directory" | "file";
|
|
377
|
+
|
|
378
|
+
function shouldRetry(err: unknown, n: number) {
|
|
379
|
+
if (isEnoent(err) || hasFsCode(err, "ENFILE") || hasFsCode(err, "EMFILE")) return false;
|
|
380
|
+
if (hasFsCode(err, "EINTR")) return n < EINTR_MAX_RETRIES;
|
|
381
|
+
if (n > EINTR_MAX_RETRIES) throw err;
|
|
382
|
+
throw err;
|
|
383
|
+
}
|
|
384
|
+
|
|
385
|
+
/**
|
|
386
|
+
* Bounded retry for synchronous I/O against `EINTR`. POSIX permits short syscalls
|
|
387
|
+
* to be interrupted by signals; when that happens libc traditionally retries.
|
|
388
|
+
* Node's sync wrappers surface the raw `EINTR` so we replicate the retry locally.
|
|
389
|
+
* Any other error (and persistent EINTR after `EINTR_MAX_RETRIES`) is rethrown
|
|
390
|
+
* for the caller's normal "optional metadata" classifier to handle.
|
|
391
|
+
*/
|
|
392
|
+
const EINTR_MAX_RETRIES = 3;
|
|
393
|
+
function retryOnEintrSync<T>(op: () => T): T | null {
|
|
394
|
+
for (let attempt = 0; attempt <= EINTR_MAX_RETRIES; attempt += 1) {
|
|
395
|
+
try {
|
|
396
|
+
return op();
|
|
397
|
+
} catch (err) {
|
|
398
|
+
if (shouldRetry(err, attempt)) continue;
|
|
399
|
+
return null;
|
|
400
|
+
}
|
|
401
|
+
}
|
|
402
|
+
throw new Error("retryOnEintrSync: exhausted without resolution");
|
|
403
|
+
}
|
|
404
|
+
async function retryOnEintr<T>(op: () => Promise<T>): Promise<T | null> {
|
|
405
|
+
for (let attempt = 0; attempt <= EINTR_MAX_RETRIES; attempt += 1) {
|
|
406
|
+
try {
|
|
407
|
+
return await op();
|
|
408
|
+
} catch (err) {
|
|
409
|
+
if (shouldRetry(err, attempt)) continue;
|
|
410
|
+
return null;
|
|
411
|
+
}
|
|
412
|
+
}
|
|
413
|
+
throw new Error("retryOnEintr: exhausted without resolution");
|
|
414
|
+
}
|
|
415
|
+
|
|
416
|
+
function getEntryTypeSync(gitEntryPath: string): EntryType | null {
|
|
417
|
+
return retryOnEintrSync(() => {
|
|
418
|
+
const stat = fs.statSync(gitEntryPath);
|
|
419
|
+
if (stat.isDirectory()) return "directory";
|
|
420
|
+
if (stat.isFile()) return "file";
|
|
421
|
+
return null;
|
|
422
|
+
});
|
|
423
|
+
}
|
|
424
|
+
|
|
425
|
+
async function getEntryType(gitEntryPath: string): Promise<EntryType | null> {
|
|
426
|
+
return retryOnEintr(async () => {
|
|
427
|
+
const stat = await fs.promises.stat(gitEntryPath);
|
|
428
|
+
if (stat.isDirectory()) return "directory";
|
|
429
|
+
if (stat.isFile()) return "file";
|
|
430
|
+
return null;
|
|
431
|
+
});
|
|
432
|
+
}
|
|
433
|
+
|
|
434
|
+
function readOptionalTextSync(filePath: string): string | null {
|
|
435
|
+
return retryOnEintrSync(() => fs.readFileSync(filePath, "utf8"));
|
|
436
|
+
}
|
|
437
|
+
|
|
438
|
+
async function readOptionalText(filePath: string): Promise<string | null> {
|
|
439
|
+
return retryOnEintr(async () => await Bun.file(filePath).text());
|
|
440
|
+
}
|
|
441
|
+
|
|
442
|
+
function parseGitDirPointer(content: string): string | null {
|
|
443
|
+
const match = /^gitdir:\s*(.+)\s*$/iu.exec(content.trim());
|
|
444
|
+
return match?.[1] ?? null;
|
|
445
|
+
}
|
|
446
|
+
|
|
447
|
+
function resolveGitDirSync(gitEntryPath: string, entryType: EntryType): string | null {
|
|
448
|
+
if (entryType === "directory") return gitEntryPath;
|
|
449
|
+
const content = readOptionalTextSync(gitEntryPath);
|
|
450
|
+
if (content === null) return null;
|
|
451
|
+
const parsed = parseGitDirPointer(content);
|
|
452
|
+
if (!parsed) return null;
|
|
453
|
+
const gitDir = path.resolve(path.dirname(gitEntryPath), parsed);
|
|
454
|
+
return getEntryTypeSync(gitDir) === "directory" ? gitDir : null;
|
|
455
|
+
}
|
|
456
|
+
|
|
457
|
+
async function resolveGitDir(gitEntryPath: string, entryType: EntryType): Promise<string | null> {
|
|
458
|
+
if (entryType === "directory") return gitEntryPath;
|
|
459
|
+
const content = await readOptionalText(gitEntryPath);
|
|
460
|
+
if (content === null) return null;
|
|
461
|
+
const parsed = parseGitDirPointer(content);
|
|
462
|
+
if (!parsed) return null;
|
|
463
|
+
const gitDir = path.resolve(path.dirname(gitEntryPath), parsed);
|
|
464
|
+
return (await getEntryType(gitDir)) === "directory" ? gitDir : null;
|
|
465
|
+
}
|
|
466
|
+
|
|
467
|
+
function resolveCommonDirSync(gitDir: string): string {
|
|
468
|
+
const content = readOptionalTextSync(path.join(gitDir, "commondir"));
|
|
469
|
+
const relative = content?.trim();
|
|
470
|
+
if (!relative) return gitDir;
|
|
471
|
+
return path.resolve(gitDir, relative);
|
|
472
|
+
}
|
|
473
|
+
|
|
474
|
+
async function resolveCommonDir(gitDir: string): Promise<string> {
|
|
475
|
+
const content = await readOptionalText(path.join(gitDir, "commondir"));
|
|
476
|
+
const relative = content?.trim();
|
|
477
|
+
if (!relative) return gitDir;
|
|
478
|
+
return path.resolve(gitDir, relative);
|
|
479
|
+
}
|
|
480
|
+
function isLinkedWorktree(repository: GitRepository): boolean {
|
|
481
|
+
return (
|
|
482
|
+
repository.gitDir !== repository.commonDir &&
|
|
483
|
+
getEntryTypeSync(path.join(repository.gitDir, "commondir")) === "file"
|
|
484
|
+
);
|
|
485
|
+
}
|
|
486
|
+
|
|
487
|
+
async function isLinkedWorktreeAsync(repository: GitRepository): Promise<boolean> {
|
|
488
|
+
return (
|
|
489
|
+
repository.gitDir !== repository.commonDir &&
|
|
490
|
+
(await getEntryType(path.join(repository.gitDir, "commondir"))) === "file"
|
|
491
|
+
);
|
|
492
|
+
}
|
|
493
|
+
|
|
494
|
+
function primaryRootFromRepositorySync(repository: GitRepository): string {
|
|
495
|
+
if (path.basename(repository.commonDir) === ".git") return path.dirname(repository.commonDir);
|
|
496
|
+
if (isLinkedWorktree(repository)) return repository.commonDir;
|
|
497
|
+
return repository.repoRoot;
|
|
498
|
+
}
|
|
499
|
+
|
|
500
|
+
async function primaryRootFromRepository(repository: GitRepository): Promise<string> {
|
|
501
|
+
if (path.basename(repository.commonDir) === ".git") return path.dirname(repository.commonDir);
|
|
502
|
+
if (await isLinkedWorktreeAsync(repository)) return repository.commonDir;
|
|
503
|
+
return repository.repoRoot;
|
|
504
|
+
}
|
|
505
|
+
|
|
506
|
+
function resolveRepoFromEntrySync(repoRoot: string, gitEntryPath: string, entryType: EntryType): GitRepository | null {
|
|
507
|
+
const gitDir = resolveGitDirSync(gitEntryPath, entryType);
|
|
508
|
+
if (!gitDir) return null;
|
|
509
|
+
return {
|
|
510
|
+
commonDir: resolveCommonDirSync(gitDir),
|
|
511
|
+
gitDir,
|
|
512
|
+
gitEntryPath,
|
|
513
|
+
headPath: path.join(gitDir, "HEAD"),
|
|
514
|
+
repoRoot,
|
|
515
|
+
};
|
|
516
|
+
}
|
|
517
|
+
|
|
518
|
+
async function resolveRepoFromEntry(
|
|
519
|
+
repoRoot: string,
|
|
520
|
+
gitEntryPath: string,
|
|
521
|
+
entryType: EntryType,
|
|
522
|
+
): Promise<GitRepository | null> {
|
|
523
|
+
const gitDir = await resolveGitDir(gitEntryPath, entryType);
|
|
524
|
+
if (!gitDir) return null;
|
|
525
|
+
return {
|
|
526
|
+
commonDir: await resolveCommonDir(gitDir),
|
|
527
|
+
gitDir,
|
|
528
|
+
gitEntryPath,
|
|
529
|
+
headPath: path.join(gitDir, "HEAD"),
|
|
530
|
+
repoRoot,
|
|
531
|
+
};
|
|
532
|
+
}
|
|
533
|
+
|
|
534
|
+
function resolveRepositorySync(startDir: string): GitRepository | null {
|
|
535
|
+
let current = path.resolve(startDir);
|
|
536
|
+
while (true) {
|
|
537
|
+
const gitEntryPath = path.join(current, ".git");
|
|
538
|
+
const entryType = getEntryTypeSync(gitEntryPath);
|
|
539
|
+
if (entryType) {
|
|
540
|
+
const repository = resolveRepoFromEntrySync(current, gitEntryPath, entryType);
|
|
541
|
+
if (repository) return repository;
|
|
542
|
+
}
|
|
543
|
+
const parent = path.dirname(current);
|
|
544
|
+
if (parent === current) return null;
|
|
545
|
+
current = parent;
|
|
546
|
+
}
|
|
547
|
+
}
|
|
548
|
+
|
|
549
|
+
async function resolveRepository(startDir: string): Promise<GitRepository | null> {
|
|
550
|
+
let current = path.resolve(startDir);
|
|
551
|
+
while (true) {
|
|
552
|
+
const gitEntryPath = path.join(current, ".git");
|
|
553
|
+
const entryType = await getEntryType(gitEntryPath);
|
|
554
|
+
if (entryType) {
|
|
555
|
+
const repository = await resolveRepoFromEntry(current, gitEntryPath, entryType);
|
|
556
|
+
if (repository) return repository;
|
|
557
|
+
}
|
|
558
|
+
const parent = path.dirname(current);
|
|
559
|
+
if (parent === current) return null;
|
|
560
|
+
current = parent;
|
|
561
|
+
}
|
|
562
|
+
}
|
|
563
|
+
|
|
564
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
565
|
+
// Internal: Ref resolution
|
|
566
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
567
|
+
|
|
568
|
+
function getRefLookupDirs(repository: GitRepository): string[] {
|
|
569
|
+
if (repository.gitDir === repository.commonDir) return [repository.gitDir];
|
|
570
|
+
return [repository.gitDir, repository.commonDir];
|
|
571
|
+
}
|
|
572
|
+
|
|
573
|
+
function normalizeRefValue(content: string | null): string | null {
|
|
574
|
+
const trimmed = content?.trim() ?? "";
|
|
575
|
+
return trimmed || null;
|
|
576
|
+
}
|
|
577
|
+
|
|
578
|
+
function parsePackedRefs(content: string | null, targetRef: string): string | null {
|
|
579
|
+
if (!content) return null;
|
|
580
|
+
for (const line of content.split("\n")) {
|
|
581
|
+
const trimmed = line.trim();
|
|
582
|
+
if (!trimmed || trimmed.startsWith("#") || trimmed.startsWith("^")) continue;
|
|
583
|
+
const [sha, refName] = trimmed.split(" ", 2);
|
|
584
|
+
if (refName === targetRef && sha) return sha;
|
|
585
|
+
}
|
|
586
|
+
return null;
|
|
587
|
+
}
|
|
588
|
+
|
|
589
|
+
function stripGitConfigComments(line: string): string {
|
|
590
|
+
let clean = "";
|
|
591
|
+
let inQuotes = false;
|
|
592
|
+
for (let i = 0; i < line.length; i++) {
|
|
593
|
+
const char = line[i];
|
|
594
|
+
if (char === '"') {
|
|
595
|
+
inQuotes = !inQuotes;
|
|
596
|
+
clean += char;
|
|
597
|
+
} else if (!inQuotes && (char === ";" || char === "#")) {
|
|
598
|
+
break;
|
|
599
|
+
} else {
|
|
600
|
+
clean += char;
|
|
601
|
+
}
|
|
602
|
+
}
|
|
603
|
+
return clean.trim();
|
|
604
|
+
}
|
|
605
|
+
|
|
606
|
+
function parseGitConfigHasReftable(content: string): boolean {
|
|
607
|
+
let inExtensions = false;
|
|
608
|
+
for (const line of content.split("\n")) {
|
|
609
|
+
const trimmed = stripGitConfigComments(line);
|
|
610
|
+
if (trimmed.startsWith("[") && trimmed.endsWith("]")) {
|
|
611
|
+
const section = trimmed.slice(1, -1).trim().toLowerCase();
|
|
612
|
+
inExtensions = section === "extensions";
|
|
613
|
+
} else if (inExtensions) {
|
|
614
|
+
const eqIndex = trimmed.indexOf("=");
|
|
615
|
+
if (eqIndex !== -1) {
|
|
616
|
+
const key = trimmed.slice(0, eqIndex).trim().toLowerCase();
|
|
617
|
+
let value = trimmed.slice(eqIndex + 1).trim();
|
|
618
|
+
if (key === "refstorage") {
|
|
619
|
+
if (value.startsWith('"') && value.endsWith('"')) {
|
|
620
|
+
value = value.slice(1, -1).trim();
|
|
621
|
+
}
|
|
622
|
+
const lowerValue = value.toLowerCase();
|
|
623
|
+
if (lowerValue === "reftable" || lowerValue.startsWith("reftable:")) {
|
|
624
|
+
return true;
|
|
625
|
+
}
|
|
626
|
+
}
|
|
627
|
+
}
|
|
628
|
+
}
|
|
629
|
+
}
|
|
630
|
+
return false;
|
|
631
|
+
}
|
|
632
|
+
|
|
633
|
+
function isReftableRepoSync(repository: GitRepository): boolean {
|
|
634
|
+
if (repository.isReftable !== undefined) return repository.isReftable;
|
|
635
|
+
const configPath = path.join(repository.commonDir, "config");
|
|
636
|
+
const content = readOptionalTextSync(configPath);
|
|
637
|
+
repository.isReftable = content ? parseGitConfigHasReftable(content) : false;
|
|
638
|
+
return repository.isReftable;
|
|
639
|
+
}
|
|
640
|
+
|
|
641
|
+
async function isReftableRepo(repository: GitRepository): Promise<boolean> {
|
|
642
|
+
if (repository.isReftable !== undefined) return repository.isReftable;
|
|
643
|
+
const configPath = path.join(repository.commonDir, "config");
|
|
644
|
+
const content = await readOptionalText(configPath);
|
|
645
|
+
repository.isReftable = content ? parseGitConfigHasReftable(content) : false;
|
|
646
|
+
return repository.isReftable;
|
|
647
|
+
}
|
|
648
|
+
|
|
649
|
+
async function resolveHeadStateReftable(repository: GitRepository, signal?: AbortSignal): Promise<GitHeadState | null> {
|
|
650
|
+
throwIfAborted(signal);
|
|
651
|
+
const symResult = await git(repository.repoRoot, ["symbolic-ref", "HEAD"], { readOnly: true, signal }).catch(err => {
|
|
652
|
+
if (signal?.aborted || (err instanceof Error && (err.name === "AbortError" || err.name === "ToolAbortError"))) {
|
|
653
|
+
throw err;
|
|
654
|
+
}
|
|
655
|
+
return null;
|
|
656
|
+
});
|
|
657
|
+
throwIfAborted(signal);
|
|
658
|
+
const revResult = await git(repository.repoRoot, ["rev-parse", "--verify", "HEAD"], {
|
|
659
|
+
readOnly: true,
|
|
660
|
+
signal,
|
|
661
|
+
}).catch(err => {
|
|
662
|
+
if (signal?.aborted || (err instanceof Error && (err.name === "AbortError" || err.name === "ToolAbortError"))) {
|
|
663
|
+
throw err;
|
|
664
|
+
}
|
|
665
|
+
return null;
|
|
666
|
+
});
|
|
667
|
+
const commit = revResult && revResult.exitCode === 0 ? revResult.stdout.trim() || null : null;
|
|
668
|
+
|
|
669
|
+
if (symResult && symResult.exitCode === 0) {
|
|
670
|
+
const ref = symResult.stdout.trim();
|
|
671
|
+
const branchName = ref.startsWith(LOCAL_BRANCH_PREFIX) ? ref.slice(LOCAL_BRANCH_PREFIX.length) : null;
|
|
672
|
+
return {
|
|
673
|
+
...repository,
|
|
674
|
+
kind: "ref",
|
|
675
|
+
ref,
|
|
676
|
+
branchName,
|
|
677
|
+
commit,
|
|
678
|
+
headContent: `${HEAD_REF_PREFIX} ${ref}`,
|
|
679
|
+
};
|
|
680
|
+
}
|
|
681
|
+
|
|
682
|
+
return {
|
|
683
|
+
...repository,
|
|
684
|
+
kind: "detached",
|
|
685
|
+
commit,
|
|
686
|
+
headContent: commit || "",
|
|
687
|
+
};
|
|
688
|
+
}
|
|
689
|
+
|
|
690
|
+
function resolveHeadStateReftableSync(repository: GitRepository): GitHeadState | null {
|
|
691
|
+
ensureAvailable();
|
|
692
|
+
const symArgs = withShortLivedGitConfig(withNoOptionalLocks(["symbolic-ref", "HEAD"]));
|
|
693
|
+
const symResult = Bun.spawnSync(["git", ...symArgs], {
|
|
694
|
+
cwd: repository.repoRoot,
|
|
695
|
+
stdout: "pipe",
|
|
696
|
+
stderr: "pipe",
|
|
697
|
+
windowsHide: true,
|
|
698
|
+
});
|
|
699
|
+
|
|
700
|
+
const revArgs = withShortLivedGitConfig(withNoOptionalLocks(["rev-parse", "--verify", "HEAD"]));
|
|
701
|
+
const revResult = Bun.spawnSync(["git", ...revArgs], {
|
|
702
|
+
cwd: repository.repoRoot,
|
|
703
|
+
stdout: "pipe",
|
|
704
|
+
stderr: "pipe",
|
|
705
|
+
windowsHide: true,
|
|
706
|
+
});
|
|
707
|
+
const commit = revResult.exitCode === 0 ? new TextDecoder().decode(revResult.stdout).trim() || null : null;
|
|
708
|
+
|
|
709
|
+
if (symResult.exitCode === 0) {
|
|
710
|
+
const ref = new TextDecoder().decode(symResult.stdout).trim();
|
|
711
|
+
const branchName = ref.startsWith(LOCAL_BRANCH_PREFIX) ? ref.slice(LOCAL_BRANCH_PREFIX.length) : null;
|
|
712
|
+
return {
|
|
713
|
+
...repository,
|
|
714
|
+
kind: "ref",
|
|
715
|
+
ref,
|
|
716
|
+
branchName,
|
|
717
|
+
commit,
|
|
718
|
+
headContent: `${HEAD_REF_PREFIX} ${ref}`,
|
|
719
|
+
};
|
|
720
|
+
}
|
|
721
|
+
|
|
722
|
+
return {
|
|
723
|
+
...repository,
|
|
724
|
+
kind: "detached",
|
|
725
|
+
commit,
|
|
726
|
+
headContent: commit || "",
|
|
727
|
+
};
|
|
728
|
+
}
|
|
729
|
+
|
|
730
|
+
function readRefSync(repository: GitRepository, targetRef: string): string | null {
|
|
731
|
+
if (isReftableRepoSync(repository)) {
|
|
732
|
+
ensureAvailable();
|
|
733
|
+
const symArgs = withShortLivedGitConfig(withNoOptionalLocks(["symbolic-ref", targetRef]));
|
|
734
|
+
const symResult = Bun.spawnSync(["git", ...symArgs], {
|
|
735
|
+
cwd: repository.repoRoot,
|
|
736
|
+
stdout: "pipe",
|
|
737
|
+
stderr: "pipe",
|
|
738
|
+
windowsHide: true,
|
|
739
|
+
});
|
|
740
|
+
if (symResult.exitCode === 0) {
|
|
741
|
+
const stdoutText = new TextDecoder().decode(symResult.stdout).trim();
|
|
742
|
+
return `${HEAD_REF_PREFIX} ${stdoutText}`;
|
|
743
|
+
}
|
|
744
|
+
const revArgs = withShortLivedGitConfig(withNoOptionalLocks(["rev-parse", "--verify", targetRef]));
|
|
745
|
+
const revResult = Bun.spawnSync(["git", ...revArgs], {
|
|
746
|
+
cwd: repository.repoRoot,
|
|
747
|
+
stdout: "pipe",
|
|
748
|
+
stderr: "pipe",
|
|
749
|
+
windowsHide: true,
|
|
750
|
+
});
|
|
751
|
+
if (revResult.exitCode === 0) {
|
|
752
|
+
return new TextDecoder().decode(revResult.stdout).trim() || null;
|
|
753
|
+
}
|
|
754
|
+
return null;
|
|
755
|
+
}
|
|
756
|
+
|
|
757
|
+
for (const dir of getRefLookupDirs(repository)) {
|
|
758
|
+
const value = normalizeRefValue(readOptionalTextSync(path.join(dir, targetRef)));
|
|
759
|
+
if (value) return value;
|
|
760
|
+
}
|
|
761
|
+
for (const dir of getRefLookupDirs(repository)) {
|
|
762
|
+
const value = parsePackedRefs(readOptionalTextSync(path.join(dir, "packed-refs")), targetRef);
|
|
763
|
+
if (value) return value;
|
|
764
|
+
}
|
|
765
|
+
return null;
|
|
766
|
+
}
|
|
767
|
+
|
|
768
|
+
async function readRef(repository: GitRepository, targetRef: string, signal?: AbortSignal): Promise<string | null> {
|
|
769
|
+
if (await isReftableRepo(repository)) {
|
|
770
|
+
throwIfAborted(signal);
|
|
771
|
+
const symResult = await git(repository.repoRoot, ["symbolic-ref", targetRef], { readOnly: true, signal }).catch(
|
|
772
|
+
err => {
|
|
773
|
+
if (
|
|
774
|
+
signal?.aborted ||
|
|
775
|
+
(err instanceof Error && (err.name === "AbortError" || err.name === "ToolAbortError"))
|
|
776
|
+
) {
|
|
777
|
+
throw err;
|
|
778
|
+
}
|
|
779
|
+
return null;
|
|
780
|
+
},
|
|
781
|
+
);
|
|
782
|
+
if (symResult && symResult.exitCode === 0) {
|
|
783
|
+
return `${HEAD_REF_PREFIX} ${symResult.stdout.trim()}`;
|
|
784
|
+
}
|
|
785
|
+
throwIfAborted(signal);
|
|
786
|
+
const revResult = await git(repository.repoRoot, ["rev-parse", "--verify", targetRef], {
|
|
787
|
+
readOnly: true,
|
|
788
|
+
signal,
|
|
789
|
+
}).catch(err => {
|
|
790
|
+
if (
|
|
791
|
+
signal?.aborted ||
|
|
792
|
+
(err instanceof Error && (err.name === "AbortError" || err.name === "ToolAbortError"))
|
|
793
|
+
) {
|
|
794
|
+
throw err;
|
|
795
|
+
}
|
|
796
|
+
return null;
|
|
797
|
+
});
|
|
798
|
+
if (revResult && revResult.exitCode === 0) {
|
|
799
|
+
return revResult.stdout.trim() || null;
|
|
800
|
+
}
|
|
801
|
+
return null;
|
|
802
|
+
}
|
|
803
|
+
|
|
804
|
+
for (const dir of getRefLookupDirs(repository)) {
|
|
805
|
+
const value = normalizeRefValue(await readOptionalText(path.join(dir, targetRef)));
|
|
806
|
+
if (value) return value;
|
|
807
|
+
}
|
|
808
|
+
for (const dir of getRefLookupDirs(repository)) {
|
|
809
|
+
const value = parsePackedRefs(await readOptionalText(path.join(dir, "packed-refs")), targetRef);
|
|
810
|
+
if (value) return value;
|
|
811
|
+
}
|
|
812
|
+
return null;
|
|
813
|
+
}
|
|
814
|
+
|
|
815
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
816
|
+
// Internal: Head state parsing
|
|
817
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
818
|
+
|
|
819
|
+
function parseHeadStateSync(repository: GitRepository, headContent: string): GitHeadState {
|
|
820
|
+
const trimmed = headContent.trim();
|
|
821
|
+
if (!trimmed?.startsWith(HEAD_REF_PREFIX)) {
|
|
822
|
+
return { ...repository, commit: trimmed || null, headContent, kind: "detached" };
|
|
823
|
+
}
|
|
824
|
+
const refValue = trimmed.slice(HEAD_REF_PREFIX.length).trim();
|
|
825
|
+
const branchName = refValue.startsWith(LOCAL_BRANCH_PREFIX) ? refValue.slice(LOCAL_BRANCH_PREFIX.length) : null;
|
|
826
|
+
return {
|
|
827
|
+
...repository,
|
|
828
|
+
branchName,
|
|
829
|
+
commit: readRefSync(repository, refValue),
|
|
830
|
+
headContent,
|
|
831
|
+
kind: "ref",
|
|
832
|
+
ref: refValue,
|
|
833
|
+
};
|
|
834
|
+
}
|
|
835
|
+
|
|
836
|
+
async function parseHeadState(repository: GitRepository, headContent: string): Promise<GitHeadState> {
|
|
837
|
+
const trimmed = headContent.trim();
|
|
838
|
+
if (!trimmed?.startsWith(HEAD_REF_PREFIX)) {
|
|
839
|
+
return { ...repository, commit: trimmed || null, headContent, kind: "detached" };
|
|
840
|
+
}
|
|
841
|
+
const refValue = trimmed.slice(HEAD_REF_PREFIX.length).trim();
|
|
842
|
+
const branchName = refValue.startsWith(LOCAL_BRANCH_PREFIX) ? refValue.slice(LOCAL_BRANCH_PREFIX.length) : null;
|
|
843
|
+
return {
|
|
844
|
+
...repository,
|
|
845
|
+
branchName,
|
|
846
|
+
commit: await readRef(repository, refValue),
|
|
847
|
+
headContent,
|
|
848
|
+
kind: "ref",
|
|
849
|
+
ref: refValue,
|
|
850
|
+
};
|
|
851
|
+
}
|
|
852
|
+
|
|
853
|
+
function parseDefaultBranchRef(refPath: string, target: string | null): string | null {
|
|
854
|
+
if (!target?.startsWith(HEAD_REF_PREFIX)) return null;
|
|
855
|
+
const resolvedRef = target.slice(HEAD_REF_PREFIX.length).trim();
|
|
856
|
+
const remotePrefix = refPath.slice(0, -"HEAD".length);
|
|
857
|
+
if (!resolvedRef.startsWith(remotePrefix)) return null;
|
|
858
|
+
return resolvedRef.slice(remotePrefix.length) || null;
|
|
859
|
+
}
|
|
860
|
+
|
|
861
|
+
function stripRemotePrefix(refValue: string): string | null {
|
|
862
|
+
const slash = refValue.indexOf("/");
|
|
863
|
+
if (slash < 0) return refValue || null;
|
|
864
|
+
return refValue.slice(slash + 1) || null;
|
|
865
|
+
}
|
|
866
|
+
|
|
867
|
+
function parseWorktreeList(text: string): GitWorktreeEntry[] {
|
|
868
|
+
const trimmed = text.trim();
|
|
869
|
+
if (!trimmed) return [];
|
|
870
|
+
return trimmed
|
|
871
|
+
.split(/\n\s*\n/)
|
|
872
|
+
.map(block => block.trim())
|
|
873
|
+
.filter(Boolean)
|
|
874
|
+
.map(block => {
|
|
875
|
+
const entry: GitWorktreeEntry = { detached: false, path: "" };
|
|
876
|
+
for (const line of block.split("\n")) {
|
|
877
|
+
if (line.startsWith("worktree ")) entry.path = line.slice("worktree ".length);
|
|
878
|
+
else if (line.startsWith("HEAD ")) entry.head = line.slice("HEAD ".length);
|
|
879
|
+
else if (line.startsWith("branch ")) entry.branch = line.slice("branch ".length);
|
|
880
|
+
else if (line === "detached") entry.detached = true;
|
|
881
|
+
}
|
|
882
|
+
return entry;
|
|
883
|
+
});
|
|
884
|
+
}
|
|
885
|
+
|
|
886
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
887
|
+
// Internal: Hunk selection
|
|
888
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
889
|
+
|
|
890
|
+
function extractFileHeader(diffText: string): string {
|
|
891
|
+
const lines = diffText.split("\n");
|
|
892
|
+
const headerLines: string[] = [];
|
|
893
|
+
for (const line of lines) {
|
|
894
|
+
if (line.startsWith("@@")) break;
|
|
895
|
+
headerLines.push(line);
|
|
896
|
+
}
|
|
897
|
+
return headerLines.join("\n");
|
|
898
|
+
}
|
|
899
|
+
|
|
900
|
+
function selectHunks(file: FileHunks, selector: HunkSelection["hunks"]): FileHunks["hunks"] {
|
|
901
|
+
if (selector.type === "indices") {
|
|
902
|
+
const wanted = new Set(selector.indices.map(v => Math.max(1, Math.floor(v))));
|
|
903
|
+
return file.hunks.filter(hunk => wanted.has(hunk.index + 1));
|
|
904
|
+
}
|
|
905
|
+
if (selector.type === "lines") {
|
|
906
|
+
const start = Math.floor(selector.start);
|
|
907
|
+
const end = Math.floor(selector.end);
|
|
908
|
+
return file.hunks.filter(hunk => hunk.newStart <= end && hunk.newStart + hunk.newLines - 1 >= start);
|
|
909
|
+
}
|
|
910
|
+
return file.hunks;
|
|
911
|
+
}
|
|
912
|
+
|
|
913
|
+
export function createHunkSelectionValidator(
|
|
914
|
+
rawDiff: string,
|
|
915
|
+
): (selections: readonly HunkSelection[]) => HunkSelectionValidationError[] {
|
|
916
|
+
const fileDiffMap = new Map(parseFileDiffs(rawDiff).map(entry => [entry.filename, entry]));
|
|
917
|
+
return selections => validateHunkSelectionsFromMap(fileDiffMap, selections);
|
|
918
|
+
}
|
|
919
|
+
|
|
920
|
+
function validateHunkSelectionsFromMap(
|
|
921
|
+
fileDiffMap: ReadonlyMap<string, FileDiff>,
|
|
922
|
+
selections: readonly HunkSelection[],
|
|
923
|
+
): HunkSelectionValidationError[] {
|
|
924
|
+
const errors: HunkSelectionValidationError[] = [];
|
|
925
|
+
|
|
926
|
+
for (const selection of selections) {
|
|
927
|
+
const fileDiff = fileDiffMap.get(selection.path);
|
|
928
|
+
if (!fileDiff) continue;
|
|
929
|
+
if (selection.hunks.type === "all") continue;
|
|
930
|
+
if (fileDiff.isBinary) {
|
|
931
|
+
errors.push({ path: selection.path, message: `Cannot select hunks for binary file ${selection.path}` });
|
|
932
|
+
continue;
|
|
933
|
+
}
|
|
934
|
+
const selected = selectHunks(parseFileHunks(fileDiff), selection.hunks);
|
|
935
|
+
if (selected.length === 0) {
|
|
936
|
+
errors.push({ path: selection.path, message: `No hunks selected for ${selection.path}` });
|
|
937
|
+
}
|
|
938
|
+
}
|
|
939
|
+
|
|
940
|
+
return errors;
|
|
941
|
+
}
|
|
942
|
+
|
|
943
|
+
export function validateHunkSelections(
|
|
944
|
+
rawDiff: string,
|
|
945
|
+
selections: readonly HunkSelection[],
|
|
946
|
+
): HunkSelectionValidationError[] {
|
|
947
|
+
return createHunkSelectionValidator(rawDiff)(selections);
|
|
948
|
+
}
|
|
949
|
+
|
|
950
|
+
function parseStatusPorcelain(text: string): GitStatusSummary {
|
|
951
|
+
let staged = 0;
|
|
952
|
+
let unstaged = 0;
|
|
953
|
+
let untracked = 0;
|
|
954
|
+
for (const line of text.split("\n")) {
|
|
955
|
+
if (!line) continue;
|
|
956
|
+
const x = line[0];
|
|
957
|
+
const y = line[1];
|
|
958
|
+
if (x === "?" && y === "?") {
|
|
959
|
+
untracked += 1;
|
|
960
|
+
continue;
|
|
961
|
+
}
|
|
962
|
+
if (x && x !== " " && x !== "?") staged += 1;
|
|
963
|
+
if (y && y !== " ") unstaged += 1;
|
|
964
|
+
}
|
|
965
|
+
return { staged, unstaged, untracked };
|
|
966
|
+
}
|
|
967
|
+
|
|
968
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
969
|
+
// API: diff
|
|
970
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
971
|
+
|
|
972
|
+
/** Run `git diff` with the given options. Returns raw diff text. */
|
|
973
|
+
export const diff = Object.assign(
|
|
974
|
+
async function diff(cwd: string, options: DiffOptions = {}): Promise<string> {
|
|
975
|
+
const args = buildDiffArgs(options);
|
|
976
|
+
if (options.allowFailure) {
|
|
977
|
+
return (await git(cwd, args, { env: options.env, readOnly: true, signal: options.signal })).stdout;
|
|
978
|
+
}
|
|
979
|
+
return runText(cwd, args, { env: options.env, readOnly: true, signal: options.signal });
|
|
980
|
+
},
|
|
981
|
+
{
|
|
982
|
+
/** List changed file paths. */
|
|
983
|
+
async changedFiles(
|
|
984
|
+
cwd: string,
|
|
985
|
+
options: Pick<DiffOptions, "cached" | "files" | "signal"> = {},
|
|
986
|
+
): Promise<string[]> {
|
|
987
|
+
return splitLines(await diff(cwd, { ...options, nameOnly: true }));
|
|
988
|
+
},
|
|
989
|
+
/** Parsed per-file add/remove counts. */
|
|
990
|
+
async numstat(cwd: string, options: Pick<DiffOptions, "cached" | "signal"> = {}): Promise<NumstatEntry[]> {
|
|
991
|
+
return parseNumstat(await diff(cwd, { ...options, numstat: true }));
|
|
992
|
+
},
|
|
993
|
+
/** Parsed diff hunks for the given files. */
|
|
994
|
+
async hunks(
|
|
995
|
+
cwd: string,
|
|
996
|
+
files: readonly string[],
|
|
997
|
+
options: { cached?: boolean; signal?: AbortSignal } = {},
|
|
998
|
+
): Promise<FileHunks[]> {
|
|
999
|
+
return parseCommitDiffHunks(
|
|
1000
|
+
await diff(cwd, { cached: options.cached ?? true, files, signal: options.signal }),
|
|
1001
|
+
);
|
|
1002
|
+
},
|
|
1003
|
+
/** Check whether a diff exists (uses `--quiet` for efficiency). */
|
|
1004
|
+
async has(cwd: string, options: Pick<DiffOptions, "cached" | "files" | "signal"> = {}): Promise<boolean> {
|
|
1005
|
+
const args = ["diff"];
|
|
1006
|
+
if (options.cached) args.push("--cached");
|
|
1007
|
+
args.push("--quiet");
|
|
1008
|
+
if (options.files?.length) args.push("--", ...options.files);
|
|
1009
|
+
const result = await git(cwd, args, { readOnly: true, signal: options.signal });
|
|
1010
|
+
if (result.exitCode === 0) return false;
|
|
1011
|
+
if (result.exitCode === 1) return true;
|
|
1012
|
+
throw new GitCommandError(args, result);
|
|
1013
|
+
},
|
|
1014
|
+
/** Diff between two tree-ish objects (`git diff-tree`). */
|
|
1015
|
+
async tree(
|
|
1016
|
+
cwd: string,
|
|
1017
|
+
base: string,
|
|
1018
|
+
headRef: string,
|
|
1019
|
+
options: { binary?: boolean; signal?: AbortSignal; allowFailure?: boolean } = {},
|
|
1020
|
+
): Promise<string> {
|
|
1021
|
+
const args = ["diff-tree", "-r", "-p"];
|
|
1022
|
+
if (options.binary) args.push("--binary");
|
|
1023
|
+
args.push(base, headRef);
|
|
1024
|
+
if (options.allowFailure) {
|
|
1025
|
+
return (await git(cwd, args, { readOnly: true, signal: options.signal })).stdout;
|
|
1026
|
+
}
|
|
1027
|
+
return runText(cwd, args, { readOnly: true, signal: options.signal });
|
|
1028
|
+
},
|
|
1029
|
+
/** Parse raw diff text into per-file diffs. */
|
|
1030
|
+
parseFiles(text: string): FileDiff[] {
|
|
1031
|
+
return parseFileDiffs(text);
|
|
1032
|
+
},
|
|
1033
|
+
/** Parse raw diff text into per-file hunks. */
|
|
1034
|
+
parseHunks(text: string): FileHunks[] {
|
|
1035
|
+
return parseCommitDiffHunks(text);
|
|
1036
|
+
},
|
|
1037
|
+
},
|
|
1038
|
+
);
|
|
1039
|
+
|
|
1040
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1041
|
+
// API: status
|
|
1042
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1043
|
+
|
|
1044
|
+
/** Run `git status --porcelain`. Returns raw status text. */
|
|
1045
|
+
export const status = Object.assign(
|
|
1046
|
+
async function status(cwd: string, options: StatusOptions = {}): Promise<string> {
|
|
1047
|
+
const args = ["status"];
|
|
1048
|
+
args.push(options.porcelainV1 ? "--porcelain=v1" : "--porcelain");
|
|
1049
|
+
if (options.z) args.push("-z");
|
|
1050
|
+
if (options.untrackedFiles) args.push(`--untracked-files=${options.untrackedFiles}`);
|
|
1051
|
+
if (options.pathspecs?.length) args.push("--", ...options.pathspecs);
|
|
1052
|
+
return runText(cwd, args, { readOnly: true, signal: options.signal });
|
|
1053
|
+
},
|
|
1054
|
+
{
|
|
1055
|
+
/** Parsed status counts (staged, unstaged, untracked). */
|
|
1056
|
+
async summary(cwd: string, signal?: AbortSignal): Promise<GitStatusSummary | null> {
|
|
1057
|
+
const result = await git(cwd, ["status", "--porcelain"], { readOnly: true, signal });
|
|
1058
|
+
if (result.exitCode !== 0) return null;
|
|
1059
|
+
return parseStatusPorcelain(result.stdout);
|
|
1060
|
+
},
|
|
1061
|
+
/** Parse porcelain status text into counts. */
|
|
1062
|
+
parse: parseStatusPorcelain,
|
|
1063
|
+
},
|
|
1064
|
+
);
|
|
1065
|
+
|
|
1066
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1067
|
+
// API: stage
|
|
1068
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1069
|
+
|
|
1070
|
+
export const stage = {
|
|
1071
|
+
/** Stage files. Empty array stages all (`git add -A`). */
|
|
1072
|
+
async files(cwd: string, files: readonly string[] = [], signal?: AbortSignal): Promise<void> {
|
|
1073
|
+
const args = files.length === 0 ? ["add", "-A"] : ["add", "--", ...files];
|
|
1074
|
+
await runEffect(cwd, args, { signal });
|
|
1075
|
+
},
|
|
1076
|
+
|
|
1077
|
+
/** Selectively stage hunks from the provided diff or the current working tree diff. */
|
|
1078
|
+
async hunks(cwd: string, selections: HunkSelection[], options: StageHunksOptions = {}): Promise<void> {
|
|
1079
|
+
if (selections.length === 0) return;
|
|
1080
|
+
const rawDiff = options.rawDiff ?? (await diff(cwd, { cached: options.diffCached, signal: options.signal }));
|
|
1081
|
+
const fileDiffs = parseFileDiffs(rawDiff);
|
|
1082
|
+
const fileDiffMap = new Map(fileDiffs.map(entry => [entry.filename, entry]));
|
|
1083
|
+
const patchParts: string[] = [];
|
|
1084
|
+
|
|
1085
|
+
for (const selection of selections) {
|
|
1086
|
+
const fileDiff = fileDiffMap.get(selection.path);
|
|
1087
|
+
if (!fileDiff) throw new Error(`No diff found for ${selection.path}`);
|
|
1088
|
+
if (fileDiff.isBinary) {
|
|
1089
|
+
if (selection.hunks.type !== "all")
|
|
1090
|
+
throw new Error(`Cannot select hunks for binary file ${selection.path}`);
|
|
1091
|
+
patchParts.push(fileDiff.content);
|
|
1092
|
+
continue;
|
|
1093
|
+
}
|
|
1094
|
+
if (selection.hunks.type === "all") {
|
|
1095
|
+
patchParts.push(fileDiff.content);
|
|
1096
|
+
continue;
|
|
1097
|
+
}
|
|
1098
|
+
const fileHunks = parseFileHunks(fileDiff);
|
|
1099
|
+
const selected = selectHunks(fileHunks, selection.hunks);
|
|
1100
|
+
if (selected.length === 0) throw new Error(`No hunks selected for ${selection.path}`);
|
|
1101
|
+
const header = extractFileHeader(fileDiff.content);
|
|
1102
|
+
patchParts.push([header, ...selected.map(h => h.content)].join("\n"));
|
|
1103
|
+
}
|
|
1104
|
+
|
|
1105
|
+
const patchText = patch.join(patchParts);
|
|
1106
|
+
if (!patchText.trim()) return;
|
|
1107
|
+
await patch.applyText(cwd, patchText, { cached: true, signal: options.signal });
|
|
1108
|
+
},
|
|
1109
|
+
|
|
1110
|
+
/** Unstage files. Empty array unstages all (`git reset`). */
|
|
1111
|
+
async reset(cwd: string, files: readonly string[] = [], signal?: AbortSignal): Promise<void> {
|
|
1112
|
+
const args = files.length === 0 ? ["reset"] : ["reset", "--", ...files];
|
|
1113
|
+
await runEffect(cwd, args, { signal });
|
|
1114
|
+
},
|
|
1115
|
+
};
|
|
1116
|
+
|
|
1117
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1118
|
+
// API: commit, push, checkout
|
|
1119
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1120
|
+
|
|
1121
|
+
/** Create a commit with the given message (passed via stdin). */
|
|
1122
|
+
export async function commit(cwd: string, message: string, options: CommitOptions = {}): Promise<GitCommandResult> {
|
|
1123
|
+
const args = ["commit", "-F", "-"];
|
|
1124
|
+
if (options.allowEmpty) args.push("--allow-empty");
|
|
1125
|
+
if (options.files?.length) args.push("--", ...options.files);
|
|
1126
|
+
return runChecked(cwd, args, { signal: options.signal, stdin: message });
|
|
1127
|
+
}
|
|
1128
|
+
|
|
1129
|
+
/** Push the current branch. */
|
|
1130
|
+
export async function push(cwd: string, options: PushOptions = {}): Promise<void> {
|
|
1131
|
+
const args = ["push"];
|
|
1132
|
+
if (options.forceWithLease) args.push("--force-with-lease");
|
|
1133
|
+
if (options.remote) args.push(options.remote);
|
|
1134
|
+
if (options.refspec) args.push(options.refspec);
|
|
1135
|
+
await runEffect(cwd, args, { signal: options.signal });
|
|
1136
|
+
}
|
|
1137
|
+
|
|
1138
|
+
/** Checkout a ref. */
|
|
1139
|
+
export async function checkout(cwd: string, ref: string, signal?: AbortSignal): Promise<void> {
|
|
1140
|
+
await runEffect(cwd, ["checkout", ref], { signal });
|
|
1141
|
+
}
|
|
1142
|
+
|
|
1143
|
+
/** Fetch a specific refspec from a remote. */
|
|
1144
|
+
export async function fetch(
|
|
1145
|
+
cwd: string,
|
|
1146
|
+
remote: string,
|
|
1147
|
+
source: string,
|
|
1148
|
+
target: string,
|
|
1149
|
+
signal?: AbortSignal,
|
|
1150
|
+
): Promise<void> {
|
|
1151
|
+
await runEffect(cwd, ["fetch", remote, `+${source}:${target}`], { signal });
|
|
1152
|
+
}
|
|
1153
|
+
|
|
1154
|
+
/** Read a tree-ish into the index. */
|
|
1155
|
+
export async function readTree(
|
|
1156
|
+
cwd: string,
|
|
1157
|
+
treeish: string,
|
|
1158
|
+
options: Pick<CommandOptions, "env" | "signal"> = {},
|
|
1159
|
+
): Promise<void> {
|
|
1160
|
+
await runEffect(cwd, ["read-tree", treeish], options);
|
|
1161
|
+
}
|
|
1162
|
+
|
|
1163
|
+
/** Write the current index as a tree and return its object id. */
|
|
1164
|
+
export async function writeTree(cwd: string, options: Pick<CommandOptions, "env" | "signal"> = {}): Promise<string> {
|
|
1165
|
+
return (await runText(cwd, ["write-tree"], options)).trim();
|
|
1166
|
+
}
|
|
1167
|
+
|
|
1168
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1169
|
+
// API: show
|
|
1170
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1171
|
+
|
|
1172
|
+
/** Run `git show` on a revision. */
|
|
1173
|
+
export const show = Object.assign(
|
|
1174
|
+
async function show(
|
|
1175
|
+
cwd: string,
|
|
1176
|
+
revision: string,
|
|
1177
|
+
options: { format?: string; signal?: AbortSignal } = {},
|
|
1178
|
+
): Promise<string> {
|
|
1179
|
+
return runText(cwd, ["show", `--format=${options.format ?? ""}`, revision], {
|
|
1180
|
+
readOnly: true,
|
|
1181
|
+
signal: options.signal,
|
|
1182
|
+
});
|
|
1183
|
+
},
|
|
1184
|
+
{
|
|
1185
|
+
/** Get the path prefix of the current directory relative to the repo root. */
|
|
1186
|
+
async prefix(cwd: string, signal?: AbortSignal): Promise<string> {
|
|
1187
|
+
return (await runText(cwd, ["rev-parse", "--show-prefix"], { readOnly: true, signal })).trim();
|
|
1188
|
+
},
|
|
1189
|
+
},
|
|
1190
|
+
);
|
|
1191
|
+
|
|
1192
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1193
|
+
// API: log
|
|
1194
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1195
|
+
|
|
1196
|
+
export const log = {
|
|
1197
|
+
/** Recent commit subjects (one-line each). */
|
|
1198
|
+
async subjects(cwd: string, count: number, signal?: AbortSignal): Promise<string[]> {
|
|
1199
|
+
return splitLines(await runText(cwd, ["log", `-n${count}`, "--pretty=format:%s"], { readOnly: true, signal }));
|
|
1200
|
+
},
|
|
1201
|
+
/** Recent commits as `<short-sha> <subject>` onelines. */
|
|
1202
|
+
async onelines(cwd: string, count: number, signal?: AbortSignal): Promise<string[]> {
|
|
1203
|
+
return splitLines(
|
|
1204
|
+
await runText(cwd, ["log", `-${count}`, "--oneline", "--no-decorate"], { readOnly: true, signal }),
|
|
1205
|
+
);
|
|
1206
|
+
},
|
|
1207
|
+
};
|
|
1208
|
+
|
|
1209
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1210
|
+
// API: branch
|
|
1211
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1212
|
+
|
|
1213
|
+
export const branch = {
|
|
1214
|
+
/** Current branch name, or null if detached/unavailable. */
|
|
1215
|
+
async current(cwd: string, signal?: AbortSignal): Promise<string | null> {
|
|
1216
|
+
const headState = await resolveHead(cwd);
|
|
1217
|
+
if (headState?.kind === "ref") return headState.branchName ?? headState.ref;
|
|
1218
|
+
const result = await git(cwd, ["symbolic-ref", "--short", "HEAD"], { readOnly: true, signal });
|
|
1219
|
+
if (result.exitCode !== 0) return null;
|
|
1220
|
+
return result.stdout.trim() || null;
|
|
1221
|
+
},
|
|
1222
|
+
|
|
1223
|
+
/** Default branch name (from remote HEAD refs). */
|
|
1224
|
+
async default(cwd: string, signal?: AbortSignal): Promise<string | null> {
|
|
1225
|
+
const repository = await resolveRepository(cwd);
|
|
1226
|
+
if (repository) {
|
|
1227
|
+
for (const refPath of DEFAULT_BRANCH_REFS) {
|
|
1228
|
+
const target = await readRef(repository, refPath, signal);
|
|
1229
|
+
const branchName = parseDefaultBranchRef(refPath, target);
|
|
1230
|
+
if (branchName) return branchName;
|
|
1231
|
+
}
|
|
1232
|
+
}
|
|
1233
|
+
for (const remoteRef of ["origin/HEAD", "upstream/HEAD"]) {
|
|
1234
|
+
const result = await git(cwd, ["rev-parse", "--abbrev-ref", remoteRef], { readOnly: true, signal });
|
|
1235
|
+
if (result.exitCode !== 0) continue;
|
|
1236
|
+
const branchName = stripRemotePrefix(result.stdout.trim());
|
|
1237
|
+
if (branchName) return branchName;
|
|
1238
|
+
}
|
|
1239
|
+
return null;
|
|
1240
|
+
},
|
|
1241
|
+
|
|
1242
|
+
/** Create a new branch at the given start point. */
|
|
1243
|
+
async create(cwd: string, name: string, startPoint = "HEAD", signal?: AbortSignal): Promise<void> {
|
|
1244
|
+
await runEffect(cwd, ["branch", name, startPoint], { signal });
|
|
1245
|
+
},
|
|
1246
|
+
|
|
1247
|
+
/** Force-move a branch to a new start point. */
|
|
1248
|
+
async force(cwd: string, name: string, startPoint: string, signal?: AbortSignal): Promise<void> {
|
|
1249
|
+
await runEffect(cwd, ["branch", "--force", name, startPoint], { signal });
|
|
1250
|
+
},
|
|
1251
|
+
|
|
1252
|
+
/** Delete a branch. Throws on failure. */
|
|
1253
|
+
async delete(cwd: string, name: string, options: { force?: boolean; signal?: AbortSignal } = {}): Promise<void> {
|
|
1254
|
+
await runEffect(cwd, ["branch", options.force === false ? "-d" : "-D", name], { signal: options.signal });
|
|
1255
|
+
},
|
|
1256
|
+
|
|
1257
|
+
/** Delete a branch. Returns false on failure instead of throwing. */
|
|
1258
|
+
async tryDelete(
|
|
1259
|
+
cwd: string,
|
|
1260
|
+
name: string,
|
|
1261
|
+
options: { force?: boolean; signal?: AbortSignal } = {},
|
|
1262
|
+
): Promise<boolean> {
|
|
1263
|
+
const result = await git(cwd, ["branch", options.force === false ? "-d" : "-D", name], {
|
|
1264
|
+
signal: options.signal,
|
|
1265
|
+
});
|
|
1266
|
+
return result.exitCode === 0;
|
|
1267
|
+
},
|
|
1268
|
+
|
|
1269
|
+
/** Create and checkout a new branch. */
|
|
1270
|
+
async checkoutNew(cwd: string, name: string, signal?: AbortSignal): Promise<void> {
|
|
1271
|
+
await runEffect(cwd, ["checkout", "-b", name], { signal });
|
|
1272
|
+
},
|
|
1273
|
+
|
|
1274
|
+
/** List branches. Pass `{ all: true }` to include remotes. */
|
|
1275
|
+
async list(cwd: string, options: { all?: boolean; signal?: AbortSignal } = {}): Promise<string[]> {
|
|
1276
|
+
const args = ["branch"];
|
|
1277
|
+
if (options.all) args.push("-a");
|
|
1278
|
+
args.push("--format=%(refname:short)");
|
|
1279
|
+
return splitLines(await runText(cwd, args, { readOnly: true, signal: options.signal }));
|
|
1280
|
+
},
|
|
1281
|
+
};
|
|
1282
|
+
|
|
1283
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1284
|
+
// API: remote
|
|
1285
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1286
|
+
|
|
1287
|
+
export const remote = {
|
|
1288
|
+
/** List remote names. */
|
|
1289
|
+
async list(cwd: string, signal?: AbortSignal): Promise<string[]> {
|
|
1290
|
+
return splitLines(await runText(cwd, ["remote"], { readOnly: true, signal }));
|
|
1291
|
+
},
|
|
1292
|
+
|
|
1293
|
+
/** Get the URL for a remote. */
|
|
1294
|
+
async url(cwd: string, name: string, signal?: AbortSignal): Promise<string | undefined> {
|
|
1295
|
+
return trimScalar(await tryText(cwd, ["remote", "get-url", name], { readOnly: true, signal }));
|
|
1296
|
+
},
|
|
1297
|
+
|
|
1298
|
+
/**
|
|
1299
|
+
* Add a remote pointing at `url`. Idempotent: if a remote named `name`
|
|
1300
|
+
* already exists with the same URL (e.g. an in-process race or a leftover
|
|
1301
|
+
* remote from a previous run), this is treated as success. Throws when the
|
|
1302
|
+
* remote exists with a different URL — that's a real conflict the caller
|
|
1303
|
+
* needs to resolve, not paper over.
|
|
1304
|
+
*/
|
|
1305
|
+
async add(cwd: string, name: string, url: string, signal?: AbortSignal): Promise<void> {
|
|
1306
|
+
const result = await git(cwd, ["remote", "add", name, url], { signal });
|
|
1307
|
+
if (result.exitCode === 0) return;
|
|
1308
|
+
if (REMOTE_ALREADY_EXISTS.test(result.stderr)) {
|
|
1309
|
+
const existing = await remote.url(cwd, name, signal);
|
|
1310
|
+
if (existing === url) return;
|
|
1311
|
+
throw new ToolError(`remote ${name} already exists with URL ${existing ?? "(unset)"}, expected ${url}`);
|
|
1312
|
+
}
|
|
1313
|
+
throw new GitCommandError(["remote", "add", name, url], result);
|
|
1314
|
+
},
|
|
1315
|
+
};
|
|
1316
|
+
|
|
1317
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1318
|
+
// API: ref
|
|
1319
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1320
|
+
|
|
1321
|
+
export const ref = {
|
|
1322
|
+
/** Check if a ref exists. */
|
|
1323
|
+
async exists(cwd: string, refName: string, signal?: AbortSignal): Promise<boolean> {
|
|
1324
|
+
if (refName === "HEAD") return (await head.sha(cwd, signal)) !== null;
|
|
1325
|
+
const repository = await resolveRepository(cwd);
|
|
1326
|
+
if (repository && refName.startsWith("refs/")) return (await readRef(repository, refName, signal)) !== null;
|
|
1327
|
+
const result = await git(cwd, ["show-ref", "--verify", "--quiet", refName], { readOnly: true, signal });
|
|
1328
|
+
return result.exitCode === 0;
|
|
1329
|
+
},
|
|
1330
|
+
|
|
1331
|
+
/** Resolve a ref to its commit SHA. */
|
|
1332
|
+
async resolve(cwd: string, refName: string, signal?: AbortSignal): Promise<string | null> {
|
|
1333
|
+
if (refName === "HEAD") return head.sha(cwd, signal);
|
|
1334
|
+
const repository = await resolveRepository(cwd);
|
|
1335
|
+
if (repository && refName.startsWith("refs/")) return readRef(repository, refName, signal);
|
|
1336
|
+
const result = await git(cwd, ["rev-parse", refName], { readOnly: true, signal });
|
|
1337
|
+
if (result.exitCode !== 0) return null;
|
|
1338
|
+
return result.stdout.trim() || null;
|
|
1339
|
+
},
|
|
1340
|
+
|
|
1341
|
+
/** Tags pointing at a ref. */
|
|
1342
|
+
async tags(cwd: string, refName = "HEAD", signal?: AbortSignal): Promise<string[]> {
|
|
1343
|
+
return splitLines(
|
|
1344
|
+
await runText(
|
|
1345
|
+
cwd,
|
|
1346
|
+
[
|
|
1347
|
+
"for-each-ref",
|
|
1348
|
+
"--points-at",
|
|
1349
|
+
refName,
|
|
1350
|
+
"--sort=-version:refname",
|
|
1351
|
+
"--format=%(refname:strip=2)",
|
|
1352
|
+
"refs/tags",
|
|
1353
|
+
],
|
|
1354
|
+
{ readOnly: true, signal },
|
|
1355
|
+
),
|
|
1356
|
+
);
|
|
1357
|
+
},
|
|
1358
|
+
};
|
|
1359
|
+
|
|
1360
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1361
|
+
// API: config
|
|
1362
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1363
|
+
|
|
1364
|
+
export const config = {
|
|
1365
|
+
async get(cwd: string, key: string, signal?: AbortSignal): Promise<string | undefined> {
|
|
1366
|
+
return trimScalar(await tryText(cwd, ["config", "--get", key], { readOnly: true, signal }));
|
|
1367
|
+
},
|
|
1368
|
+
|
|
1369
|
+
async set(cwd: string, key: string, value: string, signal?: AbortSignal): Promise<void> {
|
|
1370
|
+
await runEffect(cwd, ["config", key, value], { signal });
|
|
1371
|
+
},
|
|
1372
|
+
|
|
1373
|
+
async getBranch(cwd: string, branchName: string, key: string, signal?: AbortSignal): Promise<string | undefined> {
|
|
1374
|
+
return config.get(cwd, `branch.${branchName}.${key}`, signal);
|
|
1375
|
+
},
|
|
1376
|
+
|
|
1377
|
+
async setBranch(cwd: string, branchName: string, key: string, value: string, signal?: AbortSignal): Promise<void> {
|
|
1378
|
+
return config.set(cwd, `branch.${branchName}.${key}`, value, signal);
|
|
1379
|
+
},
|
|
1380
|
+
};
|
|
1381
|
+
|
|
1382
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1383
|
+
// API: worktree
|
|
1384
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1385
|
+
|
|
1386
|
+
export const worktree = {
|
|
1387
|
+
async add(
|
|
1388
|
+
cwd: string,
|
|
1389
|
+
worktreePath: string,
|
|
1390
|
+
refName: string,
|
|
1391
|
+
options: { detach?: boolean; signal?: AbortSignal } = {},
|
|
1392
|
+
): Promise<void> {
|
|
1393
|
+
const args = ["worktree", "add"];
|
|
1394
|
+
if (options.detach) args.push("--detach");
|
|
1395
|
+
args.push(worktreePath, refName);
|
|
1396
|
+
await runEffect(cwd, args, { signal: options.signal });
|
|
1397
|
+
},
|
|
1398
|
+
|
|
1399
|
+
async remove(
|
|
1400
|
+
cwd: string,
|
|
1401
|
+
worktreePath: string,
|
|
1402
|
+
options: { force?: boolean; signal?: AbortSignal } = {},
|
|
1403
|
+
): Promise<void> {
|
|
1404
|
+
const args = ["worktree", "remove"];
|
|
1405
|
+
if (options.force ?? true) args.push("-f");
|
|
1406
|
+
args.push(worktreePath);
|
|
1407
|
+
await runEffect(cwd, args, { signal: options.signal });
|
|
1408
|
+
},
|
|
1409
|
+
|
|
1410
|
+
async tryRemove(
|
|
1411
|
+
cwd: string,
|
|
1412
|
+
worktreePath: string,
|
|
1413
|
+
options: { force?: boolean; signal?: AbortSignal } = {},
|
|
1414
|
+
): Promise<boolean> {
|
|
1415
|
+
const args = ["worktree", "remove"];
|
|
1416
|
+
if (options.force ?? true) args.push("-f");
|
|
1417
|
+
args.push(worktreePath);
|
|
1418
|
+
const result = await git(cwd, args, { signal: options.signal });
|
|
1419
|
+
return result.exitCode === 0;
|
|
1420
|
+
},
|
|
1421
|
+
|
|
1422
|
+
async list(cwd: string, signal?: AbortSignal): Promise<GitWorktreeEntry[]> {
|
|
1423
|
+
return parseWorktreeList(await runText(cwd, ["worktree", "list", "--porcelain"], { readOnly: true, signal }));
|
|
1424
|
+
},
|
|
1425
|
+
|
|
1426
|
+
async prune(cwd: string, signal?: AbortSignal): Promise<void> {
|
|
1427
|
+
await runEffect(cwd, ["worktree", "prune"], { signal });
|
|
1428
|
+
},
|
|
1429
|
+
};
|
|
1430
|
+
|
|
1431
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1432
|
+
// API: patch
|
|
1433
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1434
|
+
|
|
1435
|
+
export const patch = {
|
|
1436
|
+
/** Apply a patch file. */
|
|
1437
|
+
async apply(cwd: string, patchPath: string, options: PatchOptions = {}): Promise<void> {
|
|
1438
|
+
await runEffect(cwd, buildApplyArgs(patchPath, options), { env: options.env, signal: options.signal });
|
|
1439
|
+
},
|
|
1440
|
+
|
|
1441
|
+
/** Apply a patch from a string (writes to a temp file). */
|
|
1442
|
+
async applyText(cwd: string, patchText: string, options: PatchOptions = {}): Promise<void> {
|
|
1443
|
+
if (!patchText.trim()) return;
|
|
1444
|
+
const tempPath = await writeTempPatch(patchText);
|
|
1445
|
+
try {
|
|
1446
|
+
await patch.apply(cwd, tempPath, options);
|
|
1447
|
+
} finally {
|
|
1448
|
+
await fs.promises.rm(tempPath, { force: true });
|
|
1449
|
+
}
|
|
1450
|
+
},
|
|
1451
|
+
|
|
1452
|
+
/** Check if a patch file can be applied cleanly. */
|
|
1453
|
+
async canApply(cwd: string, patchPath: string, options: Omit<PatchOptions, "check"> = {}): Promise<boolean> {
|
|
1454
|
+
const result = await git(cwd, buildApplyArgs(patchPath, { ...options, check: true }), {
|
|
1455
|
+
env: options.env,
|
|
1456
|
+
readOnly: true,
|
|
1457
|
+
signal: options.signal,
|
|
1458
|
+
});
|
|
1459
|
+
return result.exitCode === 0;
|
|
1460
|
+
},
|
|
1461
|
+
|
|
1462
|
+
/** Check if a patch string can be applied cleanly. */
|
|
1463
|
+
async canApplyText(cwd: string, patchText: string, options: Omit<PatchOptions, "check"> = {}): Promise<boolean> {
|
|
1464
|
+
if (!patchText.trim()) return true;
|
|
1465
|
+
const tempPath = await writeTempPatch(patchText);
|
|
1466
|
+
try {
|
|
1467
|
+
return await patch.canApply(cwd, tempPath, options);
|
|
1468
|
+
} finally {
|
|
1469
|
+
await fs.promises.rm(tempPath, { force: true });
|
|
1470
|
+
}
|
|
1471
|
+
},
|
|
1472
|
+
|
|
1473
|
+
/** Join patch parts into a single patch string. */
|
|
1474
|
+
join(parts: string[]): string {
|
|
1475
|
+
return `${parts
|
|
1476
|
+
.map(part => (part.endsWith("\n") ? part : `${part}\n`))
|
|
1477
|
+
.join("\n")
|
|
1478
|
+
.replace(/\n+$/, "")}\n`;
|
|
1479
|
+
},
|
|
1480
|
+
};
|
|
1481
|
+
|
|
1482
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1483
|
+
// API: cherryPick
|
|
1484
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1485
|
+
|
|
1486
|
+
export const cherryPick = Object.assign(
|
|
1487
|
+
async function cherryPick(cwd: string, revision: string, signal?: AbortSignal): Promise<void> {
|
|
1488
|
+
await runEffect(cwd, ["cherry-pick", revision], { signal });
|
|
1489
|
+
},
|
|
1490
|
+
{
|
|
1491
|
+
async abort(cwd: string, signal?: AbortSignal): Promise<void> {
|
|
1492
|
+
await runEffect(cwd, ["cherry-pick", "--abort"], { signal });
|
|
1493
|
+
},
|
|
1494
|
+
},
|
|
1495
|
+
);
|
|
1496
|
+
|
|
1497
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1498
|
+
// API: stash
|
|
1499
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1500
|
+
|
|
1501
|
+
export const stash = {
|
|
1502
|
+
/** Stash working tree + index changes. Returns true when git created a new stash entry. */
|
|
1503
|
+
async push(cwd: string, message?: string): Promise<boolean> {
|
|
1504
|
+
ensureAvailable();
|
|
1505
|
+
const previousStash = await ref.resolve(cwd, "refs/stash");
|
|
1506
|
+
const args = ["stash", "push", "--include-untracked"];
|
|
1507
|
+
if (message) args.push("-m", message);
|
|
1508
|
+
await runEffect(cwd, args);
|
|
1509
|
+
const nextStash = await ref.resolve(cwd, "refs/stash");
|
|
1510
|
+
return nextStash !== null && nextStash !== previousStash;
|
|
1511
|
+
},
|
|
1512
|
+
/** Pop the most recent stash entry, optionally restoring its staged state. */
|
|
1513
|
+
async pop(cwd: string, options?: { index?: boolean }): Promise<void> {
|
|
1514
|
+
const args = ["stash", "pop"];
|
|
1515
|
+
if (options?.index) args.push("--index");
|
|
1516
|
+
await runEffect(cwd, args);
|
|
1517
|
+
},
|
|
1518
|
+
};
|
|
1519
|
+
|
|
1520
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1521
|
+
// API: clone, restore, clean
|
|
1522
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1523
|
+
|
|
1524
|
+
export async function clone(url: string, targetDir: string, options: CloneOptions = {}): Promise<void> {
|
|
1525
|
+
ensureAvailable();
|
|
1526
|
+
const absoluteTarget = path.resolve(targetDir);
|
|
1527
|
+
await fs.promises.mkdir(path.dirname(absoluteTarget), { recursive: true });
|
|
1528
|
+
|
|
1529
|
+
// `git clone --depth 1 --single-branch` only fetches the tip of the target
|
|
1530
|
+
// branch, so any subsequent `git checkout <sha>` for a non-tip commit fails
|
|
1531
|
+
// with "reference is not a tree". When the caller pinned a specific SHA we
|
|
1532
|
+
// fall back to a full clone so the object is guaranteed to be present.
|
|
1533
|
+
const shallow = !options.sha;
|
|
1534
|
+
const args = ["clone"];
|
|
1535
|
+
if (shallow) args.push("--depth", "1");
|
|
1536
|
+
if (options.ref) args.push("--branch", options.ref, "--single-branch");
|
|
1537
|
+
else if (shallow) args.push("--single-branch");
|
|
1538
|
+
args.push(url, absoluteTarget);
|
|
1539
|
+
|
|
1540
|
+
try {
|
|
1541
|
+
await runEffect(path.dirname(absoluteTarget), args, { signal: options.signal });
|
|
1542
|
+
if (options.sha) {
|
|
1543
|
+
try {
|
|
1544
|
+
await checkout(absoluteTarget, options.sha, options.signal);
|
|
1545
|
+
} catch {
|
|
1546
|
+
await fs.promises.rm(absoluteTarget, { force: true, recursive: true });
|
|
1547
|
+
throw new Error(`Failed to checkout SHA ${options.sha} in cloned repository ${url}`);
|
|
1548
|
+
}
|
|
1549
|
+
}
|
|
1550
|
+
} catch (err) {
|
|
1551
|
+
await fs.promises.rm(absoluteTarget, { force: true, recursive: true });
|
|
1552
|
+
throw err;
|
|
1553
|
+
}
|
|
1554
|
+
}
|
|
1555
|
+
|
|
1556
|
+
export async function restore(cwd: string, options: RestoreOptions = {}): Promise<void> {
|
|
1557
|
+
const args = ["restore"];
|
|
1558
|
+
if (options.source) args.push(`--source=${options.source}`);
|
|
1559
|
+
if (options.staged) args.push("--staged");
|
|
1560
|
+
if (options.worktree) args.push("--worktree");
|
|
1561
|
+
if (options.files?.length) args.push("--", ...options.files);
|
|
1562
|
+
await runEffect(cwd, args, { signal: options.signal });
|
|
1563
|
+
}
|
|
1564
|
+
|
|
1565
|
+
/**
|
|
1566
|
+
* Run `git reset` with options. Default is a soft reset (no flag); pass `hard: true` for a destructive reset.
|
|
1567
|
+
*
|
|
1568
|
+
* NOTE: stage.reset() handles the per-file unstaging case. This helper exists for tree-wide resets.
|
|
1569
|
+
*/
|
|
1570
|
+
export async function reset(
|
|
1571
|
+
cwd: string,
|
|
1572
|
+
options: { hard?: boolean; mixed?: boolean; soft?: boolean; target?: string; signal?: AbortSignal } = {},
|
|
1573
|
+
): Promise<void> {
|
|
1574
|
+
const args = ["reset"];
|
|
1575
|
+
if (options.hard) args.push("--hard");
|
|
1576
|
+
else if (options.mixed) args.push("--mixed");
|
|
1577
|
+
else if (options.soft) args.push("--soft");
|
|
1578
|
+
if (options.target) args.push(options.target);
|
|
1579
|
+
await runEffect(cwd, args, { signal: options.signal });
|
|
1580
|
+
}
|
|
1581
|
+
|
|
1582
|
+
export async function clean(
|
|
1583
|
+
cwd: string,
|
|
1584
|
+
options: { ignoredOnly?: boolean; paths?: readonly string[]; signal?: AbortSignal } = {},
|
|
1585
|
+
): Promise<void> {
|
|
1586
|
+
const args = ["clean", options.ignoredOnly ? "-fdX" : "-fd"];
|
|
1587
|
+
if (options.paths?.length) args.push("--", ...options.paths);
|
|
1588
|
+
await runEffect(cwd, args, { signal: options.signal });
|
|
1589
|
+
}
|
|
1590
|
+
|
|
1591
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1592
|
+
// API: ls
|
|
1593
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1594
|
+
|
|
1595
|
+
export const ls = {
|
|
1596
|
+
/** List files tracked or untracked by git. */
|
|
1597
|
+
async files(
|
|
1598
|
+
cwd: string,
|
|
1599
|
+
options: { others?: boolean; excludeStandard?: boolean; signal?: AbortSignal } = {},
|
|
1600
|
+
): Promise<string[]> {
|
|
1601
|
+
const args = ["ls-files"];
|
|
1602
|
+
if (options.others) args.push("--others");
|
|
1603
|
+
if (options.excludeStandard) args.push("--exclude-standard");
|
|
1604
|
+
return splitLines(await runText(cwd, args, { readOnly: true, signal: options.signal }));
|
|
1605
|
+
},
|
|
1606
|
+
|
|
1607
|
+
/** List untracked files (excludes ignored). */
|
|
1608
|
+
async untracked(cwd: string, signal?: AbortSignal): Promise<string[]> {
|
|
1609
|
+
return ls.files(cwd, { others: true, excludeStandard: true, signal });
|
|
1610
|
+
},
|
|
1611
|
+
|
|
1612
|
+
/** List submodule paths (recursive). */
|
|
1613
|
+
async submodules(cwd: string, signal?: AbortSignal): Promise<string[]> {
|
|
1614
|
+
const output = await git(cwd, ["submodule", "--quiet", "foreach", "--recursive", "echo $sm_path"], {
|
|
1615
|
+
readOnly: true,
|
|
1616
|
+
signal,
|
|
1617
|
+
});
|
|
1618
|
+
return splitLines(output.stdout);
|
|
1619
|
+
},
|
|
1620
|
+
};
|
|
1621
|
+
|
|
1622
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1623
|
+
// API: head
|
|
1624
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1625
|
+
|
|
1626
|
+
export const head = {
|
|
1627
|
+
/** Full HEAD state (branch, commit, repo info). */
|
|
1628
|
+
async resolve(cwd: string, signal?: AbortSignal): Promise<GitHeadState | null> {
|
|
1629
|
+
const repository = await resolveRepository(cwd);
|
|
1630
|
+
if (!repository) return null;
|
|
1631
|
+
if (await isReftableRepo(repository)) {
|
|
1632
|
+
return resolveHeadStateReftable(repository, signal);
|
|
1633
|
+
}
|
|
1634
|
+
const content = await readOptionalText(repository.headPath);
|
|
1635
|
+
if (content === null) return null;
|
|
1636
|
+
return parseHeadState(repository, content);
|
|
1637
|
+
},
|
|
1638
|
+
|
|
1639
|
+
/** Full HEAD state (synchronous). */
|
|
1640
|
+
resolveSync(cwd: string): GitHeadState | null {
|
|
1641
|
+
const repository = resolveRepositorySync(cwd);
|
|
1642
|
+
if (!repository) return null;
|
|
1643
|
+
if (isReftableRepoSync(repository)) {
|
|
1644
|
+
return resolveHeadStateReftableSync(repository);
|
|
1645
|
+
}
|
|
1646
|
+
const content = readOptionalTextSync(repository.headPath);
|
|
1647
|
+
if (content === null) return null;
|
|
1648
|
+
return parseHeadStateSync(repository, content);
|
|
1649
|
+
},
|
|
1650
|
+
|
|
1651
|
+
/** Current HEAD commit SHA. */
|
|
1652
|
+
async sha(cwd: string, signal?: AbortSignal): Promise<string | null> {
|
|
1653
|
+
const headState = await head.resolve(cwd, signal);
|
|
1654
|
+
if (headState?.commit) return headState.commit;
|
|
1655
|
+
const result = await git(cwd, ["rev-parse", "HEAD"], { readOnly: true, signal });
|
|
1656
|
+
if (result.exitCode !== 0) return null;
|
|
1657
|
+
return result.stdout.trim() || null;
|
|
1658
|
+
},
|
|
1659
|
+
|
|
1660
|
+
/** Abbreviated HEAD commit SHA. */
|
|
1661
|
+
async short(cwd: string, length = 7, signal?: AbortSignal): Promise<string | null> {
|
|
1662
|
+
const result = await git(cwd, ["rev-parse", `--short=${length}`, "HEAD"], { readOnly: true, signal });
|
|
1663
|
+
if (result.exitCode !== 0) return null;
|
|
1664
|
+
return result.stdout.trim() || null;
|
|
1665
|
+
},
|
|
1666
|
+
};
|
|
1667
|
+
|
|
1668
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1669
|
+
// API: repo
|
|
1670
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1671
|
+
|
|
1672
|
+
export const repo = {
|
|
1673
|
+
/** Resolve the repository root (may be a worktree root). */
|
|
1674
|
+
async root(cwd: string, signal?: AbortSignal): Promise<string | null> {
|
|
1675
|
+
const repository = await resolveRepository(cwd);
|
|
1676
|
+
if (repository) return repository.repoRoot;
|
|
1677
|
+
const result = await git(cwd, ["rev-parse", "--show-toplevel"], { readOnly: true, signal });
|
|
1678
|
+
if (result.exitCode !== 0) return null;
|
|
1679
|
+
return result.stdout.trim() || null;
|
|
1680
|
+
},
|
|
1681
|
+
|
|
1682
|
+
/** Resolve the primary checkout root, or the shared common dir for bare-repo worktrees. */
|
|
1683
|
+
async primaryRoot(cwd: string, signal?: AbortSignal): Promise<string | null> {
|
|
1684
|
+
const repository = await resolveRepository(cwd);
|
|
1685
|
+
if (repository) return primaryRootFromRepository(repository);
|
|
1686
|
+
const repoRoot = await repo.root(cwd, signal);
|
|
1687
|
+
if (!repoRoot) return null;
|
|
1688
|
+
const commonDir = await runText(repoRoot, ["rev-parse", "--path-format=absolute", "--git-common-dir"], {
|
|
1689
|
+
readOnly: true,
|
|
1690
|
+
signal,
|
|
1691
|
+
});
|
|
1692
|
+
if (path.basename(commonDir.trim()) === ".git") return path.dirname(commonDir.trim());
|
|
1693
|
+
return repoRoot;
|
|
1694
|
+
},
|
|
1695
|
+
|
|
1696
|
+
/**
|
|
1697
|
+
* Sync sibling of {@link primaryRoot}. Resolves only via on-disk `.git`/
|
|
1698
|
+
* `commondir` walking — no subprocess fallback — so it stays usable from
|
|
1699
|
+
* paths where async I/O is impractical (e.g. `computeBankScope`). Returns
|
|
1700
|
+
* `null` when `cwd` is outside a repository. Bare-repo worktrees resolve to
|
|
1701
|
+
* the shared common dir (`foo.git`) because they have no primary checkout.
|
|
1702
|
+
*/
|
|
1703
|
+
primaryRootSync(cwd: string): string | null {
|
|
1704
|
+
const repository = resolveRepositorySync(cwd);
|
|
1705
|
+
if (!repository) return null;
|
|
1706
|
+
return primaryRootFromRepositorySync(repository);
|
|
1707
|
+
},
|
|
1708
|
+
|
|
1709
|
+
/** Full GitRepository metadata (sync). */
|
|
1710
|
+
resolveSync(cwd: string): GitRepository | null {
|
|
1711
|
+
return resolveRepositorySync(cwd);
|
|
1712
|
+
},
|
|
1713
|
+
|
|
1714
|
+
/** Full GitRepository metadata. */
|
|
1715
|
+
resolve(cwd: string): Promise<GitRepository | null> {
|
|
1716
|
+
return resolveRepository(cwd);
|
|
1717
|
+
},
|
|
1718
|
+
|
|
1719
|
+
/** Check if the repository uses the reftable reference storage format (sync). */
|
|
1720
|
+
isReftableSync(repository: GitRepository): boolean {
|
|
1721
|
+
return isReftableRepoSync(repository);
|
|
1722
|
+
},
|
|
1723
|
+
|
|
1724
|
+
/** Check if the repository uses the reftable reference storage format. */
|
|
1725
|
+
isReftable(repository: GitRepository): Promise<boolean> {
|
|
1726
|
+
return isReftableRepo(repository);
|
|
1727
|
+
},
|
|
1728
|
+
};
|
|
1729
|
+
|
|
1730
|
+
// Helper used during head resolution — defined here to reference `head` namespace.
|
|
1731
|
+
async function resolveHead(cwd: string, signal?: AbortSignal): Promise<GitHeadState | null> {
|
|
1732
|
+
return head.resolve(cwd, signal);
|
|
1733
|
+
}
|
|
1734
|
+
|
|
1735
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1736
|
+
// API: github (GitHub CLI)
|
|
1737
|
+
// ════════════════════════════════════════════════════════════════════════════
|
|
1738
|
+
|
|
1739
|
+
export interface GhCommandResult {
|
|
1740
|
+
exitCode: number;
|
|
1741
|
+
stdout: string;
|
|
1742
|
+
stderr: string;
|
|
1743
|
+
}
|
|
1744
|
+
|
|
1745
|
+
export interface GhCommandOptions {
|
|
1746
|
+
repoProvided?: boolean;
|
|
1747
|
+
trimOutput?: boolean;
|
|
1748
|
+
}
|
|
1749
|
+
|
|
1750
|
+
function formatGhFailure(args: readonly string[], stdout: string, stderr: string, options?: GhCommandOptions): string {
|
|
1751
|
+
const message = (stderr || stdout).trim();
|
|
1752
|
+
if (message.includes("gh auth login") || message.includes("not logged into any GitHub hosts")) {
|
|
1753
|
+
return "GitHub CLI is not authenticated. Run `gh auth login`.";
|
|
1754
|
+
}
|
|
1755
|
+
if (
|
|
1756
|
+
!options?.repoProvided &&
|
|
1757
|
+
(message.includes("not a git repository") ||
|
|
1758
|
+
message.includes("no git remotes found") ||
|
|
1759
|
+
message.includes("unable to determine current repository"))
|
|
1760
|
+
) {
|
|
1761
|
+
return "GitHub repository context is unavailable. Pass `repo` explicitly or run the tool inside a GitHub checkout.";
|
|
1762
|
+
}
|
|
1763
|
+
if (message.length > 0) return message;
|
|
1764
|
+
return `GitHub CLI command failed: gh ${args.join(" ")}`;
|
|
1765
|
+
}
|
|
1766
|
+
|
|
1767
|
+
export const github = {
|
|
1768
|
+
/** Check if `gh` CLI is installed. */
|
|
1769
|
+
available(): boolean {
|
|
1770
|
+
return Boolean($which("gh"));
|
|
1771
|
+
},
|
|
1772
|
+
|
|
1773
|
+
/** Run a raw `gh` CLI command. Does not throw on non-zero exit. */
|
|
1774
|
+
async run(cwd: string, args: string[], signal?: AbortSignal, options?: GhCommandOptions): Promise<GhCommandResult> {
|
|
1775
|
+
throwIfAborted(signal);
|
|
1776
|
+
if (!$which("gh")) {
|
|
1777
|
+
throw new ToolError("GitHub CLI (gh) is not installed. Install it from https://cli.github.com/.");
|
|
1778
|
+
}
|
|
1779
|
+
try {
|
|
1780
|
+
const child = Bun.spawn(["gh", ...args], {
|
|
1781
|
+
cwd,
|
|
1782
|
+
stdin: "ignore",
|
|
1783
|
+
stdout: "pipe",
|
|
1784
|
+
stderr: "pipe",
|
|
1785
|
+
windowsHide: true,
|
|
1786
|
+
signal,
|
|
1787
|
+
});
|
|
1788
|
+
if (!child.stdout || !child.stderr) {
|
|
1789
|
+
throw new ToolError("Failed to capture GitHub CLI output.");
|
|
1790
|
+
}
|
|
1791
|
+
const [stdout, stderr, exitCode] = await Promise.all([
|
|
1792
|
+
new Response(child.stdout).text(),
|
|
1793
|
+
new Response(child.stderr).text(),
|
|
1794
|
+
child.exited,
|
|
1795
|
+
]);
|
|
1796
|
+
throwIfAborted(signal);
|
|
1797
|
+
const trim = options?.trimOutput !== false;
|
|
1798
|
+
return {
|
|
1799
|
+
exitCode: exitCode ?? 0,
|
|
1800
|
+
stdout: trim ? stdout.trim() : stdout,
|
|
1801
|
+
stderr: trim ? stderr.trim() : stderr,
|
|
1802
|
+
};
|
|
1803
|
+
} catch (error) {
|
|
1804
|
+
if (signal?.aborted) throw new ToolAbortError();
|
|
1805
|
+
throw error;
|
|
1806
|
+
}
|
|
1807
|
+
},
|
|
1808
|
+
|
|
1809
|
+
/** Run `gh` and parse stdout as JSON. Throws on non-zero exit or invalid JSON. */
|
|
1810
|
+
async json<T>(cwd: string, args: string[], signal?: AbortSignal, options?: GhCommandOptions): Promise<T> {
|
|
1811
|
+
const result = await github.run(cwd, args, signal, options);
|
|
1812
|
+
if (result.exitCode !== 0) {
|
|
1813
|
+
throw new ToolError(formatGhFailure(args, result.stdout, result.stderr, options));
|
|
1814
|
+
}
|
|
1815
|
+
if (!result.stdout) {
|
|
1816
|
+
throw new ToolError("GitHub CLI returned empty output.");
|
|
1817
|
+
}
|
|
1818
|
+
try {
|
|
1819
|
+
return JSON.parse(result.stdout) as T;
|
|
1820
|
+
} catch {
|
|
1821
|
+
throw new ToolError("GitHub CLI returned invalid JSON output.");
|
|
1822
|
+
}
|
|
1823
|
+
},
|
|
1824
|
+
|
|
1825
|
+
/** Run `gh` and return stdout as text. Throws on non-zero exit. */
|
|
1826
|
+
async text(cwd: string, args: string[], signal?: AbortSignal, options?: GhCommandOptions): Promise<string> {
|
|
1827
|
+
const result = await github.run(cwd, args, signal, options);
|
|
1828
|
+
if (result.exitCode !== 0) {
|
|
1829
|
+
throw new ToolError(formatGhFailure(args, result.stdout, result.stderr, options));
|
|
1830
|
+
}
|
|
1831
|
+
return result.stdout;
|
|
1832
|
+
},
|
|
1833
|
+
};
|