@opengsd/gsd-pi 1.2.0-dev.b1abb545 → 1.2.0-dev.d6c5343c
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli-style.d.ts +17 -0
- package/dist/cli-style.js +28 -0
- package/dist/cli.js +1 -1
- package/dist/headless-events.d.ts +4 -2
- package/dist/headless-events.js +14 -34
- package/dist/models-resolver.d.ts +3 -13
- package/dist/models-resolver.js +3 -22
- package/dist/resource-loader.js +2 -14
- package/dist/resources/.managed-resources-content-hash +1 -1
- package/dist/resources/GSD-WORKFLOW.md +5 -4
- package/dist/resources/extensions/async-jobs/async-bash-tool.js +30 -64
- package/dist/resources/extensions/async-jobs/await-tool.js +80 -12
- package/dist/resources/extensions/async-jobs/index.js +65 -0
- package/dist/resources/extensions/async-jobs/job-manager.js +12 -1
- package/dist/resources/extensions/bg-shell/bg-shell-command.js +6 -6
- package/dist/resources/extensions/bg-shell/bg-shell-tool.js +10 -7
- package/dist/resources/extensions/bg-shell/overlay.js +9 -6
- package/dist/resources/extensions/bg-shell/process-manager.js +54 -25
- package/dist/resources/extensions/bg-shell/readiness-detector.js +11 -0
- package/dist/resources/extensions/bg-shell/utilities.js +5 -2
- package/dist/resources/extensions/browser-tools/engine/managed-gsd-browser.js +209 -88
- package/dist/resources/extensions/browser-tools/engine/selection.js +73 -5
- package/dist/resources/extensions/browser-tools/index.js +69 -12
- package/dist/resources/extensions/claude-code-cli/models.js +9 -0
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +38 -6
- package/dist/resources/extensions/gsd/auto/custom-verify-retry-store.js +17 -2
- package/dist/resources/extensions/gsd/auto/detect-stuck.js +33 -13
- package/dist/resources/extensions/gsd/auto/dispatch-history.js +105 -0
- package/dist/resources/extensions/gsd/auto/dispatch-key.js +37 -0
- package/dist/resources/extensions/gsd/auto/loop.js +4 -1
- package/dist/resources/extensions/gsd/auto/orchestrator.js +94 -48
- package/dist/resources/extensions/gsd/auto/phases.js +8 -3
- package/dist/resources/extensions/gsd/auto-direct-dispatch.js +8 -32
- package/dist/resources/extensions/gsd/auto-dispatch.js +40 -57
- package/dist/resources/extensions/gsd/auto-model-selection.js +25 -6
- package/dist/resources/extensions/gsd/auto-post-unit.js +31 -14
- package/dist/resources/extensions/gsd/auto-prompts.js +81 -19
- package/dist/resources/extensions/gsd/auto-start.js +24 -26
- package/dist/resources/extensions/gsd/auto-tool-tracking.js +18 -0
- package/dist/resources/extensions/gsd/auto-unit-tool-scope.js +12 -20
- package/dist/resources/extensions/gsd/auto-verification.js +9 -28
- package/dist/resources/extensions/gsd/auto-worktree-repair.js +10 -2
- package/dist/resources/extensions/gsd/auto-worktree.js +35 -352
- package/dist/resources/extensions/gsd/auto.js +8 -20
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +3 -2
- package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +32 -12
- package/dist/resources/extensions/gsd/bootstrap/register-extension.js +19 -0
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +229 -36
- package/dist/resources/extensions/gsd/bootstrap/write-gate.js +319 -71
- package/dist/resources/extensions/gsd/branch-patterns.js +2 -0
- package/dist/resources/extensions/gsd/browser-daemon-auto-prep.js +83 -0
- package/dist/resources/extensions/gsd/browser-evidence.js +8 -2
- package/dist/resources/extensions/gsd/captures.js +5 -15
- package/dist/resources/extensions/gsd/closeout-recovery.js +3 -2
- package/dist/resources/extensions/gsd/commands/catalog.js +6 -62
- package/dist/resources/extensions/gsd/consent-question.js +337 -0
- package/dist/resources/extensions/gsd/consent-verdict.js +63 -0
- package/dist/resources/extensions/gsd/constants.js +0 -2
- package/dist/resources/extensions/gsd/crash-recovery.js +4 -12
- package/dist/resources/extensions/gsd/db/engine.js +755 -0
- package/dist/resources/extensions/gsd/db/queries.js +398 -0
- package/dist/resources/extensions/gsd/db/sql-constants.js +11 -0
- package/dist/resources/extensions/gsd/db/writers/cascades.js +194 -0
- package/dist/resources/extensions/gsd/db/writers/import-restore.js +182 -0
- package/dist/resources/extensions/gsd/db/writers/memory.js +149 -0
- package/dist/resources/extensions/gsd/db/writers/reconcile.js +458 -0
- package/dist/resources/extensions/gsd/db/writers/status.js +70 -0
- package/dist/resources/extensions/gsd/dispatch-guard.js +10 -35
- package/dist/resources/extensions/gsd/doctor-environment.js +5 -11
- package/dist/resources/extensions/gsd/doctor-format.js +9 -6
- package/dist/resources/extensions/gsd/doctor-git-checks.js +4 -3
- package/dist/resources/extensions/gsd/doctor-runtime-checks.js +21 -16
- package/dist/resources/extensions/gsd/engine-hook-contract.js +70 -0
- package/dist/resources/extensions/gsd/error-classifier.js +9 -0
- package/dist/resources/extensions/gsd/exec-sandbox.js +30 -10
- package/dist/resources/extensions/gsd/files.js +33 -19
- package/dist/resources/extensions/gsd/git-service.js +1 -0
- package/dist/resources/extensions/gsd/gitignore.js +3 -0
- package/dist/resources/extensions/gsd/gsd-db.js +171 -2048
- package/dist/resources/extensions/gsd/guidance.js +158 -0
- package/dist/resources/extensions/gsd/guided-flow.js +51 -5
- package/dist/resources/extensions/gsd/markdown-renderer.js +10 -0
- package/dist/resources/extensions/gsd/mcp-filter.js +2 -19
- package/dist/resources/extensions/gsd/mcp-tool-name.js +5 -13
- package/dist/resources/extensions/gsd/memory-consolidation-scanner.js +1 -1
- package/dist/resources/extensions/gsd/migrate/safety.js +20 -9
- package/dist/resources/extensions/gsd/migration-auto-check.js +24 -3
- package/dist/resources/extensions/gsd/milestone-closeout.js +13 -23
- package/dist/resources/extensions/gsd/model-cost-table.js +1 -0
- package/dist/resources/extensions/gsd/model-router.js +3 -0
- package/dist/resources/extensions/gsd/notification-store.js +11 -4
- package/dist/resources/extensions/gsd/parallel-merge.js +14 -11
- package/dist/resources/extensions/gsd/parallel-monitor-overlay.js +11 -7
- package/dist/resources/extensions/gsd/parsers-legacy.js +16 -4
- package/dist/resources/extensions/gsd/paths.js +37 -24
- package/dist/resources/extensions/gsd/pre-execution-checks.js +91 -3
- package/dist/resources/extensions/gsd/preferences-models.js +14 -48
- package/dist/resources/extensions/gsd/preferences.js +14 -0
- package/dist/resources/extensions/gsd/prompts/complete-slice.md +2 -2
- package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/refine-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/replan-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/run-uat.md +6 -4
- package/dist/resources/extensions/gsd/prompts/system.md +5 -2
- package/dist/resources/extensions/gsd/provider-error-guidance.js +1 -5
- package/dist/resources/extensions/gsd/provider-switch-observer.js +1 -1
- package/dist/resources/extensions/gsd/publication.js +87 -0
- package/dist/resources/extensions/gsd/reactive-graph.js +8 -1
- package/dist/resources/extensions/gsd/recovery-classification.js +41 -87
- package/dist/resources/extensions/gsd/safety/destructive-confirmation.js +108 -0
- package/dist/resources/extensions/gsd/safety/evidence-collector.js +37 -4
- package/dist/resources/extensions/gsd/safety/evidence-cross-ref.js +7 -2
- package/dist/resources/extensions/gsd/safety/file-change-validator.js +10 -0
- package/dist/resources/extensions/gsd/state-transition-matrix.js +38 -0
- package/dist/resources/extensions/gsd/state.js +6 -20
- package/dist/resources/extensions/gsd/status-guards.js +56 -8
- package/dist/resources/extensions/gsd/stop-notice.js +57 -0
- package/dist/resources/extensions/gsd/tool-presentation-plan.js +4 -4
- package/dist/resources/extensions/gsd/tool-surface-readiness.js +56 -0
- package/dist/resources/extensions/gsd/tools/complete-slice.js +44 -53
- package/dist/resources/extensions/gsd/tools/exec-tool.js +10 -8
- package/dist/resources/extensions/gsd/tools/plan-slice.js +12 -6
- package/dist/resources/extensions/gsd/tools/reopen-milestone.js +11 -29
- package/dist/resources/extensions/gsd/tools/reopen-slice.js +14 -33
- package/dist/resources/extensions/gsd/tools/skip-slice.js +18 -36
- package/dist/resources/extensions/gsd/uat-policy.js +42 -16
- package/dist/resources/extensions/gsd/undo.js +8 -7
- package/dist/resources/extensions/gsd/unit-closeout.js +138 -0
- package/dist/resources/extensions/gsd/unit-context-composer.js +74 -1
- package/dist/resources/extensions/gsd/unit-context-manifest.js +4 -27
- package/dist/resources/extensions/gsd/unit-registry.js +337 -0
- package/dist/resources/extensions/gsd/unit-tool-contracts.js +9 -182
- package/dist/resources/extensions/gsd/verdict-parser.js +1 -1
- package/dist/resources/extensions/gsd/web-app-uat.js +45 -8
- package/dist/resources/extensions/gsd/workflow-tool-surface.js +1 -1
- package/dist/resources/extensions/gsd/worktree-git-recovery.js +293 -0
- package/dist/resources/extensions/gsd/worktree-lifecycle.js +12 -3
- package/dist/resources/extensions/gsd/worktree-manager.js +45 -28
- package/dist/resources/extensions/gsd/worktree-placement.js +59 -0
- package/dist/resources/extensions/gsd/worktree-reentry.js +12 -8
- package/dist/resources/extensions/gsd/worktree-root.js +28 -6
- package/dist/resources/extensions/gsd/worktree-safety.js +8 -5
- package/dist/resources/extensions/gsd/worktree-session-state.js +12 -11
- package/dist/resources/extensions/search-the-web/native-search.js +5 -3
- package/dist/resources/extensions/shared/browser-contract.js +59 -0
- package/dist/resources/extensions/shared/gsd-browser-cli.js +96 -5
- package/dist/resources/shared/package.json +3 -0
- package/dist/resources/skills/create-skill/references/executable-code.md +1 -1
- package/dist/resources/skills/create-skill/workflows/add-reference.md +8 -3
- package/dist/resources/skills/create-skill/workflows/add-script.md +4 -2
- package/dist/resources/skills/create-skill/workflows/add-template.md +3 -1
- package/dist/resources/skills/create-skill/workflows/add-workflow.md +8 -3
- package/dist/resources/skills/create-skill/workflows/upgrade-to-router.md +10 -5
- package/dist/resources/skills/create-skill/workflows/verify-skill.md +9 -4
- package/dist/resources/skills/gsd-browser/SKILL.md +1 -1
- package/dist/resources/skills/spike-wrap-up/SKILL.md +9 -9
- package/dist/tsconfig.extensions.tsbuildinfo +1 -1
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +8 -8
- package/dist/web/standalone/.next/build-manifest.json +3 -3
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/mcp-connections/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/notifications/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/shutdown/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/update/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +8 -8
- package/dist/web/standalone/.next/server/chunks/5124.js +1 -1
- package/dist/web/standalone/.next/server/chunks/{5047.js → 5942.js} +2 -2
- package/dist/web/standalone/.next/server/chunks/8357.js +1 -1
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +1 -1
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/dist/web/standalone/.next/static/chunks/{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/package.json +1 -1
- package/dist/worktree-cli.js +3 -6
- package/dist/worktree-status-banner.js +7 -11
- package/package.json +1 -1
- package/packages/cloud-mcp-gateway/package.json +2 -2
- package/packages/contracts/dist/rpc.d.ts +1 -0
- package/packages/contracts/dist/rpc.d.ts.map +1 -1
- package/packages/contracts/dist/rpc.js.map +1 -1
- package/packages/contracts/dist/workflow.d.ts +4 -0
- package/packages/contracts/dist/workflow.d.ts.map +1 -1
- package/packages/contracts/dist/workflow.js.map +1 -1
- package/packages/contracts/package.json +1 -1
- package/packages/daemon/package.json +4 -4
- package/packages/gsd-agent-core/package.json +5 -5
- package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts +5 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js +8 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js +7 -0
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.js +8 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.js +11 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.js +4 -4
- package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.js.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
- package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js +3 -1
- package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js.map +1 -1
- package/packages/gsd-agent-modes/package.json +7 -7
- package/packages/mcp-server/dist/cli.js +6 -3
- package/packages/mcp-server/dist/cli.js.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts +8 -0
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +46 -21
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/package.json +3 -3
- package/packages/native/package.json +1 -1
- package/packages/pi-agent-core/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/models.generated.d.ts +192 -0
- package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
- package/packages/pi-ai/dist/models.generated.js +166 -0
- package/packages/pi-ai/dist/models.generated.js.map +1 -1
- package/packages/pi-ai/package.json +3 -2
- package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +2 -2
- package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.js +19 -13
- package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/capability-patches.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/capability-patches.js +3 -1
- package/packages/pi-coding-agent/dist/core/capability-patches.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/provider-readiness.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/provider-readiness.js +13 -6
- package/packages/pi-coding-agent/dist/core/provider-readiness.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/tools/bash.d.ts +11 -0
- package/packages/pi-coding-agent/dist/core/tools/bash.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/tools/bash.js +53 -11
- package/packages/pi-coding-agent/dist/core/tools/bash.js.map +1 -1
- package/packages/pi-coding-agent/dist/index.d.ts +1 -1
- package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/index.js +1 -1
- package/packages/pi-coding-agent/dist/index.js.map +1 -1
- package/packages/pi-coding-agent/dist/utils/shell.d.ts +28 -2
- package/packages/pi-coding-agent/dist/utils/shell.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/utils/shell.js +56 -10
- package/packages/pi-coding-agent/dist/utils/shell.js.map +1 -1
- package/packages/pi-coding-agent/package.json +7 -7
- package/packages/pi-tui/dist/tui.d.ts.map +1 -1
- package/packages/pi-tui/dist/tui.js +9 -0
- package/packages/pi-tui/dist/tui.js.map +1 -1
- package/packages/pi-tui/package.json +2 -2
- package/packages/rpc-client/package.json +2 -2
- package/pkg/package.json +1 -1
- package/src/resources/GSD-WORKFLOW.md +5 -4
- package/src/resources/extensions/async-jobs/async-bash-cancel.test.ts +360 -0
- package/src/resources/extensions/async-jobs/async-bash-tool.ts +33 -56
- package/src/resources/extensions/async-jobs/await-tool.test.ts +139 -0
- package/src/resources/extensions/async-jobs/await-tool.ts +82 -12
- package/src/resources/extensions/async-jobs/index.ts +79 -0
- package/src/resources/extensions/async-jobs/job-manager.ts +21 -1
- package/src/resources/extensions/bg-shell/bg-shell-command.ts +6 -6
- package/src/resources/extensions/bg-shell/bg-shell-tool.ts +10 -6
- package/src/resources/extensions/bg-shell/overlay.ts +9 -5
- package/src/resources/extensions/bg-shell/process-manager.ts +50 -25
- package/src/resources/extensions/bg-shell/readiness-detector.ts +12 -0
- package/src/resources/extensions/bg-shell/tests/lifecycle-and-utilities.test.ts +48 -1
- package/src/resources/extensions/bg-shell/utilities.ts +5 -2
- package/src/resources/extensions/browser-tools/engine/managed-gsd-browser.ts +265 -98
- package/src/resources/extensions/browser-tools/engine/selection.ts +90 -4
- package/src/resources/extensions/browser-tools/index.ts +71 -13
- package/src/resources/extensions/browser-tools/tests/browser-engine-selection.test.mjs +83 -13
- package/src/resources/extensions/browser-tools/tests/gsd-browser-launch-config.test.mjs +29 -1
- package/src/resources/extensions/browser-tools/tests/managed-gsd-browser-tools.test.mjs +136 -0
- package/src/resources/extensions/claude-code-cli/models.ts +9 -0
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +40 -4
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +28 -0
- package/src/resources/extensions/gsd/auto/custom-verify-retry-store.ts +21 -3
- package/src/resources/extensions/gsd/auto/detect-stuck.ts +32 -9
- package/src/resources/extensions/gsd/auto/dispatch-history.ts +152 -0
- package/src/resources/extensions/gsd/auto/dispatch-key.ts +39 -0
- package/src/resources/extensions/gsd/auto/loop-deps.ts +1 -1
- package/src/resources/extensions/gsd/auto/loop.ts +4 -1
- package/src/resources/extensions/gsd/auto/orchestrator.ts +109 -51
- package/src/resources/extensions/gsd/auto/phases.ts +12 -3
- package/src/resources/extensions/gsd/auto-direct-dispatch.ts +8 -32
- package/src/resources/extensions/gsd/auto-dispatch.ts +38 -52
- package/src/resources/extensions/gsd/auto-model-selection.ts +25 -5
- package/src/resources/extensions/gsd/auto-post-unit.ts +37 -13
- package/src/resources/extensions/gsd/auto-prompts.ts +118 -35
- package/src/resources/extensions/gsd/auto-start.ts +24 -29
- package/src/resources/extensions/gsd/auto-tool-tracking.ts +19 -0
- package/src/resources/extensions/gsd/auto-unit-tool-scope.ts +14 -21
- package/src/resources/extensions/gsd/auto-verification.ts +8 -26
- package/src/resources/extensions/gsd/auto-worktree-repair.ts +13 -2
- package/src/resources/extensions/gsd/auto-worktree.ts +41 -364
- package/src/resources/extensions/gsd/auto.ts +20 -24
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +3 -5
- package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +33 -12
- package/src/resources/extensions/gsd/bootstrap/register-extension.ts +24 -0
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +270 -37
- package/src/resources/extensions/gsd/bootstrap/write-gate.ts +368 -78
- package/src/resources/extensions/gsd/branch-patterns.ts +3 -0
- package/src/resources/extensions/gsd/browser-daemon-auto-prep.ts +108 -0
- package/src/resources/extensions/gsd/browser-evidence.ts +18 -2
- package/src/resources/extensions/gsd/captures.ts +5 -16
- package/src/resources/extensions/gsd/closeout-recovery.ts +2 -1
- package/src/resources/extensions/gsd/commands/catalog.ts +6 -68
- package/src/resources/extensions/gsd/consent-question.ts +416 -0
- package/src/resources/extensions/gsd/consent-verdict.ts +86 -0
- package/src/resources/extensions/gsd/constants.ts +0 -3
- package/src/resources/extensions/gsd/crash-recovery.ts +3 -9
- package/src/resources/extensions/gsd/db/engine.ts +809 -0
- package/src/resources/extensions/gsd/db/queries.ts +490 -0
- package/src/resources/extensions/gsd/db/sql-constants.ts +12 -0
- package/src/resources/extensions/gsd/db/writers/cascades.ts +237 -0
- package/src/resources/extensions/gsd/db/writers/import-restore.ts +310 -0
- package/src/resources/extensions/gsd/db/writers/memory.ts +220 -0
- package/src/resources/extensions/gsd/db/writers/reconcile.ts +500 -0
- package/src/resources/extensions/gsd/db/writers/status.ts +88 -0
- package/src/resources/extensions/gsd/dispatch-guard.ts +8 -31
- package/src/resources/extensions/gsd/doctor-environment.ts +5 -13
- package/src/resources/extensions/gsd/doctor-format.ts +12 -7
- package/src/resources/extensions/gsd/doctor-git-checks.ts +3 -3
- package/src/resources/extensions/gsd/doctor-runtime-checks.ts +22 -17
- package/src/resources/extensions/gsd/engine-hook-contract.ts +79 -0
- package/src/resources/extensions/gsd/error-classifier.ts +11 -0
- package/src/resources/extensions/gsd/exec-sandbox.ts +49 -9
- package/src/resources/extensions/gsd/files.ts +33 -12
- package/src/resources/extensions/gsd/git-service.ts +1 -0
- package/src/resources/extensions/gsd/gitignore.ts +3 -0
- package/src/resources/extensions/gsd/gsd-db.ts +173 -2373
- package/src/resources/extensions/gsd/guidance.ts +217 -0
- package/src/resources/extensions/gsd/guided-flow.ts +50 -5
- package/src/resources/extensions/gsd/markdown-renderer.ts +11 -0
- package/src/resources/extensions/gsd/mcp-filter.ts +2 -23
- package/src/resources/extensions/gsd/mcp-tool-name.ts +6 -11
- package/src/resources/extensions/gsd/memory-consolidation-scanner.ts +1 -1
- package/src/resources/extensions/gsd/migrate/safety.ts +18 -7
- package/src/resources/extensions/gsd/migration-auto-check.ts +28 -3
- package/src/resources/extensions/gsd/milestone-closeout.ts +13 -23
- package/src/resources/extensions/gsd/model-cost-table.ts +1 -0
- package/src/resources/extensions/gsd/model-router.ts +3 -0
- package/src/resources/extensions/gsd/notification-store.ts +26 -3
- package/src/resources/extensions/gsd/parallel-merge.ts +12 -9
- package/src/resources/extensions/gsd/parallel-monitor-overlay.ts +10 -7
- package/src/resources/extensions/gsd/parsers-legacy.ts +16 -4
- package/src/resources/extensions/gsd/paths.ts +42 -22
- package/src/resources/extensions/gsd/pre-execution-checks.ts +109 -3
- package/src/resources/extensions/gsd/preferences-models.ts +12 -47
- package/src/resources/extensions/gsd/preferences.ts +18 -0
- package/src/resources/extensions/gsd/prompts/complete-slice.md +2 -2
- package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/refine-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/replan-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/run-uat.md +6 -4
- package/src/resources/extensions/gsd/prompts/system.md +5 -2
- package/src/resources/extensions/gsd/provider-error-guidance.ts +4 -9
- package/src/resources/extensions/gsd/provider-switch-observer.ts +1 -1
- package/src/resources/extensions/gsd/publication.ts +122 -0
- package/src/resources/extensions/gsd/reactive-graph.ts +11 -1
- package/src/resources/extensions/gsd/recovery-classification.ts +47 -88
- package/src/resources/extensions/gsd/safety/destructive-confirmation.ts +134 -0
- package/src/resources/extensions/gsd/safety/evidence-collector.ts +36 -4
- package/src/resources/extensions/gsd/safety/evidence-cross-ref.ts +7 -2
- package/src/resources/extensions/gsd/safety/file-change-validator.ts +14 -0
- package/src/resources/extensions/gsd/state-transition-matrix.ts +42 -0
- package/src/resources/extensions/gsd/state.ts +9 -21
- package/src/resources/extensions/gsd/status-guards.ts +59 -8
- package/src/resources/extensions/gsd/stop-notice.ts +75 -0
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +123 -0
- package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +22 -0
- package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +101 -26
- package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +3 -1
- package/src/resources/extensions/gsd/tests/auto-post-unit-evidence-crossref-4909.test.ts +46 -0
- package/src/resources/extensions/gsd/tests/auto-worktree-registry.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/auto-worktree-repair.test.ts +4 -2
- package/src/resources/extensions/gsd/tests/browser-automation-contract-fixture.ts +39 -0
- package/src/resources/extensions/gsd/tests/browser-contract.test.ts +44 -0
- package/src/resources/extensions/gsd/tests/browser-daemon-auto-prep.test.ts +144 -0
- package/src/resources/extensions/gsd/tests/checkout-branch-stash-guard.test.ts +66 -1
- package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +44 -0
- package/src/resources/extensions/gsd/tests/commands-verdict.test.ts +8 -7
- package/src/resources/extensions/gsd/tests/complete-slice-verification-gate.test.ts +42 -0
- package/src/resources/extensions/gsd/tests/consent-question.test.ts +336 -0
- package/src/resources/extensions/gsd/tests/custom-verify-retry-store.test.ts +67 -0
- package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +10 -10
- package/src/resources/extensions/gsd/tests/destructive-confirmation.test.ts +303 -0
- package/src/resources/extensions/gsd/tests/dispatch-history.test.ts +273 -0
- package/src/resources/extensions/gsd/tests/dispatch-run-uat-browser-tools.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/dynamic-bash-no-cap.test.ts +132 -0
- package/src/resources/extensions/gsd/tests/engine-hook-contract.test.ts +148 -0
- package/src/resources/extensions/gsd/tests/evidence-xref-gsd-exec.test.ts +157 -0
- package/src/resources/extensions/gsd/tests/exec-graceful-kill.test.ts +193 -0
- package/src/resources/extensions/gsd/tests/exec-tool.test.ts +29 -1
- package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +35 -1
- package/src/resources/extensions/gsd/tests/file-change-validator.test.ts +33 -1
- package/src/resources/extensions/gsd/tests/gsd-db.test.ts +27 -0
- package/src/resources/extensions/gsd/tests/guidance.test.ts +148 -0
- package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +58 -15
- package/src/resources/extensions/gsd/tests/integration/auto-worktree.test.ts +74 -59
- package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/integration/gsd-integration-fixture.ts +80 -0
- package/src/resources/extensions/gsd/tests/integration/run-uat.test.ts +199 -0
- package/src/resources/extensions/gsd/tests/mcp-project-config.test.ts +3 -1
- package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +85 -1
- package/src/resources/extensions/gsd/tests/model-unittype-mapping.test.ts +32 -1
- package/src/resources/extensions/gsd/tests/notification-store.test.ts +32 -0
- package/src/resources/extensions/gsd/tests/oauth-api-model-routing.test.ts +167 -0
- package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +18 -0
- package/src/resources/extensions/gsd/tests/parsers-legacy-importers.test.ts +139 -0
- package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +193 -1
- package/src/resources/extensions/gsd/tests/prompt-db.test.ts +124 -6
- package/src/resources/extensions/gsd/tests/provider-error-guidance.test.ts +3 -3
- package/src/resources/extensions/gsd/tests/publication.test.ts +120 -0
- package/src/resources/extensions/gsd/tests/recovery-classification-illegal-transition.test.ts +30 -0
- package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +248 -1
- package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +1 -0
- package/src/resources/extensions/gsd/tests/safety-harness-false-positives.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/session-switch-clears-pending-autostart.test.ts +108 -0
- package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +43 -6
- package/src/resources/extensions/gsd/tests/state-transition-matrix.test.ts +36 -0
- package/src/resources/extensions/gsd/tests/status-guards.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/stop-notice.test.ts +70 -0
- package/src/resources/extensions/gsd/tests/token-tool-gating.test.ts +76 -0
- package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +8 -0
- package/src/resources/extensions/gsd/tests/tool-surface-readiness.test.ts +155 -0
- package/src/resources/extensions/gsd/tests/uat-policy.test.ts +112 -29
- package/src/resources/extensions/gsd/tests/unit-closeout.test.ts +209 -0
- package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +67 -2
- package/src/resources/extensions/gsd/tests/unit-registry.test.ts +163 -0
- package/src/resources/extensions/gsd/tests/web-app-uat.test.ts +44 -1
- package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/worktree-lifecycle.test.ts +41 -4
- package/src/resources/extensions/gsd/tests/worktree-manager.test.ts +22 -1
- package/src/resources/extensions/gsd/tests/worktree-placement.test.ts +113 -0
- package/src/resources/extensions/gsd/tests/worktree-reentry.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/worktree-safety.test.ts +3 -1
- package/src/resources/extensions/gsd/tests/worktree-symlink-removal.test.ts +12 -6
- package/src/resources/extensions/gsd/tests/worktree-teardown-safety.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/write-gate-seam.test.ts +358 -0
- package/src/resources/extensions/gsd/tests/write-gate.test.ts +109 -1
- package/src/resources/extensions/gsd/tool-presentation-plan.ts +4 -4
- package/src/resources/extensions/gsd/tool-surface-readiness.ts +76 -0
- package/src/resources/extensions/gsd/tools/complete-slice.ts +43 -68
- package/src/resources/extensions/gsd/tools/exec-tool.ts +9 -8
- package/src/resources/extensions/gsd/tools/plan-slice.ts +12 -6
- package/src/resources/extensions/gsd/tools/reopen-milestone.ts +11 -38
- package/src/resources/extensions/gsd/tools/reopen-slice.ts +14 -42
- package/src/resources/extensions/gsd/tools/skip-slice.ts +18 -44
- package/src/resources/extensions/gsd/uat-policy.ts +62 -16
- package/src/resources/extensions/gsd/undo.ts +9 -8
- package/src/resources/extensions/gsd/unit-closeout.ts +201 -0
- package/src/resources/extensions/gsd/unit-context-composer.ts +111 -1
- package/src/resources/extensions/gsd/unit-context-manifest.ts +4 -28
- package/src/resources/extensions/gsd/unit-registry.ts +412 -0
- package/src/resources/extensions/gsd/unit-tool-contracts.ts +27 -192
- package/src/resources/extensions/gsd/verdict-parser.ts +1 -1
- package/src/resources/extensions/gsd/web-app-uat.ts +51 -8
- package/src/resources/extensions/gsd/workflow-tool-surface.ts +4 -1
- package/src/resources/extensions/gsd/worktree-git-recovery.ts +314 -0
- package/src/resources/extensions/gsd/worktree-lifecycle.ts +13 -9
- package/src/resources/extensions/gsd/worktree-manager.ts +47 -28
- package/src/resources/extensions/gsd/worktree-placement.ts +63 -0
- package/src/resources/extensions/gsd/worktree-reentry.ts +10 -7
- package/src/resources/extensions/gsd/worktree-root.ts +29 -6
- package/src/resources/extensions/gsd/worktree-safety.ts +8 -5
- package/src/resources/extensions/gsd/worktree-session-state.ts +11 -11
- package/src/resources/extensions/search-the-web/native-search.ts +5 -3
- package/src/resources/extensions/shared/browser-contract.ts +66 -0
- package/src/resources/extensions/shared/gsd-browser-cli.ts +119 -5
- package/src/resources/shared/package.json +3 -0
- package/src/resources/skills/create-skill/references/executable-code.md +1 -1
- package/src/resources/skills/create-skill/workflows/add-reference.md +8 -3
- package/src/resources/skills/create-skill/workflows/add-script.md +4 -2
- package/src/resources/skills/create-skill/workflows/add-template.md +3 -1
- package/src/resources/skills/create-skill/workflows/add-workflow.md +8 -3
- package/src/resources/skills/create-skill/workflows/upgrade-to-router.md +10 -5
- package/src/resources/skills/create-skill/workflows/verify-skill.md +9 -4
- package/src/resources/skills/gsd-browser/SKILL.md +1 -1
- package/src/resources/skills/spike-wrap-up/SKILL.md +9 -9
- package/dist/resources/extensions/gsd/user-input-boundary.js +0 -218
- package/src/resources/extensions/gsd/tests/user-input-boundary.test.ts +0 -173
- package/src/resources/extensions/gsd/user-input-boundary.ts +0 -216
- /package/dist/web/standalone/.next/static/{3PtrU9qGPEXwNLWkIyiqk → jmTLg6xZmAuq_LIqKOxrH}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{3PtrU9qGPEXwNLWkIyiqk → jmTLg6xZmAuq_LIqKOxrH}/_ssgManifest.js +0 -0
|
@@ -88,3 +88,49 @@ test("detects session execution tools supported by the evidence collector", () =
|
|
|
88
88
|
|
|
89
89
|
assert.equal(_hasExecutionToolCallsInSessionForTest(entries), true);
|
|
90
90
|
});
|
|
91
|
+
|
|
92
|
+
test("detects execution tool calls in bare agent-end messages (no session-entry wrapper)", () => {
|
|
93
|
+
// The auto loop passes opts.agentEndMessages as bare {role, content}
|
|
94
|
+
// messages — not {type: "message", message} session-manager entries.
|
|
95
|
+
const entries = [
|
|
96
|
+
{
|
|
97
|
+
role: "assistant",
|
|
98
|
+
content: [
|
|
99
|
+
{
|
|
100
|
+
type: "toolCall",
|
|
101
|
+
name: "Bash",
|
|
102
|
+
arguments: { command: "test -s index.html && grep -q localStorage index.html" },
|
|
103
|
+
},
|
|
104
|
+
],
|
|
105
|
+
},
|
|
106
|
+
];
|
|
107
|
+
|
|
108
|
+
assert.equal(_hasExecutionToolCallsInSessionForTest(entries), true);
|
|
109
|
+
});
|
|
110
|
+
|
|
111
|
+
test("does not suppress for bare agent-end messages without execution tools", () => {
|
|
112
|
+
const entries = [
|
|
113
|
+
{
|
|
114
|
+
role: "assistant",
|
|
115
|
+
content: [
|
|
116
|
+
{ type: "text", text: "Task complete." },
|
|
117
|
+
{ type: "toolCall", name: "Write", arguments: { file_path: "index.html" } },
|
|
118
|
+
],
|
|
119
|
+
},
|
|
120
|
+
];
|
|
121
|
+
|
|
122
|
+
assert.equal(_hasExecutionToolCallsInSessionForTest(entries), false);
|
|
123
|
+
});
|
|
124
|
+
|
|
125
|
+
test("ignores bare user messages with toolCall-shaped content", () => {
|
|
126
|
+
const entries = [
|
|
127
|
+
{
|
|
128
|
+
role: "user",
|
|
129
|
+
content: [
|
|
130
|
+
{ type: "toolCall", name: "bash", arguments: { command: "echo hi" } },
|
|
131
|
+
],
|
|
132
|
+
},
|
|
133
|
+
];
|
|
134
|
+
|
|
135
|
+
assert.equal(_hasExecutionToolCallsInSessionForTest(entries), false);
|
|
136
|
+
});
|
|
@@ -184,7 +184,7 @@ describe("auto-worktree workspace registry", () => {
|
|
|
184
184
|
git(["commit", "-m", "add milestone"], tempDir);
|
|
185
185
|
|
|
186
186
|
createAutoWorktree(tempDir, "M003");
|
|
187
|
-
const wtDir = join(tempDir, ".gsd
|
|
187
|
+
const wtDir = join(tempDir, ".gsd-worktrees", "M003");
|
|
188
188
|
writeFileSync(join(wtDir, "feature.txt"), "implemented\n");
|
|
189
189
|
git(["add", "feature.txt"], wtDir);
|
|
190
190
|
git(["commit", "-m", "feat: implement M003"], wtDir);
|
|
@@ -216,7 +216,7 @@ describe("auto-worktree workspace registry", () => {
|
|
|
216
216
|
git(["commit", "-m", "add milestone"], tempDir);
|
|
217
217
|
|
|
218
218
|
createAutoWorktree(tempDir, "M004");
|
|
219
|
-
const wtDir = join(tempDir, ".gsd
|
|
219
|
+
const wtDir = join(tempDir, ".gsd-worktrees", "M004");
|
|
220
220
|
writeFileSync(join(wtDir, "feature.txt"), "implemented\n");
|
|
221
221
|
git(["add", "feature.txt"], wtDir);
|
|
222
222
|
git(["commit", "-m", "feat: implement M004"], wtDir);
|
|
@@ -35,7 +35,8 @@ test("repair target accepts a missing expected milestone worktree", () => {
|
|
|
35
35
|
|
|
36
36
|
assert.equal(result.ok, true);
|
|
37
37
|
if (result.ok) {
|
|
38
|
-
|
|
38
|
+
// No worktree exists yet, so the expected path is the canonical container.
|
|
39
|
+
assert.equal(result.expectedPath, join(base, ".gsd-worktrees", "M001"));
|
|
39
40
|
}
|
|
40
41
|
} finally {
|
|
41
42
|
cleanup(base);
|
|
@@ -192,7 +193,8 @@ test("paused metadata path resolves to the expected worktree while paused at pro
|
|
|
192
193
|
baseIsAutoWorktree: false,
|
|
193
194
|
});
|
|
194
195
|
|
|
195
|
-
|
|
196
|
+
// No worktree exists yet, so resolution lands at the canonical container.
|
|
197
|
+
assert.equal(result, join(base, ".gsd-worktrees", "M001"));
|
|
196
198
|
} finally {
|
|
197
199
|
cleanup(base);
|
|
198
200
|
}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import assert from "node:assert/strict";
|
|
2
|
+
|
|
3
|
+
import {
|
|
4
|
+
getUatBrowserToolSupportError,
|
|
5
|
+
hasUatBrowserToolSurface,
|
|
6
|
+
type UatType,
|
|
7
|
+
} from "../uat-policy.ts";
|
|
8
|
+
|
|
9
|
+
export const BROWSER_AUTOMATION_CONTRACT_TOOLS = {
|
|
10
|
+
piProvider: ["read", "browser_navigate"],
|
|
11
|
+
externalMcpClient: ["read", "mcp__gsd-browser__browser_navigate"],
|
|
12
|
+
externalMcpWildcard: ["read", "mcp__gsd-browser__*"],
|
|
13
|
+
otherBrowserMcp: ["read", "mcp__browser-uat__*"],
|
|
14
|
+
workflowOnly: ["read", "mcp__gsd-workflow__*"],
|
|
15
|
+
withoutBrowser: ["read", "gsd_uat_exec"],
|
|
16
|
+
} as const;
|
|
17
|
+
|
|
18
|
+
export function assertBrowserAutomationContractAvailable(tools: readonly string[]): void {
|
|
19
|
+
assert.equal(hasUatBrowserToolSurface(tools), true, `${tools.join(", ")} should satisfy the Browser Automation Contract`);
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export function assertBrowserAutomationContractMissing(tools: readonly string[] | undefined): void {
|
|
23
|
+
assert.equal(hasUatBrowserToolSurface(tools), false, `${tools?.join(", ") ?? "undefined"} should not satisfy the Browser Automation Contract`);
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export function assertBrowserBackedUatCanDispatch(options: {
|
|
27
|
+
uatType: UatType;
|
|
28
|
+
activeTools: readonly string[] | undefined;
|
|
29
|
+
registeredTools?: readonly string[];
|
|
30
|
+
}): void {
|
|
31
|
+
assert.equal(
|
|
32
|
+
getUatBrowserToolSupportError({
|
|
33
|
+
...options,
|
|
34
|
+
milestoneId: "M001",
|
|
35
|
+
sliceId: "S01",
|
|
36
|
+
}),
|
|
37
|
+
null,
|
|
38
|
+
);
|
|
39
|
+
}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { describe, it } from "node:test";
|
|
2
|
+
import assert from "node:assert/strict";
|
|
3
|
+
|
|
4
|
+
import {
|
|
5
|
+
BROWSER_CONTRACT_TOOL_NAMES,
|
|
6
|
+
BROWSER_EVIDENCE_SIGNAL_TOOL_NAMES,
|
|
7
|
+
hasBrowserContractPrefix,
|
|
8
|
+
isBrowserContractToolName,
|
|
9
|
+
} from "../../shared/browser-contract.ts";
|
|
10
|
+
import { isUatBrowserToolName } from "../uat-policy.ts";
|
|
11
|
+
import { BROWSER_REQUIREMENT_RE, BROWSER_RUNTIME_RE } from "../browser-evidence.ts";
|
|
12
|
+
|
|
13
|
+
// Note: RUN_UAT_BROWSER_TOOL_NAMES and MANAGED_GSD_BROWSER_TOOL_NAMES are
|
|
14
|
+
// reference-equal aliases of BROWSER_CONTRACT_TOOL_NAMES, and the managed
|
|
15
|
+
// adapter's spec table is Record-keyed by BrowserContractToolName — both
|
|
16
|
+
// derivations are pinned by the type system, not by runtime assertions here.
|
|
17
|
+
describe("Browser Automation Contract parity", () => {
|
|
18
|
+
it("every contract name satisfies the UAT browser-tool predicate, bare and MCP-prefixed", () => {
|
|
19
|
+
for (const name of BROWSER_CONTRACT_TOOL_NAMES) {
|
|
20
|
+
assert.equal(isUatBrowserToolName(name), true, name);
|
|
21
|
+
assert.equal(isUatBrowserToolName(`mcp__gsd-browser__${name}`), true, `mcp__gsd-browser__${name}`);
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
it("contract names are canonical browser_* names with no duplicates", () => {
|
|
26
|
+
assert.equal(new Set(BROWSER_CONTRACT_TOOL_NAMES).size, BROWSER_CONTRACT_TOOL_NAMES.length);
|
|
27
|
+
for (const name of BROWSER_CONTRACT_TOOL_NAMES) {
|
|
28
|
+
assert.equal(hasBrowserContractPrefix(name), true, name);
|
|
29
|
+
assert.equal(isBrowserContractToolName(name), true, name);
|
|
30
|
+
}
|
|
31
|
+
assert.equal(isBrowserContractToolName("browser_not_a_real_tool"), false);
|
|
32
|
+
assert.equal(hasBrowserContractPrefix("gsd_uat_exec"), false);
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
it("evidence-signal names stay a subset of the contract and drive the detection regexes", () => {
|
|
36
|
+
for (const name of BROWSER_EVIDENCE_SIGNAL_TOOL_NAMES) {
|
|
37
|
+
assert.equal(isBrowserContractToolName(name), true, name);
|
|
38
|
+
// Identifier-shaped names keep the regex splice in browser-evidence.ts escape-free.
|
|
39
|
+
assert.match(name, /^browser_[a-z_]+$/);
|
|
40
|
+
assert.match(`Verified via ${name} call`, BROWSER_REQUIREMENT_RE);
|
|
41
|
+
assert.match(`Verified via ${name} call`, BROWSER_RUNTIME_RE);
|
|
42
|
+
}
|
|
43
|
+
});
|
|
44
|
+
});
|
|
@@ -0,0 +1,144 @@
|
|
|
1
|
+
import test from "node:test";
|
|
2
|
+
import assert from "node:assert/strict";
|
|
3
|
+
|
|
4
|
+
import {
|
|
5
|
+
prepareBrowserDaemonForUat,
|
|
6
|
+
shouldWarmBrowserDaemonForUat,
|
|
7
|
+
} from "../browser-daemon-auto-prep.ts";
|
|
8
|
+
import { commitBrowserEngineResolution } from "../../browser-tools/engine/selection.ts";
|
|
9
|
+
import { resolveGsdBrowserCliAvailability } from "../../shared/gsd-browser-cli.ts";
|
|
10
|
+
|
|
11
|
+
const GSD_BROWSER_ENGINE = { GSD_BROWSER_ENGINE: "gsd-browser" } as const;
|
|
12
|
+
|
|
13
|
+
test("shouldWarmBrowserDaemonForUat skips artifact-driven UAT", () => {
|
|
14
|
+
assert.equal(
|
|
15
|
+
shouldWarmBrowserDaemonForUat({
|
|
16
|
+
uatType: "artifact-driven",
|
|
17
|
+
sessionProvider: "claude-code",
|
|
18
|
+
projectRoot: "/tmp/project",
|
|
19
|
+
}),
|
|
20
|
+
false,
|
|
21
|
+
);
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
test("shouldWarmBrowserDaemonForUat enables Claude Code browser UAT when gsd-browser is available", (t) => {
|
|
25
|
+
const availability = resolveGsdBrowserCliAvailability();
|
|
26
|
+
if (!availability.available) {
|
|
27
|
+
t.skip("bundled gsd-browser CLI unavailable");
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
assert.equal(
|
|
31
|
+
shouldWarmBrowserDaemonForUat({
|
|
32
|
+
uatType: "browser-executable",
|
|
33
|
+
sessionProvider: "claude-code",
|
|
34
|
+
sessionAuthMode: "externalCli",
|
|
35
|
+
projectRoot: "/tmp/project",
|
|
36
|
+
env: GSD_BROWSER_ENGINE,
|
|
37
|
+
}),
|
|
38
|
+
true,
|
|
39
|
+
);
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
test("shouldWarmBrowserDaemonForUat enables warm-up for Claude Code oauth/apiKey when engine is gsd-browser", (t) => {
|
|
43
|
+
const availability = resolveGsdBrowserCliAvailability();
|
|
44
|
+
if (!availability.available) {
|
|
45
|
+
t.skip("bundled gsd-browser CLI unavailable");
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
for (const sessionAuthMode of ["oauth", "apiKey"] as const) {
|
|
49
|
+
assert.equal(
|
|
50
|
+
shouldWarmBrowserDaemonForUat({
|
|
51
|
+
uatType: "browser-executable",
|
|
52
|
+
sessionProvider: "claude-code",
|
|
53
|
+
sessionAuthMode,
|
|
54
|
+
sessionBaseUrl: "https://api.anthropic.com",
|
|
55
|
+
projectRoot: "/tmp/project",
|
|
56
|
+
env: GSD_BROWSER_ENGINE,
|
|
57
|
+
}),
|
|
58
|
+
true,
|
|
59
|
+
`expected warm-up for sessionAuthMode=${sessionAuthMode}`,
|
|
60
|
+
);
|
|
61
|
+
}
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
test("shouldWarmBrowserDaemonForUat skips legacy Playwright engine for Claude Code", () => {
|
|
65
|
+
assert.equal(
|
|
66
|
+
shouldWarmBrowserDaemonForUat({
|
|
67
|
+
uatType: "browser-executable",
|
|
68
|
+
sessionProvider: "claude-code",
|
|
69
|
+
sessionAuthMode: "oauth",
|
|
70
|
+
projectRoot: "/tmp/project",
|
|
71
|
+
env: { GSD_BROWSER_ENGINE: "legacy" },
|
|
72
|
+
}),
|
|
73
|
+
false,
|
|
74
|
+
);
|
|
75
|
+
});
|
|
76
|
+
|
|
77
|
+
test("shouldWarmBrowserDaemonForUat uses session-committed ambient engine for non-Claude providers", () => {
|
|
78
|
+
const projectRoot = "/tmp/ambient-engine-project";
|
|
79
|
+
commitBrowserEngineResolution(projectRoot, {
|
|
80
|
+
engine: "legacy",
|
|
81
|
+
source: "probe",
|
|
82
|
+
reason: "gsd-browser daemon connect failed (test); using legacy Playwright",
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
assert.equal(
|
|
86
|
+
shouldWarmBrowserDaemonForUat({
|
|
87
|
+
uatType: "browser-executable",
|
|
88
|
+
sessionProvider: "openai",
|
|
89
|
+
projectRoot,
|
|
90
|
+
}),
|
|
91
|
+
false,
|
|
92
|
+
);
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
test("shouldWarmBrowserDaemonForUat skips when browser MCP is disabled", () => {
|
|
96
|
+
assert.equal(
|
|
97
|
+
shouldWarmBrowserDaemonForUat({
|
|
98
|
+
uatType: "browser-executable",
|
|
99
|
+
sessionProvider: "claude-code",
|
|
100
|
+
projectRoot: "/tmp/project",
|
|
101
|
+
env: { GSD_BROWSER_MCP_ENABLED: "0" },
|
|
102
|
+
}),
|
|
103
|
+
false,
|
|
104
|
+
);
|
|
105
|
+
});
|
|
106
|
+
|
|
107
|
+
test("shouldWarmBrowserDaemonForUat skips when warm-up is disabled", () => {
|
|
108
|
+
assert.equal(
|
|
109
|
+
shouldWarmBrowserDaemonForUat({
|
|
110
|
+
uatType: "browser-executable",
|
|
111
|
+
sessionProvider: "claude-code",
|
|
112
|
+
projectRoot: "/tmp/project",
|
|
113
|
+
env: { GSD_BROWSER_WARMUP: "0" },
|
|
114
|
+
}),
|
|
115
|
+
false,
|
|
116
|
+
);
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
test("prepareBrowserDaemonForUat returns null when warm-up is not required", () => {
|
|
120
|
+
assert.equal(
|
|
121
|
+
prepareBrowserDaemonForUat({
|
|
122
|
+
uatType: "artifact-driven",
|
|
123
|
+
sessionProvider: "claude-code",
|
|
124
|
+
sessionAuthMode: "externalCli",
|
|
125
|
+
projectRoot: "/tmp/example-project",
|
|
126
|
+
}),
|
|
127
|
+
null,
|
|
128
|
+
);
|
|
129
|
+
});
|
|
130
|
+
|
|
131
|
+
test("prepareBrowserDaemonForUat returns actionable error when daemon start fails", () => {
|
|
132
|
+
const error = prepareBrowserDaemonForUat({
|
|
133
|
+
uatType: "browser-executable",
|
|
134
|
+
sessionProvider: "claude-code",
|
|
135
|
+
sessionAuthMode: "externalCli",
|
|
136
|
+
projectRoot: "/tmp/example-project",
|
|
137
|
+
env: {
|
|
138
|
+
...GSD_BROWSER_ENGINE,
|
|
139
|
+
GSD_BROWSER_MCP_COMMAND: "/definitely/missing/gsd-browser",
|
|
140
|
+
},
|
|
141
|
+
});
|
|
142
|
+
|
|
143
|
+
assert.match(error ?? "", /gsd-browser daemon failed to start/i);
|
|
144
|
+
});
|
|
@@ -5,7 +5,7 @@ import { mkdirSync, mkdtempSync, readFileSync, realpathSync, rmSync, writeFileSy
|
|
|
5
5
|
import { join } from "node:path";
|
|
6
6
|
import { tmpdir } from "node:os";
|
|
7
7
|
|
|
8
|
-
import { checkoutBranchWithStashGuard } from "../
|
|
8
|
+
import { checkoutBranchWithStashGuard } from "../worktree-git-recovery.ts";
|
|
9
9
|
|
|
10
10
|
function git(args: string[], cwd: string): string {
|
|
11
11
|
return execFileSync("git", args, {
|
|
@@ -159,4 +159,69 @@ describe("checkoutBranchWithStashGuard", () => {
|
|
|
159
159
|
const stashList = git(["stash", "list"], repo).trim();
|
|
160
160
|
assert.equal(stashList, "");
|
|
161
161
|
});
|
|
162
|
+
|
|
163
|
+
test("auto-resolves combined non-.gsd untracked collision and .gsd index conflict", (t) => {
|
|
164
|
+
// Regression for: gate checked nonGsdUnmerged.length === 0 but ignored
|
|
165
|
+
// .gsd/ index conflicts, so a failed pop with both an untracked non-.gsd/
|
|
166
|
+
// collision AND a .gsd/ tracked conflict would drop the stash while leaving
|
|
167
|
+
// .gsd/ index unmerged entries unresolved.
|
|
168
|
+
const repo = createRepo(t);
|
|
169
|
+
// Add .gsd/DECISIONS.md to base so both branches diverge from it
|
|
170
|
+
writeFileSync(join(repo, ".gsd", "DECISIONS.md"), "base\n");
|
|
171
|
+
git(["add", ".gsd/DECISIONS.md"], repo);
|
|
172
|
+
git(["commit", "-m", "add decisions"], repo);
|
|
173
|
+
|
|
174
|
+
git(["checkout", "-b", "milestone/M003"], repo);
|
|
175
|
+
// Branch has its own version of DECISIONS.md (non-JSONL .gsd/ file)
|
|
176
|
+
writeFileSync(join(repo, ".gsd", "DECISIONS.md"), "target-version\n");
|
|
177
|
+
mkdirSync(join(repo, ".harness"), { recursive: true });
|
|
178
|
+
writeFileSync(join(repo, ".harness", "settings.json"), "{\"theme\":\"dark\"}\n");
|
|
179
|
+
git(["add", ".gsd/DECISIONS.md", ".harness/settings.json"], repo);
|
|
180
|
+
git(["commit", "-m", "branch state"], repo);
|
|
181
|
+
git(["checkout", "main"], repo);
|
|
182
|
+
|
|
183
|
+
// On main: tracked .gsd/ change (creates unmerged index entry on pop) +
|
|
184
|
+
// untracked harness file (triggers "already exists, no checkout").
|
|
185
|
+
writeFileSync(join(repo, ".gsd", "DECISIONS.md"), "local-version\n");
|
|
186
|
+
mkdirSync(join(repo, ".harness"), { recursive: true });
|
|
187
|
+
writeFileSync(join(repo, ".harness", "settings.json"), "{\"theme\":\"light\"}\n");
|
|
188
|
+
|
|
189
|
+
checkoutBranchWithStashGuard(repo, "milestone/M003", "test-combined-gsd-conflict");
|
|
190
|
+
|
|
191
|
+
const branch = git(["branch", "--show-current"], repo).trim();
|
|
192
|
+
assert.equal(branch, "milestone/M003");
|
|
193
|
+
const decisions = readFileSync(join(repo, ".gsd", "DECISIONS.md"), "utf8");
|
|
194
|
+
assert.equal(decisions, "target-version\n");
|
|
195
|
+
assert.doesNotMatch(decisions, /<<<<<<<|=======|>>>>>>>/);
|
|
196
|
+
const status = git(["status", "--porcelain"], repo).trim();
|
|
197
|
+
assert.equal(status, "");
|
|
198
|
+
const stashList = git(["stash", "list"], repo).trim();
|
|
199
|
+
assert.equal(stashList, "");
|
|
200
|
+
});
|
|
201
|
+
|
|
202
|
+
test("auto-resolves non-.gsd untracked restore collisions (e.g. harness config outside .gsd/)", (t) => {
|
|
203
|
+
const repo = createRepo(t);
|
|
204
|
+
// Target branch has a harness config file committed; source (main) has it untracked.
|
|
205
|
+
// Use a path not affected by global gitignore rules (unlike .claude/settings.local.json).
|
|
206
|
+
git(["checkout", "-b", "milestone/M001"], repo);
|
|
207
|
+
mkdirSync(join(repo, ".harness"), { recursive: true });
|
|
208
|
+
writeFileSync(join(repo, ".harness", "settings.json"), "{\"theme\":\"dark\"}\n");
|
|
209
|
+
git(["add", ".harness/settings.json"], repo);
|
|
210
|
+
git(["commit", "-m", "add harness settings"], repo);
|
|
211
|
+
git(["checkout", "main"], repo);
|
|
212
|
+
|
|
213
|
+
mkdirSync(join(repo, ".harness"), { recursive: true });
|
|
214
|
+
writeFileSync(join(repo, ".harness", "settings.json"), "{\"theme\":\"light\"}\n");
|
|
215
|
+
|
|
216
|
+
checkoutBranchWithStashGuard(repo, "milestone/M001", "test-non-gsd-untracked-collision");
|
|
217
|
+
|
|
218
|
+
const branch = git(["branch", "--show-current"], repo).trim();
|
|
219
|
+
assert.equal(branch, "milestone/M001");
|
|
220
|
+
const wtContent = readFileSync(join(repo, ".harness", "settings.json"), "utf8");
|
|
221
|
+
assert.equal(wtContent, "{\"theme\":\"dark\"}\n");
|
|
222
|
+
const status = git(["status", "--porcelain"], repo).trim();
|
|
223
|
+
assert.equal(status, "");
|
|
224
|
+
const stashList = git(["stash", "list"], repo).trim();
|
|
225
|
+
assert.equal(stashList, "");
|
|
226
|
+
});
|
|
162
227
|
});
|
|
@@ -73,4 +73,48 @@ describe("clear stale pending auto-start (#3667)", () => {
|
|
|
73
73
|
"pending auto-start gate must clear stale map entries for completed discussions",
|
|
74
74
|
);
|
|
75
75
|
});
|
|
76
|
+
|
|
77
|
+
test("guided-flow recovers a finished-but-unconsumed discussion instead of dead-ending", () => {
|
|
78
|
+
// CONTEXT exists + no live turn means the discussion completed but the
|
|
79
|
+
// agent_end handoff never consumed the entry (e.g. an external-engine
|
|
80
|
+
// post-hoc gate re-arm wiped the depth verification after the save).
|
|
81
|
+
// Without recovery, every /gsd prints "Discussion already in progress"
|
|
82
|
+
// forever: the stale heuristic requires CONTEXT to be absent and
|
|
83
|
+
// discussPlanComplete requires a ROADMAP that planning never produced.
|
|
84
|
+
const source = readFileSync(join(__dirname, "..", "guided-flow.ts"), "utf-8");
|
|
85
|
+
assert.ok(
|
|
86
|
+
source.includes("milestoneHasContext && !isAgentTurnInFlight(ctx)"),
|
|
87
|
+
"pending-entry guard must have a recovery branch for CONTEXT-present, no-turn-in-flight entries",
|
|
88
|
+
);
|
|
89
|
+
assert.ok(
|
|
90
|
+
source.includes("extractDepthVerificationMilestoneId(pendingGateId) === entry.milestoneId"),
|
|
91
|
+
"recovery must only clear a pending gate belonging to the entry's own milestone",
|
|
92
|
+
);
|
|
93
|
+
assert.ok(
|
|
94
|
+
source.includes("if (checkAutoStartAfterDiscuss(basePath)) return;"),
|
|
95
|
+
"recovery must re-run the discuss→auto handoff after clearing the stale gate",
|
|
96
|
+
);
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
test("guided-flow does not treat a live discuss turn as a stale pending entry", () => {
|
|
100
|
+
const source = readFileSync(join(__dirname, "..", "guided-flow.ts"), "utf-8");
|
|
101
|
+
assert.ok(
|
|
102
|
+
source.includes("!isAgentTurnInFlight(ctx)"),
|
|
103
|
+
"stale-entry deletion must be gated on no agent turn being in flight — a dispatched " +
|
|
104
|
+
"discuss turn can think for over 30s before writing its first artifact, and deleting " +
|
|
105
|
+
"its entry re-dispatches the workflow (duplicate interview + duplicate completion message)",
|
|
106
|
+
);
|
|
107
|
+
assert.ok(
|
|
108
|
+
source.includes('const milestoneHasDraft = !!resolveMilestoneFile(basePath, entry.milestoneId, "CONTEXT-DRAFT");'),
|
|
109
|
+
"stale-entry check must treat an existing CONTEXT-DRAFT as proof of an in-progress interview",
|
|
110
|
+
);
|
|
111
|
+
assert.ok(
|
|
112
|
+
source.includes("!milestoneHasDraft"),
|
|
113
|
+
"stale-entry deletion must require the CONTEXT-DRAFT to be absent",
|
|
114
|
+
);
|
|
115
|
+
assert.ok(
|
|
116
|
+
source.includes("ctx.hasPendingMessages"),
|
|
117
|
+
"in-flight detection must also cover dispatched-but-not-yet-started queued messages",
|
|
118
|
+
);
|
|
119
|
+
});
|
|
76
120
|
});
|
|
@@ -472,18 +472,19 @@ test("auto-dispatch needs-attention pause message references /gsd verdict", asyn
|
|
|
472
472
|
}
|
|
473
473
|
});
|
|
474
474
|
|
|
475
|
-
test("
|
|
475
|
+
test("guidance.ts needs-remediation blocker messages reference /gsd verdict", async () => {
|
|
476
476
|
// We don't need to invoke deriveState — just assert the substring is in the
|
|
477
|
-
// source. The blocker strings are constructed
|
|
478
|
-
// verbatim, so a static check is sufficient and avoids
|
|
479
|
-
|
|
480
|
-
|
|
477
|
+
// source. The blocker strings are constructed in the guidance catalog and
|
|
478
|
+
// shipped to the user verbatim, so a static check is sufficient and avoids
|
|
479
|
+
// fragile DB setup.
|
|
480
|
+
const guidanceSource = readFileSync(
|
|
481
|
+
new URL("../guidance.ts", import.meta.url).pathname,
|
|
481
482
|
"utf-8",
|
|
482
483
|
);
|
|
483
|
-
const occurrences =
|
|
484
|
+
const occurrences = guidanceSource.match(/`\/gsd verdict /g) ?? [];
|
|
484
485
|
assert.ok(
|
|
485
486
|
occurrences.length >= 2,
|
|
486
|
-
`expected at least 2 references to /gsd verdict in
|
|
487
|
+
`expected at least 2 references to /gsd verdict in guidance.ts blockers, found ${occurrences.length}`,
|
|
487
488
|
);
|
|
488
489
|
});
|
|
489
490
|
|
|
@@ -258,6 +258,48 @@ describe('complete-slice verification gate (#3580)', () => {
|
|
|
258
258
|
}
|
|
259
259
|
});
|
|
260
260
|
|
|
261
|
+
test('allows a browser UAT declared as a bare keyword under ## UAT Type (M006/S01 format drift)', async () => {
|
|
262
|
+
// Regression: the agent wrote `## UAT Type\nbrowser-executable` (no
|
|
263
|
+
// `- UAT mode:` bullet). The old parser defaulted that to artifact-driven
|
|
264
|
+
// and the gate rejected the slice in a loop.
|
|
265
|
+
const body = [
|
|
266
|
+
'## UAT Type',
|
|
267
|
+
'browser-executable',
|
|
268
|
+
'',
|
|
269
|
+
'## Smoke Test',
|
|
270
|
+
'1. Open the page in a browser and perform add/edit/complete/delete once.',
|
|
271
|
+
].join('\n');
|
|
272
|
+
const result = await handleCompleteSlice(
|
|
273
|
+
makeParams({ uatContent: body }),
|
|
274
|
+
basePath,
|
|
275
|
+
);
|
|
276
|
+
if ('error' in result) {
|
|
277
|
+
assert.doesNotMatch(
|
|
278
|
+
result.error,
|
|
279
|
+
/requires browser verification/i,
|
|
280
|
+
`bare browser-executable declaration must pass the browser gate, got: ${result.error}`,
|
|
281
|
+
);
|
|
282
|
+
}
|
|
283
|
+
});
|
|
284
|
+
|
|
285
|
+
test('explains the missing declaration when a browser UAT has no parseable UAT mode', async () => {
|
|
286
|
+
// When nothing was declared, the error must not claim the agent
|
|
287
|
+
// "declared artifact-driven" — it must show the expected bullet format.
|
|
288
|
+
const body = [
|
|
289
|
+
'## Smoke Test',
|
|
290
|
+
'1. Open the page in a browser and perform add/edit/complete/delete once.',
|
|
291
|
+
].join('\n');
|
|
292
|
+
const result = await handleCompleteSlice(
|
|
293
|
+
makeParams({ uatContent: body }),
|
|
294
|
+
basePath,
|
|
295
|
+
);
|
|
296
|
+
assert.ok('error' in result, 'expected handler to reject an undeclared browser UAT');
|
|
297
|
+
const error = (result as { error: string }).error;
|
|
298
|
+
assert.match(error, /no parseable UAT mode declaration/i);
|
|
299
|
+
assert.match(error, /- UAT mode: browser-executable/);
|
|
300
|
+
assert.doesNotMatch(error, /but declares "UAT mode: artifact-driven"/);
|
|
301
|
+
});
|
|
302
|
+
|
|
261
303
|
test('allows a browser UAT when it is declared mixed (mixed receives browser tools)', async () => {
|
|
262
304
|
const body = BROWSER_UAT_BODY.replace('artifact-driven', 'mixed (artifact-driven + browser)');
|
|
263
305
|
const result = await handleCompleteSlice(
|