@opengsd/gsd-pi 1.2.0-dev.4c756166 → 1.2.0-dev.5457a158
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/dist/cli-style.d.ts +17 -0
- package/dist/cli-style.js +28 -0
- package/dist/cli.js +1 -1
- package/dist/headless-events.d.ts +4 -2
- package/dist/headless-events.js +7 -29
- package/dist/models-resolver.d.ts +3 -13
- package/dist/models-resolver.js +3 -22
- package/dist/resource-loader.js +2 -14
- package/dist/resources/.managed-resources-content-hash +1 -1
- package/dist/resources/extensions/bg-shell/utilities.js +5 -2
- package/dist/resources/extensions/claude-code-cli/models.js +9 -0
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +35 -4
- package/dist/resources/extensions/gsd/auto/orchestrator.js +33 -4
- package/dist/resources/extensions/gsd/auto/phases.js +6 -1
- package/dist/resources/extensions/gsd/auto-post-unit.js +19 -8
- package/dist/resources/extensions/gsd/auto-prompts.js +3 -0
- package/dist/resources/extensions/gsd/auto-start.js +12 -14
- package/dist/resources/extensions/gsd/auto-tool-tracking.js +18 -0
- package/dist/resources/extensions/gsd/auto-unit-tool-scope.js +7 -16
- package/dist/resources/extensions/gsd/auto-worktree-repair.js +10 -2
- package/dist/resources/extensions/gsd/auto-worktree.js +35 -352
- package/dist/resources/extensions/gsd/auto.js +8 -20
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +3 -2
- package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +9 -6
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +86 -6
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +30 -4
- package/dist/resources/extensions/gsd/branch-patterns.js +2 -0
- package/dist/resources/extensions/gsd/captures.js +5 -15
- package/dist/resources/extensions/gsd/closeout-recovery.js +3 -2
- package/dist/resources/extensions/gsd/commands/catalog.js +6 -62
- package/dist/resources/extensions/gsd/crash-recovery.js +4 -12
- package/dist/resources/extensions/gsd/db/engine.js +755 -0
- package/dist/resources/extensions/gsd/db/queries.js +372 -0
- package/dist/resources/extensions/gsd/db/sql-constants.js +11 -0
- package/dist/resources/extensions/gsd/db/writers/cascades.js +194 -0
- package/dist/resources/extensions/gsd/db/writers/import-restore.js +182 -0
- package/dist/resources/extensions/gsd/db/writers/memory.js +149 -0
- package/dist/resources/extensions/gsd/db/writers/reconcile.js +458 -0
- package/dist/resources/extensions/gsd/db/writers/status.js +70 -0
- package/dist/resources/extensions/gsd/doctor-environment.js +5 -11
- package/dist/resources/extensions/gsd/doctor-format.js +9 -6
- package/dist/resources/extensions/gsd/doctor-git-checks.js +4 -3
- package/dist/resources/extensions/gsd/doctor-runtime-checks.js +21 -16
- package/dist/resources/extensions/gsd/error-classifier.js +9 -0
- package/dist/resources/extensions/gsd/git-service.js +1 -0
- package/dist/resources/extensions/gsd/gitignore.js +3 -0
- package/dist/resources/extensions/gsd/gsd-db.js +171 -2048
- package/dist/resources/extensions/gsd/guidance.js +98 -0
- package/dist/resources/extensions/gsd/guided-flow.js +51 -5
- package/dist/resources/extensions/gsd/mcp-tool-name.js +5 -13
- package/dist/resources/extensions/gsd/memory-consolidation-scanner.js +1 -1
- package/dist/resources/extensions/gsd/migrate/safety.js +20 -9
- package/dist/resources/extensions/gsd/migration-auto-check.js +24 -3
- package/dist/resources/extensions/gsd/model-cost-table.js +1 -0
- package/dist/resources/extensions/gsd/model-router.js +3 -0
- package/dist/resources/extensions/gsd/notification-store.js +11 -4
- package/dist/resources/extensions/gsd/parallel-merge.js +14 -11
- package/dist/resources/extensions/gsd/parallel-monitor-overlay.js +11 -7
- package/dist/resources/extensions/gsd/paths.js +37 -24
- package/dist/resources/extensions/gsd/pre-execution-checks.js +91 -3
- package/dist/resources/extensions/gsd/preferences-models.js +12 -46
- package/dist/resources/extensions/gsd/preferences.js +14 -0
- package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/refine-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/replan-slice.md +1 -1
- package/dist/resources/extensions/gsd/provider-error-guidance.js +1 -5
- package/dist/resources/extensions/gsd/provider-switch-observer.js +1 -1
- package/dist/resources/extensions/gsd/publication.js +87 -0
- package/dist/resources/extensions/gsd/recovery-classification.js +41 -87
- package/dist/resources/extensions/gsd/safety/evidence-collector.js +37 -4
- package/dist/resources/extensions/gsd/safety/evidence-cross-ref.js +7 -2
- package/dist/resources/extensions/gsd/safety/file-change-validator.js +10 -0
- package/dist/resources/extensions/gsd/state-transition-matrix.js +38 -0
- package/dist/resources/extensions/gsd/state.js +1 -20
- package/dist/resources/extensions/gsd/status-guards.js +56 -8
- package/dist/resources/extensions/gsd/stop-notice.js +57 -0
- package/dist/resources/extensions/gsd/tool-surface-readiness.js +56 -0
- package/dist/resources/extensions/gsd/tools/complete-slice.js +24 -43
- package/dist/resources/extensions/gsd/tools/exec-tool.js +5 -8
- package/dist/resources/extensions/gsd/tools/plan-slice.js +12 -6
- package/dist/resources/extensions/gsd/tools/reopen-milestone.js +11 -29
- package/dist/resources/extensions/gsd/tools/reopen-slice.js +14 -33
- package/dist/resources/extensions/gsd/tools/skip-slice.js +18 -36
- package/dist/resources/extensions/gsd/undo.js +8 -7
- package/dist/resources/extensions/gsd/unit-closeout.js +138 -0
- package/dist/resources/extensions/gsd/unit-context-composer.js +9 -1
- package/dist/resources/extensions/gsd/unit-context-manifest.js +4 -27
- package/dist/resources/extensions/gsd/unit-registry.js +350 -0
- package/dist/resources/extensions/gsd/unit-tool-contracts.js +9 -182
- package/dist/resources/extensions/gsd/workflow-tool-surface.js +1 -1
- package/dist/resources/extensions/gsd/worktree-git-recovery.js +293 -0
- package/dist/resources/extensions/gsd/worktree-lifecycle.js +9 -1
- package/dist/resources/extensions/gsd/worktree-manager.js +45 -28
- package/dist/resources/extensions/gsd/worktree-placement.js +59 -0
- package/dist/resources/extensions/gsd/worktree-reentry.js +12 -8
- package/dist/resources/extensions/gsd/worktree-root.js +28 -6
- package/dist/resources/extensions/gsd/worktree-safety.js +8 -5
- package/dist/resources/extensions/gsd/worktree-session-state.js +12 -11
- package/dist/resources/skills/gsd-browser/SKILL.md +1 -1
- package/dist/tsconfig.extensions.tsbuildinfo +1 -1
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +9 -9
- package/dist/web/standalone/.next/build-manifest.json +2 -2
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/mcp-connections/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/notifications/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/shutdown/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +9 -9
- package/dist/web/standalone/.next/server/chunks/5124.js +1 -1
- package/dist/web/standalone/.next/server/chunks/{5047.js → 5942.js} +2 -2
- package/dist/web/standalone/.next/server/chunks/8357.js +1 -1
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +1 -1
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/dist/worktree-cli.js +3 -6
- package/dist/worktree-status-banner.js +7 -11
- package/package.json +1 -1
- package/packages/cloud-mcp-gateway/package.json +2 -2
- package/packages/contracts/dist/workflow.d.ts +4 -0
- package/packages/contracts/dist/workflow.d.ts.map +1 -1
- package/packages/contracts/dist/workflow.js.map +1 -1
- package/packages/contracts/package.json +1 -1
- package/packages/daemon/package.json +4 -4
- package/packages/gsd-agent-core/package.json +5 -5
- package/packages/gsd-agent-modes/package.json +7 -7
- package/packages/mcp-server/dist/cli.js +6 -3
- package/packages/mcp-server/dist/cli.js.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts +8 -0
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +46 -21
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/package.json +3 -3
- package/packages/native/package.json +1 -1
- package/packages/pi-agent-core/package.json +1 -1
- package/packages/pi-ai/dist/models.generated.d.ts +294 -239
- package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
- package/packages/pi-ai/dist/models.generated.js +260 -256
- package/packages/pi-ai/dist/models.generated.js.map +1 -1
- package/packages/pi-ai/package.json +1 -1
- package/packages/pi-coding-agent/dist/core/capability-patches.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/capability-patches.js +3 -1
- package/packages/pi-coding-agent/dist/core/capability-patches.js.map +1 -1
- package/packages/pi-coding-agent/package.json +7 -7
- package/packages/pi-tui/package.json +2 -2
- package/packages/rpc-client/package.json +2 -2
- package/pkg/package.json +1 -1
- package/src/resources/extensions/bg-shell/utilities.ts +5 -2
- package/src/resources/extensions/claude-code-cli/models.ts +9 -0
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +37 -2
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +28 -0
- package/src/resources/extensions/gsd/auto/loop-deps.ts +1 -1
- package/src/resources/extensions/gsd/auto/orchestrator.ts +39 -5
- package/src/resources/extensions/gsd/auto/phases.ts +10 -1
- package/src/resources/extensions/gsd/auto-post-unit.ts +25 -7
- package/src/resources/extensions/gsd/auto-prompts.ts +3 -0
- package/src/resources/extensions/gsd/auto-start.ts +12 -15
- package/src/resources/extensions/gsd/auto-tool-tracking.ts +19 -0
- package/src/resources/extensions/gsd/auto-unit-tool-scope.ts +10 -17
- package/src/resources/extensions/gsd/auto-worktree-repair.ts +13 -2
- package/src/resources/extensions/gsd/auto-worktree.ts +41 -364
- package/src/resources/extensions/gsd/auto.ts +20 -24
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +3 -5
- package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +10 -6
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +87 -6
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +29 -3
- package/src/resources/extensions/gsd/branch-patterns.ts +3 -0
- package/src/resources/extensions/gsd/captures.ts +5 -16
- package/src/resources/extensions/gsd/closeout-recovery.ts +2 -1
- package/src/resources/extensions/gsd/commands/catalog.ts +6 -68
- package/src/resources/extensions/gsd/crash-recovery.ts +3 -9
- package/src/resources/extensions/gsd/db/engine.ts +809 -0
- package/src/resources/extensions/gsd/db/queries.ts +453 -0
- package/src/resources/extensions/gsd/db/sql-constants.ts +12 -0
- package/src/resources/extensions/gsd/db/writers/cascades.ts +237 -0
- package/src/resources/extensions/gsd/db/writers/import-restore.ts +310 -0
- package/src/resources/extensions/gsd/db/writers/memory.ts +220 -0
- package/src/resources/extensions/gsd/db/writers/reconcile.ts +500 -0
- package/src/resources/extensions/gsd/db/writers/status.ts +88 -0
- package/src/resources/extensions/gsd/doctor-environment.ts +5 -13
- package/src/resources/extensions/gsd/doctor-format.ts +12 -7
- package/src/resources/extensions/gsd/doctor-git-checks.ts +3 -3
- package/src/resources/extensions/gsd/doctor-runtime-checks.ts +22 -17
- package/src/resources/extensions/gsd/error-classifier.ts +11 -0
- package/src/resources/extensions/gsd/git-service.ts +1 -0
- package/src/resources/extensions/gsd/gitignore.ts +3 -0
- package/src/resources/extensions/gsd/gsd-db.ts +173 -2373
- package/src/resources/extensions/gsd/guidance.ts +139 -0
- package/src/resources/extensions/gsd/guided-flow.ts +50 -5
- package/src/resources/extensions/gsd/mcp-tool-name.ts +6 -11
- package/src/resources/extensions/gsd/memory-consolidation-scanner.ts +1 -1
- package/src/resources/extensions/gsd/migrate/safety.ts +18 -7
- package/src/resources/extensions/gsd/migration-auto-check.ts +28 -3
- package/src/resources/extensions/gsd/model-cost-table.ts +1 -0
- package/src/resources/extensions/gsd/model-router.ts +3 -0
- package/src/resources/extensions/gsd/notification-store.ts +26 -3
- package/src/resources/extensions/gsd/parallel-merge.ts +12 -9
- package/src/resources/extensions/gsd/parallel-monitor-overlay.ts +10 -7
- package/src/resources/extensions/gsd/paths.ts +42 -22
- package/src/resources/extensions/gsd/pre-execution-checks.ts +109 -3
- package/src/resources/extensions/gsd/preferences-models.ts +10 -46
- package/src/resources/extensions/gsd/preferences.ts +18 -0
- package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/refine-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/replan-slice.md +1 -1
- package/src/resources/extensions/gsd/provider-error-guidance.ts +4 -9
- package/src/resources/extensions/gsd/provider-switch-observer.ts +1 -1
- package/src/resources/extensions/gsd/publication.ts +122 -0
- package/src/resources/extensions/gsd/recovery-classification.ts +47 -88
- package/src/resources/extensions/gsd/safety/evidence-collector.ts +36 -4
- package/src/resources/extensions/gsd/safety/evidence-cross-ref.ts +7 -2
- package/src/resources/extensions/gsd/safety/file-change-validator.ts +14 -0
- package/src/resources/extensions/gsd/state-transition-matrix.ts +42 -0
- package/src/resources/extensions/gsd/state.ts +4 -21
- package/src/resources/extensions/gsd/status-guards.ts +59 -8
- package/src/resources/extensions/gsd/stop-notice.ts +75 -0
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +123 -0
- package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +3 -1
- package/src/resources/extensions/gsd/tests/auto-post-unit-evidence-crossref-4909.test.ts +46 -0
- package/src/resources/extensions/gsd/tests/auto-worktree-registry.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/auto-worktree-repair.test.ts +4 -2
- package/src/resources/extensions/gsd/tests/checkout-branch-stash-guard.test.ts +66 -1
- package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +44 -0
- package/src/resources/extensions/gsd/tests/commands-verdict.test.ts +8 -7
- package/src/resources/extensions/gsd/tests/evidence-xref-gsd-exec.test.ts +157 -0
- package/src/resources/extensions/gsd/tests/file-change-validator.test.ts +33 -1
- package/src/resources/extensions/gsd/tests/guidance.test.ts +125 -0
- package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +51 -4
- package/src/resources/extensions/gsd/tests/integration/auto-worktree.test.ts +54 -1
- package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +85 -1
- package/src/resources/extensions/gsd/tests/notification-store.test.ts +32 -0
- package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +193 -1
- package/src/resources/extensions/gsd/tests/provider-error-guidance.test.ts +3 -3
- package/src/resources/extensions/gsd/tests/publication.test.ts +120 -0
- package/src/resources/extensions/gsd/tests/recovery-classification-illegal-transition.test.ts +30 -0
- package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +248 -1
- package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +1 -0
- package/src/resources/extensions/gsd/tests/safety-harness-false-positives.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/session-switch-clears-pending-autostart.test.ts +108 -0
- package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +43 -6
- package/src/resources/extensions/gsd/tests/state-transition-matrix.test.ts +36 -0
- package/src/resources/extensions/gsd/tests/status-guards.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/stop-notice.test.ts +70 -0
- package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +8 -0
- package/src/resources/extensions/gsd/tests/tool-surface-readiness.test.ts +155 -0
- package/src/resources/extensions/gsd/tests/unit-closeout.test.ts +209 -0
- package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +23 -2
- package/src/resources/extensions/gsd/tests/unit-registry.test.ts +163 -0
- package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/worktree-lifecycle.test.ts +41 -4
- package/src/resources/extensions/gsd/tests/worktree-manager.test.ts +22 -1
- package/src/resources/extensions/gsd/tests/worktree-placement.test.ts +113 -0
- package/src/resources/extensions/gsd/tests/worktree-reentry.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/worktree-safety.test.ts +3 -1
- package/src/resources/extensions/gsd/tests/worktree-symlink-removal.test.ts +12 -6
- package/src/resources/extensions/gsd/tests/worktree-teardown-safety.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/write-gate.test.ts +42 -0
- package/src/resources/extensions/gsd/tool-surface-readiness.ts +76 -0
- package/src/resources/extensions/gsd/tools/complete-slice.ts +23 -58
- package/src/resources/extensions/gsd/tools/exec-tool.ts +5 -8
- package/src/resources/extensions/gsd/tools/plan-slice.ts +12 -6
- package/src/resources/extensions/gsd/tools/reopen-milestone.ts +11 -38
- package/src/resources/extensions/gsd/tools/reopen-slice.ts +14 -42
- package/src/resources/extensions/gsd/tools/skip-slice.ts +18 -44
- package/src/resources/extensions/gsd/undo.ts +9 -8
- package/src/resources/extensions/gsd/unit-closeout.ts +201 -0
- package/src/resources/extensions/gsd/unit-context-composer.ts +12 -1
- package/src/resources/extensions/gsd/unit-context-manifest.ts +4 -28
- package/src/resources/extensions/gsd/unit-registry.ts +425 -0
- package/src/resources/extensions/gsd/unit-tool-contracts.ts +27 -192
- package/src/resources/extensions/gsd/workflow-tool-surface.ts +4 -1
- package/src/resources/extensions/gsd/worktree-git-recovery.ts +314 -0
- package/src/resources/extensions/gsd/worktree-lifecycle.ts +10 -1
- package/src/resources/extensions/gsd/worktree-manager.ts +47 -28
- package/src/resources/extensions/gsd/worktree-placement.ts +63 -0
- package/src/resources/extensions/gsd/worktree-reentry.ts +10 -7
- package/src/resources/extensions/gsd/worktree-root.ts +29 -6
- package/src/resources/extensions/gsd/worktree-safety.ts +8 -5
- package/src/resources/extensions/gsd/worktree-session-state.ts +11 -11
- package/src/resources/skills/gsd-browser/SKILL.md +1 -1
- /package/dist/web/standalone/.next/static/{DUFWcMFRH3iXh7d2fbrOF → 2p9Rv9pQflAxCBbGVI2vb}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{DUFWcMFRH3iXh7d2fbrOF → 2p9Rv9pQflAxCBbGVI2vb}/_ssgManifest.js +0 -0
|
@@ -1,187 +1,14 @@
|
|
|
1
1
|
// Project/App: gsd-pi
|
|
2
|
-
// File Purpose:
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
export const RUN_UAT_READ_ONLY_TOOL_NAMES = [
|
|
11
|
-
"find",
|
|
12
|
-
"glob",
|
|
13
|
-
"grep",
|
|
14
|
-
"ls",
|
|
15
|
-
"read",
|
|
16
|
-
];
|
|
17
|
-
export const RUN_UAT_BROWSER_TOOL_NAMES = [
|
|
18
|
-
"browser_navigate",
|
|
19
|
-
"browser_click",
|
|
20
|
-
"browser_type",
|
|
21
|
-
"browser_fill_form",
|
|
22
|
-
"browser_click_ref",
|
|
23
|
-
"browser_fill_ref",
|
|
24
|
-
"browser_wait_for",
|
|
25
|
-
"browser_assert",
|
|
26
|
-
"browser_verify",
|
|
27
|
-
"browser_screenshot",
|
|
28
|
-
"browser_snapshot_refs",
|
|
29
|
-
"browser_find",
|
|
30
|
-
"browser_get_console_logs",
|
|
31
|
-
"browser_get_network_logs",
|
|
32
|
-
"browser_evaluate",
|
|
33
|
-
"browser_reload",
|
|
34
|
-
"browser_batch",
|
|
35
|
-
"browser_act",
|
|
36
|
-
];
|
|
2
|
+
// File Purpose: Unit-to-tool contract views derived from the Unit Registry (ADR-033).
|
|
3
|
+
//
|
|
4
|
+
// The contract data lives in unit-registry.ts (one Unit Descriptor per unit
|
|
5
|
+
// type). This file keeps the established import surface: the derived
|
|
6
|
+
// `UNIT_TOOL_CONTRACTS` table and its accessor functions are unchanged for
|
|
7
|
+
// consumers.
|
|
8
|
+
import { UNIT_REGISTRY, } from "./unit-registry.js";
|
|
9
|
+
export { RUN_UAT_WORKFLOW_TOOL_NAMES, RUN_UAT_READ_ONLY_TOOL_NAMES, RUN_UAT_BROWSER_TOOL_NAMES, } from "./unit-registry.js";
|
|
37
10
|
export const RUN_UAT_TOOL_PRESENTATION_PLAN_ID = "run-uat/default-v1";
|
|
38
|
-
export const UNIT_TOOL_CONTRACTS =
|
|
39
|
-
"research-milestone": {
|
|
40
|
-
allowedGsdTools: ["gsd_summary_save", "gsd_decision_save"],
|
|
41
|
-
requiredWorkflowTools: ["gsd_summary_save"],
|
|
42
|
-
},
|
|
43
|
-
"plan-milestone": {
|
|
44
|
-
allowedGsdTools: [
|
|
45
|
-
"gsd_milestone_status",
|
|
46
|
-
"gsd_plan_milestone",
|
|
47
|
-
"gsd_plan_slice",
|
|
48
|
-
"gsd_decision_save",
|
|
49
|
-
"gsd_requirement_update",
|
|
50
|
-
],
|
|
51
|
-
requiredWorkflowTools: ["gsd_milestone_status", "gsd_plan_milestone", "gsd_plan_slice"],
|
|
52
|
-
},
|
|
53
|
-
"discuss-milestone": {
|
|
54
|
-
allowedGsdTools: [
|
|
55
|
-
"gsd_summary_save",
|
|
56
|
-
"gsd_decision_save",
|
|
57
|
-
"gsd_requirement_save",
|
|
58
|
-
"gsd_requirement_update",
|
|
59
|
-
"gsd_plan_milestone",
|
|
60
|
-
"gsd_milestone_generate_id",
|
|
61
|
-
],
|
|
62
|
-
requiredWorkflowTools: [
|
|
63
|
-
"ask_user_questions",
|
|
64
|
-
"gsd_summary_save",
|
|
65
|
-
"gsd_requirement_save",
|
|
66
|
-
"gsd_requirement_update",
|
|
67
|
-
"gsd_plan_milestone",
|
|
68
|
-
"gsd_milestone_generate_id",
|
|
69
|
-
],
|
|
70
|
-
},
|
|
71
|
-
"discuss-slice": {
|
|
72
|
-
allowedGsdTools: ["gsd_summary_save", "gsd_decision_save"],
|
|
73
|
-
requiredWorkflowTools: ["ask_user_questions", "gsd_summary_save"],
|
|
74
|
-
},
|
|
75
|
-
"validate-milestone": {
|
|
76
|
-
allowedGsdTools: ["gsd_milestone_status", "gsd_validate_milestone", "gsd_reassess_roadmap", "subagent"],
|
|
77
|
-
requiredWorkflowTools: ["gsd_milestone_status", "gsd_validate_milestone", "gsd_reassess_roadmap"],
|
|
78
|
-
},
|
|
79
|
-
"complete-milestone": {
|
|
80
|
-
allowedGsdTools: [
|
|
81
|
-
"gsd_milestone_status",
|
|
82
|
-
"gsd_requirement_update",
|
|
83
|
-
"gsd_summary_save",
|
|
84
|
-
"gsd_complete_milestone",
|
|
85
|
-
"subagent",
|
|
86
|
-
],
|
|
87
|
-
requiredWorkflowTools: [
|
|
88
|
-
"gsd_milestone_status",
|
|
89
|
-
"gsd_requirement_update",
|
|
90
|
-
"gsd_summary_save",
|
|
91
|
-
"gsd_complete_milestone",
|
|
92
|
-
],
|
|
93
|
-
},
|
|
94
|
-
"research-slice": {
|
|
95
|
-
allowedGsdTools: ["gsd_summary_save", "gsd_decision_save"],
|
|
96
|
-
requiredWorkflowTools: ["gsd_summary_save"],
|
|
97
|
-
},
|
|
98
|
-
"plan-slice": {
|
|
99
|
-
allowedGsdTools: ["gsd_plan_slice", "gsd_reassess_roadmap", "gsd_decision_save"],
|
|
100
|
-
requiredWorkflowTools: ["gsd_plan_slice", "gsd_reassess_roadmap"],
|
|
101
|
-
},
|
|
102
|
-
"refine-slice": {
|
|
103
|
-
allowedGsdTools: ["gsd_plan_slice", "gsd_decision_save"],
|
|
104
|
-
requiredWorkflowTools: ["gsd_plan_slice"],
|
|
105
|
-
},
|
|
106
|
-
"replan-slice": {
|
|
107
|
-
allowedGsdTools: ["gsd_replan_slice", "gsd_decision_save"],
|
|
108
|
-
requiredWorkflowTools: ["gsd_replan_slice"],
|
|
109
|
-
},
|
|
110
|
-
"complete-slice": {
|
|
111
|
-
allowedGsdTools: [
|
|
112
|
-
"gsd_slice_complete",
|
|
113
|
-
"gsd_task_reopen",
|
|
114
|
-
"gsd_replan_slice",
|
|
115
|
-
"gsd_decision_save",
|
|
116
|
-
"gsd_requirement_update",
|
|
117
|
-
"gsd_summary_save",
|
|
118
|
-
"subagent",
|
|
119
|
-
],
|
|
120
|
-
requiredWorkflowTools: [
|
|
121
|
-
"gsd_slice_complete",
|
|
122
|
-
"gsd_task_reopen",
|
|
123
|
-
"gsd_replan_slice",
|
|
124
|
-
"gsd_requirement_update",
|
|
125
|
-
"gsd_summary_save",
|
|
126
|
-
],
|
|
127
|
-
forbiddenGsdTools: {
|
|
128
|
-
gsd_uat_result_save: "Run UAT owns persisted UAT Assessment.",
|
|
129
|
-
},
|
|
130
|
-
},
|
|
131
|
-
"reassess-roadmap": {
|
|
132
|
-
allowedGsdTools: ["gsd_milestone_status", "gsd_reassess_roadmap"],
|
|
133
|
-
requiredWorkflowTools: ["gsd_milestone_status", "gsd_reassess_roadmap"],
|
|
134
|
-
},
|
|
135
|
-
"execute-task": {
|
|
136
|
-
allowedGsdTools: ["gsd_task_complete", "gsd_decision_save"],
|
|
137
|
-
requiredWorkflowTools: ["gsd_task_complete"],
|
|
138
|
-
},
|
|
139
|
-
"execute-task-simple": {
|
|
140
|
-
allowedGsdTools: ["gsd_task_complete", "gsd_decision_save"],
|
|
141
|
-
requiredWorkflowTools: ["gsd_task_complete"],
|
|
142
|
-
},
|
|
143
|
-
"reactive-execute": {
|
|
144
|
-
allowedGsdTools: ["gsd_task_complete", "gsd_summary_save", "gsd_decision_save"],
|
|
145
|
-
requiredWorkflowTools: ["gsd_task_complete", "gsd_summary_save"],
|
|
146
|
-
},
|
|
147
|
-
"run-uat": {
|
|
148
|
-
allowedGsdTools: [...RUN_UAT_WORKFLOW_TOOL_NAMES, "subagent"],
|
|
149
|
-
requiredWorkflowTools: [...RUN_UAT_WORKFLOW_TOOL_NAMES],
|
|
150
|
-
forbiddenGsdTools: {
|
|
151
|
-
gsd_exec: "Use gsd_uat_exec so acceptance evidence is typed as UAT-owned.",
|
|
152
|
-
gsd_save_gate_result: "gsd_uat_result_save owns the aggregate UAT gate.",
|
|
153
|
-
gsd_summary_save: "gsd_uat_result_save owns persisted UAT Assessment writes.",
|
|
154
|
-
},
|
|
155
|
-
},
|
|
156
|
-
"gate-evaluate": {
|
|
157
|
-
allowedGsdTools: ["gsd_save_gate_result"],
|
|
158
|
-
requiredWorkflowTools: ["gsd_save_gate_result"],
|
|
159
|
-
},
|
|
160
|
-
"rewrite-docs": {
|
|
161
|
-
allowedGsdTools: ["gsd_summary_save", "gsd_decision_save"],
|
|
162
|
-
requiredWorkflowTools: [],
|
|
163
|
-
},
|
|
164
|
-
"workflow-preferences": {
|
|
165
|
-
allowedGsdTools: ["gsd_summary_save"],
|
|
166
|
-
requiredWorkflowTools: [],
|
|
167
|
-
},
|
|
168
|
-
"discuss-project": {
|
|
169
|
-
allowedGsdTools: ["gsd_summary_save", "gsd_decision_save", "gsd_requirement_save"],
|
|
170
|
-
requiredWorkflowTools: ["ask_user_questions", "gsd_summary_save"],
|
|
171
|
-
},
|
|
172
|
-
"discuss-requirements": {
|
|
173
|
-
allowedGsdTools: ["gsd_requirement_save", "gsd_summary_save"],
|
|
174
|
-
requiredWorkflowTools: ["ask_user_questions", "gsd_requirement_save", "gsd_summary_save"],
|
|
175
|
-
},
|
|
176
|
-
"research-decision": {
|
|
177
|
-
allowedGsdTools: ["gsd_summary_save"],
|
|
178
|
-
requiredWorkflowTools: ["ask_user_questions"],
|
|
179
|
-
},
|
|
180
|
-
"research-project": {
|
|
181
|
-
allowedGsdTools: ["gsd_summary_save", "gsd_decision_save"],
|
|
182
|
-
requiredWorkflowTools: [],
|
|
183
|
-
},
|
|
184
|
-
};
|
|
11
|
+
export const UNIT_TOOL_CONTRACTS = Object.fromEntries(Object.entries(UNIT_REGISTRY).flatMap(([unitType, descriptor]) => descriptor.toolContract ? [[unitType, descriptor.toolContract]] : []));
|
|
185
12
|
export const AUTO_UNIT_SCOPED_TOOLS = Object.fromEntries(Object.entries(UNIT_TOOL_CONTRACTS).map(([unitType, contract]) => [unitType, contract.allowedGsdTools]));
|
|
186
13
|
export function getUnitToolSurfaceContract(unitType) {
|
|
187
14
|
return UNIT_TOOL_CONTRACTS[unitType];
|
|
@@ -8,7 +8,7 @@ export const WORKFLOW_TOOL_ALIAS_NAMES = CONTRACT_WORKFLOW_TOOL_ALIAS_NAMES;
|
|
|
8
8
|
export const DB_WORKFLOW_TOOL_NAMES = CONTRACT_WORKFLOW_TOOL_NAMES;
|
|
9
9
|
export const WORKFLOW_TOOL_ALIAS_PAIRS = WORKFLOW_TOOL_CONTRACTS.flatMap((tool) => tool.aliases.map((alias) => ({ canonical: tool.canonicalName, alias })));
|
|
10
10
|
export const WORKFLOW_TOOL_ALIAS_TO_CANONICAL = Object.fromEntries(WORKFLOW_TOOL_ALIAS_PAIRS.map(({ alias, canonical }) => [alias, canonical]));
|
|
11
|
-
const WORKFLOW_MCP_ADAPTER_TOOL_NAMES = [
|
|
11
|
+
export const WORKFLOW_MCP_ADAPTER_TOOL_NAMES = [
|
|
12
12
|
"gsd_cancel",
|
|
13
13
|
"gsd_captures",
|
|
14
14
|
"ask_user_questions",
|
|
@@ -0,0 +1,293 @@
|
|
|
1
|
+
// Project/App: gsd-pi
|
|
2
|
+
// File Purpose: Git checkout/stash/merge-state recovery primitives for worktree operations.
|
|
3
|
+
/**
|
|
4
|
+
* Worktree Git Recovery — the recurring-bug hot spot, in one place.
|
|
5
|
+
*
|
|
6
|
+
* Owns the verbs that recover a repository from interrupted or conflicting
|
|
7
|
+
* git operations during worktree transitions:
|
|
8
|
+
*
|
|
9
|
+
* - `checkoutBranchWithStashGuard` — branch switch with stash protection,
|
|
10
|
+
* including the stash-pop EEXIST collision recovery for untracked files
|
|
11
|
+
* (force-checkout + targeted stash drop; #645 broadened it beyond `.gsd/`,
|
|
12
|
+
* guarded by "no non-.gsd unmerged entries remain").
|
|
13
|
+
* - `removeMergeStateFiles` — clears SQUASH_MSG / MERGE_HEAD / etc. left by
|
|
14
|
+
* a failed merge so subsequent merges don't fail on stale state.
|
|
15
|
+
* - `cleanupConflictState` — merge-abort + index reset + state-file cleanup
|
|
16
|
+
* after a conflicted (including squash) merge.
|
|
17
|
+
* - stash helpers (`popStashByRef`, `stashRefFromError`,
|
|
18
|
+
* `stashAlreadyExistsFilesFromError`, `gsdJsonlFilesWithConflictMarkers`)
|
|
19
|
+
* used by the merge pipeline in auto-worktree.ts.
|
|
20
|
+
*
|
|
21
|
+
* Extracted from auto-worktree.ts so recovery fixes land here instead of as
|
|
22
|
+
* embedded special cases in a 2,600-line orchestration module, and so the
|
|
23
|
+
* rules can be tested against scripted git states.
|
|
24
|
+
*
|
|
25
|
+
* The State Reconciliation drift repair (`state-reconciliation/drift/
|
|
26
|
+
* merge-state.ts`) keeps its own merge-state primitive by design — drift
|
|
27
|
+
* repairs own their raw primitives (see CONTEXT.md, Drift repair).
|
|
28
|
+
*/
|
|
29
|
+
import { execFileSync } from "node:child_process";
|
|
30
|
+
import { existsSync, readFileSync, unlinkSync } from "node:fs";
|
|
31
|
+
import { isAbsolute, join, resolve } from "node:path";
|
|
32
|
+
import { debugLog } from "./debug-logger.js";
|
|
33
|
+
import { logError, logWarning } from "./workflow-logger.js";
|
|
34
|
+
import { nativeAddPaths, nativeCheckoutBranch, nativeConflictFiles, nativeLsFiles, nativeMergeAbort, nativeWorkingTreeStatus, } from "./native-git-bridge.js";
|
|
35
|
+
import { resolveGitDir } from "./worktree-manager.js";
|
|
36
|
+
/**
|
|
37
|
+
* Pop the stash entry created with `stashMarker` in its subject, resolving it
|
|
38
|
+
* to a concrete `stash@{n}` ref first so a concurrent stash push cannot make
|
|
39
|
+
* `git stash pop` grab the wrong entry.
|
|
40
|
+
*
|
|
41
|
+
* If `stashMarker` is null or no longer present in the stash list (e.g. a
|
|
42
|
+
* concurrent process popped/dropped it), leaves the stash list untouched and
|
|
43
|
+
* returns null.
|
|
44
|
+
*
|
|
45
|
+
* Throws on pop failure so callers can handle conflict cases the same way
|
|
46
|
+
* they would with the prior `git stash pop` form. When throwing after a
|
|
47
|
+
* targeted pop attempt, the error is annotated with the targeted stash ref.
|
|
48
|
+
*
|
|
49
|
+
* (Issue #4980 HIGH-6)
|
|
50
|
+
*/
|
|
51
|
+
export function popStashByRef(basePath, stashMarker) {
|
|
52
|
+
let popArg = null;
|
|
53
|
+
if (stashMarker) {
|
|
54
|
+
try {
|
|
55
|
+
const list = execFileSync("git", ["stash", "list", "--format=%gd%x00%s"], {
|
|
56
|
+
cwd: basePath,
|
|
57
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
58
|
+
encoding: "utf-8",
|
|
59
|
+
}).trim().split("\n").filter(Boolean);
|
|
60
|
+
for (const entry of list) {
|
|
61
|
+
const [ref, subject] = entry.split("\0");
|
|
62
|
+
if (ref && subject?.includes(stashMarker)) {
|
|
63
|
+
popArg = ref;
|
|
64
|
+
break;
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
catch (err) {
|
|
69
|
+
logWarning("worktree", `stash list lookup failed; leaving stash untouched: ${err instanceof Error ? err.message : String(err)}`);
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
if (!popArg) {
|
|
73
|
+
logWarning("worktree", "recorded stash entry could not be resolved; skipping automatic pop");
|
|
74
|
+
return null;
|
|
75
|
+
}
|
|
76
|
+
try {
|
|
77
|
+
execFileSync("git", ["stash", "pop", popArg], {
|
|
78
|
+
cwd: basePath,
|
|
79
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
80
|
+
encoding: "utf-8",
|
|
81
|
+
});
|
|
82
|
+
}
|
|
83
|
+
catch (err) {
|
|
84
|
+
if (err && typeof err === "object") {
|
|
85
|
+
err.stashRef = popArg;
|
|
86
|
+
}
|
|
87
|
+
throw err;
|
|
88
|
+
}
|
|
89
|
+
return popArg;
|
|
90
|
+
}
|
|
91
|
+
/**
|
|
92
|
+
* Extract a stash ref annotation injected by popStashByRef() when git stash
|
|
93
|
+
* pop fails and we need to conditionally drop the exact stash entry later.
|
|
94
|
+
*/
|
|
95
|
+
export function stashRefFromError(err) {
|
|
96
|
+
if (!err || typeof err !== "object")
|
|
97
|
+
return null;
|
|
98
|
+
const stashRef = err.stashRef;
|
|
99
|
+
return typeof stashRef === "string" && stashRef.length > 0 ? stashRef : null;
|
|
100
|
+
}
|
|
101
|
+
export function stashAlreadyExistsFilesFromError(err) {
|
|
102
|
+
if (!err || typeof err !== "object")
|
|
103
|
+
return [];
|
|
104
|
+
const stderr = err.stderr;
|
|
105
|
+
const stderrText = typeof stderr === "string"
|
|
106
|
+
? stderr
|
|
107
|
+
: stderr instanceof Uint8Array
|
|
108
|
+
? Buffer.from(stderr).toString("utf-8")
|
|
109
|
+
: "";
|
|
110
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
111
|
+
const text = `${stderrText}\n${message}`;
|
|
112
|
+
const files = new Set();
|
|
113
|
+
for (const line of text.split("\n")) {
|
|
114
|
+
const m = line.match(/^(.*?)\s+already exists, no checkout\s*$/i);
|
|
115
|
+
if (!m)
|
|
116
|
+
continue;
|
|
117
|
+
const filePath = m[1]?.trim();
|
|
118
|
+
if (filePath)
|
|
119
|
+
files.add(filePath);
|
|
120
|
+
}
|
|
121
|
+
return [...files];
|
|
122
|
+
}
|
|
123
|
+
/**
|
|
124
|
+
* Detect whether an on-disk file still contains unresolved merge conflict
|
|
125
|
+
* markers from a failed stash-pop or merge attempt.
|
|
126
|
+
*
|
|
127
|
+
* Returns false when the file cannot be read.
|
|
128
|
+
*/
|
|
129
|
+
export function hasConflictMarkers(filePath) {
|
|
130
|
+
try {
|
|
131
|
+
const content = readFileSync(filePath, "utf-8");
|
|
132
|
+
return content.includes("<<<<<<<") && content.includes("=======") && content.includes(">>>>>>>");
|
|
133
|
+
}
|
|
134
|
+
catch {
|
|
135
|
+
return false;
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
export function gsdJsonlFilesWithConflictMarkers(basePath) {
|
|
139
|
+
return nativeLsFiles(basePath, ".gsd/*.jsonl").filter((f) => hasConflictMarkers(join(basePath, f)));
|
|
140
|
+
}
|
|
141
|
+
export function removeMergeStateFiles(basePath, contextLabel) {
|
|
142
|
+
try {
|
|
143
|
+
for (const f of ["SQUASH_MSG", "MERGE_MSG", "MERGE_MODE", "MERGE_HEAD", "AUTO_MERGE"]) {
|
|
144
|
+
const rawPath = execFileSync("git", ["rev-parse", "--git-path", f], {
|
|
145
|
+
cwd: basePath,
|
|
146
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
147
|
+
encoding: "utf-8",
|
|
148
|
+
}).trim();
|
|
149
|
+
const p = rawPath.length > 0
|
|
150
|
+
? (isAbsolute(rawPath) ? rawPath : resolve(basePath, rawPath))
|
|
151
|
+
: join(resolveGitDir(basePath), f);
|
|
152
|
+
if (existsSync(p))
|
|
153
|
+
unlinkSync(p);
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
catch (err) {
|
|
157
|
+
logError("worktree", `${contextLabel} merge state cleanup failed: ${err instanceof Error ? err.message : String(err)}`);
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
export function cleanupConflictState(basePath) {
|
|
161
|
+
// Merge conflicts can leave unmerged index entries; merge-abort alone is not
|
|
162
|
+
// enough for squash merges (MERGE_HEAD is never written). Reset the merge
|
|
163
|
+
// index, then remove merge message files that native/libgit2 paths may have
|
|
164
|
+
// created.
|
|
165
|
+
try {
|
|
166
|
+
nativeMergeAbort(basePath);
|
|
167
|
+
}
|
|
168
|
+
catch (err) {
|
|
169
|
+
// MERGE_HEAD absent (squash merge path) — abort is a no-op, which is fine.
|
|
170
|
+
debugLog("conflict-cleanup:merge-abort-skipped", {
|
|
171
|
+
error: err instanceof Error ? err.message : String(err),
|
|
172
|
+
});
|
|
173
|
+
}
|
|
174
|
+
try {
|
|
175
|
+
execFileSync("git", ["reset", "--merge"], {
|
|
176
|
+
cwd: basePath,
|
|
177
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
178
|
+
encoding: "utf-8",
|
|
179
|
+
});
|
|
180
|
+
}
|
|
181
|
+
catch (err) {
|
|
182
|
+
logError("worktree", `git reset --merge failed after merge conflict: ${err instanceof Error ? err.message : String(err)}`);
|
|
183
|
+
}
|
|
184
|
+
removeMergeStateFiles(basePath, "conflict");
|
|
185
|
+
}
|
|
186
|
+
export function checkoutBranchWithStashGuard(basePath, branch, reason) {
|
|
187
|
+
let stashMarker = null;
|
|
188
|
+
let stashed = false;
|
|
189
|
+
const status = nativeWorkingTreeStatus(basePath).trim();
|
|
190
|
+
if (status.length > 0) {
|
|
191
|
+
stashMarker = `gsd-checkout-stash:${reason}:${process.pid}:${Date.now()}:${process.hrtime.bigint().toString(36)}`;
|
|
192
|
+
const stashListBefore = execFileSync("git", ["stash", "list"], {
|
|
193
|
+
cwd: basePath,
|
|
194
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
195
|
+
encoding: "utf-8",
|
|
196
|
+
});
|
|
197
|
+
execFileSync("git", ["stash", "push", "--include-untracked", "-m", `gsd: checkout stash [${stashMarker}]`], {
|
|
198
|
+
cwd: basePath,
|
|
199
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
200
|
+
encoding: "utf-8",
|
|
201
|
+
});
|
|
202
|
+
const stashListAfter = execFileSync("git", ["stash", "list"], {
|
|
203
|
+
cwd: basePath,
|
|
204
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
205
|
+
encoding: "utf-8",
|
|
206
|
+
});
|
|
207
|
+
stashed = stashListAfter !== stashListBefore;
|
|
208
|
+
}
|
|
209
|
+
// Checkout and stash-restore are split so we can distinguish two failure
|
|
210
|
+
// modes: (a) checkout failed → HEAD did not move, restore stash and rethrow;
|
|
211
|
+
// (b) checkout succeeded but stash pop failed → HEAD moved to `branch` but
|
|
212
|
+
// the working-tree changes remain in the stash list. We surface a distinct
|
|
213
|
+
// error in case (b) so callers don't assume the branch switch was rolled back.
|
|
214
|
+
try {
|
|
215
|
+
nativeCheckoutBranch(basePath, branch);
|
|
216
|
+
}
|
|
217
|
+
catch (checkoutErr) {
|
|
218
|
+
if (stashed) {
|
|
219
|
+
try {
|
|
220
|
+
popStashByRef(basePath, stashMarker);
|
|
221
|
+
}
|
|
222
|
+
catch (restoreErr) {
|
|
223
|
+
logWarning("worktree", `git stash pop failed during checkout restore: ${restoreErr instanceof Error ? restoreErr.message : String(restoreErr)}`);
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
throw checkoutErr;
|
|
227
|
+
}
|
|
228
|
+
if (stashed) {
|
|
229
|
+
try {
|
|
230
|
+
popStashByRef(basePath, stashMarker);
|
|
231
|
+
}
|
|
232
|
+
catch (popErr) {
|
|
233
|
+
const msg = popErr instanceof Error ? popErr.message : String(popErr);
|
|
234
|
+
const stderr = popErr && typeof popErr === "object"
|
|
235
|
+
? popErr.stderr
|
|
236
|
+
: undefined;
|
|
237
|
+
const stderrText = typeof stderr === "string"
|
|
238
|
+
? stderr
|
|
239
|
+
: stderr instanceof Uint8Array
|
|
240
|
+
? Buffer.from(stderr).toString("utf-8")
|
|
241
|
+
: "";
|
|
242
|
+
const stashPopMessage = `${stderrText}\n${msg}`.trim();
|
|
243
|
+
const alreadyExists = stashAlreadyExistsFilesFromError(popErr);
|
|
244
|
+
const isUntrackedRestoreFailure = stashPopMessage.includes("could not restore untracked files from stash");
|
|
245
|
+
const stashRefForDrop = stashRefFromError(popErr);
|
|
246
|
+
const allConflictFiles = nativeConflictFiles(basePath);
|
|
247
|
+
const nonGsdUnmerged = allConflictFiles.filter((f) => !f.startsWith(".gsd/"));
|
|
248
|
+
const gsdUnmerged = allConflictFiles.filter((f) => f.startsWith(".gsd/"));
|
|
249
|
+
const gsdContentConflicts = isUntrackedRestoreFailure
|
|
250
|
+
? gsdJsonlFilesWithConflictMarkers(basePath)
|
|
251
|
+
: [];
|
|
252
|
+
// Resolve ALL untracked-collision files by accepting HEAD — files in
|
|
253
|
+
// alreadyExists were untracked on the source branch by definition of the
|
|
254
|
+
// "already exists, no checkout" failure, so target HEAD is authoritative.
|
|
255
|
+
// gsdUnmerged: .gsd/ index conflicts left by the partial stash pop are
|
|
256
|
+
// also resolved via HEAD — .gsd/ runtime state is always authoritative
|
|
257
|
+
// on the target branch, so accepting HEAD is safe here too.
|
|
258
|
+
const resolvable = [...new Set([...alreadyExists, ...gsdContentConflicts, ...gsdUnmerged])];
|
|
259
|
+
if (isUntrackedRestoreFailure &&
|
|
260
|
+
resolvable.length > 0 &&
|
|
261
|
+
nonGsdUnmerged.length === 0) {
|
|
262
|
+
for (const f of resolvable) {
|
|
263
|
+
execFileSync("git", ["checkout", "HEAD", "--", f], {
|
|
264
|
+
cwd: basePath,
|
|
265
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
266
|
+
encoding: "utf-8",
|
|
267
|
+
});
|
|
268
|
+
nativeAddPaths(basePath, [f]);
|
|
269
|
+
}
|
|
270
|
+
if (stashRefForDrop) {
|
|
271
|
+
try {
|
|
272
|
+
execFileSync("git", ["stash", "drop", stashRefForDrop], {
|
|
273
|
+
cwd: basePath,
|
|
274
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
275
|
+
encoding: "utf-8",
|
|
276
|
+
});
|
|
277
|
+
}
|
|
278
|
+
catch (err) { /* stash may already be consumed */
|
|
279
|
+
logWarning("worktree", `git stash drop failed: ${err instanceof Error ? err.message : String(err)}`);
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
else {
|
|
283
|
+
logWarning("worktree", "recorded stash entry could not be resolved; skipping automatic drop");
|
|
284
|
+
}
|
|
285
|
+
return;
|
|
286
|
+
}
|
|
287
|
+
const wrapped = new Error(`checkout to '${branch}' succeeded but stash restore failed; working tree changes remain in the stash list. Original error: ${msg}`);
|
|
288
|
+
if (stashRefForDrop)
|
|
289
|
+
wrapped.stashRef = stashRefForDrop;
|
|
290
|
+
throw wrapped;
|
|
291
|
+
}
|
|
292
|
+
}
|
|
293
|
+
}
|
|
@@ -327,7 +327,15 @@ export function _enterMilestoneCore(s, deps, milestoneId, ctx, opts = {}) {
|
|
|
327
327
|
// Handles the case where originalBasePath is falsy and basePath is itself
|
|
328
328
|
// a worktree path — prevents double-nested worktree paths (#3729).
|
|
329
329
|
const basePath = resolveWorktreeProjectRoot(s.basePath, s.originalBasePath);
|
|
330
|
-
|
|
330
|
+
// A stranded-recovery session that adopted the milestone branch in the
|
|
331
|
+
// project root must keep re-entering in that mode: the root checkout holds
|
|
332
|
+
// the branch, so creating the canonical worktree would fail with "already
|
|
333
|
+
// in use by another worktree". The override clears when the recovered
|
|
334
|
+
// milestone merges (_mergeAndExit), restoring configured isolation for
|
|
335
|
+
// subsequent milestones.
|
|
336
|
+
const mode = opts.modeOverride ??
|
|
337
|
+
s.strandedRecoveryIsolationMode ??
|
|
338
|
+
getIsolationMode(basePath);
|
|
331
339
|
if (s.isolationDegraded) {
|
|
332
340
|
if (mode === "worktree") {
|
|
333
341
|
try {
|