@opengsd/gsd-pi 1.2.0-dev.fb12b103 → 1.2.0-dev.fbdca60b
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-model-override.d.ts +15 -0
- package/dist/cli-model-override.js +21 -0
- package/dist/cli.js +1 -18
- package/dist/headless-events.js +7 -5
- package/dist/loader.js +6 -4
- package/dist/mcp-server.js +2 -1
- package/dist/register-agent-bundles.d.ts +11 -2
- package/dist/register-agent-bundles.js +18 -4
- package/dist/resource-loader.d.ts +10 -5
- package/dist/resource-loader.js +121 -6
- package/dist/resources/.managed-resources-content-hash +1 -1
- package/dist/resources/GSD-WORKFLOW.md +5 -4
- package/dist/resources/extensions/ask-user-questions.js +3 -2
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +447 -215
- package/dist/resources/extensions/claude-code-cli/turn-assembler.js +33 -1
- package/dist/resources/extensions/gsd/auto/closeout.js +215 -0
- 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 +120 -0
- package/dist/resources/extensions/gsd/auto/dispatch-key.js +37 -0
- package/dist/resources/extensions/gsd/auto/dispatch.js +365 -0
- package/dist/resources/extensions/gsd/auto/finalize.js +347 -0
- package/dist/resources/extensions/gsd/auto/loop.js +7 -1
- package/dist/resources/extensions/gsd/auto/milestone-lease-reclaim.js +56 -0
- package/dist/resources/extensions/gsd/auto/orchestrator.js +167 -64
- package/dist/resources/extensions/gsd/auto/phase-helpers.js +146 -0
- package/dist/resources/extensions/gsd/auto/phases.js +17 -2329
- package/dist/resources/extensions/gsd/auto/pre-dispatch.js +534 -0
- package/dist/resources/extensions/gsd/auto/session.js +3 -0
- package/dist/resources/extensions/gsd/auto/unit-phase.js +694 -0
- package/dist/resources/extensions/gsd/auto/workflow-unit-dispatch.js +1 -1
- package/dist/resources/extensions/gsd/auto/worktree-safety-phase.js +125 -0
- package/dist/resources/extensions/gsd/auto-direct-dispatch.js +11 -34
- package/dist/resources/extensions/gsd/auto-dispatch.js +39 -58
- package/dist/resources/extensions/gsd/auto-model-selection.js +11 -7
- package/dist/resources/extensions/gsd/auto-post-unit.js +30 -12
- package/dist/resources/extensions/gsd/auto-prompts.js +66 -9
- package/dist/resources/extensions/gsd/auto-start.js +26 -8
- package/dist/resources/extensions/gsd/auto-unit-closeout.js +45 -21
- package/dist/resources/extensions/gsd/auto-unit-tool-scope.js +5 -4
- package/dist/resources/extensions/gsd/auto-verification.js +23 -30
- package/dist/resources/extensions/gsd/auto-worktree.js +15 -2
- package/dist/resources/extensions/gsd/auto.js +52 -2
- package/dist/resources/extensions/gsd/blocked-models.js +28 -0
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +26 -6
- package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +37 -7
- package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +2 -2
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +107 -45
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +302 -80
- package/dist/resources/extensions/gsd/closeout-wizard.js +92 -0
- package/dist/resources/extensions/gsd/commands/context.js +16 -2
- package/dist/resources/extensions/gsd/commands-handlers.js +46 -3
- package/dist/resources/extensions/gsd/commands-mcp-status.js +2 -2
- package/dist/resources/extensions/gsd/commands-workflow-templates.js +9 -2
- 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/crash-recovery.js +8 -3
- package/dist/resources/extensions/gsd/db/engine.js +24 -6
- package/dist/resources/extensions/gsd/db/queries.js +56 -0
- package/dist/resources/extensions/gsd/db-migration-backup.js +51 -8
- package/dist/resources/extensions/gsd/db-transaction.js +27 -23
- package/dist/resources/extensions/gsd/db-writer.js +8 -17
- package/dist/resources/extensions/gsd/dispatch-guard.js +10 -35
- package/dist/resources/extensions/gsd/doctor-engine-checks.js +5 -5
- package/dist/resources/extensions/gsd/doctor-environment.js +256 -125
- package/dist/resources/extensions/gsd/doctor-git-checks.js +2 -18
- package/dist/resources/extensions/gsd/engine-hook-contract.js +70 -0
- package/dist/resources/extensions/gsd/files.js +33 -19
- package/dist/resources/extensions/gsd/gsd-command-home.js +22 -12
- package/dist/resources/extensions/gsd/gsd-db.js +17 -21
- package/dist/resources/extensions/gsd/guidance.js +60 -0
- package/dist/resources/extensions/gsd/guided-flow.js +93 -4
- package/dist/resources/extensions/gsd/health-widget.js +87 -28
- package/dist/resources/extensions/gsd/markdown-renderer.js +10 -0
- package/dist/resources/extensions/gsd/mcp-bridge.js +10 -0
- package/dist/resources/extensions/gsd/memory-relations.js +1 -1
- package/dist/resources/extensions/gsd/milestone-closeout.js +85 -24
- package/dist/resources/extensions/gsd/milestone-planning-persistence.js +2 -2
- package/dist/resources/extensions/gsd/milestone-reopen-events.js +3 -5
- package/dist/resources/extensions/gsd/milestone-settlement.js +2 -2
- package/dist/resources/extensions/gsd/notifications.js +12 -7
- package/dist/resources/extensions/gsd/parsers-legacy.js +16 -4
- package/dist/resources/extensions/gsd/projection-flush.js +7 -0
- package/dist/resources/extensions/gsd/prompts/complete-slice.md +3 -3
- package/dist/resources/extensions/gsd/prompts/execute-task.md +3 -2
- package/dist/resources/extensions/gsd/prompts/plan-milestone.md +1 -1
- package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/quick-task.md +1 -1
- package/dist/resources/extensions/gsd/prompts/reassess-roadmap.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/research-milestone.md +1 -1
- package/dist/resources/extensions/gsd/prompts/research-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
- package/dist/resources/extensions/gsd/prompts/run-uat.md +8 -4
- package/dist/resources/extensions/gsd/prompts/triage-captures.md +1 -1
- package/dist/resources/extensions/gsd/prompts/validate-milestone.md +1 -1
- package/dist/resources/extensions/gsd/prompts/workflow-start.md +2 -1
- package/dist/resources/extensions/gsd/reactive-graph.js +8 -1
- package/dist/resources/extensions/gsd/roadmap-slices.js +25 -3
- package/dist/resources/extensions/gsd/session-lock.js +1 -1
- package/dist/resources/extensions/gsd/skill-activation.js +3 -6
- package/dist/resources/extensions/gsd/state.js +11 -2
- package/dist/resources/extensions/gsd/tool-contract.js +14 -3
- package/dist/resources/extensions/gsd/tool-presentation-plan.js +4 -4
- package/dist/resources/extensions/gsd/tool-surface-readiness.js +83 -31
- package/dist/resources/extensions/gsd/tools/complete-milestone.js +3 -2
- package/dist/resources/extensions/gsd/tools/complete-slice.js +22 -12
- package/dist/resources/extensions/gsd/tools/complete-task.js +65 -2
- package/dist/resources/extensions/gsd/tools/plan-slice.js +2 -2
- package/dist/resources/extensions/gsd/tools/plan-task.js +2 -2
- package/dist/resources/extensions/gsd/tools/reassess-roadmap.js +2 -2
- package/dist/resources/extensions/gsd/tools/reopen-milestone.js +2 -2
- package/dist/resources/extensions/gsd/tools/reopen-slice.js +2 -2
- package/dist/resources/extensions/gsd/tools/reopen-task.js +2 -2
- package/dist/resources/extensions/gsd/tools/replan-slice.js +2 -2
- package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +67 -2
- package/dist/resources/extensions/gsd/uat-policy.js +40 -15
- package/dist/resources/extensions/gsd/unit-context-composer.js +1 -1
- package/dist/resources/extensions/gsd/unit-registry.js +34 -4
- package/dist/resources/extensions/gsd/verdict-parser.js +1 -1
- package/dist/resources/extensions/gsd/verification-verdict.js +2 -1
- package/dist/resources/extensions/gsd/workflow-event-ledger.js +91 -0
- package/dist/resources/extensions/gsd/workflow-event-vocabulary.js +46 -0
- package/dist/resources/extensions/gsd/workflow-events.js +6 -18
- package/dist/resources/extensions/gsd/workflow-mcp-auto-prep.js +2 -0
- package/dist/resources/extensions/gsd/workflow-mcp-readiness-cache.js +105 -0
- package/dist/resources/extensions/gsd/workflow-reconcile.js +21 -56
- package/dist/resources/extensions/gsd/worktree-lifecycle.js +3 -2
- package/dist/resources/extensions/gsd/worktree-manager.js +7 -1
- package/dist/resources/extensions/gsd/worktree-safety.js +28 -26
- package/dist/resources/extensions/gsd/worktree.js +8 -1
- package/dist/resources/extensions/mcp-client/manager.js +6 -1
- package/dist/resources/extensions/shared/gsd-browser-cli.js +21 -2
- package/dist/resources/shared/gsd-browser-path-sync.js +214 -0
- package/dist/resources/shared/package-manager-detection.js +1 -1
- package/dist/resources/skills/create-skill/SKILL.md +3 -0
- package/dist/resources/skills/create-skill/references/skill-structure.md +1 -0
- package/dist/runtime-checks.d.ts +10 -0
- package/dist/runtime-checks.js +27 -0
- 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 +6 -6
- package/dist/web/standalone/.next/build-manifest.json +2 -2
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/mcp-connections/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/notifications/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/shutdown/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +6 -6
- package/dist/web/standalone/.next/server/chunks/{5942.js → 1128.js} +1 -1
- 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/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/node_modules/node-pty/build/Makefile +1 -1
- package/dist/web/standalone/node_modules/postcss/lib/container.js +26 -18
- package/dist/web/standalone/node_modules/postcss/lib/css-syntax-error.js +47 -14
- package/dist/web/standalone/node_modules/postcss/lib/declaration.js +4 -4
- package/dist/web/standalone/node_modules/postcss/lib/fromJSON.js +3 -3
- package/dist/web/standalone/node_modules/postcss/lib/input.js +54 -29
- package/dist/web/standalone/node_modules/postcss/lib/lazy-result.js +47 -37
- package/dist/web/standalone/node_modules/postcss/lib/map-generator.js +26 -9
- package/dist/web/standalone/node_modules/postcss/lib/no-work-result.js +57 -55
- package/dist/web/standalone/node_modules/postcss/lib/node.js +99 -31
- package/dist/web/standalone/node_modules/postcss/lib/parse.js +1 -1
- package/dist/web/standalone/node_modules/postcss/lib/parser.js +10 -9
- package/dist/web/standalone/node_modules/postcss/lib/postcss.js +12 -12
- package/dist/web/standalone/node_modules/postcss/lib/previous-map.js +30 -11
- package/dist/web/standalone/node_modules/postcss/lib/processor.js +7 -7
- package/dist/web/standalone/node_modules/postcss/lib/result.js +5 -5
- package/dist/web/standalone/node_modules/postcss/lib/rule.js +6 -6
- package/dist/web/standalone/node_modules/postcss/lib/stringifier.js +69 -28
- package/dist/web/standalone/node_modules/postcss/lib/tokenize.js +6 -2
- package/dist/web/standalone/node_modules/postcss/package.json +48 -48
- package/package.json +3 -3
- 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/sdk.d.ts.map +1 -1
- package/packages/gsd-agent-core/dist/sdk.js +6 -4
- package/packages/gsd-agent-core/dist/sdk.js.map +1 -1
- package/packages/gsd-agent-core/package.json +5 -5
- package/packages/gsd-agent-modes/dist/modes/interactive/components/settings-selector.d.ts +2 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/components/settings-selector.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/settings-selector.js +10 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/components/settings-selector.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts +8 -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 +50 -6
- package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/transcript-design.d.ts +2 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/components/transcript-design.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/transcript-design.js +34 -5
- package/packages/gsd-agent-modes/dist/modes/interactive/components/transcript-design.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode.d.ts +1 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode.js +17 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-settings.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-settings.js +4 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-settings.js.map +1 -1
- package/packages/gsd-agent-modes/package.json +7 -7
- package/packages/mcp-server/README.md +12 -3
- package/packages/mcp-server/dist/cli-runner.d.ts +40 -0
- package/packages/mcp-server/dist/cli-runner.d.ts.map +1 -0
- package/packages/mcp-server/dist/cli-runner.js +137 -0
- package/packages/mcp-server/dist/cli-runner.js.map +1 -0
- package/packages/mcp-server/dist/cli.js +2 -53
- 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/pid-registry.d.ts +46 -0
- package/packages/mcp-server/dist/pid-registry.d.ts.map +1 -0
- package/packages/mcp-server/dist/pid-registry.js +459 -0
- package/packages/mcp-server/dist/pid-registry.js.map +1 -0
- package/packages/mcp-server/dist/probe-mode.d.ts +4 -0
- package/packages/mcp-server/dist/probe-mode.d.ts.map +1 -0
- package/packages/mcp-server/dist/probe-mode.js +10 -0
- package/packages/mcp-server/dist/probe-mode.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/stdio-watchdog.d.ts +8 -0
- package/packages/mcp-server/dist/stdio-watchdog.d.ts.map +1 -0
- package/packages/mcp-server/dist/stdio-watchdog.js +40 -0
- package/packages/mcp-server/dist/stdio-watchdog.js.map +1 -0
- package/packages/mcp-server/dist/workflow-tools.d.ts +18 -18
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +161 -81
- 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/agent-loop.js +43 -2
- package/packages/pi-agent-core/dist/agent-loop.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/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/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 +419 -221
- package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
- package/packages/pi-ai/dist/models.generated.js +468 -269
- 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/settings-manager.d.ts +3 -0
- package/packages/pi-coding-agent/dist/core/settings-manager.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/settings-manager.js +11 -0
- package/packages/pi-coding-agent/dist/core/settings-manager.js.map +1 -1
- package/packages/pi-coding-agent/dist/theme/theme.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/theme/theme.js +45 -17
- package/packages/pi-coding-agent/dist/theme/theme.js.map +1 -1
- package/packages/pi-coding-agent/package.json +7 -7
- package/packages/pi-tui/README.md +15 -0
- package/packages/pi-tui/dist/index.d.ts +2 -2
- package/packages/pi-tui/dist/index.d.ts.map +1 -1
- package/packages/pi-tui/dist/index.js +2 -2
- package/packages/pi-tui/dist/index.js.map +1 -1
- package/packages/pi-tui/dist/terminal-image.d.ts +33 -0
- package/packages/pi-tui/dist/terminal-image.d.ts.map +1 -1
- package/packages/pi-tui/dist/terminal-image.js +54 -2
- package/packages/pi-tui/dist/terminal-image.js.map +1 -1
- package/packages/pi-tui/dist/terminal.d.ts +12 -0
- package/packages/pi-tui/dist/terminal.d.ts.map +1 -1
- package/packages/pi-tui/dist/terminal.js +70 -25
- package/packages/pi-tui/dist/terminal.js.map +1 -1
- package/packages/pi-tui/dist/tui.d.ts +15 -0
- package/packages/pi-tui/dist/tui.d.ts.map +1 -1
- package/packages/pi-tui/dist/tui.js +106 -21
- package/packages/pi-tui/dist/tui.js.map +1 -1
- package/packages/pi-tui/dist/utils.d.ts.map +1 -1
- package/packages/pi-tui/dist/utils.js +110 -36
- package/packages/pi-tui/dist/utils.js.map +1 -1
- package/packages/pi-tui/package.json +2 -2
- package/packages/rpc-client/package.json +2 -2
- package/pkg/dist/theme/theme.d.ts.map +1 -1
- package/pkg/dist/theme/theme.js +45 -17
- package/pkg/dist/theme/theme.js.map +1 -1
- package/pkg/package.json +1 -1
- package/src/resources/GSD-WORKFLOW.md +5 -4
- package/src/resources/extensions/ask-user-questions.ts +7 -2
- package/src/resources/extensions/browser-tools/tests/gsd-browser-launch-config.test.mjs +11 -0
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +531 -226
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +672 -7
- package/src/resources/extensions/claude-code-cli/turn-assembler.ts +38 -1
- package/src/resources/extensions/gsd/auto/closeout.ts +309 -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 +168 -0
- package/src/resources/extensions/gsd/auto/dispatch-key.ts +39 -0
- package/src/resources/extensions/gsd/auto/dispatch.ts +449 -0
- package/src/resources/extensions/gsd/auto/finalize.ts +445 -0
- package/src/resources/extensions/gsd/auto/loop.ts +7 -1
- package/src/resources/extensions/gsd/auto/milestone-lease-reclaim.ts +74 -0
- package/src/resources/extensions/gsd/auto/orchestrator.ts +186 -66
- package/src/resources/extensions/gsd/auto/phase-helpers.ts +199 -0
- package/src/resources/extensions/gsd/auto/phases.ts +58 -3022
- package/src/resources/extensions/gsd/auto/pre-dispatch.ts +704 -0
- package/src/resources/extensions/gsd/auto/session.ts +3 -0
- package/src/resources/extensions/gsd/auto/unit-phase.ts +910 -0
- package/src/resources/extensions/gsd/auto/workflow-unit-dispatch.ts +1 -1
- package/src/resources/extensions/gsd/auto/worktree-safety-phase.ts +149 -0
- package/src/resources/extensions/gsd/auto-direct-dispatch.ts +18 -48
- package/src/resources/extensions/gsd/auto-dispatch.ts +37 -62
- package/src/resources/extensions/gsd/auto-model-selection.ts +16 -7
- package/src/resources/extensions/gsd/auto-post-unit.ts +33 -12
- package/src/resources/extensions/gsd/auto-prompts.ts +78 -9
- package/src/resources/extensions/gsd/auto-start.ts +27 -11
- package/src/resources/extensions/gsd/auto-unit-closeout.ts +83 -28
- package/src/resources/extensions/gsd/auto-unit-tool-scope.ts +4 -4
- package/src/resources/extensions/gsd/auto-verification.ts +26 -28
- package/src/resources/extensions/gsd/auto-worktree.ts +15 -2
- package/src/resources/extensions/gsd/auto.ts +64 -2
- package/src/resources/extensions/gsd/blocked-models.ts +49 -0
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +34 -5
- package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +56 -6
- package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +2 -2
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +118 -50
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +350 -86
- package/src/resources/extensions/gsd/closeout-wizard.ts +102 -0
- package/src/resources/extensions/gsd/commands/context.ts +16 -2
- package/src/resources/extensions/gsd/commands-handlers.ts +46 -3
- package/src/resources/extensions/gsd/commands-mcp-status.ts +2 -2
- package/src/resources/extensions/gsd/commands-workflow-templates.ts +11 -4
- 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/crash-recovery.ts +10 -2
- package/src/resources/extensions/gsd/db/engine.ts +26 -6
- package/src/resources/extensions/gsd/db/queries.ts +66 -0
- package/src/resources/extensions/gsd/db-migration-backup.ts +56 -7
- package/src/resources/extensions/gsd/db-transaction.ts +37 -20
- package/src/resources/extensions/gsd/db-writer.ts +11 -19
- package/src/resources/extensions/gsd/dispatch-guard.ts +8 -31
- package/src/resources/extensions/gsd/doctor-engine-checks.ts +5 -4
- package/src/resources/extensions/gsd/doctor-environment.ts +267 -142
- package/src/resources/extensions/gsd/doctor-git-checks.ts +2 -19
- package/src/resources/extensions/gsd/engine-hook-contract.ts +79 -0
- package/src/resources/extensions/gsd/files.ts +33 -12
- package/src/resources/extensions/gsd/gsd-command-home.ts +13 -3
- package/src/resources/extensions/gsd/gsd-db.ts +19 -22
- package/src/resources/extensions/gsd/guidance.ts +78 -0
- package/src/resources/extensions/gsd/guided-flow.ts +145 -24
- package/src/resources/extensions/gsd/health-widget.ts +91 -27
- package/src/resources/extensions/gsd/markdown-renderer.ts +11 -0
- package/src/resources/extensions/gsd/mcp-bridge.ts +39 -0
- package/src/resources/extensions/gsd/memory-relations.ts +1 -1
- package/src/resources/extensions/gsd/milestone-closeout.ts +109 -24
- package/src/resources/extensions/gsd/milestone-planning-persistence.ts +2 -2
- package/src/resources/extensions/gsd/milestone-reopen-events.ts +3 -6
- package/src/resources/extensions/gsd/milestone-settlement.ts +2 -2
- package/src/resources/extensions/gsd/notifications.ts +13 -6
- package/src/resources/extensions/gsd/parsers-legacy.ts +16 -4
- package/src/resources/extensions/gsd/projection-flush.ts +20 -0
- package/src/resources/extensions/gsd/prompts/complete-slice.md +3 -3
- package/src/resources/extensions/gsd/prompts/execute-task.md +3 -2
- package/src/resources/extensions/gsd/prompts/plan-milestone.md +1 -1
- package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/quick-task.md +1 -1
- package/src/resources/extensions/gsd/prompts/reassess-roadmap.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/research-milestone.md +1 -1
- package/src/resources/extensions/gsd/prompts/research-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
- package/src/resources/extensions/gsd/prompts/run-uat.md +8 -4
- package/src/resources/extensions/gsd/prompts/triage-captures.md +1 -1
- package/src/resources/extensions/gsd/prompts/validate-milestone.md +1 -1
- package/src/resources/extensions/gsd/prompts/workflow-start.md +2 -1
- package/src/resources/extensions/gsd/reactive-graph.ts +11 -1
- package/src/resources/extensions/gsd/roadmap-slices.ts +28 -3
- package/src/resources/extensions/gsd/session-lock.ts +1 -1
- package/src/resources/extensions/gsd/skill-activation.ts +3 -6
- package/src/resources/extensions/gsd/state.ts +12 -1
- package/src/resources/extensions/gsd/tests/auto-abort-pause-regression.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/auto-blocked-remediation-message.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +206 -22
- package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +75 -1
- package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +257 -18
- package/src/resources/extensions/gsd/tests/auto-pause-double-entry-guard.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +77 -1
- package/src/resources/extensions/gsd/tests/auto-phases-lifecycle.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/auto-remote-session-lock-cleanup.test.ts +65 -3
- package/src/resources/extensions/gsd/tests/auto-start-orphan-bootstrap.test.ts +236 -0
- package/src/resources/extensions/gsd/tests/auto-unit-closeout.test.ts +169 -1
- package/src/resources/extensions/gsd/tests/blocked-models.test.ts +19 -0
- package/src/resources/extensions/gsd/tests/complete-slice-verification-gate.test.ts +42 -0
- package/src/resources/extensions/gsd/tests/complete-task.test.ts +141 -5
- 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/db-migration-backup.test.ts +68 -19
- package/src/resources/extensions/gsd/tests/db-transaction.test.ts +59 -0
- package/src/resources/extensions/gsd/tests/db-writer.test.ts +15 -4
- package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +12 -11
- package/src/resources/extensions/gsd/tests/derive-state-helpers.test.ts +62 -0
- package/src/resources/extensions/gsd/tests/discuss-routing-fixes.test.ts +12 -2
- package/src/resources/extensions/gsd/tests/dispatch-history.test.ts +328 -0
- package/src/resources/extensions/gsd/tests/dist-redirect.mjs +8 -0
- package/src/resources/extensions/gsd/tests/doctor-git-checks-terminal.test.ts +73 -0
- package/src/resources/extensions/gsd/tests/engine-hook-contract.test.ts +148 -0
- package/src/resources/extensions/gsd/tests/engine-interfaces-contract.test.ts +117 -91
- package/src/resources/extensions/gsd/tests/ensure-db-open.test.ts +113 -0
- package/src/resources/extensions/gsd/tests/gsd-command-home.test.ts +120 -0
- package/src/resources/extensions/gsd/tests/gsd-db.test.ts +46 -0
- package/src/resources/extensions/gsd/tests/guidance.test.ts +23 -0
- package/src/resources/extensions/gsd/tests/guided-dispatch-root.test.ts +18 -6
- package/src/resources/extensions/gsd/tests/integration/auto-worktree.test.ts +15 -0
- package/src/resources/extensions/gsd/tests/integration/doctor-environment-async.test.ts +104 -0
- package/src/resources/extensions/gsd/tests/integration/run-uat.test.ts +217 -0
- package/src/resources/extensions/gsd/tests/journal-integration.test.ts +47 -16
- package/src/resources/extensions/gsd/tests/mcp-readiness-preflight.test.ts +205 -0
- package/src/resources/extensions/gsd/tests/mcp-status.test.ts +6 -5
- package/src/resources/extensions/gsd/tests/milestone-closeout.test.ts +95 -4
- package/src/resources/extensions/gsd/tests/milestone-merge-stash-restore.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/milestone-report-path.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/milestone-settlement.test.ts +92 -0
- package/src/resources/extensions/gsd/tests/milestone-transition-state-rebuild.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/notifications.test.ts +64 -9
- package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +18 -0
- package/src/resources/extensions/gsd/tests/parallel-skill-prompt-integration.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/parsers-legacy-importers.test.ts +143 -0
- package/src/resources/extensions/gsd/tests/phases-merge-error-stops-auto.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/phases-terminal-complete-idempotent.test.ts +242 -0
- package/src/resources/extensions/gsd/tests/plan-gate-failed-doctor-heal-hint.test.ts +3 -3
- package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +63 -2
- package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +10 -2
- package/src/resources/extensions/gsd/tests/prompt-db.test.ts +124 -6
- package/src/resources/extensions/gsd/tests/provider-errors.test.ts +2 -4
- package/src/resources/extensions/gsd/tests/remote-notification-from-desktop.test.ts +31 -81
- package/src/resources/extensions/gsd/tests/roadmap-slices.test.ts +68 -0
- package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +26 -2
- package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +170 -48
- package/src/resources/extensions/gsd/tests/skill-activation.test.ts +20 -17
- package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +7 -3
- package/src/resources/extensions/gsd/tests/stop-auto-race-null-unit.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/teardown-chdir-failure-clears-registry.test.ts +17 -0
- package/src/resources/extensions/gsd/tests/token-tool-gating.test.ts +4 -2
- package/src/resources/extensions/gsd/tests/tool-surface-readiness.test.ts +184 -10
- package/src/resources/extensions/gsd/tests/tool-unavailable-retry.test.ts +33 -0
- package/src/resources/extensions/gsd/tests/transport-gate-double-complete.test.ts +139 -0
- package/src/resources/extensions/gsd/tests/uat-policy.test.ts +88 -0
- package/src/resources/extensions/gsd/tests/uok-audit-unified.test.ts +8 -0
- package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/verification-verdict.test.ts +2 -0
- package/src/resources/extensions/gsd/tests/workflow-events.test.ts +19 -0
- package/src/resources/extensions/gsd/tests/workflow-mcp-readiness-cache.test.ts +119 -0
- package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +65 -2
- package/src/resources/extensions/gsd/tests/workflow-phase-contract-matrix.test.ts +332 -0
- package/src/resources/extensions/gsd/tests/workflow-reconcile.test.ts +20 -0
- package/src/resources/extensions/gsd/tests/workflow-templates.test.ts +92 -0
- package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +273 -38
- package/src/resources/extensions/gsd/tests/worktree-health-dispatch.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/worktree-project-root-degrade.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/worktree-safety-phase.test.ts +100 -0
- package/src/resources/extensions/gsd/tests/worktree-safety.test.ts +72 -0
- package/src/resources/extensions/gsd/tests/worktree-teardown-safety.test.ts +22 -0
- package/src/resources/extensions/gsd/tests/worktree.test.ts +18 -0
- package/src/resources/extensions/gsd/tests/write-gate-seam.test.ts +358 -0
- package/src/resources/extensions/gsd/tests/write-gate.test.ts +67 -1
- package/src/resources/extensions/gsd/tool-contract.ts +38 -3
- package/src/resources/extensions/gsd/tool-presentation-plan.ts +4 -4
- package/src/resources/extensions/gsd/tool-surface-readiness.ts +126 -19
- package/src/resources/extensions/gsd/tools/complete-milestone.ts +3 -2
- package/src/resources/extensions/gsd/tools/complete-slice.ts +22 -12
- package/src/resources/extensions/gsd/tools/complete-task.ts +90 -2
- package/src/resources/extensions/gsd/tools/plan-slice.ts +2 -2
- package/src/resources/extensions/gsd/tools/plan-task.ts +2 -2
- package/src/resources/extensions/gsd/tools/reassess-roadmap.ts +2 -2
- package/src/resources/extensions/gsd/tools/reopen-milestone.ts +2 -2
- package/src/resources/extensions/gsd/tools/reopen-slice.ts +2 -2
- package/src/resources/extensions/gsd/tools/reopen-task.ts +2 -2
- package/src/resources/extensions/gsd/tools/replan-slice.ts +2 -2
- package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +81 -2
- package/src/resources/extensions/gsd/uat-policy.ts +60 -15
- package/src/resources/extensions/gsd/unit-context-composer.ts +1 -1
- package/src/resources/extensions/gsd/unit-registry.ts +34 -4
- package/src/resources/extensions/gsd/verdict-parser.ts +1 -1
- package/src/resources/extensions/gsd/verification-verdict.ts +4 -2
- package/src/resources/extensions/gsd/workflow-event-ledger.ts +131 -0
- package/src/resources/extensions/gsd/workflow-event-vocabulary.ts +59 -0
- package/src/resources/extensions/gsd/workflow-events.ts +12 -20
- package/src/resources/extensions/gsd/workflow-mcp-auto-prep.ts +2 -0
- package/src/resources/extensions/gsd/workflow-mcp-readiness-cache.ts +150 -0
- package/src/resources/extensions/gsd/workflow-reconcile.ts +29 -62
- package/src/resources/extensions/gsd/worktree-lifecycle.ts +3 -8
- package/src/resources/extensions/gsd/worktree-manager.ts +6 -1
- package/src/resources/extensions/gsd/worktree-safety.ts +41 -39
- package/src/resources/extensions/gsd/worktree.ts +7 -1
- package/src/resources/extensions/mcp-client/manager.ts +7 -1
- package/src/resources/extensions/shared/gsd-browser-cli.ts +23 -2
- package/src/resources/shared/gsd-browser-path-sync.ts +273 -0
- package/src/resources/shared/package-manager-detection.ts +1 -1
- package/src/resources/skills/create-skill/SKILL.md +3 -0
- package/src/resources/skills/create-skill/references/skill-structure.md +1 -0
- package/dist/resources/extensions/gsd/user-input-boundary.js +0 -218
- package/dist/resources/skills/gsd-browser/SKILL.md +0 -41
- 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/src/resources/skills/gsd-browser/SKILL.md +0 -41
- /package/dist/web/standalone/.next/static/{mU4QIDlpVHDdjDpeEKh5W → 2T9IOdiiM3o3gZ4UbPi8E}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{mU4QIDlpVHDdjDpeEKh5W → 2T9IOdiiM3o3gZ4UbPi8E}/_ssgManifest.js +0 -0
|
@@ -7,7 +7,7 @@
|
|
|
7
7
|
import { existsSync, readFileSync, mkdirSync } from "node:fs";
|
|
8
8
|
import { execFileSync } from "node:child_process";
|
|
9
9
|
import { createRequire } from "node:module";
|
|
10
|
-
import { join, resolve as resolvePath, sep } from "node:path";
|
|
10
|
+
import { join, resolve as resolvePath, sep, win32 as pathWin32 } from "node:path";
|
|
11
11
|
import { homedir } from "node:os";
|
|
12
12
|
import { deriveState } from "./state.js";
|
|
13
13
|
import { gsdRoot } from "./paths.js";
|
|
@@ -20,6 +20,7 @@ import { getAutoWorktreePath } from "./auto-worktree.js";
|
|
|
20
20
|
import { currentDirectoryRoot, projectRoot } from "./commands/context.js";
|
|
21
21
|
import { loadPrompt } from "./prompt-loader.js";
|
|
22
22
|
import { buildClaudeRuntimeFloorAdvisory } from "../../shared/claude-runtime-floor.js";
|
|
23
|
+
import { reconcileGsdBrowserPathAfterInstall } from "../../shared/gsd-browser-path-sync.js";
|
|
23
24
|
import { isPnpmInstall } from "../../shared/package-manager-detection.js";
|
|
24
25
|
import { buildDoctorHealIssuePayload, buildDoctorHealSummary, buildWorkflowDispatchContent, } from "./workflow-protocol.js";
|
|
25
26
|
import { restoreGsdWorkflowTools, scopeGsdWorkflowToolsForDispatch, } from "./bootstrap/register-hooks.js";
|
|
@@ -51,8 +52,31 @@ function resolveInstallCommand(pkg) {
|
|
|
51
52
|
return `bun add -g ${pkg}`;
|
|
52
53
|
if (isPnpmInstall())
|
|
53
54
|
return `pnpm add -g ${pkg}`;
|
|
55
|
+
const npmPrefix = resolveWindowsNpmGlobalPrefix();
|
|
56
|
+
if (npmPrefix)
|
|
57
|
+
return `npm --prefix ${quoteWindowsArg(npmPrefix)} install -g ${pkg}`;
|
|
54
58
|
return `npm install -g ${pkg}`;
|
|
55
59
|
}
|
|
60
|
+
function resolveWindowsNpmGlobalPrefix(argv1 = process.argv[1], platform = process.platform) {
|
|
61
|
+
if (platform !== "win32" || !argv1)
|
|
62
|
+
return null;
|
|
63
|
+
const normalized = pathWin32.normalize(argv1);
|
|
64
|
+
const marker = `${pathWin32.sep}node_modules${pathWin32.sep}`;
|
|
65
|
+
const index = normalized.toLowerCase().lastIndexOf(marker);
|
|
66
|
+
if (index <= 0)
|
|
67
|
+
return null;
|
|
68
|
+
const prefix = normalized.slice(0, index);
|
|
69
|
+
// Verify this is a real npm global prefix: such a directory always contains
|
|
70
|
+
// npm's own bin shim (`npm.cmd`) as a sibling of `node_modules/`. Local
|
|
71
|
+
// project `node_modules/`, npx caches, and other non-global layouts do not,
|
|
72
|
+
// so without this check `--prefix` would target the wrong directory.
|
|
73
|
+
if (!existsSync(pathWin32.join(prefix, "npm.cmd")))
|
|
74
|
+
return null;
|
|
75
|
+
return prefix;
|
|
76
|
+
}
|
|
77
|
+
function quoteWindowsArg(value) {
|
|
78
|
+
return `"${value.replace(/"/g, '\\"')}"`;
|
|
79
|
+
}
|
|
56
80
|
function notifyClaudeRuntimeFloorAdvisory(ctx) {
|
|
57
81
|
let advisory = null;
|
|
58
82
|
try {
|
|
@@ -484,11 +508,30 @@ export async function handleUpdate(ctx, args = "") {
|
|
|
484
508
|
execSync(installCmd, {
|
|
485
509
|
stdio: ["ignore", "pipe", "ignore"],
|
|
486
510
|
});
|
|
511
|
+
let reconcile = null;
|
|
512
|
+
if (browserUpdate) {
|
|
513
|
+
try {
|
|
514
|
+
reconcile = reconcileGsdBrowserPathAfterInstall({
|
|
515
|
+
latestVersion: latest,
|
|
516
|
+
compareSemver: compareSemverLocal,
|
|
517
|
+
resolvePathVersion: resolveGsdBrowserPathVersionForCommand,
|
|
518
|
+
});
|
|
519
|
+
}
|
|
520
|
+
catch {
|
|
521
|
+
// Reconciliation is best-effort: the install above already succeeded,
|
|
522
|
+
// so a reconcile failure must not flip the result to "Update failed".
|
|
523
|
+
reconcile = null;
|
|
524
|
+
}
|
|
525
|
+
}
|
|
487
526
|
const newPathVersion = browserUpdate ? resolveGsdBrowserPathVersionForCommand() : null;
|
|
488
|
-
const
|
|
527
|
+
const pathNote = browserUpdate && !(newPathVersion && compareSemverLocal(newPathVersion, latest) >= 0)
|
|
528
|
+
? (reconcile?.message
|
|
529
|
+
?? "Ensure the npm global bin directory is on your PATH so MCP automation uses the updated binary.")
|
|
530
|
+
: "";
|
|
489
531
|
ctx.ui.notify(browserUpdate
|
|
490
532
|
? `Updated gsd-browser to v${latest}. Restart your GSD session to use the new browser automation version.` +
|
|
491
|
-
(
|
|
533
|
+
(reconcile?.action === "synced" && reconcile.message ? `\n${reconcile.message}` : "") +
|
|
534
|
+
(pathNote ? `\nNote: ${pathNote}` : "")
|
|
492
535
|
: `Updated to v${latest}. Restart your GSD session to use the new version.`, "info");
|
|
493
536
|
if (!browserUpdate)
|
|
494
537
|
notifyClaudeRuntimeFloorAdvisory(ctx);
|
|
@@ -59,7 +59,7 @@ export function formatMcpStatusReport(servers) {
|
|
|
59
59
|
: s.connected
|
|
60
60
|
? `connected — ${s.toolCount} tools`
|
|
61
61
|
: s.available
|
|
62
|
-
? `available — ${s.toolCount} tools`
|
|
62
|
+
? `probe available — ${s.toolCount} tools`
|
|
63
63
|
: "disconnected";
|
|
64
64
|
const warningText = s.envWarnings?.length ? ` — ${s.envWarnings.length} warning(s)` : "";
|
|
65
65
|
lines.push(` ${icon} ${s.name} (${s.transport}) — ${status}${warningText}`);
|
|
@@ -83,7 +83,7 @@ export function formatMcpServerDetail(server) {
|
|
|
83
83
|
lines.push(` Error: ${server.error}`);
|
|
84
84
|
}
|
|
85
85
|
else if (server.connected || server.available) {
|
|
86
|
-
lines.push(` Status: ${server.connected ? "connected" : "available"}`);
|
|
86
|
+
lines.push(` Status: ${server.connected ? "connected" : "probe available"}`);
|
|
87
87
|
lines.push(` Tools: ${server.toolCount}`);
|
|
88
88
|
if (server.tools.length > 0) {
|
|
89
89
|
lines.push("");
|
|
@@ -59,6 +59,13 @@ function datePrefix() {
|
|
|
59
59
|
const dd = String(d.getDate()).padStart(2, "0");
|
|
60
60
|
return `${yy}${mm}${dd}`;
|
|
61
61
|
}
|
|
62
|
+
export function isWorkflowStateComplete(state) {
|
|
63
|
+
if (state.completedAt)
|
|
64
|
+
return true;
|
|
65
|
+
return Array.isArray(state.phases) &&
|
|
66
|
+
state.phases.length > 0 &&
|
|
67
|
+
state.phases.every((phase) => phase.status === "completed");
|
|
68
|
+
}
|
|
62
69
|
/**
|
|
63
70
|
* Write a STATE.json file to track workflow execution state.
|
|
64
71
|
*/
|
|
@@ -85,7 +92,7 @@ function writeWorkflowState(artifactDir, templateId, templateName, phases, descr
|
|
|
85
92
|
* Scan all workflow artifact directories for in-progress STATE.json files.
|
|
86
93
|
* Returns workflows that were started but not completed.
|
|
87
94
|
*/
|
|
88
|
-
function findInProgressWorkflows(basePath) {
|
|
95
|
+
export function findInProgressWorkflows(basePath) {
|
|
89
96
|
const workflowsRoot = join(gsdRoot(basePath), "workflows");
|
|
90
97
|
if (!existsSync(workflowsRoot))
|
|
91
98
|
return [];
|
|
@@ -105,7 +112,7 @@ function findInProgressWorkflows(basePath) {
|
|
|
105
112
|
try {
|
|
106
113
|
const raw = readFileSync(statePath, "utf-8");
|
|
107
114
|
const state = JSON.parse(raw);
|
|
108
|
-
if (!state
|
|
115
|
+
if (!isWorkflowStateComplete(state)) {
|
|
109
116
|
results.push(state);
|
|
110
117
|
}
|
|
111
118
|
}
|
|
@@ -0,0 +1,353 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Consent Question module — the single home for the user-question lifecycle:
|
|
3
|
+
* classification → gating → answer validation → cancellation.
|
|
4
|
+
*
|
|
5
|
+
* Every question the assistant puts to a human is classified into a kind, and
|
|
6
|
+
* the kind alone decides the fail policy:
|
|
7
|
+
*
|
|
8
|
+
* - "gate" — mechanical write gates (depth verification, destructive
|
|
9
|
+
* confirm). Fail-closed; structural answer validation is
|
|
10
|
+
* delegated to the write-gate validators.
|
|
11
|
+
* - "consent" — approval/confirmation questions ("ready to write?",
|
|
12
|
+
* "is this correct?"). Fail-closed.
|
|
13
|
+
* - "decision" — explicit user decisions (research vs skip). Fail-closed.
|
|
14
|
+
* - "informational" — anything that is not asking the user to consent or
|
|
15
|
+
* decide. Fail-open.
|
|
16
|
+
*
|
|
17
|
+
* Fail-closed means an empty/missing answer is NEVER treated as an answer —
|
|
18
|
+
* evaluateAnswer returns "waiting" so callers pause instead of proceeding.
|
|
19
|
+
* This fixes #528 (empty `selected` on a non-gate question used to pass
|
|
20
|
+
* through as a real answer) by construction.
|
|
21
|
+
*
|
|
22
|
+
* shouldPauseForQuestion replaces the old unit-type allowlist: a classified
|
|
23
|
+
* consent/decision question pauses regardless of unit type, including
|
|
24
|
+
* interactive mode where no unit is active. This fixes #682 (prose approval
|
|
25
|
+
* questions outside the 4 allowlisted discuss units rendered as un-gated
|
|
26
|
+
* prose menus) by construction.
|
|
27
|
+
*/
|
|
28
|
+
import { isGateQuestionId } from "./bootstrap/write-gate.js";
|
|
29
|
+
import { evaluateGateAnswer, hasNotesValue, hasSelectedValue, } from "./consent-verdict.js";
|
|
30
|
+
import { isDestructiveConfirmGateId } from "./safety/destructive-confirmation.js";
|
|
31
|
+
/** Fail policy is derived from the kind — there is no per-question override. */
|
|
32
|
+
export function failPolicyForKind(kind) {
|
|
33
|
+
return kind === "informational" ? "open" : "closed";
|
|
34
|
+
}
|
|
35
|
+
// ── Prose detectors (moved from user-input-boundary) ────────────────────────
|
|
36
|
+
const REMOTE_QUESTION_FAILURE_RE = /(?:Remote (?:auth failed|questions failed|channel configured but returned no result|questions timed out|questions timed out or failed)|Failed to send questions via)/i;
|
|
37
|
+
const APPROVAL_WAIT_RE = /\bwait(?:ing)?\s+for\s+(?:your\s+)?(?:confirmation|approval|input|response|answer)\b/i;
|
|
38
|
+
const APPROVAL_QUESTION_RE = /\b(?:confirm|confirmation|approve|approval|approved|captured|correct|correctly|happy\s+with|ready\s+to\s+(?:write|save|proceed|ship)|(?:want|need)\s+to\s+adjust|should\s+I\s+(?:write|save|proceed)|do\s+you\s+want\s+me\s+to\s+(?:write|save|proceed)|ship\s+it)\b/i;
|
|
39
|
+
const APPROVAL_RIGHT_QUESTION_RE = /\b(?:does|do|is|are|was|were|did)\b[^\n?]{0,120}\bright\b/i;
|
|
40
|
+
const APPROVAL_CHANGE_QUESTION_RE = /\b(?:anything\s+else|anything|something)\s+to\s+(?:adjust|add|remove|reclassify)\b/i;
|
|
41
|
+
const RESEARCH_DECISION_QUESTION_RE = /\b(?:research|skip)\b/i;
|
|
42
|
+
const ASK_USER_QUESTIONS_CANCELLED_RE = /ask_user_questions was cancelled before receiving a response/i;
|
|
43
|
+
/** Scan question-mark-terminated fragments of `text` against `patterns`. */
|
|
44
|
+
function hasQuestionMatching(text, patterns) {
|
|
45
|
+
for (let i = 0; i < text.length; i++) {
|
|
46
|
+
if (text[i] !== "?")
|
|
47
|
+
continue;
|
|
48
|
+
const previousBreak = Math.max(text.lastIndexOf("\n", i), text.lastIndexOf(".", i), text.lastIndexOf("!", i), text.lastIndexOf("?", i - 1));
|
|
49
|
+
const fragment = text.slice(previousBreak + 1, i + 1);
|
|
50
|
+
if (patterns.some((pattern) => pattern.test(fragment)))
|
|
51
|
+
return true;
|
|
52
|
+
}
|
|
53
|
+
return false;
|
|
54
|
+
}
|
|
55
|
+
export function hasApprovalQuestion(text) {
|
|
56
|
+
return hasQuestionMatching(text, [
|
|
57
|
+
APPROVAL_QUESTION_RE,
|
|
58
|
+
APPROVAL_RIGHT_QUESTION_RE,
|
|
59
|
+
APPROVAL_CHANGE_QUESTION_RE,
|
|
60
|
+
]);
|
|
61
|
+
}
|
|
62
|
+
export function hasResearchDecisionQuestion(text) {
|
|
63
|
+
return hasQuestionMatching(text, [RESEARCH_DECISION_QUESTION_RE]);
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Detect a plain-text "Next steps:" menu — numbered options with an "Other"
|
|
67
|
+
* choice — emitted as prose instead of a structured ask_user_questions call.
|
|
68
|
+
* Without this, auto-mode treats the menu as informational and loops on its
|
|
69
|
+
* own turn until tokens are exhausted (#454).
|
|
70
|
+
*/
|
|
71
|
+
export function hasPlainTextNextStepsMenu(lines) {
|
|
72
|
+
const nextStepsIndex = lines.findIndex((line) => /^next steps\s*:?$/i.test(line));
|
|
73
|
+
if (nextStepsIndex < 0)
|
|
74
|
+
return false;
|
|
75
|
+
const menuLines = lines.slice(nextStepsIndex + 1);
|
|
76
|
+
const numberedOptions = menuLines.filter((line) => /^\d+[.)]\s+\S/.test(line));
|
|
77
|
+
return numberedOptions.length >= 2 && numberedOptions.some((line) => /\bother\b/i.test(line));
|
|
78
|
+
}
|
|
79
|
+
// ── Message text extraction (moved from user-input-boundary) ────────────────
|
|
80
|
+
function extractMessageText(msg, includeThinking) {
|
|
81
|
+
if (!msg || typeof msg !== "object")
|
|
82
|
+
return "";
|
|
83
|
+
const content = msg.content;
|
|
84
|
+
if (typeof content === "string")
|
|
85
|
+
return content;
|
|
86
|
+
if (!Array.isArray(content))
|
|
87
|
+
return "";
|
|
88
|
+
const parts = [];
|
|
89
|
+
for (const block of content) {
|
|
90
|
+
if (!block || typeof block !== "object")
|
|
91
|
+
continue;
|
|
92
|
+
const typed = block;
|
|
93
|
+
if (typed.type === "text" && typeof typed.text === "string") {
|
|
94
|
+
parts.push(typed.text);
|
|
95
|
+
}
|
|
96
|
+
// Thinking blocks are internal reasoning, not user-visible — included only
|
|
97
|
+
// when the caller asks for the full transcript text.
|
|
98
|
+
if (includeThinking && typed.type === "thinking" && typeof typed.thinking === "string") {
|
|
99
|
+
parts.push(typed.thinking);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
return parts.join("\n");
|
|
103
|
+
}
|
|
104
|
+
function lastAssistantMessageText(messages, includeThinking) {
|
|
105
|
+
if (!Array.isArray(messages))
|
|
106
|
+
return "";
|
|
107
|
+
for (let i = messages.length - 1; i >= 0; i--) {
|
|
108
|
+
const msg = messages[i];
|
|
109
|
+
if (!msg || typeof msg !== "object")
|
|
110
|
+
continue;
|
|
111
|
+
if (msg.role !== "assistant")
|
|
112
|
+
continue;
|
|
113
|
+
const text = extractMessageText(msg, includeThinking).trim();
|
|
114
|
+
if (text)
|
|
115
|
+
return text;
|
|
116
|
+
}
|
|
117
|
+
return "";
|
|
118
|
+
}
|
|
119
|
+
export function lastAssistantText(messages) {
|
|
120
|
+
return lastAssistantMessageText(messages, true);
|
|
121
|
+
}
|
|
122
|
+
function lastAssistantVisibleText(messages) {
|
|
123
|
+
return lastAssistantMessageText(messages, false);
|
|
124
|
+
}
|
|
125
|
+
function anyMessageMatches(messages, patterns) {
|
|
126
|
+
if (!Array.isArray(messages))
|
|
127
|
+
return false;
|
|
128
|
+
return messages.some((msg) => {
|
|
129
|
+
if (!msg || typeof msg !== "object")
|
|
130
|
+
return false;
|
|
131
|
+
if (msg.role === "user")
|
|
132
|
+
return false;
|
|
133
|
+
const text = extractMessageText(msg, false);
|
|
134
|
+
return patterns.some((pattern) => pattern.test(text));
|
|
135
|
+
});
|
|
136
|
+
}
|
|
137
|
+
// ── Classification ──────────────────────────────────────────────────────────
|
|
138
|
+
const APPROVAL_GATE_ID_RE = /^depth_verification_.+_confirm$/;
|
|
139
|
+
/**
|
|
140
|
+
* Classify a question — structured (by id) or prose (by text) — into a kind.
|
|
141
|
+
*
|
|
142
|
+
* Structured questions with a recognized gate id are gates; every other
|
|
143
|
+
* structured question is asking the user something, so it classifies as
|
|
144
|
+
* consent (fail-closed). Prose classifies by the approval/decision detectors;
|
|
145
|
+
* prose that matches neither is informational (fail-open).
|
|
146
|
+
*/
|
|
147
|
+
export function classifyQuestion(input) {
|
|
148
|
+
if (isDestructiveConfirmGateId(input.id)) {
|
|
149
|
+
return { kind: "gate", gateSubKind: "destructive-confirm" };
|
|
150
|
+
}
|
|
151
|
+
if (typeof input.id === "string" && isGateQuestionId(input.id)) {
|
|
152
|
+
return {
|
|
153
|
+
kind: "gate",
|
|
154
|
+
gateSubKind: APPROVAL_GATE_ID_RE.test(input.id) ? "approval" : "depth-verification",
|
|
155
|
+
};
|
|
156
|
+
}
|
|
157
|
+
// Any other structured question is a real elicitation of the user.
|
|
158
|
+
if (typeof input.id === "string" || Array.isArray(input.options)) {
|
|
159
|
+
return { kind: "consent" };
|
|
160
|
+
}
|
|
161
|
+
const text = input.text ?? "";
|
|
162
|
+
if (input.unitType === "research-decision" && hasResearchDecisionQuestion(text)) {
|
|
163
|
+
return { kind: "decision" };
|
|
164
|
+
}
|
|
165
|
+
if (hasApprovalQuestion(text)) {
|
|
166
|
+
return { kind: "consent" };
|
|
167
|
+
}
|
|
168
|
+
return { kind: "informational" };
|
|
169
|
+
}
|
|
170
|
+
/**
|
|
171
|
+
* THE single policy point for whether a question's answer counts as answered.
|
|
172
|
+
*
|
|
173
|
+
* - cancelled rounds → "cancelled" for every kind.
|
|
174
|
+
* - gate: delegates to evaluateGateAnswer in the consent-verdict leaf — the
|
|
175
|
+
* same verdict engine write-gate's applyAskUserQuestionsGateResult consumes;
|
|
176
|
+
* confirm option → "verified", any other real selection → "declined",
|
|
177
|
+
* empty/missing → "waiting" (fail-closed).
|
|
178
|
+
* - consent/decision: a non-empty selection or non-empty notes → "answered";
|
|
179
|
+
* empty/missing → "waiting" (fail-closed; fixes #528).
|
|
180
|
+
* - informational: always "answered" (fail-open).
|
|
181
|
+
*/
|
|
182
|
+
export function evaluateAnswer(options) {
|
|
183
|
+
const { question, details } = options;
|
|
184
|
+
if (details.cancelled)
|
|
185
|
+
return "cancelled";
|
|
186
|
+
const { kind } = classifyQuestion({ id: question.id, options: question.options });
|
|
187
|
+
if (failPolicyForKind(kind) === "open")
|
|
188
|
+
return "answered";
|
|
189
|
+
if (kind === "gate") {
|
|
190
|
+
// Gates keep strict structural validation: only the confirmation option
|
|
191
|
+
// verifies; notes never satisfy a gate.
|
|
192
|
+
return evaluateGateAnswer(question, details);
|
|
193
|
+
}
|
|
194
|
+
const questionId = typeof question.id === "string" ? question.id : "";
|
|
195
|
+
const answer = details.response?.answers?.[questionId];
|
|
196
|
+
if (hasSelectedValue(answer?.selected))
|
|
197
|
+
return "answered";
|
|
198
|
+
// Notes-only is a real user utterance for consent/decision questions, but
|
|
199
|
+
// never for gates (handled above).
|
|
200
|
+
if (hasNotesValue(answer?.notes))
|
|
201
|
+
return "answered";
|
|
202
|
+
return "waiting";
|
|
203
|
+
}
|
|
204
|
+
const OUTCOME_PRECEDENCE = [
|
|
205
|
+
"cancelled",
|
|
206
|
+
"waiting",
|
|
207
|
+
"declined",
|
|
208
|
+
"verified",
|
|
209
|
+
"answered",
|
|
210
|
+
];
|
|
211
|
+
/**
|
|
212
|
+
* Evaluate a whole ask_user_questions round. The round outcome is the most
|
|
213
|
+
* blocking per-question outcome (cancelled > waiting > declined > verified >
|
|
214
|
+
* answered). An empty round with a response is "answered"; an empty round
|
|
215
|
+
* without a response is "waiting" only when cancelled is not set and there is
|
|
216
|
+
* nothing to validate — callers treat a missing response with no questions as
|
|
217
|
+
* a no-op, so it reports "answered" here.
|
|
218
|
+
*/
|
|
219
|
+
export function evaluateAskUserQuestionsRound(questions, details) {
|
|
220
|
+
if (details.cancelled)
|
|
221
|
+
return "cancelled";
|
|
222
|
+
let worst = "answered";
|
|
223
|
+
for (const question of questions) {
|
|
224
|
+
const outcome = evaluateAnswer({ question, details });
|
|
225
|
+
if (OUTCOME_PRECEDENCE.indexOf(outcome) < OUTCOME_PRECEDENCE.indexOf(worst)) {
|
|
226
|
+
worst = outcome;
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
return worst;
|
|
230
|
+
}
|
|
231
|
+
export function formatUnansweredConsentQuestionMessage(questions) {
|
|
232
|
+
const ids = questions
|
|
233
|
+
.map((question) => (typeof question.id === "string" ? question.id : null))
|
|
234
|
+
.filter((id) => Boolean(id));
|
|
235
|
+
return [
|
|
236
|
+
`ask_user_questions returned without a selection${ids.length ? ` for ${ids.join(", ")}` : ""}.`,
|
|
237
|
+
"An empty answer is not consent — do not infer approval or proceed.",
|
|
238
|
+
"Re-call ask_user_questions with the same question(s) and wait for the user's response.",
|
|
239
|
+
].join(" ");
|
|
240
|
+
}
|
|
241
|
+
// ── Pause gating (replaces the unit-type allowlist) ─────────────────────────
|
|
242
|
+
/**
|
|
243
|
+
* Shared preamble for the awaiting-input predicates: cancellation and remote
|
|
244
|
+
* delivery failures always pause (an undelivered question can never be
|
|
245
|
+
* answered, so proceeding would be fail-open), as does an explicit
|
|
246
|
+
* "waiting for your approval/input" phrase in the last assistant text.
|
|
247
|
+
*
|
|
248
|
+
* Returns `forced: true` when the boundary is unconditional, plus the last
|
|
249
|
+
* assistant visible text for the caller's own classification.
|
|
250
|
+
*/
|
|
251
|
+
function awaitingBoundary(messages) {
|
|
252
|
+
if (anyMessageMatches(messages, [ASK_USER_QUESTIONS_CANCELLED_RE, REMOTE_QUESTION_FAILURE_RE])) {
|
|
253
|
+
return { forced: true, text: "" };
|
|
254
|
+
}
|
|
255
|
+
const text = lastAssistantVisibleText(messages);
|
|
256
|
+
if (text && APPROVAL_WAIT_RE.test(text))
|
|
257
|
+
return { forced: true, text };
|
|
258
|
+
return { forced: false, text };
|
|
259
|
+
}
|
|
260
|
+
/**
|
|
261
|
+
* Decide whether the assistant should pause for a prose user question.
|
|
262
|
+
*
|
|
263
|
+
* Unlike the retired USER_APPROVAL_UNIT_TYPES allowlist, this pauses for any
|
|
264
|
+
* classified consent/decision question regardless of unit type — including
|
|
265
|
+
* interactive mode where no unit is active (#682).
|
|
266
|
+
*/
|
|
267
|
+
export function shouldPauseForQuestion(unitType, messages) {
|
|
268
|
+
const { forced, text } = awaitingBoundary(messages);
|
|
269
|
+
if (forced)
|
|
270
|
+
return true;
|
|
271
|
+
// Streaming hot path: this runs on every message_update for every unit type.
|
|
272
|
+
// The classifiers only ever match question-mark-terminated fragments, so
|
|
273
|
+
// text without a "?" can never classify as consent/decision — bail before
|
|
274
|
+
// the multi-regex scan.
|
|
275
|
+
if (!text || !text.includes("?"))
|
|
276
|
+
return false;
|
|
277
|
+
const { kind } = classifyQuestion({ text, unitType });
|
|
278
|
+
return kind === "consent" || kind === "decision";
|
|
279
|
+
}
|
|
280
|
+
// ── Awaiting-input boundaries (moved from user-input-boundary) ──────────────
|
|
281
|
+
export function isAwaitingUserInput(messages) {
|
|
282
|
+
const { forced, text } = awaitingBoundary(messages);
|
|
283
|
+
if (forced)
|
|
284
|
+
return true;
|
|
285
|
+
if (!text)
|
|
286
|
+
return false;
|
|
287
|
+
const lines = text.split(/\r?\n/).map((line) => line.trim()).filter(Boolean);
|
|
288
|
+
if (lines.some((line) => line.endsWith("?")))
|
|
289
|
+
return true;
|
|
290
|
+
if (hasPlainTextNextStepsMenu(lines))
|
|
291
|
+
return true;
|
|
292
|
+
return hasApprovalQuestion(text);
|
|
293
|
+
}
|
|
294
|
+
export function isAwaitingApprovalBoundary(messages) {
|
|
295
|
+
// With no unit type, classification reduces to the approval detectors —
|
|
296
|
+
// exactly the approval boundary.
|
|
297
|
+
return shouldPauseForQuestion(undefined, messages);
|
|
298
|
+
}
|
|
299
|
+
// ── Approval gate ids + explicit responses (moved from user-input-boundary) ─
|
|
300
|
+
export function approvalGateIdForUnit(unitType, unitId) {
|
|
301
|
+
if (!unitType)
|
|
302
|
+
return null;
|
|
303
|
+
if (unitType === "discuss-project")
|
|
304
|
+
return "depth_verification_project_confirm";
|
|
305
|
+
if (unitType === "discuss-requirements")
|
|
306
|
+
return "depth_verification_requirements_confirm";
|
|
307
|
+
if (unitType === "research-decision")
|
|
308
|
+
return "depth_verification_research_decision_confirm";
|
|
309
|
+
if (unitType === "discuss-milestone") {
|
|
310
|
+
const safeUnitId = typeof unitId === "string" && /^[A-Za-z0-9_-]+$/.test(unitId)
|
|
311
|
+
? unitId
|
|
312
|
+
: "milestone";
|
|
313
|
+
return `depth_verification_${safeUnitId}_confirm`;
|
|
314
|
+
}
|
|
315
|
+
return null;
|
|
316
|
+
}
|
|
317
|
+
const CHANGE_REQUEST_RESPONSE_RE = /\b(?:no|nope|nah|not\s+yet|don't|do\s+not|change|add|remove|reclassify|adjust|clarify|missing|instead|but|however|wait|hold)\b/i;
|
|
318
|
+
const APPROVAL_RESPONSE_RE = /^(?:y|yes|yeah|yep|approve|approved|confirm|confirmed|correct|right|looks\s+(?:good|right)|sounds\s+good|all\s+good|ok|okay|go\s+ahead|proceed|write\s+it|save\s+it|do\s+it)\b/i;
|
|
319
|
+
const RESEARCH_DECISION_RESPONSE_RE = /^(?:research|run\s+research|do\s+research|skip|skip\s+research|no\s+research)\b/i;
|
|
320
|
+
export function isExplicitApprovalResponse(input, pendingGateId) {
|
|
321
|
+
const text = input?.trim() ?? "";
|
|
322
|
+
if (!text)
|
|
323
|
+
return false;
|
|
324
|
+
if (pendingGateId?.includes("research_decision")) {
|
|
325
|
+
return RESEARCH_DECISION_RESPONSE_RE.test(text);
|
|
326
|
+
}
|
|
327
|
+
if (CHANGE_REQUEST_RESPONSE_RE.test(text))
|
|
328
|
+
return false;
|
|
329
|
+
return APPROVAL_RESPONSE_RE.test(text);
|
|
330
|
+
}
|
|
331
|
+
/** True when an assistant message already has an in-flight ask_user_questions tool call. */
|
|
332
|
+
export function messageHasPendingAskUserQuestionsTool(message) {
|
|
333
|
+
if (!message || typeof message !== "object")
|
|
334
|
+
return false;
|
|
335
|
+
const content = message.content;
|
|
336
|
+
if (!Array.isArray(content))
|
|
337
|
+
return false;
|
|
338
|
+
return content.some((block) => {
|
|
339
|
+
if (!block || typeof block !== "object")
|
|
340
|
+
return false;
|
|
341
|
+
// Claude Code marks completion by attaching externalResult, not by setting state.
|
|
342
|
+
// Streaming blocks often carry no state; serverToolUse is the claude-code-cli MCP path.
|
|
343
|
+
const tool = block;
|
|
344
|
+
if (tool.type !== "toolCall" && tool.type !== "serverToolUse")
|
|
345
|
+
return false;
|
|
346
|
+
const name = String(tool.name ?? "").toLowerCase();
|
|
347
|
+
if (!name.includes("ask_user_questions"))
|
|
348
|
+
return false;
|
|
349
|
+
if (tool.externalResult !== undefined)
|
|
350
|
+
return false;
|
|
351
|
+
return tool.state !== "completed" && tool.state !== "done";
|
|
352
|
+
});
|
|
353
|
+
}
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Consent verdict leaf — the single per-question verdict engine shared by the
|
|
3
|
+
* write gate (bootstrap/write-gate.ts) and the Consent Question module
|
|
4
|
+
* (consent-question.ts).
|
|
5
|
+
*
|
|
6
|
+
* This module is a dependency leaf on purpose: write-gate consumes
|
|
7
|
+
* evaluateGateAnswer here while consent-question imports write-gate's id
|
|
8
|
+
* predicates, so putting the verdict anywhere else would create an import
|
|
9
|
+
* cycle. It must not import from either module (ADR-039).
|
|
10
|
+
*/
|
|
11
|
+
/**
|
|
12
|
+
* Check whether a depth_verification answer confirms the discussion is complete.
|
|
13
|
+
* Uses structural validation: the selected answer must exactly match the first
|
|
14
|
+
* option label from the question definition (the confirmation option by convention).
|
|
15
|
+
* This rejects free-form "Other" text, decline options, and garbage input without
|
|
16
|
+
* coupling to any specific label substring.
|
|
17
|
+
*
|
|
18
|
+
* @param selected The answer's selected value from details.response.answers[id].selected
|
|
19
|
+
* @param options The question's options array from event.input.questions[n].options
|
|
20
|
+
*/
|
|
21
|
+
export function isDepthConfirmationAnswer(selected, options) {
|
|
22
|
+
const value = Array.isArray(selected) ? selected[0] : selected;
|
|
23
|
+
if (typeof value !== "string" || !value)
|
|
24
|
+
return false;
|
|
25
|
+
// If options are available, structurally validate: selected must exactly match
|
|
26
|
+
// the first option (confirmation) label. Rejects free-form "Other" and decline options.
|
|
27
|
+
if (Array.isArray(options) && options.length > 0) {
|
|
28
|
+
const confirmLabel = options[0]?.label;
|
|
29
|
+
return typeof confirmLabel === "string" && value === confirmLabel;
|
|
30
|
+
}
|
|
31
|
+
// Fail-closed: no options means we cannot structurally validate the answer.
|
|
32
|
+
// Returning false prevents any free-form string from unlocking the gate.
|
|
33
|
+
return false;
|
|
34
|
+
}
|
|
35
|
+
export function hasSelectedValue(selected) {
|
|
36
|
+
if (Array.isArray(selected)) {
|
|
37
|
+
return selected.some((value) => typeof value === "string" && value.length > 0);
|
|
38
|
+
}
|
|
39
|
+
return typeof selected === "string" && selected.length > 0;
|
|
40
|
+
}
|
|
41
|
+
export function hasNotesValue(notes) {
|
|
42
|
+
return typeof notes === "string" && notes.trim().length > 0;
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* THE per-question verdict for gate questions (fail-closed):
|
|
46
|
+
*
|
|
47
|
+
* - cancelled rounds → "cancelled".
|
|
48
|
+
* - the confirmation option (structural match) → "verified".
|
|
49
|
+
* - any other real selection → "declined".
|
|
50
|
+
* - empty/missing selection → "waiting" — an empty answer is NEVER an answer,
|
|
51
|
+
* so notes can never satisfy a gate either.
|
|
52
|
+
*/
|
|
53
|
+
export function evaluateGateAnswer(question, details) {
|
|
54
|
+
if (details.cancelled)
|
|
55
|
+
return "cancelled";
|
|
56
|
+
const questionId = typeof question.id === "string" ? question.id : "";
|
|
57
|
+
const answer = details.response?.answers?.[questionId];
|
|
58
|
+
if (isDepthConfirmationAnswer(answer?.selected, question.options))
|
|
59
|
+
return "verified";
|
|
60
|
+
if (hasSelectedValue(answer?.selected))
|
|
61
|
+
return "declined";
|
|
62
|
+
return "waiting";
|
|
63
|
+
}
|
|
@@ -186,6 +186,7 @@ export function writeLock(basePath, unitType, unitId, sessionFile) {
|
|
|
186
186
|
* stale session-file pointer.
|
|
187
187
|
*/
|
|
188
188
|
export function clearLock(basePath) {
|
|
189
|
+
const legacyLock = readLegacyLock(basePath);
|
|
189
190
|
clearLegacyLockFile(basePath);
|
|
190
191
|
if (!isDbAvailable())
|
|
191
192
|
return;
|
|
@@ -198,9 +199,13 @@ export function clearLock(basePath) {
|
|
|
198
199
|
deleteRuntimeKv("worker", staleWorker.worker_id, SESSION_FILE_KV_KEY);
|
|
199
200
|
return;
|
|
200
201
|
}
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
202
|
+
if (legacyLock?.pid) {
|
|
203
|
+
markWorkerStoppingByPid(projectRoot, legacyLock.pid);
|
|
204
|
+
const workerByLegacyPid = getAllAutoWorkers().find((w) => w.pid === legacyLock.pid
|
|
205
|
+
&& normalizeRealPath(w.project_root_realpath) === projectRoot);
|
|
206
|
+
if (workerByLegacyPid)
|
|
207
|
+
forceReleaseLeasesForWorker(workerByLegacyPid.worker_id);
|
|
208
|
+
}
|
|
204
209
|
const worker = findActiveWorkerForCurrentProcess(projectRoot);
|
|
205
210
|
if (worker)
|
|
206
211
|
deleteRuntimeKv("worker", worker.worker_id, SESSION_FILE_KV_KEY);
|
|
@@ -1,11 +1,13 @@
|
|
|
1
1
|
// Project/App: gsd-pi
|
|
2
2
|
// File Purpose: GSD engine — connection ownership, lifecycle, schema/migrations,
|
|
3
3
|
// and transaction primitives for the single-writer layer. The shared handle
|
|
4
|
-
// (currentDb) lives here;
|
|
5
|
-
// (db/queries.ts) read
|
|
4
|
+
// (currentDb) lives here; domain writers, allowlisted coordination/runtime
|
|
5
|
+
// writers, schema/migration helpers, and the Query Module (db/queries.ts) read
|
|
6
|
+
// it through getDb()/getDbOrNull().
|
|
6
7
|
//
|
|
7
8
|
// This file legitimately holds DDL and BEGIN/COMMIT control, so it is
|
|
8
|
-
// allowlisted in tests/single-writer-invariant.test.ts alongside
|
|
9
|
+
// allowlisted in tests/single-writer-invariant.test.ts alongside the explicit
|
|
10
|
+
// writer layer.
|
|
9
11
|
import { createRequire } from "node:module";
|
|
10
12
|
import { existsSync, copyFileSync, mkdirSync, realpathSync } from "node:fs";
|
|
11
13
|
import { dirname, join } from "node:path";
|
|
@@ -15,7 +17,7 @@ import { createDbAdapter } from "../db-adapter.js";
|
|
|
15
17
|
import { createBaseSchemaObjects } from "../db-base-schema.js";
|
|
16
18
|
import { createCoordinationTablesV24 } from "../db-coordination-schema.js";
|
|
17
19
|
import { createDbConnectionCache } from "../db-connection-cache.js";
|
|
18
|
-
import { backupDatabaseBeforeMigration } from "../db-migration-backup.js";
|
|
20
|
+
import { backupDatabaseBeforeMigration, isMigrationBackupError } from "../db-migration-backup.js";
|
|
19
21
|
import { applyMigrationV2Artifacts, applyMigrationV3Memories, applyMigrationV4DecisionMadeBy, applyMigrationV5HierarchyTables, applyMigrationV6SliceSummaries, applyMigrationV7Dependencies, applyMigrationV8PlanningFields, applyMigrationV9Ordering, applyMigrationV10ReplanTrigger, applyMigrationV11TaskPlanning, applyMigrationV12QualityGates, applyMigrationV13HotPathIndexes, applyMigrationV14SliceDependencies, applyMigrationV15AuditTables, applyMigrationV16EscalationSource, applyMigrationV17TaskEscalation, applyMigrationV18MemorySources, applyMigrationV19MemoryFts, applyMigrationV20MemoryRelations, applyMigrationV21StructuredMemories, applyMigrationV22QualityGateRepair, applyMigrationV23MilestoneQueue, applyMigrationV26MilestoneCommitAttributions, applyMigrationV27ArtifactHash, applyMigrationV28MemoryLastHitAt, applyMigrationV29RepositoryTargets, } from "../db-migration-steps.js";
|
|
20
22
|
import { isMemoriesFtsAvailableSchema, tryCreateMemoriesFtsSchema } from "../db-memory-fts-schema.js";
|
|
21
23
|
import { createDbOpenState } from "../db-open-state.js";
|
|
@@ -526,8 +528,9 @@ export function openDatabase(path) {
|
|
|
526
528
|
}
|
|
527
529
|
catch (err) {
|
|
528
530
|
// Corrupt freelist: DDL fails with "malformed" but VACUUM can rebuild.
|
|
529
|
-
//
|
|
530
|
-
|
|
531
|
+
// Pre-migration backup failures are already pre-DDL and must propagate
|
|
532
|
+
// instead of being masked by VACUUM recovery (see #2519).
|
|
533
|
+
if (shouldAttemptVacuumRecovery(fileBacked, err)) {
|
|
531
534
|
try {
|
|
532
535
|
adapter.exec("VACUUM");
|
|
533
536
|
initSchema(adapter, fileBacked, path);
|
|
@@ -572,6 +575,10 @@ export function openDatabase(path) {
|
|
|
572
575
|
}
|
|
573
576
|
return true;
|
|
574
577
|
}
|
|
578
|
+
function shouldAttemptVacuumRecovery(fileBacked, err) {
|
|
579
|
+
return fileBacked && err instanceof Error && err.message.includes("malformed") && !isMigrationBackupError(err);
|
|
580
|
+
}
|
|
581
|
+
export const _shouldAttemptVacuumRecoveryForTest = shouldAttemptVacuumRecovery;
|
|
575
582
|
export function closeDatabase() {
|
|
576
583
|
if (currentDb) {
|
|
577
584
|
try {
|
|
@@ -687,6 +694,7 @@ function createTransactionControls(db) {
|
|
|
687
694
|
return {
|
|
688
695
|
begin: () => db.exec("BEGIN"),
|
|
689
696
|
beginRead: () => db.exec("BEGIN DEFERRED"),
|
|
697
|
+
beginImmediate: () => db.exec("BEGIN IMMEDIATE"),
|
|
690
698
|
commit: () => db.exec("COMMIT"),
|
|
691
699
|
rollback: () => db.exec("ROLLBACK"),
|
|
692
700
|
};
|
|
@@ -705,6 +713,16 @@ export function transaction(fn) {
|
|
|
705
713
|
throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
|
|
706
714
|
return _transactionRunner.transaction(createTransactionControls(currentDb), fn);
|
|
707
715
|
}
|
|
716
|
+
/**
|
|
717
|
+
* Run a BEGIN IMMEDIATE write transaction for operations that need SQLite's
|
|
718
|
+
* reserved writer lock before issuing updates. Re-entrant like transaction():
|
|
719
|
+
* nested calls run inside the outer transaction without a nested BEGIN.
|
|
720
|
+
*/
|
|
721
|
+
export function immediateTransaction(fn) {
|
|
722
|
+
if (!currentDb)
|
|
723
|
+
throw new GSDError(GSD_STALE_STATE, "gsd-db: No database open");
|
|
724
|
+
return _transactionRunner.immediateTransaction(createTransactionControls(currentDb), fn);
|
|
725
|
+
}
|
|
708
726
|
/**
|
|
709
727
|
* Wrap a block of reads in a DEFERRED transaction so that all SELECTs observe
|
|
710
728
|
* a consistent snapshot of the DB even if a concurrent writer commits between
|