gsd-pi 2.71.0 → 2.72.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +57 -17
- package/dist/cli.js +29 -3
- package/dist/headless-events.d.ts +2 -0
- package/dist/headless-events.js +7 -0
- package/dist/headless.js +16 -3
- package/dist/mcp-server.js +40 -17
- package/dist/provider-migrations.d.ts +10 -0
- package/dist/provider-migrations.js +12 -0
- package/dist/resource-loader.js +139 -13
- package/dist/resources/GSD-WORKFLOW.md +1 -1
- package/dist/resources/agents/debugger.md +58 -0
- package/dist/resources/agents/doc-writer.md +43 -0
- package/dist/resources/agents/git-ops.md +56 -0
- package/dist/resources/agents/javascript-pro.md +46 -271
- package/dist/resources/agents/planner.md +55 -0
- package/dist/resources/agents/refactorer.md +47 -0
- package/dist/resources/agents/reviewer.md +48 -0
- package/dist/resources/agents/security.md +59 -0
- package/dist/resources/agents/tester.md +50 -0
- package/dist/resources/agents/typescript-pro.md +41 -235
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +113 -10
- package/dist/resources/extensions/gsd/auto/infra-errors.js +34 -0
- package/dist/resources/extensions/gsd/auto/loop.js +32 -1
- package/dist/resources/extensions/gsd/auto/phases.js +5 -1
- package/dist/resources/extensions/gsd/auto/session.js +11 -0
- package/dist/resources/extensions/gsd/auto-dashboard.js +22 -16
- package/dist/resources/extensions/gsd/auto-model-selection.js +10 -2
- package/dist/resources/extensions/gsd/auto-prompts.js +88 -33
- package/dist/resources/extensions/gsd/auto-start.js +34 -7
- package/dist/resources/extensions/gsd/auto-tool-tracking.js +1 -1
- package/dist/resources/extensions/gsd/auto-worktree.js +1 -1
- package/dist/resources/extensions/gsd/auto.js +56 -0
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +3 -3
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +2 -0
- package/dist/resources/extensions/gsd/bootstrap/register-shortcuts.js +63 -51
- package/dist/resources/extensions/gsd/bootstrap/system-context.js +6 -0
- package/dist/resources/extensions/gsd/commands/context.js +15 -6
- package/dist/resources/extensions/gsd/commands/dispatcher.js +12 -2
- package/dist/resources/extensions/gsd/commands/handlers/auto.js +10 -33
- package/dist/resources/extensions/gsd/commands/handlers/core.js +56 -11
- package/dist/resources/extensions/gsd/commands/handlers/notifications-handler.js +15 -6
- package/dist/resources/extensions/gsd/commands/handlers/workflow.js +4 -10
- package/dist/resources/extensions/gsd/dashboard-overlay.js +8 -3
- package/dist/resources/extensions/gsd/dispatch-guard.js +18 -1
- package/dist/resources/extensions/gsd/doctor-providers.js +23 -0
- package/dist/resources/extensions/gsd/error-classifier.js +5 -2
- package/dist/resources/extensions/gsd/forensics.js +19 -6
- package/dist/resources/extensions/gsd/gate-registry.js +208 -0
- package/dist/resources/extensions/gsd/gsd-db.js +41 -0
- package/dist/resources/extensions/gsd/guided-flow.js +5 -10
- package/dist/resources/extensions/gsd/metrics.js +1 -0
- package/dist/resources/extensions/gsd/milestone-actions.js +10 -4
- package/dist/resources/extensions/gsd/milestone-validation-gates.js +11 -12
- package/dist/resources/extensions/gsd/notification-overlay.js +42 -13
- package/dist/resources/extensions/gsd/notification-store.js +56 -5
- package/dist/resources/extensions/gsd/notification-widget.js +5 -13
- package/dist/resources/extensions/gsd/parallel-monitor-overlay.js +8 -3
- package/dist/resources/extensions/gsd/pre-execution-checks.js +35 -2
- package/dist/resources/extensions/gsd/prompt-validation.js +126 -0
- package/dist/resources/extensions/gsd/prompts/complete-slice.md +5 -3
- package/dist/resources/extensions/gsd/prompts/discuss.md +2 -0
- package/dist/resources/extensions/gsd/prompts/execute-task.md +22 -19
- package/dist/resources/extensions/gsd/prompts/guided-discuss-milestone.md +2 -0
- package/dist/resources/extensions/gsd/prompts/guided-discuss-slice.md +2 -0
- package/dist/resources/extensions/gsd/prompts/guided-resume-task.md +1 -1
- package/dist/resources/extensions/gsd/prompts/queue.md +3 -2
- package/dist/resources/extensions/gsd/prompts/system.md +1 -0
- package/dist/resources/extensions/gsd/prompts/validate-milestone.md +4 -1
- package/dist/resources/extensions/gsd/session-model-override.js +25 -0
- package/dist/resources/extensions/gsd/shortcut-defs.js +40 -0
- package/dist/resources/extensions/gsd/state.js +9 -2
- package/dist/resources/extensions/gsd/tools/complete-slice.js +52 -1
- package/dist/resources/extensions/gsd/tools/complete-task.js +51 -1
- package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +4 -1
- package/dist/resources/extensions/ollama/index.js +13 -5
- package/dist/resources/extensions/shared/gsd-phase-state.js +35 -0
- package/dist/resources/extensions/subagent/agents.js +8 -0
- package/dist/resources/extensions/subagent/index.js +17 -0
- package/dist/resources/skills/create-skill/SKILL.md +2 -0
- package/dist/startup-model-validation.d.ts +0 -1
- package/dist/startup-model-validation.js +6 -2
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +8 -8
- 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/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 +8 -8
- package/dist/web/standalone/.next/server/middleware-build-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/package.json +1 -1
- package/packages/mcp-server/dist/server.d.ts +12 -1
- package/packages/mcp-server/dist/server.d.ts.map +1 -1
- package/packages/mcp-server/dist/server.js +90 -42
- package/packages/mcp-server/dist/server.js.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +22 -12
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/src/server.ts +110 -38
- package/packages/mcp-server/src/workflow-tools.test.ts +110 -0
- package/packages/mcp-server/src/workflow-tools.ts +32 -12
- package/packages/pi-ai/dist/providers/amazon-bedrock.js +11 -2
- package/packages/pi-ai/dist/providers/amazon-bedrock.js.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic-auth.test.d.ts +2 -0
- package/packages/pi-ai/dist/providers/anthropic-auth.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/anthropic-auth.test.js +20 -0
- package/packages/pi-ai/dist/providers/anthropic-auth.test.js.map +1 -0
- package/packages/pi-ai/dist/providers/anthropic-shared.d.ts +4 -1
- package/packages/pi-ai/dist/providers/anthropic-shared.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic-shared.js +8 -3
- package/packages/pi-ai/dist/providers/anthropic-shared.js.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic-shared.test.js +44 -1
- package/packages/pi-ai/dist/providers/anthropic-shared.test.js.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic.d.ts +2 -1
- package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic.js +7 -4
- package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
- package/packages/pi-ai/dist/providers/openai-completions.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/openai-completions.js +11 -0
- package/packages/pi-ai/dist/providers/openai-completions.js.map +1 -1
- package/packages/pi-ai/src/providers/amazon-bedrock.ts +13 -1
- package/packages/pi-ai/src/providers/anthropic-auth.test.ts +32 -0
- package/packages/pi-ai/src/providers/anthropic-shared.test.ts +55 -1
- package/packages/pi-ai/src/providers/anthropic-shared.ts +14 -3
- package/packages/pi-ai/src/providers/anthropic.ts +8 -4
- package/packages/pi-ai/src/providers/openai-completions.ts +14 -0
- package/packages/pi-coding-agent/dist/core/agent-session-renderable-tools.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/agent-session-renderable-tools.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/agent-session-renderable-tools.test.js +61 -0
- package/packages/pi-coding-agent/dist/core/agent-session-renderable-tools.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session.js +2 -1
- package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +10 -0
- package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.js +27 -0
- package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.test.js +85 -0
- package/packages/pi-coding-agent/dist/core/auth-storage.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-resolver-initial-model-auth.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/model-resolver-initial-model-auth.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/model-resolver-initial-model-auth.test.js +64 -0
- package/packages/pi-coding-agent/dist/core/model-resolver-initial-model-auth.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/model-resolver.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-resolver.js +22 -18
- package/packages/pi-coding-agent/dist/core/model-resolver.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-resolver.test.d.ts +8 -0
- package/packages/pi-coding-agent/dist/core/model-resolver.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/model-resolver.test.js +75 -0
- package/packages/pi-coding-agent/dist/core/model-resolver.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/retry-handler.d.ts +5 -0
- package/packages/pi-coding-agent/dist/core/retry-handler.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.js +55 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.test.js +57 -0
- package/packages/pi-coding-agent/dist/core/retry-handler.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.d.ts +11 -0
- package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.js +38 -5
- package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/sdk.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/sdk.test.js +71 -0
- package/packages/pi-coding-agent/dist/core/sdk.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/index.d.ts +1 -1
- package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/index.js +1 -1
- package/packages/pi-coding-agent/dist/index.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/login-dialog.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/login-dialog.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/login-dialog.test.js +13 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/login-dialog.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.d.ts +4 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.js +24 -2
- package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js +9 -2
- package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts +4 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +43 -0
- 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 +7 -2
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/model-controller.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/model-controller.js +6 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/model-controller.js.map +1 -1
- 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 +4 -3
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.js +4 -2
- package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.js.map +1 -1
- package/packages/pi-coding-agent/package.json +1 -1
- package/packages/pi-coding-agent/src/core/agent-session-renderable-tools.test.ts +70 -0
- package/packages/pi-coding-agent/src/core/agent-session.ts +2 -1
- package/packages/pi-coding-agent/src/core/auth-storage.test.ts +108 -0
- package/packages/pi-coding-agent/src/core/auth-storage.ts +30 -0
- package/packages/pi-coding-agent/src/core/model-resolver-initial-model-auth.test.ts +78 -0
- package/packages/pi-coding-agent/src/core/model-resolver.test.ts +85 -0
- package/packages/pi-coding-agent/src/core/model-resolver.ts +22 -18
- package/packages/pi-coding-agent/src/core/retry-handler.test.ts +83 -0
- package/packages/pi-coding-agent/src/core/retry-handler.ts +60 -1
- package/packages/pi-coding-agent/src/core/sdk.test.ts +89 -0
- package/packages/pi-coding-agent/src/core/sdk.ts +45 -9
- package/packages/pi-coding-agent/src/index.ts +1 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/login-dialog.test.ts +24 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/login-dialog.ts +30 -2
- package/packages/pi-coding-agent/src/modes/interactive/components/model-selector.ts +15 -6
- package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +47 -0
- package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +7 -2
- package/packages/pi-coding-agent/src/modes/interactive/controllers/model-controller.ts +6 -1
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +4 -3
- package/packages/pi-coding-agent/src/modes/interactive/slash-command-handlers.ts +4 -2
- package/pkg/package.json +1 -1
- package/src/resources/GSD-WORKFLOW.md +1 -1
- package/src/resources/agents/debugger.md +58 -0
- package/src/resources/agents/doc-writer.md +43 -0
- package/src/resources/agents/git-ops.md +56 -0
- package/src/resources/agents/javascript-pro.md +46 -271
- package/src/resources/agents/planner.md +55 -0
- package/src/resources/agents/refactorer.md +47 -0
- package/src/resources/agents/reviewer.md +48 -0
- package/src/resources/agents/security.md +59 -0
- package/src/resources/agents/tester.md +50 -0
- package/src/resources/agents/typescript-pro.md +41 -235
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +122 -8
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +189 -6
- package/src/resources/extensions/gsd/auto/infra-errors.ts +38 -0
- package/src/resources/extensions/gsd/auto/loop-deps.ts +2 -0
- package/src/resources/extensions/gsd/auto/loop.ts +45 -1
- package/src/resources/extensions/gsd/auto/phases.ts +6 -0
- package/src/resources/extensions/gsd/auto/session.ts +11 -0
- package/src/resources/extensions/gsd/auto-dashboard.ts +29 -18
- package/src/resources/extensions/gsd/auto-model-selection.ts +9 -1
- package/src/resources/extensions/gsd/auto-prompts.ts +111 -33
- package/src/resources/extensions/gsd/auto-start.ts +41 -7
- package/src/resources/extensions/gsd/auto-tool-tracking.ts +1 -1
- package/src/resources/extensions/gsd/auto-worktree.ts +1 -1
- package/src/resources/extensions/gsd/auto.ts +72 -0
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +3 -3
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +2 -0
- package/src/resources/extensions/gsd/bootstrap/register-shortcuts.ts +79 -60
- package/src/resources/extensions/gsd/bootstrap/system-context.ts +7 -0
- package/src/resources/extensions/gsd/commands/context.ts +16 -5
- package/src/resources/extensions/gsd/commands/dispatcher.ts +14 -2
- package/src/resources/extensions/gsd/commands/handlers/auto.ts +10 -36
- package/src/resources/extensions/gsd/commands/handlers/core.ts +58 -11
- package/src/resources/extensions/gsd/commands/handlers/notifications-handler.ts +17 -7
- package/src/resources/extensions/gsd/commands/handlers/workflow.ts +4 -10
- package/src/resources/extensions/gsd/dashboard-overlay.ts +10 -3
- package/src/resources/extensions/gsd/dispatch-guard.ts +18 -1
- package/src/resources/extensions/gsd/doctor-providers.ts +24 -0
- package/src/resources/extensions/gsd/error-classifier.ts +5 -2
- package/src/resources/extensions/gsd/forensics.ts +23 -7
- package/src/resources/extensions/gsd/gate-registry.ts +251 -0
- package/src/resources/extensions/gsd/gsd-db.ts +51 -0
- package/src/resources/extensions/gsd/guided-flow.ts +5 -10
- package/src/resources/extensions/gsd/interrupted-session.ts +1 -0
- package/src/resources/extensions/gsd/metrics.ts +12 -1
- package/src/resources/extensions/gsd/milestone-actions.ts +10 -3
- package/src/resources/extensions/gsd/milestone-validation-gates.ts +11 -13
- package/src/resources/extensions/gsd/notification-overlay.ts +47 -14
- package/src/resources/extensions/gsd/notification-store.ts +54 -5
- package/src/resources/extensions/gsd/notification-widget.ts +5 -14
- package/src/resources/extensions/gsd/parallel-monitor-overlay.ts +10 -3
- package/src/resources/extensions/gsd/pre-execution-checks.ts +39 -2
- package/src/resources/extensions/gsd/prompt-validation.ts +157 -0
- package/src/resources/extensions/gsd/prompts/complete-slice.md +5 -3
- package/src/resources/extensions/gsd/prompts/discuss.md +2 -0
- package/src/resources/extensions/gsd/prompts/execute-task.md +22 -19
- package/src/resources/extensions/gsd/prompts/guided-discuss-milestone.md +2 -0
- package/src/resources/extensions/gsd/prompts/guided-discuss-slice.md +2 -0
- package/src/resources/extensions/gsd/prompts/guided-resume-task.md +1 -1
- package/src/resources/extensions/gsd/prompts/queue.md +3 -2
- package/src/resources/extensions/gsd/prompts/system.md +1 -0
- package/src/resources/extensions/gsd/prompts/validate-milestone.md +4 -1
- package/src/resources/extensions/gsd/session-model-override.ts +36 -0
- package/src/resources/extensions/gsd/shortcut-defs.ts +56 -0
- package/src/resources/extensions/gsd/state.ts +13 -2
- package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +25 -9
- package/src/resources/extensions/gsd/tests/auto-start-worktree-db-path.test.ts +28 -0
- package/src/resources/extensions/gsd/tests/bootstrap-derive-state-db-open.test.ts +39 -0
- package/src/resources/extensions/gsd/tests/complete-slice-gate-closure.test.ts +167 -0
- package/src/resources/extensions/gsd/tests/complete-slice-prompt-task-summary-layout.test.ts +18 -0
- package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +27 -0
- package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +36 -0
- package/src/resources/extensions/gsd/tests/execute-task-prompt-existing-artifact-guard.test.ts +33 -0
- package/src/resources/extensions/gsd/tests/forensics-stuck-loops.test.ts +62 -0
- package/src/resources/extensions/gsd/tests/format-shortcut.test.ts +31 -0
- package/src/resources/extensions/gsd/tests/gate-dispatch.test.ts +27 -0
- package/src/resources/extensions/gsd/tests/gate-registry.test.ts +140 -0
- package/src/resources/extensions/gsd/tests/gsd-no-project-error.test.ts +73 -0
- package/src/resources/extensions/gsd/tests/infra-errors-cooldown.test.ts +180 -0
- package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +66 -1
- package/src/resources/extensions/gsd/tests/model-isolation.test.ts +36 -51
- package/src/resources/extensions/gsd/tests/notification-store.test.ts +35 -0
- package/src/resources/extensions/gsd/tests/notification-widget.test.ts +26 -0
- package/src/resources/extensions/gsd/tests/notifications-handler.test.ts +90 -0
- package/src/resources/extensions/gsd/tests/parallel-monitor-overlay.test.ts +1 -0
- package/src/resources/extensions/gsd/tests/park-db-sync.test.ts +18 -0
- package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +49 -0
- package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +19 -0
- package/src/resources/extensions/gsd/tests/prompt-system-gate-coverage.test.ts +208 -0
- package/src/resources/extensions/gsd/tests/provider-errors.test.ts +16 -0
- package/src/resources/extensions/gsd/tests/register-shortcuts.test.ts +63 -5
- package/src/resources/extensions/gsd/tests/session-model-override.test.ts +35 -0
- package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +90 -0
- package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +7 -0
- package/src/resources/extensions/gsd/tests/validate-milestone-prompt-verification-classes.test.ts +18 -0
- package/src/resources/extensions/gsd/tools/complete-slice.ts +63 -0
- package/src/resources/extensions/gsd/tools/complete-task.ts +63 -0
- package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +4 -1
- package/src/resources/extensions/gsd/types.ts +26 -0
- package/src/resources/extensions/ollama/index.ts +13 -3
- package/src/resources/extensions/ollama/ollama-status-indicator.test.ts +28 -0
- package/src/resources/extensions/shared/gsd-phase-state.ts +42 -0
- package/src/resources/extensions/shared/tests/gsd-phase-state.test.ts +48 -0
- package/src/resources/extensions/subagent/agents.ts +10 -0
- package/src/resources/extensions/subagent/index.ts +18 -0
- package/src/resources/extensions/subagent/tests/agents-conflicts.test.ts +33 -0
- package/src/resources/skills/create-skill/SKILL.md +2 -0
- /package/dist/web/standalone/.next/static/{nPky_WQC28aBD77eZsRAB → Y0I7CjXJl-tWoV__KidV4}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{nPky_WQC28aBD77eZsRAB → Y0I7CjXJl-tWoV__KidV4}/_ssgManifest.js +0 -0
|
@@ -0,0 +1,208 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Prompt-system gate coverage tests.
|
|
3
|
+
*
|
|
4
|
+
* These tests pin the invariants the plan file documents:
|
|
5
|
+
* 1. Every pending slice-scoped gate is routed to exactly one owner turn.
|
|
6
|
+
* Q8 (owned by complete-slice) MUST NOT leak into gate-evaluate and
|
|
7
|
+
* get silently dropped the way it used to before the registry landed.
|
|
8
|
+
* 2. getPendingGatesForTurn filters by the registry's owner turn, not
|
|
9
|
+
* just the DB scope column.
|
|
10
|
+
* 3. Output validators recognize artifacts that contain the required
|
|
11
|
+
* gate section headings, and flag ones that don't.
|
|
12
|
+
* 4. Prompt output produced by the validators reflects MV01-MV04.
|
|
13
|
+
*
|
|
14
|
+
* They also assert the VALIDATION.md renderer still produces headings
|
|
15
|
+
* matching the registry's promptSection strings, so future renderer
|
|
16
|
+
* edits that drift from the registry fail the suite loudly.
|
|
17
|
+
*/
|
|
18
|
+
|
|
19
|
+
import { describe, test, beforeEach, afterEach } from "node:test";
|
|
20
|
+
import assert from "node:assert/strict";
|
|
21
|
+
import { mkdtempSync, rmSync } from "node:fs";
|
|
22
|
+
import { join } from "node:path";
|
|
23
|
+
import { tmpdir } from "node:os";
|
|
24
|
+
|
|
25
|
+
import {
|
|
26
|
+
openDatabase,
|
|
27
|
+
closeDatabase,
|
|
28
|
+
insertMilestone,
|
|
29
|
+
insertSlice,
|
|
30
|
+
insertTask,
|
|
31
|
+
insertGateRow,
|
|
32
|
+
getPendingGates,
|
|
33
|
+
getPendingGatesForTurn,
|
|
34
|
+
} from "../gsd-db.ts";
|
|
35
|
+
import {
|
|
36
|
+
GATE_REGISTRY,
|
|
37
|
+
getGatesForTurn,
|
|
38
|
+
type OwnerTurn,
|
|
39
|
+
} from "../gate-registry.ts";
|
|
40
|
+
import {
|
|
41
|
+
validateSliceSummaryOutput,
|
|
42
|
+
validateTaskSummaryOutput,
|
|
43
|
+
validateMilestoneValidationOutput,
|
|
44
|
+
validateGateSections,
|
|
45
|
+
} from "../prompt-validation.ts";
|
|
46
|
+
|
|
47
|
+
function setupTestDb(): string {
|
|
48
|
+
const tmpDir = mkdtempSync(join(tmpdir(), "prompt-gate-coverage-"));
|
|
49
|
+
const dbPath = join(tmpDir, "gsd.db");
|
|
50
|
+
openDatabase(dbPath);
|
|
51
|
+
insertMilestone({ id: "M001", title: "Test", status: "active" });
|
|
52
|
+
insertSlice({
|
|
53
|
+
milestoneId: "M001",
|
|
54
|
+
id: "S01",
|
|
55
|
+
title: "Test Slice",
|
|
56
|
+
status: "pending",
|
|
57
|
+
risk: "medium",
|
|
58
|
+
depends: [],
|
|
59
|
+
});
|
|
60
|
+
insertTask({
|
|
61
|
+
id: "T01",
|
|
62
|
+
sliceId: "S01",
|
|
63
|
+
milestoneId: "M001",
|
|
64
|
+
title: "Test Task",
|
|
65
|
+
status: "pending",
|
|
66
|
+
});
|
|
67
|
+
return tmpDir;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
describe("getPendingGatesForTurn routes by owner turn, not scope column", () => {
|
|
71
|
+
let tmpDir: string;
|
|
72
|
+
beforeEach(() => {
|
|
73
|
+
tmpDir = setupTestDb();
|
|
74
|
+
});
|
|
75
|
+
afterEach(() => {
|
|
76
|
+
closeDatabase();
|
|
77
|
+
rmSync(tmpDir, { recursive: true, force: true });
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
test("Q8 stored as scope:'slice' is owned by complete-slice, not gate-evaluate", () => {
|
|
81
|
+
// Seed the three slice-scoped gates plan-slice writes today.
|
|
82
|
+
insertGateRow({ milestoneId: "M001", sliceId: "S01", gateId: "Q3", scope: "slice" });
|
|
83
|
+
insertGateRow({ milestoneId: "M001", sliceId: "S01", gateId: "Q4", scope: "slice" });
|
|
84
|
+
insertGateRow({ milestoneId: "M001", sliceId: "S01", gateId: "Q8", scope: "slice" });
|
|
85
|
+
|
|
86
|
+
// getPendingGates(..., "slice") returns all three (unchanged).
|
|
87
|
+
const allSlicePending = getPendingGates("M001", "S01", "slice");
|
|
88
|
+
assert.equal(allSlicePending.length, 3);
|
|
89
|
+
|
|
90
|
+
// But the turn-aware helper routes them correctly.
|
|
91
|
+
const gateEval = getPendingGatesForTurn("M001", "S01", "gate-evaluate");
|
|
92
|
+
assert.deepEqual(gateEval.map((g) => g.gate_id).sort(), ["Q3", "Q4"]);
|
|
93
|
+
|
|
94
|
+
const completeSlice = getPendingGatesForTurn("M001", "S01", "complete-slice");
|
|
95
|
+
assert.deepEqual(completeSlice.map((g) => g.gate_id), ["Q8"]);
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
test("task-scoped gates are scoped to the requested task id", () => {
|
|
99
|
+
insertGateRow({ milestoneId: "M001", sliceId: "S01", gateId: "Q5", scope: "task", taskId: "T01" });
|
|
100
|
+
insertGateRow({ milestoneId: "M001", sliceId: "S01", gateId: "Q6", scope: "task", taskId: "T01" });
|
|
101
|
+
insertGateRow({ milestoneId: "M001", sliceId: "S01", gateId: "Q5", scope: "task", taskId: "T02" });
|
|
102
|
+
|
|
103
|
+
const t1 = getPendingGatesForTurn("M001", "S01", "execute-task", "T01");
|
|
104
|
+
assert.equal(t1.length, 2);
|
|
105
|
+
assert.ok(t1.every((g) => g.gate_id === "Q5" || g.gate_id === "Q6"));
|
|
106
|
+
|
|
107
|
+
const t2 = getPendingGatesForTurn("M001", "S01", "execute-task", "T02");
|
|
108
|
+
assert.equal(t2.length, 1);
|
|
109
|
+
assert.equal(t2[0].gate_id, "Q5");
|
|
110
|
+
});
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
describe("per-turn output validators", () => {
|
|
114
|
+
test("validateSliceSummaryOutput flags missing Operational Readiness", () => {
|
|
115
|
+
const md = `# S01: Test Slice\n\n## What Happened\nstuff\n\n## Verification\nstuff\n`;
|
|
116
|
+
const result = validateSliceSummaryOutput(md);
|
|
117
|
+
assert.equal(result.valid, false);
|
|
118
|
+
assert.ok(result.missing.some((m) => m.includes("Q8")));
|
|
119
|
+
assert.ok(result.missing.some((m) => m.includes("Operational Readiness")));
|
|
120
|
+
});
|
|
121
|
+
|
|
122
|
+
test("validateSliceSummaryOutput passes when Operational Readiness heading is present", () => {
|
|
123
|
+
const md = `# S01\n\n## Operational Readiness\n- Health: /health\n- Failure: alert\n`;
|
|
124
|
+
const result = validateSliceSummaryOutput(md);
|
|
125
|
+
assert.equal(result.valid, true);
|
|
126
|
+
assert.equal(result.missing.length, 0);
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
test("validateMilestoneValidationOutput requires all four MV headings", () => {
|
|
130
|
+
// Missing Requirement Coverage.
|
|
131
|
+
const md = [
|
|
132
|
+
"# Milestone Validation: M001",
|
|
133
|
+
"## Success Criteria Checklist",
|
|
134
|
+
"ok",
|
|
135
|
+
"## Slice Delivery Audit",
|
|
136
|
+
"ok",
|
|
137
|
+
"## Cross-Slice Integration",
|
|
138
|
+
"ok",
|
|
139
|
+
].join("\n\n");
|
|
140
|
+
const result = validateMilestoneValidationOutput(md);
|
|
141
|
+
assert.equal(result.valid, false);
|
|
142
|
+
assert.ok(result.missing.some((m) => m.includes("MV04")));
|
|
143
|
+
});
|
|
144
|
+
|
|
145
|
+
test("validateMilestoneValidationOutput passes for a complete VALIDATION.md", () => {
|
|
146
|
+
const md = [
|
|
147
|
+
"# Milestone Validation: M001",
|
|
148
|
+
"## Success Criteria Checklist",
|
|
149
|
+
"ok",
|
|
150
|
+
"## Slice Delivery Audit",
|
|
151
|
+
"ok",
|
|
152
|
+
"## Cross-Slice Integration",
|
|
153
|
+
"ok",
|
|
154
|
+
"## Requirement Coverage",
|
|
155
|
+
"ok",
|
|
156
|
+
].join("\n\n");
|
|
157
|
+
const result = validateMilestoneValidationOutput(md);
|
|
158
|
+
assert.equal(result.valid, true, `unexpected missing: ${result.missing.join(", ")}`);
|
|
159
|
+
});
|
|
160
|
+
|
|
161
|
+
test("validateTaskSummaryOutput flags missing task-gate sections", () => {
|
|
162
|
+
const md = `# T01\n\n## What Happened\nstuff\n\n## Verification\nstuff\n`;
|
|
163
|
+
const result = validateTaskSummaryOutput(md);
|
|
164
|
+
assert.equal(result.valid, false);
|
|
165
|
+
const idsInMissing = result.missing.join(" ");
|
|
166
|
+
assert.ok(idsInMissing.includes("Q5"));
|
|
167
|
+
assert.ok(idsInMissing.includes("Q6"));
|
|
168
|
+
assert.ok(idsInMissing.includes("Q7"));
|
|
169
|
+
});
|
|
170
|
+
|
|
171
|
+
test("validateGateSections returns empty missing when gate bucket is empty", () => {
|
|
172
|
+
// Build a phoney owner turn that owns nothing (simulate by validating
|
|
173
|
+
// against a real turn against an artifact containing every section).
|
|
174
|
+
const fullMd = getGatesForTurn("validate-milestone")
|
|
175
|
+
.map((g) => `## ${g.promptSection}\n\nstuff`)
|
|
176
|
+
.join("\n\n");
|
|
177
|
+
const result = validateGateSections(fullMd, "validate-milestone");
|
|
178
|
+
assert.equal(result.valid, true);
|
|
179
|
+
});
|
|
180
|
+
});
|
|
181
|
+
|
|
182
|
+
describe("registry / renderer parity", () => {
|
|
183
|
+
test("MV promptSections match the validate-milestone renderer H2 headings", () => {
|
|
184
|
+
// Mirror the string literals from tools/validate-milestone.ts
|
|
185
|
+
// renderValidationMarkdown() so a rename there flips this test red.
|
|
186
|
+
const expectedHeadings = [
|
|
187
|
+
"Success Criteria Checklist",
|
|
188
|
+
"Slice Delivery Audit",
|
|
189
|
+
"Cross-Slice Integration",
|
|
190
|
+
"Requirement Coverage",
|
|
191
|
+
];
|
|
192
|
+
const registryHeadings = getGatesForTurn("validate-milestone").map((g) => g.promptSection);
|
|
193
|
+
assert.deepEqual(registryHeadings.sort(), [...expectedHeadings].sort());
|
|
194
|
+
});
|
|
195
|
+
|
|
196
|
+
test("Q8 promptSection matches the complete-slice renderer H2 heading", () => {
|
|
197
|
+
// Mirror the slice-summary H2 introduced in tools/complete-slice.ts.
|
|
198
|
+
assert.equal(GATE_REGISTRY.Q8.promptSection, "Operational Readiness");
|
|
199
|
+
});
|
|
200
|
+
|
|
201
|
+
test("registry owner turns cover every turn gate-registry.ts declares", () => {
|
|
202
|
+
const ownerTurns = new Set<OwnerTurn>(Object.values(GATE_REGISTRY).map((g) => g.ownerTurn));
|
|
203
|
+
assert.ok(ownerTurns.has("gate-evaluate"));
|
|
204
|
+
assert.ok(ownerTurns.has("execute-task"));
|
|
205
|
+
assert.ok(ownerTurns.has("complete-slice"));
|
|
206
|
+
assert.ok(ownerTurns.has("validate-milestone"));
|
|
207
|
+
});
|
|
208
|
+
});
|
|
@@ -32,6 +32,15 @@ test("classifyError detects rate limit from message", () => {
|
|
|
32
32
|
assert.equal(result.kind, "rate-limit");
|
|
33
33
|
});
|
|
34
34
|
|
|
35
|
+
test("classifyError treats OpenRouter affordability errors as transient rate-limit class", () => {
|
|
36
|
+
const result = classifyError(
|
|
37
|
+
"402 This request requires more credits, or fewer max_tokens. You requested up to 32000 tokens, but can only afford 329.",
|
|
38
|
+
);
|
|
39
|
+
assert.ok(isTransient(result));
|
|
40
|
+
assert.equal(result.kind, "rate-limit");
|
|
41
|
+
assert.ok("retryAfterMs" in result && result.retryAfterMs > 0);
|
|
42
|
+
});
|
|
43
|
+
|
|
35
44
|
test("classifyError extracts reset delay from message", () => {
|
|
36
45
|
const result = classifyError("rate limit exceeded, reset in 45s");
|
|
37
46
|
assert.equal(result.kind, "rate-limit");
|
|
@@ -101,6 +110,13 @@ test("classifyError detects quota exceeded as permanent", () => {
|
|
|
101
110
|
assert.ok(!isTransient(result));
|
|
102
111
|
});
|
|
103
112
|
|
|
113
|
+
test("classifyError treats plain 'Connection error.' as transient connection failure (#3594)", () => {
|
|
114
|
+
const result = classifyError("Connection error.");
|
|
115
|
+
assert.ok(isTransient(result));
|
|
116
|
+
assert.equal(result.kind, "connection");
|
|
117
|
+
assert.ok("retryAfterMs" in result && result.retryAfterMs === 15_000);
|
|
118
|
+
});
|
|
119
|
+
|
|
104
120
|
test("classifyError treats unknown error as not transient", () => {
|
|
105
121
|
const result = classifyError("something went wrong");
|
|
106
122
|
assert.ok(!isTransient(result));
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import test from "node:test";
|
|
2
2
|
import assert from "node:assert/strict";
|
|
3
|
-
import { mkdirSync, rmSync } from "node:fs";
|
|
3
|
+
import { mkdirSync, realpathSync, rmSync } from "node:fs";
|
|
4
4
|
import { join } from "node:path";
|
|
5
5
|
import { tmpdir } from "node:os";
|
|
6
6
|
|
|
@@ -37,10 +37,10 @@ test("dashboard shortcut resolves the project root instead of the current worktr
|
|
|
37
37
|
});
|
|
38
38
|
|
|
39
39
|
let capturedHandler: ((ctx: any) => Promise<void>) | null = null;
|
|
40
|
-
const shortcuts: Array<{ description: string; handler: (ctx: any) => Promise<void> }> = [];
|
|
40
|
+
const shortcuts: Array<{ key: string; description: string; handler: (ctx: any) => Promise<void> }> = [];
|
|
41
41
|
const pi = {
|
|
42
|
-
registerShortcut: (
|
|
43
|
-
shortcuts.push(shortcut);
|
|
42
|
+
registerShortcut: (key: unknown, shortcut: { description: string; handler: (ctx: any) => Promise<void> }) => {
|
|
43
|
+
shortcuts.push({ key: String(key), ...shortcut });
|
|
44
44
|
if (!capturedHandler) {
|
|
45
45
|
capturedHandler = shortcut.handler;
|
|
46
46
|
}
|
|
@@ -69,5 +69,63 @@ test("dashboard shortcut resolves the project root instead of the current worktr
|
|
|
69
69
|
|
|
70
70
|
assert.ok(customCalls > 0, "shortcut opens the dashboard overlay when project root is resolved");
|
|
71
71
|
assert.equal(notices.length, 0, "shortcut does not fall back to the missing-.gsd warning");
|
|
72
|
-
assert.equal(shortcuts.length,
|
|
72
|
+
assert.equal(shortcuts.length, 5, "all GSD shortcuts are still registered");
|
|
73
|
+
const keys = shortcuts.map((shortcut) => shortcut.key);
|
|
74
|
+
assert.ok(keys.includes("ctrl+alt+g"), "primary dashboard shortcut is registered");
|
|
75
|
+
assert.ok(keys.includes("ctrl+shift+g"), "fallback dashboard shortcut is registered");
|
|
76
|
+
assert.ok(keys.includes("ctrl+alt+n"), "primary notifications shortcut is registered");
|
|
77
|
+
assert.ok(keys.includes("ctrl+shift+n"), "fallback notifications shortcut is registered");
|
|
78
|
+
assert.ok(keys.includes("ctrl+alt+p"), "primary parallel shortcut is registered");
|
|
79
|
+
// No Ctrl+Shift+P fallback — conflicts with cycleModelBackward (shift+ctrl+p)
|
|
80
|
+
assert.ok(!keys.includes("ctrl+shift+p"), "parallel fallback must not be registered (conflicts with cycleModelBackward)");
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
test("parallel shortcut passes resolved project root into overlay", async (t) => {
|
|
84
|
+
const base = makeTempDir("parallel-root");
|
|
85
|
+
const worktreeRoot = join(base, ".gsd", "worktrees", "M001");
|
|
86
|
+
mkdirSync(join(base, ".gsd", "parallel"), { recursive: true });
|
|
87
|
+
mkdirSync(worktreeRoot, { recursive: true });
|
|
88
|
+
|
|
89
|
+
const originalCwd = process.cwd();
|
|
90
|
+
process.chdir(worktreeRoot);
|
|
91
|
+
t.after(() => {
|
|
92
|
+
process.chdir(originalCwd);
|
|
93
|
+
cleanup(base);
|
|
94
|
+
});
|
|
95
|
+
|
|
96
|
+
const shortcuts: Array<{ key: string; description: string; handler: (ctx: any) => Promise<void> }> = [];
|
|
97
|
+
registerShortcuts({
|
|
98
|
+
registerShortcut: (key: unknown, shortcut: { description: string; handler: (ctx: any) => Promise<void> }) => {
|
|
99
|
+
shortcuts.push({ key: String(key), ...shortcut });
|
|
100
|
+
},
|
|
101
|
+
} as any);
|
|
102
|
+
|
|
103
|
+
const parallelShortcut = shortcuts.find((shortcut) => shortcut.key === "ctrl+alt+p");
|
|
104
|
+
assert.ok(parallelShortcut, "parallel shortcut is registered");
|
|
105
|
+
|
|
106
|
+
let capturedBasePath: string | undefined;
|
|
107
|
+
await parallelShortcut!.handler({
|
|
108
|
+
hasUI: true,
|
|
109
|
+
ui: {
|
|
110
|
+
custom: async (factory: any) => {
|
|
111
|
+
const overlay = factory(
|
|
112
|
+
{ requestRender() {} },
|
|
113
|
+
{ fg: (_color: string, text: string) => text, bold: (text: string) => text },
|
|
114
|
+
null,
|
|
115
|
+
() => {},
|
|
116
|
+
);
|
|
117
|
+
capturedBasePath = (overlay as any).basePath;
|
|
118
|
+
overlay.dispose?.();
|
|
119
|
+
return true;
|
|
120
|
+
},
|
|
121
|
+
notify: () => {},
|
|
122
|
+
},
|
|
123
|
+
});
|
|
124
|
+
|
|
125
|
+
assert.ok(capturedBasePath, "parallel shortcut should construct overlay with a basePath");
|
|
126
|
+
assert.equal(
|
|
127
|
+
realpathSync(capturedBasePath),
|
|
128
|
+
realpathSync(base),
|
|
129
|
+
"parallel overlay should use the resolved project root, not the worktree cwd",
|
|
130
|
+
);
|
|
73
131
|
});
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import test from "node:test";
|
|
2
|
+
import assert from "node:assert/strict";
|
|
3
|
+
import { readFileSync } from "node:fs";
|
|
4
|
+
import { join } from "node:path";
|
|
5
|
+
|
|
6
|
+
import {
|
|
7
|
+
clearSessionModelOverride,
|
|
8
|
+
getSessionModelOverride,
|
|
9
|
+
setSessionModelOverride,
|
|
10
|
+
} from "../session-model-override.js";
|
|
11
|
+
|
|
12
|
+
const phasesSource = readFileSync(join(import.meta.dirname, "..", "auto", "phases.ts"), "utf-8");
|
|
13
|
+
|
|
14
|
+
test("setSessionModelOverride stores provider/model for the session", () => {
|
|
15
|
+
const sessionId = `session-override-${Date.now()}`;
|
|
16
|
+
setSessionModelOverride(sessionId, { provider: "openai-codex", id: "gpt-5.4" });
|
|
17
|
+
|
|
18
|
+
const override = getSessionModelOverride(sessionId);
|
|
19
|
+
assert.equal(override?.provider, "openai-codex");
|
|
20
|
+
assert.equal(override?.id, "gpt-5.4");
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
test("clearSessionModelOverride removes the session override", () => {
|
|
24
|
+
const sessionId = `session-clear-${Date.now()}`;
|
|
25
|
+
setSessionModelOverride(sessionId, { provider: "anthropic", id: "claude-sonnet-4-6" });
|
|
26
|
+
clearSessionModelOverride(sessionId);
|
|
27
|
+
assert.equal(getSessionModelOverride(sessionId), undefined);
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
test("auto dispatch threads manual session model override into selectAndApplyModel", () => {
|
|
31
|
+
assert.ok(
|
|
32
|
+
phasesSource.includes("s.manualSessionModelOverride"),
|
|
33
|
+
"auto/phases.ts should pass s.manualSessionModelOverride into selectAndApplyModel",
|
|
34
|
+
);
|
|
35
|
+
});
|
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
import test from "node:test";
|
|
2
|
+
import assert from "node:assert/strict";
|
|
3
|
+
import { readFileSync } from "node:fs";
|
|
4
|
+
import { resolve } from "node:path";
|
|
5
|
+
|
|
6
|
+
const gsdDir = resolve(import.meta.dirname, "..");
|
|
7
|
+
|
|
8
|
+
function readGsdFile(relativePath: string): string {
|
|
9
|
+
return readFileSync(resolve(gsdDir, relativePath), "utf-8");
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
test("command entrypoints use startAutoDetached instead of awaiting startAuto (#3733)", () => {
|
|
13
|
+
const autoHandlerSrc = readGsdFile("commands/handlers/auto.ts");
|
|
14
|
+
const workflowHandlerSrc = readGsdFile("commands/handlers/workflow.ts");
|
|
15
|
+
const guidedFlowSrc = readGsdFile("guided-flow.ts");
|
|
16
|
+
|
|
17
|
+
assert.ok(
|
|
18
|
+
!autoHandlerSrc.includes("await startAuto("),
|
|
19
|
+
"auto command handler should not await startAuto from the active agent turn",
|
|
20
|
+
);
|
|
21
|
+
assert.ok(
|
|
22
|
+
!workflowHandlerSrc.includes("await startAuto("),
|
|
23
|
+
"workflow command handler should not await startAuto from the active agent turn",
|
|
24
|
+
);
|
|
25
|
+
assert.ok(
|
|
26
|
+
!guidedFlowSrc.includes("await startAuto("),
|
|
27
|
+
"guided flow should not await startAuto from the active agent turn",
|
|
28
|
+
);
|
|
29
|
+
|
|
30
|
+
assert.ok(
|
|
31
|
+
autoHandlerSrc.includes("startAutoDetached("),
|
|
32
|
+
"auto command handler should launch auto-mode through startAutoDetached",
|
|
33
|
+
);
|
|
34
|
+
assert.ok(
|
|
35
|
+
workflowHandlerSrc.includes("startAutoDetached("),
|
|
36
|
+
"workflow handler should launch auto-mode through startAutoDetached",
|
|
37
|
+
);
|
|
38
|
+
assert.ok(
|
|
39
|
+
guidedFlowSrc.includes("startAutoDetached("),
|
|
40
|
+
"guided flow should launch auto-mode through startAutoDetached",
|
|
41
|
+
);
|
|
42
|
+
});
|
|
43
|
+
|
|
44
|
+
test("startAutoDetached reports failures asynchronously (#3733)", () => {
|
|
45
|
+
const autoSrc = readGsdFile("auto.ts");
|
|
46
|
+
|
|
47
|
+
assert.ok(
|
|
48
|
+
autoSrc.includes("export function startAutoDetached"),
|
|
49
|
+
"auto.ts should export startAutoDetached",
|
|
50
|
+
);
|
|
51
|
+
assert.ok(
|
|
52
|
+
autoSrc.includes("void startAuto(ctx, pi, base, verboseMode, options).catch"),
|
|
53
|
+
"startAutoDetached should launch startAuto without awaiting it",
|
|
54
|
+
);
|
|
55
|
+
assert.ok(
|
|
56
|
+
autoSrc.includes("ctx.ui.notify(`Auto-start failed: ${message}`, \"error\")"),
|
|
57
|
+
"startAutoDetached should surface async startup failures to the user",
|
|
58
|
+
);
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
test("detached auto-start preserves milestone lock across pause/stop cleanup (#3733)", () => {
|
|
62
|
+
const autoSrc = readGsdFile("auto.ts");
|
|
63
|
+
const sessionSrc = readGsdFile("auto/session.ts");
|
|
64
|
+
|
|
65
|
+
assert.ok(
|
|
66
|
+
autoSrc.includes("milestoneLock?: string | null"),
|
|
67
|
+
"startAuto/startAutoDetached options should carry an explicit milestone lock",
|
|
68
|
+
);
|
|
69
|
+
assert.ok(
|
|
70
|
+
autoSrc.includes("s.sessionMilestoneLock = options.milestoneLock ?? null;"),
|
|
71
|
+
"startAuto should capture the requested milestone lock before async work begins",
|
|
72
|
+
);
|
|
73
|
+
assert.ok(
|
|
74
|
+
autoSrc.includes("milestoneLock: s.sessionMilestoneLock ?? undefined"),
|
|
75
|
+
"pause metadata should persist the detached milestone lock for resume",
|
|
76
|
+
);
|
|
77
|
+
assert.ok(
|
|
78
|
+
autoSrc.includes("s.sessionMilestoneLock = meta.milestoneLock ?? null;"),
|
|
79
|
+
"resume should restore the persisted milestone lock",
|
|
80
|
+
);
|
|
81
|
+
assert.ok(
|
|
82
|
+
autoSrc.includes("restoreMilestoneLockEnv();"),
|
|
83
|
+
"auto cleanup should restore the previous process milestone-lock env",
|
|
84
|
+
);
|
|
85
|
+
|
|
86
|
+
assert.ok(
|
|
87
|
+
sessionSrc.includes("sessionMilestoneLock: string | null = null;"),
|
|
88
|
+
"AutoSession should track the detached milestone lock explicitly",
|
|
89
|
+
);
|
|
90
|
+
});
|
|
@@ -61,6 +61,13 @@ describe("#2883: isToolInvocationError classification", () => {
|
|
|
61
61
|
);
|
|
62
62
|
});
|
|
63
63
|
|
|
64
|
+
test("detects Node v18+ JSON parse variant with property-value text", () => {
|
|
65
|
+
assert.equal(
|
|
66
|
+
isToolInvocationError("Expected ',' or '}' after property value in JSON at position 4096"),
|
|
67
|
+
true,
|
|
68
|
+
);
|
|
69
|
+
});
|
|
70
|
+
|
|
64
71
|
test("detects Unexpected end of JSON input", () => {
|
|
65
72
|
assert.equal(
|
|
66
73
|
isToolInvocationError("Unexpected end of JSON input"),
|
package/src/resources/extensions/gsd/tests/validate-milestone-prompt-verification-classes.test.ts
ADDED
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import test from "node:test";
|
|
2
|
+
import assert from "node:assert/strict";
|
|
3
|
+
import { readFileSync } from "node:fs";
|
|
4
|
+
import { join } from "node:path";
|
|
5
|
+
|
|
6
|
+
const promptPath = join(process.cwd(), "src/resources/extensions/gsd/prompts/validate-milestone.md");
|
|
7
|
+
const prompt = readFileSync(promptPath, "utf-8");
|
|
8
|
+
|
|
9
|
+
test("validate-milestone reviewer C requires canonical verification class names", () => {
|
|
10
|
+
assert.match(prompt, /\*\*Reviewer C[\s\S]*Verification Classes/i);
|
|
11
|
+
assert.match(prompt, /exact class names [`']?Contract[`']?, [`']?Integration[`']?, [`']?Operational[`']?, and [`']?UAT[`']?/i);
|
|
12
|
+
assert.match(prompt, /If no verification classes were planned, say that explicitly/i);
|
|
13
|
+
});
|
|
14
|
+
|
|
15
|
+
test("validate-milestone prompt routes verification class analysis into verificationClasses", () => {
|
|
16
|
+
assert.match(prompt, /pass it in `verificationClasses`/i);
|
|
17
|
+
assert.match(prompt, /Extract the `Verification Classes` subsection from Reviewer C and pass it verbatim in `verificationClasses`/);
|
|
18
|
+
});
|
|
@@ -21,7 +21,10 @@ import {
|
|
|
21
21
|
getMilestone,
|
|
22
22
|
updateSliceStatus,
|
|
23
23
|
setSliceSummaryMd,
|
|
24
|
+
saveGateResult,
|
|
25
|
+
getPendingGatesForTurn,
|
|
24
26
|
} from "../gsd-db.js";
|
|
27
|
+
import { getGatesForTurn } from "../gate-registry.js";
|
|
25
28
|
import { resolveSliceFile, resolveSlicePath, clearPathCache } from "../paths.js";
|
|
26
29
|
import { checkOwnership, sliceUnitKey } from "../unit-ownership.js";
|
|
27
30
|
import { saveFile, clearParseCache } from "../files.js";
|
|
@@ -39,6 +42,23 @@ export interface CompleteSliceResult {
|
|
|
39
42
|
uatPath: string;
|
|
40
43
|
}
|
|
41
44
|
|
|
45
|
+
/**
|
|
46
|
+
* Map a complete-slice-owned gate id to the CompleteSliceParams field
|
|
47
|
+
* whose presence drives `pass` vs. `omitted`. Keep this in lockstep with
|
|
48
|
+
* the gates declared in gate-registry.ts under ownerTurn "complete-slice".
|
|
49
|
+
*/
|
|
50
|
+
function sliceGateFieldForId(
|
|
51
|
+
id: string,
|
|
52
|
+
params: CompleteSliceParams,
|
|
53
|
+
): string | undefined {
|
|
54
|
+
switch (id) {
|
|
55
|
+
case "Q8":
|
|
56
|
+
return params.operationalReadiness;
|
|
57
|
+
default:
|
|
58
|
+
return undefined;
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
|
|
42
62
|
/**
|
|
43
63
|
* Render slice summary markdown matching the template format.
|
|
44
64
|
* YAML frontmatter uses snake_case keys for parseSummary() compatibility.
|
|
@@ -169,6 +189,10 @@ ${reqSurfaced}
|
|
|
169
189
|
|
|
170
190
|
${reqInvalidated}
|
|
171
191
|
|
|
192
|
+
## Operational Readiness
|
|
193
|
+
|
|
194
|
+
${params.operationalReadiness?.trim() || "None."}
|
|
195
|
+
|
|
172
196
|
## Deviations
|
|
173
197
|
|
|
174
198
|
${params.deviations || "None."}
|
|
@@ -330,6 +354,45 @@ export async function handleCompleteSlice(
|
|
|
330
354
|
// Store rendered markdown in DB for D004 recovery
|
|
331
355
|
setSliceSummaryMd(params.milestoneId, params.sliceId, summaryMd, uatMd);
|
|
332
356
|
|
|
357
|
+
// ── Close gates owned by complete-slice (Q8) ───────────────────────────
|
|
358
|
+
// Each owned gate maps to a specific summary section via the registry.
|
|
359
|
+
// If the caller populated the corresponding field, record `pass`; if the
|
|
360
|
+
// field is empty, record `omitted`. Without this loop, Q8 would stay
|
|
361
|
+
// pending forever and block future state derivation (see gate-registry).
|
|
362
|
+
try {
|
|
363
|
+
const pendingGates = getPendingGatesForTurn(
|
|
364
|
+
params.milestoneId,
|
|
365
|
+
params.sliceId,
|
|
366
|
+
"complete-slice",
|
|
367
|
+
);
|
|
368
|
+
if (pendingGates.length > 0) {
|
|
369
|
+
const ownedDefs = new Map(getGatesForTurn("complete-slice").map((g) => [g.id, g] as const));
|
|
370
|
+
for (const row of pendingGates) {
|
|
371
|
+
const def = ownedDefs.get(row.gate_id);
|
|
372
|
+
if (!def) continue;
|
|
373
|
+
// Map gate id → param field it maps to. Keep the map local so
|
|
374
|
+
// adding a new complete-slice gate is a single place change.
|
|
375
|
+
const field = sliceGateFieldForId(def.id, params);
|
|
376
|
+
const hasContent = typeof field === "string" && field.trim().length > 0;
|
|
377
|
+
saveGateResult({
|
|
378
|
+
milestoneId: params.milestoneId,
|
|
379
|
+
sliceId: params.sliceId,
|
|
380
|
+
gateId: def.id,
|
|
381
|
+
verdict: hasContent ? "pass" : "omitted",
|
|
382
|
+
rationale: hasContent
|
|
383
|
+
? `${def.promptSection} section populated in slice summary`
|
|
384
|
+
: `${def.promptSection} section left empty — recorded as omitted`,
|
|
385
|
+
findings: hasContent ? (field as string).trim() : "",
|
|
386
|
+
});
|
|
387
|
+
}
|
|
388
|
+
}
|
|
389
|
+
} catch (gateErr) {
|
|
390
|
+
logWarning(
|
|
391
|
+
"tool",
|
|
392
|
+
`complete-slice gate close warning for ${params.milestoneId}/${params.sliceId}: ${(gateErr as Error).message}`,
|
|
393
|
+
);
|
|
394
|
+
}
|
|
395
|
+
|
|
333
396
|
// Invalidate all caches
|
|
334
397
|
invalidateStateCache();
|
|
335
398
|
clearPathCache();
|
|
@@ -24,7 +24,10 @@ import {
|
|
|
24
24
|
updateTaskStatus,
|
|
25
25
|
setTaskSummaryMd,
|
|
26
26
|
deleteVerificationEvidence,
|
|
27
|
+
saveGateResult,
|
|
28
|
+
getPendingGatesForTurn,
|
|
27
29
|
} from "../gsd-db.js";
|
|
30
|
+
import { getGatesForTurn } from "../gate-registry.js";
|
|
28
31
|
import { resolveSliceFile, resolveTasksDir, clearPathCache } from "../paths.js";
|
|
29
32
|
import { checkOwnership, taskUnitKey } from "../unit-ownership.js";
|
|
30
33
|
import { saveFile, clearParseCache } from "../files.js";
|
|
@@ -44,6 +47,27 @@ export interface CompleteTaskResult {
|
|
|
44
47
|
|
|
45
48
|
import type { TaskRow } from "../gsd-db.js";
|
|
46
49
|
|
|
50
|
+
/**
|
|
51
|
+
* Map an execute-task-owned gate id to the CompleteTaskParams field whose
|
|
52
|
+
* presence drives `pass` vs. `omitted`. Keep in lockstep with the gates
|
|
53
|
+
* declared in gate-registry.ts under ownerTurn "execute-task".
|
|
54
|
+
*/
|
|
55
|
+
function taskGateFieldForId(
|
|
56
|
+
id: string,
|
|
57
|
+
params: CompleteTaskParams,
|
|
58
|
+
): string | undefined {
|
|
59
|
+
switch (id) {
|
|
60
|
+
case "Q5":
|
|
61
|
+
return params.failureModes;
|
|
62
|
+
case "Q6":
|
|
63
|
+
return params.loadProfile;
|
|
64
|
+
case "Q7":
|
|
65
|
+
return params.negativeTests;
|
|
66
|
+
default:
|
|
67
|
+
return undefined;
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
|
|
47
71
|
/**
|
|
48
72
|
* Normalize a list parameter that may arrive as a string (newline-delimited
|
|
49
73
|
* bullet list from the LLM) into a string array (#3361).
|
|
@@ -236,6 +260,45 @@ export async function handleCompleteTask(
|
|
|
236
260
|
// Store rendered markdown in DB for D004 recovery
|
|
237
261
|
setTaskSummaryMd(params.milestoneId, params.sliceId, params.taskId, summaryMd);
|
|
238
262
|
|
|
263
|
+
// ── Close gates owned by execute-task (Q5/Q6/Q7) for this task ────────
|
|
264
|
+
// Each gate id maps to a specific params field via taskGateFieldForId.
|
|
265
|
+
// When the model populates the field, record `pass`; when it's empty,
|
|
266
|
+
// record `omitted`. Task-scoped rows are filtered by taskId so a single
|
|
267
|
+
// task's completion doesn't touch sibling tasks' gate rows.
|
|
268
|
+
try {
|
|
269
|
+
const pendingGates = getPendingGatesForTurn(
|
|
270
|
+
params.milestoneId,
|
|
271
|
+
params.sliceId,
|
|
272
|
+
"execute-task",
|
|
273
|
+
params.taskId,
|
|
274
|
+
);
|
|
275
|
+
if (pendingGates.length > 0) {
|
|
276
|
+
const ownedDefs = new Map(getGatesForTurn("execute-task").map((g) => [g.id, g] as const));
|
|
277
|
+
for (const row of pendingGates) {
|
|
278
|
+
const def = ownedDefs.get(row.gate_id);
|
|
279
|
+
if (!def) continue;
|
|
280
|
+
const field = taskGateFieldForId(def.id, params);
|
|
281
|
+
const hasContent = typeof field === "string" && field.trim().length > 0;
|
|
282
|
+
saveGateResult({
|
|
283
|
+
milestoneId: params.milestoneId,
|
|
284
|
+
sliceId: params.sliceId,
|
|
285
|
+
taskId: params.taskId,
|
|
286
|
+
gateId: def.id,
|
|
287
|
+
verdict: hasContent ? "pass" : "omitted",
|
|
288
|
+
rationale: hasContent
|
|
289
|
+
? `${def.promptSection} section populated in task summary`
|
|
290
|
+
: `${def.promptSection} section left empty — recorded as omitted`,
|
|
291
|
+
findings: hasContent ? (field as string).trim() : "",
|
|
292
|
+
});
|
|
293
|
+
}
|
|
294
|
+
}
|
|
295
|
+
} catch (gateErr) {
|
|
296
|
+
logWarning(
|
|
297
|
+
"tool",
|
|
298
|
+
`complete-task gate close warning for ${params.milestoneId}/${params.sliceId}/${params.taskId}: ${(gateErr as Error).message}`,
|
|
299
|
+
);
|
|
300
|
+
}
|
|
301
|
+
|
|
239
302
|
// Invalidate all caches
|
|
240
303
|
invalidateStateCache();
|
|
241
304
|
clearPathCache();
|
|
@@ -8,6 +8,7 @@ import {
|
|
|
8
8
|
_getAdapter,
|
|
9
9
|
saveGateResult,
|
|
10
10
|
} from "../gsd-db.js";
|
|
11
|
+
import { GATE_REGISTRY } from "../gate-registry.js";
|
|
11
12
|
import { saveArtifactToDb } from "../db-writer.js";
|
|
12
13
|
import type { CompleteMilestoneParams } from "./complete-milestone.js";
|
|
13
14
|
import { handleCompleteMilestone } from "./complete-milestone.js";
|
|
@@ -427,7 +428,9 @@ export async function executeSaveGateResult(
|
|
|
427
428
|
};
|
|
428
429
|
}
|
|
429
430
|
|
|
430
|
-
|
|
431
|
+
// Source of truth: gate-registry.ts. Every declared GateId is accepted,
|
|
432
|
+
// so adding a new gate in one place automatically flows through here.
|
|
433
|
+
const validGates = Object.keys(GATE_REGISTRY);
|
|
431
434
|
if (!validGates.includes(params.gateId)) {
|
|
432
435
|
return {
|
|
433
436
|
content: [{ type: "text", text: `Error: Invalid gateId "${params.gateId}". Must be one of: ${validGates.join(", ")}` }],
|