gsd-pi 2.82.0-dev.725028083 → 2.82.0-dev.98ea09b1e
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/README.md +4 -3
- package/dist/resources/.managed-resources-content-hash +1 -1
- package/dist/resources/GSD-WORKFLOW.md +10 -1
- package/dist/resources/extensions/claude-code-cli/partial-builder.js +2 -1
- package/dist/resources/extensions/cmux/index.js +5 -0
- package/dist/resources/extensions/gsd/auto/infra-errors.js +9 -3
- package/dist/resources/extensions/gsd/auto/loop.js +5 -5
- package/dist/resources/extensions/gsd/auto/orchestrator.js +124 -6
- package/dist/resources/extensions/gsd/auto/phases.js +8 -1
- package/dist/resources/extensions/gsd/auto/workflow-memory-pressure.js +12 -0
- package/dist/resources/extensions/gsd/auto-dispatch.js +13 -6
- package/dist/resources/extensions/gsd/auto-model-selection.js +2 -0
- package/dist/resources/extensions/gsd/auto-post-unit.js +233 -127
- package/dist/resources/extensions/gsd/auto-prompts.js +2 -2
- package/dist/resources/extensions/gsd/auto-recovery.js +31 -1
- package/dist/resources/extensions/gsd/auto-start.js +85 -12
- package/dist/resources/extensions/gsd/auto-verification.js +28 -22
- package/dist/resources/extensions/gsd/auto-worktree.js +111 -1
- package/dist/resources/extensions/gsd/auto.js +158 -55
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +4 -1
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +9 -8
- package/dist/resources/extensions/gsd/bootstrap/subagent-input.js +21 -9
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +16 -2
- package/dist/resources/extensions/gsd/clean-root-preflight.js +170 -8
- package/dist/resources/extensions/gsd/commands/catalog.js +4 -1
- package/dist/resources/extensions/gsd/commands/handlers/core.js +37 -0
- package/dist/resources/extensions/gsd/commands-bootstrap.js +5 -0
- package/dist/resources/extensions/gsd/crash-recovery.js +31 -5
- package/dist/resources/extensions/gsd/db/unit-dispatches.js +3 -2
- package/dist/resources/extensions/gsd/dispatch-guard.js +2 -2
- package/dist/resources/extensions/gsd/doctor-runtime-checks.js +28 -11
- package/dist/resources/extensions/gsd/doctor.js +2 -28
- package/dist/resources/extensions/gsd/export-html.js +27 -425
- package/dist/resources/extensions/gsd/git-service.js +39 -1
- package/dist/resources/extensions/gsd/gsd-db.js +1 -0
- package/dist/resources/extensions/gsd/guided-flow.js +13 -6
- package/dist/resources/extensions/gsd/md-importer.js +1 -1
- package/dist/resources/extensions/gsd/migrate/command.js +5 -0
- package/dist/resources/extensions/gsd/migrate/parsers.js +10 -0
- package/dist/resources/extensions/gsd/migrate/preview.js +9 -0
- package/dist/resources/extensions/gsd/migrate/transformer.js +51 -4
- package/dist/resources/extensions/gsd/migrate/writer.js +11 -1
- package/dist/resources/extensions/gsd/migration-auto-check.js +12 -17
- package/dist/resources/extensions/gsd/milestone-actions.js +11 -4
- package/dist/resources/extensions/gsd/native-git-bridge.js +48 -12
- package/dist/resources/extensions/gsd/post-execution-checks.js +73 -2
- package/dist/resources/extensions/gsd/pre-execution-checks.js +28 -1
- package/dist/resources/extensions/gsd/prompt-loader.js +1 -1
- package/dist/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
- package/dist/resources/extensions/gsd/prompts/complete-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/discuss-headless.md +8 -8
- package/dist/resources/extensions/gsd/prompts/discuss.md +9 -9
- package/dist/resources/extensions/gsd/prompts/guided-discuss-project.md +4 -4
- package/dist/resources/extensions/gsd/prompts/guided-discuss-requirements.md +3 -3
- package/dist/resources/extensions/gsd/prompts/plan-slice.md +4 -4
- package/dist/resources/extensions/gsd/prompts/queue.md +4 -4
- package/dist/resources/extensions/gsd/prompts/refine-slice.md +2 -2
- package/dist/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
- package/dist/resources/extensions/gsd/state-reconciliation/drift/merge-state.js +6 -1
- package/dist/resources/extensions/gsd/state-reconciliation/drift/project-md.js +9 -14
- package/dist/resources/extensions/gsd/state-reconciliation/drift/roadmap.js +19 -24
- package/dist/resources/extensions/gsd/status-guards.js +4 -0
- package/dist/resources/extensions/gsd/templates/plan.md +8 -5
- package/dist/resources/extensions/gsd/templates/task-plan.md +4 -2
- package/dist/resources/extensions/gsd/tools/complete-milestone.js +6 -8
- package/dist/resources/extensions/gsd/tools/complete-slice.js +6 -8
- package/dist/resources/extensions/gsd/tools/plan-milestone.js +7 -1
- package/dist/resources/extensions/gsd/tools/plan-slice.js +89 -14
- package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +119 -0
- package/dist/resources/extensions/gsd/unit-context-manifest.js +32 -10
- package/dist/resources/extensions/gsd/validation.js +23 -1
- package/dist/resources/extensions/gsd/verification-gate.js +68 -7
- package/dist/resources/extensions/gsd/verification-verdict.js +26 -0
- package/dist/resources/extensions/gsd/workflow-projections.js +6 -8
- package/dist/resources/extensions/gsd/worktree-lifecycle.js +54 -10
- package/dist/resources/extensions/shared/html-shell.js +388 -0
- package/dist/resources/extensions/subagent/index.js +448 -78
- package/dist/resources/extensions/subagent/launch.js +77 -0
- package/dist/resources/extensions/subagent/run-store.js +148 -0
- package/dist/resources/extensions/visual-brief/artifact-policy.js +29 -0
- package/dist/resources/extensions/visual-brief/extension-manifest.json +8 -0
- package/dist/resources/extensions/visual-brief/index.js +5 -0
- package/dist/resources/extensions/visual-brief/page-contract.js +124 -0
- package/dist/resources/extensions/visual-brief/prompts.js +140 -0
- 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 +14 -14
- package/dist/web/standalone/.next/build-manifest.json +3 -3
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/react-loadable-manifest.json +3 -3
- package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
- 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/page.js +2 -2
- package/dist/web/standalone/.next/server/app/_not-found/page.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +4 -7
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +4 -7
- 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 +4 -5
- 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 +2 -5
- package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +4 -7
- 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 +4 -7
- 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 +4 -5
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +2 -5
- package/dist/web/standalone/.next/server/app/page.js +2 -2
- package/dist/web/standalone/.next/server/app/page.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +14 -14
- package/dist/web/standalone/.next/server/chunks/4266.js +2 -0
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
- package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
- package/dist/web/standalone/.next/server/next-font-manifest.json +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/web/standalone/.next/static/chunks/2973.33f26573894b6153.js +2 -0
- package/dist/web/standalone/.next/static/chunks/{8359.e059d86b255fce1c.js → 8359.7eb3bb8f8ecf4c01.js} +2 -2
- package/dist/web/standalone/.next/static/chunks/app/layout-8c10ec293ae0f1d5.js +1 -0
- package/dist/web/standalone/.next/static/chunks/{webpack-de742b64187e13fe.js → webpack-9a4db269f9ed63ad.js} +1 -1
- package/dist/web/standalone/.next/static/css/746ee28c929d1880.css +1 -0
- package/package.json +4 -4
- package/packages/contracts/dist/rpc.test.js +7 -0
- package/packages/contracts/dist/rpc.test.js.map +1 -1
- package/packages/contracts/dist/workflow.d.ts +21 -0
- package/packages/contracts/dist/workflow.d.ts.map +1 -1
- package/packages/contracts/dist/workflow.js +24 -0
- package/packages/contracts/dist/workflow.js.map +1 -1
- package/packages/contracts/src/rpc.test.ts +8 -0
- package/packages/contracts/src/workflow.ts +24 -0
- package/packages/mcp-server/README.md +13 -4
- package/packages/mcp-server/dist/workflow-tools.d.ts +0 -3
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +80 -0
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/src/workflow-tools.test.ts +23 -1
- package/packages/mcp-server/src/workflow-tools.ts +168 -0
- package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
- package/packages/native/tsconfig.json +2 -1
- package/packages/native/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-ai/dist/providers/openai-codex-responses.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/openai-codex-responses.js +82 -1
- package/packages/pi-ai/dist/providers/openai-codex-responses.js.map +1 -1
- package/packages/pi-ai/dist/providers/openai-codex-responses.test.d.ts +2 -0
- package/packages/pi-ai/dist/providers/openai-codex-responses.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/openai-codex-responses.test.js +52 -0
- package/packages/pi-ai/dist/providers/openai-codex-responses.test.js.map +1 -0
- package/packages/pi-ai/dist/providers/simple-options.d.ts +2 -4
- package/packages/pi-ai/dist/providers/simple-options.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/simple-options.js +5 -6
- package/packages/pi-ai/dist/providers/simple-options.js.map +1 -1
- package/packages/pi-ai/dist/providers/simple-options.test.d.ts +2 -0
- package/packages/pi-ai/dist/providers/simple-options.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/simple-options.test.js +50 -0
- package/packages/pi-ai/dist/providers/simple-options.test.js.map +1 -0
- package/packages/pi-ai/src/providers/openai-codex-responses.test.ts +63 -0
- package/packages/pi-ai/src/providers/openai-codex-responses.ts +91 -1
- package/packages/pi-ai/src/providers/simple-options.test.ts +60 -0
- package/packages/pi-ai/src/providers/simple-options.ts +5 -6
- package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.js +66 -0
- package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/agent-session.js +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
- package/packages/pi-coding-agent/src/core/agent-session-thinking-level.test.ts +79 -0
- package/packages/pi-coding-agent/src/core/agent-session.ts +1 -1
- package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-tui/dist/tui.d.ts.map +1 -1
- package/packages/pi-tui/dist/tui.js +5 -0
- package/packages/pi-tui/dist/tui.js.map +1 -1
- package/packages/pi-tui/src/tui.ts +6 -0
- package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
- package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
- package/src/resources/GSD-WORKFLOW.md +10 -1
- package/src/resources/extensions/claude-code-cli/partial-builder.ts +2 -1
- package/src/resources/extensions/claude-code-cli/tests/partial-builder.test.ts +19 -2
- package/src/resources/extensions/cmux/index.ts +6 -0
- package/src/resources/extensions/gsd/auto/contracts.ts +59 -16
- package/src/resources/extensions/gsd/auto/infra-errors.ts +9 -3
- package/src/resources/extensions/gsd/auto/loop.ts +8 -5
- package/src/resources/extensions/gsd/auto/orchestrator.ts +129 -6
- package/src/resources/extensions/gsd/auto/phases.ts +7 -1
- package/src/resources/extensions/gsd/auto/workflow-memory-pressure.ts +13 -0
- package/src/resources/extensions/gsd/auto-dispatch.ts +14 -6
- package/src/resources/extensions/gsd/auto-model-selection.ts +2 -1
- package/src/resources/extensions/gsd/auto-post-unit.ts +266 -139
- package/src/resources/extensions/gsd/auto-prompts.ts +2 -2
- package/src/resources/extensions/gsd/auto-recovery.ts +29 -0
- package/src/resources/extensions/gsd/auto-start.ts +92 -9
- package/src/resources/extensions/gsd/auto-verification.ts +36 -34
- package/src/resources/extensions/gsd/auto-worktree.ts +119 -1
- package/src/resources/extensions/gsd/auto.ts +167 -53
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +6 -1
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +9 -8
- package/src/resources/extensions/gsd/bootstrap/subagent-input.ts +19 -7
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +19 -3
- package/src/resources/extensions/gsd/clean-root-preflight.ts +174 -8
- package/src/resources/extensions/gsd/commands/catalog.ts +4 -1
- package/src/resources/extensions/gsd/commands/handlers/core.ts +40 -0
- package/src/resources/extensions/gsd/commands-bootstrap.ts +10 -0
- package/src/resources/extensions/gsd/crash-recovery.ts +30 -4
- package/src/resources/extensions/gsd/db/unit-dispatches.ts +4 -3
- package/src/resources/extensions/gsd/dispatch-guard.ts +2 -2
- package/src/resources/extensions/gsd/doctor-runtime-checks.ts +25 -13
- package/src/resources/extensions/gsd/doctor.ts +2 -27
- package/src/resources/extensions/gsd/export-html.ts +27 -427
- package/src/resources/extensions/gsd/git-service.ts +45 -1
- package/src/resources/extensions/gsd/gsd-db.ts +3 -0
- package/src/resources/extensions/gsd/guided-flow.ts +14 -7
- package/src/resources/extensions/gsd/md-importer.ts +1 -1
- package/src/resources/extensions/gsd/migrate/command.ts +5 -0
- package/src/resources/extensions/gsd/migrate/parsers.ts +11 -0
- package/src/resources/extensions/gsd/migrate/preview.ts +10 -0
- package/src/resources/extensions/gsd/migrate/transformer.ts +58 -4
- package/src/resources/extensions/gsd/migrate/writer.ts +14 -1
- package/src/resources/extensions/gsd/migration-auto-check.ts +15 -23
- package/src/resources/extensions/gsd/milestone-actions.ts +10 -4
- package/src/resources/extensions/gsd/native-git-bridge.ts +54 -12
- package/src/resources/extensions/gsd/post-execution-checks.ts +87 -2
- package/src/resources/extensions/gsd/pre-execution-checks.ts +32 -1
- package/src/resources/extensions/gsd/prompt-loader.ts +1 -1
- package/src/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
- package/src/resources/extensions/gsd/prompts/complete-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/discuss-headless.md +8 -8
- package/src/resources/extensions/gsd/prompts/discuss.md +9 -9
- package/src/resources/extensions/gsd/prompts/guided-discuss-project.md +4 -4
- package/src/resources/extensions/gsd/prompts/guided-discuss-requirements.md +3 -3
- package/src/resources/extensions/gsd/prompts/plan-slice.md +4 -4
- package/src/resources/extensions/gsd/prompts/queue.md +4 -4
- package/src/resources/extensions/gsd/prompts/refine-slice.md +2 -2
- package/src/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
- package/src/resources/extensions/gsd/state-reconciliation/drift/merge-state.ts +8 -1
- package/src/resources/extensions/gsd/state-reconciliation/drift/project-md.ts +12 -15
- package/src/resources/extensions/gsd/state-reconciliation/drift/roadmap.ts +17 -25
- package/src/resources/extensions/gsd/status-guards.ts +5 -0
- package/src/resources/extensions/gsd/templates/plan.md +8 -5
- package/src/resources/extensions/gsd/templates/task-plan.md +4 -2
- package/src/resources/extensions/gsd/tests/auto-deterministic-error-classification-4973.test.ts +116 -0
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +54 -0
- package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +487 -4
- package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +12 -11
- package/src/resources/extensions/gsd/tests/auto-post-unit-step-message.test.ts +12 -1
- package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +15 -1
- package/src/resources/extensions/gsd/tests/auto-runtime-state.test.ts +4 -4
- package/src/resources/extensions/gsd/tests/auto-start-orphan-bootstrap.test.ts +1 -0
- package/src/resources/extensions/gsd/tests/auto-worktree-registry.test.ts +69 -1
- package/src/resources/extensions/gsd/tests/brief-command.test.ts +89 -0
- package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +107 -2
- package/src/resources/extensions/gsd/tests/closeout-git-deferral.test.ts +16 -0
- package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +4 -1
- package/src/resources/extensions/gsd/tests/complete-slice.test.ts +5 -9
- package/src/resources/extensions/gsd/tests/complete-task.test.ts +3 -1
- package/src/resources/extensions/gsd/tests/crash-recovery-via-db.test.ts +43 -2
- package/src/resources/extensions/gsd/tests/db-authority-regression.test.ts +208 -0
- package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +59 -2
- package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +39 -0
- package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +27 -0
- package/src/resources/extensions/gsd/tests/evidence-cross-ref.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/export-html-enhancements.test.ts +8 -0
- package/src/resources/extensions/gsd/tests/guided-discuss-project-prompt-rendering.test.ts +2 -0
- package/src/resources/extensions/gsd/tests/guided-flow.test.ts +21 -0
- package/src/resources/extensions/gsd/tests/headless-milestone-parity.test.ts +6 -6
- package/src/resources/extensions/gsd/tests/hook-model-resolution.test.ts +5 -0
- package/src/resources/extensions/gsd/tests/infra-error.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/infra-errors-cooldown.test.ts +9 -0
- package/src/resources/extensions/gsd/tests/integration/doctor-runtime.test.ts +20 -0
- package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +103 -1
- package/src/resources/extensions/gsd/tests/integration/migrate-command.test.ts +48 -3
- package/src/resources/extensions/gsd/tests/integration/state-machine-runtime-failures.test.ts +6 -1
- package/src/resources/extensions/gsd/tests/migrate-transformer.test.ts +5 -1
- package/src/resources/extensions/gsd/tests/migrate-validator-parsers.test.ts +24 -1
- package/src/resources/extensions/gsd/tests/migrate-writer-integration.test.ts +6 -1
- package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +26 -18
- package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +63 -2
- package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +121 -1
- package/src/resources/extensions/gsd/tests/park-db-sync.test.ts +55 -1
- package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +26 -0
- package/src/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +2 -0
- package/src/resources/extensions/gsd/tests/plan-slice.test.ts +225 -1
- package/src/resources/extensions/gsd/tests/plan-task.test.ts +17 -0
- package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
- package/src/resources/extensions/gsd/tests/post-execution-checks.test.ts +86 -0
- package/src/resources/extensions/gsd/tests/post-unit-git-failure.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +53 -0
- package/src/resources/extensions/gsd/tests/prompt-loader.test.ts +23 -0
- package/src/resources/extensions/gsd/tests/remediation-completion-guard.test.ts +46 -2
- package/src/resources/extensions/gsd/tests/session-switch-abort-misclassification.test.ts +10 -0
- package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +31 -1
- package/src/resources/extensions/gsd/tests/state-corruption-2945.test.ts +6 -0
- package/src/resources/extensions/gsd/tests/state-reconciliation-drift.test.ts +119 -23
- package/src/resources/extensions/gsd/tests/stuck-state-via-db.test.ts +64 -1
- package/src/resources/extensions/gsd/tests/summary-render-parity.test.ts +7 -3
- package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +86 -7
- package/src/resources/extensions/gsd/tests/verification-gate.test.ts +110 -1
- package/src/resources/extensions/gsd/tests/verification-verdict.test.ts +78 -0
- package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/workflow-memory-pressure.test.ts +21 -1
- package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/worktree-git-pathspec.test.ts +39 -0
- package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +64 -12
- package/src/resources/extensions/gsd/tests/worktree-lifecycle.test.ts +25 -0
- package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +54 -0
- package/src/resources/extensions/gsd/tools/complete-milestone.ts +8 -10
- package/src/resources/extensions/gsd/tools/complete-slice.ts +6 -8
- package/src/resources/extensions/gsd/tools/plan-milestone.ts +5 -1
- package/src/resources/extensions/gsd/tools/plan-slice.ts +98 -12
- package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +135 -0
- package/src/resources/extensions/gsd/types.ts +1 -1
- package/src/resources/extensions/gsd/unit-context-manifest.ts +47 -11
- package/src/resources/extensions/gsd/validation.ts +23 -1
- package/src/resources/extensions/gsd/verification-gate.ts +78 -6
- package/src/resources/extensions/gsd/verification-verdict.ts +47 -0
- package/src/resources/extensions/gsd/workflow-projections.ts +6 -8
- package/src/resources/extensions/gsd/worktree-lifecycle.ts +61 -10
- package/src/resources/extensions/shared/html-shell.ts +412 -0
- package/src/resources/extensions/subagent/index.ts +567 -103
- package/src/resources/extensions/subagent/launch.ts +131 -0
- package/src/resources/extensions/subagent/run-store.ts +218 -0
- package/src/resources/extensions/subagent/tests/launch.test.ts +115 -0
- package/src/resources/extensions/subagent/tests/run-store.test.ts +111 -0
- package/src/resources/extensions/visual-brief/artifact-policy.ts +41 -0
- package/src/resources/extensions/visual-brief/extension-manifest.json +8 -0
- package/src/resources/extensions/visual-brief/index.ts +8 -0
- package/src/resources/extensions/visual-brief/page-contract.ts +136 -0
- package/src/resources/extensions/visual-brief/prompts.ts +183 -0
- package/src/resources/extensions/visual-brief/tests/visual-brief.test.ts +212 -0
- package/dist/web/standalone/.next/server/chunks/5822.js +0 -2
- package/dist/web/standalone/.next/static/chunks/2556.0527fea66e123b7f.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/layout-a16c7a7ecdf0c2cf.js +0 -1
- package/dist/web/standalone/.next/static/css/54ec2745c1da488b.css +0 -1
- package/dist/web/standalone/.next/static/css/de70bee13400563f.css +0 -1
- package/dist/web/standalone/.next/static/media/4cf2300e9c8272f7-s.p.woff2 +0 -0
- package/dist/web/standalone/.next/static/media/747892c23ea88013-s.woff2 +0 -0
- package/dist/web/standalone/.next/static/media/8d697b304b401681-s.woff2 +0 -0
- package/dist/web/standalone/.next/static/media/93f479601ee12b01-s.p.woff2 +0 -0
- package/dist/web/standalone/.next/static/media/9610d9e46709d722-s.woff2 +0 -0
- package/dist/web/standalone/.next/static/media/ba015fad6dcf6784-s.woff2 +0 -0
- /package/dist/web/standalone/.next/static/{KDRTXR-22LPCsa80X9dey → euQ0CLP_v8V4e76Tu3odJ}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{KDRTXR-22LPCsa80X9dey → euQ0CLP_v8V4e76Tu3odJ}/_ssgManifest.js +0 -0
|
@@ -5,7 +5,7 @@ import assert from 'node:assert/strict';
|
|
|
5
5
|
import { mkdtempSync, mkdirSync, writeFileSync, rmSync, existsSync, symlinkSync, readFileSync } from "node:fs";
|
|
6
6
|
import { join, dirname } from "node:path";
|
|
7
7
|
import { tmpdir } from "node:os";
|
|
8
|
-
import { execSync } from "node:child_process";
|
|
8
|
+
import { execFileSync, execSync } from "node:child_process";
|
|
9
9
|
|
|
10
10
|
import {
|
|
11
11
|
inferCommitType,
|
|
@@ -371,6 +371,18 @@ describe('git-service', async () => {
|
|
|
371
371
|
return dir;
|
|
372
372
|
}
|
|
373
373
|
|
|
374
|
+
function gitRun(args: string[], cwd: string): string {
|
|
375
|
+
return execFileSync("git", args, {
|
|
376
|
+
cwd,
|
|
377
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
378
|
+
encoding: "utf-8",
|
|
379
|
+
env: {
|
|
380
|
+
...process.env,
|
|
381
|
+
GIT_ALLOW_PROTOCOL: "file",
|
|
382
|
+
},
|
|
383
|
+
}).trim();
|
|
384
|
+
}
|
|
385
|
+
|
|
374
386
|
// ─── GitServiceImpl: smart staging ─────────────────────────────────────
|
|
375
387
|
|
|
376
388
|
test('GitServiceImpl: smart staging', () => {
|
|
@@ -413,6 +425,96 @@ describe('git-service', async () => {
|
|
|
413
425
|
rmSync(repo, { recursive: true, force: true });
|
|
414
426
|
});
|
|
415
427
|
|
|
428
|
+
test('GitServiceImpl: task autoCommit skips keyFiles inside submodules', () => {
|
|
429
|
+
const repo = initTempRepo();
|
|
430
|
+
const subSrc = mkdtempSync(join(tmpdir(), "gsd-git-submodule-src-"));
|
|
431
|
+
|
|
432
|
+
try {
|
|
433
|
+
gitRun(["init", "-b", "main"], subSrc);
|
|
434
|
+
gitRun(["config", "user.name", "Pi Test"], subSrc);
|
|
435
|
+
gitRun(["config", "user.email", "pi@example.com"], subSrc);
|
|
436
|
+
createFile(subSrc, "tracked.txt", "initial\n");
|
|
437
|
+
gitRun(["add", "-A"], subSrc);
|
|
438
|
+
gitRun(["commit", "-m", "init submodule"], subSrc);
|
|
439
|
+
|
|
440
|
+
gitRun(["-c", "protocol.file.allow=always", "submodule", "add", `file://${subSrc}`, "sub"], repo);
|
|
441
|
+
gitRun(["commit", "-m", "add submodule"], repo);
|
|
442
|
+
|
|
443
|
+
createFile(repo, "sub/copied.txt", "copied from source\n");
|
|
444
|
+
createFile(repo, "src/feature.ts", "export const feature = true;\n");
|
|
445
|
+
createFile(repo, "src/unrelated.ts", "export const unrelated = true;\n");
|
|
446
|
+
|
|
447
|
+
const svc = new GitServiceImpl(repo);
|
|
448
|
+
const taskContext: TaskCommitContext = {
|
|
449
|
+
taskId: "S01/T01",
|
|
450
|
+
taskDisplayId: "T01",
|
|
451
|
+
taskTitle: "fix submodule staging",
|
|
452
|
+
milestoneId: "M001",
|
|
453
|
+
milestoneTitle: "Submodule auto commit",
|
|
454
|
+
sliceId: "S01",
|
|
455
|
+
sliceTitle: "Commit scoped files",
|
|
456
|
+
oneLiner: "Fixed auto commit when key files include submodule paths",
|
|
457
|
+
keyFiles: ["sub/copied.txt", "src/feature.ts"],
|
|
458
|
+
};
|
|
459
|
+
|
|
460
|
+
const result = svc.autoCommit("execute-task", "M001/S01/T01", [], taskContext);
|
|
461
|
+
|
|
462
|
+
assert.ok(result !== null, "autoCommit should commit non-submodule changes");
|
|
463
|
+
const committed = gitRun(["show", "--name-only", "--format=", "HEAD"], repo);
|
|
464
|
+
assert.ok(committed.includes("src/feature.ts"), "non-submodule keyFile is committed");
|
|
465
|
+
assert.ok(!committed.includes("sub/copied.txt"), "submodule inner keyFile is not pathspec-staged");
|
|
466
|
+
assert.ok(!committed.includes("src/unrelated.ts"), "scoped staging does not fall back to smartStage");
|
|
467
|
+
} finally {
|
|
468
|
+
rmSync(repo, { recursive: true, force: true });
|
|
469
|
+
rmSync(subSrc, { recursive: true, force: true });
|
|
470
|
+
}
|
|
471
|
+
});
|
|
472
|
+
|
|
473
|
+
test('GitServiceImpl: all keyFiles inside submodules falls back to smartStage', () => {
|
|
474
|
+
const repo = initTempRepo();
|
|
475
|
+
const subSrc = mkdtempSync(join(tmpdir(), "gsd-git-all-submodule-src-"));
|
|
476
|
+
|
|
477
|
+
try {
|
|
478
|
+
gitRun(["init", "-b", "main"], subSrc);
|
|
479
|
+
gitRun(["config", "user.name", "Pi Test"], subSrc);
|
|
480
|
+
gitRun(["config", "user.email", "pi@example.com"], subSrc);
|
|
481
|
+
createFile(subSrc, "tracked.txt", "initial\n");
|
|
482
|
+
gitRun(["add", "-A"], subSrc);
|
|
483
|
+
gitRun(["commit", "-m", "init submodule"], subSrc);
|
|
484
|
+
|
|
485
|
+
gitRun(["-c", "protocol.file.allow=always", "submodule", "add", `file://${subSrc}`, "sub"], repo);
|
|
486
|
+
gitRun(["commit", "-m", "add submodule"], repo);
|
|
487
|
+
|
|
488
|
+
createFile(repo, "sub/file1.txt", "inside submodule\n");
|
|
489
|
+
createFile(repo, "sub/file2.txt", "also inside\n");
|
|
490
|
+
createFile(repo, "src/real.ts", "export const real = true;\n");
|
|
491
|
+
|
|
492
|
+
const svc = new GitServiceImpl(repo);
|
|
493
|
+
const taskContext: TaskCommitContext = {
|
|
494
|
+
taskId: "S01/T02",
|
|
495
|
+
taskDisplayId: "T02",
|
|
496
|
+
taskTitle: "all keyFiles inside submodule",
|
|
497
|
+
milestoneId: "M001",
|
|
498
|
+
milestoneTitle: "Submodule auto commit",
|
|
499
|
+
sliceId: "S01",
|
|
500
|
+
sliceTitle: "Commit scoped files",
|
|
501
|
+
oneLiner: "Fell back when all key files are inside submodules",
|
|
502
|
+
keyFiles: ["sub", "sub/file1.txt", "sub/file2.txt"],
|
|
503
|
+
};
|
|
504
|
+
|
|
505
|
+
const result = svc.autoCommit("execute-task", "M001/S01/T02", [], taskContext);
|
|
506
|
+
|
|
507
|
+
assert.ok(result !== null, "autoCommit falls back to smartStage when all keyFiles are filtered");
|
|
508
|
+
const committed = gitRun(["show", "--name-only", "--format=", "HEAD"], repo);
|
|
509
|
+
assert.ok(!committed.includes("sub/file1.txt"), "submodule keyFile is not committed");
|
|
510
|
+
assert.ok(!committed.includes("sub/file2.txt"), "submodule keyFile is not committed");
|
|
511
|
+
assert.ok(committed.includes("src/real.ts"), "smartStage fallback commits other dirty files");
|
|
512
|
+
} finally {
|
|
513
|
+
rmSync(repo, { recursive: true, force: true });
|
|
514
|
+
rmSync(subSrc, { recursive: true, force: true });
|
|
515
|
+
}
|
|
516
|
+
});
|
|
517
|
+
|
|
416
518
|
// ─── GitServiceImpl: smart staging excludes tracked runtime files ──────
|
|
417
519
|
|
|
418
520
|
test('GitServiceImpl: smart staging excludes tracked runtime files', () => {
|
|
@@ -14,7 +14,9 @@ import {
|
|
|
14
14
|
generatePreview,
|
|
15
15
|
writeGSDDirectory,
|
|
16
16
|
} from '../../migrate/index.ts';
|
|
17
|
+
import { importWrittenMigrationToDb } from '../../migrate/command.ts';
|
|
17
18
|
import { deriveState } from '../../state.ts';
|
|
19
|
+
import { closeDatabase, getDecisionById, getRequirementCounts } from '../../gsd-db.ts';
|
|
18
20
|
import { describe, test, beforeEach, afterEach } from 'node:test';
|
|
19
21
|
import assert from 'node:assert/strict';
|
|
20
22
|
|
|
@@ -52,6 +54,16 @@ const SAMPLE_REQUIREMENTS = `# Requirements
|
|
|
52
54
|
- Description: Output matches GSD format.
|
|
53
55
|
`;
|
|
54
56
|
|
|
57
|
+
const SAMPLE_REQUIREMENTS_LEGACY_IDS = `# Requirements
|
|
58
|
+
|
|
59
|
+
## Active
|
|
60
|
+
|
|
61
|
+
- [ ] **CORE-PIPELINE**: Pipeline must work end-to-end.
|
|
62
|
+
- [ ] **OUTPUT-FORMAT**: Output matches GSD format.
|
|
63
|
+
- [ ] **IMPORT-DB**: Migration imports requirements into the DB.
|
|
64
|
+
- [ ] **STATUS-WIDGET**: Status can query migrated requirements.
|
|
65
|
+
`;
|
|
66
|
+
|
|
55
67
|
const SAMPLE_STATE = `# State
|
|
56
68
|
|
|
57
69
|
**Current Phase:** 20-features
|
|
@@ -166,14 +178,14 @@ Depends on foundation work.
|
|
|
166
178
|
</context>
|
|
167
179
|
`;
|
|
168
180
|
|
|
169
|
-
function createCompleteFixture(): string {
|
|
181
|
+
function createCompleteFixture(requirementsContent: string = SAMPLE_REQUIREMENTS): string {
|
|
170
182
|
const base = mkdtempSync(join(tmpdir(), 'gsd-cmd-test-'));
|
|
171
183
|
const planning = join(base, '.planning');
|
|
172
184
|
mkdirSync(planning, { recursive: true });
|
|
173
185
|
|
|
174
186
|
writeFileSync(join(planning, 'PROJECT.md'), SAMPLE_PROJECT);
|
|
175
187
|
writeFileSync(join(planning, 'ROADMAP.md'), SAMPLE_ROADMAP);
|
|
176
|
-
writeFileSync(join(planning, 'REQUIREMENTS.md'),
|
|
188
|
+
writeFileSync(join(planning, 'REQUIREMENTS.md'), requirementsContent);
|
|
177
189
|
writeFileSync(join(planning, 'STATE.md'), SAMPLE_STATE);
|
|
178
190
|
writeFileSync(join(planning, 'config.json'), SAMPLE_CONFIG);
|
|
179
191
|
|
|
@@ -303,6 +315,7 @@ test('Full pipeline: parse → transform → preview → write → deriveState',
|
|
|
303
315
|
const expectedTaskPct = totalTasks > 0 ? Math.round((doneTasks / totalTasks) * 100) : 0;
|
|
304
316
|
assert.deepStrictEqual(preview.sliceCompletionPct, expectedSlicePct, 'pipeline: preview sliceCompletionPct');
|
|
305
317
|
assert.deepStrictEqual(preview.taskCompletionPct, expectedTaskPct, 'pipeline: preview taskCompletionPct');
|
|
318
|
+
assert.deepStrictEqual(preview.decisions.total, 1, 'pipeline: preview decisions total');
|
|
306
319
|
|
|
307
320
|
// Requirements in preview
|
|
308
321
|
assert.deepStrictEqual(preview.requirements.active, 1, 'pipeline: preview requirements active');
|
|
@@ -342,6 +355,39 @@ test('Full pipeline: parse → transform → preview → write → deriveState',
|
|
|
342
355
|
}
|
|
343
356
|
});
|
|
344
357
|
|
|
358
|
+
test('Full pipeline: legacy requirement IDs import into DB with canonical IDs', async () => {
|
|
359
|
+
const base = createCompleteFixture(SAMPLE_REQUIREMENTS_LEGACY_IDS);
|
|
360
|
+
const writeTarget = mkdtempSync(join(tmpdir(), 'gsd-cmd-legacy-reqs-'));
|
|
361
|
+
try {
|
|
362
|
+
const parsed = await parsePlanningDirectory(join(base, '.planning'));
|
|
363
|
+
const project = transformToGSD(parsed);
|
|
364
|
+
const preview = generatePreview(project);
|
|
365
|
+
|
|
366
|
+
assert.deepStrictEqual(
|
|
367
|
+
project.requirements.map((req) => req.id),
|
|
368
|
+
['R001', 'R002', 'R003', 'R004'],
|
|
369
|
+
'legacy-reqs: transform assigns canonical R IDs',
|
|
370
|
+
);
|
|
371
|
+
assert.ok(
|
|
372
|
+
project.requirements[0]?.description.includes('Legacy ID: CORE-PIPELINE'),
|
|
373
|
+
'legacy-reqs: original ID survives in migrated requirement content',
|
|
374
|
+
);
|
|
375
|
+
|
|
376
|
+
await writeGSDDirectory(project, writeTarget);
|
|
377
|
+
const imported = await importWrittenMigrationToDb(writeTarget, preview);
|
|
378
|
+
const counts = getRequirementCounts();
|
|
379
|
+
|
|
380
|
+
assert.deepStrictEqual(imported.decisions, 1, 'legacy-reqs: DB import includes migrated decisions');
|
|
381
|
+
assert.deepStrictEqual(imported.requirements, 4, 'legacy-reqs: DB import count matches preview');
|
|
382
|
+
assert.ok(getDecisionById('D001') !== null, 'legacy-reqs: migrated decision is queryable');
|
|
383
|
+
assert.deepStrictEqual(counts.total, 4, 'legacy-reqs: DB stores all migrated requirements');
|
|
384
|
+
} finally {
|
|
385
|
+
closeDatabase();
|
|
386
|
+
rmSync(base, { recursive: true, force: true });
|
|
387
|
+
rmSync(writeTarget, { recursive: true, force: true });
|
|
388
|
+
}
|
|
389
|
+
});
|
|
390
|
+
|
|
345
391
|
// ─── Test 6: .gsd/ exists detection ────────────────────────────────────
|
|
346
392
|
|
|
347
393
|
test('.gsd/ exists detection', () => {
|
|
@@ -357,4 +403,3 @@ test('.gsd/ exists detection', () => {
|
|
|
357
403
|
rmSync(base, { recursive: true, force: true });
|
|
358
404
|
}
|
|
359
405
|
});
|
|
360
|
-
|
package/src/resources/extensions/gsd/tests/integration/state-machine-runtime-failures.test.ts
CHANGED
|
@@ -195,6 +195,11 @@ describe("infrastructure error detection", () => {
|
|
|
195
195
|
assert.equal(isInfrastructureError(err), "EAGAIN");
|
|
196
196
|
});
|
|
197
197
|
|
|
198
|
+
test("ENOBUFS (no buffer space available) is detected", () => {
|
|
199
|
+
const err = Object.assign(new Error("spawnSync git ENOBUFS"), { code: "ENOBUFS" });
|
|
200
|
+
assert.equal(isInfrastructureError(err), "ENOBUFS");
|
|
201
|
+
});
|
|
202
|
+
|
|
198
203
|
test("SQLite WAL corruption is detected via message scan", () => {
|
|
199
204
|
const err = new Error("database disk image is malformed");
|
|
200
205
|
assert.equal(isInfrastructureError(err), "SQLITE_CORRUPT");
|
|
@@ -223,7 +228,7 @@ describe("infrastructure error detection", () => {
|
|
|
223
228
|
test("all INFRA_ERROR_CODES are covered", () => {
|
|
224
229
|
const expectedCodes = [
|
|
225
230
|
"ENOSPC", "ENOMEM", "EROFS", "EDQUOT", "EMFILE",
|
|
226
|
-
"ENFILE", "EAGAIN", "ECONNREFUSED", "ENOTFOUND", "ENETUNREACH",
|
|
231
|
+
"ENFILE", "EAGAIN", "ENOBUFS", "ECONNREFUSED", "ENOTFOUND", "ENETUNREACH",
|
|
227
232
|
];
|
|
228
233
|
for (const code of expectedCodes) {
|
|
229
234
|
assert.ok(INFRA_ERROR_CODES.has(code), `${code} should be in INFRA_ERROR_CODES`);
|
|
@@ -497,6 +497,7 @@ test('Scenario 11: Requirements edge cases', () => {
|
|
|
497
497
|
makeRequirement('', 'Another No ID', 'validated'),
|
|
498
498
|
makeRequirement('R005', 'Has ID', 'something-weird'),
|
|
499
499
|
makeRequirement('R006', 'Deferred One', 'DEFERRED'),
|
|
500
|
+
makeRequirement('AUTH-7', 'Legacy ID', 'active'),
|
|
500
501
|
],
|
|
501
502
|
phases: {
|
|
502
503
|
'1-req-edge': makePhase('1-req-edge', 1, 'req-edge'),
|
|
@@ -510,6 +511,8 @@ test('Scenario 11: Requirements edge cases', () => {
|
|
|
510
511
|
assert.deepStrictEqual(result.requirements[2]?.id, 'R005', 'req-edge: existing id preserved');
|
|
511
512
|
assert.deepStrictEqual(result.requirements[2]?.status, 'active', 'req-edge: unknown status normalized to active');
|
|
512
513
|
assert.deepStrictEqual(result.requirements[3]?.status, 'deferred', 'req-edge: uppercase DEFERRED normalized');
|
|
514
|
+
assert.deepStrictEqual(result.requirements[4]?.id, 'R003', 'req-edge: non-R legacy id gets next canonical id');
|
|
515
|
+
assert.ok(result.requirements[4]?.description.includes('Legacy ID: AUTH-7'), 'req-edge: original legacy id is preserved in description');
|
|
513
516
|
});
|
|
514
517
|
|
|
515
518
|
// ─── Scenario 12: Vision derivation ────────────────────────────────────────
|
|
@@ -553,6 +556,8 @@ test('Scenario 13: Decisions content', () => {
|
|
|
553
556
|
const result = transformToGSD(project);
|
|
554
557
|
|
|
555
558
|
assert.ok(result.decisionsContent.includes('decision-01'), 'decisions: extracts key-decisions from summaries');
|
|
559
|
+
assert.ok(result.decisionsContent.includes('| D001 |'), 'decisions: writes DB-importable decision ID');
|
|
560
|
+
assert.ok(result.decisionsContent.includes('| # | When | Scope | Decision | Choice | Rationale | Revisable? | Made By |'), 'decisions: writes canonical table header');
|
|
556
561
|
});
|
|
557
562
|
|
|
558
563
|
// ─── Scenario 14: No undefined values in output ───────────────────────────
|
|
@@ -616,4 +621,3 @@ test('Scenario 15: Empty research', () => {
|
|
|
616
621
|
});
|
|
617
622
|
|
|
618
623
|
// ─── Results ───────────────────────────────────────────────────────────────
|
|
619
|
-
|
|
@@ -41,6 +41,15 @@ const SAMPLE_ROADMAP = `# Project Roadmap
|
|
|
41
41
|
- [ ] 31 — Notifications
|
|
42
42
|
`;
|
|
43
43
|
|
|
44
|
+
const SAMPLE_VERSION_PREFIX_ROADMAP = `# Project Roadmap
|
|
45
|
+
|
|
46
|
+
## Phases
|
|
47
|
+
|
|
48
|
+
- ✅ **v1.0 MVP** — Phases 1-6 (shipped 2026-02-24)
|
|
49
|
+
- ✅ **v1.1 Onboarding** — Phases 7-9 (shipped 2026-03-01)
|
|
50
|
+
- 🚧 **v1.8 Production** — Phases 44-53
|
|
51
|
+
`;
|
|
52
|
+
|
|
44
53
|
const SAMPLE_PROJECT = `# My Project
|
|
45
54
|
|
|
46
55
|
A sample project for testing the migration parser.
|
|
@@ -246,6 +255,21 @@ test('parseOldRoadmap: flat format', () => {
|
|
|
246
255
|
assert.deepStrictEqual(roadmap.phases[1].done, false, 'flat roadmap: second phase not done');
|
|
247
256
|
});
|
|
248
257
|
|
|
258
|
+
test('parseOldRoadmap: emoji version-prefix phase ranges', () => {
|
|
259
|
+
const roadmap = parseOldRoadmap(SAMPLE_VERSION_PREFIX_ROADMAP);
|
|
260
|
+
assert.deepStrictEqual(roadmap.milestones.length, 0, 'version roadmap: no milestone sections');
|
|
261
|
+
assert.deepStrictEqual(roadmap.phases.length, 3, 'version roadmap: 3 phase ranges');
|
|
262
|
+
assert.deepStrictEqual(roadmap.phases[0].number, 1, 'version roadmap: first range starts at phase 1');
|
|
263
|
+
assert.deepStrictEqual(roadmap.phases[0].title, 'MVP', 'version roadmap: first title');
|
|
264
|
+
assert.deepStrictEqual(roadmap.phases[0].done, true, 'version roadmap: first range done');
|
|
265
|
+
assert.deepStrictEqual(roadmap.phases[1].number, 7, 'version roadmap: second range starts at phase 7');
|
|
266
|
+
assert.deepStrictEqual(roadmap.phases[1].title, 'Onboarding', 'version roadmap: second title');
|
|
267
|
+
assert.deepStrictEqual(roadmap.phases[1].done, true, 'version roadmap: second range done');
|
|
268
|
+
assert.deepStrictEqual(roadmap.phases[2].number, 44, 'version roadmap: third range starts at phase 44');
|
|
269
|
+
assert.deepStrictEqual(roadmap.phases[2].title, 'Production', 'version roadmap: third title');
|
|
270
|
+
assert.deepStrictEqual(roadmap.phases[2].done, false, 'version roadmap: third range in progress');
|
|
271
|
+
});
|
|
272
|
+
|
|
249
273
|
test('parseOldRoadmap: milestone-sectioned with <details>', () => {
|
|
250
274
|
const roadmap = parseOldRoadmap(SAMPLE_MILESTONE_SECTIONED_ROADMAP);
|
|
251
275
|
assert.ok(roadmap.milestones.length >= 2, 'ms roadmap: has milestone sections');
|
|
@@ -387,4 +411,3 @@ test('parseOldProject', () => {
|
|
|
387
411
|
const project = parseOldProject(SAMPLE_PROJECT);
|
|
388
412
|
assert.deepStrictEqual(project, SAMPLE_PROJECT, 'project: returns raw content');
|
|
389
413
|
});
|
|
390
|
-
|
|
@@ -14,7 +14,7 @@ import { parseSummary } from '../files.ts';
|
|
|
14
14
|
import { deriveState } from '../state.ts';
|
|
15
15
|
import { invalidateAllCaches } from '../cache.ts';
|
|
16
16
|
import { ensureDbOpen } from '../bootstrap/dynamic-tools.ts';
|
|
17
|
-
import { closeDatabase, getAllMilestones } from '../gsd-db.ts';
|
|
17
|
+
import { closeDatabase, getAllMilestones, getArtifact } from '../gsd-db.ts';
|
|
18
18
|
import { importWrittenMigrationToDb } from '../migrate/command.ts';
|
|
19
19
|
import type {
|
|
20
20
|
GSDProject,
|
|
@@ -336,7 +336,12 @@ test('Scenario 2: Fully complete project — deriveState phase', async () => {
|
|
|
336
336
|
assert.deepStrictEqual(preview.taskCompletionPct, 100, 'complete: preview taskCompletionPct');
|
|
337
337
|
assert.deepStrictEqual(preview.requirements.total, 0, 'complete: preview requirements total');
|
|
338
338
|
|
|
339
|
+
const imported = await importWrittenMigrationToDb(base, preview);
|
|
340
|
+
assert.ok(imported.artifacts >= 6, 'complete: imports generated milestone artifacts');
|
|
341
|
+
assert.ok(getArtifact('milestones/M001/M001-VALIDATION.md') !== null, 'complete: M001-VALIDATION.md imported as artifact');
|
|
342
|
+
assert.ok(getArtifact('milestones/M001/M001-SUMMARY.md') !== null, 'complete: M001-SUMMARY.md imported as artifact');
|
|
339
343
|
} finally {
|
|
344
|
+
closeDatabase();
|
|
340
345
|
rmSync(base, { recursive: true, force: true });
|
|
341
346
|
}
|
|
342
347
|
});
|
|
@@ -6,13 +6,15 @@ import test from "node:test";
|
|
|
6
6
|
|
|
7
7
|
import { ensureDbOpen } from "../bootstrap/dynamic-tools.ts";
|
|
8
8
|
import {
|
|
9
|
-
_getAdapter,
|
|
10
9
|
closeDatabase,
|
|
11
10
|
getAllMilestones,
|
|
11
|
+
insertMilestone,
|
|
12
|
+
insertSlice,
|
|
13
|
+
insertTask,
|
|
12
14
|
getSliceTasks,
|
|
13
15
|
} from "../gsd-db.ts";
|
|
14
16
|
import {
|
|
15
|
-
|
|
17
|
+
checkMarkdownHierarchyAgainstDb,
|
|
16
18
|
countMarkdownHierarchy,
|
|
17
19
|
} from "../migration-auto-check.ts";
|
|
18
20
|
import { writeGSDDirectory } from "../migrate/writer.ts";
|
|
@@ -70,7 +72,7 @@ function projectFixture(): GSDProject {
|
|
|
70
72
|
};
|
|
71
73
|
}
|
|
72
74
|
|
|
73
|
-
test("migration auto-check
|
|
75
|
+
test("migration auto-check preserves empty DB and reports explicit recovery", async () => {
|
|
74
76
|
const base = makeBase();
|
|
75
77
|
try {
|
|
76
78
|
await writeGSDDirectory(projectFixture(), base);
|
|
@@ -79,32 +81,35 @@ test("migration auto-check imports markdown hierarchy when DB is empty", async (
|
|
|
79
81
|
assert.equal(await ensureDbOpen(base), true);
|
|
80
82
|
assert.equal(getAllMilestones().length, 0, "fresh authoritative DB starts empty");
|
|
81
83
|
|
|
82
|
-
const result = await
|
|
83
|
-
assert.equal(result.action, "
|
|
84
|
+
const result = await checkMarkdownHierarchyAgainstDb(base);
|
|
85
|
+
assert.equal(result.action, "recovery-required");
|
|
84
86
|
assert.equal(result.reason, "db-empty");
|
|
85
|
-
assert.deepEqual(result.afterDb, { milestones:
|
|
86
|
-
assert.equal(
|
|
87
|
-
assert.
|
|
87
|
+
assert.deepEqual(result.afterDb, { milestones: 0, slices: 0, tasks: 0 });
|
|
88
|
+
assert.equal(result.recoveryCommand, "gsd recover");
|
|
89
|
+
assert.match(result.message ?? "", /will not import markdown automatically/);
|
|
90
|
+
assert.equal(getAllMilestones().length, 0);
|
|
91
|
+
assert.equal(getSliceTasks("M001", "S01").length, 0);
|
|
88
92
|
} finally {
|
|
89
93
|
cleanup(base);
|
|
90
94
|
}
|
|
91
95
|
});
|
|
92
96
|
|
|
93
|
-
test("migration auto-check
|
|
97
|
+
test("migration auto-check preserves DB on hierarchy count mismatch", async () => {
|
|
94
98
|
const base = makeBase();
|
|
95
99
|
try {
|
|
96
100
|
await writeGSDDirectory(projectFixture(), base);
|
|
97
|
-
await
|
|
98
|
-
|
|
99
|
-
|
|
101
|
+
assert.equal(await ensureDbOpen(base), true);
|
|
102
|
+
insertMilestone({ id: "M001", title: "Legacy Milestone", status: "active" });
|
|
103
|
+
insertSlice({ id: "S01", milestoneId: "M001", title: "Legacy Slice", status: "pending", risk: "medium", depends: [], demo: "Legacy slice demo", sequence: 1 });
|
|
100
104
|
assert.equal(getSliceTasks("M001", "S01").length, 0, "test fixture simulates stale DB task count");
|
|
101
105
|
|
|
102
|
-
const result = await
|
|
103
|
-
assert.equal(result.action, "
|
|
106
|
+
const result = await checkMarkdownHierarchyAgainstDb(base);
|
|
107
|
+
assert.equal(result.action, "recovery-required");
|
|
104
108
|
assert.equal(result.reason, "count-mismatch");
|
|
105
109
|
assert.deepEqual(result.beforeDb, { milestones: 1, slices: 1, tasks: 0 });
|
|
106
|
-
assert.deepEqual(result.afterDb, { milestones: 1, slices: 1, tasks:
|
|
107
|
-
assert.equal(
|
|
110
|
+
assert.deepEqual(result.afterDb, { milestones: 1, slices: 1, tasks: 0 });
|
|
111
|
+
assert.equal(result.recoveryCommand, "gsd recover");
|
|
112
|
+
assert.equal(getSliceTasks("M001", "S01").length, 0);
|
|
108
113
|
} finally {
|
|
109
114
|
cleanup(base);
|
|
110
115
|
}
|
|
@@ -114,9 +119,12 @@ test("migration auto-check leaves matching DB hierarchy alone", async () => {
|
|
|
114
119
|
const base = makeBase();
|
|
115
120
|
try {
|
|
116
121
|
await writeGSDDirectory(projectFixture(), base);
|
|
117
|
-
await
|
|
122
|
+
assert.equal(await ensureDbOpen(base), true);
|
|
123
|
+
insertMilestone({ id: "M001", title: "Legacy Milestone", status: "active" });
|
|
124
|
+
insertSlice({ id: "S01", milestoneId: "M001", title: "Legacy Slice", status: "pending", risk: "medium", depends: [], demo: "Legacy slice demo", sequence: 1 });
|
|
125
|
+
insertTask({ id: "T01", sliceId: "S01", milestoneId: "M001", title: "Legacy Task", status: "pending" });
|
|
118
126
|
|
|
119
|
-
const result = await
|
|
127
|
+
const result = await checkMarkdownHierarchyAgainstDb(base);
|
|
120
128
|
assert.equal(result.action, "none");
|
|
121
129
|
assert.equal(result.reason, "in-sync");
|
|
122
130
|
assert.deepEqual(result.markdown, { milestones: 1, slices: 1, tasks: 1 });
|
|
@@ -13,17 +13,19 @@
|
|
|
13
13
|
import { describe, test, beforeEach, afterEach } from "node:test";
|
|
14
14
|
import assert from "node:assert/strict";
|
|
15
15
|
import { chmodSync, existsSync, mkdtempSync, writeFileSync, readFileSync, rmSync } from "node:fs";
|
|
16
|
-
import { join } from "node:path";
|
|
16
|
+
import { delimiter, join } from "node:path";
|
|
17
17
|
import { tmpdir } from "node:os";
|
|
18
18
|
import { execFileSync } from "node:child_process";
|
|
19
19
|
import {
|
|
20
20
|
assertWorktreeMaterialized,
|
|
21
21
|
nativeBranchDelete,
|
|
22
22
|
nativeCommit,
|
|
23
|
+
nativeGetCurrentBranch,
|
|
23
24
|
nativeIsRepo,
|
|
24
25
|
nativeResetHard,
|
|
25
26
|
nativeWorktreeAdd,
|
|
26
27
|
} from "../native-git-bridge.js";
|
|
28
|
+
import { GIT_NO_PROMPT_ENV } from "../git-constants.js";
|
|
27
29
|
|
|
28
30
|
// Note: prior static-analysis tests that scanned native-git-bridge.ts for
|
|
29
31
|
// the raw shell-spawn pattern were removed under #4827 — the integration
|
|
@@ -78,6 +80,55 @@ describe("native-git-bridge #4180: fallback runtime behaviour", () => {
|
|
|
78
80
|
assert.equal(subject, "test: regression commit #4180");
|
|
79
81
|
});
|
|
80
82
|
|
|
83
|
+
test("nativeCommit retries once after transient ENOBUFS from git", (t) => {
|
|
84
|
+
const bin = mkdtempSync(join(tmpdir(), "ngb-enobufs-bin-"));
|
|
85
|
+
t.after(() => rmSync(bin, { recursive: true, force: true }));
|
|
86
|
+
|
|
87
|
+
const realGit = execFileSync("git", ["--exec-path"], { encoding: "utf-8" }).trim();
|
|
88
|
+
const attempts = join(bin, "attempts.txt");
|
|
89
|
+
const fakeGit = join(bin, "fake-git.cjs");
|
|
90
|
+
writeFileSync(fakeGit, `
|
|
91
|
+
const { appendFileSync, readFileSync } = require("node:fs");
|
|
92
|
+
const { spawnSync } = require("node:child_process");
|
|
93
|
+
const attempts = ${JSON.stringify(attempts)};
|
|
94
|
+
const realGit = ${JSON.stringify(join(realGit, process.platform === "win32" ? "git.exe" : "git"))};
|
|
95
|
+
appendFileSync(attempts, "1");
|
|
96
|
+
if (process.argv[2] === "commit" && readFileSync(attempts, "utf-8").length === 1) {
|
|
97
|
+
console.error("spawnSync git ENOBUFS");
|
|
98
|
+
process.exit(1);
|
|
99
|
+
}
|
|
100
|
+
const result = spawnSync(realGit, process.argv.slice(2), { stdio: "inherit" });
|
|
101
|
+
process.exit(result.status ?? 1);
|
|
102
|
+
`, "utf-8");
|
|
103
|
+
|
|
104
|
+
if (process.platform === "win32") {
|
|
105
|
+
writeFileSync(join(bin, "git.cmd"), `@echo off\r\nnode "${fakeGit}" %*\r\n`, "utf-8");
|
|
106
|
+
} else {
|
|
107
|
+
const shim = join(bin, "git");
|
|
108
|
+
writeFileSync(shim, `#!/bin/sh\nexec node "${fakeGit}" "$@"\n`, "utf-8");
|
|
109
|
+
chmodSync(shim, 0o755);
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
writeFileSync(join(repo, "file.txt"), "retry commit\n");
|
|
113
|
+
git(["add", "."], repo);
|
|
114
|
+
|
|
115
|
+
const originalPath = process.env.PATH ?? "";
|
|
116
|
+
const gitEnv = GIT_NO_PROMPT_ENV as NodeJS.ProcessEnv;
|
|
117
|
+
const originalGitEnvPath = gitEnv.PATH;
|
|
118
|
+
try {
|
|
119
|
+
process.env.PATH = `${bin}${delimiter}${originalPath}`;
|
|
120
|
+
gitEnv.PATH = process.env.PATH;
|
|
121
|
+
const result = nativeCommit(repo, "test: retry ENOBUFS commit");
|
|
122
|
+
assert.ok(result !== null, "commit should succeed after retry");
|
|
123
|
+
} finally {
|
|
124
|
+
process.env.PATH = originalPath;
|
|
125
|
+
gitEnv.PATH = originalGitEnvPath;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
assert.equal(readFileSync(attempts, "utf-8").length, 2);
|
|
129
|
+
assert.equal(git(["log", "-1", "--format=%s"], repo), "test: retry ENOBUFS commit");
|
|
130
|
+
});
|
|
131
|
+
|
|
81
132
|
test("nativeCommit runs commit hooks", () => {
|
|
82
133
|
const hookPath = join(repo, ".git", "hooks", "commit-msg");
|
|
83
134
|
const marker = join(repo, "hook-ran.txt");
|
|
@@ -119,7 +170,17 @@ describe("native-git-bridge #4180: fallback runtime behaviour", () => {
|
|
|
119
170
|
test("nativeBranchDelete throws when git cannot delete the branch", () => {
|
|
120
171
|
assert.throws(
|
|
121
172
|
() => nativeBranchDelete(repo, "does-not-exist"),
|
|
122
|
-
/
|
|
173
|
+
/git branch -D does-not-exist failed[\s\S]*does-not-exist/,
|
|
174
|
+
);
|
|
175
|
+
});
|
|
176
|
+
|
|
177
|
+
test("nativeGetCurrentBranch preserves git stderr in fallback errors", (t) => {
|
|
178
|
+
const dir = mkdtempSync(join(tmpdir(), "ngb-stderr-notrepo-"));
|
|
179
|
+
t.after(() => rmSync(dir, { recursive: true, force: true }));
|
|
180
|
+
|
|
181
|
+
assert.throws(
|
|
182
|
+
() => nativeGetCurrentBranch(dir),
|
|
183
|
+
/git branch --show-current failed[\s\S]*(not a git repository|not a git repo|fatal:)/,
|
|
123
184
|
);
|
|
124
185
|
});
|
|
125
186
|
|
|
@@ -243,7 +243,7 @@ describe("auditOrphanedMilestoneBranches", () => {
|
|
|
243
243
|
);
|
|
244
244
|
});
|
|
245
245
|
|
|
246
|
-
test("
|
|
246
|
+
test("milestone in DB, no branch, no worktree dir → no-op", () => {
|
|
247
247
|
insertMilestone({ id: "M001", title: "Test", status: "complete" });
|
|
248
248
|
|
|
249
249
|
const result = auditOrphanedMilestoneBranches(dir, "worktree");
|
|
@@ -251,4 +251,124 @@ describe("auditOrphanedMilestoneBranches", () => {
|
|
|
251
251
|
assert.deepStrictEqual(result.recovered, []);
|
|
252
252
|
assert.deepStrictEqual(result.warnings, []);
|
|
253
253
|
});
|
|
254
|
+
|
|
255
|
+
test("#5879 — cleans orphaned worktree dir for complete milestone whose branch was already deleted", () => {
|
|
256
|
+
// Reproduces the postflight-stash-restore-failed scenario:
|
|
257
|
+
// 1. An earlier audit deleted milestone/M001 (merged + complete).
|
|
258
|
+
// 2. The worktree dir cleanup failed silently (logWarning only).
|
|
259
|
+
// 3. On the next startup the branch is gone, so the existing branch-keyed
|
|
260
|
+
// loop is invisible to the orphan dir. Without the second pass, the
|
|
261
|
+
// directory lives forever.
|
|
262
|
+
insertMilestone({ id: "M001", title: "Test", status: "complete" });
|
|
263
|
+
|
|
264
|
+
const wtDir = join(dir, ".gsd", "worktrees", "M001");
|
|
265
|
+
mkdirSync(wtDir, { recursive: true });
|
|
266
|
+
writeFileSync(join(wtDir, "leftover.txt"), "stranded from a prior session\n");
|
|
267
|
+
|
|
268
|
+
// No milestone/M001 branch — already deleted on a previous run.
|
|
269
|
+
const branches = run("git branch --list milestone/M001", dir);
|
|
270
|
+
assert.equal(branches, "", "test fixture: branch should not exist");
|
|
271
|
+
|
|
272
|
+
const result = auditOrphanedMilestoneBranches(dir, "worktree");
|
|
273
|
+
|
|
274
|
+
assert.ok(
|
|
275
|
+
result.recovered.some((r) => r.includes("M001") && r.includes("branch already deleted")),
|
|
276
|
+
`should report branch-less orphan cleanup; got: ${JSON.stringify(result.recovered)}`,
|
|
277
|
+
);
|
|
278
|
+
assert.ok(!existsSync(wtDir), "branch-less orphan worktree dir should be removed");
|
|
279
|
+
});
|
|
280
|
+
|
|
281
|
+
test("#5879 — branch list failure still cleans complete orphan only after branch absence is verified", () => {
|
|
282
|
+
insertMilestone({ id: "M001", title: "Test", status: "complete" });
|
|
283
|
+
|
|
284
|
+
const wtDir = join(dir, ".gsd", "worktrees", "M001");
|
|
285
|
+
mkdirSync(wtDir, { recursive: true });
|
|
286
|
+
writeFileSync(join(wtDir, "leftover.txt"), "stranded from a prior session\n");
|
|
287
|
+
|
|
288
|
+
const result = auditOrphanedMilestoneBranches(dir, "worktree", {
|
|
289
|
+
branchList: () => {
|
|
290
|
+
throw new Error("branch list failed");
|
|
291
|
+
},
|
|
292
|
+
branchExists: (_basePath, branch) => {
|
|
293
|
+
assert.equal(branch, "milestone/M001");
|
|
294
|
+
return false;
|
|
295
|
+
},
|
|
296
|
+
});
|
|
297
|
+
|
|
298
|
+
assert.ok(
|
|
299
|
+
result.recovered.some((r) => r.includes("M001") && r.includes("branch already deleted")),
|
|
300
|
+
`should report verified branch-less orphan cleanup; got: ${JSON.stringify(result.recovered)}`,
|
|
301
|
+
);
|
|
302
|
+
assert.ok(!existsSync(wtDir), "verified branch-less orphan worktree dir should be removed");
|
|
303
|
+
});
|
|
304
|
+
|
|
305
|
+
test("#5879 — branch list failure preserves complete worktree when branch still exists", () => {
|
|
306
|
+
insertMilestone({ id: "M001", title: "Test", status: "complete" });
|
|
307
|
+
|
|
308
|
+
const wtDir = join(dir, ".gsd", "worktrees", "M001");
|
|
309
|
+
mkdirSync(wtDir, { recursive: true });
|
|
310
|
+
writeFileSync(join(wtDir, "live-work.txt"), "do not delete\n");
|
|
311
|
+
|
|
312
|
+
const result = auditOrphanedMilestoneBranches(dir, "worktree", {
|
|
313
|
+
branchList: () => {
|
|
314
|
+
throw new Error("branch list failed");
|
|
315
|
+
},
|
|
316
|
+
branchExists: (_basePath, branch) => {
|
|
317
|
+
assert.equal(branch, "milestone/M001");
|
|
318
|
+
return true;
|
|
319
|
+
},
|
|
320
|
+
});
|
|
321
|
+
|
|
322
|
+
assert.deepStrictEqual(result.recovered, []);
|
|
323
|
+
assert.ok(existsSync(wtDir), "worktree dir must be preserved when branch existence is verified");
|
|
324
|
+
});
|
|
325
|
+
|
|
326
|
+
test("#5879 — skips branch-less orphan for milestone that is not complete", () => {
|
|
327
|
+
// Defensive: only `complete` milestones get the branch-less cleanup. An
|
|
328
|
+
// `active` milestone with no branch but a worktree dir is a different
|
|
329
|
+
// state (probably mid-recovery) and should not be silently wiped.
|
|
330
|
+
insertMilestone({ id: "M001", title: "Test", status: "active" });
|
|
331
|
+
|
|
332
|
+
const wtDir = join(dir, ".gsd", "worktrees", "M001");
|
|
333
|
+
mkdirSync(wtDir, { recursive: true });
|
|
334
|
+
writeFileSync(join(wtDir, "live-work.txt"), "do not delete\n");
|
|
335
|
+
|
|
336
|
+
const result = auditOrphanedMilestoneBranches(dir, "worktree");
|
|
337
|
+
|
|
338
|
+
assert.deepStrictEqual(result.recovered, []);
|
|
339
|
+
assert.ok(existsSync(wtDir), "active milestone worktree dir must be preserved");
|
|
340
|
+
});
|
|
341
|
+
|
|
342
|
+
test("#5879 — branch list failure does not delete worktree for milestone that is not complete", () => {
|
|
343
|
+
insertMilestone({ id: "M001", title: "Test", status: "active" });
|
|
344
|
+
|
|
345
|
+
const wtDir = join(dir, ".gsd", "worktrees", "M001");
|
|
346
|
+
mkdirSync(wtDir, { recursive: true });
|
|
347
|
+
writeFileSync(join(wtDir, "live-work.txt"), "do not delete\n");
|
|
348
|
+
|
|
349
|
+
const result = auditOrphanedMilestoneBranches(dir, "worktree", {
|
|
350
|
+
branchList: () => {
|
|
351
|
+
throw new Error("branch list failed");
|
|
352
|
+
},
|
|
353
|
+
branchExists: () => {
|
|
354
|
+
throw new Error("branchExists should not be called for active milestones");
|
|
355
|
+
},
|
|
356
|
+
});
|
|
357
|
+
|
|
358
|
+
assert.deepStrictEqual(result.recovered, []);
|
|
359
|
+
assert.ok(existsSync(wtDir), "active milestone worktree dir must be preserved");
|
|
360
|
+
});
|
|
361
|
+
|
|
362
|
+
test("#5879 — skips branch-less orphan in 'none' isolation mode", () => {
|
|
363
|
+
insertMilestone({ id: "M001", title: "Test", status: "complete" });
|
|
364
|
+
|
|
365
|
+
const wtDir = join(dir, ".gsd", "worktrees", "M001");
|
|
366
|
+
mkdirSync(wtDir, { recursive: true });
|
|
367
|
+
writeFileSync(join(wtDir, "leftover.txt"), "stranded\n");
|
|
368
|
+
|
|
369
|
+
const result = auditOrphanedMilestoneBranches(dir, "none");
|
|
370
|
+
|
|
371
|
+
assert.deepStrictEqual(result.recovered, []);
|
|
372
|
+
assert.ok(existsSync(wtDir), "'none' mode must not touch worktree dirs");
|
|
373
|
+
});
|
|
254
374
|
});
|