@opengsd/gsd-pi 1.2.0-dev.b1abb545 → 1.2.0-dev.e8563f58
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli-style.d.ts +17 -0
- package/dist/cli-style.js +28 -0
- package/dist/cli.js +1 -1
- package/dist/headless-events.d.ts +4 -2
- package/dist/headless-events.js +14 -34
- package/dist/mcp-server.js +2 -1
- package/dist/models-resolver.d.ts +3 -13
- package/dist/models-resolver.js +3 -22
- package/dist/resource-loader.js +2 -14
- package/dist/resources/.managed-resources-content-hash +1 -1
- package/dist/resources/GSD-WORKFLOW.md +5 -4
- package/dist/resources/extensions/async-jobs/async-bash-tool.js +30 -64
- package/dist/resources/extensions/async-jobs/await-tool.js +80 -12
- package/dist/resources/extensions/async-jobs/index.js +65 -0
- package/dist/resources/extensions/async-jobs/job-manager.js +12 -1
- package/dist/resources/extensions/bg-shell/bg-shell-command.js +6 -6
- package/dist/resources/extensions/bg-shell/bg-shell-tool.js +10 -7
- package/dist/resources/extensions/bg-shell/overlay.js +9 -6
- package/dist/resources/extensions/bg-shell/process-manager.js +54 -25
- package/dist/resources/extensions/bg-shell/readiness-detector.js +11 -0
- package/dist/resources/extensions/bg-shell/utilities.js +5 -2
- package/dist/resources/extensions/browser-tools/engine/managed-gsd-browser.js +209 -88
- package/dist/resources/extensions/browser-tools/engine/selection.js +73 -5
- package/dist/resources/extensions/browser-tools/index.js +69 -12
- package/dist/resources/extensions/claude-code-cli/models.js +9 -0
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +38 -6
- package/dist/resources/extensions/gsd/auto/custom-verify-retry-store.js +17 -2
- package/dist/resources/extensions/gsd/auto/detect-stuck.js +33 -13
- package/dist/resources/extensions/gsd/auto/dispatch-history.js +105 -0
- package/dist/resources/extensions/gsd/auto/dispatch-key.js +37 -0
- package/dist/resources/extensions/gsd/auto/loop.js +4 -1
- package/dist/resources/extensions/gsd/auto/orchestrator.js +122 -58
- package/dist/resources/extensions/gsd/auto/phases.js +8 -3
- package/dist/resources/extensions/gsd/auto-direct-dispatch.js +8 -32
- package/dist/resources/extensions/gsd/auto-dispatch.js +40 -57
- package/dist/resources/extensions/gsd/auto-model-selection.js +36 -13
- package/dist/resources/extensions/gsd/auto-post-unit.js +31 -14
- package/dist/resources/extensions/gsd/auto-prompts.js +81 -19
- package/dist/resources/extensions/gsd/auto-start.js +24 -26
- package/dist/resources/extensions/gsd/auto-tool-tracking.js +18 -0
- package/dist/resources/extensions/gsd/auto-unit-tool-scope.js +12 -20
- package/dist/resources/extensions/gsd/auto-verification.js +9 -28
- package/dist/resources/extensions/gsd/auto-worktree-repair.js +10 -2
- package/dist/resources/extensions/gsd/auto-worktree.js +35 -352
- package/dist/resources/extensions/gsd/auto.js +15 -20
- package/dist/resources/extensions/gsd/blocked-models.js +28 -0
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +29 -8
- package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +32 -12
- package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +2 -2
- package/dist/resources/extensions/gsd/bootstrap/register-extension.js +19 -0
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +229 -36
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +319 -71
- package/dist/resources/extensions/gsd/branch-patterns.js +2 -0
- package/dist/resources/extensions/gsd/browser-daemon-auto-prep.js +83 -0
- package/dist/resources/extensions/gsd/browser-evidence.js +8 -2
- package/dist/resources/extensions/gsd/captures.js +5 -15
- package/dist/resources/extensions/gsd/closeout-recovery.js +3 -2
- package/dist/resources/extensions/gsd/closeout-wizard.js +92 -0
- package/dist/resources/extensions/gsd/commands/catalog.js +6 -62
- package/dist/resources/extensions/gsd/commands-handlers.js +46 -3
- package/dist/resources/extensions/gsd/consent-question.js +353 -0
- package/dist/resources/extensions/gsd/consent-verdict.js +63 -0
- package/dist/resources/extensions/gsd/constants.js +0 -2
- package/dist/resources/extensions/gsd/crash-recovery.js +4 -12
- package/dist/resources/extensions/gsd/db/engine.js +755 -0
- package/dist/resources/extensions/gsd/db/queries.js +398 -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/dispatch-guard.js +10 -35
- package/dist/resources/extensions/gsd/doctor-environment.js +5 -11
- package/dist/resources/extensions/gsd/doctor-format.js +9 -6
- package/dist/resources/extensions/gsd/doctor-git-checks.js +6 -21
- package/dist/resources/extensions/gsd/doctor-runtime-checks.js +21 -16
- package/dist/resources/extensions/gsd/engine-hook-contract.js +70 -0
- package/dist/resources/extensions/gsd/error-classifier.js +9 -0
- package/dist/resources/extensions/gsd/exec-sandbox.js +30 -10
- package/dist/resources/extensions/gsd/files.js +33 -19
- 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-command-home.js +22 -12
- package/dist/resources/extensions/gsd/gsd-db.js +172 -2048
- package/dist/resources/extensions/gsd/guidance.js +158 -0
- package/dist/resources/extensions/gsd/guided-flow.js +51 -5
- package/dist/resources/extensions/gsd/markdown-renderer.js +10 -0
- package/dist/resources/extensions/gsd/mcp-filter.js +2 -19
- package/dist/resources/extensions/gsd/mcp-tool-name.js +5 -13
- package/dist/resources/extensions/gsd/memory-consolidation-scanner.js +1 -1
- package/dist/resources/extensions/gsd/migrate/safety.js +20 -9
- package/dist/resources/extensions/gsd/migration-auto-check.js +24 -3
- package/dist/resources/extensions/gsd/milestone-closeout.js +85 -24
- package/dist/resources/extensions/gsd/model-cost-table.js +1 -0
- package/dist/resources/extensions/gsd/model-router.js +3 -0
- package/dist/resources/extensions/gsd/notification-store.js +11 -4
- package/dist/resources/extensions/gsd/parallel-merge.js +14 -11
- package/dist/resources/extensions/gsd/parallel-monitor-overlay.js +11 -7
- package/dist/resources/extensions/gsd/parsers-legacy.js +16 -4
- package/dist/resources/extensions/gsd/paths.js +37 -24
- package/dist/resources/extensions/gsd/pre-execution-checks.js +91 -3
- package/dist/resources/extensions/gsd/preferences-models.js +14 -48
- package/dist/resources/extensions/gsd/preferences.js +14 -0
- package/dist/resources/extensions/gsd/prompts/complete-slice.md +2 -2
- package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/refine-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/replan-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/run-uat.md +6 -4
- package/dist/resources/extensions/gsd/prompts/system.md +5 -2
- package/dist/resources/extensions/gsd/provider-error-guidance.js +1 -5
- package/dist/resources/extensions/gsd/provider-switch-observer.js +1 -1
- package/dist/resources/extensions/gsd/publication.js +87 -0
- package/dist/resources/extensions/gsd/reactive-graph.js +8 -1
- package/dist/resources/extensions/gsd/recovery-classification.js +41 -87
- package/dist/resources/extensions/gsd/safety/destructive-confirmation.js +108 -0
- package/dist/resources/extensions/gsd/safety/evidence-collector.js +37 -4
- package/dist/resources/extensions/gsd/safety/evidence-cross-ref.js +7 -2
- package/dist/resources/extensions/gsd/safety/file-change-validator.js +10 -0
- package/dist/resources/extensions/gsd/state-transition-matrix.js +38 -0
- package/dist/resources/extensions/gsd/state.js +6 -20
- package/dist/resources/extensions/gsd/status-guards.js +56 -8
- package/dist/resources/extensions/gsd/stop-notice.js +57 -0
- package/dist/resources/extensions/gsd/tool-presentation-plan.js +4 -4
- package/dist/resources/extensions/gsd/tool-surface-readiness.js +56 -0
- package/dist/resources/extensions/gsd/tools/complete-slice.js +44 -53
- package/dist/resources/extensions/gsd/tools/exec-tool.js +10 -8
- package/dist/resources/extensions/gsd/tools/plan-slice.js +12 -6
- package/dist/resources/extensions/gsd/tools/reopen-milestone.js +11 -29
- package/dist/resources/extensions/gsd/tools/reopen-slice.js +14 -33
- package/dist/resources/extensions/gsd/tools/skip-slice.js +18 -36
- package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +67 -2
- package/dist/resources/extensions/gsd/uat-policy.js +42 -16
- package/dist/resources/extensions/gsd/undo.js +8 -7
- package/dist/resources/extensions/gsd/unit-closeout.js +138 -0
- package/dist/resources/extensions/gsd/unit-context-composer.js +74 -1
- package/dist/resources/extensions/gsd/unit-context-manifest.js +4 -27
- package/dist/resources/extensions/gsd/unit-registry.js +337 -0
- package/dist/resources/extensions/gsd/unit-tool-contracts.js +9 -182
- package/dist/resources/extensions/gsd/verdict-parser.js +1 -1
- package/dist/resources/extensions/gsd/web-app-uat.js +45 -8
- package/dist/resources/extensions/gsd/workflow-tool-surface.js +1 -1
- package/dist/resources/extensions/gsd/worktree-git-recovery.js +293 -0
- package/dist/resources/extensions/gsd/worktree-lifecycle.js +12 -3
- package/dist/resources/extensions/gsd/worktree-manager.js +45 -28
- package/dist/resources/extensions/gsd/worktree-placement.js +59 -0
- package/dist/resources/extensions/gsd/worktree-reentry.js +12 -8
- package/dist/resources/extensions/gsd/worktree-root.js +28 -6
- package/dist/resources/extensions/gsd/worktree-safety.js +8 -5
- package/dist/resources/extensions/gsd/worktree-session-state.js +12 -11
- package/dist/resources/extensions/search-the-web/native-search.js +5 -3
- package/dist/resources/extensions/shared/browser-contract.js +59 -0
- package/dist/resources/extensions/shared/gsd-browser-cli.js +116 -6
- package/dist/resources/shared/gsd-browser-path-sync.js +214 -0
- package/dist/resources/shared/package-manager-detection.js +1 -1
- package/dist/resources/shared/package.json +3 -0
- package/dist/resources/skills/create-skill/references/executable-code.md +1 -1
- package/dist/resources/skills/create-skill/workflows/add-reference.md +8 -3
- package/dist/resources/skills/create-skill/workflows/add-script.md +4 -2
- package/dist/resources/skills/create-skill/workflows/add-template.md +3 -1
- package/dist/resources/skills/create-skill/workflows/add-workflow.md +8 -3
- package/dist/resources/skills/create-skill/workflows/upgrade-to-router.md +10 -5
- package/dist/resources/skills/create-skill/workflows/verify-skill.md +9 -4
- package/dist/resources/skills/gsd-browser/SKILL.md +1 -1
- package/dist/resources/skills/spike-wrap-up/SKILL.md +9 -9
- package/dist/tsconfig.extensions.tsbuildinfo +1 -1
- package/dist/update-check.d.ts +2 -0
- package/dist/update-check.js +24 -1
- package/dist/update-cmd.js +20 -3
- 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 +3 -3
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.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/update/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/update/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 +8 -8
- package/dist/web/standalone/.next/server/chunks/5124.js +1 -1
- package/dist/web/standalone/.next/server/chunks/{5047.js → 5942.js} +2 -2
- package/dist/web/standalone/.next/server/chunks/8357.js +2 -2
- 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/{796.cf859a427a2cb2ac.js → 796.e0bdc932325d7e03.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/{webpack-fbea77b5f9953368.js → webpack-f0285ce91d4ec9ef.js} +1 -1
- package/dist/web/standalone/node_modules/node-pty/build/Makefile +1 -1
- package/dist/web/standalone/package.json +1 -1
- package/dist/worktree-cli.js +3 -6
- package/dist/worktree-status-banner.js +7 -11
- package/package.json +1 -1
- package/packages/cloud-mcp-gateway/package.json +2 -2
- package/packages/contracts/dist/rpc.d.ts +1 -0
- package/packages/contracts/dist/rpc.d.ts.map +1 -1
- package/packages/contracts/dist/rpc.js.map +1 -1
- package/packages/contracts/dist/workflow.d.ts +4 -0
- package/packages/contracts/dist/workflow.d.ts.map +1 -1
- package/packages/contracts/dist/workflow.js.map +1 -1
- package/packages/contracts/package.json +1 -1
- package/packages/daemon/package.json +4 -4
- package/packages/gsd-agent-core/package.json +5 -5
- package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts +5 -0
- 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 +8 -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 +7 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.js +8 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.js +11 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.js +4 -4
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js +3 -1
- package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js.map +1 -1
- package/packages/gsd-agent-modes/package.json +7 -7
- package/packages/mcp-server/dist/cli.js +6 -3
- package/packages/mcp-server/dist/cli.js.map +1 -1
- package/packages/mcp-server/dist/moonshot-tool-schema.d.ts +29 -0
- package/packages/mcp-server/dist/moonshot-tool-schema.d.ts.map +1 -0
- package/packages/mcp-server/dist/moonshot-tool-schema.js +50 -0
- package/packages/mcp-server/dist/moonshot-tool-schema.js.map +1 -0
- package/packages/mcp-server/dist/server.d.ts.map +1 -1
- package/packages/mcp-server/dist/server.js +4 -0
- package/packages/mcp-server/dist/server.js.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts +26 -18
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +145 -59
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/package.json +5 -4
- package/packages/native/package.json +1 -1
- package/packages/pi-agent-core/dist/harness/env/nodejs.d.ts +1 -0
- package/packages/pi-agent-core/dist/harness/env/nodejs.d.ts.map +1 -1
- package/packages/pi-agent-core/dist/harness/env/nodejs.js +34 -3
- package/packages/pi-agent-core/dist/harness/env/nodejs.js.map +1 -1
- package/packages/pi-agent-core/dist/index.d.ts +1 -0
- package/packages/pi-agent-core/dist/index.d.ts.map +1 -1
- package/packages/pi-agent-core/dist/index.js +3 -0
- package/packages/pi-agent-core/dist/index.js.map +1 -1
- package/packages/pi-agent-core/package.json +1 -1
- package/packages/pi-ai/README.md +1 -0
- package/packages/pi-ai/dist/index.d.ts +2 -0
- package/packages/pi-ai/dist/index.d.ts.map +1 -1
- package/packages/pi-ai/dist/index.js +2 -0
- package/packages/pi-ai/dist/index.js.map +1 -1
- package/packages/pi-ai/dist/models.generated.d.ts +192 -0
- package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
- package/packages/pi-ai/dist/models.generated.js +166 -0
- package/packages/pi-ai/dist/models.generated.js.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic.js +12 -7
- package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
- package/packages/pi-ai/dist/providers/google-shared.d.ts +5 -0
- package/packages/pi-ai/dist/providers/google-shared.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/google-shared.js +12 -3
- package/packages/pi-ai/dist/providers/google-shared.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 +7 -3
- package/packages/pi-ai/dist/providers/openai-completions.js.map +1 -1
- package/packages/pi-ai/dist/utils/moonshot-tool-schema.d.ts +9 -0
- package/packages/pi-ai/dist/utils/moonshot-tool-schema.d.ts.map +1 -0
- package/packages/pi-ai/dist/utils/moonshot-tool-schema.js +34 -0
- package/packages/pi-ai/dist/utils/moonshot-tool-schema.js.map +1 -0
- package/packages/pi-ai/dist/utils/oauth/github-copilot.d.ts.map +1 -1
- package/packages/pi-ai/dist/utils/oauth/github-copilot.js +6 -2
- package/packages/pi-ai/dist/utils/oauth/github-copilot.js.map +1 -1
- package/packages/pi-ai/package.json +3 -2
- package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +2 -2
- package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.js +19 -13
- package/packages/pi-coding-agent/dist/core/auth-storage.js.map +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/dist/core/provider-readiness.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/provider-readiness.js +13 -6
- package/packages/pi-coding-agent/dist/core/provider-readiness.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/tools/bash.d.ts +11 -0
- package/packages/pi-coding-agent/dist/core/tools/bash.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/tools/bash.js +53 -11
- package/packages/pi-coding-agent/dist/core/tools/bash.js.map +1 -1
- 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/utils/shell.d.ts +28 -2
- package/packages/pi-coding-agent/dist/utils/shell.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/utils/shell.js +56 -10
- package/packages/pi-coding-agent/dist/utils/shell.js.map +1 -1
- package/packages/pi-coding-agent/package.json +7 -7
- package/packages/pi-tui/dist/tui.d.ts.map +1 -1
- package/packages/pi-tui/dist/tui.js +9 -0
- package/packages/pi-tui/dist/tui.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/GSD-WORKFLOW.md +5 -4
- package/src/resources/extensions/async-jobs/async-bash-cancel.test.ts +360 -0
- package/src/resources/extensions/async-jobs/async-bash-tool.ts +33 -56
- package/src/resources/extensions/async-jobs/await-tool.test.ts +139 -0
- package/src/resources/extensions/async-jobs/await-tool.ts +82 -12
- package/src/resources/extensions/async-jobs/index.ts +79 -0
- package/src/resources/extensions/async-jobs/job-manager.ts +21 -1
- package/src/resources/extensions/bg-shell/bg-shell-command.ts +6 -6
- package/src/resources/extensions/bg-shell/bg-shell-tool.ts +10 -6
- package/src/resources/extensions/bg-shell/overlay.ts +9 -5
- package/src/resources/extensions/bg-shell/process-manager.ts +50 -25
- package/src/resources/extensions/bg-shell/readiness-detector.ts +12 -0
- package/src/resources/extensions/bg-shell/tests/lifecycle-and-utilities.test.ts +48 -1
- package/src/resources/extensions/bg-shell/utilities.ts +5 -2
- package/src/resources/extensions/browser-tools/engine/managed-gsd-browser.ts +265 -98
- package/src/resources/extensions/browser-tools/engine/selection.ts +90 -4
- package/src/resources/extensions/browser-tools/index.ts +71 -13
- package/src/resources/extensions/browser-tools/tests/browser-engine-selection.test.mjs +83 -13
- package/src/resources/extensions/browser-tools/tests/gsd-browser-launch-config.test.mjs +40 -1
- package/src/resources/extensions/browser-tools/tests/managed-gsd-browser-tools.test.mjs +136 -0
- package/src/resources/extensions/claude-code-cli/models.ts +9 -0
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +40 -4
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +28 -0
- package/src/resources/extensions/gsd/auto/custom-verify-retry-store.ts +21 -3
- package/src/resources/extensions/gsd/auto/detect-stuck.ts +32 -9
- package/src/resources/extensions/gsd/auto/dispatch-history.ts +152 -0
- package/src/resources/extensions/gsd/auto/dispatch-key.ts +39 -0
- package/src/resources/extensions/gsd/auto/loop-deps.ts +1 -1
- package/src/resources/extensions/gsd/auto/loop.ts +4 -1
- package/src/resources/extensions/gsd/auto/orchestrator.ts +137 -61
- package/src/resources/extensions/gsd/auto/phases.ts +12 -3
- package/src/resources/extensions/gsd/auto-direct-dispatch.ts +8 -32
- package/src/resources/extensions/gsd/auto-dispatch.ts +38 -52
- package/src/resources/extensions/gsd/auto-model-selection.ts +41 -12
- package/src/resources/extensions/gsd/auto-post-unit.ts +37 -13
- package/src/resources/extensions/gsd/auto-prompts.ts +118 -35
- package/src/resources/extensions/gsd/auto-start.ts +24 -29
- package/src/resources/extensions/gsd/auto-tool-tracking.ts +19 -0
- package/src/resources/extensions/gsd/auto-unit-tool-scope.ts +14 -21
- package/src/resources/extensions/gsd/auto-verification.ts +8 -26
- package/src/resources/extensions/gsd/auto-worktree-repair.ts +13 -2
- package/src/resources/extensions/gsd/auto-worktree.ts +41 -364
- package/src/resources/extensions/gsd/auto.ts +28 -24
- package/src/resources/extensions/gsd/blocked-models.ts +49 -0
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +37 -10
- package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +33 -12
- package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +2 -2
- package/src/resources/extensions/gsd/bootstrap/register-extension.ts +24 -0
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +270 -37
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +368 -78
- package/src/resources/extensions/gsd/branch-patterns.ts +3 -0
- package/src/resources/extensions/gsd/browser-daemon-auto-prep.ts +108 -0
- package/src/resources/extensions/gsd/browser-evidence.ts +18 -2
- package/src/resources/extensions/gsd/captures.ts +5 -16
- package/src/resources/extensions/gsd/closeout-recovery.ts +2 -1
- package/src/resources/extensions/gsd/closeout-wizard.ts +102 -0
- package/src/resources/extensions/gsd/commands/catalog.ts +6 -68
- package/src/resources/extensions/gsd/commands-handlers.ts +46 -3
- package/src/resources/extensions/gsd/consent-question.ts +431 -0
- package/src/resources/extensions/gsd/consent-verdict.ts +86 -0
- package/src/resources/extensions/gsd/constants.ts +0 -3
- package/src/resources/extensions/gsd/crash-recovery.ts +3 -9
- package/src/resources/extensions/gsd/db/engine.ts +809 -0
- package/src/resources/extensions/gsd/db/queries.ts +490 -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/dispatch-guard.ts +8 -31
- package/src/resources/extensions/gsd/doctor-environment.ts +5 -13
- package/src/resources/extensions/gsd/doctor-format.ts +12 -7
- package/src/resources/extensions/gsd/doctor-git-checks.ts +5 -22
- package/src/resources/extensions/gsd/doctor-runtime-checks.ts +22 -17
- package/src/resources/extensions/gsd/engine-hook-contract.ts +79 -0
- package/src/resources/extensions/gsd/error-classifier.ts +11 -0
- package/src/resources/extensions/gsd/exec-sandbox.ts +49 -9
- package/src/resources/extensions/gsd/files.ts +33 -12
- 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-command-home.ts +13 -3
- package/src/resources/extensions/gsd/gsd-db.ts +176 -2375
- package/src/resources/extensions/gsd/guidance.ts +217 -0
- package/src/resources/extensions/gsd/guided-flow.ts +50 -5
- package/src/resources/extensions/gsd/markdown-renderer.ts +11 -0
- package/src/resources/extensions/gsd/mcp-filter.ts +2 -23
- package/src/resources/extensions/gsd/mcp-tool-name.ts +6 -11
- package/src/resources/extensions/gsd/memory-consolidation-scanner.ts +1 -1
- package/src/resources/extensions/gsd/migrate/safety.ts +18 -7
- package/src/resources/extensions/gsd/migration-auto-check.ts +28 -3
- package/src/resources/extensions/gsd/milestone-closeout.ts +109 -24
- package/src/resources/extensions/gsd/model-cost-table.ts +1 -0
- package/src/resources/extensions/gsd/model-router.ts +3 -0
- package/src/resources/extensions/gsd/notification-store.ts +26 -3
- package/src/resources/extensions/gsd/parallel-merge.ts +12 -9
- package/src/resources/extensions/gsd/parallel-monitor-overlay.ts +10 -7
- package/src/resources/extensions/gsd/parsers-legacy.ts +16 -4
- package/src/resources/extensions/gsd/paths.ts +42 -22
- package/src/resources/extensions/gsd/pre-execution-checks.ts +109 -3
- package/src/resources/extensions/gsd/preferences-models.ts +12 -47
- package/src/resources/extensions/gsd/preferences.ts +18 -0
- package/src/resources/extensions/gsd/prompts/complete-slice.md +2 -2
- package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/refine-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/replan-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/run-uat.md +6 -4
- package/src/resources/extensions/gsd/prompts/system.md +5 -2
- package/src/resources/extensions/gsd/provider-error-guidance.ts +4 -9
- package/src/resources/extensions/gsd/provider-switch-observer.ts +1 -1
- package/src/resources/extensions/gsd/publication.ts +122 -0
- package/src/resources/extensions/gsd/reactive-graph.ts +11 -1
- package/src/resources/extensions/gsd/recovery-classification.ts +47 -88
- package/src/resources/extensions/gsd/safety/destructive-confirmation.ts +134 -0
- package/src/resources/extensions/gsd/safety/evidence-collector.ts +36 -4
- package/src/resources/extensions/gsd/safety/evidence-cross-ref.ts +7 -2
- package/src/resources/extensions/gsd/safety/file-change-validator.ts +14 -0
- package/src/resources/extensions/gsd/state-transition-matrix.ts +42 -0
- package/src/resources/extensions/gsd/state.ts +9 -21
- package/src/resources/extensions/gsd/status-guards.ts +59 -8
- package/src/resources/extensions/gsd/stop-notice.ts +75 -0
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +123 -0
- package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +91 -0
- package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +198 -26
- package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +3 -1
- package/src/resources/extensions/gsd/tests/auto-post-unit-evidence-crossref-4909.test.ts +46 -0
- package/src/resources/extensions/gsd/tests/auto-worktree-registry.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/auto-worktree-repair.test.ts +4 -2
- package/src/resources/extensions/gsd/tests/blocked-models.test.ts +19 -0
- package/src/resources/extensions/gsd/tests/browser-automation-contract-fixture.ts +39 -0
- package/src/resources/extensions/gsd/tests/browser-contract.test.ts +44 -0
- package/src/resources/extensions/gsd/tests/browser-daemon-auto-prep.test.ts +144 -0
- package/src/resources/extensions/gsd/tests/checkout-branch-stash-guard.test.ts +66 -1
- package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +44 -0
- package/src/resources/extensions/gsd/tests/commands-verdict.test.ts +8 -7
- package/src/resources/extensions/gsd/tests/complete-slice-verification-gate.test.ts +42 -0
- package/src/resources/extensions/gsd/tests/consent-question.test.ts +351 -0
- package/src/resources/extensions/gsd/tests/custom-verify-retry-store.test.ts +67 -0
- package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +10 -10
- package/src/resources/extensions/gsd/tests/destructive-confirmation.test.ts +303 -0
- package/src/resources/extensions/gsd/tests/dispatch-history.test.ts +273 -0
- package/src/resources/extensions/gsd/tests/dispatch-run-uat-browser-tools.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/doctor-git-checks-terminal.test.ts +73 -0
- package/src/resources/extensions/gsd/tests/dynamic-bash-no-cap.test.ts +132 -0
- package/src/resources/extensions/gsd/tests/engine-hook-contract.test.ts +148 -0
- package/src/resources/extensions/gsd/tests/evidence-xref-gsd-exec.test.ts +157 -0
- package/src/resources/extensions/gsd/tests/exec-graceful-kill.test.ts +193 -0
- package/src/resources/extensions/gsd/tests/exec-tool.test.ts +29 -1
- package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +35 -1
- package/src/resources/extensions/gsd/tests/file-change-validator.test.ts +33 -1
- package/src/resources/extensions/gsd/tests/gsd-command-home.test.ts +120 -0
- package/src/resources/extensions/gsd/tests/gsd-db.test.ts +27 -0
- package/src/resources/extensions/gsd/tests/guidance.test.ts +148 -0
- package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +58 -15
- package/src/resources/extensions/gsd/tests/integration/auto-worktree.test.ts +74 -59
- package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/integration/gsd-integration-fixture.ts +80 -0
- package/src/resources/extensions/gsd/tests/integration/run-uat.test.ts +199 -0
- package/src/resources/extensions/gsd/tests/mcp-project-config.test.ts +3 -1
- package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +85 -1
- package/src/resources/extensions/gsd/tests/milestone-closeout.test.ts +95 -4
- package/src/resources/extensions/gsd/tests/model-unittype-mapping.test.ts +32 -1
- package/src/resources/extensions/gsd/tests/notification-store.test.ts +32 -0
- package/src/resources/extensions/gsd/tests/oauth-api-model-routing.test.ts +167 -0
- package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +18 -0
- package/src/resources/extensions/gsd/tests/parsers-legacy-importers.test.ts +138 -0
- package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +193 -1
- package/src/resources/extensions/gsd/tests/prompt-db.test.ts +124 -6
- package/src/resources/extensions/gsd/tests/provider-error-guidance.test.ts +3 -3
- package/src/resources/extensions/gsd/tests/publication.test.ts +120 -0
- package/src/resources/extensions/gsd/tests/recovery-classification-illegal-transition.test.ts +30 -0
- package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +248 -1
- package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +1 -0
- package/src/resources/extensions/gsd/tests/safety-harness-false-positives.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/session-switch-clears-pending-autostart.test.ts +108 -0
- package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +43 -6
- package/src/resources/extensions/gsd/tests/state-transition-matrix.test.ts +36 -0
- package/src/resources/extensions/gsd/tests/status-guards.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/stop-notice.test.ts +70 -0
- package/src/resources/extensions/gsd/tests/token-tool-gating.test.ts +76 -0
- package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +8 -0
- package/src/resources/extensions/gsd/tests/tool-surface-readiness.test.ts +155 -0
- package/src/resources/extensions/gsd/tests/uat-policy.test.ts +112 -29
- package/src/resources/extensions/gsd/tests/unit-closeout.test.ts +209 -0
- package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +67 -2
- package/src/resources/extensions/gsd/tests/unit-registry.test.ts +163 -0
- package/src/resources/extensions/gsd/tests/web-app-uat.test.ts +44 -1
- package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +275 -40
- package/src/resources/extensions/gsd/tests/worktree-lifecycle.test.ts +41 -4
- package/src/resources/extensions/gsd/tests/worktree-manager.test.ts +22 -1
- package/src/resources/extensions/gsd/tests/worktree-placement.test.ts +113 -0
- package/src/resources/extensions/gsd/tests/worktree-reentry.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/worktree-safety.test.ts +3 -1
- package/src/resources/extensions/gsd/tests/worktree-symlink-removal.test.ts +12 -6
- package/src/resources/extensions/gsd/tests/worktree-teardown-safety.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/write-gate-seam.test.ts +358 -0
- package/src/resources/extensions/gsd/tests/write-gate.test.ts +109 -1
- package/src/resources/extensions/gsd/tool-presentation-plan.ts +4 -4
- package/src/resources/extensions/gsd/tool-surface-readiness.ts +76 -0
- package/src/resources/extensions/gsd/tools/complete-slice.ts +43 -68
- package/src/resources/extensions/gsd/tools/exec-tool.ts +9 -8
- package/src/resources/extensions/gsd/tools/plan-slice.ts +12 -6
- package/src/resources/extensions/gsd/tools/reopen-milestone.ts +11 -38
- package/src/resources/extensions/gsd/tools/reopen-slice.ts +14 -42
- package/src/resources/extensions/gsd/tools/skip-slice.ts +18 -44
- package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +81 -2
- package/src/resources/extensions/gsd/uat-policy.ts +62 -16
- package/src/resources/extensions/gsd/undo.ts +9 -8
- package/src/resources/extensions/gsd/unit-closeout.ts +201 -0
- package/src/resources/extensions/gsd/unit-context-composer.ts +111 -1
- package/src/resources/extensions/gsd/unit-context-manifest.ts +4 -28
- package/src/resources/extensions/gsd/unit-registry.ts +412 -0
- package/src/resources/extensions/gsd/unit-tool-contracts.ts +27 -192
- package/src/resources/extensions/gsd/verdict-parser.ts +1 -1
- package/src/resources/extensions/gsd/web-app-uat.ts +51 -8
- package/src/resources/extensions/gsd/workflow-tool-surface.ts +4 -1
- package/src/resources/extensions/gsd/worktree-git-recovery.ts +314 -0
- package/src/resources/extensions/gsd/worktree-lifecycle.ts +13 -9
- package/src/resources/extensions/gsd/worktree-manager.ts +47 -28
- package/src/resources/extensions/gsd/worktree-placement.ts +63 -0
- package/src/resources/extensions/gsd/worktree-reentry.ts +10 -7
- package/src/resources/extensions/gsd/worktree-root.ts +29 -6
- package/src/resources/extensions/gsd/worktree-safety.ts +8 -5
- package/src/resources/extensions/gsd/worktree-session-state.ts +11 -11
- package/src/resources/extensions/search-the-web/native-search.ts +5 -3
- package/src/resources/extensions/shared/browser-contract.ts +66 -0
- package/src/resources/extensions/shared/gsd-browser-cli.ts +141 -6
- package/src/resources/shared/gsd-browser-path-sync.ts +273 -0
- package/src/resources/shared/package-manager-detection.ts +1 -1
- package/src/resources/shared/package.json +3 -0
- package/src/resources/skills/create-skill/references/executable-code.md +1 -1
- package/src/resources/skills/create-skill/workflows/add-reference.md +8 -3
- package/src/resources/skills/create-skill/workflows/add-script.md +4 -2
- package/src/resources/skills/create-skill/workflows/add-template.md +3 -1
- package/src/resources/skills/create-skill/workflows/add-workflow.md +8 -3
- package/src/resources/skills/create-skill/workflows/upgrade-to-router.md +10 -5
- package/src/resources/skills/create-skill/workflows/verify-skill.md +9 -4
- package/src/resources/skills/gsd-browser/SKILL.md +1 -1
- package/src/resources/skills/spike-wrap-up/SKILL.md +9 -9
- package/dist/resources/extensions/gsd/user-input-boundary.js +0 -218
- package/src/resources/extensions/gsd/tests/user-input-boundary.test.ts +0 -173
- package/src/resources/extensions/gsd/user-input-boundary.ts +0 -216
- /package/dist/web/standalone/.next/static/{3PtrU9qGPEXwNLWkIyiqk → LDHRKiRBIVZmiuMjrL1Vy}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{3PtrU9qGPEXwNLWkIyiqk → LDHRKiRBIVZmiuMjrL1Vy}/_ssgManifest.js +0 -0
|
@@ -1,13 +1,16 @@
|
|
|
1
1
|
// gsd-pi - Write gate runtime persistence and policy guards.
|
|
2
|
-
import { copyFileSync, existsSync, lstatSync, mkdirSync,
|
|
2
|
+
import { copyFileSync, existsSync, lstatSync, mkdirSync, readlinkSync, realpathSync, renameSync, unlinkSync, writeFileSync } from "node:fs";
|
|
3
3
|
import { isAbsolute, join, relative, resolve, sep } from "node:path";
|
|
4
4
|
import { minimatch } from "minimatch";
|
|
5
5
|
import { GSD_PHASE_SCOPE_DISPLAY_REASON, shouldBlockAutoUnitToolCall } from "../auto-unit-tool-scope.js";
|
|
6
|
-
import {
|
|
6
|
+
import { canonicalToolName } from "../engine-hook-contract.js";
|
|
7
|
+
import { loadJsonFileOrNull } from "../json-persistence.js";
|
|
7
8
|
import { getIsolationMode } from "../preferences.js";
|
|
8
9
|
import { compileSubagentPermissionContract } from "../unit-context-manifest.js";
|
|
9
10
|
import { logWarning } from "../workflow-logger.js";
|
|
10
11
|
import { isGsdWorktreePath, resolveWorktreeProjectRoot } from "../worktree-root.js";
|
|
12
|
+
import { worktreesDirs } from "../worktree-placement.js";
|
|
13
|
+
import { evaluateGateAnswer } from "../consent-verdict.js";
|
|
11
14
|
/**
|
|
12
15
|
* Regex matching milestone CONTEXT.md file names in both legacy M001
|
|
13
16
|
* and unique M001-abc123 formats. Exported so regex-hardening tests
|
|
@@ -102,9 +105,6 @@ const GATE_QUESTION_PATTERNS = [
|
|
|
102
105
|
const GATE_SAFE_TOOLS = new Set([
|
|
103
106
|
"ask_user_questions",
|
|
104
107
|
]);
|
|
105
|
-
export function canonicalToolName(toolName) {
|
|
106
|
-
return stripMcpToolPrefix(toolName);
|
|
107
|
-
}
|
|
108
108
|
/**
|
|
109
109
|
* Persistence is ON by default (opt-out).
|
|
110
110
|
* Set GSD_PERSIST_WRITE_GATE_STATE="0" or GSD_PERSIST_WRITE_GATE_STATE="false"
|
|
@@ -143,13 +143,11 @@ function currentWriteGateSnapshot(basePath = process.cwd()) {
|
|
|
143
143
|
pendingGateId: state.pendingGateId,
|
|
144
144
|
};
|
|
145
145
|
}
|
|
146
|
-
function
|
|
147
|
-
if (!shouldPersistWriteGateSnapshot())
|
|
148
|
-
return;
|
|
146
|
+
function writeSnapshotFileAtomic(basePath, snapshot) {
|
|
149
147
|
const path = writeGateSnapshotPath(basePath);
|
|
150
148
|
ensureWriteGateSnapshotDirectory(basePath);
|
|
151
149
|
const tempPath = `${path}.${process.pid}.${Date.now()}.${Math.random().toString(36).slice(2)}.tmp`;
|
|
152
|
-
writeFileSync(tempPath, JSON.stringify(
|
|
150
|
+
writeFileSync(tempPath, JSON.stringify(snapshot, null, 2), "utf-8");
|
|
153
151
|
try {
|
|
154
152
|
renameSync(tempPath, path);
|
|
155
153
|
}
|
|
@@ -165,6 +163,18 @@ function persistWriteGateSnapshot(basePath) {
|
|
|
165
163
|
}
|
|
166
164
|
}
|
|
167
165
|
}
|
|
166
|
+
/**
|
|
167
|
+
* Persist the current in-memory state for `basePath`, stamped with the
|
|
168
|
+
* writer provenance tag. Callers (mutateWriteGateState / childMutate) are
|
|
169
|
+
* responsible for first reconciling in-memory content against the disk
|
|
170
|
+
* snapshot — the read-merge-write sequence is synchronous, so a concurrent
|
|
171
|
+
* writer's update is folded in by that unconditional pre-persist read.
|
|
172
|
+
*/
|
|
173
|
+
function persistWriteGateSnapshot(basePath, writer) {
|
|
174
|
+
if (!shouldPersistWriteGateSnapshot())
|
|
175
|
+
return;
|
|
176
|
+
writeSnapshotFileAtomic(basePath, { ...currentWriteGateSnapshot(basePath), writer });
|
|
177
|
+
}
|
|
168
178
|
function clearPersistedWriteGateSnapshot(basePath) {
|
|
169
179
|
if (!shouldPersistWriteGateSnapshot())
|
|
170
180
|
return;
|
|
@@ -189,6 +199,7 @@ function normalizeWriteGateSnapshot(value) {
|
|
|
189
199
|
verifiedApprovalGates: [...new Set(verifiedGates)].sort(),
|
|
190
200
|
activeQueuePhase: record.activeQueuePhase === true,
|
|
191
201
|
pendingGateId: typeof record.pendingGateId === "string" ? record.pendingGateId : null,
|
|
202
|
+
...(record.writer === "host" || record.writer === "child" ? { writer: record.writer } : {}),
|
|
192
203
|
};
|
|
193
204
|
}
|
|
194
205
|
const EMPTY_SNAPSHOT = {
|
|
@@ -197,6 +208,14 @@ const EMPTY_SNAPSHOT = {
|
|
|
197
208
|
activeQueuePhase: false,
|
|
198
209
|
pendingGateId: null,
|
|
199
210
|
};
|
|
211
|
+
/**
|
|
212
|
+
* Read the snapshot file as it exists on disk. Returns null when the file
|
|
213
|
+
* is missing or unparseable (no in-memory fallback — callers decide).
|
|
214
|
+
*/
|
|
215
|
+
function readDiskSnapshot(basePath) {
|
|
216
|
+
const raw = loadJsonFileOrNull(writeGateSnapshotPath(basePath), (data) => typeof data === "object" && data !== null);
|
|
217
|
+
return raw === null ? null : normalizeWriteGateSnapshot(raw);
|
|
218
|
+
}
|
|
200
219
|
export function loadWriteGateSnapshot(basePath) {
|
|
201
220
|
const path = writeGateSnapshotPath(basePath);
|
|
202
221
|
if (!existsSync(path)) {
|
|
@@ -207,12 +226,112 @@ export function loadWriteGateSnapshot(basePath) {
|
|
|
207
226
|
return EMPTY_SNAPSHOT;
|
|
208
227
|
return currentWriteGateSnapshot(basePath);
|
|
209
228
|
}
|
|
210
|
-
|
|
211
|
-
|
|
229
|
+
const disk = readDiskSnapshot(basePath);
|
|
230
|
+
return disk ?? currentWriteGateSnapshot(basePath);
|
|
231
|
+
}
|
|
232
|
+
/**
|
|
233
|
+
* Reconcile a disk snapshot into the in-memory state. This is THE merge
|
|
234
|
+
* rule for the two-process seam:
|
|
235
|
+
*
|
|
236
|
+
* - verifications are a grow-only union: a milestone/gate verified by
|
|
237
|
+
* either process is never lost to a stale write from the other;
|
|
238
|
+
* - activeQueuePhase and pendingGateId take the disk value (last writer
|
|
239
|
+
* wins — matches the long-standing refresh semantics, and keeps
|
|
240
|
+
* "delete the snapshot file to clear the HARD BLOCK" working);
|
|
241
|
+
* - verified wins over pending: if the merged verified sets already cover
|
|
242
|
+
* the pending gate (gate id verified, or its depth milestone verified),
|
|
243
|
+
* the pending re-arm is dropped. This generalizes the ad-hoc re-arm
|
|
244
|
+
* guard that previously protected only the tool_execution_start window.
|
|
245
|
+
*/
|
|
246
|
+
function mergeSnapshotIntoState(state, disk) {
|
|
247
|
+
for (const milestone of disk.verifiedDepthMilestones)
|
|
248
|
+
state.verifiedDepthMilestones.add(milestone);
|
|
249
|
+
for (const gate of disk.verifiedApprovalGates ?? [])
|
|
250
|
+
state.verifiedApprovalGates.add(gate);
|
|
251
|
+
state.activeQueuePhase = disk.activeQueuePhase;
|
|
252
|
+
state.pendingGateId = disk.pendingGateId;
|
|
253
|
+
dropVerifiedPendingGate(state);
|
|
254
|
+
}
|
|
255
|
+
/** Verified-on-disk wins over a pending re-arm (see mergeSnapshotIntoState). */
|
|
256
|
+
function dropVerifiedPendingGate(state) {
|
|
257
|
+
const pending = state.pendingGateId;
|
|
258
|
+
if (!pending)
|
|
259
|
+
return;
|
|
260
|
+
const milestoneId = extractDepthVerificationMilestoneId(pending);
|
|
261
|
+
if (state.verifiedApprovalGates.has(pending) ||
|
|
262
|
+
(milestoneId !== null && state.verifiedDepthMilestones.has(milestoneId))) {
|
|
263
|
+
state.pendingGateId = null;
|
|
212
264
|
}
|
|
213
|
-
|
|
265
|
+
}
|
|
266
|
+
function replaceStateFromSnapshot(state, snapshot) {
|
|
267
|
+
state.pendingGateId = snapshot.pendingGateId;
|
|
268
|
+
state.activeQueuePhase = snapshot.activeQueuePhase;
|
|
269
|
+
state.verifiedDepthMilestones = new Set(snapshot.verifiedDepthMilestones);
|
|
270
|
+
state.verifiedApprovalGates = new Set(snapshot.verifiedApprovalGates ?? []);
|
|
271
|
+
}
|
|
272
|
+
/**
|
|
273
|
+
* Reconcile the persisted write-gate snapshot into the in-process Map entry.
|
|
274
|
+
* The workflow MCP server runs in a child process and records depth
|
|
275
|
+
* verification there; without this refresh the extension host keeps stale
|
|
276
|
+
* pending-gate memory and `activateDeferredApprovalGate` can re-arm a gate
|
|
277
|
+
* that the subprocess already cleared on disk.
|
|
278
|
+
*
|
|
279
|
+
* Uses the union merge rule (mergeSnapshotIntoState) when a readable
|
|
280
|
+
* snapshot file exists; a missing or unparseable file is a full reset
|
|
281
|
+
* (replace with empty) so deleting the file still clears all gate state,
|
|
282
|
+
* and a corrupt file does not leave stale `pendingGateId` in memory for
|
|
283
|
+
* the next mutation to persist back.
|
|
284
|
+
*
|
|
285
|
+
* Returns the reconciled snapshot so callers that need to inspect it
|
|
286
|
+
* (e.g. re-arm guards) avoid a second disk read.
|
|
287
|
+
*/
|
|
288
|
+
export function refreshWriteGateStateFromDisk(basePath) {
|
|
289
|
+
if (!shouldPersistWriteGateSnapshot())
|
|
214
290
|
return currentWriteGateSnapshot(basePath);
|
|
291
|
+
const state = getWriteGateState(basePath);
|
|
292
|
+
const disk = readDiskSnapshot(basePath);
|
|
293
|
+
if (disk) {
|
|
294
|
+
mergeSnapshotIntoState(state, disk);
|
|
295
|
+
}
|
|
296
|
+
else {
|
|
297
|
+
replaceStateFromSnapshot(state, EMPTY_SNAPSHOT);
|
|
298
|
+
}
|
|
299
|
+
return currentWriteGateSnapshot(basePath);
|
|
300
|
+
}
|
|
301
|
+
/**
|
|
302
|
+
* Read-modify-write primitive for gate mutations. Reconciles the disk
|
|
303
|
+
* snapshot into memory (union merge), applies the mutation on top, then
|
|
304
|
+
* persists. The whole sequence is synchronous, so the reconcile read
|
|
305
|
+
* doubles as the pre-persist merge of concurrent writes — there is no
|
|
306
|
+
* version field; the read-merge-write is simply unconditional.
|
|
307
|
+
*
|
|
308
|
+
* The mutate callback sees the already-reconciled state, so policy checks
|
|
309
|
+
* (e.g. host setPending's verified-on-disk-wins guard) can live inside it
|
|
310
|
+
* without a second disk read. Returning `false` from the callback aborts:
|
|
311
|
+
* nothing is persisted and this function returns false.
|
|
312
|
+
*
|
|
313
|
+
* `reconcile: false` skips the disk merge — used for intentional full
|
|
314
|
+
* resets where re-unioning disk verifications would defeat the reset.
|
|
315
|
+
*/
|
|
316
|
+
function mutateWriteGateState(basePath, mutate, opts) {
|
|
317
|
+
const state = getWriteGateState(basePath);
|
|
318
|
+
if (shouldPersistWriteGateSnapshot() && (opts?.reconcile ?? true)) {
|
|
319
|
+
const disk = readDiskSnapshot(basePath);
|
|
320
|
+
if (disk) {
|
|
321
|
+
mergeSnapshotIntoState(state, disk);
|
|
322
|
+
}
|
|
323
|
+
else {
|
|
324
|
+
// Missing OR unparseable on disk: treat as a full reset. Keeping
|
|
325
|
+
// stale in-memory state across a corrupt snapshot would persist it
|
|
326
|
+
// back on this very mutation and defeat the documented
|
|
327
|
+
// "delete the file to clear the HARD BLOCK gate" escape hatch.
|
|
328
|
+
replaceStateFromSnapshot(state, EMPTY_SNAPSHOT);
|
|
329
|
+
}
|
|
215
330
|
}
|
|
331
|
+
if (mutate(state) === false)
|
|
332
|
+
return false;
|
|
333
|
+
persistWriteGateSnapshot(basePath, opts?.writer ?? defaultWriteGateWriter());
|
|
334
|
+
return true;
|
|
216
335
|
}
|
|
217
336
|
export function isDepthVerified(basePath = process.cwd()) {
|
|
218
337
|
return getWriteGateState(basePath).verifiedDepthMilestones.size > 0;
|
|
@@ -223,6 +342,7 @@ export function isDepthVerified(basePath = process.cwd()) {
|
|
|
223
342
|
export function isMilestoneDepthVerified(milestoneId, basePath = process.cwd()) {
|
|
224
343
|
if (!milestoneId)
|
|
225
344
|
return false;
|
|
345
|
+
refreshWriteGateStateFromDisk(basePath);
|
|
226
346
|
return getWriteGateState(basePath).verifiedDepthMilestones.has(milestoneId);
|
|
227
347
|
}
|
|
228
348
|
export function isMilestoneDepthVerifiedInSnapshot(snapshot, milestoneId) {
|
|
@@ -234,31 +354,34 @@ export function isQueuePhaseActive(basePath = process.cwd()) {
|
|
|
234
354
|
return getWriteGateState(basePath).activeQueuePhase;
|
|
235
355
|
}
|
|
236
356
|
export function setQueuePhaseActive(active, basePath) {
|
|
237
|
-
|
|
238
|
-
|
|
357
|
+
mutateWriteGateState(basePath, (state) => {
|
|
358
|
+
state.activeQueuePhase = active;
|
|
359
|
+
});
|
|
239
360
|
}
|
|
240
361
|
export function resetWriteGateState(basePath) {
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
362
|
+
mutateWriteGateState(basePath, (state) => {
|
|
363
|
+
state.verifiedDepthMilestones.clear();
|
|
364
|
+
state.verifiedApprovalGates.clear();
|
|
365
|
+
state.pendingGateId = null;
|
|
366
|
+
}, { reconcile: false });
|
|
246
367
|
}
|
|
247
368
|
export function clearDiscussionFlowState(basePath) {
|
|
248
369
|
writeGateStatesByBasePath.delete(writeGateStateKey(basePath));
|
|
249
370
|
clearPersistedWriteGateSnapshot(basePath);
|
|
250
371
|
}
|
|
372
|
+
/**
|
|
373
|
+
* Ambient (env-sniffed) export, reserved for the child's dynamic-import
|
|
374
|
+
* surface (packages/mcp-server) and module-internal use. Host-owned modules
|
|
375
|
+
* (register-hooks, auto-dispatch, …) must call
|
|
376
|
+
* hostWriteGateAdapter.markDepthVerified explicitly so a leaked
|
|
377
|
+
* GSD_WORKFLOW_* env variable cannot silently flip them to child semantics.
|
|
378
|
+
*/
|
|
251
379
|
export function markDepthVerified(milestoneId, basePath = process.cwd()) {
|
|
252
|
-
|
|
253
|
-
return;
|
|
254
|
-
getWriteGateState(basePath).verifiedDepthMilestones.add(milestoneId);
|
|
255
|
-
persistWriteGateSnapshot(basePath);
|
|
380
|
+
defaultWriteGateAdapter().markDepthVerified(milestoneId, basePath);
|
|
256
381
|
}
|
|
382
|
+
/** Ambient export for the child's dynamic-import surface — see markDepthVerified. */
|
|
257
383
|
export function markApprovalGateVerified(gateId, basePath = process.cwd()) {
|
|
258
|
-
|
|
259
|
-
return;
|
|
260
|
-
getWriteGateState(basePath).verifiedApprovalGates.add(gateId);
|
|
261
|
-
persistWriteGateSnapshot(basePath);
|
|
384
|
+
defaultWriteGateAdapter().markApprovalGateVerified(gateId, basePath);
|
|
262
385
|
}
|
|
263
386
|
export function isApprovalGateVerifiedInSnapshot(snapshot, gateId) {
|
|
264
387
|
if (!gateId)
|
|
@@ -287,29 +410,147 @@ function extractContextMilestoneId(inputPath) {
|
|
|
287
410
|
return match?.[1] ?? null;
|
|
288
411
|
}
|
|
289
412
|
/**
|
|
290
|
-
* Mark a gate as pending (called when ask_user_questions is invoked with a
|
|
413
|
+
* Mark a gate as pending (called when ask_user_questions is invoked with a
|
|
414
|
+
* gate ID). Delegates to the process's default adapter: in the workflow MCP
|
|
415
|
+
* child the arm is unconditional (a fresh question intentionally revokes
|
|
416
|
+
* prior verification); in the host the adapter's verified-on-disk-wins
|
|
417
|
+
* guard applies and a suppressed arm returns false.
|
|
418
|
+
*
|
|
419
|
+
* Ambient (env-sniffed) export, reserved for the child's dynamic-import
|
|
420
|
+
* surface (packages/mcp-server). Host-owned modules must call
|
|
421
|
+
* hostWriteGateAdapter.setPending explicitly.
|
|
291
422
|
*/
|
|
292
423
|
export function setPendingGate(gateId, basePath) {
|
|
293
|
-
|
|
424
|
+
return defaultWriteGateAdapter().setPending(gateId, basePath);
|
|
425
|
+
}
|
|
426
|
+
/** Arm `gateId` on a reconciled state, revoking its prior verification. */
|
|
427
|
+
function armPendingGate(state, gateId) {
|
|
294
428
|
state.pendingGateId = gateId;
|
|
295
429
|
state.verifiedApprovalGates.delete(gateId);
|
|
296
430
|
const milestoneId = extractDepthVerificationMilestoneId(gateId);
|
|
297
431
|
if (milestoneId)
|
|
298
432
|
state.verifiedDepthMilestones.delete(milestoneId);
|
|
299
|
-
persistWriteGateSnapshot(basePath);
|
|
300
433
|
}
|
|
301
434
|
/**
|
|
302
435
|
* Clear the pending gate (called when the user confirms).
|
|
436
|
+
* Ambient export for the child's dynamic-import surface — host-owned
|
|
437
|
+
* modules must call hostWriteGateAdapter.clearPending explicitly.
|
|
303
438
|
*/
|
|
304
439
|
export function clearPendingGate(basePath) {
|
|
305
|
-
|
|
306
|
-
persistWriteGateSnapshot(basePath);
|
|
440
|
+
defaultWriteGateAdapter().clearPending(basePath);
|
|
307
441
|
}
|
|
308
442
|
/**
|
|
309
443
|
* Get the currently pending gate, if any.
|
|
444
|
+
* Ambient export for the child's dynamic-import surface — host-owned
|
|
445
|
+
* modules should prefer hostWriteGateAdapter.readState.
|
|
310
446
|
*/
|
|
311
447
|
export function getPendingGate(basePath = process.cwd()) {
|
|
312
|
-
return
|
|
448
|
+
return defaultWriteGateAdapter().readState(basePath).pendingGateId;
|
|
449
|
+
}
|
|
450
|
+
/**
|
|
451
|
+
* HOST adapter: in-memory state + reconcile-on-read. Disk verifications are
|
|
452
|
+
* never lost (union merge), and a gate verified on disk wins over an
|
|
453
|
+
* in-memory pending re-arm — setPending refuses to clobber a verification
|
|
454
|
+
* the child already recorded. This generalizes the ad-hoc re-arm guard that
|
|
455
|
+
* previously protected only the tool_execution_start window in
|
|
456
|
+
* register-hooks.ts.
|
|
457
|
+
*/
|
|
458
|
+
export const hostWriteGateAdapter = {
|
|
459
|
+
writer: "host",
|
|
460
|
+
readState(basePath) {
|
|
461
|
+
return refreshWriteGateStateFromDisk(basePath);
|
|
462
|
+
},
|
|
463
|
+
markDepthVerified(milestoneId, basePath) {
|
|
464
|
+
if (!milestoneId)
|
|
465
|
+
return;
|
|
466
|
+
mutateWriteGateState(basePath, (state) => {
|
|
467
|
+
state.verifiedDepthMilestones.add(milestoneId);
|
|
468
|
+
}, { writer: "host" });
|
|
469
|
+
},
|
|
470
|
+
markApprovalGateVerified(gateId, basePath) {
|
|
471
|
+
if (!gateId)
|
|
472
|
+
return;
|
|
473
|
+
mutateWriteGateState(basePath, (state) => {
|
|
474
|
+
state.verifiedApprovalGates.add(gateId);
|
|
475
|
+
}, { writer: "host" });
|
|
476
|
+
},
|
|
477
|
+
setPending(gateId, basePath) {
|
|
478
|
+
// The verified-check runs inside the mutate callback so the single
|
|
479
|
+
// reconcile read in mutateWriteGateState serves both the guard and the
|
|
480
|
+
// pre-persist merge. A suppressed arm aborts without persisting.
|
|
481
|
+
return mutateWriteGateState(basePath, (state) => {
|
|
482
|
+
if (state.verifiedApprovalGates.has(gateId))
|
|
483
|
+
return false;
|
|
484
|
+
const milestoneId = extractDepthVerificationMilestoneId(gateId);
|
|
485
|
+
if (milestoneId && state.verifiedDepthMilestones.has(milestoneId))
|
|
486
|
+
return false;
|
|
487
|
+
armPendingGate(state, gateId);
|
|
488
|
+
}, { writer: "host" });
|
|
489
|
+
},
|
|
490
|
+
clearPending(basePath) {
|
|
491
|
+
mutateWriteGateState(basePath, (state) => {
|
|
492
|
+
state.pendingGateId = null;
|
|
493
|
+
}, { writer: "host" });
|
|
494
|
+
},
|
|
495
|
+
};
|
|
496
|
+
/**
|
|
497
|
+
* CHILD adapter: write-through with always-fresh reads — load the disk
|
|
498
|
+
* snapshot, mutate, persist. No cross-turn in-memory state is trusted: the
|
|
499
|
+
* in-process Map entry is replaced from disk before every mutation so a
|
|
500
|
+
* long-lived MCP child never resurrects state the host has since changed.
|
|
501
|
+
*/
|
|
502
|
+
export const childWriteGateAdapter = {
|
|
503
|
+
writer: "child",
|
|
504
|
+
readState(basePath) {
|
|
505
|
+
return loadWriteGateSnapshot(basePath);
|
|
506
|
+
},
|
|
507
|
+
markDepthVerified(milestoneId, basePath) {
|
|
508
|
+
if (!milestoneId)
|
|
509
|
+
return;
|
|
510
|
+
childMutate(basePath, (state) => {
|
|
511
|
+
state.verifiedDepthMilestones.add(milestoneId);
|
|
512
|
+
});
|
|
513
|
+
},
|
|
514
|
+
markApprovalGateVerified(gateId, basePath) {
|
|
515
|
+
if (!gateId)
|
|
516
|
+
return;
|
|
517
|
+
childMutate(basePath, (state) => {
|
|
518
|
+
state.verifiedApprovalGates.add(gateId);
|
|
519
|
+
});
|
|
520
|
+
},
|
|
521
|
+
setPending(gateId, basePath) {
|
|
522
|
+
childMutate(basePath, (state) => armPendingGate(state, gateId));
|
|
523
|
+
return true;
|
|
524
|
+
},
|
|
525
|
+
clearPending(basePath) {
|
|
526
|
+
childMutate(basePath, (state) => {
|
|
527
|
+
state.pendingGateId = null;
|
|
528
|
+
});
|
|
529
|
+
},
|
|
530
|
+
};
|
|
531
|
+
function childMutate(basePath, mutate) {
|
|
532
|
+
const state = getWriteGateState(basePath);
|
|
533
|
+
if (shouldPersistWriteGateSnapshot()) {
|
|
534
|
+
// Always-fresh: disk is the only truth for the child; discard any
|
|
535
|
+
// cross-turn in-memory residue before mutating.
|
|
536
|
+
replaceStateFromSnapshot(state, readDiskSnapshot(basePath) ?? EMPTY_SNAPSHOT);
|
|
537
|
+
}
|
|
538
|
+
mutate(state);
|
|
539
|
+
persistWriteGateSnapshot(basePath, "child");
|
|
540
|
+
}
|
|
541
|
+
/**
|
|
542
|
+
* Which adapter the module-level convenience exports (markDepthVerified,
|
|
543
|
+
* setPendingGate, …) delegate to. The workflow MCP child is spawned with
|
|
544
|
+
* GSD_WORKFLOW_WRITE_GATE_MODULE / GSD_WORKFLOW_PROJECT_ROOT in its
|
|
545
|
+
* environment (workflow-mcp.ts), so when this module is dynamically imported
|
|
546
|
+
* inside that process the child adapter is selected automatically; the
|
|
547
|
+
* extension host process has neither variable and stays on the host adapter.
|
|
548
|
+
*/
|
|
549
|
+
export function defaultWriteGateWriter(env = process.env) {
|
|
550
|
+
return env.GSD_WORKFLOW_WRITE_GATE_MODULE || env.GSD_WORKFLOW_PROJECT_ROOT ? "child" : "host";
|
|
551
|
+
}
|
|
552
|
+
function defaultWriteGateAdapter() {
|
|
553
|
+
return defaultWriteGateWriter() === "child" ? childWriteGateAdapter : hostWriteGateAdapter;
|
|
313
554
|
}
|
|
314
555
|
/**
|
|
315
556
|
* Check whether a tool call should be blocked because a discussion gate
|
|
@@ -359,36 +600,15 @@ export function shouldBlockPendingGateBashInSnapshot(snapshot, command, _milesto
|
|
|
359
600
|
].join(" "),
|
|
360
601
|
};
|
|
361
602
|
}
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
368
|
-
*
|
|
369
|
-
* @param selected The answer's selected value from details.response.answers[id].selected
|
|
370
|
-
* @param options The question's options array from event.input.questions[n].options
|
|
371
|
-
*/
|
|
372
|
-
export function isDepthConfirmationAnswer(selected, options) {
|
|
373
|
-
const value = Array.isArray(selected) ? selected[0] : selected;
|
|
374
|
-
if (typeof value !== "string" || !value)
|
|
375
|
-
return false;
|
|
376
|
-
// If options are available, structurally validate: selected must exactly match
|
|
377
|
-
// the first option (confirmation) label. Rejects free-form "Other" and decline options.
|
|
378
|
-
if (Array.isArray(options) && options.length > 0) {
|
|
379
|
-
const confirmLabel = options[0]?.label;
|
|
380
|
-
return typeof confirmLabel === "string" && value === confirmLabel;
|
|
381
|
-
}
|
|
382
|
-
// Fail-closed: no options means we cannot structurally validate the answer.
|
|
383
|
-
// Returning false prevents any free-form string from unlocking the gate.
|
|
384
|
-
return false;
|
|
385
|
-
}
|
|
603
|
+
// The structural depth-confirmation validator lives in the consent-verdict
|
|
604
|
+
// leaf (../consent-verdict.ts) so the write gate and the Consent Question
|
|
605
|
+
// module share one verdict engine without an import cycle. Re-exported here
|
|
606
|
+
// because the workflow MCP child loads this module by dist path and validates
|
|
607
|
+
// the function is present (packages/mcp-server/src/server.ts).
|
|
608
|
+
export { isDepthConfirmationAnswer } from "../consent-verdict.js";
|
|
386
609
|
function findGateQuestion(questions, gateId) {
|
|
387
610
|
return questions.find((question) => question?.id === gateId);
|
|
388
611
|
}
|
|
389
|
-
function readSelectedGateAnswer(details, questionId) {
|
|
390
|
-
return details.response?.answers?.[questionId]?.selected;
|
|
391
|
-
}
|
|
392
612
|
function verifyAnsweredGate(basePath, question, fallbackMilestoneId) {
|
|
393
613
|
const gateId = typeof question.id === "string" ? question.id : "";
|
|
394
614
|
const milestoneId = extractDepthVerificationMilestoneId(gateId) ?? fallbackMilestoneId ?? null;
|
|
@@ -397,6 +617,30 @@ function verifyAnsweredGate(basePath, question, fallbackMilestoneId) {
|
|
|
397
617
|
clearPendingGate(basePath);
|
|
398
618
|
return { status: "verified", gateId, milestoneId };
|
|
399
619
|
}
|
|
620
|
+
/** Map an unresolved (non-verified) gate verdict to the caller-facing result. */
|
|
621
|
+
function unresolvedGateResult(verdict, gateId, details) {
|
|
622
|
+
if (verdict === "declined")
|
|
623
|
+
return { status: "declined", gateId };
|
|
624
|
+
// "waiting" (and the unreachable post-cancel case): an empty selection is
|
|
625
|
+
// not an answer — keep the gate pending and make the caller pause.
|
|
626
|
+
return {
|
|
627
|
+
status: "waiting",
|
|
628
|
+
pendingGateId: gateId,
|
|
629
|
+
interrupted: details.interrupted === true,
|
|
630
|
+
};
|
|
631
|
+
}
|
|
632
|
+
/**
|
|
633
|
+
* Apply an ask_user_questions round to durable gate state. The per-question
|
|
634
|
+
* VERDICT comes from the consent-verdict leaf (evaluateGateAnswer) — the same
|
|
635
|
+
* engine the Consent Question module uses — so write-gate only owns the
|
|
636
|
+
* persistence/arming side effects:
|
|
637
|
+
*
|
|
638
|
+
* - "verified" verdict → markApprovalGateVerified/markDepthVerified/clearPendingGate.
|
|
639
|
+
* - "declined" verdict → no state change; the gate (if armed) stays pending.
|
|
640
|
+
* - "waiting" verdict (empty/missing selection) → no state change; reported as
|
|
641
|
+
* "waiting" so callers pause instead of proceeding (fail-closed; an empty
|
|
642
|
+
* answer is never an answer).
|
|
643
|
+
*/
|
|
400
644
|
export function applyAskUserQuestionsGateResult(options) {
|
|
401
645
|
const { basePath, questions, details, fallbackMilestoneId } = options;
|
|
402
646
|
const currentPendingGate = getPendingGate(basePath);
|
|
@@ -410,11 +654,11 @@ export function applyAskUserQuestionsGateResult(options) {
|
|
|
410
654
|
}
|
|
411
655
|
const pendingQuestion = findGateQuestion(questions, currentPendingGate);
|
|
412
656
|
if (pendingQuestion) {
|
|
413
|
-
const
|
|
414
|
-
if (
|
|
657
|
+
const verdict = evaluateGateAnswer(pendingQuestion, details);
|
|
658
|
+
if (verdict === "verified") {
|
|
415
659
|
return verifyAnsweredGate(basePath, pendingQuestion, fallbackMilestoneId);
|
|
416
660
|
}
|
|
417
|
-
return
|
|
661
|
+
return unresolvedGateResult(verdict, currentPendingGate, details);
|
|
418
662
|
}
|
|
419
663
|
}
|
|
420
664
|
if (details.cancelled || !details.response)
|
|
@@ -422,12 +666,14 @@ export function applyAskUserQuestionsGateResult(options) {
|
|
|
422
666
|
for (const question of questions) {
|
|
423
667
|
if (typeof question.id !== "string" || !isGateQuestionId(question.id))
|
|
424
668
|
continue;
|
|
425
|
-
const
|
|
426
|
-
if (
|
|
427
|
-
return
|
|
669
|
+
const verdict = evaluateGateAnswer(question, details);
|
|
670
|
+
if (verdict !== "verified") {
|
|
671
|
+
return unresolvedGateResult(verdict, question.id, details);
|
|
428
672
|
}
|
|
429
673
|
if (currentPendingGate && question.id !== currentPendingGate) {
|
|
430
|
-
|
|
674
|
+
// A different gate than the armed one was confirmed — the armed gate is
|
|
675
|
+
// still unresolved, so do not verify and let discussion continue.
|
|
676
|
+
return { status: "declined", gateId: currentPendingGate };
|
|
431
677
|
}
|
|
432
678
|
return verifyAnsweredGate(basePath, question, fallbackMilestoneId);
|
|
433
679
|
}
|
|
@@ -919,10 +1165,12 @@ export function shouldBlockWorktreeWrite(toolName, targetPath, effectiveBasePath
|
|
|
919
1165
|
const realTarget = realpathOrResolve(absTarget);
|
|
920
1166
|
const realRoot = realpathOrResolve(projectRoot);
|
|
921
1167
|
const realGsd = realpathOrResolve(join(projectRoot, ".gsd"));
|
|
922
|
-
|
|
923
|
-
//
|
|
924
|
-
|
|
925
|
-
|
|
1168
|
+
// Allow writes inside a legitimate worktrees subtree (canonical
|
|
1169
|
+
// .gsd-worktrees/ or legacy .gsd/worktrees/).
|
|
1170
|
+
for (const container of worktreesDirs(projectRoot)) {
|
|
1171
|
+
if (isPathContained(realTarget, realpathOrResolve(container)))
|
|
1172
|
+
return { block: false };
|
|
1173
|
+
}
|
|
926
1174
|
// Allow writes to .gsd/ planning artifacts, but reject siblings whose name
|
|
927
1175
|
// starts with "worktrees" (the worktrees-extra prefix trick — case 4).
|
|
928
1176
|
if (isPathContained(realTarget, realGsd)) {
|
|
@@ -11,3 +11,5 @@ export const SLICE_BRANCH_RE = /^gsd\/(?:([a-zA-Z0-9_-]+)\/)?(M\d+(?:-[a-z0-9]{6
|
|
|
11
11
|
export const QUICK_BRANCH_RE = /^gsd\/quick\//;
|
|
12
12
|
/** Matches GSD-generated workflow template branches, not arbitrary user gsd/* branches. */
|
|
13
13
|
export const WORKFLOW_BRANCH_RE = /^gsd\/(?:hotfix|bugfix|small-feature|refactor|spike|security-audit|dep-upgrade|full-project)\//;
|
|
14
|
+
/** Auto-mode milestone branch prefix: milestone/<MID>. */
|
|
15
|
+
export const MILESTONE_BRANCH_PREFIX = "milestone/";
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
import { execFileSync } from "node:child_process";
|
|
2
|
+
import { resolve } from "node:path";
|
|
3
|
+
import { resolveAmbientBrowserEngineResolution, resolveBrowserEngineResolution, } from "../browser-tools/engine/selection.js";
|
|
4
|
+
import { resolveGsdBrowserCliAvailability, resolveGsdBrowserDaemonStartInvocation, } from "../shared/gsd-browser-cli.js";
|
|
5
|
+
import { uatTypeIncludesBrowser } from "./uat-policy.js";
|
|
6
|
+
const DEFAULT_DAEMON_START_TIMEOUT_MS = 30_000;
|
|
7
|
+
function isEnvDisabled(value) {
|
|
8
|
+
if (!value)
|
|
9
|
+
return false;
|
|
10
|
+
const normalized = value.trim().toLowerCase();
|
|
11
|
+
return normalized === "0" || normalized === "false" || normalized === "off";
|
|
12
|
+
}
|
|
13
|
+
function isWarmUpDisabled(env = process.env) {
|
|
14
|
+
const value = env.GSD_BROWSER_WARMUP?.trim().toLowerCase();
|
|
15
|
+
return value === "0" || value === "false" || value === "off";
|
|
16
|
+
}
|
|
17
|
+
/** Active engine for warm-up: explicit env override, else session-committed ambient resolution. */
|
|
18
|
+
function resolveActiveBrowserEngine(projectRoot, env) {
|
|
19
|
+
if (env.GSD_BROWSER_ENGINE?.trim()) {
|
|
20
|
+
return resolveBrowserEngineResolution(env, projectRoot).engine;
|
|
21
|
+
}
|
|
22
|
+
return resolveAmbientBrowserEngineResolution(projectRoot).engine;
|
|
23
|
+
}
|
|
24
|
+
export function shouldWarmBrowserDaemonForUat(ctx) {
|
|
25
|
+
if (!uatTypeIncludesBrowser(ctx.uatType))
|
|
26
|
+
return false;
|
|
27
|
+
const env = ctx.env ?? process.env;
|
|
28
|
+
if (isWarmUpDisabled(env))
|
|
29
|
+
return false;
|
|
30
|
+
if (isEnvDisabled(env.GSD_BROWSER_MCP_ENABLED))
|
|
31
|
+
return false;
|
|
32
|
+
const availability = resolveGsdBrowserCliAvailability(env);
|
|
33
|
+
if (!availability.available)
|
|
34
|
+
return false;
|
|
35
|
+
const projectRoot = resolve(ctx.projectRoot);
|
|
36
|
+
return resolveActiveBrowserEngine(projectRoot, env) === "gsd-browser";
|
|
37
|
+
}
|
|
38
|
+
export function ensureBrowserDaemonStarted(projectRoot, options = {}) {
|
|
39
|
+
const env = options.env ?? process.env;
|
|
40
|
+
const availability = resolveGsdBrowserCliAvailability(env);
|
|
41
|
+
if (!availability.available) {
|
|
42
|
+
return { ok: false, error: availability.detail };
|
|
43
|
+
}
|
|
44
|
+
let invocation;
|
|
45
|
+
try {
|
|
46
|
+
invocation = resolveGsdBrowserDaemonStartInvocation(projectRoot, env);
|
|
47
|
+
}
|
|
48
|
+
catch (error) {
|
|
49
|
+
return {
|
|
50
|
+
ok: false,
|
|
51
|
+
error: error instanceof Error ? error.message : String(error),
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
try {
|
|
55
|
+
execFileSync(invocation.command, invocation.args, {
|
|
56
|
+
cwd: invocation.cwd,
|
|
57
|
+
env: { ...process.env, ...env, ...(invocation.env ?? {}) },
|
|
58
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
59
|
+
timeout: options.timeoutMs ?? DEFAULT_DAEMON_START_TIMEOUT_MS,
|
|
60
|
+
encoding: "utf-8",
|
|
61
|
+
});
|
|
62
|
+
return { ok: true };
|
|
63
|
+
}
|
|
64
|
+
catch (error) {
|
|
65
|
+
return {
|
|
66
|
+
ok: false,
|
|
67
|
+
error: error instanceof Error ? error.message : String(error),
|
|
68
|
+
};
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Best-effort pre-warm of the gsd-browser session daemon before browser-backed
|
|
73
|
+
* run-uat dispatch. Returns an actionable stop reason when warm-up is required
|
|
74
|
+
* but fails; returns null when warm-up is skipped or succeeds.
|
|
75
|
+
*/
|
|
76
|
+
export function prepareBrowserDaemonForUat(ctx) {
|
|
77
|
+
if (!shouldWarmBrowserDaemonForUat(ctx))
|
|
78
|
+
return null;
|
|
79
|
+
const result = ensureBrowserDaemonStarted(ctx.projectRoot, { env: ctx.env });
|
|
80
|
+
if (result.ok)
|
|
81
|
+
return null;
|
|
82
|
+
return `Cannot dispatch browser-backed run-uat: gsd-browser daemon failed to start (${result.error}). Ensure Chrome/Chromium is installed, run \`gsd-browser daemon health\` with the project session flags from .mcp.json, or set GSD_BROWSER_PATH to a Chromium binary.`;
|
|
83
|
+
}
|
|
@@ -1,8 +1,14 @@
|
|
|
1
1
|
// Project/App: gsd-pi
|
|
2
2
|
// File Purpose: Shared browser-observable UAT requirement and evidence detection.
|
|
3
|
-
|
|
3
|
+
import { BROWSER_EVIDENCE_SIGNAL_TOOL_NAMES } from "../shared/browser-contract.js";
|
|
4
|
+
// Alternation fragment over the contract's evidence-signal names, e.g.
|
|
5
|
+
// `browser_(?:assert|batch|...)`. The names are `browser_`-prefixed
|
|
6
|
+
// identifiers (pinned by tests/browser-contract.test.ts), so no escaping is
|
|
7
|
+
// needed.
|
|
8
|
+
const BROWSER_TOOL_SIGNAL = `browser_(?:${BROWSER_EVIDENCE_SIGNAL_TOOL_NAMES.map((name) => name.slice("browser_".length)).join("|")})`;
|
|
9
|
+
export const BROWSER_REQUIREMENT_RE = new RegExp(String.raw `\b(?:file://|localhost|playwright|chrome|screenshot|snapshot|${BROWSER_TOOL_SIGNAL})\b|\b(?:open|launch|navigate|load|visit|serve|start)\b.{0,80}\b(?:browser|page|localhost|file://)\b|\bbrowser\s+(?:check|session|test|uat|tool|automation|interaction|flow)\b`, "i");
|
|
4
10
|
export const NO_BROWSER_EVIDENCE_RE = /\b(?:no|without|not|wasn'?t|isn'?t)\s+(?:automated\s+)?(?:live\s+)?browser(?:\s+(?:session|test|uat))?|\bno\s+automated\s+browser\b|\bnot\s+conducted\b/i;
|
|
5
|
-
export const BROWSER_RUNTIME_RE =
|
|
11
|
+
export const BROWSER_RUNTIME_RE = new RegExp(String.raw `\b(?:browser|playwright|chrome|camoufox|${BROWSER_TOOL_SIGNAL}|screenshot|snapshot|file://|localhost)\b`, "i");
|
|
6
12
|
export const BROWSER_ACTION_RE = /\b(?:open(?:ed)?|navigate(?:d)?|click(?:ed)?|type(?:d)?|reload(?:ed)?|capture(?:d)?|screenshot|snapshot)\b/i;
|
|
7
13
|
export const BROWSER_ASSERTION_RE = /\b(?:assert(?:ed|ion)?|observed|confirmed|verified|expected|visible|text|count|label|strikethrough|localstorage|screenshot|snapshot|passed)\b/i;
|
|
8
14
|
const NON_REQUIREMENT_BROWSER_HEADING_RE = /^(?:not\s+proven|not\s+covered|out\s+of\s+scope|deferred|follow-?ups?|known\s+limitations|notes\s+for\s+tester)\b/i;
|