gsd-pi 2.82.0-dev.ed17d078d → 3.0.0-dev.1b44e695b
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 +93 -18
- package/dist/cli.js +20 -9
- package/dist/headless-ui.js +13 -6
- package/dist/headless.js +9 -2
- 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/claude-code-cli/stream-adapter.js +44 -6
- package/dist/resources/extensions/cmux/index.js +5 -0
- package/dist/resources/extensions/gsd/auto/detect-stuck.js +1 -1
- package/dist/resources/extensions/gsd/auto/infra-errors.js +9 -3
- package/dist/resources/extensions/gsd/auto/loop.js +122 -40
- package/dist/resources/extensions/gsd/auto/orchestrator.js +15 -4
- package/dist/resources/extensions/gsd/auto/phases.js +134 -49
- package/dist/resources/extensions/gsd/auto/session.js +6 -0
- package/dist/resources/extensions/gsd/auto/unit-runner-events.js +7 -1
- package/dist/resources/extensions/gsd/auto/workflow-kernel.js +3 -0
- package/dist/resources/extensions/gsd/auto/workflow-memory-pressure.js +12 -0
- package/dist/resources/extensions/gsd/auto-budget.js +9 -0
- package/dist/resources/extensions/gsd/auto-dashboard.js +66 -1
- package/dist/resources/extensions/gsd/auto-direct-dispatch.js +1 -0
- package/dist/resources/extensions/gsd/auto-dispatch.js +144 -30
- package/dist/resources/extensions/gsd/auto-model-selection.js +2 -0
- package/dist/resources/extensions/gsd/auto-post-unit.js +329 -137
- package/dist/resources/extensions/gsd/auto-prompts.js +36 -10
- package/dist/resources/extensions/gsd/auto-recovery.js +82 -16
- package/dist/resources/extensions/gsd/auto-start.js +99 -16
- package/dist/resources/extensions/gsd/auto-timers.js +11 -3
- package/dist/resources/extensions/gsd/auto-verification.js +146 -34
- package/dist/resources/extensions/gsd/auto-worktree.js +185 -26
- package/dist/resources/extensions/gsd/auto.js +135 -74
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +65 -10
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +13 -10
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +14 -4
- package/dist/resources/extensions/gsd/bootstrap/subagent-input.js +21 -9
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +24 -9
- package/dist/resources/extensions/gsd/clean-root-preflight.js +170 -8
- package/dist/resources/extensions/gsd/commands/catalog.js +10 -1
- package/dist/resources/extensions/gsd/commands/handlers/core.js +38 -0
- package/dist/resources/extensions/gsd/commands/handlers/ops.js +20 -0
- package/dist/resources/extensions/gsd/commands-bootstrap.js +5 -0
- package/dist/resources/extensions/gsd/commands-handlers.js +2 -0
- package/dist/resources/extensions/gsd/commands-mcp-status.js +9 -0
- package/dist/resources/extensions/gsd/commands-prefs-wizard.js +10 -2
- package/dist/resources/extensions/gsd/commands-verdict.js +139 -0
- package/dist/resources/extensions/gsd/crash-recovery.js +55 -7
- package/dist/resources/extensions/gsd/db/auto-workers.js +30 -0
- package/dist/resources/extensions/gsd/db/milestone-leases.js +24 -0
- package/dist/resources/extensions/gsd/db/unit-dispatches.js +3 -2
- package/dist/resources/extensions/gsd/db-base-schema.js +2 -0
- package/dist/resources/extensions/gsd/db-migration-steps.js +4 -0
- package/dist/resources/extensions/gsd/db-task-slice-rows.js +2 -0
- package/dist/resources/extensions/gsd/dispatch-guard.js +46 -2
- package/dist/resources/extensions/gsd/docs/preferences-reference.md +10 -0
- package/dist/resources/extensions/gsd/doctor-git-checks.js +46 -1
- 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/forensics.js +10 -3
- package/dist/resources/extensions/gsd/git-service.js +152 -15
- package/dist/resources/extensions/gsd/gsd-db.js +76 -33
- package/dist/resources/extensions/gsd/guided-flow-queue.js +4 -3
- package/dist/resources/extensions/gsd/guided-flow.js +110 -117
- package/dist/resources/extensions/gsd/guided-unit-context.js +23 -0
- package/dist/resources/extensions/gsd/init-wizard.js +17 -2
- package/dist/resources/extensions/gsd/markdown-renderer.js +14 -11
- package/dist/resources/extensions/gsd/mcp-filter.js +58 -0
- package/dist/resources/extensions/gsd/migrate/parsers.js +121 -2
- 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 +57 -14
- package/dist/resources/extensions/gsd/parallel-orchestrator.js +3 -0
- package/dist/resources/extensions/gsd/paths.js +4 -0
- package/dist/resources/extensions/gsd/pending-auto-start.js +52 -0
- package/dist/resources/extensions/gsd/planning-path-scope.js +9 -3
- package/dist/resources/extensions/gsd/post-execution-checks.js +73 -9
- package/dist/resources/extensions/gsd/pre-execution-checks.js +54 -19
- package/dist/resources/extensions/gsd/preferences-mcp.js +19 -0
- package/dist/resources/extensions/gsd/preferences-types.js +3 -0
- package/dist/resources/extensions/gsd/preferences-validation.js +147 -0
- package/dist/resources/extensions/gsd/preferences.js +6 -0
- 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/forensics.md +3 -3
- 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/reactive-execute.md +1 -1
- 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/queue-reorder-ui.js +30 -13
- package/dist/resources/extensions/gsd/repo-identity.js +39 -22
- package/dist/resources/extensions/gsd/repository-registry.js +44 -0
- package/dist/resources/extensions/gsd/safety/evidence-collector.js +2 -0
- package/dist/resources/extensions/gsd/safety/evidence-cross-ref.js +42 -18
- package/dist/resources/extensions/gsd/session-lock.js +15 -2
- package/dist/resources/extensions/gsd/slice-parallel-conflict.js +2 -2
- package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +84 -5
- package/dist/resources/extensions/gsd/smart-entry-routing.js +36 -0
- 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/state.js +28 -7
- package/dist/resources/extensions/gsd/status-guards.js +14 -2
- package/dist/resources/extensions/gsd/templates/PREFERENCES.md +1 -0
- package/dist/resources/extensions/gsd/templates/plan.md +9 -5
- package/dist/resources/extensions/gsd/templates/task-plan.md +10 -2
- package/dist/resources/extensions/gsd/tools/complete-milestone.js +15 -9
- package/dist/resources/extensions/gsd/tools/complete-slice.js +56 -10
- package/dist/resources/extensions/gsd/tools/exec-tool.js +87 -5
- package/dist/resources/extensions/gsd/tools/plan-milestone.js +7 -1
- package/dist/resources/extensions/gsd/tools/plan-slice.js +151 -15
- package/dist/resources/extensions/gsd/tools/validate-milestone.js +32 -1
- package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +185 -40
- package/dist/resources/extensions/gsd/unit-context-composer.js +2 -0
- package/dist/resources/extensions/gsd/unit-context-manifest.js +69 -17
- package/dist/resources/extensions/gsd/validation.js +23 -1
- package/dist/resources/extensions/gsd/verification-gate.js +142 -7
- package/dist/resources/extensions/gsd/verification-verdict.js +26 -0
- package/dist/resources/extensions/gsd/workflow-manifest.js +2 -0
- package/dist/resources/extensions/gsd/workflow-mcp.js +17 -1
- package/dist/resources/extensions/gsd/workflow-projections.js +6 -8
- package/dist/resources/extensions/gsd/worktree-lifecycle.js +86 -19
- package/dist/resources/extensions/gsd/worktree-manager.js +11 -2
- package/dist/resources/extensions/gsd/worktree-safety.js +43 -4
- package/dist/resources/extensions/gsd/worktree-state-projection.js +31 -0
- package/dist/resources/extensions/gsd/worktree-telemetry.js +32 -0
- package/dist/resources/extensions/shared/html-shell.js +388 -0
- package/dist/resources/extensions/shared/interview-ui.js +6 -4
- package/dist/resources/extensions/shared/next-action-ui.js +13 -5
- 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/ttsr/ttsr-manager.js +3 -1
- 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/resources/skills/forensics/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 +7 -7
- 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 +5 -5
- package/dist/web/standalone/.next/required-server-files.json +1 -1
- 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/browse-directories/route.js +1 -1
- 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 +7 -7
- 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.65b24fac92188a6b.js +10 -0
- package/dist/web/standalone/.next/static/chunks/9441.ff70bb53f6835771.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/layout-8c10ec293ae0f1d5.js +1 -0
- package/dist/web/standalone/.next/static/chunks/{webpack-de742b64187e13fe.js → webpack-855d616060cb6e59.js} +1 -1
- package/dist/web/standalone/.next/static/css/746ee28c929d1880.css +1 -0
- package/dist/web/standalone/server.js +1 -1
- 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/daemon/package.json +2 -2
- 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/package.json +2 -2
- 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/package.json +1 -1
- package/packages/native/tsconfig.json +2 -1
- package/packages/native/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-agent-core/package.json +1 -1
- package/packages/pi-ai/dist/providers/google-gemini-cli.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/google-gemini-cli.js +5 -0
- package/packages/pi-ai/dist/providers/google-gemini-cli.js.map +1 -1
- package/packages/pi-ai/dist/providers/google-gemini-cli.test.d.ts +2 -0
- package/packages/pi-ai/dist/providers/google-gemini-cli.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/google-gemini-cli.test.js +41 -0
- package/packages/pi-ai/dist/providers/google-gemini-cli.test.js.map +1 -0
- 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/package.json +1 -1
- package/packages/pi-ai/src/providers/google-gemini-cli.test.ts +49 -0
- package/packages/pi-ai/src/providers/google-gemini-cli.ts +7 -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/dist/core/chat-controller-ordering.test.js +44 -3
- package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts +6 -1
- package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction/compaction.js +7 -2
- package/packages/pi-coding-agent/dist/core/compaction/compaction.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction/compaction.test.js +14 -1
- package/packages/pi-coding-agent/dist/core/compaction/compaction.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.js +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +8 -2
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/footer.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js +24 -6
- package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +82 -97
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js +7 -7
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +11 -0
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js +25 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +2 -0
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +24 -10
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/packages/pi-coding-agent/package.json +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/src/core/chat-controller-ordering.test.ts +53 -3
- package/packages/pi-coding-agent/src/core/compaction/compaction.test.ts +23 -1
- package/packages/pi-coding-agent/src/core/compaction/compaction.ts +7 -2
- package/packages/pi-coding-agent/src/core/sdk.ts +1 -1
- package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +17 -1
- package/packages/pi-coding-agent/src/modes/interactive/components/footer.ts +23 -7
- package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +1 -1
- package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +91 -102
- package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +15 -1
- package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.ts +9 -9
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-ordering.test.ts +30 -1
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +29 -10
- package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-tui/dist/__tests__/terminal.test.d.ts +2 -0
- package/packages/pi-tui/dist/__tests__/terminal.test.d.ts.map +1 -0
- package/packages/pi-tui/dist/__tests__/terminal.test.js +103 -0
- package/packages/pi-tui/dist/__tests__/terminal.test.js.map +1 -0
- package/packages/pi-tui/dist/__tests__/tui.test.js +45 -2
- package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
- package/packages/pi-tui/dist/terminal.d.ts +2 -0
- package/packages/pi-tui/dist/terminal.d.ts.map +1 -1
- package/packages/pi-tui/dist/terminal.js +12 -0
- package/packages/pi-tui/dist/terminal.js.map +1 -1
- package/packages/pi-tui/dist/tui.d.ts.map +1 -1
- package/packages/pi-tui/dist/tui.js +106 -27
- package/packages/pi-tui/dist/tui.js.map +1 -1
- package/packages/pi-tui/package.json +1 -1
- package/packages/pi-tui/src/__tests__/terminal.test.ts +121 -0
- package/packages/pi-tui/src/__tests__/tui.test.ts +59 -2
- package/packages/pi-tui/src/terminal.ts +11 -0
- package/packages/pi-tui/src/tui.ts +108 -27
- package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
- package/packages/rpc-client/package.json +1 -1
- package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
- package/pkg/package.json +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/stream-adapter.ts +52 -6
- package/src/resources/extensions/claude-code-cli/tests/partial-builder.test.ts +19 -2
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +49 -2
- package/src/resources/extensions/cmux/index.ts +6 -0
- package/src/resources/extensions/gsd/auto/contracts.ts +19 -6
- package/src/resources/extensions/gsd/auto/detect-stuck.ts +1 -0
- package/src/resources/extensions/gsd/auto/infra-errors.ts +9 -3
- package/src/resources/extensions/gsd/auto/loop.ts +123 -40
- package/src/resources/extensions/gsd/auto/orchestrator.ts +15 -4
- package/src/resources/extensions/gsd/auto/phases.ts +160 -60
- package/src/resources/extensions/gsd/auto/session.ts +16 -0
- package/src/resources/extensions/gsd/auto/types.ts +3 -0
- package/src/resources/extensions/gsd/auto/unit-runner-events.ts +6 -2
- package/src/resources/extensions/gsd/auto/workflow-kernel.ts +5 -1
- package/src/resources/extensions/gsd/auto/workflow-memory-pressure.ts +13 -0
- package/src/resources/extensions/gsd/auto-budget.ts +11 -0
- package/src/resources/extensions/gsd/auto-dashboard.ts +72 -1
- package/src/resources/extensions/gsd/auto-direct-dispatch.ts +1 -0
- package/src/resources/extensions/gsd/auto-dispatch.ts +164 -29
- package/src/resources/extensions/gsd/auto-model-selection.ts +2 -1
- package/src/resources/extensions/gsd/auto-post-unit.ts +369 -148
- package/src/resources/extensions/gsd/auto-prompts.ts +36 -13
- package/src/resources/extensions/gsd/auto-recovery.ts +86 -13
- package/src/resources/extensions/gsd/auto-start.ts +109 -14
- package/src/resources/extensions/gsd/auto-timers.ts +10 -3
- package/src/resources/extensions/gsd/auto-verification.ts +174 -42
- package/src/resources/extensions/gsd/auto-worktree.ts +202 -30
- package/src/resources/extensions/gsd/auto.ts +172 -81
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +66 -10
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +13 -10
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +13 -4
- package/src/resources/extensions/gsd/bootstrap/subagent-input.ts +19 -7
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +27 -10
- package/src/resources/extensions/gsd/clean-root-preflight.ts +174 -8
- package/src/resources/extensions/gsd/commands/catalog.ts +10 -1
- package/src/resources/extensions/gsd/commands/handlers/core.ts +41 -0
- package/src/resources/extensions/gsd/commands/handlers/ops.ts +21 -0
- package/src/resources/extensions/gsd/commands-bootstrap.ts +10 -0
- package/src/resources/extensions/gsd/commands-handlers.ts +2 -0
- package/src/resources/extensions/gsd/commands-mcp-status.ts +8 -0
- package/src/resources/extensions/gsd/commands-prefs-wizard.ts +11 -3
- package/src/resources/extensions/gsd/commands-verdict.ts +202 -0
- package/src/resources/extensions/gsd/crash-recovery.ts +55 -6
- package/src/resources/extensions/gsd/db/auto-workers.ts +37 -0
- package/src/resources/extensions/gsd/db/milestone-leases.ts +26 -0
- package/src/resources/extensions/gsd/db/unit-dispatches.ts +4 -3
- package/src/resources/extensions/gsd/db-base-schema.ts +2 -0
- package/src/resources/extensions/gsd/db-migration-steps.ts +5 -0
- package/src/resources/extensions/gsd/db-task-slice-rows.ts +4 -0
- package/src/resources/extensions/gsd/dispatch-guard.ts +60 -2
- package/src/resources/extensions/gsd/docs/preferences-reference.md +10 -0
- package/src/resources/extensions/gsd/doctor-git-checks.ts +45 -1
- package/src/resources/extensions/gsd/doctor-runtime-checks.ts +25 -13
- package/src/resources/extensions/gsd/doctor-types.ts +1 -0
- package/src/resources/extensions/gsd/doctor.ts +2 -27
- package/src/resources/extensions/gsd/export-html.ts +27 -427
- package/src/resources/extensions/gsd/forensics.ts +9 -3
- package/src/resources/extensions/gsd/git-service.ts +182 -16
- package/src/resources/extensions/gsd/gsd-db.ts +80 -31
- package/src/resources/extensions/gsd/guided-flow-queue.ts +4 -3
- package/src/resources/extensions/gsd/guided-flow.ts +142 -134
- package/src/resources/extensions/gsd/guided-unit-context.ts +30 -0
- package/src/resources/extensions/gsd/init-wizard.ts +17 -2
- package/src/resources/extensions/gsd/journal.ts +8 -1
- package/src/resources/extensions/gsd/markdown-renderer.ts +14 -11
- package/src/resources/extensions/gsd/mcp-filter.ts +80 -0
- package/src/resources/extensions/gsd/migrate/parsers.ts +139 -2
- 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 +63 -14
- package/src/resources/extensions/gsd/parallel-orchestrator.ts +3 -0
- package/src/resources/extensions/gsd/paths.ts +5 -0
- package/src/resources/extensions/gsd/pending-auto-start.ts +79 -0
- package/src/resources/extensions/gsd/planning-path-scope.ts +10 -2
- package/src/resources/extensions/gsd/post-execution-checks.ts +87 -12
- package/src/resources/extensions/gsd/pre-execution-checks.ts +67 -19
- package/src/resources/extensions/gsd/preferences-mcp.ts +27 -0
- package/src/resources/extensions/gsd/preferences-types.ts +35 -0
- package/src/resources/extensions/gsd/preferences-validation.ts +154 -0
- package/src/resources/extensions/gsd/preferences.ts +9 -0
- 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/forensics.md +3 -3
- 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/reactive-execute.md +1 -1
- 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/queue-reorder-ui.ts +31 -13
- package/src/resources/extensions/gsd/repo-identity.ts +45 -25
- package/src/resources/extensions/gsd/repository-registry.ts +77 -0
- package/src/resources/extensions/gsd/safety/evidence-collector.ts +2 -0
- package/src/resources/extensions/gsd/safety/evidence-cross-ref.ts +54 -19
- package/src/resources/extensions/gsd/session-lock.ts +15 -2
- package/src/resources/extensions/gsd/slice-parallel-conflict.ts +2 -2
- package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +75 -3
- package/src/resources/extensions/gsd/smart-entry-routing.ts +77 -0
- 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/state.ts +33 -7
- package/src/resources/extensions/gsd/status-guards.ts +16 -2
- package/src/resources/extensions/gsd/templates/PREFERENCES.md +1 -0
- package/src/resources/extensions/gsd/templates/plan.md +9 -5
- package/src/resources/extensions/gsd/templates/task-plan.md +10 -2
- package/src/resources/extensions/gsd/tests/artifact-retry-cap.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/auto-abort-pause-regression.test.ts +10 -1
- package/src/resources/extensions/gsd/tests/auto-dashboard.test.ts +71 -0
- 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 +775 -34
- package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +245 -28
- package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +151 -12
- package/src/resources/extensions/gsd/tests/auto-phases-lifecycle.test.ts +53 -2
- package/src/resources/extensions/gsd/tests/auto-post-unit-step-message.test.ts +18 -6
- package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +136 -13
- package/src/resources/extensions/gsd/tests/auto-remote-session-lock-cleanup.test.ts +64 -0
- package/src/resources/extensions/gsd/tests/auto-retry-mcp-churn-fixes.test.ts +12 -0
- package/src/resources/extensions/gsd/tests/auto-start-orphan-bootstrap.test.ts +1 -0
- package/src/resources/extensions/gsd/tests/auto-stop-notification.test.ts +20 -0
- package/src/resources/extensions/gsd/tests/auto-workers.test.ts +29 -0
- package/src/resources/extensions/gsd/tests/auto-worktree-registry.test.ts +69 -1
- package/src/resources/extensions/gsd/tests/autocomplete-regressions-1675.test.ts +21 -0
- package/src/resources/extensions/gsd/tests/brief-command.test.ts +89 -0
- package/src/resources/extensions/gsd/tests/checkout-branch-stash-guard.test.ts +87 -0
- package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +107 -2
- package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +32 -4
- package/src/resources/extensions/gsd/tests/closeout-git-deferral.test.ts +16 -0
- package/src/resources/extensions/gsd/tests/commands-verdict.test.ts +378 -0
- package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +55 -2
- package/src/resources/extensions/gsd/tests/complete-slice.test.ts +60 -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 +104 -2
- package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +2 -0
- package/src/resources/extensions/gsd/tests/db-authority-regression.test.ts +208 -0
- package/src/resources/extensions/gsd/tests/db-task-slice-rows.test.ts +1 -0
- package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +61 -2
- package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +111 -1
- package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +68 -1
- package/src/resources/extensions/gsd/tests/dispatch-missing-task-plans.test.ts +140 -1
- package/src/resources/extensions/gsd/tests/doctor-empty-worktree.test.ts +65 -0
- package/src/resources/extensions/gsd/tests/doctor-forensics-db-open-regression.test.ts +50 -0
- package/src/resources/extensions/gsd/tests/evidence-cross-ref.test.ts +97 -0
- package/src/resources/extensions/gsd/tests/exec-sandbox.test.ts +99 -1
- package/src/resources/extensions/gsd/tests/execution-entry-missing-context-4671.test.ts +15 -1
- package/src/resources/extensions/gsd/tests/export-html-enhancements.test.ts +8 -0
- package/src/resources/extensions/gsd/tests/gsd-db.test.ts +28 -0
- package/src/resources/extensions/gsd/tests/gsdroot-worktree-detection.test.ts +5 -2
- package/src/resources/extensions/gsd/tests/guided-discuss-project-prompt-rendering.test.ts +2 -0
- package/src/resources/extensions/gsd/tests/guided-dispatch-root.test.ts +106 -0
- package/src/resources/extensions/gsd/tests/guided-flow-session-isolation.test.ts +59 -11
- package/src/resources/extensions/gsd/tests/guided-flow.test.ts +21 -0
- package/src/resources/extensions/gsd/tests/guided-tool-contract.test.ts +65 -0
- package/src/resources/extensions/gsd/tests/headless-milestone-parity.test.ts +7 -7
- 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/init-prefs-routing.test.ts +22 -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 +226 -2
- package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +5 -21
- package/src/resources/extensions/gsd/tests/integration/state-machine-live-validation.test.ts +15 -0
- package/src/resources/extensions/gsd/tests/integration/state-machine-runtime-failures.test.ts +6 -1
- package/src/resources/extensions/gsd/tests/interrupted-session-auto.test.ts +40 -0
- package/src/resources/extensions/gsd/tests/journal-integration.test.ts +49 -3
- package/src/resources/extensions/gsd/tests/journal.test.ts +32 -0
- package/src/resources/extensions/gsd/tests/mcp-filter.test.ts +287 -0
- package/src/resources/extensions/gsd/tests/mcp-status.test.ts +11 -0
- package/src/resources/extensions/gsd/tests/merge-db-cycle.test.ts +179 -0
- package/src/resources/extensions/gsd/tests/merge-self-branch-guard.test.ts +21 -40
- package/src/resources/extensions/gsd/tests/migrate-validator-parsers.test.ts +59 -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 +80 -2
- package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +121 -1
- package/src/resources/extensions/gsd/tests/parallel-orchestrator-zombie-cleanup.test.ts +55 -0
- package/src/resources/extensions/gsd/tests/park-db-sync.test.ts +55 -1
- package/src/resources/extensions/gsd/tests/pending-autostart-scope.test.ts +29 -5
- package/src/resources/extensions/gsd/tests/pipeline-variant-dispatch.test.ts +2 -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 +343 -3
- package/src/resources/extensions/gsd/tests/plan-task.test.ts +18 -1
- 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 +105 -3
- package/src/resources/extensions/gsd/tests/post-unit-git-failure.test.ts +8 -1
- package/src/resources/extensions/gsd/tests/post-unit-state-rebuild.test.ts +84 -0
- package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +147 -0
- package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +54 -0
- package/src/resources/extensions/gsd/tests/preferences-mcp.test.ts +128 -0
- package/src/resources/extensions/gsd/tests/preferences.test.ts +75 -0
- package/src/resources/extensions/gsd/tests/prefs-wizard-coverage.test.ts +79 -0
- package/src/resources/extensions/gsd/tests/progressive-planning.test.ts +42 -1
- package/src/resources/extensions/gsd/tests/prompt-loader.test.ts +23 -0
- package/src/resources/extensions/gsd/tests/provider-errors.test.ts +37 -1
- package/src/resources/extensions/gsd/tests/quality-gates.test.ts +6 -0
- package/src/resources/extensions/gsd/tests/queue-reorder-ui.test.ts +54 -0
- package/src/resources/extensions/gsd/tests/remediation-completion-guard.test.ts +89 -2
- package/src/resources/extensions/gsd/tests/repo-identity-worktree.test.ts +28 -1
- package/src/resources/extensions/gsd/tests/repository-registry.test.ts +52 -0
- package/src/resources/extensions/gsd/tests/run-uat-replay-cap.test.ts +2 -3
- package/src/resources/extensions/gsd/tests/session-lock-regression.test.ts +35 -0
- package/src/resources/extensions/gsd/tests/session-switch-abort-misclassification.test.ts +59 -1
- package/src/resources/extensions/gsd/tests/slice-parallel-conflict.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +112 -1
- package/src/resources/extensions/gsd/tests/smart-entry-routing.test.ts +113 -0
- package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +94 -2
- 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/status-guards.test.ts +17 -1
- package/src/resources/extensions/gsd/tests/stuck-state-via-db.test.ts +64 -1
- package/src/resources/extensions/gsd/tests/subagent-model-dispatch.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/summary-render-parity.test.ts +7 -3
- package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +1 -0
- package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +131 -9
- package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +111 -1
- package/src/resources/extensions/gsd/tests/validate-milestone-stuck-guard.test.ts +29 -2
- package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +68 -0
- package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +42 -1
- package/src/resources/extensions/gsd/tests/verification-gate.test.ts +188 -1
- package/src/resources/extensions/gsd/tests/verification-verdict.test.ts +78 -0
- package/src/resources/extensions/gsd/tests/workflow-kernel.test.ts +7 -0
- package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +19 -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 +153 -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 +73 -2
- package/src/resources/extensions/gsd/tests/worktree-preferences-sync.test.ts +25 -0
- package/src/resources/extensions/gsd/tests/worktree-safety.test.ts +90 -0
- package/src/resources/extensions/gsd/tests/worktree-sync-milestones.test.ts +95 -0
- package/src/resources/extensions/gsd/tests/worktree-telemetry.test.ts +16 -0
- package/src/resources/extensions/gsd/tests/worktree-write-gate.test.ts +27 -0
- package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +59 -0
- package/src/resources/extensions/gsd/tools/complete-milestone.ts +18 -10
- package/src/resources/extensions/gsd/tools/complete-slice.ts +57 -10
- package/src/resources/extensions/gsd/tools/exec-tool.ts +98 -5
- package/src/resources/extensions/gsd/tools/plan-milestone.ts +5 -1
- package/src/resources/extensions/gsd/tools/plan-slice.ts +172 -12
- package/src/resources/extensions/gsd/tools/validate-milestone.ts +31 -0
- package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +166 -17
- package/src/resources/extensions/gsd/types.ts +1 -1
- package/src/resources/extensions/gsd/unit-context-composer.ts +3 -0
- package/src/resources/extensions/gsd/unit-context-manifest.ts +86 -19
- package/src/resources/extensions/gsd/validation.ts +23 -1
- package/src/resources/extensions/gsd/verification-gate.ts +170 -6
- package/src/resources/extensions/gsd/verification-verdict.ts +47 -0
- package/src/resources/extensions/gsd/workflow-manifest.ts +2 -0
- package/src/resources/extensions/gsd/workflow-mcp.ts +18 -1
- package/src/resources/extensions/gsd/workflow-projections.ts +6 -8
- package/src/resources/extensions/gsd/worktree-lifecycle.ts +98 -20
- package/src/resources/extensions/gsd/worktree-manager.ts +14 -2
- package/src/resources/extensions/gsd/worktree-safety.ts +57 -10
- package/src/resources/extensions/gsd/worktree-state-projection.ts +43 -0
- package/src/resources/extensions/gsd/worktree-telemetry.ts +39 -0
- package/src/resources/extensions/shared/html-shell.ts +412 -0
- package/src/resources/extensions/shared/interview-ui.ts +6 -4
- package/src/resources/extensions/shared/next-action-ui.ts +11 -5
- package/src/resources/extensions/shared/tests/interview-notes-loop.test.ts +15 -0
- package/src/resources/extensions/shared/tests/next-action-ui-hasui.test.ts +32 -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/ttsr/ttsr-manager.ts +5 -1
- 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/src/resources/skills/forensics/SKILL.md +1 -1
- 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/8359.e059d86b255fce1c.js +0 -10
- package/dist/web/standalone/.next/static/chunks/9441.1081da1125d1764f.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/{YEvjuT-fsFfYQhDSWtueS → Z8H5evS-hDo0qdP22XJPA}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{YEvjuT-fsFfYQhDSWtueS → Z8H5evS-hDo0qdP22XJPA}/_ssgManifest.js +0 -0
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
import { describe, it } from "node:test";
|
|
4
4
|
import assert from "node:assert/strict";
|
|
5
5
|
import { execFileSync, spawn } from "node:child_process";
|
|
6
|
-
import { existsSync, mkdirSync, mkdtempSync, readFileSync, rmSync, writeFileSync } from "node:fs";
|
|
6
|
+
import { existsSync, lstatSync, mkdirSync, mkdtempSync, readFileSync, rmSync, writeFileSync } from "node:fs";
|
|
7
7
|
import { join } from "node:path";
|
|
8
8
|
import { tmpdir } from "node:os";
|
|
9
9
|
|
|
@@ -12,8 +12,13 @@ import { validatePreferences } from "../preferences-validation.ts";
|
|
|
12
12
|
import {
|
|
13
13
|
_buildSliceWorkerEnvForTest,
|
|
14
14
|
_resolveSliceParallelMaxWorkersForTest,
|
|
15
|
+
getSliceOrchestratorState,
|
|
16
|
+
isSliceParallelActive,
|
|
15
17
|
restoreSliceState,
|
|
18
|
+
resetSliceOrchestrator,
|
|
16
19
|
SLICE_WORKER_AUTO_ARGS,
|
|
20
|
+
startSliceParallel,
|
|
21
|
+
stopSliceParallel,
|
|
17
22
|
} from "../slice-parallel-orchestrator.ts";
|
|
18
23
|
|
|
19
24
|
function readLinuxProcessStartFingerprint(pid: number): string | null {
|
|
@@ -50,6 +55,23 @@ function makeTempProject(): string {
|
|
|
50
55
|
return basePath;
|
|
51
56
|
}
|
|
52
57
|
|
|
58
|
+
function runGit(basePath: string, args: string[]): string {
|
|
59
|
+
return execFileSync("git", args, {
|
|
60
|
+
cwd: basePath,
|
|
61
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
62
|
+
encoding: "utf-8",
|
|
63
|
+
}).trim();
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
function initGitProject(basePath: string): void {
|
|
67
|
+
runGit(basePath, ["init", "--initial-branch=main"]);
|
|
68
|
+
runGit(basePath, ["config", "user.email", "test@example.com"]);
|
|
69
|
+
runGit(basePath, ["config", "user.name", "Test User"]);
|
|
70
|
+
writeFileSync(join(basePath, "README.md"), "initial\n", "utf-8");
|
|
71
|
+
runGit(basePath, ["add", "README.md"]);
|
|
72
|
+
runGit(basePath, ["commit", "-m", "initial"]);
|
|
73
|
+
}
|
|
74
|
+
|
|
53
75
|
function writeSliceOrchestratorState(
|
|
54
76
|
basePath: string,
|
|
55
77
|
worker: {
|
|
@@ -111,6 +133,56 @@ describe("slice worker launch contract", () => {
|
|
|
111
133
|
});
|
|
112
134
|
});
|
|
113
135
|
|
|
136
|
+
describe("slice-parallel stale worktree handling", () => {
|
|
137
|
+
it("replaces a stale slice worktree before spawning the worker", async () => {
|
|
138
|
+
const basePath = makeTempProject();
|
|
139
|
+
const oldGsdBinPath = process.env.GSD_BIN_PATH;
|
|
140
|
+
try {
|
|
141
|
+
initGitProject(basePath);
|
|
142
|
+
const workerBin = join(basePath, "fake-worker.js");
|
|
143
|
+
writeFileSync(
|
|
144
|
+
workerBin,
|
|
145
|
+
[
|
|
146
|
+
"process.on('SIGTERM', () => process.exit(0));",
|
|
147
|
+
"setInterval(() => {}, 1000);",
|
|
148
|
+
].join("\n"),
|
|
149
|
+
"utf-8",
|
|
150
|
+
);
|
|
151
|
+
process.env.GSD_BIN_PATH = workerBin;
|
|
152
|
+
|
|
153
|
+
const staleWorktree = join(basePath, ".gsd", "worktrees", "M900-S01");
|
|
154
|
+
mkdirSync(staleWorktree, { recursive: true });
|
|
155
|
+
writeFileSync(join(staleWorktree, ".git"), "gitdir: /tmp/not-this-repo\n", "utf-8");
|
|
156
|
+
writeFileSync(join(staleWorktree, "stale-marker"), "stale\n", "utf-8");
|
|
157
|
+
|
|
158
|
+
const result = await startSliceParallel(basePath, "M900", [{ id: "S01" }], { maxWorkers: 1 });
|
|
159
|
+
|
|
160
|
+
assert.deepEqual(result, { started: ["S01"], errors: [] });
|
|
161
|
+
assert.equal(existsSync(join(staleWorktree, "stale-marker")), false);
|
|
162
|
+
assert.equal(lstatSync(join(staleWorktree, ".git")).isFile(), true);
|
|
163
|
+
assert.match(readFileSync(join(staleWorktree, ".git"), "utf-8"), /^gitdir: /);
|
|
164
|
+
assert.equal(getSliceOrchestratorState()?.active, true);
|
|
165
|
+
} finally {
|
|
166
|
+
const child = getSliceOrchestratorState()?.workers.get("S01")?.process;
|
|
167
|
+
if (child && child.exitCode === null) {
|
|
168
|
+
await new Promise<void>((resolve) => {
|
|
169
|
+
const timeout = setTimeout(resolve, 500);
|
|
170
|
+
child.once("exit", () => {
|
|
171
|
+
clearTimeout(timeout);
|
|
172
|
+
resolve();
|
|
173
|
+
});
|
|
174
|
+
child.kill("SIGTERM");
|
|
175
|
+
});
|
|
176
|
+
}
|
|
177
|
+
stopSliceParallel();
|
|
178
|
+
resetSliceOrchestrator();
|
|
179
|
+
if (oldGsdBinPath === undefined) delete process.env.GSD_BIN_PATH;
|
|
180
|
+
else process.env.GSD_BIN_PATH = oldGsdBinPath;
|
|
181
|
+
rmSync(basePath, { recursive: true, force: true });
|
|
182
|
+
}
|
|
183
|
+
});
|
|
184
|
+
});
|
|
185
|
+
|
|
114
186
|
describe("slice-parallel-orchestrator recovery identity", () => {
|
|
115
187
|
it("rejects a live PID when the process start fingerprint does not match", () => {
|
|
116
188
|
const basePath = makeTempProject();
|
|
@@ -165,6 +237,45 @@ describe("slice-parallel-orchestrator recovery identity", () => {
|
|
|
165
237
|
rmSync(basePath, { recursive: true, force: true });
|
|
166
238
|
}
|
|
167
239
|
});
|
|
240
|
+
|
|
241
|
+
it("treats persisted non-running workers as inactive and clears stale state file", () => {
|
|
242
|
+
const basePath = makeTempProject();
|
|
243
|
+
try {
|
|
244
|
+
writeFileSync(
|
|
245
|
+
join(basePath, ".gsd", "slice-orchestrator.json"),
|
|
246
|
+
JSON.stringify({
|
|
247
|
+
active: true,
|
|
248
|
+
workers: [{
|
|
249
|
+
milestoneId: "M900",
|
|
250
|
+
sliceId: "S01",
|
|
251
|
+
pid: 0,
|
|
252
|
+
workerToken: "done-worker",
|
|
253
|
+
processStartFingerprint: null,
|
|
254
|
+
worktreePath: join(basePath, ".gsd", "worktrees", "M900-S01"),
|
|
255
|
+
startedAt: Date.now(),
|
|
256
|
+
state: "stopped",
|
|
257
|
+
completedUnits: 1,
|
|
258
|
+
cost: 0,
|
|
259
|
+
}],
|
|
260
|
+
totalCost: 0,
|
|
261
|
+
maxWorkers: 1,
|
|
262
|
+
startedAt: Date.now(),
|
|
263
|
+
basePath,
|
|
264
|
+
}),
|
|
265
|
+
"utf-8",
|
|
266
|
+
);
|
|
267
|
+
|
|
268
|
+
assert.equal(isSliceParallelActive(basePath), false);
|
|
269
|
+
assert.equal(
|
|
270
|
+
existsSync(join(basePath, ".gsd", "slice-orchestrator.json")),
|
|
271
|
+
false,
|
|
272
|
+
"stale non-running persisted state should be removed",
|
|
273
|
+
);
|
|
274
|
+
} finally {
|
|
275
|
+
resetSliceOrchestrator();
|
|
276
|
+
rmSync(basePath, { recursive: true, force: true });
|
|
277
|
+
}
|
|
278
|
+
});
|
|
168
279
|
});
|
|
169
280
|
|
|
170
281
|
describe("slice_parallel preference and state gating", () => {
|
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
// GSD-2 — Smart entry routing behavior tests.
|
|
2
|
+
// Verifies guided wizard choices resolve to the correct execution path.
|
|
3
|
+
|
|
4
|
+
import test from "node:test";
|
|
5
|
+
import assert from "node:assert/strict";
|
|
6
|
+
|
|
7
|
+
import {
|
|
8
|
+
resolveActiveTaskChoiceRoute,
|
|
9
|
+
resolveGuidedExecuteLaunchMode,
|
|
10
|
+
type ActiveTaskChoice,
|
|
11
|
+
type ActiveTaskRoute,
|
|
12
|
+
type SmartEntryIsolationMode,
|
|
13
|
+
} from "../smart-entry-routing.ts";
|
|
14
|
+
|
|
15
|
+
test("guided execute route enters auto step bootstrap only for worktree isolation", () => {
|
|
16
|
+
const cases: Array<{
|
|
17
|
+
isolationMode: SmartEntryIsolationMode;
|
|
18
|
+
expectedRoute: ActiveTaskRoute;
|
|
19
|
+
}> = [
|
|
20
|
+
{
|
|
21
|
+
isolationMode: "worktree",
|
|
22
|
+
expectedRoute: {
|
|
23
|
+
kind: "auto-bootstrap",
|
|
24
|
+
verboseMode: false,
|
|
25
|
+
options: {
|
|
26
|
+
step: true,
|
|
27
|
+
milestoneLock: "M001",
|
|
28
|
+
},
|
|
29
|
+
},
|
|
30
|
+
},
|
|
31
|
+
{
|
|
32
|
+
isolationMode: "none",
|
|
33
|
+
expectedRoute: {
|
|
34
|
+
kind: "guided-dispatch",
|
|
35
|
+
unitType: "execute-task",
|
|
36
|
+
},
|
|
37
|
+
},
|
|
38
|
+
{
|
|
39
|
+
isolationMode: "branch",
|
|
40
|
+
expectedRoute: {
|
|
41
|
+
kind: "guided-dispatch",
|
|
42
|
+
unitType: "execute-task",
|
|
43
|
+
},
|
|
44
|
+
},
|
|
45
|
+
];
|
|
46
|
+
|
|
47
|
+
for (const testCase of cases) {
|
|
48
|
+
assert.deepEqual(
|
|
49
|
+
resolveActiveTaskChoiceRoute({
|
|
50
|
+
choice: "execute",
|
|
51
|
+
isolationMode: testCase.isolationMode,
|
|
52
|
+
milestoneId: "M001",
|
|
53
|
+
}),
|
|
54
|
+
testCase.expectedRoute,
|
|
55
|
+
);
|
|
56
|
+
}
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
test("active task smart entry choices resolve to explicit routes", () => {
|
|
60
|
+
const cases: Array<{
|
|
61
|
+
choice: ActiveTaskChoice;
|
|
62
|
+
expectedRoute: ActiveTaskRoute;
|
|
63
|
+
}> = [
|
|
64
|
+
{
|
|
65
|
+
choice: "auto",
|
|
66
|
+
expectedRoute: {
|
|
67
|
+
kind: "auto-bootstrap",
|
|
68
|
+
verboseMode: false,
|
|
69
|
+
},
|
|
70
|
+
},
|
|
71
|
+
{
|
|
72
|
+
choice: "status",
|
|
73
|
+
expectedRoute: {
|
|
74
|
+
kind: "status",
|
|
75
|
+
},
|
|
76
|
+
},
|
|
77
|
+
{
|
|
78
|
+
choice: "milestone_actions",
|
|
79
|
+
expectedRoute: {
|
|
80
|
+
kind: "milestone-actions",
|
|
81
|
+
},
|
|
82
|
+
},
|
|
83
|
+
];
|
|
84
|
+
|
|
85
|
+
for (const testCase of cases) {
|
|
86
|
+
assert.deepEqual(
|
|
87
|
+
resolveActiveTaskChoiceRoute({
|
|
88
|
+
choice: testCase.choice,
|
|
89
|
+
isolationMode: "worktree",
|
|
90
|
+
milestoneId: "M001",
|
|
91
|
+
}),
|
|
92
|
+
testCase.expectedRoute,
|
|
93
|
+
);
|
|
94
|
+
}
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
test("active task route rejects invalid choices from untyped callers", () => {
|
|
98
|
+
assert.throws(
|
|
99
|
+
() =>
|
|
100
|
+
resolveActiveTaskChoiceRoute({
|
|
101
|
+
choice: "not_yet" as ActiveTaskChoice,
|
|
102
|
+
isolationMode: "worktree",
|
|
103
|
+
milestoneId: "M001",
|
|
104
|
+
}),
|
|
105
|
+
/Invalid ActiveTaskChoice: not_yet/,
|
|
106
|
+
);
|
|
107
|
+
});
|
|
108
|
+
|
|
109
|
+
test("guided execute launch mode remains a small compatibility helper", () => {
|
|
110
|
+
assert.equal(resolveGuidedExecuteLaunchMode("worktree"), "auto-step");
|
|
111
|
+
assert.equal(resolveGuidedExecuteLaunchMode("none"), "guided-dispatch");
|
|
112
|
+
assert.equal(resolveGuidedExecuteLaunchMode("branch"), "guided-dispatch");
|
|
113
|
+
});
|
|
@@ -4,7 +4,10 @@ import { readFileSync } from "node:fs";
|
|
|
4
4
|
import { resolve } from "node:path";
|
|
5
5
|
|
|
6
6
|
import { _withDetachedAutoKeepaliveForTest } from "../auto.ts";
|
|
7
|
-
import {
|
|
7
|
+
import {
|
|
8
|
+
_scheduleAutoStartAfterIdleForTest,
|
|
9
|
+
resolveGuidedExecuteLaunchMode,
|
|
10
|
+
} from "../guided-flow.ts";
|
|
8
11
|
|
|
9
12
|
const gsdDir = resolve(import.meta.dirname, "..");
|
|
10
13
|
|
|
@@ -64,6 +67,24 @@ test("bare /gsd stays in the foreground smart-entry flow (#5125 regression)", ()
|
|
|
64
67
|
);
|
|
65
68
|
});
|
|
66
69
|
|
|
70
|
+
test("guided execute uses auto step bootstrap when worktree isolation is enabled", () => {
|
|
71
|
+
assert.equal(
|
|
72
|
+
resolveGuidedExecuteLaunchMode("worktree"),
|
|
73
|
+
"auto-step",
|
|
74
|
+
"guided execute must enter auto bootstrap so the milestone worktree is created before execution",
|
|
75
|
+
);
|
|
76
|
+
assert.equal(
|
|
77
|
+
resolveGuidedExecuteLaunchMode("none"),
|
|
78
|
+
"guided-dispatch",
|
|
79
|
+
"non-isolated projects can keep the foreground guided dispatch path",
|
|
80
|
+
);
|
|
81
|
+
assert.equal(
|
|
82
|
+
resolveGuidedExecuteLaunchMode("branch"),
|
|
83
|
+
"guided-dispatch",
|
|
84
|
+
"this regression fix is scoped to worktree isolation",
|
|
85
|
+
);
|
|
86
|
+
});
|
|
87
|
+
|
|
67
88
|
test("auto bootstrap validates blocked directories before touching .gsd migration state", () => {
|
|
68
89
|
const autoSrc = readGsdFile("auto.ts");
|
|
69
90
|
const autoStartSrc = readGsdFile("auto-start.ts");
|
|
@@ -84,6 +105,8 @@ test("auto bootstrap validates blocked directories before touching .gsd migratio
|
|
|
84
105
|
const bootstrapIdx = autoStartSrc.indexOf("export async function bootstrapAutoSession(");
|
|
85
106
|
const bootstrapBody = autoStartSrc.slice(bootstrapIdx);
|
|
86
107
|
const bootstrapValidationIdx = bootstrapBody.indexOf("validateDirectory(base)");
|
|
108
|
+
const staleCrashReadIdx = bootstrapBody.indexOf("const startupLock = readCrashLock(base)");
|
|
109
|
+
const staleCrashClearIdx = bootstrapBody.indexOf("clearLock(base);");
|
|
87
110
|
const lockIdx = bootstrapBody.indexOf("acquireSessionLock(base)");
|
|
88
111
|
const bootstrapMigrationIdx = bootstrapBody.indexOf("migrateToExternalState(base)");
|
|
89
112
|
|
|
@@ -91,27 +114,63 @@ test("auto bootstrap validates blocked directories before touching .gsd migratio
|
|
|
91
114
|
assert.ok(bootstrapValidationIdx > -1, "bootstrapAutoSession should validate the base directory");
|
|
92
115
|
assert.ok(lockIdx > -1, "bootstrapAutoSession should acquire a session lock for safe projects");
|
|
93
116
|
assert.ok(bootstrapMigrationIdx > -1, "bootstrapAutoSession should still migrate safe projects");
|
|
117
|
+
assert.ok(staleCrashReadIdx > -1, "bootstrapAutoSession should probe stale crash lock state before lock acquisition");
|
|
118
|
+
assert.ok(staleCrashClearIdx > -1, "bootstrapAutoSession should clear stale crash lock state when detected");
|
|
94
119
|
assert.ok(
|
|
95
120
|
bootstrapValidationIdx < lockIdx && bootstrapValidationIdx < bootstrapMigrationIdx,
|
|
96
121
|
"fresh bootstrap must reject blocked directories before locking or migrating .gsd state",
|
|
97
122
|
);
|
|
123
|
+
assert.ok(
|
|
124
|
+
staleCrashReadIdx < lockIdx && staleCrashClearIdx < lockIdx,
|
|
125
|
+
"fresh bootstrap must auto-clear stale crash lock state before session lock acquisition",
|
|
126
|
+
);
|
|
98
127
|
});
|
|
99
128
|
|
|
100
129
|
test("fresh start registers the auto worker before bootstrap enters worktree flow (#5405)", () => {
|
|
101
130
|
const autoSrc = readGsdFile("auto.ts");
|
|
131
|
+
const autoStartSrc = readGsdFile("auto-start.ts");
|
|
102
132
|
const startAutoIdx = autoSrc.indexOf("export async function startAuto(");
|
|
103
133
|
const startAutoBody = autoSrc.slice(startAutoIdx);
|
|
134
|
+
const bootstrapIdx = autoStartSrc.indexOf("export async function bootstrapAutoSession(");
|
|
135
|
+
const bootstrapBody = autoStartSrc.slice(bootstrapIdx);
|
|
104
136
|
|
|
105
|
-
const preBootstrapRegisterIdx = startAutoBody.indexOf("registerAutoWorkerForSession(s, base);");
|
|
106
137
|
const bootstrapCallIdx = startAutoBody.indexOf("const ready = await bootstrapAutoSession(");
|
|
138
|
+
const preBootstrapBody = startAutoBody.slice(0, bootstrapCallIdx);
|
|
139
|
+
const preBootstrapRegisterIdx = preBootstrapBody.lastIndexOf("registerAutoWorkerForSession(s, base);");
|
|
140
|
+
const resumeSectionIdx = startAutoBody.indexOf("if (s.paused) {");
|
|
141
|
+
const freshStartSectionIdx = startAutoBody.indexOf("// ── Fresh start path — delegated to auto-start.ts ──");
|
|
142
|
+
const resumeBody = startAutoBody.slice(resumeSectionIdx, freshStartSectionIdx);
|
|
143
|
+
const resumeDbOpenIdx = resumeBody.indexOf("await openProjectDbIfPresent(base);");
|
|
144
|
+
const resumeRegisterIdx = resumeBody.indexOf("registerAutoWorkerForSession(s, base);");
|
|
145
|
+
const resumeEnterMilestoneIdx = resumeBody.indexOf("buildLifecycle().enterMilestone");
|
|
146
|
+
const dbOpenIdx = bootstrapBody.indexOf("await openProjectDbIfPresent(base);");
|
|
147
|
+
const bootstrapRegisterIdx = bootstrapBody.indexOf("registerAutoWorkerForSession(base);");
|
|
148
|
+
const enterMilestoneIdx = bootstrapBody.indexOf("buildLifecycle().enterMilestone");
|
|
107
149
|
|
|
108
150
|
assert.ok(startAutoIdx > -1, "startAuto should exist");
|
|
109
151
|
assert.ok(preBootstrapRegisterIdx > -1, "startAuto should register worker before bootstrap");
|
|
110
152
|
assert.ok(bootstrapCallIdx > -1, "startAuto should call bootstrapAutoSession");
|
|
153
|
+
assert.ok(resumeSectionIdx > -1, "startAuto should have resume milestone entry flow");
|
|
154
|
+
assert.ok(freshStartSectionIdx > resumeSectionIdx, "resume assertions should be scoped before fresh start");
|
|
155
|
+
assert.ok(resumeDbOpenIdx > -1, "resume should open DB before milestone entry");
|
|
156
|
+
assert.ok(resumeRegisterIdx > -1, "resume should register worker before milestone entry");
|
|
157
|
+
assert.ok(resumeEnterMilestoneIdx > -1, "resume should enter milestones through lifecycle");
|
|
158
|
+
assert.ok(bootstrapIdx > -1, "bootstrapAutoSession should exist");
|
|
159
|
+
assert.ok(dbOpenIdx > -1, "bootstrap should open the project DB");
|
|
160
|
+
assert.ok(bootstrapRegisterIdx > -1, "bootstrap should register worker after DB open");
|
|
161
|
+
assert.ok(enterMilestoneIdx > -1, "bootstrap should enter milestones through lifecycle");
|
|
111
162
|
assert.ok(
|
|
112
163
|
preBootstrapRegisterIdx < bootstrapCallIdx,
|
|
113
164
|
"worker registration must happen before bootstrap so enterMilestone can claim milestone leases on first entry",
|
|
114
165
|
);
|
|
166
|
+
assert.ok(
|
|
167
|
+
dbOpenIdx < bootstrapRegisterIdx && bootstrapRegisterIdx < enterMilestoneIdx,
|
|
168
|
+
"bootstrap must open DB and register worker before first enterMilestone",
|
|
169
|
+
);
|
|
170
|
+
assert.ok(
|
|
171
|
+
resumeDbOpenIdx < resumeRegisterIdx && resumeRegisterIdx < resumeEnterMilestoneIdx,
|
|
172
|
+
"resume must open DB and register worker before first enterMilestone",
|
|
173
|
+
);
|
|
115
174
|
});
|
|
116
175
|
|
|
117
176
|
test("startAutoDetached reports failures asynchronously (#3733)", () => {
|
|
@@ -240,3 +299,36 @@ test("discussion auto-start waits for the current command context to become idle
|
|
|
240
299
|
assert.equal(launches[0][2], "/tmp/gsd-auto-start-idle-test");
|
|
241
300
|
assert.deepEqual(launches[0][4], { step: true });
|
|
242
301
|
});
|
|
302
|
+
|
|
303
|
+
test("resume path only hard-exits on blocked stop, not blocked pause (#6154)", () => {
|
|
304
|
+
const autoSrc = readGsdFile("auto.ts");
|
|
305
|
+
const startAutoIdx = autoSrc.indexOf("export async function startAuto(");
|
|
306
|
+
const startAutoBody = autoSrc.slice(startAutoIdx);
|
|
307
|
+
const resumeSectionIdx = startAutoBody.indexOf("if (s.paused) {");
|
|
308
|
+
const freshStartSectionIdx = startAutoBody.indexOf("// ── Fresh start path — delegated to auto-start.ts ──");
|
|
309
|
+
const resumeBody = startAutoBody.slice(resumeSectionIdx, freshStartSectionIdx);
|
|
310
|
+
|
|
311
|
+
assert.ok(startAutoIdx > -1, "startAuto should exist");
|
|
312
|
+
assert.ok(resumeSectionIdx > -1, "resume path should exist");
|
|
313
|
+
assert.ok(freshStartSectionIdx > resumeSectionIdx, "resume assertions should be scoped before fresh start");
|
|
314
|
+
assert.ok(
|
|
315
|
+
resumeBody.includes('if (resumeResult?.kind === "blocked" && resumeResult.action === "stop")'),
|
|
316
|
+
"resume path should only hard-stop blocked resume when action is stop",
|
|
317
|
+
);
|
|
318
|
+
assert.ok(
|
|
319
|
+
resumeBody.includes('if (resumeResult?.kind === "blocked")'),
|
|
320
|
+
"resume path should still notify blocked resume results",
|
|
321
|
+
);
|
|
322
|
+
});
|
|
323
|
+
|
|
324
|
+
test("prepareForUnit skips worktree safety when isolation is not worktree (#6154)", () => {
|
|
325
|
+
const autoSrc = readGsdFile("auto.ts");
|
|
326
|
+
const prepareForUnitIdx = autoSrc.indexOf("async prepareForUnit(unitType, unitId) {");
|
|
327
|
+
const prepareForUnitBody = autoSrc.slice(prepareForUnitIdx, autoSrc.indexOf("async syncAfterUnit() {}", prepareForUnitIdx));
|
|
328
|
+
|
|
329
|
+
assert.ok(prepareForUnitIdx > -1, "prepareForUnit should exist");
|
|
330
|
+
assert.ok(
|
|
331
|
+
prepareForUnitBody.includes('if (getIsolationMode(runtimeBasePath) !== "worktree")'),
|
|
332
|
+
"prepareForUnit should bypass worktree safety validation outside worktree isolation mode",
|
|
333
|
+
);
|
|
334
|
+
});
|
|
@@ -271,6 +271,11 @@ describe("#2945 Bug 3: mergeAndExit must teardown worktree after successful merg
|
|
|
271
271
|
// a real git fixture and verify the worktree directory is removed
|
|
272
272
|
// from disk after the merge.
|
|
273
273
|
const tmpBase = realpathSync(mkdtempSync(join(tmpdir(), "gsd-2945-bug3-")));
|
|
274
|
+
// ADR-016 phase 3 (#5693): Lifecycle.restoreToProjectRoot now chdirs to
|
|
275
|
+
// s.originalBasePath. Save cwd before the test so we can restore it
|
|
276
|
+
// before rmSync removes tmpBase — otherwise the next test in this file
|
|
277
|
+
// inherits a deleted cwd and process.cwd() throws ENOENT (uv_cwd).
|
|
278
|
+
const prevCwd = process.cwd();
|
|
274
279
|
try {
|
|
275
280
|
const git = (args: string[]): void => {
|
|
276
281
|
execFileSync("git", args, { cwd: tmpBase, stdio: "pipe" });
|
|
@@ -325,6 +330,7 @@ describe("#2945 Bug 3: mergeAndExit must teardown worktree after successful merg
|
|
|
325
330
|
`teardownAutoWorktree must be called after successful merge — worktree directory at ${wt} should be removed`,
|
|
326
331
|
);
|
|
327
332
|
} finally {
|
|
333
|
+
try { process.chdir(prevCwd); } catch { /* noop */ }
|
|
328
334
|
try { rmSync(tmpBase, { recursive: true, force: true }); } catch { /* noop */ }
|
|
329
335
|
}
|
|
330
336
|
});
|
|
@@ -544,7 +544,7 @@ test("ADR-017 (#5703): live worker lock is not cleared", async (t) => {
|
|
|
544
544
|
|
|
545
545
|
// ─── #5704: unregistered-milestone drift ────────────────────────────────────
|
|
546
546
|
|
|
547
|
-
test("ADR-017 (#5704): unregistered-milestone drift
|
|
547
|
+
test("ADR-017 (#5704): unregistered-milestone drift fails closed without importing markdown", async (t) => {
|
|
548
548
|
const base = mkdtempSync(join(tmpdir(), "gsd-adr017-projmd-"));
|
|
549
549
|
const milestoneDir = join(base, ".gsd", "milestones", "M042");
|
|
550
550
|
mkdirSync(milestoneDir, { recursive: true });
|
|
@@ -571,20 +571,22 @@ test("ADR-017 (#5704): unregistered-milestone drift detected and DB row inserted
|
|
|
571
571
|
// Pre-condition: filesystem has the milestone, DB does NOT.
|
|
572
572
|
assert.equal(getMilestone("M042"), null, "pre: DB has no row for M042");
|
|
573
573
|
|
|
574
|
-
|
|
575
|
-
|
|
576
|
-
|
|
577
|
-
|
|
578
|
-
|
|
579
|
-
|
|
580
|
-
|
|
581
|
-
|
|
582
|
-
|
|
574
|
+
await assert.rejects(
|
|
575
|
+
reconcileBeforeDispatch(base, {
|
|
576
|
+
invalidateStateCache: () => {},
|
|
577
|
+
deriveState: async () => makeState(),
|
|
578
|
+
}),
|
|
579
|
+
(err: unknown) => {
|
|
580
|
+
assert.ok(err instanceof ReconciliationFailedError);
|
|
581
|
+
assert.match(String(err.message), /unregistered-milestone/);
|
|
582
|
+
assert.equal(err.failures[0]?.drift.kind, "unregistered-milestone");
|
|
583
|
+
assert.match(String(err.failures[0]?.cause), /M042/);
|
|
584
|
+
assert.match(String(err.failures[0]?.cause), /markdown projection/);
|
|
585
|
+
assert.match(String(err.failures[0]?.cause), /recovery\/migration/);
|
|
586
|
+
return true;
|
|
587
|
+
},
|
|
583
588
|
);
|
|
584
|
-
assert.
|
|
585
|
-
if (milestoneRepaired?.kind === "unregistered-milestone") {
|
|
586
|
-
assert.equal(milestoneRepaired.milestoneId, "M042");
|
|
587
|
-
}
|
|
589
|
+
assert.equal(getMilestone("M042"), null, "post: DB still has no row for M042");
|
|
588
590
|
});
|
|
589
591
|
|
|
590
592
|
test("ADR-017 (#5704): registered milestone (DB row present) → no drift", async (t) => {
|
|
@@ -627,13 +629,14 @@ test("ADR-017 (#5704): registered milestone (DB row present) → no drift", asyn
|
|
|
627
629
|
|
|
628
630
|
// ─── #5705: roadmap-divergence drift ─────────────────────────────────────────
|
|
629
631
|
|
|
630
|
-
test("ADR-017 (#5705): roadmap-divergence
|
|
632
|
+
test("ADR-017 (#5705): roadmap-divergence re-renders projection without syncing depends into DB", async (t) => {
|
|
631
633
|
const base = mkdtempSync(join(tmpdir(), "gsd-adr017-roadmap-"));
|
|
632
634
|
const milestoneDir = join(base, ".gsd", "milestones", "M001");
|
|
635
|
+
const roadmapPath = join(milestoneDir, "M001-ROADMAP.md");
|
|
633
636
|
mkdirSync(milestoneDir, { recursive: true });
|
|
634
637
|
// ROADMAP.md declares S02 depends on [S01]
|
|
635
638
|
writeFileSync(
|
|
636
|
-
|
|
639
|
+
roadmapPath,
|
|
637
640
|
[
|
|
638
641
|
"# M001: Test",
|
|
639
642
|
"",
|
|
@@ -665,7 +668,12 @@ test("ADR-017 (#5705): roadmap-divergence drift detected and DB depends synced",
|
|
|
665
668
|
});
|
|
666
669
|
|
|
667
670
|
assert.equal(result.ok, true);
|
|
668
|
-
assert.deepEqual(getSlice("M001", "S02")?.depends, [
|
|
671
|
+
assert.deepEqual(getSlice("M001", "S02")?.depends, [], "post: DB depends remains authoritative");
|
|
672
|
+
assert.match(
|
|
673
|
+
readFileSync(roadmapPath, "utf-8"),
|
|
674
|
+
/- \[ \] \*\*S02: Feature\*\* `risk:medium` `depends:\[\]`/,
|
|
675
|
+
"post: ROADMAP projection is regenerated from DB depends",
|
|
676
|
+
);
|
|
669
677
|
const roadmapRepaired = result.repaired.find((d) => d.kind === "roadmap-divergence");
|
|
670
678
|
assert.ok(roadmapRepaired, "repaired list should include the roadmap-divergence drift");
|
|
671
679
|
if (roadmapRepaired?.kind === "roadmap-divergence") {
|
|
@@ -673,13 +681,14 @@ test("ADR-017 (#5705): roadmap-divergence drift detected and DB depends synced",
|
|
|
673
681
|
}
|
|
674
682
|
});
|
|
675
683
|
|
|
676
|
-
test("ADR-017 (#5705): ROADMAP
|
|
684
|
+
test("ADR-017 (#5705): ROADMAP-only slice is removed from projection and not inserted into DB", async (t) => {
|
|
677
685
|
const base = mkdtempSync(join(tmpdir(), "gsd-adr017-roadmap-newslice-"));
|
|
678
686
|
const milestoneDir = join(base, ".gsd", "milestones", "M001");
|
|
687
|
+
const roadmapPath = join(milestoneDir, "M001-ROADMAP.md");
|
|
679
688
|
mkdirSync(milestoneDir, { recursive: true });
|
|
680
689
|
// ROADMAP.md declares S01 and S02; DB will only have S01.
|
|
681
690
|
writeFileSync(
|
|
682
|
-
|
|
691
|
+
roadmapPath,
|
|
683
692
|
[
|
|
684
693
|
"# M001: Test",
|
|
685
694
|
"",
|
|
@@ -710,10 +719,10 @@ test("ADR-017 (#5705): ROADMAP declares slice missing from DB → slice inserted
|
|
|
710
719
|
});
|
|
711
720
|
|
|
712
721
|
assert.equal(result.ok, true);
|
|
713
|
-
|
|
714
|
-
|
|
715
|
-
assert.
|
|
716
|
-
assert.
|
|
722
|
+
assert.equal(getSlice("M001", "S02"), null, "post: S02 still has no DB row");
|
|
723
|
+
const rendered = readFileSync(roadmapPath, "utf-8");
|
|
724
|
+
assert.match(rendered, /- \[ \] \*\*S01: Foundation\*\*/);
|
|
725
|
+
assert.doesNotMatch(rendered, /S02: Feature/, "post: ROADMAP-only S02 removed from projection");
|
|
717
726
|
const roadmapRepaired = result.repaired.find((d) => d.kind === "roadmap-divergence");
|
|
718
727
|
assert.ok(roadmapRepaired, "repaired list should include the roadmap-divergence drift");
|
|
719
728
|
if (roadmapRepaired?.kind === "roadmap-divergence") {
|
|
@@ -721,6 +730,93 @@ test("ADR-017 (#5705): ROADMAP declares slice missing from DB → slice inserted
|
|
|
721
730
|
}
|
|
722
731
|
});
|
|
723
732
|
|
|
733
|
+
test("ADR-017 (#5705): ROADMAP sequence drift re-renders from DB order without mutating DB", async (t) => {
|
|
734
|
+
const base = mkdtempSync(join(tmpdir(), "gsd-adr017-roadmap-sequence-"));
|
|
735
|
+
const milestoneDir = join(base, ".gsd", "milestones", "M001");
|
|
736
|
+
const roadmapPath = join(milestoneDir, "M001-ROADMAP.md");
|
|
737
|
+
mkdirSync(milestoneDir, { recursive: true });
|
|
738
|
+
writeFileSync(
|
|
739
|
+
roadmapPath,
|
|
740
|
+
[
|
|
741
|
+
"# M001: Test",
|
|
742
|
+
"",
|
|
743
|
+
"**Vision:** Verify sequence drift",
|
|
744
|
+
"",
|
|
745
|
+
"## Slices",
|
|
746
|
+
"",
|
|
747
|
+
"- [ ] **S02: Feature** `risk:medium` `depends:[]`",
|
|
748
|
+
"- [ ] **S01: Foundation** `risk:medium` `depends:[]`",
|
|
749
|
+
"",
|
|
750
|
+
].join("\n"),
|
|
751
|
+
);
|
|
752
|
+
t.after(() => {
|
|
753
|
+
try { closeDatabase(); } catch { /* noop */ }
|
|
754
|
+
rmSync(base, { recursive: true, force: true });
|
|
755
|
+
});
|
|
756
|
+
|
|
757
|
+
openDatabase(join(base, ".gsd", "gsd.db"));
|
|
758
|
+
insertMilestone({ id: "M001", title: "Test", status: "active" });
|
|
759
|
+
insertSlice({ id: "S01", milestoneId: "M001", title: "Foundation", status: "pending", risk: "medium", depends: [], demo: "", sequence: 1 });
|
|
760
|
+
insertSlice({ id: "S02", milestoneId: "M001", title: "Feature", status: "pending", risk: "medium", depends: [], demo: "", sequence: 2 });
|
|
761
|
+
|
|
762
|
+
const result = await reconcileBeforeDispatch(base, {
|
|
763
|
+
invalidateStateCache: () => {},
|
|
764
|
+
deriveState: async () => makeState(),
|
|
765
|
+
});
|
|
766
|
+
|
|
767
|
+
assert.equal(result.ok, true);
|
|
768
|
+
assert.equal(getSlice("M001", "S01")?.sequence, 1, "post: S01 DB sequence remains authoritative");
|
|
769
|
+
assert.equal(getSlice("M001", "S02")?.sequence, 2, "post: S02 DB sequence remains authoritative");
|
|
770
|
+
const rendered = readFileSync(roadmapPath, "utf-8");
|
|
771
|
+
assert.ok(
|
|
772
|
+
rendered.indexOf("S01: Foundation") < rendered.indexOf("S02: Feature"),
|
|
773
|
+
"post: ROADMAP projection follows DB sequence",
|
|
774
|
+
);
|
|
775
|
+
assert.ok(result.repaired.some((d) => d.kind === "roadmap-divergence"));
|
|
776
|
+
});
|
|
777
|
+
|
|
778
|
+
test("ADR-017 (#5705): ROADMAP checkbox drift re-renders from DB status without mutating DB", async (t) => {
|
|
779
|
+
const base = mkdtempSync(join(tmpdir(), "gsd-adr017-roadmap-checkbox-"));
|
|
780
|
+
const milestoneDir = join(base, ".gsd", "milestones", "M001");
|
|
781
|
+
const roadmapPath = join(milestoneDir, "M001-ROADMAP.md");
|
|
782
|
+
mkdirSync(milestoneDir, { recursive: true });
|
|
783
|
+
writeFileSync(
|
|
784
|
+
roadmapPath,
|
|
785
|
+
[
|
|
786
|
+
"# M001: Test",
|
|
787
|
+
"",
|
|
788
|
+
"**Vision:** Verify checkbox drift",
|
|
789
|
+
"",
|
|
790
|
+
"## Slices",
|
|
791
|
+
"",
|
|
792
|
+
"- [x] **S01: Foundation** `risk:medium` `depends:[]`",
|
|
793
|
+
"",
|
|
794
|
+
].join("\n"),
|
|
795
|
+
);
|
|
796
|
+
t.after(() => {
|
|
797
|
+
try { closeDatabase(); } catch { /* noop */ }
|
|
798
|
+
rmSync(base, { recursive: true, force: true });
|
|
799
|
+
});
|
|
800
|
+
|
|
801
|
+
openDatabase(join(base, ".gsd", "gsd.db"));
|
|
802
|
+
insertMilestone({ id: "M001", title: "Test", status: "active" });
|
|
803
|
+
insertSlice({ id: "S01", milestoneId: "M001", title: "Foundation", status: "pending", risk: "medium", depends: [], demo: "", sequence: 1 });
|
|
804
|
+
|
|
805
|
+
const result = await reconcileBeforeDispatch(base, {
|
|
806
|
+
invalidateStateCache: () => {},
|
|
807
|
+
deriveState: async () => makeState(),
|
|
808
|
+
});
|
|
809
|
+
|
|
810
|
+
assert.equal(result.ok, true);
|
|
811
|
+
assert.equal(getSlice("M001", "S01")?.status, "pending", "post: DB status remains authoritative");
|
|
812
|
+
assert.match(
|
|
813
|
+
readFileSync(roadmapPath, "utf-8"),
|
|
814
|
+
/- \[ \] \*\*S01: Foundation\*\*/,
|
|
815
|
+
"post: ROADMAP checkbox reflects DB status",
|
|
816
|
+
);
|
|
817
|
+
assert.ok(result.repaired.some((d) => d.kind === "roadmap-divergence"));
|
|
818
|
+
});
|
|
819
|
+
|
|
724
820
|
test("ADR-017 (#5705): in-sync ROADMAP and DB → no roadmap-divergence drift", async (t) => {
|
|
725
821
|
const base = mkdtempSync(join(tmpdir(), "gsd-adr017-roadmap-clean-"));
|
|
726
822
|
const milestoneDir = join(base, ".gsd", "milestones", "M001");
|
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
import test from 'node:test';
|
|
4
4
|
import assert from 'node:assert/strict';
|
|
5
5
|
|
|
6
|
-
import { isClosedStatus } from '../status-guards.ts';
|
|
6
|
+
import { isClosedStatus, isFutureMilestoneStatus } from '../status-guards.ts';
|
|
7
7
|
|
|
8
8
|
test('isClosedStatus: "complete" returns true', () => {
|
|
9
9
|
assert.equal(isClosedStatus('complete'), true);
|
|
@@ -17,6 +17,10 @@ test('isClosedStatus: "skipped" returns true', () => {
|
|
|
17
17
|
assert.equal(isClosedStatus('skipped'), true);
|
|
18
18
|
});
|
|
19
19
|
|
|
20
|
+
test('isClosedStatus: "closed" returns true', () => {
|
|
21
|
+
assert.equal(isClosedStatus('closed'), true);
|
|
22
|
+
});
|
|
23
|
+
|
|
20
24
|
test('isClosedStatus: "pending" returns false', () => {
|
|
21
25
|
assert.equal(isClosedStatus('pending'), false);
|
|
22
26
|
});
|
|
@@ -32,3 +36,15 @@ test('isClosedStatus: "active" returns false', () => {
|
|
|
32
36
|
test('isClosedStatus: "" (empty string) returns false', () => {
|
|
33
37
|
assert.equal(isClosedStatus(''), false);
|
|
34
38
|
});
|
|
39
|
+
|
|
40
|
+
test('isFutureMilestoneStatus includes future milestone aliases', () => {
|
|
41
|
+
assert.equal(isFutureMilestoneStatus('pending'), true);
|
|
42
|
+
assert.equal(isFutureMilestoneStatus('queued'), true);
|
|
43
|
+
assert.equal(isFutureMilestoneStatus('planned'), true);
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
test('isFutureMilestoneStatus excludes active and closed milestones', () => {
|
|
47
|
+
assert.equal(isFutureMilestoneStatus('active'), false);
|
|
48
|
+
assert.equal(isFutureMilestoneStatus('complete'), false);
|
|
49
|
+
assert.equal(isFutureMilestoneStatus('parked'), false);
|
|
50
|
+
});
|