@opengsd/gsd-pi 1.2.0-dev.844675c9 → 1.2.0-dev.955e4da0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli-web-branch.d.ts +2 -0
- package/dist/cli-web-branch.js +9 -2
- package/dist/help-text.js +5 -0
- package/dist/resources/.managed-resources-content-hash +1 -1
- package/dist/resources/extensions/ask-user-questions.js +78 -23
- package/dist/resources/extensions/bg-shell/utilities.js +2 -2
- package/dist/resources/extensions/claude-code-cli/models.js +9 -0
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +92 -230
- package/dist/resources/extensions/claude-code-cli/turn-assembler.js +224 -0
- package/dist/resources/extensions/github-sync/templates.js +3 -3
- package/dist/resources/extensions/gsd/artifact-projection.js +14 -0
- package/dist/resources/extensions/gsd/auto/loop.js +74 -56
- package/dist/resources/extensions/gsd/auto/orchestrator.js +142 -15
- package/dist/resources/extensions/gsd/auto/phases.js +34 -4
- package/dist/resources/extensions/gsd/auto/run-unit.js +2 -1
- package/dist/resources/extensions/gsd/auto/session.js +3 -0
- package/dist/resources/extensions/gsd/auto-dashboard.js +16 -4
- package/dist/resources/extensions/gsd/auto-dispatch.js +6 -5
- package/dist/resources/extensions/gsd/auto-model-selection.js +8 -0
- package/dist/resources/extensions/gsd/auto-post-unit.js +12 -9
- package/dist/resources/extensions/gsd/auto-prompts.js +81 -8
- package/dist/resources/extensions/gsd/auto-recovery.js +48 -49
- package/dist/resources/extensions/gsd/auto-runtime-state.js +14 -0
- package/dist/resources/extensions/gsd/auto-start.js +20 -36
- package/dist/resources/extensions/gsd/auto-timers.js +16 -2
- package/dist/resources/extensions/gsd/auto-tool-tracking.js +32 -0
- package/dist/resources/extensions/gsd/auto-unit-tool-scope.js +4 -29
- package/dist/resources/extensions/gsd/auto-verification.js +7 -7
- package/dist/resources/extensions/gsd/auto-worktree-repair.js +10 -2
- package/dist/resources/extensions/gsd/auto-worktree.js +34 -289
- package/dist/resources/extensions/gsd/auto.js +15 -14
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +28 -37
- package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +20 -43
- package/dist/resources/extensions/gsd/bootstrap/query-tools.js +2 -2
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +131 -140
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +89 -8
- package/dist/resources/extensions/gsd/captures.js +5 -13
- package/dist/resources/extensions/gsd/closeout-consistency-gate.js +21 -4
- package/dist/resources/extensions/gsd/closeout-recovery.js +3 -2
- package/dist/resources/extensions/gsd/codebase-generator.js +8 -4
- package/dist/resources/extensions/gsd/commands/catalog.js +6 -62
- package/dist/resources/extensions/gsd/commands/handlers/auto.js +3 -0
- package/dist/resources/extensions/gsd/commands-handlers.js +20 -0
- package/dist/resources/extensions/gsd/commands-inspect.js +4 -8
- package/dist/resources/extensions/gsd/commands-maintenance.js +61 -41
- package/dist/resources/extensions/gsd/commands-ship.js +2 -2
- package/dist/resources/extensions/gsd/commands-verdict.js +12 -2
- package/dist/resources/extensions/gsd/db/engine.js +755 -0
- package/dist/resources/extensions/gsd/db/queries.js +372 -0
- package/dist/resources/extensions/gsd/db/sql-constants.js +11 -0
- package/dist/resources/extensions/gsd/db/writers/cascades.js +194 -0
- package/dist/resources/extensions/gsd/db/writers/import-restore.js +182 -0
- package/dist/resources/extensions/gsd/db/writers/memory.js +149 -0
- package/dist/resources/extensions/gsd/db/writers/reconcile.js +458 -0
- package/dist/resources/extensions/gsd/db/writers/status.js +70 -0
- package/dist/resources/extensions/gsd/db-workspace.js +103 -0
- package/dist/resources/extensions/gsd/delegation-policy.js +2 -10
- package/dist/resources/extensions/gsd/discussion-handoff.js +218 -0
- package/dist/resources/extensions/gsd/docs/preferences-reference.md +9 -0
- package/dist/resources/extensions/gsd/doctor-environment.js +8 -10
- package/dist/resources/extensions/gsd/doctor-git-checks.js +4 -3
- package/dist/resources/extensions/gsd/doctor-runtime-checks.js +9 -2
- package/dist/resources/extensions/gsd/doctor.js +16 -9
- package/dist/resources/extensions/gsd/error-classifier.js +1 -1
- package/dist/resources/extensions/gsd/git-conflict-state.js +16 -1
- package/dist/resources/extensions/gsd/git-service.js +1 -0
- package/dist/resources/extensions/gsd/gitignore.js +3 -0
- package/dist/resources/extensions/gsd/gsd-db.js +183 -2048
- package/dist/resources/extensions/gsd/guided-flow.js +68 -471
- package/dist/resources/extensions/gsd/guided-unit-completion.js +225 -0
- package/dist/resources/extensions/gsd/markdown-renderer.js +2 -1
- package/dist/resources/extensions/gsd/mcp-filter.js +2 -1
- package/dist/resources/extensions/gsd/mcp-tool-name.js +26 -0
- package/dist/resources/extensions/gsd/md-importer.js +4 -3
- package/dist/resources/extensions/gsd/migrate/safety.js +19 -11
- package/dist/resources/extensions/gsd/migration-auto-check.js +27 -5
- package/dist/resources/extensions/gsd/milestone-closeout-proof.js +72 -0
- package/dist/resources/extensions/gsd/milestone-closeout.js +12 -4
- package/dist/resources/extensions/gsd/milestone-merge-transaction.js +10 -0
- package/dist/resources/extensions/gsd/milestone-planning-persistence.js +156 -0
- package/dist/resources/extensions/gsd/milestone-readiness.js +77 -0
- package/dist/resources/extensions/gsd/milestone-settlement.js +50 -0
- package/dist/resources/extensions/gsd/milestone-validation-evidence.js +73 -0
- package/dist/resources/extensions/gsd/milestone-validation-verdict.js +57 -0
- package/dist/resources/extensions/gsd/model-cost-table.js +1 -0
- package/dist/resources/extensions/gsd/model-router.js +3 -0
- package/dist/resources/extensions/gsd/parallel-eligibility.js +3 -6
- package/dist/resources/extensions/gsd/parallel-merge.js +14 -11
- package/dist/resources/extensions/gsd/parallel-monitor-overlay.js +7 -5
- package/dist/resources/extensions/gsd/parallel-orchestrator.js +3 -2
- package/dist/resources/extensions/gsd/paths.js +10 -24
- package/dist/resources/extensions/gsd/preferences-diagnostics.js +67 -0
- package/dist/resources/extensions/gsd/preferences.js +161 -29
- package/dist/resources/extensions/gsd/prompts/complete-slice.md +1 -0
- package/dist/resources/extensions/gsd/prompts/execute-task.md +2 -0
- package/dist/resources/extensions/gsd/prompts/guided-discuss-project.md +3 -1
- package/dist/resources/extensions/gsd/prompts/plan-milestone.md +2 -0
- package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -0
- package/dist/resources/extensions/gsd/prompts/refine-slice.md +1 -0
- package/dist/resources/extensions/gsd/prompts/system.md +1 -1
- package/dist/resources/extensions/gsd/provider-payload-policy.js +83 -0
- package/dist/resources/extensions/gsd/pull-request-process.js +13 -0
- package/dist/resources/extensions/gsd/quality-gate-closure.js +109 -0
- package/dist/resources/extensions/gsd/question-transport.js +86 -0
- package/dist/resources/extensions/gsd/recovery-classification.js +12 -1
- package/dist/resources/extensions/gsd/roadmap-slices.js +8 -2
- package/dist/resources/extensions/gsd/safety/evidence-collector.js +37 -4
- package/dist/resources/extensions/gsd/safety/evidence-cross-ref.js +7 -2
- package/dist/resources/extensions/gsd/safety/file-change-validator.js +10 -0
- package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +3 -2
- package/dist/resources/extensions/gsd/state-transition-matrix.js +38 -0
- package/dist/resources/extensions/gsd/state.js +13 -5
- package/dist/resources/extensions/gsd/status-guards.js +56 -8
- package/dist/resources/extensions/gsd/templates/plan.md +7 -0
- package/dist/resources/extensions/gsd/templates/project.md +1 -0
- package/dist/resources/extensions/gsd/templates/roadmap.md +1 -1
- package/dist/resources/extensions/gsd/templates/uat.md +5 -1
- package/dist/resources/extensions/gsd/tool-contract.js +52 -8
- package/dist/resources/extensions/gsd/tool-presentation-plan.js +15 -34
- package/dist/resources/extensions/gsd/tool-surface-snapshot.js +17 -0
- package/dist/resources/extensions/gsd/tools/complete-slice.js +24 -43
- package/dist/resources/extensions/gsd/tools/exec-tool.js +5 -5
- package/dist/resources/extensions/gsd/tools/plan-milestone.js +15 -143
- package/dist/resources/extensions/gsd/tools/reassess-roadmap.js +39 -0
- package/dist/resources/extensions/gsd/tools/reopen-milestone.js +11 -29
- package/dist/resources/extensions/gsd/tools/reopen-slice.js +14 -33
- package/dist/resources/extensions/gsd/tools/skip-slice.js +18 -36
- package/dist/resources/extensions/gsd/tools/validate-milestone.js +15 -78
- package/dist/resources/extensions/gsd/uat-policy.js +16 -10
- package/dist/resources/extensions/gsd/uat-run.js +9 -14
- package/dist/resources/extensions/gsd/undo.js +8 -7
- package/dist/resources/extensions/gsd/unit-context-composer.js +40 -20
- package/dist/resources/extensions/gsd/unit-runtime.js +3 -2
- package/dist/resources/extensions/gsd/unit-tool-contracts.js +2 -1
- package/dist/resources/extensions/gsd/user-input-boundary.js +23 -0
- package/dist/resources/extensions/gsd/validation-block-guard.js +2 -0
- package/dist/resources/extensions/gsd/web-app-uat.js +80 -0
- package/dist/resources/extensions/gsd/workflow-mcp.js +15 -102
- package/dist/resources/extensions/gsd/workflow-reconcile.js +4 -3
- package/dist/resources/extensions/gsd/workflow-tool-surface.js +46 -0
- package/dist/resources/extensions/gsd/workspace-git-guard.js +2 -0
- package/dist/resources/extensions/gsd/worktree-git-recovery.js +287 -0
- package/dist/resources/extensions/gsd/worktree-lifecycle.js +9 -1
- package/dist/resources/extensions/gsd/worktree-manager.js +45 -28
- package/dist/resources/extensions/gsd/worktree-placement.js +59 -0
- package/dist/resources/extensions/gsd/worktree-reentry.js +12 -8
- package/dist/resources/extensions/gsd/worktree-root.js +17 -6
- package/dist/resources/extensions/gsd/worktree-safety.js +8 -5
- package/dist/resources/extensions/gsd/worktree-session-state.js +12 -10
- package/dist/resources/extensions/gsd/worktree-state-projection.js +33 -4
- package/dist/resources/extensions/gsd/worktree-telemetry.js +12 -0
- package/dist/resources/extensions/shared/interview-ui.js +2 -2
- package/dist/resources/shared/claude-runtime-floor.js +182 -0
- package/dist/resources/skills/gsd-browser/SKILL.md +1 -1
- package/dist/tsconfig.extensions.tsbuildinfo +1 -1
- package/dist/update-cmd.js +20 -0
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +12 -12
- 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 +8 -8
- package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/mcp-connections/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/notifications/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/shutdown/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +12 -12
- package/dist/web/standalone/.next/server/chunks/{5047.js → 5942.js} +2 -2
- package/dist/web/standalone/.next/server/chunks/8357.js +1 -1
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +1 -1
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/dist/web/standalone/.next/static/chunks/2659.b7b129ee6a769448.js +1 -0
- package/dist/web/standalone/.next/static/chunks/2772.bfa657f49f955239.js +1 -0
- package/dist/web/standalone/.next/static/chunks/{3616.4113d484a994e411.js → 3616.3c60753b8ffcbd2e.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/4283.e4873b058df143a1.js +2 -0
- package/dist/web/standalone/.next/static/chunks/5826.a46ecdd1cfe8dabc.js +1 -0
- package/dist/web/standalone/.next/static/chunks/796.cf859a427a2cb2ac.js +10 -0
- package/dist/web/standalone/.next/static/chunks/8785.2e5a118797fb2dd2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/{webpack-dda80a1ef5587410.js → webpack-fbea77b5f9953368.js} +1 -1
- package/dist/web/standalone/node_modules/node-pty/build/Makefile +1 -1
- package/dist/web-mode.d.ts +2 -0
- package/dist/web-mode.js +20 -8
- package/dist/worktree-status-banner.js +7 -3
- package/package.json +2 -1
- package/packages/cloud-mcp-gateway/package.json +2 -2
- package/packages/contracts/package.json +1 -1
- package/packages/daemon/package.json +4 -4
- package/packages/gsd-agent-core/dist/session/agent-session-extensions.d.ts +2 -0
- package/packages/gsd-agent-core/dist/session/agent-session-extensions.d.ts.map +1 -1
- package/packages/gsd-agent-core/dist/session/agent-session-extensions.js +14 -0
- package/packages/gsd-agent-core/dist/session/agent-session-extensions.js.map +1 -1
- package/packages/gsd-agent-core/package.json +5 -5
- package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js +3 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js +106 -40
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-extension-widgets.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-extension-widgets.js +6 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-extension-widgets.js.map +1 -1
- package/packages/gsd-agent-modes/package.json +7 -7
- package/packages/mcp-server/dist/server.d.ts +10 -0
- package/packages/mcp-server/dist/server.d.ts.map +1 -1
- package/packages/mcp-server/dist/server.js +8 -0
- package/packages/mcp-server/dist/server.js.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts +41 -0
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +32 -22
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/package.json +3 -3
- package/packages/native/package.json +1 -1
- package/packages/pi-agent-core/package.json +1 -1
- package/packages/pi-ai/dist/image-models.generated.d.ts +2 -2
- package/packages/pi-ai/dist/image-models.generated.js +6 -6
- package/packages/pi-ai/dist/image-models.generated.js.map +1 -1
- package/packages/pi-ai/dist/models.generated.d.ts +295 -98
- package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
- package/packages/pi-ai/dist/models.generated.js +309 -154
- package/packages/pi-ai/dist/models.generated.js.map +1 -1
- package/packages/pi-ai/package.json +1 -1
- package/packages/pi-coding-agent/dist/core/capability-patches.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/capability-patches.js +3 -1
- package/packages/pi-coding-agent/dist/core/capability-patches.js.map +1 -1
- package/packages/pi-coding-agent/package.json +7 -7
- package/packages/pi-tui/dist/components/input.js +1 -1
- package/packages/pi-tui/dist/components/input.js.map +1 -1
- package/packages/pi-tui/dist/keys.d.ts.map +1 -1
- package/packages/pi-tui/dist/keys.js +39 -30
- package/packages/pi-tui/dist/keys.js.map +1 -1
- package/packages/pi-tui/dist/stdin-buffer.d.ts.map +1 -1
- package/packages/pi-tui/dist/stdin-buffer.js +22 -0
- package/packages/pi-tui/dist/stdin-buffer.js.map +1 -1
- package/packages/pi-tui/package.json +2 -2
- package/packages/rpc-client/package.json +2 -2
- package/pkg/package.json +1 -1
- package/src/resources/extensions/ask-user-questions.ts +87 -24
- package/src/resources/extensions/bg-shell/utilities.ts +2 -2
- package/src/resources/extensions/claude-code-cli/models.ts +9 -0
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +114 -281
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +268 -0
- package/src/resources/extensions/claude-code-cli/turn-assembler.ts +287 -0
- package/src/resources/extensions/github-sync/templates.ts +3 -3
- package/src/resources/extensions/github-sync/tests/templates.test.ts +2 -2
- package/src/resources/extensions/gsd/artifact-projection.ts +31 -0
- package/src/resources/extensions/gsd/auto/contracts.ts +32 -2
- package/src/resources/extensions/gsd/auto/loop-deps.ts +3 -1
- package/src/resources/extensions/gsd/auto/loop.ts +83 -61
- package/src/resources/extensions/gsd/auto/orchestrator.ts +164 -17
- package/src/resources/extensions/gsd/auto/phases.ts +45 -4
- package/src/resources/extensions/gsd/auto/run-unit.ts +2 -1
- package/src/resources/extensions/gsd/auto/session.ts +4 -0
- package/src/resources/extensions/gsd/auto-dashboard.ts +18 -4
- package/src/resources/extensions/gsd/auto-dispatch.ts +20 -7
- package/src/resources/extensions/gsd/auto-model-selection.ts +8 -0
- package/src/resources/extensions/gsd/auto-post-unit.ts +16 -8
- package/src/resources/extensions/gsd/auto-prompts.ts +107 -9
- package/src/resources/extensions/gsd/auto-recovery.ts +50 -50
- package/src/resources/extensions/gsd/auto-runtime-state.ts +26 -0
- package/src/resources/extensions/gsd/auto-start.ts +25 -34
- package/src/resources/extensions/gsd/auto-timers.ts +16 -2
- package/src/resources/extensions/gsd/auto-tool-tracking.ts +35 -0
- package/src/resources/extensions/gsd/auto-unit-tool-scope.ts +9 -30
- package/src/resources/extensions/gsd/auto-verification.ts +7 -8
- package/src/resources/extensions/gsd/auto-worktree-repair.ts +13 -2
- package/src/resources/extensions/gsd/auto-worktree.ts +53 -306
- package/src/resources/extensions/gsd/auto.ts +27 -17
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +29 -37
- package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +20 -43
- package/src/resources/extensions/gsd/bootstrap/query-tools.ts +2 -2
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +147 -153
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +132 -6
- package/src/resources/extensions/gsd/captures.ts +5 -14
- package/src/resources/extensions/gsd/closeout-consistency-gate.ts +27 -5
- package/src/resources/extensions/gsd/closeout-recovery.ts +2 -1
- package/src/resources/extensions/gsd/codebase-generator.ts +9 -5
- package/src/resources/extensions/gsd/commands/catalog.ts +6 -68
- package/src/resources/extensions/gsd/commands/handlers/auto.ts +3 -0
- package/src/resources/extensions/gsd/commands-handlers.ts +18 -0
- package/src/resources/extensions/gsd/commands-inspect.ts +7 -8
- package/src/resources/extensions/gsd/commands-maintenance.ts +74 -40
- package/src/resources/extensions/gsd/commands-ship.ts +2 -2
- package/src/resources/extensions/gsd/commands-verdict.ts +19 -2
- package/src/resources/extensions/gsd/db/engine.ts +809 -0
- package/src/resources/extensions/gsd/db/queries.ts +453 -0
- package/src/resources/extensions/gsd/db/sql-constants.ts +12 -0
- package/src/resources/extensions/gsd/db/writers/cascades.ts +237 -0
- package/src/resources/extensions/gsd/db/writers/import-restore.ts +310 -0
- package/src/resources/extensions/gsd/db/writers/memory.ts +220 -0
- package/src/resources/extensions/gsd/db/writers/reconcile.ts +500 -0
- package/src/resources/extensions/gsd/db/writers/status.ts +88 -0
- package/src/resources/extensions/gsd/db-workspace.ts +170 -0
- package/src/resources/extensions/gsd/delegation-policy.ts +3 -11
- package/src/resources/extensions/gsd/discussion-handoff.ts +276 -0
- package/src/resources/extensions/gsd/docs/preferences-reference.md +9 -0
- package/src/resources/extensions/gsd/doctor-environment.ts +8 -11
- package/src/resources/extensions/gsd/doctor-git-checks.ts +3 -3
- package/src/resources/extensions/gsd/doctor-runtime-checks.ts +10 -3
- package/src/resources/extensions/gsd/doctor.ts +15 -5
- package/src/resources/extensions/gsd/error-classifier.ts +1 -1
- package/src/resources/extensions/gsd/git-conflict-state.ts +17 -1
- package/src/resources/extensions/gsd/git-service.ts +1 -0
- package/src/resources/extensions/gsd/gitignore.ts +3 -0
- package/src/resources/extensions/gsd/gsd-db.ts +185 -2373
- package/src/resources/extensions/gsd/guided-flow.ts +81 -561
- package/src/resources/extensions/gsd/guided-unit-completion.ts +275 -0
- package/src/resources/extensions/gsd/markdown-renderer.ts +2 -1
- package/src/resources/extensions/gsd/mcp-filter.ts +2 -1
- package/src/resources/extensions/gsd/mcp-tool-name.ts +35 -0
- package/src/resources/extensions/gsd/md-importer.ts +3 -3
- package/src/resources/extensions/gsd/migrate/safety.ts +17 -9
- package/src/resources/extensions/gsd/migration-auto-check.ts +30 -5
- package/src/resources/extensions/gsd/milestone-closeout-proof.ts +131 -0
- package/src/resources/extensions/gsd/milestone-closeout.ts +12 -4
- package/src/resources/extensions/gsd/milestone-merge-transaction.ts +47 -0
- package/src/resources/extensions/gsd/milestone-planning-persistence.ts +224 -0
- package/src/resources/extensions/gsd/milestone-readiness.ts +125 -0
- package/src/resources/extensions/gsd/milestone-settlement.ts +81 -0
- package/src/resources/extensions/gsd/milestone-validation-evidence.ts +95 -0
- package/src/resources/extensions/gsd/milestone-validation-verdict.ts +80 -0
- package/src/resources/extensions/gsd/model-cost-table.ts +1 -0
- package/src/resources/extensions/gsd/model-router.ts +3 -0
- package/src/resources/extensions/gsd/parallel-eligibility.ts +4 -5
- package/src/resources/extensions/gsd/parallel-merge.ts +12 -9
- package/src/resources/extensions/gsd/parallel-monitor-overlay.ts +6 -5
- package/src/resources/extensions/gsd/parallel-orchestrator.ts +6 -2
- package/src/resources/extensions/gsd/paths.ts +9 -22
- package/src/resources/extensions/gsd/preferences-diagnostics.ts +98 -0
- package/src/resources/extensions/gsd/preferences-types.ts +16 -0
- package/src/resources/extensions/gsd/preferences.ts +191 -28
- package/src/resources/extensions/gsd/prompts/complete-slice.md +1 -0
- package/src/resources/extensions/gsd/prompts/execute-task.md +2 -0
- package/src/resources/extensions/gsd/prompts/guided-discuss-project.md +3 -1
- package/src/resources/extensions/gsd/prompts/plan-milestone.md +2 -0
- package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -0
- package/src/resources/extensions/gsd/prompts/refine-slice.md +1 -0
- package/src/resources/extensions/gsd/prompts/system.md +1 -1
- package/src/resources/extensions/gsd/provider-payload-policy.ts +140 -0
- package/src/resources/extensions/gsd/pull-request-process.ts +41 -0
- package/src/resources/extensions/gsd/quality-gate-closure.ts +140 -0
- package/src/resources/extensions/gsd/question-transport.ts +138 -0
- package/src/resources/extensions/gsd/recovery-classification.ts +14 -1
- package/src/resources/extensions/gsd/roadmap-slices.ts +8 -2
- package/src/resources/extensions/gsd/safety/evidence-collector.ts +36 -4
- package/src/resources/extensions/gsd/safety/evidence-cross-ref.ts +7 -2
- package/src/resources/extensions/gsd/safety/file-change-validator.ts +14 -0
- package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +6 -2
- package/src/resources/extensions/gsd/state-transition-matrix.ts +42 -0
- package/src/resources/extensions/gsd/state.ts +15 -5
- package/src/resources/extensions/gsd/status-guards.ts +59 -8
- package/src/resources/extensions/gsd/templates/plan.md +7 -0
- package/src/resources/extensions/gsd/templates/project.md +1 -0
- package/src/resources/extensions/gsd/templates/roadmap.md +1 -1
- package/src/resources/extensions/gsd/templates/uat.md +5 -1
- package/src/resources/extensions/gsd/tests/ask-user-questions-render.test.ts +92 -0
- package/src/resources/extensions/gsd/tests/auto-dashboard.test.ts +29 -1
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +444 -5
- package/src/resources/extensions/gsd/tests/auto-milestone-target.test.ts +23 -0
- package/src/resources/extensions/gsd/tests/auto-model-selection-tool-poisoning.test.ts +18 -0
- package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +133 -4
- package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +3 -1
- package/src/resources/extensions/gsd/tests/auto-post-unit-evidence-crossref-4909.test.ts +46 -0
- package/src/resources/extensions/gsd/tests/auto-runtime-state.test.ts +34 -0
- package/src/resources/extensions/gsd/tests/auto-worktree-registry.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/auto-worktree-repair.test.ts +4 -2
- package/src/resources/extensions/gsd/tests/canonical-milestone-root.test.ts +20 -0
- package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +22 -0
- package/src/resources/extensions/gsd/tests/codebase-generator.test.ts +22 -0
- package/src/resources/extensions/gsd/tests/commands-dispatcher-workspace-git.test.ts +11 -0
- package/src/resources/extensions/gsd/tests/commands-verdict.test.ts +38 -1
- package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +34 -3
- package/src/resources/extensions/gsd/tests/dispatch-run-uat-browser-tools.test.ts +88 -0
- package/src/resources/extensions/gsd/tests/doctor-scope-db-unavailable.test.ts +18 -0
- package/src/resources/extensions/gsd/tests/evidence-xref-gsd-exec.test.ts +157 -0
- package/src/resources/extensions/gsd/tests/execute-task-rendering.test.ts +1 -0
- package/src/resources/extensions/gsd/tests/file-change-validator.test.ts +33 -1
- package/src/resources/extensions/gsd/tests/fixtures/pr-body/swarm-lane-no-blockers.md +1 -5
- package/src/resources/extensions/gsd/tests/fixtures/pr-body/swarm-lane-with-blockers.md +1 -5
- package/src/resources/extensions/gsd/tests/gate-state-canonicalization.test.ts +48 -1
- package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +5 -4
- package/src/resources/extensions/gsd/tests/integration/auto-worktree.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/mcp-tool-name.test.ts +34 -0
- package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +143 -1
- package/src/resources/extensions/gsd/tests/milestone-closeout-proof.test.ts +99 -0
- package/src/resources/extensions/gsd/tests/milestone-closeout.test.ts +25 -0
- package/src/resources/extensions/gsd/tests/milestone-merge-transaction.test.ts +46 -0
- package/src/resources/extensions/gsd/tests/milestone-readiness.test.ts +65 -0
- package/src/resources/extensions/gsd/tests/milestone-validation-evidence.test.ts +41 -0
- package/src/resources/extensions/gsd/tests/milestone-validation-verdict.test.ts +55 -0
- package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +45 -0
- package/src/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +2 -0
- package/src/resources/extensions/gsd/tests/planning-crossval.test.ts +45 -0
- package/src/resources/extensions/gsd/tests/preferences-diagnostics.test.ts +67 -0
- package/src/resources/extensions/gsd/tests/preferences.test.ts +183 -0
- package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +46 -0
- package/src/resources/extensions/gsd/tests/provider-errors.test.ts +9 -0
- package/src/resources/extensions/gsd/tests/provider-payload-policy.test.ts +165 -0
- package/src/resources/extensions/gsd/tests/pull-request-process.test.ts +47 -0
- package/src/resources/extensions/gsd/tests/recovery-classification-illegal-transition.test.ts +30 -0
- package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +185 -1
- package/src/resources/extensions/gsd/tests/roadmap-parse-regression.test.ts +40 -0
- package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +25 -1
- package/src/resources/extensions/gsd/tests/safety-harness-false-positives.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +80 -0
- package/src/resources/extensions/gsd/tests/session-switch-clears-pending-autostart.test.ts +108 -0
- package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +144 -7
- package/src/resources/extensions/gsd/tests/stale-queued-milestone.test.ts +27 -0
- package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/state-transition-matrix.test.ts +36 -0
- package/src/resources/extensions/gsd/tests/status-guards.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/tool-availability-audit.test.ts +35 -0
- package/src/resources/extensions/gsd/tests/tool-naming.test.ts +35 -42
- package/src/resources/extensions/gsd/tests/uat-policy.test.ts +23 -0
- package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +47 -0
- package/src/resources/extensions/gsd/tests/user-input-boundary.test.ts +86 -1
- package/src/resources/extensions/gsd/tests/validate-milestone-stuck-guard.test.ts +39 -0
- package/src/resources/extensions/gsd/tests/web-app-uat.test.ts +150 -0
- package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +126 -9
- package/src/resources/extensions/gsd/tests/workspace-git-preflight.test.ts +15 -0
- package/src/resources/extensions/gsd/tests/worktree-lifecycle.test.ts +41 -4
- package/src/resources/extensions/gsd/tests/worktree-manager.test.ts +43 -1
- package/src/resources/extensions/gsd/tests/worktree-placement.test.ts +113 -0
- package/src/resources/extensions/gsd/tests/worktree-projection-writers.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/worktree-reentry.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/worktree-safety.test.ts +3 -1
- package/src/resources/extensions/gsd/tests/worktree-symlink-removal.test.ts +12 -6
- package/src/resources/extensions/gsd/tests/worktree-teardown-safety.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/worktree-telemetry.test.ts +22 -0
- package/src/resources/extensions/gsd/tests/write-gate.test.ts +121 -0
- package/src/resources/extensions/gsd/tool-contract.ts +86 -8
- package/src/resources/extensions/gsd/tool-presentation-plan.ts +16 -33
- package/src/resources/extensions/gsd/tool-surface-snapshot.ts +47 -0
- package/src/resources/extensions/gsd/tools/complete-slice.ts +23 -58
- package/src/resources/extensions/gsd/tools/exec-tool.ts +5 -5
- package/src/resources/extensions/gsd/tools/plan-milestone.ts +19 -160
- package/src/resources/extensions/gsd/tools/reassess-roadmap.ts +43 -0
- package/src/resources/extensions/gsd/tools/reopen-milestone.ts +11 -38
- package/src/resources/extensions/gsd/tools/reopen-slice.ts +14 -42
- package/src/resources/extensions/gsd/tools/skip-slice.ts +18 -44
- package/src/resources/extensions/gsd/tools/validate-milestone.ts +25 -84
- package/src/resources/extensions/gsd/uat-policy.ts +19 -10
- package/src/resources/extensions/gsd/uat-run.ts +10 -14
- package/src/resources/extensions/gsd/undo.ts +9 -8
- package/src/resources/extensions/gsd/unit-context-composer.ts +85 -20
- package/src/resources/extensions/gsd/unit-runtime.ts +3 -2
- package/src/resources/extensions/gsd/unit-tool-contracts.ts +2 -1
- package/src/resources/extensions/gsd/user-input-boundary.ts +18 -0
- package/src/resources/extensions/gsd/validation-block-guard.ts +2 -0
- package/src/resources/extensions/gsd/web-app-uat.ts +101 -0
- package/src/resources/extensions/gsd/workflow-mcp.ts +22 -110
- package/src/resources/extensions/gsd/workflow-reconcile.ts +3 -3
- package/src/resources/extensions/gsd/workflow-tool-surface.ts +73 -0
- package/src/resources/extensions/gsd/workspace-git-guard.ts +1 -0
- package/src/resources/extensions/gsd/worktree-git-recovery.ts +308 -0
- package/src/resources/extensions/gsd/worktree-lifecycle.ts +17 -17
- package/src/resources/extensions/gsd/worktree-manager.ts +47 -28
- package/src/resources/extensions/gsd/worktree-placement.ts +63 -0
- package/src/resources/extensions/gsd/worktree-reentry.ts +10 -7
- package/src/resources/extensions/gsd/worktree-root.ts +17 -6
- package/src/resources/extensions/gsd/worktree-safety.ts +8 -5
- package/src/resources/extensions/gsd/worktree-session-state.ts +12 -10
- package/src/resources/extensions/gsd/worktree-state-projection.ts +55 -7
- package/src/resources/extensions/gsd/worktree-telemetry.ts +16 -0
- package/src/resources/extensions/shared/interview-ui.ts +15 -2
- package/src/resources/shared/claude-runtime-floor.ts +248 -0
- package/src/resources/skills/gsd-browser/SKILL.md +1 -1
- package/dist/web/standalone/.next/static/chunks/2659.feb6499ca863ebfc.js +0 -1
- package/dist/web/standalone/.next/static/chunks/2772.151789db0edea835.js +0 -1
- package/dist/web/standalone/.next/static/chunks/4283.10a065467b5340d8.js +0 -2
- package/dist/web/standalone/.next/static/chunks/5826.960dc4634cc9b0d3.js +0 -1
- package/dist/web/standalone/.next/static/chunks/796.46f811c0fac23aab.js +0 -10
- package/dist/web/standalone/.next/static/chunks/8785.d32f7a61f55c1600.js +0 -1
- /package/dist/web/standalone/.next/static/{Qbr81pQ-pbQXP4bq2VXLv → C24pqUd-aru-l0Dp0gLZP}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{Qbr81pQ-pbQXP4bq2VXLv → C24pqUd-aru-l0Dp0gLZP}/_ssgManifest.js +0 -0
|
@@ -36,6 +36,8 @@ After the opening answer, classify project shape as **`simple`** or **`complex`*
|
|
|
36
36
|
|
|
37
37
|
Persist the verdict through `gsd_summary_save` with `artifact_type: "PROJECT"` into `## Project Shape`; downstream `discuss-requirements`, `discuss-milestone`, and `discuss-slice` read the rendered projection.
|
|
38
38
|
|
|
39
|
+
When the project is browser-facing (web app, SPA, static site with UI, Next/React/Vue, etc.), also record **`Web stack:`** under `## Project Shape` (framework + whether Playwright or browser UAT is expected). Downstream planning uses this to choose `browser-executable` / `runtime-executable` UAT instead of deferring UI proof to humans.
|
|
40
|
+
|
|
39
41
|
### Before deeper rounds
|
|
40
42
|
|
|
41
43
|
Ground your questions in the **Preparation Context snapshot above** (stack, structure, greenfield/brownfield and framework signals) plus any prior `.planning/` or `.gsd/` artifacts — those are authoritative. **Do not survey the codebase** with `rg`/`find`/`scout` before asking; read a specific file only when a question's answer genuinely hinges on it.
|
|
@@ -103,7 +105,7 @@ Once the user confirms depth:
|
|
|
103
105
|
|
|
104
106
|
1. Use the **Project** output template (inlined above).
|
|
105
107
|
2. Call `gsd_summary_save` with `artifact_type: "PROJECT"` and full project markdown as `content`; omit `milestone_id`. The tool persists the DB-backed PROJECT artifact and renders `.gsd/PROJECT.md`. Preserve the user's terms and framing.
|
|
106
|
-
3. The `## Project Shape` section MUST contain `**Complexity:** simple` or `**Complexity:** complex` (matching the verdict you announced) plus a one-line `**Why:**` rationale. Downstream stages read
|
|
108
|
+
3. The `## Project Shape` section MUST contain `**Complexity:** simple` or `**Complexity:** complex` (matching the verdict you announced) plus a one-line `**Why:**` rationale. For browser-facing projects, also include `**Web stack:**` (framework and expected browser/Playwright UAT approach). Downstream stages read these lines.
|
|
107
109
|
4. The `## Capability Contract` section MUST reference `.gsd/REQUIREMENTS.md` — that file does not yet exist; the next stage (`discuss-requirements`) will produce it.
|
|
108
110
|
5. The `## Milestone Sequence` MUST list at least M001 with title and one-liner. Subsequent milestones may be listed as known intents; they will be elaborated in their own discuss-milestone stages.
|
|
109
111
|
6. Do NOT use `artifact_type: "CONTEXT"` and do NOT pass `milestone_id: "PROJECT"`; that creates a fake milestone named PROJECT.
|
|
@@ -47,6 +47,8 @@ If milestone research is inlined, trust it and skip redundant exploration. If fi
|
|
|
47
47
|
|
|
48
48
|
Narrate decomposition reasoning in complete sentences: grouping, risk order, verification strategy.
|
|
49
49
|
|
|
50
|
+
**Web apps:** when inlined Web App UAT guidance is present, set milestone `Verification Classes` → UAT to browser-observable acceptance (Playwright spec or `browser_*` checks). Order an early slice to add Playwright smoke scaffolding when the dependency is missing.
|
|
51
|
+
|
|
50
52
|
Then:
|
|
51
53
|
1. Use the **Roadmap** output template from the inlined context above
|
|
52
54
|
2. {{skillActivation}}
|
|
@@ -41,6 +41,7 @@ If slice research is inlined, trust its architectural findings, but verify every
|
|
|
41
41
|
3. Use the inlined Output Template sections already present in this prompt. Do not read template files from disk.
|
|
42
42
|
4. {{skillActivation}} Record expected executor skills in each task plan's `skills_used` frontmatter.
|
|
43
43
|
5. Define slice verification before tasks. Non-trivial slices need real tests or executable assertions; boundary contracts need contract-exercising checks. Tests must not read .gitignore/gitignored paths such as `.gsd/`, `.planning/`, or `.audits/`.
|
|
44
|
+
**Web apps:** when inlined Web App UAT guidance is present, follow it — add Playwright smoke scaffolding on the first UI slice if missing, name localhost preconditions, and plan verification commands that match browser-capable UAT modes at slice closeout.
|
|
44
45
|
6. Include Threat Surface (Q3), Requirement Impact (Q4), proof level, observability, integration closure, Failure Modes (Q5), Load Profile (Q6), and Negative Tests (Q7) only where applicable.
|
|
45
46
|
7. Right-size tasks. Simple slices can be one task; split only when context, ownership, or verification boundaries justify it.
|
|
46
47
|
8. Task `verify` commands must be safe, simple commands. Do not use shell pipes, redirects, semicolons, backticks, command substitution, output trimming, or grep regex alternation with `|`. If multiple checks are needed, create a small test file and run it with `node --test` or a package test script, or use separate simple commands joined only with `&&`. For absence checks, verify a pattern does not exist with `! grep -q 'pattern' file` or `! rg -q 'pattern' file`; do not use `grep -c` or `rg -c` to assert zero matches because count commands exit 1 when they find zero matches, and the verification gate treats that as failure.
|
|
@@ -63,6 +63,7 @@ Then:
|
|
|
63
63
|
- `{{taskPlanTemplatePath}}`
|
|
64
64
|
2. {{skillActivation}} Record the installed skills you expect executors to use in each task plan's `skills_used` frontmatter.
|
|
65
65
|
3. Define slice-level verification: the objective stopping condition. Plan real test files with real assertions; for simple slices, executable commands are fine.
|
|
66
|
+
**Web apps:** when inlined Web App UAT guidance is present, follow it for Playwright scaffolding and browser-capable verification commands.
|
|
66
67
|
4. For non-trivial slices, plan observability / proof level / integration closure, threat surface, and requirement impact. Omit entirely for simple slices.
|
|
67
68
|
5. Decompose the slice into tasks that fit one context window each. Every task passed to `gsd_plan_slice` must use the exact keys `taskId`, `title`, `description`, `estimate`, `files`, `verify`, `inputs`, `expectedOutput`, and optional `observabilityImpact`. Put Why / Do / Done-when detail in `description`. `files`, `inputs`, and `expectedOutput` must be JSON arrays of strings, even for one path (for example, `"expectedOutput": ["src/index.ts"]`, never `"expectedOutput": "src/index.ts"`). `expectedOutput` is path-only: list only files the task creates or overwrites, and use `[]` for pure verification tasks.
|
|
68
69
|
6. **Persist planning state through `gsd_plan_slice`.** Call it with the full payload. The tool writes to the DB and renders `{{outputPath}}` and `{{slicePath}}/tasks/T##-PLAN.md` automatically. Do NOT rely on direct `PLAN.md` writes.
|
|
@@ -158,7 +158,7 @@ Fix root causes, not symptoms. If applying temporary mitigation, label it and pr
|
|
|
158
158
|
- State uncertainty plainly: "Not sure this handles X - testing it." No performed confidence, no hedging paragraphs.
|
|
159
159
|
- All user-visible narration must be grammatical English. Do not emit compressed planner notes like "Need inspect X". If it fits a commit comment or standup note, it is acceptable.
|
|
160
160
|
- When debugging, stay curious. Problems are puzzles. Say what's interesting about the failure before reaching for fixes.
|
|
161
|
-
- After completing a task, give a brief summary and 2-4 numbered next-step options; last option is always "Other". Omit the list for strict output formats.
|
|
161
|
+
- After completing a task, give a brief summary and 2-4 numbered next-step options; last option is always "Other". Omit the list for strict output formats, or when the active workflow prompt already ends with its own explicit "Next steps:" handoff block — in that case follow the workflow's handoff and do not add a second list.
|
|
162
162
|
|
|
163
163
|
If any next step is destructive/outward-facing, present it via `ask_user_questions` and wait for the user's answer before execution. Do not execute a next-step item from a prior plain-text numbered list without fresh confirmation.
|
|
164
164
|
|
|
@@ -0,0 +1,140 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Provider Payload Policy - ordered shaping of provider request payloads.
|
|
3
|
+
*
|
|
4
|
+
* The order is intentional:
|
|
5
|
+
* 1. observation budgeting masks old tool results in auto-mode,
|
|
6
|
+
* 2. display truncation caps tool-result text for every mode,
|
|
7
|
+
* 3. the protected Source Context Block is appended after truncation,
|
|
8
|
+
* 4. supported models receive the configured service tier.
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import type { ContextManagementConfig } from "./preferences-types.js";
|
|
12
|
+
import type { ServiceTierSetting } from "./service-tier.js";
|
|
13
|
+
|
|
14
|
+
import {
|
|
15
|
+
createObservationMask,
|
|
16
|
+
createResponsesInputObservationMask,
|
|
17
|
+
truncateContextResultMessages,
|
|
18
|
+
truncateResponsesInputResultItems,
|
|
19
|
+
} from "./context-masker.js";
|
|
20
|
+
import { getSourceObservationStore, isAutoActive } from "./auto-runtime-state.js";
|
|
21
|
+
import { loadEffectiveGSDPreferences } from "./preferences.js";
|
|
22
|
+
import { getEffectiveServiceTier, supportsServiceTier } from "./service-tier.js";
|
|
23
|
+
import { injectSourceContextBlockIntoPayload } from "./source-observations.js";
|
|
24
|
+
|
|
25
|
+
const DEFAULT_OBSERVATION_MASK_TURNS = 8;
|
|
26
|
+
const DEFAULT_TOOL_RESULT_MAX_CHARS = 800;
|
|
27
|
+
|
|
28
|
+
type MessagePayload = Parameters<ReturnType<typeof createObservationMask>>[0];
|
|
29
|
+
type ResponsesInputPayload = Parameters<ReturnType<typeof createResponsesInputObservationMask>>[0];
|
|
30
|
+
|
|
31
|
+
export interface ProviderPayloadPolicyDeps {
|
|
32
|
+
isAutoActive(): boolean;
|
|
33
|
+
loadContextManagementConfig(): ContextManagementConfig | undefined;
|
|
34
|
+
renderSourceContextBlock(): string | null;
|
|
35
|
+
getEffectiveServiceTier(): ServiceTierSetting;
|
|
36
|
+
supportsServiceTier(modelId: string): boolean;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
export interface ProviderPayloadPolicyInput {
|
|
40
|
+
payload: Record<string, unknown>;
|
|
41
|
+
modelId?: string;
|
|
42
|
+
deps?: Partial<ProviderPayloadPolicyDeps>;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
export const DEFAULT_PROVIDER_PAYLOAD_POLICY_DEPS: ProviderPayloadPolicyDeps = {
|
|
46
|
+
isAutoActive,
|
|
47
|
+
loadContextManagementConfig: () => loadEffectiveGSDPreferences()?.preferences.context_management,
|
|
48
|
+
renderSourceContextBlock: () => getSourceObservationStore().renderActiveBlock(),
|
|
49
|
+
getEffectiveServiceTier,
|
|
50
|
+
supportsServiceTier,
|
|
51
|
+
};
|
|
52
|
+
|
|
53
|
+
export function applyProviderPayloadPolicy({
|
|
54
|
+
payload,
|
|
55
|
+
modelId,
|
|
56
|
+
deps: overrides,
|
|
57
|
+
}: ProviderPayloadPolicyInput): Record<string, unknown> {
|
|
58
|
+
const deps = { ...DEFAULT_PROVIDER_PAYLOAD_POLICY_DEPS, ...overrides };
|
|
59
|
+
|
|
60
|
+
try {
|
|
61
|
+
applyContextManagement(payload, deps);
|
|
62
|
+
} catch {
|
|
63
|
+
// Provider payload shaping should not block a request when optional
|
|
64
|
+
// context management preferences or adapters fail.
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
try {
|
|
68
|
+
applySourceContextBlock(payload, deps);
|
|
69
|
+
} catch {
|
|
70
|
+
// Source observations are opportunistic; execution can continue without
|
|
71
|
+
// an injected block.
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
applyServiceTier(payload, modelId, deps);
|
|
75
|
+
return payload;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
function applyContextManagement(
|
|
79
|
+
payload: Record<string, unknown>,
|
|
80
|
+
deps: ProviderPayloadPolicyDeps,
|
|
81
|
+
): void {
|
|
82
|
+
const config = deps.loadContextManagementConfig();
|
|
83
|
+
applyObservationBudget(payload, config, deps.isAutoActive());
|
|
84
|
+
applyDisplayTruncation(payload, config);
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
function applyObservationBudget(
|
|
88
|
+
payload: Record<string, unknown>,
|
|
89
|
+
config: ContextManagementConfig | undefined,
|
|
90
|
+
autoActive: boolean,
|
|
91
|
+
): void {
|
|
92
|
+
if (!autoActive || config?.observation_masking === false) return;
|
|
93
|
+
|
|
94
|
+
const keepTurns = config?.observation_mask_turns ?? DEFAULT_OBSERVATION_MASK_TURNS;
|
|
95
|
+
if (Array.isArray(payload.messages)) {
|
|
96
|
+
payload.messages = createObservationMask(keepTurns)(payload.messages as MessagePayload);
|
|
97
|
+
}
|
|
98
|
+
if (Array.isArray(payload.input)) {
|
|
99
|
+
payload.input = createResponsesInputObservationMask(keepTurns)(payload.input as ResponsesInputPayload);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
function applyDisplayTruncation(
|
|
104
|
+
payload: Record<string, unknown>,
|
|
105
|
+
config: ContextManagementConfig | undefined,
|
|
106
|
+
): void {
|
|
107
|
+
const maxChars = config?.tool_result_max_chars ?? DEFAULT_TOOL_RESULT_MAX_CHARS;
|
|
108
|
+
|
|
109
|
+
if (Array.isArray(payload.messages)) {
|
|
110
|
+
payload.messages = truncateContextResultMessages(payload.messages as MessagePayload, maxChars);
|
|
111
|
+
}
|
|
112
|
+
if (Array.isArray(payload.input)) {
|
|
113
|
+
payload.input = truncateResponsesInputResultItems(payload.input as ResponsesInputPayload, maxChars);
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
function applySourceContextBlock(
|
|
118
|
+
payload: Record<string, unknown>,
|
|
119
|
+
deps: ProviderPayloadPolicyDeps,
|
|
120
|
+
): void {
|
|
121
|
+
if (!deps.isAutoActive()) return;
|
|
122
|
+
|
|
123
|
+
const sourceContextBlock = deps.renderSourceContextBlock();
|
|
124
|
+
if (!sourceContextBlock) return;
|
|
125
|
+
|
|
126
|
+
Object.assign(payload, injectSourceContextBlockIntoPayload(payload, sourceContextBlock));
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
function applyServiceTier(
|
|
130
|
+
payload: Record<string, unknown>,
|
|
131
|
+
modelId: string | undefined,
|
|
132
|
+
deps: ProviderPayloadPolicyDeps,
|
|
133
|
+
): void {
|
|
134
|
+
if (!modelId) return;
|
|
135
|
+
|
|
136
|
+
const tier = deps.getEffectiveServiceTier();
|
|
137
|
+
if (!tier || !deps.supportsServiceTier(modelId)) return;
|
|
138
|
+
|
|
139
|
+
payload.service_tier = tier;
|
|
140
|
+
}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
// Project/App: gsd-pi
|
|
2
|
+
// File Purpose: Process-level pull request policy for GSD-generated PRs.
|
|
3
|
+
|
|
4
|
+
import { createDraftPR } from "./git-service.js";
|
|
5
|
+
import {
|
|
6
|
+
buildPrEvidence,
|
|
7
|
+
type PrEvidence,
|
|
8
|
+
type PrEvidenceInput,
|
|
9
|
+
} from "./pr-evidence.js";
|
|
10
|
+
|
|
11
|
+
export interface DraftPullRequestOptions {
|
|
12
|
+
head: string;
|
|
13
|
+
base: string;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
export interface DraftPullRequestDeps {
|
|
17
|
+
createDraftPR: (
|
|
18
|
+
basePath: string,
|
|
19
|
+
milestoneId: string,
|
|
20
|
+
title: string,
|
|
21
|
+
body: string,
|
|
22
|
+
opts: DraftPullRequestOptions,
|
|
23
|
+
) => string | null;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export function buildPullRequestEvidence(input: PrEvidenceInput): PrEvidence {
|
|
27
|
+
return buildPrEvidence({
|
|
28
|
+
...input,
|
|
29
|
+
aiAssisted: false,
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
export function createDraftPullRequestFromEvidence(
|
|
34
|
+
basePath: string,
|
|
35
|
+
milestoneId: string,
|
|
36
|
+
evidence: PrEvidence,
|
|
37
|
+
options: DraftPullRequestOptions,
|
|
38
|
+
deps: DraftPullRequestDeps = { createDraftPR },
|
|
39
|
+
): string | null {
|
|
40
|
+
return deps.createDraftPR(basePath, milestoneId, evidence.title, evidence.body, options);
|
|
41
|
+
}
|
|
@@ -0,0 +1,140 @@
|
|
|
1
|
+
// Project/App: gsd-pi
|
|
2
|
+
// File Purpose: Canonical quality-gate closure from durable DB and artifact evidence.
|
|
3
|
+
|
|
4
|
+
import { existsSync, readFileSync } from "node:fs";
|
|
5
|
+
|
|
6
|
+
import { extractSection } from "./files.js";
|
|
7
|
+
import { getGateDefinition } from "./gate-registry.js";
|
|
8
|
+
import { getMilestoneSlices, getPendingGates, saveGateResult } from "./gsd-db.js";
|
|
9
|
+
import { resolveSliceFile, resolveTaskFile } from "./paths.js";
|
|
10
|
+
import type { GateId, GateRow, GateVerdict } from "./types.js";
|
|
11
|
+
|
|
12
|
+
export interface QualityGateClosureOptions {
|
|
13
|
+
artifactBasePath?: string;
|
|
14
|
+
milestoneValidationPassed?: boolean;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
export interface QualityGateClosureResult {
|
|
18
|
+
repaired: Array<{ gateId: GateId; sliceId: string; taskId?: string; verdict: GateVerdict }>;
|
|
19
|
+
unresolved: GateRow[];
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
interface GateEvidence {
|
|
23
|
+
verdict: GateVerdict;
|
|
24
|
+
rationale: string;
|
|
25
|
+
findings: string;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
const GATE_SECTION_HEADINGS: Partial<Record<GateId, string[]>> = {
|
|
29
|
+
Q3: ["Threat Surface", "Abuse Surface"],
|
|
30
|
+
Q4: ["Requirement Impact", "Broken Promises"],
|
|
31
|
+
Q5: ["Failure Modes"],
|
|
32
|
+
Q6: ["Load Profile"],
|
|
33
|
+
Q7: ["Negative Tests"],
|
|
34
|
+
Q8: ["Operational Readiness"],
|
|
35
|
+
};
|
|
36
|
+
|
|
37
|
+
function readFile(path: string | null): string | null {
|
|
38
|
+
if (!path || !existsSync(path)) return null;
|
|
39
|
+
return readFileSync(path, "utf-8");
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
function firstSection(content: string | null, gateId: GateId): string | null {
|
|
43
|
+
if (!content) return null;
|
|
44
|
+
for (const heading of GATE_SECTION_HEADINGS[gateId] ?? []) {
|
|
45
|
+
const section = extractSection(content, heading);
|
|
46
|
+
if (section) return section;
|
|
47
|
+
}
|
|
48
|
+
return null;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
function evidenceArtifactContent(row: GateRow, basePath: string): string | null {
|
|
52
|
+
const def = getGateDefinition(row.gate_id);
|
|
53
|
+
switch (def?.ownerTurn) {
|
|
54
|
+
case "gate-evaluate":
|
|
55
|
+
return readFile(resolveSliceFile(basePath, row.milestone_id, row.slice_id, "PLAN"));
|
|
56
|
+
case "execute-task":
|
|
57
|
+
return readFile(resolveTaskFile(basePath, row.milestone_id, row.slice_id, row.task_id, "SUMMARY"));
|
|
58
|
+
case "complete-slice":
|
|
59
|
+
return readFile(resolveSliceFile(basePath, row.milestone_id, row.slice_id, "SUMMARY"));
|
|
60
|
+
default:
|
|
61
|
+
return null;
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
function closureEvidence(row: GateRow, options: QualityGateClosureOptions): GateEvidence | null {
|
|
66
|
+
const def = getGateDefinition(row.gate_id);
|
|
67
|
+
if (!def) return null;
|
|
68
|
+
|
|
69
|
+
if (def.ownerTurn === "validate-milestone" && options.milestoneValidationPassed) {
|
|
70
|
+
return {
|
|
71
|
+
verdict: "pass",
|
|
72
|
+
rationale: `${def.promptSection} covered by passing milestone validation`,
|
|
73
|
+
findings: "",
|
|
74
|
+
};
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
if (!options.artifactBasePath) return null;
|
|
78
|
+
|
|
79
|
+
const section = firstSection(evidenceArtifactContent(row, options.artifactBasePath), row.gate_id);
|
|
80
|
+
if (section) {
|
|
81
|
+
return {
|
|
82
|
+
verdict: "pass",
|
|
83
|
+
rationale: `${def.promptSection} evidence found in durable artifact`,
|
|
84
|
+
findings: section,
|
|
85
|
+
};
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
if (!options.milestoneValidationPassed) return null;
|
|
89
|
+
return {
|
|
90
|
+
verdict: "omitted",
|
|
91
|
+
rationale: `${def.promptSection} has no durable artifact section; milestone validation passed`,
|
|
92
|
+
findings: "",
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
function closeGate(row: GateRow, evidence: GateEvidence): void {
|
|
97
|
+
saveGateResult({
|
|
98
|
+
milestoneId: row.milestone_id,
|
|
99
|
+
sliceId: row.slice_id,
|
|
100
|
+
gateId: row.gate_id,
|
|
101
|
+
taskId: row.task_id,
|
|
102
|
+
verdict: evidence.verdict,
|
|
103
|
+
rationale: evidence.rationale,
|
|
104
|
+
findings: evidence.findings,
|
|
105
|
+
});
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
export function closeQualityGatesFromEvidence(
|
|
109
|
+
milestoneId: string,
|
|
110
|
+
options: QualityGateClosureOptions = {},
|
|
111
|
+
): QualityGateClosureResult {
|
|
112
|
+
const repaired: QualityGateClosureResult["repaired"] = [];
|
|
113
|
+
const unresolved: GateRow[] = [];
|
|
114
|
+
|
|
115
|
+
for (const slice of getMilestoneSlices(milestoneId)) {
|
|
116
|
+
const sliceId = slice.id;
|
|
117
|
+
for (const row of getPendingGates(milestoneId, sliceId)) {
|
|
118
|
+
if (!getGateDefinition(row.gate_id)) {
|
|
119
|
+
unresolved.push(row);
|
|
120
|
+
continue;
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
const evidence = closureEvidence(row, options);
|
|
124
|
+
if (!evidence) {
|
|
125
|
+
unresolved.push(row);
|
|
126
|
+
continue;
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
closeGate(row, evidence);
|
|
130
|
+
repaired.push({
|
|
131
|
+
gateId: row.gate_id,
|
|
132
|
+
sliceId: row.slice_id,
|
|
133
|
+
...(row.task_id ? { taskId: row.task_id } : {}),
|
|
134
|
+
verdict: evidence.verdict,
|
|
135
|
+
});
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
return { repaired, unresolved };
|
|
140
|
+
}
|
|
@@ -0,0 +1,138 @@
|
|
|
1
|
+
// Project/App: gsd-pi
|
|
2
|
+
// File Purpose: Resolve how workflow units should ask the user for input.
|
|
3
|
+
|
|
4
|
+
import { parseMcpToolName, toMcpToolName } from "./mcp-tool-name.js";
|
|
5
|
+
|
|
6
|
+
export type StructuredQuestionsFlag = "true" | "false";
|
|
7
|
+
|
|
8
|
+
export interface QuestionTransportOptions {
|
|
9
|
+
activeTools: string[];
|
|
10
|
+
authMode?: "apiKey" | "oauth" | "externalCli" | "none";
|
|
11
|
+
baseUrl?: string;
|
|
12
|
+
env?: NodeJS.ProcessEnv;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export interface QuestionTransportResolution {
|
|
16
|
+
structuredQuestionsAvailable: StructuredQuestionsFlag;
|
|
17
|
+
questionToolAvailable: boolean;
|
|
18
|
+
usesWorkflowMcp: boolean;
|
|
19
|
+
reason: "question-tool-available" | "question-tool-missing" | "workflow-mcp-disabled";
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export interface WorkflowQuestionToolSurfaceOptions {
|
|
23
|
+
workflowServerName?: string;
|
|
24
|
+
workflowExplicitlyBlocked?: boolean;
|
|
25
|
+
workflowMcpTools: string[];
|
|
26
|
+
exactWorkflowMcpTools: string[];
|
|
27
|
+
env?: NodeJS.ProcessEnv;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export interface WorkflowQuestionToolSurfaceResolution extends QuestionTransportResolution {
|
|
31
|
+
questionToolName?: string;
|
|
32
|
+
workflowQuestionsEnabled: boolean;
|
|
33
|
+
disallowedTools: string[];
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
function isWorkflowMcpServerName(serverName: string): boolean {
|
|
37
|
+
const normalized = serverName.toLowerCase();
|
|
38
|
+
return normalized === "gsd" || normalized.includes("workflow");
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
export function usesWorkflowMcpTransport(
|
|
42
|
+
authMode: QuestionTransportOptions["authMode"],
|
|
43
|
+
baseUrl: string | undefined,
|
|
44
|
+
): boolean {
|
|
45
|
+
return authMode === "externalCli" && typeof baseUrl === "string" && baseUrl.startsWith("local://");
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
export function hasAskUserQuestionsTool(activeTools: string[]): boolean {
|
|
49
|
+
return activeTools.some((toolName) => {
|
|
50
|
+
if (toolName === "ask_user_questions") return true;
|
|
51
|
+
const mcp = parseMcpToolName(toolName);
|
|
52
|
+
if (!mcp) return false;
|
|
53
|
+
if (mcp.toolName === "ask_user_questions") return true;
|
|
54
|
+
return mcp.toolName === "*" && isWorkflowMcpServerName(mcp.serverName);
|
|
55
|
+
});
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
function workflowMcpStructuredQuestionsEnabled(env: NodeJS.ProcessEnv = process.env): boolean {
|
|
59
|
+
const value = env.GSD_WORKFLOW_MCP_STRUCTURED_QUESTIONS?.trim().toLowerCase();
|
|
60
|
+
return value !== "0" && value !== "false" && value !== "off";
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
export function resolveQuestionTransport(
|
|
64
|
+
options: QuestionTransportOptions,
|
|
65
|
+
): QuestionTransportResolution {
|
|
66
|
+
const questionToolAvailable = hasAskUserQuestionsTool(options.activeTools);
|
|
67
|
+
const usesWorkflowMcp = usesWorkflowMcpTransport(options.authMode, options.baseUrl);
|
|
68
|
+
|
|
69
|
+
if (!questionToolAvailable) {
|
|
70
|
+
return {
|
|
71
|
+
structuredQuestionsAvailable: "false",
|
|
72
|
+
questionToolAvailable,
|
|
73
|
+
usesWorkflowMcp,
|
|
74
|
+
reason: "question-tool-missing",
|
|
75
|
+
};
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
if (options.authMode === "externalCli" && !workflowMcpStructuredQuestionsEnabled(options.env)) {
|
|
79
|
+
return {
|
|
80
|
+
structuredQuestionsAvailable: "false",
|
|
81
|
+
questionToolAvailable,
|
|
82
|
+
usesWorkflowMcp,
|
|
83
|
+
reason: "workflow-mcp-disabled",
|
|
84
|
+
};
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
return {
|
|
88
|
+
structuredQuestionsAvailable: "true",
|
|
89
|
+
questionToolAvailable,
|
|
90
|
+
usesWorkflowMcp,
|
|
91
|
+
reason: "question-tool-available",
|
|
92
|
+
};
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
export function supportsStructuredQuestions(
|
|
96
|
+
activeTools: string[],
|
|
97
|
+
options: Omit<QuestionTransportOptions, "activeTools"> = {},
|
|
98
|
+
): boolean {
|
|
99
|
+
return resolveQuestionTransport({
|
|
100
|
+
...options,
|
|
101
|
+
activeTools,
|
|
102
|
+
}).structuredQuestionsAvailable === "true";
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
export function resolveWorkflowQuestionToolSurface(
|
|
106
|
+
options: WorkflowQuestionToolSurfaceOptions,
|
|
107
|
+
): WorkflowQuestionToolSurfaceResolution {
|
|
108
|
+
const questionToolName = options.workflowServerName && !options.workflowExplicitlyBlocked
|
|
109
|
+
? toMcpToolName(options.workflowServerName, "ask_user_questions")
|
|
110
|
+
: undefined;
|
|
111
|
+
const activeTools = [
|
|
112
|
+
...options.exactWorkflowMcpTools,
|
|
113
|
+
...options.workflowMcpTools,
|
|
114
|
+
...(questionToolName ? [questionToolName] : []),
|
|
115
|
+
];
|
|
116
|
+
const transport = resolveQuestionTransport({
|
|
117
|
+
activeTools,
|
|
118
|
+
authMode: "externalCli",
|
|
119
|
+
baseUrl: "local://claude-code",
|
|
120
|
+
env: options.env,
|
|
121
|
+
});
|
|
122
|
+
const exactQuestionToolAllowed =
|
|
123
|
+
!!questionToolName && options.exactWorkflowMcpTools.includes(questionToolName);
|
|
124
|
+
const workflowQuestionsEnabled =
|
|
125
|
+
transport.structuredQuestionsAvailable === "true" &&
|
|
126
|
+
(options.workflowMcpTools.length > 0 || exactQuestionToolAllowed);
|
|
127
|
+
const disallowedTools =
|
|
128
|
+
options.workflowServerName && transport.questionToolAvailable && !workflowQuestionsEnabled
|
|
129
|
+
? [toMcpToolName(options.workflowServerName, "ask_user_questions")]
|
|
130
|
+
: [];
|
|
131
|
+
|
|
132
|
+
return {
|
|
133
|
+
...transport,
|
|
134
|
+
questionToolName,
|
|
135
|
+
workflowQuestionsEnabled,
|
|
136
|
+
disallowedTools,
|
|
137
|
+
};
|
|
138
|
+
}
|
|
@@ -3,6 +3,7 @@
|
|
|
3
3
|
|
|
4
4
|
import { classifyError, isTransient, type ErrorClass } from "./error-classifier.js";
|
|
5
5
|
import { ReconciliationFailedError } from "./state-reconciliation.js";
|
|
6
|
+
import { IllegalPhaseTransitionError } from "./state-transition-matrix.js";
|
|
6
7
|
|
|
7
8
|
export type RecoveryFailureKind =
|
|
8
9
|
| "tool-schema"
|
|
@@ -13,6 +14,7 @@ export type RecoveryFailureKind =
|
|
|
13
14
|
| "worktree-invalid"
|
|
14
15
|
| "verification-drift"
|
|
15
16
|
| "reconciliation-drift"
|
|
17
|
+
| "illegal-transition"
|
|
16
18
|
| "provider"
|
|
17
19
|
| "runtime-unknown";
|
|
18
20
|
|
|
@@ -43,7 +45,9 @@ export function classifyFailure(input: RecoveryClassificationInput): RecoveryCla
|
|
|
43
45
|
const failureKind =
|
|
44
46
|
input.error instanceof ReconciliationFailedError
|
|
45
47
|
? "reconciliation-drift"
|
|
46
|
-
: input.
|
|
48
|
+
: input.error instanceof IllegalPhaseTransitionError
|
|
49
|
+
? "illegal-transition"
|
|
50
|
+
: input.failureKind ?? inferFailureKind(message);
|
|
47
51
|
|
|
48
52
|
switch (failureKind) {
|
|
49
53
|
case "tool-schema":
|
|
@@ -111,6 +115,15 @@ export function classifyFailure(input: RecoveryClassificationInput): RecoveryCla
|
|
|
111
115
|
remediation:
|
|
112
116
|
"Inspect the persistent or repair-failed drift kinds reported by the State Reconciliation Module before resuming.",
|
|
113
117
|
};
|
|
118
|
+
case "illegal-transition":
|
|
119
|
+
return {
|
|
120
|
+
failureKind,
|
|
121
|
+
action: "escalate",
|
|
122
|
+
reason: `Illegal phase transition${unitSuffix(input)}: ${message}`,
|
|
123
|
+
exitReason: "illegal-transition",
|
|
124
|
+
remediation:
|
|
125
|
+
"A derived Phase edge rejected by the Phase Transition Invariant survived reconciliation; inspect deriveState and the State Reconciliation Module before resuming.",
|
|
126
|
+
};
|
|
114
127
|
case "provider": {
|
|
115
128
|
const providerClass = classifyError(message, input.retryAfterMs);
|
|
116
129
|
return {
|
|
@@ -189,9 +189,15 @@ export function parseRoadmapSlices(content: string): RoadmapSliceEntry[] {
|
|
|
189
189
|
const risk = (riskMatch ? riskMatch[1] : "low") as RiskLevel;
|
|
190
190
|
|
|
191
191
|
const depsMatch = rest.match(/`depends:\[([^\]]*)\]`/);
|
|
192
|
-
|
|
193
|
-
|
|
192
|
+
// Recovery fallback: double-bracket form `[[id]]` from serialized bracket-wrapped IDs
|
|
193
|
+
const fallbackDepsMatch = depsMatch ? null : rest.match(/`depends:\[(\[(?:[^\]]*)\](?:,\[(?:[^\]]*)\])*)\]`/);
|
|
194
|
+
const rawDepContent = (depsMatch ?? fallbackDepsMatch)?.[1] ?? "";
|
|
195
|
+
const SLICE_ID_RE = /^[A-Za-z0-9][A-Za-z0-9-]*$/;
|
|
196
|
+
const RANGE_RE = /^[A-Za-z]+\d+(?:-|\.\.)[A-Za-z]+\d+$/;
|
|
197
|
+
const rawDepParts = rawDepContent.trim()
|
|
198
|
+
? rawDepContent.replace(/\[|\]/g, "").split(",").map(s => s.trim()).filter(s => SLICE_ID_RE.test(s) || RANGE_RE.test(s))
|
|
194
199
|
: [];
|
|
200
|
+
const depends = expandDependencies(rawDepParts);
|
|
195
201
|
|
|
196
202
|
// ADR-011: the renderer writes a `[sketch]` badge for sketch slices.
|
|
197
203
|
// Parse it back so the is_sketch flag survives a markdown → DB re-import
|
|
@@ -57,9 +57,10 @@ const EXECUTION_TOOL_NAMES = new Set([
|
|
|
57
57
|
"functions.exec_command",
|
|
58
58
|
"gsd_exec",
|
|
59
59
|
"gsd_exec_search",
|
|
60
|
+
"gsd_uat_exec",
|
|
60
61
|
"powershell",
|
|
61
62
|
]);
|
|
62
|
-
const MCP_EXECUTION_TOOL_RE = /^mcp__.+
|
|
63
|
+
const MCP_EXECUTION_TOOL_RE = /^mcp__.+__gsd_(?:uat_)?exec(?:_search)?$/;
|
|
63
64
|
|
|
64
65
|
// ─── Module State ───────────────────────────────────────────────────────────
|
|
65
66
|
|
|
@@ -206,11 +207,17 @@ export function clearEvidenceFromDisk(
|
|
|
206
207
|
* Exit codes and output are filled in by recordToolResult after execution.
|
|
207
208
|
*/
|
|
208
209
|
export function recordToolCall(toolCallId: string, toolName: string, input: Record<string, unknown>): void {
|
|
210
|
+
// Idempotent by toolCallId: native tools reach this via both
|
|
211
|
+
// tool_execution_start and tool_call; external (pre-executed) tools only
|
|
212
|
+
// via tool_execution_start. First recording wins.
|
|
213
|
+
if (unitEvidence.some(e => e.toolCallId === toolCallId)) return;
|
|
209
214
|
if (isExecutionToolName(toolName)) {
|
|
210
215
|
unitEvidence.push({
|
|
211
216
|
kind: "bash",
|
|
212
217
|
toolCallId,
|
|
213
|
-
|
|
218
|
+
// gsd_exec / gsd_uat_exec carry the script body in `script` (or `code`);
|
|
219
|
+
// bash-style tools use `command`/`cmd`; gsd_exec_search uses `query`.
|
|
220
|
+
command: String(input.command ?? input.script ?? input.cmd ?? input.code ?? input.query ?? ""),
|
|
214
221
|
exitCode: -1,
|
|
215
222
|
outputSnippet: "",
|
|
216
223
|
timestamp: Date.now(),
|
|
@@ -249,11 +256,36 @@ export function recordToolResult(
|
|
|
249
256
|
if (entry.kind === "bash") {
|
|
250
257
|
const text = extractResultText(result);
|
|
251
258
|
entry.outputSnippet = text.slice(0, 500);
|
|
252
|
-
|
|
253
|
-
entry.exitCode = exitMatch ? Number(exitMatch[1]) : (isError ? 1 : 0);
|
|
259
|
+
entry.exitCode = resolveExitCode(text, isError);
|
|
254
260
|
}
|
|
255
261
|
}
|
|
256
262
|
|
|
263
|
+
/**
|
|
264
|
+
* Resolve the exit code from a tool result's text. Handles the bash tool's
|
|
265
|
+
* prose marker, the gsd_exec / gsd_uat_exec JSON envelope (`"exit_code": N`),
|
|
266
|
+
* and a last-resort read of the run's persisted `.gsd/exec/<id>.meta.json`
|
|
267
|
+
* (covers truncated result text).
|
|
268
|
+
*/
|
|
269
|
+
function resolveExitCode(text: string, isError: boolean): number {
|
|
270
|
+
const proseMatch = text.match(/Command exited with code (\d+)/);
|
|
271
|
+
if (proseMatch) return Number(proseMatch[1]);
|
|
272
|
+
|
|
273
|
+
const jsonMatch = text.match(/"exit_code"\s*:\s*(-?\d+)/);
|
|
274
|
+
if (jsonMatch) return Number(jsonMatch[1]);
|
|
275
|
+
|
|
276
|
+
const metaMatch = text.match(/"meta_path"\s*:\s*"([^"]+)"/);
|
|
277
|
+
if (metaMatch) {
|
|
278
|
+
try {
|
|
279
|
+
const meta = JSON.parse(readFileSync(metaMatch[1], "utf-8")) as Record<string, unknown>;
|
|
280
|
+
if (typeof meta.exit_code === "number") return meta.exit_code;
|
|
281
|
+
} catch {
|
|
282
|
+
// Fall through to the isError heuristic
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
|
|
286
|
+
return isError ? 1 : 0;
|
|
287
|
+
}
|
|
288
|
+
|
|
257
289
|
// ─── Internals ──────────────────────────────────────────────────────────────
|
|
258
290
|
|
|
259
291
|
function extractResultText(result: unknown): string {
|
|
@@ -121,9 +121,14 @@ function findMatches(
|
|
|
121
121
|
const exact = bashCalls.filter(b => b.command.trim() === normalized);
|
|
122
122
|
if (exact.length > 0) return exact;
|
|
123
123
|
|
|
124
|
-
// Substring match: claimed is contained in actual or actual in claimed
|
|
124
|
+
// Substring match: claimed is contained in actual or actual in claimed.
|
|
125
|
+
// A claimed verification command typically appears verbatim inside a
|
|
126
|
+
// larger gsd_exec script body (cd prefix, multi-line scripts), so
|
|
127
|
+
// script-containing-claim is the common direction. Blank-command entries
|
|
128
|
+
// must be excluded — `"x".includes("")` is true, so they'd match anything.
|
|
125
129
|
const substring = bashCalls.filter(
|
|
126
|
-
b => b.command.
|
|
130
|
+
b => b.command.trim().length > 0 &&
|
|
131
|
+
(b.command.includes(normalized) || normalized.includes(b.command)),
|
|
127
132
|
);
|
|
128
133
|
if (substring.length > 0) return substring;
|
|
129
134
|
|