@opengsd/gsd-pi 1.2.0-dev.5457a158 → 1.2.0-dev.822c9439
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/headless-events.js +7 -5
- package/dist/mcp-server.js +2 -1
- package/dist/resource-loader.d.ts +9 -5
- package/dist/resource-loader.js +114 -6
- 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/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/stream-adapter.js +3 -2
- 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 +89 -54
- package/dist/resources/extensions/gsd/auto/phases.js +49 -6
- package/dist/resources/extensions/gsd/auto/session.js +3 -0
- package/dist/resources/extensions/gsd/auto-direct-dispatch.js +11 -34
- package/dist/resources/extensions/gsd/auto-dispatch.js +50 -58
- package/dist/resources/extensions/gsd/auto-model-selection.js +36 -13
- package/dist/resources/extensions/gsd/auto-post-unit.js +30 -12
- package/dist/resources/extensions/gsd/auto-prompts.js +78 -19
- package/dist/resources/extensions/gsd/auto-start.js +12 -12
- 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 +14 -1
- package/dist/resources/extensions/gsd/auto.js +37 -1
- 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 +23 -6
- 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 +172 -59
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +302 -80
- 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/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/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 +8 -3
- package/dist/resources/extensions/gsd/db/queries.js +26 -0
- 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-git-checks.js +2 -18
- package/dist/resources/extensions/gsd/engine-hook-contract.js +70 -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/gsd-command-home.js +22 -12
- package/dist/resources/extensions/gsd/gsd-db.js +2 -1
- package/dist/resources/extensions/gsd/guidance.js +60 -0
- package/dist/resources/extensions/gsd/guided-flow.js +6 -3
- 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/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/parsers-legacy.js +16 -4
- package/dist/resources/extensions/gsd/preferences-models.js +2 -2
- 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 +1 -1
- 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 +7 -5
- package/dist/resources/extensions/gsd/prompts/system.md +5 -2
- 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/reactive-graph.js +8 -1
- package/dist/resources/extensions/gsd/roadmap-slices.js +25 -3
- package/dist/resources/extensions/gsd/safety/destructive-confirmation.js +108 -0
- package/dist/resources/extensions/gsd/session-lock.js +1 -1
- package/dist/resources/extensions/gsd/state.js +5 -0
- 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/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 +3 -2
- package/dist/resources/extensions/gsd/tools/exec-tool.js +5 -0
- 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 +42 -16
- package/dist/resources/extensions/gsd/unit-context-composer.js +65 -0
- package/dist/resources/extensions/gsd/unit-registry.js +7 -20
- 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/web-app-uat.js +45 -8
- 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-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.js +8 -1
- 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/SKILL.md +3 -0
- package/dist/resources/skills/create-skill/references/executable-code.md +1 -1
- package/dist/resources/skills/create-skill/references/skill-structure.md +1 -0
- 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/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 +11 -11
- 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/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/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 +11 -11
- 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/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/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 +5 -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 +10 -5
- 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 +18 -18
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +99 -38
- 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/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 +239 -153
- package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
- package/packages/pi-ai/dist/models.generated.js +256 -145
- 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/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/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/stream-adapter.ts +3 -2
- 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.ts +4 -1
- package/src/resources/extensions/gsd/auto/orchestrator.ts +98 -56
- package/src/resources/extensions/gsd/auto/phases.ts +65 -26
- package/src/resources/extensions/gsd/auto/session.ts +3 -0
- package/src/resources/extensions/gsd/auto-direct-dispatch.ts +18 -48
- package/src/resources/extensions/gsd/auto-dispatch.ts +48 -61
- package/src/resources/extensions/gsd/auto-model-selection.ts +41 -12
- package/src/resources/extensions/gsd/auto-post-unit.ts +33 -12
- package/src/resources/extensions/gsd/auto-prompts.ts +115 -35
- package/src/resources/extensions/gsd/auto-start.ts +12 -14
- 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 +14 -1
- package/src/resources/extensions/gsd/auto.ts +44 -1
- 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 +23 -6
- 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 +211 -59
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +350 -86
- 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/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/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 +10 -2
- package/src/resources/extensions/gsd/db/queries.ts +37 -0
- 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-git-checks.ts +2 -19
- package/src/resources/extensions/gsd/engine-hook-contract.ts +79 -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/gsd-command-home.ts +13 -3
- package/src/resources/extensions/gsd/gsd-db.ts +4 -3
- package/src/resources/extensions/gsd/guidance.ts +78 -0
- package/src/resources/extensions/gsd/guided-flow.ts +21 -26
- 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/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/parsers-legacy.ts +16 -4
- package/src/resources/extensions/gsd/preferences-models.ts +2 -1
- 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 +1 -1
- 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 +7 -5
- package/src/resources/extensions/gsd/prompts/system.md +5 -2
- 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/reactive-graph.ts +11 -1
- package/src/resources/extensions/gsd/roadmap-slices.ts +28 -3
- package/src/resources/extensions/gsd/safety/destructive-confirmation.ts +134 -0
- package/src/resources/extensions/gsd/session-lock.ts +1 -1
- package/src/resources/extensions/gsd/state.ts +5 -0
- package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +97 -1
- package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +198 -26
- package/src/resources/extensions/gsd/tests/auto-remote-session-lock-cleanup.test.ts +65 -3
- 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/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/discuss-routing-fixes.test.ts +12 -2
- 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/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/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 +23 -0
- package/src/resources/extensions/gsd/tests/guided-dispatch-root.test.ts +2 -6
- package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +7 -11
- package/src/resources/extensions/gsd/tests/integration/auto-worktree.test.ts +20 -58
- 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/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/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/phases-terminal-complete-idempotent.test.ts +242 -0
- package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +63 -2
- package/src/resources/extensions/gsd/tests/prompt-db.test.ts +124 -6
- package/src/resources/extensions/gsd/tests/roadmap-slices.test.ts +68 -0
- package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +19 -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 +76 -0
- 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 +112 -29
- package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +44 -0
- package/src/resources/extensions/gsd/tests/uok-audit-unified.test.ts +8 -0
- package/src/resources/extensions/gsd/tests/verification-verdict.test.ts +2 -0
- package/src/resources/extensions/gsd/tests/web-app-uat.test.ts +44 -1
- package/src/resources/extensions/gsd/tests/workflow-events.test.ts +19 -0
- package/src/resources/extensions/gsd/tests/workflow-reconcile.test.ts +20 -0
- package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +273 -38
- 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/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 +3 -2
- package/src/resources/extensions/gsd/tools/exec-tool.ts +4 -0
- 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 +62 -16
- package/src/resources/extensions/gsd/unit-context-composer.ts +99 -0
- package/src/resources/extensions/gsd/unit-registry.ts +7 -20
- 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/web-app-uat.ts +51 -8
- 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-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.ts +7 -1
- 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/SKILL.md +3 -0
- package/src/resources/skills/create-skill/references/executable-code.md +1 -1
- package/src/resources/skills/create-skill/references/skill-structure.md +1 -0
- 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/spike-wrap-up/SKILL.md +9 -9
- 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/{2p9Rv9pQflAxCBbGVI2vb → yWwBo-w09Y_W-nmeeWFRp}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{2p9Rv9pQflAxCBbGVI2vb → yWwBo-w09Y_W-nmeeWFRp}/_ssgManifest.js +0 -0
|
@@ -1,22 +1,107 @@
|
|
|
1
1
|
// Project/App: gsd-pi
|
|
2
2
|
// File Purpose: Shared closeout detection and merge actions for /gsd home and smart entry.
|
|
3
|
+
import { existsSync, readdirSync, statSync } from "node:fs";
|
|
4
|
+
import { join } from "node:path";
|
|
3
5
|
import { setAutoOutcomeWidget } from "./auto-dashboard.js";
|
|
4
6
|
import { invalidateAllCaches } from "./cache.js";
|
|
7
|
+
import { isDbAvailable } from "./db/engine.js";
|
|
8
|
+
import { getMilestone } from "./db/queries.js";
|
|
9
|
+
import { MILESTONE_ID_RE } from "./milestone-ids.js";
|
|
5
10
|
import { mergeCompletedMilestone } from "./parallel-merge.js";
|
|
6
11
|
import { cleanupQuickBranch, detectStrandedQuickBranch } from "./quick.js";
|
|
12
|
+
import { isClosedStatus } from "./status-guards.js";
|
|
7
13
|
import { findUnmergedCompletedMilestones, } from "./unmerged-milestone-guard.js";
|
|
8
14
|
import { appendRequirementsBacklogToSummary } from "./requirements-backlog.js";
|
|
15
|
+
import { nativeBranchList, nativeIsRepo } from "./native-git-bridge.js";
|
|
16
|
+
import { allWorktreesDirs } from "./worktree-manager.js";
|
|
9
17
|
const MILESTONE_MERGE_CLOSEOUT_COMMANDS = [
|
|
10
18
|
"/gsd status for overview",
|
|
11
19
|
"/gsd visualize to inspect",
|
|
12
20
|
"/gsd notifications for history",
|
|
13
21
|
"/gsd start for new work",
|
|
14
22
|
];
|
|
23
|
+
function listMilestoneWorktreeIds(basePath) {
|
|
24
|
+
const ids = new Set();
|
|
25
|
+
for (const wtDir of allWorktreesDirs(basePath)) {
|
|
26
|
+
if (!existsSync(wtDir))
|
|
27
|
+
continue;
|
|
28
|
+
for (const entry of readdirSync(wtDir)) {
|
|
29
|
+
if (!MILESTONE_ID_RE.test(entry))
|
|
30
|
+
continue;
|
|
31
|
+
try {
|
|
32
|
+
if (statSync(join(wtDir, entry)).isDirectory())
|
|
33
|
+
ids.add(entry);
|
|
34
|
+
}
|
|
35
|
+
catch {
|
|
36
|
+
// skip unreadable entries
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
return [...ids].sort();
|
|
41
|
+
}
|
|
42
|
+
function listMilestoneBranchIds(basePath) {
|
|
43
|
+
try {
|
|
44
|
+
return nativeBranchList(basePath, "milestone/*")
|
|
45
|
+
.map((branch) => branch.replace(/^milestone\//, ""))
|
|
46
|
+
.filter((id) => MILESTONE_ID_RE.test(id))
|
|
47
|
+
.sort();
|
|
48
|
+
}
|
|
49
|
+
catch {
|
|
50
|
+
return [];
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* A milestone ID is "stranded residue" only when its worktree/branch artifacts
|
|
55
|
+
* exist for a milestone the DB does not consider currently in flight — i.e. the
|
|
56
|
+
* row is closed (complete/done/skipped/closed) or absent. Active, pending,
|
|
57
|
+
* blocked, parked, queued, and deferred rows describe normal in-flight or
|
|
58
|
+
* intentionally-preserved state, never residue. Returning `false` skips the ID;
|
|
59
|
+
* returning `true` keeps it in the hint.
|
|
60
|
+
*/
|
|
61
|
+
function isStrandedMilestoneId(milestoneId) {
|
|
62
|
+
if (!isDbAvailable())
|
|
63
|
+
return true;
|
|
64
|
+
const row = getMilestone(milestoneId);
|
|
65
|
+
if (!row)
|
|
66
|
+
return true;
|
|
67
|
+
return isClosedStatus(row.status);
|
|
68
|
+
}
|
|
69
|
+
/** Surface stranded milestone git residue when closeout guards did not classify it. */
|
|
70
|
+
export function detectIdleMilestoneResidueHint(basePath) {
|
|
71
|
+
if (!nativeIsRepo(basePath))
|
|
72
|
+
return null;
|
|
73
|
+
const gsdDir = join(basePath, ".gsd");
|
|
74
|
+
const dbPath = join(gsdDir, "gsd.db");
|
|
75
|
+
if (!existsSync(gsdDir) || !existsSync(dbPath)) {
|
|
76
|
+
return {
|
|
77
|
+
milestoneIds: [],
|
|
78
|
+
message: "This git repo has no local GSD workflow database (.gsd/gsd.db). " +
|
|
79
|
+
"Workflow state may live in an external worktree, or run /gsd new-project to initialize here.",
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
const worktreeIds = listMilestoneWorktreeIds(basePath);
|
|
83
|
+
const branchIds = listMilestoneBranchIds(basePath);
|
|
84
|
+
const candidateIds = [...new Set([...worktreeIds, ...branchIds])].sort();
|
|
85
|
+
const milestoneIds = candidateIds.filter(isStrandedMilestoneId);
|
|
86
|
+
if (milestoneIds.length === 0)
|
|
87
|
+
return null;
|
|
88
|
+
const listed = milestoneIds.join(", ");
|
|
89
|
+
const recovery = milestoneIds.length === 1
|
|
90
|
+
? `/gsd dispatch complete-milestone ${milestoneIds[0]}`
|
|
91
|
+
: "/gsd doctor --fix";
|
|
92
|
+
return {
|
|
93
|
+
milestoneIds,
|
|
94
|
+
message: `Stranded milestone git residue detected (${listed}: worktree dir and/or milestone/* branch). ` +
|
|
95
|
+
`Run ${recovery} or /gsd status to recover closeout before starting new work.`,
|
|
96
|
+
};
|
|
97
|
+
}
|
|
15
98
|
export async function loadCloseoutContext(basePath) {
|
|
16
99
|
const unmergedMilestones = await findUnmergedCompletedMilestones(basePath);
|
|
100
|
+
const idleResidueHint = unmergedMilestones.length === 0 ? detectIdleMilestoneResidueHint(basePath) : null;
|
|
17
101
|
return {
|
|
18
102
|
strandedQuick: detectStrandedQuickBranch(basePath),
|
|
19
103
|
unmergedMilestones,
|
|
104
|
+
idleResidueHint,
|
|
20
105
|
};
|
|
21
106
|
}
|
|
22
107
|
export function getPrimaryCloseoutRecommendation(closeout) {
|
|
@@ -62,6 +147,13 @@ export function buildIdleMenuSummary(state, closeout) {
|
|
|
62
147
|
`${blocker.milestoneId} is complete but not merged into ${blocker.integrationBranch}.`,
|
|
63
148
|
];
|
|
64
149
|
}
|
|
150
|
+
// Surface idle residue before the completion summary so smart entry shows
|
|
151
|
+
// the same recovery text /gsd home would: a closed/unknown milestone with
|
|
152
|
+
// lingering worktree/branch artifacts must not be hidden behind the
|
|
153
|
+
// "all milestones complete" message.
|
|
154
|
+
if (closeout.idleResidueHint) {
|
|
155
|
+
return [closeout.idleResidueHint.message];
|
|
156
|
+
}
|
|
65
157
|
if (state.phase === "complete") {
|
|
66
158
|
const last = state.lastCompletedMilestone;
|
|
67
159
|
return appendRequirementsBacklogToSummary(state, [
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { checkRemoteAutoSession, isAutoActive, isAutoPaused, stopAutoRemote } from "../auto.js";
|
|
1
|
+
import { checkRemoteAutoSession, forceStopAutoRemote, isAutoActive, isAutoPaused, stopAutoRemote } from "../auto.js";
|
|
2
2
|
import { validateDirectory } from "../validate-directory.js";
|
|
3
3
|
import { resolveProjectRoot } from "../worktree.js";
|
|
4
4
|
import { showNextAction } from "../../shared/tui.js";
|
|
@@ -135,5 +135,19 @@ export async function guardRemoteSession(ctx, pi) {
|
|
|
135
135
|
}
|
|
136
136
|
return false;
|
|
137
137
|
}
|
|
138
|
-
|
|
138
|
+
if (choice === "force") {
|
|
139
|
+
const result = forceStopAutoRemote(projectRoot());
|
|
140
|
+
if (result.error) {
|
|
141
|
+
ctx.ui.notify(`Failed to force-stop remote auto-mode: ${result.error}`, "error");
|
|
142
|
+
return false;
|
|
143
|
+
}
|
|
144
|
+
if (result.found) {
|
|
145
|
+
ctx.ui.notify(`Force-stopped auto-mode session (PID ${result.pid}). Starting a new session.`, "warning");
|
|
146
|
+
}
|
|
147
|
+
else {
|
|
148
|
+
ctx.ui.notify("Remote session is no longer running. Starting a new session.", "info");
|
|
149
|
+
}
|
|
150
|
+
return true;
|
|
151
|
+
}
|
|
152
|
+
return false;
|
|
139
153
|
}
|
|
@@ -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);
|
|
@@ -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
|
+
}
|
|
@@ -6,8 +6,6 @@
|
|
|
6
6
|
// ─── Timeouts ─────────────────────────────────────────────────────────────────
|
|
7
7
|
/** Default timeout for verification-gate commands (ms). */
|
|
8
8
|
export const DEFAULT_COMMAND_TIMEOUT_MS = 120_000;
|
|
9
|
-
/** Default timeout for the dynamic bash tool (seconds). */
|
|
10
|
-
export const DEFAULT_BASH_TIMEOUT_SECS = 120;
|
|
11
9
|
// ─── Cache Sizes ──────────────────────────────────────────────────────────────
|
|
12
10
|
/** Max directory-listing cache entries before eviction (#611). */
|
|
13
11
|
export const DIR_CACHE_MAX = 200;
|