gsd-pi 2.63.0-dev.026d309 → 2.63.0-dev.786f0ff
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/README.md +46 -134
- package/dist/cli.js +48 -6
- package/dist/headless-query.js +11 -1
- package/dist/help-text.js +4 -1
- package/dist/onboarding.js +15 -8
- package/dist/resource-loader.js +18 -3
- package/dist/resources/extensions/cmux/index.js +21 -12
- package/dist/resources/extensions/gsd/auto/detect-stuck.js +27 -0
- package/dist/resources/extensions/gsd/auto/finalize-timeout.js +40 -0
- package/dist/resources/extensions/gsd/auto/loop.js +4 -0
- package/dist/resources/extensions/gsd/auto/phases.js +157 -22
- package/dist/resources/extensions/gsd/auto/session.js +12 -0
- package/dist/resources/extensions/gsd/auto-dashboard.js +9 -3
- package/dist/resources/extensions/gsd/auto-model-selection.js +32 -0
- package/dist/resources/extensions/gsd/auto-post-unit.js +124 -10
- package/dist/resources/extensions/gsd/auto-prompts.js +25 -0
- package/dist/resources/extensions/gsd/auto-recovery.js +15 -7
- package/dist/resources/extensions/gsd/auto-start.js +10 -21
- package/dist/resources/extensions/gsd/auto-timers.js +2 -1
- package/dist/resources/extensions/gsd/auto-tool-tracking.js +17 -0
- package/dist/resources/extensions/gsd/auto-worktree.js +13 -7
- package/dist/resources/extensions/gsd/auto.js +19 -2
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +147 -75
- package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +13 -0
- package/dist/resources/extensions/gsd/bootstrap/query-tools.js +85 -0
- package/dist/resources/extensions/gsd/bootstrap/register-extension.js +3 -0
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +32 -1
- package/dist/resources/extensions/gsd/bootstrap/sanitize-complete-milestone.js +54 -0
- package/dist/resources/extensions/gsd/bootstrap/system-context.js +30 -2
- package/dist/resources/extensions/gsd/commands-handlers.js +9 -4
- package/dist/resources/extensions/gsd/constants.js +42 -0
- package/dist/resources/extensions/gsd/db-writer.js +72 -4
- package/dist/resources/extensions/gsd/forensics.js +20 -4
- package/dist/resources/extensions/gsd/gsd-db.js +64 -17
- package/dist/resources/extensions/gsd/guided-flow.js +19 -0
- package/dist/resources/extensions/gsd/metrics.js +27 -1
- package/dist/resources/extensions/gsd/native-git-bridge.js +5 -3
- package/dist/resources/extensions/gsd/preferences-types.js +2 -0
- package/dist/resources/extensions/gsd/preferences.js +7 -2
- package/dist/resources/extensions/gsd/prompt-loader.js +7 -0
- package/dist/resources/extensions/gsd/prompts/complete-milestone.md +2 -0
- package/dist/resources/extensions/gsd/prompts/complete-slice.md +2 -0
- package/dist/resources/extensions/gsd/prompts/doctor-heal.md +1 -0
- package/dist/resources/extensions/gsd/prompts/forensics.md +2 -0
- package/dist/resources/extensions/gsd/prompts/reassess-roadmap.md +2 -0
- package/dist/resources/extensions/gsd/prompts/system.md +4 -7
- package/dist/resources/extensions/gsd/prompts/validate-milestone.md +2 -0
- package/dist/resources/extensions/gsd/roadmap-mutations.js +1 -1
- package/dist/resources/extensions/gsd/roadmap-slices.js +9 -5
- package/dist/resources/extensions/gsd/safety/content-validator.js +73 -0
- package/dist/resources/extensions/gsd/safety/destructive-guard.js +34 -0
- package/dist/resources/extensions/gsd/safety/evidence-collector.js +109 -0
- package/dist/resources/extensions/gsd/safety/evidence-cross-ref.js +83 -0
- package/dist/resources/extensions/gsd/safety/file-change-validator.js +71 -0
- package/dist/resources/extensions/gsd/safety/git-checkpoint.js +91 -0
- package/dist/resources/extensions/gsd/safety/safety-harness.js +64 -0
- package/dist/resources/extensions/gsd/slice-parallel-conflict.js +67 -0
- package/dist/resources/extensions/gsd/slice-parallel-eligibility.js +51 -0
- package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +378 -0
- package/dist/resources/extensions/gsd/state.js +74 -14
- package/dist/resources/extensions/gsd/status-guards.js +11 -0
- package/dist/resources/extensions/gsd/tools/complete-milestone.js +17 -12
- package/dist/resources/extensions/gsd/tools/complete-slice.js +40 -26
- package/dist/resources/extensions/gsd/tools/complete-task.js +12 -12
- package/dist/resources/extensions/gsd/tools/plan-milestone.js +33 -25
- package/dist/resources/extensions/gsd/tools/plan-slice.js +5 -8
- package/dist/resources/extensions/gsd/workflow-projections.js +21 -5
- package/dist/resources/extensions/gsd/worktree-manager.js +82 -29
- package/dist/resources/extensions/gsd/worktree-resolver.js +4 -3
- package/dist/resources/extensions/mcp-client/auth.js +101 -0
- package/dist/resources/extensions/mcp-client/index.js +10 -1
- package/dist/resources/extensions/ollama/index.js +28 -22
- package/dist/resources/extensions/ollama/model-capabilities.js +37 -34
- package/dist/resources/extensions/ollama/ndjson-stream.js +54 -0
- package/dist/resources/extensions/ollama/ollama-chat-provider.js +380 -0
- package/dist/resources/extensions/ollama/ollama-client.js +23 -32
- package/dist/resources/extensions/ollama/ollama-discovery.js +2 -7
- package/dist/resources/extensions/ollama/ollama-tool.js +62 -0
- package/dist/resources/extensions/ollama/thinking-parser.js +104 -0
- package/dist/update-cmd.js +4 -2
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +20 -20
- package/dist/web/standalone/.next/build-manifest.json +2 -2
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/server/app/_global-error.html +2 -2
- 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 +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 +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 +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 +2 -2
- 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 +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 +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 +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 +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 +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 +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 +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 +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 +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 +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 +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 +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/onboarding/route.js +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 +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 +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 +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 +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 +2 -2
- 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 +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 +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/skill-health/route.js +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 +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 +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 +2 -2
- 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 +2 -2
- 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 +1 -1
- package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/visualizer/route.js +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 +20 -20
- package/dist/web/standalone/.next/server/chunks/6897.js +12 -0
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +2 -2
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/package.json +1 -1
- package/packages/pi-agent-core/dist/agent-loop.d.ts +8 -0
- package/packages/pi-agent-core/dist/agent-loop.d.ts.map +1 -1
- package/packages/pi-agent-core/dist/agent-loop.js +50 -0
- package/packages/pi-agent-core/dist/agent-loop.js.map +1 -1
- package/packages/pi-agent-core/src/agent-loop.test.ts +221 -5
- package/packages/pi-agent-core/src/agent-loop.ts +53 -0
- package/packages/pi-ai/dist/types.d.ts +16 -1
- package/packages/pi-ai/dist/types.d.ts.map +1 -1
- package/packages/pi-ai/dist/types.js.map +1 -1
- package/packages/pi-ai/src/types.ts +18 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +9 -0
- package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.js +50 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.test.js +41 -0
- package/packages/pi-coding-agent/dist/core/auth-storage.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +7 -0
- package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/loader.js +31 -4
- package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/loader.test.js +28 -1
- package/packages/pi-coding-agent/dist/core/extensions/loader.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/provider-registration.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/extensions/provider-registration.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/extensions/provider-registration.test.js +46 -0
- package/packages/pi-coding-agent/dist/core/extensions/provider-registration.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry.d.ts +1 -0
- package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry.js +12 -0
- package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-resolver.js +3 -3
- package/packages/pi-coding-agent/dist/core/model-resolver.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/resource-loader.d.ts +23 -1
- package/packages/pi-coding-agent/dist/core/resource-loader.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/resource-loader.js +80 -56
- package/packages/pi-coding-agent/dist/core/resource-loader.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.js +9 -0
- package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
- package/packages/pi-coding-agent/src/core/auth-storage.test.ts +53 -0
- package/packages/pi-coding-agent/src/core/auth-storage.ts +66 -1
- package/packages/pi-coding-agent/src/core/extensions/loader.test.ts +39 -1
- package/packages/pi-coding-agent/src/core/extensions/loader.ts +34 -4
- package/packages/pi-coding-agent/src/core/extensions/provider-registration.test.ts +81 -0
- package/packages/pi-coding-agent/src/core/extensions/types.ts +2 -0
- package/packages/pi-coding-agent/src/core/model-registry.ts +14 -0
- package/packages/pi-coding-agent/src/core/model-resolver.ts +3 -3
- package/packages/pi-coding-agent/src/core/resource-loader.ts +89 -56
- package/packages/pi-coding-agent/src/core/sdk.ts +10 -0
- package/src/resources/extensions/cmux/index.ts +18 -12
- package/src/resources/extensions/gsd/auto/detect-stuck.ts +27 -0
- package/src/resources/extensions/gsd/auto/finalize-timeout.ts +46 -0
- package/src/resources/extensions/gsd/auto/loop.ts +5 -0
- package/src/resources/extensions/gsd/auto/phases.ts +194 -33
- package/src/resources/extensions/gsd/auto/session.ts +14 -0
- package/src/resources/extensions/gsd/auto-dashboard.ts +11 -3
- package/src/resources/extensions/gsd/auto-model-selection.ts +36 -0
- package/src/resources/extensions/gsd/auto-post-unit.ts +141 -12
- package/src/resources/extensions/gsd/auto-prompts.ts +21 -0
- package/src/resources/extensions/gsd/auto-recovery.ts +9 -8
- package/src/resources/extensions/gsd/auto-start.ts +11 -20
- package/src/resources/extensions/gsd/auto-timers.ts +2 -1
- package/src/resources/extensions/gsd/auto-tool-tracking.ts +19 -0
- package/src/resources/extensions/gsd/auto-worktree.ts +14 -6
- package/src/resources/extensions/gsd/auto.ts +22 -1
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +160 -88
- package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +15 -0
- package/src/resources/extensions/gsd/bootstrap/query-tools.ts +98 -0
- package/src/resources/extensions/gsd/bootstrap/register-extension.ts +4 -0
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +36 -1
- package/src/resources/extensions/gsd/bootstrap/sanitize-complete-milestone.ts +57 -0
- package/src/resources/extensions/gsd/bootstrap/system-context.ts +31 -2
- package/src/resources/extensions/gsd/commands-handlers.ts +10 -4
- package/src/resources/extensions/gsd/constants.ts +44 -0
- package/src/resources/extensions/gsd/db-writer.ts +78 -4
- package/src/resources/extensions/gsd/forensics.ts +21 -5
- package/src/resources/extensions/gsd/gsd-db.ts +64 -17
- package/src/resources/extensions/gsd/guided-flow.ts +22 -0
- package/src/resources/extensions/gsd/metrics.ts +28 -1
- package/src/resources/extensions/gsd/native-git-bridge.ts +5 -3
- package/src/resources/extensions/gsd/preferences-types.ts +16 -0
- package/src/resources/extensions/gsd/preferences.ts +9 -2
- package/src/resources/extensions/gsd/prompt-loader.ts +8 -0
- package/src/resources/extensions/gsd/prompts/complete-milestone.md +2 -0
- package/src/resources/extensions/gsd/prompts/complete-slice.md +2 -0
- package/src/resources/extensions/gsd/prompts/doctor-heal.md +1 -0
- package/src/resources/extensions/gsd/prompts/forensics.md +2 -0
- package/src/resources/extensions/gsd/prompts/reassess-roadmap.md +2 -0
- package/src/resources/extensions/gsd/prompts/system.md +4 -7
- package/src/resources/extensions/gsd/prompts/validate-milestone.md +2 -0
- package/src/resources/extensions/gsd/roadmap-mutations.ts +1 -1
- package/src/resources/extensions/gsd/roadmap-slices.ts +10 -5
- package/src/resources/extensions/gsd/safety/content-validator.ts +98 -0
- package/src/resources/extensions/gsd/safety/destructive-guard.ts +49 -0
- package/src/resources/extensions/gsd/safety/evidence-collector.ts +151 -0
- package/src/resources/extensions/gsd/safety/evidence-cross-ref.ts +120 -0
- package/src/resources/extensions/gsd/safety/file-change-validator.ts +108 -0
- package/src/resources/extensions/gsd/safety/git-checkpoint.ts +106 -0
- package/src/resources/extensions/gsd/safety/safety-harness.ts +105 -0
- package/src/resources/extensions/gsd/slice-parallel-conflict.ts +86 -0
- package/src/resources/extensions/gsd/slice-parallel-eligibility.ts +73 -0
- package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +477 -0
- package/src/resources/extensions/gsd/state.ts +67 -12
- package/src/resources/extensions/gsd/status-guards.ts +13 -0
- package/src/resources/extensions/gsd/tests/artifact-corruption-2630.test.ts +288 -0
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +34 -13
- package/src/resources/extensions/gsd/tests/cmux.test.ts +58 -0
- package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +51 -0
- package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +140 -0
- package/src/resources/extensions/gsd/tests/complete-slice-string-coercion.test.ts +211 -0
- package/src/resources/extensions/gsd/tests/complete-task.test.ts +39 -0
- package/src/resources/extensions/gsd/tests/dashboard-model-label-ordering.test.ts +107 -0
- package/src/resources/extensions/gsd/tests/db-access-guardrails.test.ts +109 -0
- package/src/resources/extensions/gsd/tests/db-path-worktree-symlink.test.ts +13 -9
- package/src/resources/extensions/gsd/tests/db-writer.test.ts +134 -0
- package/src/resources/extensions/gsd/tests/deferred-slice-dispatch.test.ts +203 -0
- package/src/resources/extensions/gsd/tests/discuss-tool-scoping.test.ts +130 -0
- package/src/resources/extensions/gsd/tests/doctor-fix-flag.test.ts +92 -0
- package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +116 -0
- package/src/resources/extensions/gsd/tests/flat-rate-routing-guard.test.ts +50 -0
- package/src/resources/extensions/gsd/tests/forensics-stuck-loops.test.ts +103 -0
- package/src/resources/extensions/gsd/tests/git-checkpoint.test.ts +94 -0
- package/src/resources/extensions/gsd/tests/insert-slice-no-wipe.test.ts +88 -0
- package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +27 -7
- package/src/resources/extensions/gsd/tests/integration/idle-recovery.test.ts +34 -0
- package/src/resources/extensions/gsd/tests/metrics.test.ts +116 -1
- package/src/resources/extensions/gsd/tests/milestone-status-tool.test.ts +201 -0
- package/src/resources/extensions/gsd/tests/plan-milestone-title.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +82 -18
- package/src/resources/extensions/gsd/tests/preferences.test.ts +10 -0
- package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +25 -0
- package/src/resources/extensions/gsd/tests/roadmap-slices.test.ts +69 -0
- package/src/resources/extensions/gsd/tests/shared-wal.test.ts +30 -0
- package/src/resources/extensions/gsd/tests/slice-context-injection.test.ts +50 -0
- package/src/resources/extensions/gsd/tests/slice-parallel-conflict.test.ts +92 -0
- package/src/resources/extensions/gsd/tests/slice-parallel-eligibility.test.ts +95 -0
- package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +83 -0
- package/src/resources/extensions/gsd/tests/stuck-detection-coverage.test.ts +42 -0
- package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +103 -0
- package/src/resources/extensions/gsd/tests/tool-param-optionality.test.ts +349 -0
- package/src/resources/extensions/gsd/tests/worktree-health-dispatch.test.ts +35 -2
- package/src/resources/extensions/gsd/tests/worktree-health-monorepo.test.ts +73 -0
- package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +34 -0
- package/src/resources/extensions/gsd/tests/worktree-submodule-safety.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/worktree-teardown-safety.test.ts +148 -0
- package/src/resources/extensions/gsd/tools/complete-milestone.ts +34 -20
- package/src/resources/extensions/gsd/tools/complete-slice.ts +41 -26
- package/src/resources/extensions/gsd/tools/complete-task.ts +12 -12
- package/src/resources/extensions/gsd/tools/plan-milestone.ts +55 -30
- package/src/resources/extensions/gsd/tools/plan-slice.ts +13 -8
- package/src/resources/extensions/gsd/types.ts +44 -22
- package/src/resources/extensions/gsd/workflow-logger.ts +2 -1
- package/src/resources/extensions/gsd/workflow-projections.ts +23 -5
- package/src/resources/extensions/gsd/worktree-manager.ts +76 -28
- package/src/resources/extensions/gsd/worktree-resolver.ts +4 -3
- package/src/resources/extensions/mcp-client/auth.ts +149 -0
- package/src/resources/extensions/mcp-client/index.ts +16 -1
- package/src/resources/extensions/ollama/index.ts +26 -25
- package/src/resources/extensions/ollama/model-capabilities.ts +41 -34
- package/src/resources/extensions/ollama/ndjson-stream.ts +63 -0
- package/src/resources/extensions/ollama/ollama-auth-mode.test.ts +20 -0
- package/src/resources/extensions/ollama/ollama-chat-provider.ts +459 -0
- package/src/resources/extensions/ollama/ollama-client.ts +30 -30
- package/src/resources/extensions/ollama/ollama-discovery.ts +5 -8
- package/src/resources/extensions/ollama/ollama-tool.ts +69 -0
- package/src/resources/extensions/ollama/tests/ollama-chat-provider-stream.test.ts +82 -0
- package/src/resources/extensions/ollama/tests/ollama-discovery.test.ts +0 -27
- package/src/resources/extensions/ollama/thinking-parser.ts +116 -0
- package/src/resources/extensions/ollama/types.ts +23 -0
- package/dist/web/standalone/.next/server/chunks/2229.js +0 -12
- /package/dist/web/standalone/.next/static/{TTlAguZQ5vR9EOv6G8cel → SDB1T-4NqkMjYirjjqQhr}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{TTlAguZQ5vR9EOv6G8cel → SDB1T-4NqkMjYirjjqQhr}/_ssgManifest.js +0 -0
|
@@ -24,6 +24,7 @@ import {
|
|
|
24
24
|
saveDecisionToDb,
|
|
25
25
|
updateRequirementInDb,
|
|
26
26
|
saveArtifactToDb,
|
|
27
|
+
extractDeferredSliceRef,
|
|
27
28
|
} from '../db-writer.ts';
|
|
28
29
|
import type { Decision, Requirement } from '../types.ts';
|
|
29
30
|
|
|
@@ -475,6 +476,71 @@ describe('db-writer', () => {
|
|
|
475
476
|
}
|
|
476
477
|
});
|
|
477
478
|
|
|
479
|
+
test('updateRequirementInDb — seeds from REQUIREMENTS.md when DB empty (#3346)', async () => {
|
|
480
|
+
const tmpDir = makeTmpDir();
|
|
481
|
+
const dbPath = path.join(tmpDir, '.gsd', 'gsd.db');
|
|
482
|
+
openDatabase(dbPath);
|
|
483
|
+
|
|
484
|
+
try {
|
|
485
|
+
// Write a REQUIREMENTS.md with real content (simulating discussion phase output)
|
|
486
|
+
const reqContent = [
|
|
487
|
+
'# Requirements',
|
|
488
|
+
'',
|
|
489
|
+
'## Active',
|
|
490
|
+
'',
|
|
491
|
+
'### R005 — User authentication',
|
|
492
|
+
'- Class: functional',
|
|
493
|
+
'- Why: Users need secure access',
|
|
494
|
+
'- Source: user-research',
|
|
495
|
+
'- Primary owner: M001/S02',
|
|
496
|
+
'',
|
|
497
|
+
'### R007 — API rate limiting',
|
|
498
|
+
'- Class: non-functional',
|
|
499
|
+
'- Why: Prevent abuse',
|
|
500
|
+
'- Source: architecture',
|
|
501
|
+
'- Primary owner: M001/S03',
|
|
502
|
+
'',
|
|
503
|
+
'## Validated',
|
|
504
|
+
'',
|
|
505
|
+
'### R001 — Database schema',
|
|
506
|
+
'- Class: functional',
|
|
507
|
+
'- Why: Foundation for storage',
|
|
508
|
+
'- Source: design',
|
|
509
|
+
'- Validation: S01 verified',
|
|
510
|
+
].join('\n');
|
|
511
|
+
fs.writeFileSync(path.join(tmpDir, '.gsd', 'REQUIREMENTS.md'), reqContent);
|
|
512
|
+
|
|
513
|
+
// DB is empty — no requirements seeded. Update R005 to "validated".
|
|
514
|
+
// Before #3346 fix: this would create a skeleton with empty fields.
|
|
515
|
+
// After fix: this seeds all 3 requirements from REQUIREMENTS.md first.
|
|
516
|
+
await updateRequirementInDb('R005', {
|
|
517
|
+
status: 'validated',
|
|
518
|
+
validation: 'S02 — auth flow verified',
|
|
519
|
+
}, tmpDir);
|
|
520
|
+
|
|
521
|
+
// R005 should have the update AND the original content from markdown
|
|
522
|
+
const r005 = getRequirementById('R005');
|
|
523
|
+
assert.ok(r005, 'R005 should exist');
|
|
524
|
+
assert.equal(r005!.status, 'validated', 'status should be updated');
|
|
525
|
+
assert.equal(r005!.validation, 'S02 — auth flow verified', 'validation should be updated');
|
|
526
|
+
assert.equal(r005!.class, 'functional', 'class should be preserved from REQUIREMENTS.md');
|
|
527
|
+
assert.ok(r005!.description?.includes('authentication') || r005!.full_content?.includes('authentication'),
|
|
528
|
+
'original content should be preserved');
|
|
529
|
+
|
|
530
|
+
// R007 and R001 should also be seeded (not just the one being updated)
|
|
531
|
+
const r007 = getRequirementById('R007');
|
|
532
|
+
assert.ok(r007, 'R007 should be seeded from REQUIREMENTS.md');
|
|
533
|
+
assert.equal(r007!.status, 'active', 'R007 status should be active');
|
|
534
|
+
|
|
535
|
+
const r001 = getRequirementById('R001');
|
|
536
|
+
assert.ok(r001, 'R001 should be seeded from REQUIREMENTS.md');
|
|
537
|
+
assert.equal(r001!.status, 'validated', 'R001 status should be validated (from section heading)');
|
|
538
|
+
} finally {
|
|
539
|
+
closeDatabase();
|
|
540
|
+
cleanupDir(tmpDir);
|
|
541
|
+
}
|
|
542
|
+
});
|
|
543
|
+
|
|
478
544
|
// ═══════════════════════════════════════════════════════════════════════════
|
|
479
545
|
// saveArtifactToDb Tests
|
|
480
546
|
// ═══════════════════════════════════════════════════════════════════════════
|
|
@@ -694,4 +760,72 @@ describe('db-writer', () => {
|
|
|
694
760
|
|
|
695
761
|
// ═══════════════════════════════════════════════════════════════════════════
|
|
696
762
|
|
|
763
|
+
// ═══════════════════════════════════════════════════════════════════════════
|
|
764
|
+
// extractDeferredSliceRef
|
|
765
|
+
// ═══════════════════════════════════════════════════════════════════════════
|
|
766
|
+
|
|
767
|
+
describe('extractDeferredSliceRef', () => {
|
|
768
|
+
const fields = (scope: string, choice: string, decision: string) => ({
|
|
769
|
+
scope,
|
|
770
|
+
choice,
|
|
771
|
+
decision,
|
|
772
|
+
});
|
|
773
|
+
|
|
774
|
+
test('detects deferral in scope with M###/S## pattern in choice', () => {
|
|
775
|
+
const result = extractDeferredSliceRef(
|
|
776
|
+
fields('deferral of low-priority work', 'Move M001/S03 to backlog', ''),
|
|
777
|
+
);
|
|
778
|
+
assert.deepStrictEqual(result, { milestoneId: 'M001', sliceId: 'S03' });
|
|
779
|
+
});
|
|
780
|
+
|
|
781
|
+
test('detects deferral in choice field', () => {
|
|
782
|
+
const result = extractDeferredSliceRef(
|
|
783
|
+
fields('slice prioritization', 'defer M002/S01 until next sprint', ''),
|
|
784
|
+
);
|
|
785
|
+
assert.deepStrictEqual(result, { milestoneId: 'M002', sliceId: 'S01' });
|
|
786
|
+
});
|
|
787
|
+
|
|
788
|
+
test('detects deferral in decision field', () => {
|
|
789
|
+
const result = extractDeferredSliceRef(
|
|
790
|
+
fields('resource constraints', '', 'deferred M010/S12 pending review'),
|
|
791
|
+
);
|
|
792
|
+
assert.deepStrictEqual(result, { milestoneId: 'M010', sliceId: 'S12' });
|
|
793
|
+
});
|
|
794
|
+
|
|
795
|
+
test('returns null when no M###/S## pattern is present', () => {
|
|
796
|
+
const result = extractDeferredSliceRef(
|
|
797
|
+
fields('deferral of work', 'will revisit later', 'deferred indefinitely'),
|
|
798
|
+
);
|
|
799
|
+
assert.strictEqual(result, null);
|
|
800
|
+
});
|
|
801
|
+
|
|
802
|
+
test('recognises "deferring" variant', () => {
|
|
803
|
+
const result = extractDeferredSliceRef(
|
|
804
|
+
fields('deferring this slice', 'M005/S02 can wait', ''),
|
|
805
|
+
);
|
|
806
|
+
assert.deepStrictEqual(result, { milestoneId: 'M005', sliceId: 'S02' });
|
|
807
|
+
});
|
|
808
|
+
|
|
809
|
+
test('recognises "defers" variant', () => {
|
|
810
|
+
const result = extractDeferredSliceRef(
|
|
811
|
+
fields('team defers slice', 'M100/S10 not urgent', ''),
|
|
812
|
+
);
|
|
813
|
+
assert.deepStrictEqual(result, { milestoneId: 'M100', sliceId: 'S10' });
|
|
814
|
+
});
|
|
815
|
+
|
|
816
|
+
test('returns first M###/S## match when multiple patterns exist', () => {
|
|
817
|
+
const result = extractDeferredSliceRef(
|
|
818
|
+
fields('', 'defer M003/S01 and M003/S02', ''),
|
|
819
|
+
);
|
|
820
|
+
assert.deepStrictEqual(result, { milestoneId: 'M003', sliceId: 'S01' });
|
|
821
|
+
});
|
|
822
|
+
|
|
823
|
+
test('returns null when no deferral keyword is present', () => {
|
|
824
|
+
const result = extractDeferredSliceRef(
|
|
825
|
+
fields('approved work', 'M001/S01 is ready', 'proceed with M001/S01'),
|
|
826
|
+
);
|
|
827
|
+
assert.strictEqual(result, null);
|
|
828
|
+
});
|
|
829
|
+
});
|
|
830
|
+
|
|
697
831
|
});
|
|
@@ -0,0 +1,203 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Regression test for #2661: Auto-mode dispatches deferred slices.
|
|
3
|
+
*
|
|
4
|
+
* When a decision defers a slice, the dispatcher must skip it and advance
|
|
5
|
+
* to the next eligible slice. This tests both:
|
|
6
|
+
* 1. deriveStateFromDb skips slices with status "deferred"
|
|
7
|
+
* 2. saveDecisionToDb updates the slice status when the decision is a deferral
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
import { describe, test } from "node:test";
|
|
11
|
+
import assert from "node:assert/strict";
|
|
12
|
+
import { mkdtempSync, mkdirSync, rmSync, writeFileSync } from "node:fs";
|
|
13
|
+
import { join } from "node:path";
|
|
14
|
+
import { tmpdir } from "node:os";
|
|
15
|
+
|
|
16
|
+
import { deriveStateFromDb, invalidateStateCache } from "../state.ts";
|
|
17
|
+
import {
|
|
18
|
+
openDatabase,
|
|
19
|
+
closeDatabase,
|
|
20
|
+
isDbAvailable,
|
|
21
|
+
insertMilestone,
|
|
22
|
+
insertSlice,
|
|
23
|
+
insertTask,
|
|
24
|
+
insertArtifact,
|
|
25
|
+
updateSliceStatus,
|
|
26
|
+
} from "../gsd-db.ts";
|
|
27
|
+
import { isDeferredStatus } from "../status-guards.ts";
|
|
28
|
+
|
|
29
|
+
// ─── Helpers ──────────────────────────────────────────────────────────────
|
|
30
|
+
|
|
31
|
+
function createFixtureBase(): string {
|
|
32
|
+
const base = mkdtempSync(join(tmpdir(), "gsd-deferred-dispatch-"));
|
|
33
|
+
mkdirSync(join(base, ".gsd", "milestones"), { recursive: true });
|
|
34
|
+
return base;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
function writeFile(base: string, relativePath: string, content: string): void {
|
|
38
|
+
const full = join(base, ".gsd", relativePath);
|
|
39
|
+
mkdirSync(join(full, ".."), { recursive: true });
|
|
40
|
+
writeFileSync(full, content);
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
function cleanup(base: string): void {
|
|
44
|
+
rmSync(base, { recursive: true, force: true });
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
// ─── Tests ────────────────────────────────────────────────────────────────
|
|
48
|
+
|
|
49
|
+
describe("deferred-slice-dispatch (#2661)", () => {
|
|
50
|
+
test("isDeferredStatus returns true for 'deferred'", () => {
|
|
51
|
+
assert.ok(isDeferredStatus("deferred"), "should recognize 'deferred'");
|
|
52
|
+
assert.ok(!isDeferredStatus("active"), "should not match 'active'");
|
|
53
|
+
assert.ok(!isDeferredStatus("complete"), "should not match 'complete'");
|
|
54
|
+
assert.ok(!isDeferredStatus("pending"), "should not match 'pending'");
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
test("deriveStateFromDb skips deferred slice and picks next eligible", async () => {
|
|
58
|
+
const base = createFixtureBase();
|
|
59
|
+
try {
|
|
60
|
+
openDatabase(":memory:");
|
|
61
|
+
assert.ok(isDbAvailable());
|
|
62
|
+
|
|
63
|
+
// M001 with three slices: S01 complete, S02 deferred, S03 pending
|
|
64
|
+
insertMilestone({ id: "M001", title: "Test Milestone", status: "active" });
|
|
65
|
+
|
|
66
|
+
insertSlice({ id: "S01", milestoneId: "M001", title: "Done Slice", status: "complete", risk: "low", depends: [] });
|
|
67
|
+
insertSlice({ id: "S02", milestoneId: "M001", title: "Deferred Slice", status: "deferred", risk: "low", depends: [] });
|
|
68
|
+
insertSlice({ id: "S03", milestoneId: "M001", title: "Next Slice", status: "pending", risk: "low", depends: [] });
|
|
69
|
+
|
|
70
|
+
// S01 needs a SUMMARY file to count as complete for milestone-level checks
|
|
71
|
+
writeFile(base, "milestones/M001/M001-ROADMAP.md", `# M001: Test Milestone
|
|
72
|
+
|
|
73
|
+
**Vision:** Test deferred slices.
|
|
74
|
+
|
|
75
|
+
## Slices
|
|
76
|
+
|
|
77
|
+
- [x] **S01: Done Slice** \`risk:low\` \`depends:[]\`
|
|
78
|
+
> Done.
|
|
79
|
+
|
|
80
|
+
- [ ] **S02: Deferred Slice** \`risk:low\` \`depends:[]\`
|
|
81
|
+
> Deferred.
|
|
82
|
+
|
|
83
|
+
- [ ] **S03: Next Slice** \`risk:low\` \`depends:[]\`
|
|
84
|
+
> Next.
|
|
85
|
+
`);
|
|
86
|
+
writeFile(base, "milestones/M001/slices/S01/S01-SUMMARY.md", "# S01 Summary\nDone.");
|
|
87
|
+
|
|
88
|
+
invalidateStateCache();
|
|
89
|
+
const state = await deriveStateFromDb(base);
|
|
90
|
+
|
|
91
|
+
// The active slice must be S03, NOT S02 (which is deferred)
|
|
92
|
+
assert.equal(state.activeMilestone?.id, "M001", "active milestone is M001");
|
|
93
|
+
assert.equal(state.activeSlice?.id, "S03", "active slice should skip deferred S02 and land on S03");
|
|
94
|
+
assert.notEqual(state.activeSlice?.id, "S02", "active slice must NOT be the deferred S02");
|
|
95
|
+
|
|
96
|
+
closeDatabase();
|
|
97
|
+
} finally {
|
|
98
|
+
closeDatabase();
|
|
99
|
+
cleanup(base);
|
|
100
|
+
}
|
|
101
|
+
});
|
|
102
|
+
|
|
103
|
+
test("deriveStateFromDb does not count deferred slices as done for progress", async () => {
|
|
104
|
+
const base = createFixtureBase();
|
|
105
|
+
try {
|
|
106
|
+
openDatabase(":memory:");
|
|
107
|
+
|
|
108
|
+
insertMilestone({ id: "M001", title: "Test", status: "active" });
|
|
109
|
+
insertSlice({ id: "S01", milestoneId: "M001", title: "Complete", status: "complete", risk: "low", depends: [] });
|
|
110
|
+
insertSlice({ id: "S02", milestoneId: "M001", title: "Deferred", status: "deferred", risk: "low", depends: [] });
|
|
111
|
+
insertSlice({ id: "S03", milestoneId: "M001", title: "Pending", status: "pending", risk: "low", depends: [] });
|
|
112
|
+
|
|
113
|
+
writeFile(base, "milestones/M001/M001-ROADMAP.md", `# M001
|
|
114
|
+
## Slices
|
|
115
|
+
- [x] **S01: Complete** \`risk:low\` \`depends:[]\`
|
|
116
|
+
- [ ] **S02: Deferred** \`risk:low\` \`depends:[]\`
|
|
117
|
+
- [ ] **S03: Pending** \`risk:low\` \`depends:[]\`
|
|
118
|
+
`);
|
|
119
|
+
writeFile(base, "milestones/M001/slices/S01/S01-SUMMARY.md", "# Done");
|
|
120
|
+
|
|
121
|
+
invalidateStateCache();
|
|
122
|
+
const state = await deriveStateFromDb(base);
|
|
123
|
+
|
|
124
|
+
// Deferred slices should not count as "done" in progress
|
|
125
|
+
// Only S01 (complete) counts as done
|
|
126
|
+
assert.equal(state.progress?.slices?.done, 1, "only 1 slice (S01) should be done");
|
|
127
|
+
// Total should still be 3 (deferred slices are still part of the milestone)
|
|
128
|
+
assert.equal(state.progress?.slices?.total, 3, "all 3 slices counted in total");
|
|
129
|
+
|
|
130
|
+
closeDatabase();
|
|
131
|
+
} finally {
|
|
132
|
+
closeDatabase();
|
|
133
|
+
cleanup(base);
|
|
134
|
+
}
|
|
135
|
+
});
|
|
136
|
+
|
|
137
|
+
test("all slices deferred results in blocked state", async () => {
|
|
138
|
+
const base = createFixtureBase();
|
|
139
|
+
try {
|
|
140
|
+
openDatabase(":memory:");
|
|
141
|
+
|
|
142
|
+
insertMilestone({ id: "M001", title: "Test", status: "active" });
|
|
143
|
+
insertSlice({ id: "S01", milestoneId: "M001", title: "Deferred A", status: "deferred", risk: "low", depends: [] });
|
|
144
|
+
insertSlice({ id: "S02", milestoneId: "M001", title: "Deferred B", status: "deferred", risk: "low", depends: [] });
|
|
145
|
+
|
|
146
|
+
writeFile(base, "milestones/M001/M001-ROADMAP.md", `# M001
|
|
147
|
+
## Slices
|
|
148
|
+
- [ ] **S01: Deferred A** \`risk:low\` \`depends:[]\`
|
|
149
|
+
- [ ] **S02: Deferred B** \`risk:low\` \`depends:[]\`
|
|
150
|
+
`);
|
|
151
|
+
|
|
152
|
+
invalidateStateCache();
|
|
153
|
+
const state = await deriveStateFromDb(base);
|
|
154
|
+
|
|
155
|
+
// No eligible slice — should be blocked
|
|
156
|
+
assert.equal(state.activeSlice, null, "no active slice when all deferred");
|
|
157
|
+
assert.equal(state.phase, "blocked", "phase should be blocked when all slices deferred");
|
|
158
|
+
|
|
159
|
+
closeDatabase();
|
|
160
|
+
} finally {
|
|
161
|
+
closeDatabase();
|
|
162
|
+
cleanup(base);
|
|
163
|
+
}
|
|
164
|
+
});
|
|
165
|
+
|
|
166
|
+
test("saveDecisionToDb marks slice as deferred when decision is a deferral", async () => {
|
|
167
|
+
const base = createFixtureBase();
|
|
168
|
+
try {
|
|
169
|
+
openDatabase(":memory:");
|
|
170
|
+
|
|
171
|
+
insertMilestone({ id: "M001", title: "Test", status: "active" });
|
|
172
|
+
insertSlice({ id: "S03", milestoneId: "M001", title: "Target Slice", status: "active", risk: "low", depends: [] });
|
|
173
|
+
|
|
174
|
+
writeFile(base, "milestones/M001/M001-ROADMAP.md", `# M001
|
|
175
|
+
## Slices
|
|
176
|
+
- [ ] **S03: Target Slice** \`risk:low\` \`depends:[]\`
|
|
177
|
+
`);
|
|
178
|
+
|
|
179
|
+
const { saveDecisionToDb } = await import("../db-writer.ts");
|
|
180
|
+
const { getSlice } = await import("../gsd-db.ts");
|
|
181
|
+
|
|
182
|
+
// Save a deferral decision that references M001/S03
|
|
183
|
+
await saveDecisionToDb(
|
|
184
|
+
{
|
|
185
|
+
scope: "deferral",
|
|
186
|
+
decision: "Defer S03 to focus on higher priority work",
|
|
187
|
+
choice: "defer M001/S03",
|
|
188
|
+
rationale: "Not ready yet",
|
|
189
|
+
},
|
|
190
|
+
base,
|
|
191
|
+
);
|
|
192
|
+
|
|
193
|
+
// The slice status should now be "deferred"
|
|
194
|
+
const slice = getSlice("M001", "S03");
|
|
195
|
+
assert.equal(slice?.status, "deferred", "slice status should be updated to 'deferred' after deferral decision");
|
|
196
|
+
|
|
197
|
+
closeDatabase();
|
|
198
|
+
} finally {
|
|
199
|
+
closeDatabase();
|
|
200
|
+
cleanup(base);
|
|
201
|
+
}
|
|
202
|
+
});
|
|
203
|
+
});
|
|
@@ -0,0 +1,130 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* discuss-tool-scoping.test.ts — Tests for #2949.
|
|
3
|
+
*
|
|
4
|
+
* xAI/Grok returns "Grammar is too complex" (400) when the combined tool
|
|
5
|
+
* schemas exceed the provider's grammar limit. The GSD discuss flow only
|
|
6
|
+
* needs a small subset of tools (summary_save, decision_save, etc.), but
|
|
7
|
+
* was sending ALL ~30+ tools to the provider.
|
|
8
|
+
*
|
|
9
|
+
* These tests verify:
|
|
10
|
+
* 1. DISCUSS_TOOLS_ALLOWLIST is exported and contains only the tools
|
|
11
|
+
* needed during discuss flows (no heavy planning/execution/completion tools).
|
|
12
|
+
* 2. Heavy execution tools are NOT in the allowlist.
|
|
13
|
+
* 3. The allowlist includes the tools actually referenced by discuss prompts.
|
|
14
|
+
* 4. dispatchWorkflow scopes tools when unitType is a discuss variant.
|
|
15
|
+
*/
|
|
16
|
+
|
|
17
|
+
import { describe, test } from "node:test";
|
|
18
|
+
import assert from "node:assert/strict";
|
|
19
|
+
import { readFileSync } from "node:fs";
|
|
20
|
+
import { join, dirname } from "node:path";
|
|
21
|
+
import { fileURLToPath } from "node:url";
|
|
22
|
+
|
|
23
|
+
import { DISCUSS_TOOLS_ALLOWLIST } from "../constants.ts";
|
|
24
|
+
|
|
25
|
+
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
26
|
+
const promptsDir = join(__dirname, "..", "prompts");
|
|
27
|
+
const guidedFlowPath = join(__dirname, "..", "guided-flow.ts");
|
|
28
|
+
|
|
29
|
+
// ─── Heavy tools that should NOT be in discuss scope ─────────────────────────
|
|
30
|
+
|
|
31
|
+
/** Tools that are only needed during planning, execution, or completion phases */
|
|
32
|
+
const HEAVY_TOOLS = [
|
|
33
|
+
"gsd_plan_slice",
|
|
34
|
+
"gsd_slice_plan",
|
|
35
|
+
"gsd_plan_task",
|
|
36
|
+
"gsd_task_plan",
|
|
37
|
+
"gsd_task_complete",
|
|
38
|
+
"gsd_complete_task",
|
|
39
|
+
"gsd_slice_complete",
|
|
40
|
+
"gsd_complete_slice",
|
|
41
|
+
"gsd_complete_milestone",
|
|
42
|
+
"gsd_milestone_complete",
|
|
43
|
+
"gsd_validate_milestone",
|
|
44
|
+
"gsd_milestone_validate",
|
|
45
|
+
"gsd_replan_slice",
|
|
46
|
+
"gsd_slice_replan",
|
|
47
|
+
"gsd_reassess_roadmap",
|
|
48
|
+
"gsd_roadmap_reassess",
|
|
49
|
+
"gsd_save_gate_result",
|
|
50
|
+
];
|
|
51
|
+
|
|
52
|
+
// ─── Tools that discuss prompts reference ────────────────────────────────────
|
|
53
|
+
|
|
54
|
+
/** Tools explicitly called by discuss prompt templates */
|
|
55
|
+
const DISCUSS_REQUIRED_TOOLS = [
|
|
56
|
+
"gsd_summary_save", // guided-discuss-slice.md, guided-discuss-milestone.md, discuss.md
|
|
57
|
+
"gsd_decision_save", // discuss.md output phase
|
|
58
|
+
"gsd_plan_milestone", // discuss.md output phase (single + multi milestone)
|
|
59
|
+
"gsd_milestone_generate_id", // discuss.md multi-milestone Phase 1
|
|
60
|
+
"gsd_requirement_update", // used during discuss for requirement updates
|
|
61
|
+
];
|
|
62
|
+
|
|
63
|
+
// ─── Tests ───────────────────────────────────────────────────────────────────
|
|
64
|
+
|
|
65
|
+
describe("discuss tool scoping (#2949)", () => {
|
|
66
|
+
test("DISCUSS_TOOLS_ALLOWLIST is exported and non-empty", () => {
|
|
67
|
+
assert.ok(Array.isArray(DISCUSS_TOOLS_ALLOWLIST), "should be an array");
|
|
68
|
+
assert.ok(DISCUSS_TOOLS_ALLOWLIST.length > 0, "should not be empty");
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
test("DISCUSS_TOOLS_ALLOWLIST excludes heavy execution/completion tools", () => {
|
|
72
|
+
for (const heavy of HEAVY_TOOLS) {
|
|
73
|
+
assert.ok(
|
|
74
|
+
!DISCUSS_TOOLS_ALLOWLIST.includes(heavy),
|
|
75
|
+
`allowlist should NOT include heavy tool "${heavy}"`,
|
|
76
|
+
);
|
|
77
|
+
}
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
test("DISCUSS_TOOLS_ALLOWLIST includes tools referenced by discuss prompts", () => {
|
|
81
|
+
for (const required of DISCUSS_REQUIRED_TOOLS) {
|
|
82
|
+
assert.ok(
|
|
83
|
+
DISCUSS_TOOLS_ALLOWLIST.includes(required),
|
|
84
|
+
`allowlist should include "${required}" (used by discuss prompts)`,
|
|
85
|
+
);
|
|
86
|
+
}
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
test("DISCUSS_TOOLS_ALLOWLIST is significantly smaller than full tool set", () => {
|
|
90
|
+
// Full set is 27 DB tools + dynamic + journal = 33+
|
|
91
|
+
// Discuss set should be roughly 10 GSD tools (5 canonical + 5 aliases)
|
|
92
|
+
assert.ok(
|
|
93
|
+
DISCUSS_TOOLS_ALLOWLIST.length <= 12,
|
|
94
|
+
`allowlist should have at most 12 GSD tools, got ${DISCUSS_TOOLS_ALLOWLIST.length}`,
|
|
95
|
+
);
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
test("guided-discuss-slice.md references gsd_summary_save", () => {
|
|
99
|
+
const prompt = readFileSync(join(promptsDir, "guided-discuss-slice.md"), "utf-8");
|
|
100
|
+
assert.ok(
|
|
101
|
+
prompt.includes("gsd_summary_save"),
|
|
102
|
+
"guided-discuss-slice.md should reference gsd_summary_save",
|
|
103
|
+
);
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
test("discuss.md references gsd_plan_milestone and gsd_decision_save", () => {
|
|
107
|
+
const prompt = readFileSync(join(promptsDir, "discuss.md"), "utf-8");
|
|
108
|
+
assert.ok(
|
|
109
|
+
prompt.includes("gsd_plan_milestone"),
|
|
110
|
+
"discuss.md should reference gsd_plan_milestone",
|
|
111
|
+
);
|
|
112
|
+
assert.ok(
|
|
113
|
+
prompt.includes("gsd_decision_save"),
|
|
114
|
+
"discuss.md should reference gsd_decision_save",
|
|
115
|
+
);
|
|
116
|
+
});
|
|
117
|
+
|
|
118
|
+
test("dispatchWorkflow source code scopes tools for discuss unit types", () => {
|
|
119
|
+
const source = readFileSync(guidedFlowPath, "utf-8");
|
|
120
|
+
// Verify that dispatchWorkflow references the allowlist for tool scoping
|
|
121
|
+
assert.ok(
|
|
122
|
+
source.includes("DISCUSS_TOOLS_ALLOWLIST"),
|
|
123
|
+
"guided-flow.ts should reference DISCUSS_TOOLS_ALLOWLIST for tool scoping",
|
|
124
|
+
);
|
|
125
|
+
assert.ok(
|
|
126
|
+
source.includes("setActiveTools"),
|
|
127
|
+
"guided-flow.ts should call setActiveTools to scope tools during discuss",
|
|
128
|
+
);
|
|
129
|
+
});
|
|
130
|
+
});
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Regression test for #1919: --fix flag not stripped before positional parse.
|
|
3
|
+
*
|
|
4
|
+
* parseDoctorArgs("--fix") must:
|
|
5
|
+
* 1. Set fixFlag = true
|
|
6
|
+
* 2. Not leak "--fix" into requestedScope
|
|
7
|
+
* 3. Keep mode as "doctor" (the flag is not a positional subcommand)
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
import { parseDoctorArgs } from "../commands-handlers.js";
|
|
11
|
+
import { createTestContext } from "./test-helpers.ts";
|
|
12
|
+
|
|
13
|
+
const { assertEq, assertTrue, report } = createTestContext();
|
|
14
|
+
|
|
15
|
+
async function main(): Promise<void> {
|
|
16
|
+
// ── 1. Bare --fix flag ──────────────────────────────────────────────────────
|
|
17
|
+
console.log("\n=== bare --fix flag (#1919) ===");
|
|
18
|
+
{
|
|
19
|
+
const r = parseDoctorArgs("--fix");
|
|
20
|
+
assertTrue(r.fixFlag, "--fix sets fixFlag to true");
|
|
21
|
+
assertEq(r.mode, "doctor", "--fix does not change mode from doctor");
|
|
22
|
+
assertEq(r.requestedScope, undefined, "--fix is stripped and does not become requestedScope");
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
// ── 2. --fix with a scope ──────────────────────────────────────────────────
|
|
26
|
+
console.log("\n=== --fix with scope ===");
|
|
27
|
+
{
|
|
28
|
+
const r = parseDoctorArgs("--fix M001/S01");
|
|
29
|
+
assertTrue(r.fixFlag, "--fix M001/S01 sets fixFlag to true");
|
|
30
|
+
assertEq(r.mode, "doctor", "--fix M001/S01 keeps mode as doctor");
|
|
31
|
+
assertEq(r.requestedScope, "M001/S01", "scope is M001/S01 after stripping --fix");
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
// ── 3. Positional fix still works ──────────────────────────────────────────
|
|
35
|
+
console.log("\n=== positional fix subcommand ===");
|
|
36
|
+
{
|
|
37
|
+
const r = parseDoctorArgs("fix");
|
|
38
|
+
assertEq(r.fixFlag, false, "positional fix does not set fixFlag");
|
|
39
|
+
assertEq(r.mode, "fix", "positional fix sets mode to fix");
|
|
40
|
+
assertEq(r.requestedScope, undefined, "no scope with bare positional fix");
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
// ── 4. Positional fix with scope ───────────────────────────────────────────
|
|
44
|
+
console.log("\n=== positional fix with scope ===");
|
|
45
|
+
{
|
|
46
|
+
const r = parseDoctorArgs("fix M001");
|
|
47
|
+
assertEq(r.mode, "fix", "fix M001 sets mode to fix");
|
|
48
|
+
assertEq(r.requestedScope, "M001", "fix M001 parses scope as M001");
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
// ── 5. --fix combined with other flags ─────────────────────────────────────
|
|
52
|
+
console.log("\n=== --fix combined with --dry-run ===");
|
|
53
|
+
{
|
|
54
|
+
const r = parseDoctorArgs("--fix --dry-run");
|
|
55
|
+
assertTrue(r.fixFlag, "--fix --dry-run sets fixFlag");
|
|
56
|
+
assertTrue(r.dryRun, "--fix --dry-run sets dryRun");
|
|
57
|
+
assertEq(r.requestedScope, undefined, "no scope leaked from combined flags");
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
// ── 6. --fix combined with --json ──────────────────────────────────────────
|
|
61
|
+
console.log("\n=== --fix with --json ===");
|
|
62
|
+
{
|
|
63
|
+
const r = parseDoctorArgs("--fix --json");
|
|
64
|
+
assertTrue(r.fixFlag, "--fix --json sets fixFlag");
|
|
65
|
+
assertTrue(r.jsonMode, "--fix --json sets jsonMode");
|
|
66
|
+
assertEq(r.requestedScope, undefined, "no scope leaked from --fix --json");
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// ── 7. Empty args (baseline) ───────────────────────────────────────────────
|
|
70
|
+
console.log("\n=== empty args baseline ===");
|
|
71
|
+
{
|
|
72
|
+
const r = parseDoctorArgs("");
|
|
73
|
+
assertEq(r.fixFlag, false, "empty args: fixFlag false");
|
|
74
|
+
assertEq(r.mode, "doctor", "empty args: mode is doctor");
|
|
75
|
+
assertEq(r.requestedScope, undefined, "empty args: no scope");
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
// ── 8. heal and audit modes unaffected ─────────────────────────────────────
|
|
79
|
+
console.log("\n=== heal and audit modes ===");
|
|
80
|
+
{
|
|
81
|
+
const rh = parseDoctorArgs("heal M001/S01");
|
|
82
|
+
assertEq(rh.mode, "heal", "heal mode parsed correctly");
|
|
83
|
+
assertEq(rh.requestedScope, "M001/S01", "heal scope parsed correctly");
|
|
84
|
+
|
|
85
|
+
const ra = parseDoctorArgs("audit");
|
|
86
|
+
assertEq(ra.mode, "audit", "audit mode parsed correctly");
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
report();
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
main();
|