@opengsd/gsd-pi 1.2.0-dev.4813ead6 → 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 +6 -6
- 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 +6 -6
- 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/image-models.generated.d.ts +0 -30
- package/packages/pi-ai/dist/image-models.generated.d.ts.map +1 -1
- package/packages/pi-ai/dist/image-models.generated.js +0 -30
- package/packages/pi-ai/dist/image-models.generated.js.map +1 -1
- package/packages/pi-ai/dist/models.generated.d.ts +361 -255
- package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
- package/packages/pi-ai/dist/models.generated.js +311 -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/{tkLHUSzPA2kMmWz4DmGwI → 2p9Rv9pQflAxCBbGVI2vb}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{tkLHUSzPA2kMmWz4DmGwI → 2p9Rv9pQflAxCBbGVI2vb}/_ssgManifest.js +0 -0
|
@@ -157,7 +157,7 @@ describe("auto-worktree lifecycle", () => {
|
|
|
157
157
|
try {
|
|
158
158
|
const wtPath = createAutoWorktree(tempDir, "M001");
|
|
159
159
|
const realWtPath = realpathSync(wtPath);
|
|
160
|
-
assert.
|
|
160
|
+
assert.equal(realWtPath, join(tempDir, ".gsd-worktrees", "M001"), "worktree uses canonical path under project root, not through the .gsd symlink");
|
|
161
161
|
|
|
162
162
|
_resetAutoWorktreeOriginalBaseForTests();
|
|
163
163
|
process.chdir(realWtPath);
|
|
@@ -192,6 +192,59 @@ describe("auto-worktree lifecycle", () => {
|
|
|
192
192
|
}
|
|
193
193
|
});
|
|
194
194
|
|
|
195
|
+
test("legacy symlink-resolved auto worktree is detected after module state reset", () => {
|
|
196
|
+
tempDir = createTempRepo();
|
|
197
|
+
const savedGsdHome = process.env.GSD_HOME;
|
|
198
|
+
const fakeHome = realpathSync(mkdtempSync(join(tmpdir(), "auto-wt-home-legacy-")));
|
|
199
|
+
const storage = join(fakeHome, ".gsd", "projects", "abc123def456");
|
|
200
|
+
mkdirSync(join(storage, "milestones", "M001"), { recursive: true });
|
|
201
|
+
writeFileSync(join(storage, "milestones", "M001", "CONTEXT.md"), "# M001\n");
|
|
202
|
+
symlinkSync(storage, join(tempDir, ".gsd"));
|
|
203
|
+
process.env.GSD_HOME = join(fakeHome, ".gsd");
|
|
204
|
+
|
|
205
|
+
try {
|
|
206
|
+
// Worktrees created by older versions live at .gsd/worktrees/<MID>;
|
|
207
|
+
// git resolves the symlink and registers them under external state.
|
|
208
|
+
// Detection must keep working for them — canonical .gsd-worktrees/
|
|
209
|
+
// creation never crosses the symlink, so create the legacy worktree
|
|
210
|
+
// explicitly.
|
|
211
|
+
const wtPath = join(tempDir, ".gsd", "worktrees", "M001");
|
|
212
|
+
run(`git worktree add -b milestone/M001 "${wtPath}"`, tempDir);
|
|
213
|
+
const realWtPath = realpathSync(wtPath);
|
|
214
|
+
assert.ok(realWtPath.startsWith(storage), "git registered the symlink-resolved worktree path");
|
|
215
|
+
|
|
216
|
+
_resetAutoWorktreeOriginalBaseForTests();
|
|
217
|
+
process.chdir(realWtPath);
|
|
218
|
+
|
|
219
|
+
assert.ok(isInAutoWorktree(tempDir), "structural detection works without module originalBase");
|
|
220
|
+
const resolved = getAutoWorktreePath(realWtPath, "M001");
|
|
221
|
+
assert.ok(resolved, "existing legacy worktree is found when basePath is the worktree path");
|
|
222
|
+
assert.equal(realpathSync(resolved!), realWtPath);
|
|
223
|
+
|
|
224
|
+
enterAutoWorktree(tempDir, "M001");
|
|
225
|
+
process.chdir(realWtPath);
|
|
226
|
+
assert.deepStrictEqual(
|
|
227
|
+
getActiveAutoWorktreeContext(),
|
|
228
|
+
{
|
|
229
|
+
originalBase: tempDir,
|
|
230
|
+
worktreeName: "M001",
|
|
231
|
+
branch: "milestone/M001",
|
|
232
|
+
},
|
|
233
|
+
"active context is detected from a symlink-resolved legacy worktree cwd",
|
|
234
|
+
);
|
|
235
|
+
} finally {
|
|
236
|
+
process.chdir(tempDir);
|
|
237
|
+
try {
|
|
238
|
+
teardownAutoWorktree(tempDir, "M001");
|
|
239
|
+
} catch {
|
|
240
|
+
// Best-effort cleanup for partially-created temp worktrees.
|
|
241
|
+
}
|
|
242
|
+
if (savedGsdHome === undefined) delete process.env.GSD_HOME;
|
|
243
|
+
else process.env.GSD_HOME = savedGsdHome;
|
|
244
|
+
rmSync(fakeHome, { recursive: true, force: true });
|
|
245
|
+
}
|
|
246
|
+
});
|
|
247
|
+
|
|
195
248
|
test("coexistence with manual worktree", async () => {
|
|
196
249
|
tempDir = createTempRepo();
|
|
197
250
|
const msDir = join(tempDir, ".gsd", "milestones", "M003");
|
|
@@ -293,11 +293,12 @@ describe('git-service', async () => {
|
|
|
293
293
|
|
|
294
294
|
assert.deepStrictEqual(
|
|
295
295
|
RUNTIME_EXCLUSION_PATHS.length,
|
|
296
|
-
|
|
297
|
-
"exactly
|
|
296
|
+
17,
|
|
297
|
+
"exactly 17 runtime exclusion paths"
|
|
298
298
|
);
|
|
299
299
|
|
|
300
300
|
const expectedPaths = [
|
|
301
|
+
".gsd-worktrees/",
|
|
301
302
|
".gsd/activity/",
|
|
302
303
|
".gsd/audit/",
|
|
303
304
|
".gsd/forensics/",
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import assert from "node:assert/strict";
|
|
2
2
|
import { createRequire } from "node:module";
|
|
3
|
-
import { copyFileSync, mkdtempSync, renameSync, rmSync } from "node:fs";
|
|
3
|
+
import { copyFileSync, mkdirSync, mkdtempSync, renameSync, rmSync, writeFileSync } from "node:fs";
|
|
4
4
|
import { join } from "node:path";
|
|
5
5
|
import { tmpdir } from "node:os";
|
|
6
6
|
import test from "node:test";
|
|
@@ -288,6 +288,90 @@ test("migration auto-check refreshes a stale open DB handle before comparing", a
|
|
|
288
288
|
}
|
|
289
289
|
});
|
|
290
290
|
|
|
291
|
+
function writeScratchMilestoneDir(base: string, milestoneId: string, file?: string): void {
|
|
292
|
+
const dir = join(base, ".gsd", "milestones", milestoneId);
|
|
293
|
+
mkdirSync(dir, { recursive: true });
|
|
294
|
+
if (file) writeFileSync(join(dir, file), `# ${milestoneId} discussion context\n`);
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
test("migration auto-check ignores discussion-scratch milestone dirs (CONTEXT only, no DB row)", async () => {
|
|
298
|
+
const base = makeBase();
|
|
299
|
+
try {
|
|
300
|
+
await writeGSDDirectory(projectFixture(), base); // markdown: M001 / S01 / T01
|
|
301
|
+
assert.equal(await ensureDbOpen(base), true);
|
|
302
|
+
insertMilestone({ id: "M001", title: "Legacy Milestone", status: "active" });
|
|
303
|
+
insertSlice({ id: "S01", milestoneId: "M001", title: "Legacy Slice", status: "pending", risk: "medium", depends: [], demo: "Legacy slice demo", sequence: 1 });
|
|
304
|
+
insertTask({ id: "T01", sliceId: "S01", milestoneId: "M001", title: "Legacy Task", status: "pending" });
|
|
305
|
+
|
|
306
|
+
// Mid-discussion artifacts: dirs with no ROADMAP and no DB row. The queued
|
|
307
|
+
// DB row is only inserted at discussion handoff, so these are expected to
|
|
308
|
+
// be DB-less — not drift, and recover must not be recommended (it would
|
|
309
|
+
// import them as ghost active milestones).
|
|
310
|
+
writeScratchMilestoneDir(base, "M002", "M002-CONTEXT.md");
|
|
311
|
+
writeScratchMilestoneDir(base, "M003", "M003-CONTEXT-DRAFT.md");
|
|
312
|
+
writeScratchMilestoneDir(base, "M004"); // empty dir
|
|
313
|
+
|
|
314
|
+
const result = await checkMarkdownHierarchyAgainstDb(base);
|
|
315
|
+
assert.equal(result.action, "none");
|
|
316
|
+
assert.equal(result.reason, "in-sync");
|
|
317
|
+
assert.deepEqual(result.markdown, { milestones: 1, slices: 1, tasks: 1 });
|
|
318
|
+
} finally {
|
|
319
|
+
cleanup(base);
|
|
320
|
+
}
|
|
321
|
+
});
|
|
322
|
+
|
|
323
|
+
test("migration auto-check stays quiet mid-first-discussion (scratch dir over empty DB)", async () => {
|
|
324
|
+
const base = makeBase();
|
|
325
|
+
try {
|
|
326
|
+
await writeGSDDirectory({ projectContent: "# P\n", decisionsContent: "", requirements: [], milestones: [] }, base);
|
|
327
|
+
assert.equal(await ensureDbOpen(base), true);
|
|
328
|
+
writeScratchMilestoneDir(base, "M001", "M001-CONTEXT.md");
|
|
329
|
+
|
|
330
|
+
const result = await checkMarkdownHierarchyAgainstDb(base);
|
|
331
|
+
assert.equal(result.action, "none");
|
|
332
|
+
assert.equal(result.reason, "no-markdown");
|
|
333
|
+
} finally {
|
|
334
|
+
cleanup(base);
|
|
335
|
+
}
|
|
336
|
+
});
|
|
337
|
+
|
|
338
|
+
test("migration auto-check still reports real drift with scratch dirs excluded from counts", async () => {
|
|
339
|
+
const base = makeBase();
|
|
340
|
+
try {
|
|
341
|
+
await writeGSDDirectory(projectFixture(), base); // markdown: M001 / S01 / T01, DB empty
|
|
342
|
+
assert.equal(await ensureDbOpen(base), true);
|
|
343
|
+
writeScratchMilestoneDir(base, "M002", "M002-CONTEXT.md");
|
|
344
|
+
|
|
345
|
+
const result = await checkMarkdownHierarchyAgainstDb(base);
|
|
346
|
+
assert.equal(result.action, "recovery-required");
|
|
347
|
+
assert.equal(result.reason, "db-empty");
|
|
348
|
+
assert.equal(result.recoveryCommand, "/gsd recover --confirm");
|
|
349
|
+
// The scratch dir must not inflate the reported markdown count.
|
|
350
|
+
assert.deepEqual(result.markdown, { milestones: 1, slices: 1, tasks: 1 });
|
|
351
|
+
} finally {
|
|
352
|
+
cleanup(base);
|
|
353
|
+
}
|
|
354
|
+
});
|
|
355
|
+
|
|
356
|
+
test("migration auto-check still compares a roadmapless milestone that HAS a DB row", async () => {
|
|
357
|
+
const base = makeBase();
|
|
358
|
+
try {
|
|
359
|
+
await writeGSDDirectory({ projectContent: "# P\n", decisionsContent: "", requirements: [], milestones: [] }, base);
|
|
360
|
+
assert.equal(await ensureDbOpen(base), true);
|
|
361
|
+
// Post-handoff queued milestone: CONTEXT-only dir WITH a DB row. It must
|
|
362
|
+
// stay in the comparison (both sides have it → in-sync).
|
|
363
|
+
insertMilestone({ id: "M001", title: "M001", status: "queued" });
|
|
364
|
+
writeScratchMilestoneDir(base, "M001", "M001-CONTEXT.md");
|
|
365
|
+
|
|
366
|
+
const result = await checkMarkdownHierarchyAgainstDb(base);
|
|
367
|
+
assert.equal(result.action, "none");
|
|
368
|
+
assert.equal(result.reason, "in-sync");
|
|
369
|
+
assert.deepEqual(result.markdown, { milestones: 1, slices: 0, tasks: 0 });
|
|
370
|
+
} finally {
|
|
371
|
+
cleanup(base);
|
|
372
|
+
}
|
|
373
|
+
});
|
|
374
|
+
|
|
291
375
|
test("rebuildMarkdownProjectionsFromDb realigns markdown when DB holds extra rows", async () => {
|
|
292
376
|
const base = makeBase();
|
|
293
377
|
try {
|
|
@@ -306,6 +306,38 @@ describe("notification-store", () => {
|
|
|
306
306
|
rmSync(lockPath, { force: true });
|
|
307
307
|
});
|
|
308
308
|
|
|
309
|
+
test("structured meta persists kind and scope on the entry", () => {
|
|
310
|
+
initNotificationStore(tmp);
|
|
311
|
+
appendNotification("Auto-mode blocked — validation gate", "warning", "notify", { kind: "auto-stop", scope: "M005" });
|
|
312
|
+
|
|
313
|
+
const entries = readNotifications();
|
|
314
|
+
assert.equal(entries.length, 1);
|
|
315
|
+
assert.equal(entries[0].kind, "auto-stop");
|
|
316
|
+
assert.equal(entries[0].scope, "M005");
|
|
317
|
+
});
|
|
318
|
+
|
|
319
|
+
test("dedup keys on kind+scope when present, not on prose", () => {
|
|
320
|
+
initNotificationStore(tmp);
|
|
321
|
+
appendNotification("Auto-mode blocked — validation gate", "warning", "notify", { kind: "auto-stop", scope: "M005" });
|
|
322
|
+
// Rephrased message, same structured identity → deduped within the window
|
|
323
|
+
appendNotification("Auto-mode blocked — gate rejected", "warning", "notify", { kind: "auto-stop", scope: "M005" });
|
|
324
|
+
// Same kind, different scope → distinct
|
|
325
|
+
appendNotification("Auto-mode blocked — validation gate", "warning", "notify", { kind: "auto-stop", scope: "M006" });
|
|
326
|
+
|
|
327
|
+
assert.equal(readNotifications().length, 2);
|
|
328
|
+
});
|
|
329
|
+
|
|
330
|
+
test("readNotifications filters by kind and scope", () => {
|
|
331
|
+
initNotificationStore(tmp);
|
|
332
|
+
appendNotification("a", "info", "notify", { kind: "auto-stop", scope: "M005" });
|
|
333
|
+
appendNotification("b", "info", "notify", { kind: "provider-error-pause", scope: "M005" });
|
|
334
|
+
appendNotification("c", "info");
|
|
335
|
+
|
|
336
|
+
assert.equal(readNotifications(tmp, { kind: "auto-stop" }).length, 1);
|
|
337
|
+
assert.equal(readNotifications(tmp, { scope: "M005" }).length, 2);
|
|
338
|
+
assert.equal(readNotifications(tmp, { kind: "provider-error-pause", scope: "M005" })[0].message, "b");
|
|
339
|
+
});
|
|
340
|
+
|
|
309
341
|
test("listeners are notified on append, markAllRead, and clear", () => {
|
|
310
342
|
initNotificationStore(tmp);
|
|
311
343
|
let calls = 0;
|
|
@@ -21,6 +21,7 @@ import { join } from "node:path";
|
|
|
21
21
|
import {
|
|
22
22
|
extractPackageReferences,
|
|
23
23
|
checkFilePathConsistency,
|
|
24
|
+
checkPlanningArtifactReferences,
|
|
24
25
|
checkTaskOrdering,
|
|
25
26
|
checkInterfaceContracts,
|
|
26
27
|
checkVerificationCommands,
|
|
@@ -973,7 +974,12 @@ describe("runPreExecutionChecks", () => {
|
|
|
973
974
|
}
|
|
974
975
|
});
|
|
975
976
|
|
|
976
|
-
|
|
977
|
+
// Supersedes #5492: .gsd metadata inputs used to be accepted in worktree mode
|
|
978
|
+
// by resolving from the canonical project root. Task inputs are source files
|
|
979
|
+
// only (CONTEXT.md "Task Input") — planning artifacts are now blocked with a
|
|
980
|
+
// removal message regardless of where they resolve. The canonical-root
|
|
981
|
+
// resolution itself remains for merged-but-unsynced source files.
|
|
982
|
+
test("blocks .gsd metadata inputs even when resolvable from canonical project root in worktree mode", async () => {
|
|
977
983
|
const projectRoot = join(tmpdir(), `pre-exec-project-root-${Date.now()}`);
|
|
978
984
|
const worktreeRoot = join(tmpdir(), `pre-exec-worktree-root-${Date.now()}`);
|
|
979
985
|
mkdirSync(join(projectRoot, ".gsd"), { recursive: true });
|
|
@@ -990,6 +996,35 @@ describe("runPreExecutionChecks", () => {
|
|
|
990
996
|
}),
|
|
991
997
|
];
|
|
992
998
|
|
|
999
|
+
const result = await runPreExecutionChecks(tasks, worktreeRoot, {
|
|
1000
|
+
canonicalProjectRoot: projectRoot,
|
|
1001
|
+
});
|
|
1002
|
+
assert.equal(result.status, "fail");
|
|
1003
|
+
assert.equal(result.checks.length, 1);
|
|
1004
|
+
assert.ok(result.checks[0].message.includes("preloaded as context"));
|
|
1005
|
+
} finally {
|
|
1006
|
+
rmSync(projectRoot, { recursive: true, force: true });
|
|
1007
|
+
rmSync(worktreeRoot, { recursive: true, force: true });
|
|
1008
|
+
}
|
|
1009
|
+
});
|
|
1010
|
+
|
|
1011
|
+
test("resolves source-file inputs from canonical project root in worktree mode (#5492)", async () => {
|
|
1012
|
+
const projectRoot = join(tmpdir(), `pre-exec-project-root-src-${Date.now()}`);
|
|
1013
|
+
const worktreeRoot = join(tmpdir(), `pre-exec-worktree-root-src-${Date.now()}`);
|
|
1014
|
+
mkdirSync(join(projectRoot, "src"), { recursive: true });
|
|
1015
|
+
mkdirSync(worktreeRoot, { recursive: true });
|
|
1016
|
+
writeFileSync(join(projectRoot, "src", "merged.ts"), "// merged upstream");
|
|
1017
|
+
|
|
1018
|
+
try {
|
|
1019
|
+
const tasks = [
|
|
1020
|
+
createTask({
|
|
1021
|
+
id: "T01",
|
|
1022
|
+
files: [],
|
|
1023
|
+
inputs: ["src/merged.ts"],
|
|
1024
|
+
expected_output: [],
|
|
1025
|
+
}),
|
|
1026
|
+
];
|
|
1027
|
+
|
|
993
1028
|
const result = await runPreExecutionChecks(tasks, worktreeRoot, {
|
|
994
1029
|
canonicalProjectRoot: projectRoot,
|
|
995
1030
|
});
|
|
@@ -2263,3 +2298,160 @@ describe("checkFilePathConsistency quote-wrapped annotation (#3747)", () => {
|
|
|
2263
2298
|
);
|
|
2264
2299
|
});
|
|
2265
2300
|
});
|
|
2301
|
+
|
|
2302
|
+
// ─── Planning Artifact Reference Tests ───────────────────────────────────────
|
|
2303
|
+
|
|
2304
|
+
describe("checkPlanningArtifactReferences", () => {
|
|
2305
|
+
function withTempDir(run: (tempDir: string) => void): void {
|
|
2306
|
+
const tempDir = join(tmpdir(), `pre-exec-artifact-test-${Date.now()}-${Math.random().toString(36).slice(2)}`);
|
|
2307
|
+
mkdirSync(tempDir, { recursive: true });
|
|
2308
|
+
try {
|
|
2309
|
+
run(tempDir);
|
|
2310
|
+
} finally {
|
|
2311
|
+
rmSync(tempDir, { recursive: true, force: true });
|
|
2312
|
+
}
|
|
2313
|
+
}
|
|
2314
|
+
|
|
2315
|
+
test("blocks .gsd/ paths in inputs even when the file exists", () => {
|
|
2316
|
+
withTempDir((tempDir) => {
|
|
2317
|
+
const artifactDir = join(tempDir, ".gsd", "milestones", "M001");
|
|
2318
|
+
mkdirSync(artifactDir, { recursive: true });
|
|
2319
|
+
writeFileSync(join(artifactDir, "M001-CONTEXT.md"), "# context");
|
|
2320
|
+
|
|
2321
|
+
const tasks = [
|
|
2322
|
+
createTask({ id: "T01", inputs: [".gsd/milestones/M001/M001-CONTEXT.md"] }),
|
|
2323
|
+
];
|
|
2324
|
+
|
|
2325
|
+
const results = checkPlanningArtifactReferences(tasks, tempDir);
|
|
2326
|
+
assert.equal(results.length, 1);
|
|
2327
|
+
assert.equal(results[0].blocking, true);
|
|
2328
|
+
assert.ok(results[0].message.includes("preloaded as context"));
|
|
2329
|
+
});
|
|
2330
|
+
});
|
|
2331
|
+
|
|
2332
|
+
test("blocks bare artifact names that do not resolve to a real file, with the truthful message", () => {
|
|
2333
|
+
withTempDir((tempDir) => {
|
|
2334
|
+
const tasks = [createTask({ id: "T01", inputs: ["M001-CONTEXT.md"] })];
|
|
2335
|
+
|
|
2336
|
+
const results = checkPlanningArtifactReferences(tasks, tempDir);
|
|
2337
|
+
assert.equal(results.length, 1);
|
|
2338
|
+
assert.ok(results[0].message.includes("preloaded as context"));
|
|
2339
|
+
assert.ok(!results[0].message.includes("doesn't exist"));
|
|
2340
|
+
|
|
2341
|
+
// The consistency check must not double-report the same entry with the
|
|
2342
|
+
// misleading "doesn't exist" message.
|
|
2343
|
+
const consistency = checkFilePathConsistency(tasks, tempDir);
|
|
2344
|
+
assert.deepEqual(consistency, []);
|
|
2345
|
+
});
|
|
2346
|
+
});
|
|
2347
|
+
|
|
2348
|
+
test("allows a bare artifact-style name that exists as a real source file", () => {
|
|
2349
|
+
withTempDir((tempDir) => {
|
|
2350
|
+
writeFileSync(join(tempDir, "M001-CONTEXT.md"), "# tracked source file");
|
|
2351
|
+
|
|
2352
|
+
const tasks = [createTask({ id: "T01", inputs: ["M001-CONTEXT.md"] })];
|
|
2353
|
+
assert.deepEqual(checkPlanningArtifactReferences(tasks, tempDir), []);
|
|
2354
|
+
});
|
|
2355
|
+
});
|
|
2356
|
+
|
|
2357
|
+
test("leaves artifact-style names with directory components to the existence checks", () => {
|
|
2358
|
+
withTempDir((tempDir) => {
|
|
2359
|
+
const tasks = [createTask({ id: "T01", inputs: ["tests/fixtures/M001-CONTEXT.md"] })];
|
|
2360
|
+
|
|
2361
|
+
assert.deepEqual(checkPlanningArtifactReferences(tasks, tempDir), []);
|
|
2362
|
+
const consistency = checkFilePathConsistency(tasks, tempDir);
|
|
2363
|
+
assert.equal(consistency.length, 1);
|
|
2364
|
+
assert.ok(consistency[0].message.includes("doesn't exist"));
|
|
2365
|
+
});
|
|
2366
|
+
});
|
|
2367
|
+
|
|
2368
|
+
test("blocks .planning/ and .audits/ paths in files", () => {
|
|
2369
|
+
withTempDir((tempDir) => {
|
|
2370
|
+
const tasks = [
|
|
2371
|
+
createTask({ id: "T01", files: [".planning/codebase/STACK.md", ".audits/security.md"] }),
|
|
2372
|
+
];
|
|
2373
|
+
|
|
2374
|
+
const results = checkPlanningArtifactReferences(tasks, tempDir);
|
|
2375
|
+
assert.equal(results.length, 2);
|
|
2376
|
+
assert.ok(results.every((r) => r.blocking && r.message.includes("never task files")));
|
|
2377
|
+
});
|
|
2378
|
+
});
|
|
2379
|
+
|
|
2380
|
+
test("blocks .gsd/ paths in expectedOutput with the workflow-tools message and no ordering duplicate", () => {
|
|
2381
|
+
withTempDir((tempDir) => {
|
|
2382
|
+
const tasks = [
|
|
2383
|
+
createTask({
|
|
2384
|
+
id: "T01",
|
|
2385
|
+
sequence: 0,
|
|
2386
|
+
expected_output: [".gsd/milestones/M001/M001-SUMMARY.md"],
|
|
2387
|
+
}),
|
|
2388
|
+
createTask({
|
|
2389
|
+
id: "T02",
|
|
2390
|
+
sequence: 1,
|
|
2391
|
+
inputs: [".gsd/milestones/M001/M001-SUMMARY.md"],
|
|
2392
|
+
}),
|
|
2393
|
+
];
|
|
2394
|
+
|
|
2395
|
+
const results = checkPlanningArtifactReferences(tasks, tempDir);
|
|
2396
|
+
assert.equal(results.length, 2);
|
|
2397
|
+
const outputFinding = results.find((r) => r.message.includes("expectedOutput"));
|
|
2398
|
+
assert.ok(outputFinding);
|
|
2399
|
+
assert.ok(outputFinding.message.includes("written by workflow tools"));
|
|
2400
|
+
|
|
2401
|
+
assert.deepEqual(checkTaskOrdering(tasks, tempDir), []);
|
|
2402
|
+
});
|
|
2403
|
+
});
|
|
2404
|
+
|
|
2405
|
+
test("skips prose, sentinels, and runtime-only entries", () => {
|
|
2406
|
+
withTempDir((tempDir) => {
|
|
2407
|
+
const tasks = [
|
|
2408
|
+
createTask({
|
|
2409
|
+
id: "T01",
|
|
2410
|
+
inputs: ["none", "Current enum shape", "server logs (runtime)"],
|
|
2411
|
+
}),
|
|
2412
|
+
];
|
|
2413
|
+
|
|
2414
|
+
assert.deepEqual(checkPlanningArtifactReferences(tasks, tempDir), []);
|
|
2415
|
+
});
|
|
2416
|
+
});
|
|
2417
|
+
|
|
2418
|
+
test("runPreExecutionChecks surfaces artifact findings as blocking failures", async () => {
|
|
2419
|
+
const tempDir = join(tmpdir(), `pre-exec-artifact-run-${Date.now()}`);
|
|
2420
|
+
mkdirSync(tempDir, { recursive: true });
|
|
2421
|
+
try {
|
|
2422
|
+
const tasks = [createTask({ id: "T01", inputs: ["M001-CONTEXT.md"] })];
|
|
2423
|
+
const result: PreExecutionResult = await runPreExecutionChecks(tasks, tempDir);
|
|
2424
|
+
assert.equal(result.status, "fail");
|
|
2425
|
+
assert.ok(
|
|
2426
|
+
result.checks.some((c) => !c.passed && c.blocking && c.message.includes("preloaded as context")),
|
|
2427
|
+
);
|
|
2428
|
+
} finally {
|
|
2429
|
+
rmSync(tempDir, { recursive: true, force: true });
|
|
2430
|
+
}
|
|
2431
|
+
});
|
|
2432
|
+
|
|
2433
|
+
test("allows a bare artifact-style name that exists only at canonicalProjectRoot (not at worktree basePath)", () => {
|
|
2434
|
+
withTempDir((canonicalRoot) => {
|
|
2435
|
+
withTempDir((worktreeRoot) => {
|
|
2436
|
+
writeFileSync(join(canonicalRoot, "M001-CONTEXT.md"), "# tracked source file");
|
|
2437
|
+
const tasks = [createTask({ id: "T01", inputs: ["M001-CONTEXT.md"] })];
|
|
2438
|
+
assert.deepEqual(
|
|
2439
|
+
checkPlanningArtifactReferences(tasks, worktreeRoot, { canonicalProjectRoot: canonicalRoot }),
|
|
2440
|
+
[],
|
|
2441
|
+
);
|
|
2442
|
+
});
|
|
2443
|
+
});
|
|
2444
|
+
});
|
|
2445
|
+
|
|
2446
|
+
test("blocks .GSD/ paths with mixed casing (case-insensitive metadata dir match)", () => {
|
|
2447
|
+
withTempDir((tempDir) => {
|
|
2448
|
+
const tasks = [
|
|
2449
|
+
createTask({ id: "T01", inputs: [".GSD/milestones/M001/M001-CONTEXT.md"] }),
|
|
2450
|
+
];
|
|
2451
|
+
const results = checkPlanningArtifactReferences(tasks, tempDir);
|
|
2452
|
+
assert.equal(results.length, 1);
|
|
2453
|
+
assert.equal(results[0].blocking, true);
|
|
2454
|
+
assert.ok(results[0].message.includes("preloaded as context"));
|
|
2455
|
+
});
|
|
2456
|
+
});
|
|
2457
|
+
});
|
|
@@ -3,10 +3,10 @@ import assert from "node:assert/strict";
|
|
|
3
3
|
|
|
4
4
|
import { classifyError, isTransient } from "../error-classifier.ts";
|
|
5
5
|
import {
|
|
6
|
-
formatProviderErrorGuidance,
|
|
7
6
|
resolveProviderErrorGuidance,
|
|
8
7
|
unitTypeToPrefsPhaseKey,
|
|
9
8
|
} from "../provider-error-guidance.ts";
|
|
9
|
+
import { formatGuidance } from "../guidance.ts";
|
|
10
10
|
|
|
11
11
|
test("classifyError: Cloud Code Assist 400 invalid argument is model-error", () => {
|
|
12
12
|
const result = classifyError(
|
|
@@ -75,8 +75,8 @@ test("resolveProviderErrorGuidance suggests gemini-3-flash for antigravity pro-h
|
|
|
75
75
|
assert.ok(guidance.steps.some((step) => step.includes("fallbacks")));
|
|
76
76
|
});
|
|
77
77
|
|
|
78
|
-
test("
|
|
79
|
-
const text =
|
|
78
|
+
test("formatGuidance numbers steps", () => {
|
|
79
|
+
const text = formatGuidance({
|
|
80
80
|
summary: "Provider error on test/model.",
|
|
81
81
|
steps: ["Change model", "Run /gsd next"],
|
|
82
82
|
});
|
|
@@ -0,0 +1,120 @@
|
|
|
1
|
+
// gsd-pi — ADR-034 Publication module tests.
|
|
2
|
+
//
|
|
3
|
+
// Exercises publishMilestone against real local git fixtures (a bare repo as
|
|
4
|
+
// the remote) — no network, no gh. The PR path is only tested up to its
|
|
5
|
+
// non-fatal failure contract, since createDraftPullRequestFromEvidence shells
|
|
6
|
+
// out to gh.
|
|
7
|
+
|
|
8
|
+
import test from "node:test";
|
|
9
|
+
import assert from "node:assert/strict";
|
|
10
|
+
import { rmSync } from "node:fs";
|
|
11
|
+
import { join } from "node:path";
|
|
12
|
+
|
|
13
|
+
import { gitRemoteExists, publishMilestone } from "../publication.ts";
|
|
14
|
+
import { git, makeTempDir, makeTempRepo } from "./test-utils.ts";
|
|
15
|
+
|
|
16
|
+
function makeRepoWithBareRemote(): { repo: string; bare: string; cleanup: () => void } {
|
|
17
|
+
const repo = makeTempRepo("gsd-publication-test-");
|
|
18
|
+
const bare = join(makeTempDir("gsd-publication-remote-"), "remote.git");
|
|
19
|
+
git(repo, "init", "--bare", bare);
|
|
20
|
+
git(repo, "remote", "add", "origin", bare);
|
|
21
|
+
return {
|
|
22
|
+
repo,
|
|
23
|
+
bare,
|
|
24
|
+
cleanup: () => {
|
|
25
|
+
rmSync(repo, { recursive: true, force: true });
|
|
26
|
+
rmSync(bare, { recursive: true, force: true });
|
|
27
|
+
},
|
|
28
|
+
};
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
const NO_PUBLISH_PREFS = { autoPush: false, autoPr: false };
|
|
32
|
+
|
|
33
|
+
function makeRequest(repo: string, prefs: { autoPush: boolean; autoPr: boolean; remote?: string; prTargetBranch?: string }) {
|
|
34
|
+
return {
|
|
35
|
+
basePath: repo,
|
|
36
|
+
milestoneId: "M001",
|
|
37
|
+
milestoneTitle: "Test milestone",
|
|
38
|
+
integrationBranch: "main",
|
|
39
|
+
milestoneBranch: "milestone/M001",
|
|
40
|
+
sliceSummaries: ["### S01\nSlice one"],
|
|
41
|
+
nothingToCommit: false,
|
|
42
|
+
prefs,
|
|
43
|
+
};
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
test("gitRemoteExists distinguishes configured from missing remotes", () => {
|
|
47
|
+
const { repo, cleanup } = makeRepoWithBareRemote();
|
|
48
|
+
try {
|
|
49
|
+
assert.equal(gitRemoteExists(repo, "origin"), true);
|
|
50
|
+
assert.equal(gitRemoteExists(repo, "upstream"), false);
|
|
51
|
+
} finally {
|
|
52
|
+
cleanup();
|
|
53
|
+
}
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
test("auto-push pushes the integration branch to the remote", () => {
|
|
57
|
+
const { repo, bare, cleanup } = makeRepoWithBareRemote();
|
|
58
|
+
try {
|
|
59
|
+
const result = publishMilestone(makeRequest(repo, { autoPush: true, autoPr: false }));
|
|
60
|
+
assert.equal(result.pushed, true);
|
|
61
|
+
assert.equal(result.prCreated, false);
|
|
62
|
+
const remoteHeads = git(repo, "ls-remote", "--heads", bare);
|
|
63
|
+
assert.match(remoteHeads, /refs\/heads\/main/);
|
|
64
|
+
} finally {
|
|
65
|
+
cleanup();
|
|
66
|
+
}
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
test("auto-push is suppressed when auto-PR owns the remote interaction", () => {
|
|
70
|
+
const { repo, bare, cleanup } = makeRepoWithBareRemote();
|
|
71
|
+
try {
|
|
72
|
+
git(repo, "branch", "milestone/M001");
|
|
73
|
+
const result = publishMilestone(makeRequest(repo, { autoPush: true, autoPr: true }));
|
|
74
|
+
// PR creation pushes the milestone branch, then fails non-fatally at gh.
|
|
75
|
+
assert.equal(result.pushed, false);
|
|
76
|
+
assert.equal(result.prCreated, false);
|
|
77
|
+
assert.equal(result.prUrl, undefined);
|
|
78
|
+
const remoteHeads = git(repo, "ls-remote", "--heads", bare);
|
|
79
|
+
assert.match(remoteHeads, /refs\/heads\/milestone\/M001/);
|
|
80
|
+
assert.doesNotMatch(remoteHeads, /refs\/heads\/main/);
|
|
81
|
+
} finally {
|
|
82
|
+
cleanup();
|
|
83
|
+
}
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
test("nothing-to-commit short-circuits all publication", () => {
|
|
87
|
+
const { repo, bare, cleanup } = makeRepoWithBareRemote();
|
|
88
|
+
try {
|
|
89
|
+
const result = publishMilestone({
|
|
90
|
+
...makeRequest(repo, { autoPush: true, autoPr: false }),
|
|
91
|
+
nothingToCommit: true,
|
|
92
|
+
});
|
|
93
|
+
assert.deepEqual(result, { pushed: false, prCreated: false });
|
|
94
|
+
const remoteHeads = git(repo, "ls-remote", "--heads", bare);
|
|
95
|
+
assert.equal(remoteHeads.trim(), "");
|
|
96
|
+
} finally {
|
|
97
|
+
cleanup();
|
|
98
|
+
}
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
test("missing remote makes publication a silent no-op", () => {
|
|
102
|
+
const { repo, cleanup } = makeRepoWithBareRemote();
|
|
103
|
+
try {
|
|
104
|
+
git(repo, "remote", "remove", "origin");
|
|
105
|
+
const result = publishMilestone(makeRequest(repo, { autoPush: true, autoPr: true }));
|
|
106
|
+
assert.deepEqual(result, { pushed: false, prCreated: false });
|
|
107
|
+
} finally {
|
|
108
|
+
cleanup();
|
|
109
|
+
}
|
|
110
|
+
});
|
|
111
|
+
|
|
112
|
+
test("publication disabled returns the empty result without touching git", () => {
|
|
113
|
+
const { repo, cleanup } = makeRepoWithBareRemote();
|
|
114
|
+
try {
|
|
115
|
+
const result = publishMilestone(makeRequest(repo, NO_PUBLISH_PREFS));
|
|
116
|
+
assert.deepEqual(result, { pushed: false, prCreated: false });
|
|
117
|
+
} finally {
|
|
118
|
+
cleanup();
|
|
119
|
+
}
|
|
120
|
+
});
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
// GSD — recovery-classification: illegal-transition kind (ADR-030)
|
|
2
|
+
|
|
3
|
+
import test from "node:test";
|
|
4
|
+
import assert from "node:assert/strict";
|
|
5
|
+
|
|
6
|
+
import { classifyFailure } from "../recovery-classification.ts";
|
|
7
|
+
import { IllegalPhaseTransitionError } from "../state-transition-matrix.ts";
|
|
8
|
+
|
|
9
|
+
test("classifyFailure recognizes IllegalPhaseTransitionError by class and escalates", () => {
|
|
10
|
+
const classification = classifyFailure({
|
|
11
|
+
error: new IllegalPhaseTransitionError("executing", "complete"),
|
|
12
|
+
unitType: "execute-task",
|
|
13
|
+
unitId: "T-1",
|
|
14
|
+
});
|
|
15
|
+
|
|
16
|
+
assert.equal(classification.failureKind, "illegal-transition");
|
|
17
|
+
assert.equal(classification.action, "escalate");
|
|
18
|
+
assert.equal(classification.exitReason, "illegal-transition");
|
|
19
|
+
assert.match(classification.reason, /Illegal phase transition/);
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
test("classifyFailure routes an explicit illegal-transition failureKind to the same case", () => {
|
|
23
|
+
const classification = classifyFailure({
|
|
24
|
+
error: new Error("derived edge rejected"),
|
|
25
|
+
failureKind: "illegal-transition",
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
assert.equal(classification.failureKind, "illegal-transition");
|
|
29
|
+
assert.equal(classification.action, "escalate");
|
|
30
|
+
});
|