gsd-pi 2.44.0 → 2.45.0
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 +30 -12
- package/dist/resources/extensions/gsd/activity-log.js +7 -0
- package/dist/resources/extensions/gsd/auto/infra-errors.js +3 -0
- package/dist/resources/extensions/gsd/auto/phases.js +37 -36
- package/dist/resources/extensions/gsd/auto-prompts.js +24 -1
- package/dist/resources/extensions/gsd/auto-start.js +31 -2
- package/dist/resources/extensions/gsd/auto-timers.js +57 -3
- package/dist/resources/extensions/gsd/auto-worktree-sync.js +4 -0
- package/dist/resources/extensions/gsd/auto-worktree.js +9 -6
- package/dist/resources/extensions/gsd/auto.js +30 -3
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +156 -0
- package/dist/resources/extensions/gsd/bootstrap/system-context.js +46 -12
- package/dist/resources/extensions/gsd/commands/catalog.js +7 -1
- package/dist/resources/extensions/gsd/commands/handlers/core.js +2 -0
- package/dist/resources/extensions/gsd/commands/handlers/ops.js +10 -0
- package/dist/resources/extensions/gsd/commands/handlers/workflow.js +5 -0
- package/dist/resources/extensions/gsd/commands-mcp-status.js +187 -0
- package/dist/resources/extensions/gsd/db-writer.js +34 -16
- package/dist/resources/extensions/gsd/doctor.js +8 -0
- package/dist/resources/extensions/gsd/git-service.js +8 -3
- package/dist/resources/extensions/gsd/gsd-db.js +12 -1
- package/dist/resources/extensions/gsd/markdown-renderer.js +1 -1
- package/dist/resources/extensions/gsd/preferences.js +9 -1
- package/dist/resources/extensions/gsd/prompts/complete-milestone.md +2 -4
- package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/reassess-roadmap.md +6 -6
- package/dist/resources/extensions/gsd/prompts/replan-slice.md +3 -14
- package/dist/resources/extensions/gsd/prompts/rethink.md +78 -0
- package/dist/resources/extensions/gsd/prompts/validate-milestone.md +7 -37
- package/dist/resources/extensions/gsd/provider-error-pause.js +7 -0
- package/dist/resources/extensions/gsd/repo-identity.js +45 -7
- package/dist/resources/extensions/gsd/rethink.js +115 -0
- package/dist/resources/extensions/gsd/state.js +41 -3
- package/dist/resources/extensions/gsd/tools/plan-slice.js +1 -0
- package/dist/resources/extensions/gsd/tools/plan-task.js +1 -0
- package/dist/resources/extensions/gsd/tools/replan-slice.js +2 -0
- package/dist/resources/extensions/gsd/tools/validate-milestone.js +88 -0
- package/dist/resources/extensions/gsd/worktree-manager.js +32 -2
- package/dist/resources/extensions/gsd/worktree-resolver.js +6 -0
- package/dist/resources/extensions/mcp-client/index.js +14 -0
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +7 -7
- 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 +2 -2
- package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
- 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/page.js +1 -1
- package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +3 -3
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +3 -3
- 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 +2 -2
- 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 +2 -2
- package/dist/web/standalone/.next/server/app/api/boot/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/browse-directories/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/captures/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/cleanup/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/dev-mode/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/doctor/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/export-data/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/files/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/forensics/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/git/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/history/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/hooks/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/inspect/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/knowledge/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/live-state/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/onboarding/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/preferences/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/recovery/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/remote-questions/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/browser/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/command/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/events/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/manage/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/settings-data/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/skill-health/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/steer/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/switch-root/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/input/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/resize/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/sessions/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/upload/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/undo/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/visualizer/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +4 -4
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -4
- 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 +2 -2
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/page.js +1 -1
- package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +7 -7
- 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 +2 -2
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/dist/web/standalone/.next/static/chunks/4024.11ca5c01938e5948.js +9 -0
- package/dist/web/standalone/.next/static/chunks/{3721.bf31263de6d5fa46.js → 485.243af25f0cdf50d6.js} +2 -2
- package/dist/web/standalone/.next/static/chunks/app/{page-7e9530a7122506c5.js → page-12dd5ece0df4badc.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/webpack-0a4cd455ec4197d2.js +1 -0
- package/dist/web/standalone/.next/static/css/dd4ae3f58ac9b600.css +1 -0
- package/package.json +1 -1
- package/packages/native/dist/stream-process/index.js +2 -2
- package/packages/native/src/__tests__/stream-process.test.mjs +34 -0
- package/packages/native/src/stream-process/index.ts +2 -2
- package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +3 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.js +15 -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 +6 -8
- package/packages/pi-coding-agent/dist/core/auth-storage.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/runner.test.js +24 -26
- package/packages/pi-coding-agent/dist/core/extensions/runner.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/fs-utils.test.js +29 -48
- package/packages/pi-coding-agent/dist/core/fs-utils.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/local-model-check.d.ts +15 -0
- package/packages/pi-coding-agent/dist/core/local-model-check.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/local-model-check.js +41 -0
- package/packages/pi-coding-agent/dist/core/local-model-check.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/model-registry.d.ts +11 -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 +20 -1
- package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/resolve-config-value.test.js +34 -44
- package/packages/pi-coding-agent/dist/core/resolve-config-value.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/session-manager.test.js +30 -34
- package/packages/pi-coding-agent/dist/core/session-manager.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/settings-manager.d.ts +3 -0
- package/packages/pi-coding-agent/dist/core/settings-manager.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/settings-manager.js +6 -0
- package/packages/pi-coding-agent/dist/core/settings-manager.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/tools/edit-diff.test.js +10 -12
- package/packages/pi-coding-agent/dist/core/tools/edit-diff.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/main.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/main.js +17 -0
- package/packages/pi-coding-agent/dist/main.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/timestamp.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/timestamp.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/timestamp.test.js +32 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/timestamp.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts +3 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js +8 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.d.ts +2 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.js +12 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/timestamp.d.ts +15 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/timestamp.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/timestamp.js +40 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/timestamp.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +4 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.d.ts +5 -2
- package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js +13 -2
- package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +17 -8
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +7 -3
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/packages/pi-coding-agent/dist/resources/extensions/memory/storage.test.js +43 -47
- package/packages/pi-coding-agent/dist/resources/extensions/memory/storage.test.js.map +1 -1
- package/packages/pi-coding-agent/package.json +1 -1
- package/packages/pi-coding-agent/src/core/auth-storage.test.ts +7 -7
- package/packages/pi-coding-agent/src/core/auth-storage.ts +15 -1
- package/packages/pi-coding-agent/src/core/extensions/runner.test.ts +26 -26
- package/packages/pi-coding-agent/src/core/fs-utils.test.ts +31 -43
- package/packages/pi-coding-agent/src/core/local-model-check.ts +45 -0
- package/packages/pi-coding-agent/src/core/model-registry.ts +21 -1
- package/packages/pi-coding-agent/src/core/resolve-config-value.test.ts +40 -45
- package/packages/pi-coding-agent/src/core/session-manager.test.ts +33 -33
- package/packages/pi-coding-agent/src/core/settings-manager.ts +9 -0
- package/packages/pi-coding-agent/src/core/tools/edit-diff.test.ts +17 -17
- package/packages/pi-coding-agent/src/main.ts +19 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/timestamp.test.ts +38 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +10 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/settings-selector.ts +15 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/timestamp.ts +48 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +3 -1
- package/packages/pi-coding-agent/src/modes/interactive/components/user-message.ts +18 -3
- package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +16 -7
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +8 -1
- package/packages/pi-coding-agent/src/resources/extensions/memory/storage.test.ts +74 -74
- package/pkg/package.json +1 -1
- package/src/resources/extensions/gsd/activity-log.ts +1 -0
- package/src/resources/extensions/gsd/auto/infra-errors.ts +3 -0
- package/src/resources/extensions/gsd/auto/phases.ts +46 -48
- package/src/resources/extensions/gsd/auto-prompts.ts +24 -1
- package/src/resources/extensions/gsd/auto-start.ts +39 -2
- package/src/resources/extensions/gsd/auto-timers.ts +64 -3
- package/src/resources/extensions/gsd/auto-worktree-sync.ts +5 -0
- package/src/resources/extensions/gsd/auto-worktree.ts +9 -6
- package/src/resources/extensions/gsd/auto.ts +37 -3
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +148 -0
- package/src/resources/extensions/gsd/bootstrap/system-context.ts +48 -11
- package/src/resources/extensions/gsd/commands/catalog.ts +7 -1
- package/src/resources/extensions/gsd/commands/handlers/core.ts +2 -0
- package/src/resources/extensions/gsd/commands/handlers/ops.ts +10 -0
- package/src/resources/extensions/gsd/commands/handlers/workflow.ts +8 -0
- package/src/resources/extensions/gsd/commands-mcp-status.ts +247 -0
- package/src/resources/extensions/gsd/db-writer.ts +39 -17
- package/src/resources/extensions/gsd/doctor.ts +7 -1
- package/src/resources/extensions/gsd/git-service.ts +6 -2
- package/src/resources/extensions/gsd/gsd-db.ts +16 -1
- package/src/resources/extensions/gsd/markdown-renderer.ts +1 -1
- package/src/resources/extensions/gsd/preferences.ts +11 -1
- package/src/resources/extensions/gsd/prompts/complete-milestone.md +2 -4
- package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/reassess-roadmap.md +6 -6
- package/src/resources/extensions/gsd/prompts/replan-slice.md +3 -14
- package/src/resources/extensions/gsd/prompts/rethink.md +78 -0
- package/src/resources/extensions/gsd/prompts/validate-milestone.md +7 -37
- package/src/resources/extensions/gsd/provider-error-pause.ts +9 -0
- package/src/resources/extensions/gsd/repo-identity.ts +46 -7
- package/src/resources/extensions/gsd/rethink.ts +154 -0
- package/src/resources/extensions/gsd/state.ts +41 -1
- package/src/resources/extensions/gsd/tests/all-milestones-complete-merge.test.ts +99 -99
- package/src/resources/extensions/gsd/tests/auto-lock-creation.test.ts +14 -16
- package/src/resources/extensions/gsd/tests/auto-paused-session-validation.test.ts +43 -57
- package/src/resources/extensions/gsd/tests/auto-pr-bugs.test.ts +88 -0
- package/src/resources/extensions/gsd/tests/auto-preflight.test.ts +11 -13
- package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +465 -523
- package/src/resources/extensions/gsd/tests/auto-secrets-gate.test.ts +73 -75
- package/src/resources/extensions/gsd/tests/auto-start-needs-discussion.test.ts +34 -56
- package/src/resources/extensions/gsd/tests/auto-worktree-milestone-merge.test.ts +533 -656
- package/src/resources/extensions/gsd/tests/auto-worktree.test.ts +165 -143
- package/src/resources/extensions/gsd/tests/cache-staleness-regression.test.ts +29 -52
- package/src/resources/extensions/gsd/tests/captures.test.ts +148 -176
- package/src/resources/extensions/gsd/tests/claude-import-tui.test.ts +32 -33
- package/src/resources/extensions/gsd/tests/collect-from-manifest.test.ts +141 -143
- package/src/resources/extensions/gsd/tests/commands-inspect-open-db.test.ts +25 -25
- package/src/resources/extensions/gsd/tests/commands-logs.test.ts +81 -81
- package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +38 -59
- package/src/resources/extensions/gsd/tests/complete-slice.test.ts +228 -263
- package/src/resources/extensions/gsd/tests/complete-task.test.ts +250 -302
- package/src/resources/extensions/gsd/tests/completed-units-metrics-sync.test.ts +114 -0
- package/src/resources/extensions/gsd/tests/context-store.test.ts +354 -367
- package/src/resources/extensions/gsd/tests/continue-here.test.ts +68 -72
- package/src/resources/extensions/gsd/tests/cost-projection.test.ts +92 -106
- package/src/resources/extensions/gsd/tests/crash-recovery.test.ts +27 -35
- package/src/resources/extensions/gsd/tests/dashboard-budget.test.ts +220 -237
- package/src/resources/extensions/gsd/tests/db-writer.test.ts +465 -416
- package/src/resources/extensions/gsd/tests/definition-loader.test.ts +76 -92
- package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +68 -83
- package/src/resources/extensions/gsd/tests/derive-state-db-disk-reconcile.test.ts +121 -0
- package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +210 -181
- package/src/resources/extensions/gsd/tests/derive-state-deps.test.ts +78 -101
- package/src/resources/extensions/gsd/tests/derive-state.test.ts +192 -227
- package/src/resources/extensions/gsd/tests/detection.test.ts +232 -278
- package/src/resources/extensions/gsd/tests/dev-engine-wrapper.test.ts +30 -34
- package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +164 -180
- package/src/resources/extensions/gsd/tests/dispatch-missing-task-plans.test.ts +43 -49
- package/src/resources/extensions/gsd/tests/dispatch-uat-last-completed.test.ts +28 -32
- package/src/resources/extensions/gsd/tests/doctor-completion-deferral.test.ts +27 -29
- package/src/resources/extensions/gsd/tests/doctor-delimiter-fix.test.ts +34 -38
- package/src/resources/extensions/gsd/tests/doctor-enhancements.test.ts +54 -75
- package/src/resources/extensions/gsd/tests/doctor-environment-worktree.test.ts +21 -32
- package/src/resources/extensions/gsd/tests/doctor-environment.test.ts +72 -97
- package/src/resources/extensions/gsd/tests/doctor-fixlevel.test.ts +38 -44
- package/src/resources/extensions/gsd/tests/doctor-git.test.ts +104 -145
- package/src/resources/extensions/gsd/tests/doctor-proactive.test.ts +84 -106
- package/src/resources/extensions/gsd/tests/doctor-roadmap-summary-atomicity.test.ts +54 -60
- package/src/resources/extensions/gsd/tests/doctor-runtime.test.ts +72 -93
- package/src/resources/extensions/gsd/tests/doctor.test.ts +104 -134
- package/src/resources/extensions/gsd/tests/ensure-db-open.test.ts +123 -131
- package/src/resources/extensions/gsd/tests/est-annotation-timeout.test.ts +120 -0
- package/src/resources/extensions/gsd/tests/exit-command.test.ts +20 -24
- package/src/resources/extensions/gsd/tests/feature-branch-lifecycle-integration.test.ts +48 -57
- package/src/resources/extensions/gsd/tests/files-loadfile-eisdir.test.ts +5 -7
- package/src/resources/extensions/gsd/tests/flag-file-db.test.ts +30 -42
- package/src/resources/extensions/gsd/tests/freeform-decisions.test.ts +198 -206
- package/src/resources/extensions/gsd/tests/git-locale.test.ts +13 -27
- package/src/resources/extensions/gsd/tests/git-service.test.ts +285 -388
- package/src/resources/extensions/gsd/tests/gitignore-tracked-gsd.test.ts +31 -39
- package/src/resources/extensions/gsd/tests/graph-operations.test.ts +63 -69
- package/src/resources/extensions/gsd/tests/gsd-db.test.ts +255 -264
- package/src/resources/extensions/gsd/tests/gsd-inspect.test.ts +108 -119
- package/src/resources/extensions/gsd/tests/gsd-recover.test.ts +81 -103
- package/src/resources/extensions/gsd/tests/gsd-tools.test.ts +229 -262
- package/src/resources/extensions/gsd/tests/headless-answers.test.ts +13 -13
- package/src/resources/extensions/gsd/tests/health-widget.test.ts +29 -37
- package/src/resources/extensions/gsd/tests/idle-recovery.test.ts +81 -102
- package/src/resources/extensions/gsd/tests/infra-error.test.ts +20 -2
- package/src/resources/extensions/gsd/tests/inherited-repo-home-dir.test.ts +121 -0
- package/src/resources/extensions/gsd/tests/init-wizard.test.ts +16 -18
- package/src/resources/extensions/gsd/tests/integration-edge.test.ts +41 -46
- package/src/resources/extensions/gsd/tests/integration-lifecycle.test.ts +42 -53
- package/src/resources/extensions/gsd/tests/integration-mixed-milestones.test.ts +75 -91
- package/src/resources/extensions/gsd/tests/integration-proof.test.ts +18 -18
- package/src/resources/extensions/gsd/tests/knowledge.test.ts +89 -0
- package/src/resources/extensions/gsd/tests/markdown-renderer.test.ts +150 -194
- package/src/resources/extensions/gsd/tests/mcp-status.test.ts +103 -0
- package/src/resources/extensions/gsd/tests/md-importer.test.ts +101 -125
- package/src/resources/extensions/gsd/tests/memory-extractor.test.ts +45 -54
- package/src/resources/extensions/gsd/tests/memory-store.test.ts +80 -93
- package/src/resources/extensions/gsd/tests/merge-conflict-stops-loop.test.ts +66 -0
- package/src/resources/extensions/gsd/tests/migrate-command.test.ts +57 -66
- package/src/resources/extensions/gsd/tests/migrate-hierarchy.test.ts +83 -93
- package/src/resources/extensions/gsd/tests/migrate-parser.test.ts +161 -170
- package/src/resources/extensions/gsd/tests/migrate-transformer.test.ts +125 -141
- package/src/resources/extensions/gsd/tests/migrate-validator-parsers.test.ts +107 -131
- package/src/resources/extensions/gsd/tests/migrate-writer-integration.test.ts +87 -96
- package/src/resources/extensions/gsd/tests/migrate-writer.test.ts +125 -164
- package/src/resources/extensions/gsd/tests/must-have-parser.test.ts +81 -94
- package/src/resources/extensions/gsd/tests/none-mode-gates.test.ts +35 -36
- package/src/resources/extensions/gsd/tests/overrides.test.ts +99 -106
- package/src/resources/extensions/gsd/tests/parallel-crash-recovery.test.ts +40 -47
- package/src/resources/extensions/gsd/tests/parallel-worker-monitoring.test.ts +25 -28
- package/src/resources/extensions/gsd/tests/parallel-workers-multi-milestone-e2e.test.ts +66 -83
- package/src/resources/extensions/gsd/tests/park-edge-cases.test.ts +54 -77
- package/src/resources/extensions/gsd/tests/park-milestone.test.ts +68 -115
- package/src/resources/extensions/gsd/tests/parsers.test.ts +546 -611
- package/src/resources/extensions/gsd/tests/paths.test.ts +72 -87
- package/src/resources/extensions/gsd/tests/post-unit-hooks.test.ts +77 -117
- package/src/resources/extensions/gsd/tests/preferences.test.ts +27 -0
- package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +11 -7
- package/src/resources/extensions/gsd/tests/prompt-db.test.ts +56 -56
- package/src/resources/extensions/gsd/tests/queue-draft-detection.test.ts +93 -119
- package/src/resources/extensions/gsd/tests/queue-order.test.ts +70 -82
- package/src/resources/extensions/gsd/tests/queue-reorder-e2e.test.ts +42 -55
- package/src/resources/extensions/gsd/tests/quick-auto-guard.test.ts +100 -0
- package/src/resources/extensions/gsd/tests/quick-branch-lifecycle.test.ts +45 -73
- package/src/resources/extensions/gsd/tests/reassess-prompt.test.ts +28 -38
- package/src/resources/extensions/gsd/tests/recovery-attempts-reset.test.ts +176 -0
- package/src/resources/extensions/gsd/tests/replan-slice.test.ts +73 -80
- package/src/resources/extensions/gsd/tests/repo-identity-worktree.test.ts +71 -74
- package/src/resources/extensions/gsd/tests/requirements.test.ts +70 -75
- package/src/resources/extensions/gsd/tests/retry-state-reset.test.ts +44 -66
- package/src/resources/extensions/gsd/tests/roadmap-parse-regression.test.ts +114 -181
- package/src/resources/extensions/gsd/tests/rule-registry.test.ts +63 -65
- package/src/resources/extensions/gsd/tests/run-uat.test.ts +66 -128
- package/src/resources/extensions/gsd/tests/session-lock-multipath.test.ts +18 -25
- package/src/resources/extensions/gsd/tests/session-lock-regression.test.ts +37 -44
- package/src/resources/extensions/gsd/tests/shared-wal.test.ts +19 -26
- package/src/resources/extensions/gsd/tests/sqlite-unavailable-gate.test.ts +63 -0
- package/src/resources/extensions/gsd/tests/stalled-tool-recovery.test.ts +6 -8
- package/src/resources/extensions/gsd/tests/stop-auto-merge-back.test.ts +67 -0
- package/src/resources/extensions/gsd/tests/survivor-branch-complete.test.ts +108 -0
- package/src/resources/extensions/gsd/tests/symlink-numbered-variants.test.ts +22 -28
- package/src/resources/extensions/gsd/tests/terminated-transient.test.ts +49 -0
- package/src/resources/extensions/gsd/tests/token-savings.test.ts +54 -56
- package/src/resources/extensions/gsd/tests/tool-call-loop-guard.test.ts +23 -25
- package/src/resources/extensions/gsd/tests/tool-naming.test.ts +10 -11
- package/src/resources/extensions/gsd/tests/unique-milestone-ids.test.ts +66 -82
- package/src/resources/extensions/gsd/tests/unit-runtime.test.ts +46 -47
- package/src/resources/extensions/gsd/tests/visualizer-critical-path.test.ts +20 -22
- package/src/resources/extensions/gsd/tests/visualizer-data.test.ts +84 -86
- package/src/resources/extensions/gsd/tests/visualizer-overlay.test.ts +41 -43
- package/src/resources/extensions/gsd/tests/visualizer-views.test.ts +94 -96
- package/src/resources/extensions/gsd/tests/windows-path-normalization.test.ts +11 -13
- package/src/resources/extensions/gsd/tests/worker-registry.test.ts +27 -29
- package/src/resources/extensions/gsd/tests/workflow-templates.test.ts +50 -52
- package/src/resources/extensions/gsd/tests/worktree-bugfix.test.ts +10 -13
- package/src/resources/extensions/gsd/tests/worktree-db-integration.test.ts +14 -18
- package/src/resources/extensions/gsd/tests/worktree-db.test.ts +38 -39
- package/src/resources/extensions/gsd/tests/worktree-e2e.test.ts +17 -21
- package/src/resources/extensions/gsd/tests/worktree-health.test.ts +25 -30
- package/src/resources/extensions/gsd/tests/worktree-integration.test.ts +30 -37
- package/src/resources/extensions/gsd/tests/worktree-submodule-safety.test.ts +65 -0
- package/src/resources/extensions/gsd/tests/worktree-symlink-removal.test.ts +15 -22
- package/src/resources/extensions/gsd/tests/worktree-sync-milestones.test.ts +59 -66
- package/src/resources/extensions/gsd/tests/worktree.test.ts +44 -50
- package/src/resources/extensions/gsd/tools/plan-slice.ts +2 -0
- package/src/resources/extensions/gsd/tools/plan-task.ts +2 -0
- package/src/resources/extensions/gsd/tools/replan-slice.ts +3 -0
- package/src/resources/extensions/gsd/tools/validate-milestone.ts +127 -0
- package/src/resources/extensions/gsd/worktree-manager.ts +43 -2
- package/src/resources/extensions/gsd/worktree-resolver.ts +7 -0
- package/src/resources/extensions/mcp-client/index.ts +20 -0
- package/dist/web/standalone/.next/static/chunks/4024.0de81b543b28b9fe.js +0 -9
- package/dist/web/standalone/.next/static/chunks/webpack-9014b5adb127a98a.js +0 -1
- package/dist/web/standalone/.next/static/css/8a727f372cf53002.css +0 -1
- /package/dist/web/standalone/.next/static/{mgkxN0mGP6gSUmGPEzbk_ → wUzEX1U3CmFcMry2SUDJn}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{mgkxN0mGP6gSUmGPEzbk_ → wUzEX1U3CmFcMry2SUDJn}/_ssgManifest.js +0 -0
|
@@ -510,13 +510,18 @@ export class GitServiceImpl {
|
|
|
510
510
|
* Returns the PR URL on success, or null on failure.
|
|
511
511
|
* Non-fatal: callers should treat failure as best-effort.
|
|
512
512
|
*/
|
|
513
|
-
export function createDraftPR(basePath, milestoneId, title, body) {
|
|
513
|
+
export function createDraftPR(basePath, milestoneId, title, body, opts) {
|
|
514
514
|
try {
|
|
515
|
-
const
|
|
515
|
+
const args = [
|
|
516
516
|
"pr", "create", "--draft",
|
|
517
517
|
"--title", title,
|
|
518
518
|
"--body", body,
|
|
519
|
-
]
|
|
519
|
+
];
|
|
520
|
+
if (opts?.head)
|
|
521
|
+
args.push("--head", opts.head);
|
|
522
|
+
if (opts?.base)
|
|
523
|
+
args.push("--base", opts.base);
|
|
524
|
+
const result = execFileSync("gh", args, { cwd: basePath, encoding: "utf8", timeout: 30000, env: GIT_NO_PROMPT_ENV });
|
|
520
525
|
return result.trim();
|
|
521
526
|
}
|
|
522
527
|
catch {
|
|
@@ -253,6 +253,7 @@ function initSchema(db, fileBacked) {
|
|
|
253
253
|
inputs TEXT NOT NULL DEFAULT '[]',
|
|
254
254
|
expected_output TEXT NOT NULL DEFAULT '[]',
|
|
255
255
|
observability_impact TEXT NOT NULL DEFAULT '',
|
|
256
|
+
full_plan_md TEXT NOT NULL DEFAULT '',
|
|
256
257
|
sequence INTEGER DEFAULT 0, -- DEAD CODE: no tool exposes sequence — always 0
|
|
257
258
|
PRIMARY KEY (milestone_id, slice_id, id),
|
|
258
259
|
FOREIGN KEY (milestone_id, slice_id) REFERENCES slices(milestone_id, id)
|
|
@@ -542,6 +543,13 @@ function migrateSchema(db) {
|
|
|
542
543
|
":applied_at": new Date().toISOString(),
|
|
543
544
|
});
|
|
544
545
|
}
|
|
546
|
+
if (currentVersion < 11) {
|
|
547
|
+
ensureColumn(db, "tasks", "full_plan_md", `ALTER TABLE tasks ADD COLUMN full_plan_md TEXT NOT NULL DEFAULT ''`);
|
|
548
|
+
db.prepare("INSERT INTO schema_version (version, applied_at) VALUES (:version, :applied_at)").run({
|
|
549
|
+
":version": 11,
|
|
550
|
+
":applied_at": new Date().toISOString(),
|
|
551
|
+
});
|
|
552
|
+
}
|
|
545
553
|
db.exec("COMMIT");
|
|
546
554
|
}
|
|
547
555
|
catch (err) {
|
|
@@ -997,7 +1005,8 @@ export function upsertTaskPlanning(milestoneId, sliceId, taskId, planning) {
|
|
|
997
1005
|
verify = COALESCE(:verify, verify),
|
|
998
1006
|
inputs = COALESCE(:inputs, inputs),
|
|
999
1007
|
expected_output = COALESCE(:expected_output, expected_output),
|
|
1000
|
-
observability_impact = COALESCE(:observability_impact, observability_impact)
|
|
1008
|
+
observability_impact = COALESCE(:observability_impact, observability_impact),
|
|
1009
|
+
full_plan_md = COALESCE(:full_plan_md, full_plan_md)
|
|
1001
1010
|
WHERE milestone_id = :milestone_id AND slice_id = :slice_id AND id = :id`).run({
|
|
1002
1011
|
":milestone_id": milestoneId,
|
|
1003
1012
|
":slice_id": sliceId,
|
|
@@ -1010,6 +1019,7 @@ export function upsertTaskPlanning(milestoneId, sliceId, taskId, planning) {
|
|
|
1010
1019
|
":inputs": planning.inputs ? JSON.stringify(planning.inputs) : null,
|
|
1011
1020
|
":expected_output": planning.expectedOutput ? JSON.stringify(planning.expectedOutput) : null,
|
|
1012
1021
|
":observability_impact": planning.observabilityImpact ?? null,
|
|
1022
|
+
":full_plan_md": planning.fullPlanMd ?? null,
|
|
1013
1023
|
});
|
|
1014
1024
|
}
|
|
1015
1025
|
function rowToSlice(row) {
|
|
@@ -1078,6 +1088,7 @@ function rowToTask(row) {
|
|
|
1078
1088
|
inputs: JSON.parse(row["inputs"] || "[]"),
|
|
1079
1089
|
expected_output: JSON.parse(row["expected_output"] || "[]"),
|
|
1080
1090
|
observability_impact: row["observability_impact"] ?? "",
|
|
1091
|
+
full_plan_md: row["full_plan_md"] ?? "",
|
|
1081
1092
|
sequence: row["sequence"] ?? 0,
|
|
1082
1093
|
};
|
|
1083
1094
|
}
|
|
@@ -298,7 +298,7 @@ export async function renderTaskPlanFromDb(basePath, milestoneId, sliceId, taskI
|
|
|
298
298
|
mkdirSync(tasksDir, { recursive: true });
|
|
299
299
|
const absPath = join(tasksDir, buildTaskFileName(taskId, "PLAN"));
|
|
300
300
|
const artifactPath = toArtifactPath(absPath, basePath);
|
|
301
|
-
const content = renderTaskPlanMarkdown(task);
|
|
301
|
+
const content = task.full_plan_md.trim() ? task.full_plan_md : renderTaskPlanMarkdown(task);
|
|
302
302
|
await writeAndStore(absPath, artifactPath, content, {
|
|
303
303
|
artifact_type: "PLAN",
|
|
304
304
|
milestone_id: milestoneId,
|
|
@@ -125,6 +125,11 @@ function loadPreferencesFile(path, scope) {
|
|
|
125
125
|
...(allWarnings.length > 0 ? { warnings: allWarnings } : {}),
|
|
126
126
|
};
|
|
127
127
|
}
|
|
128
|
+
let _warnedUnrecognizedFormat = false;
|
|
129
|
+
/** @internal Reset the warn-once flag — exported for testing only. */
|
|
130
|
+
export function _resetParseWarningFlag() {
|
|
131
|
+
_warnedUnrecognizedFormat = false;
|
|
132
|
+
}
|
|
128
133
|
/** @internal Exported for testing only */
|
|
129
134
|
export function parsePreferencesMarkdown(content) {
|
|
130
135
|
// Use indexOf instead of [\s\S]*? regex to avoid backtracking (#468)
|
|
@@ -142,7 +147,10 @@ export function parsePreferencesMarkdown(content) {
|
|
|
142
147
|
if (/^##\s+\w/m.test(content)) {
|
|
143
148
|
return parseHeadingListFormat(content);
|
|
144
149
|
}
|
|
145
|
-
|
|
150
|
+
if (!_warnedUnrecognizedFormat) {
|
|
151
|
+
_warnedUnrecognizedFormat = true;
|
|
152
|
+
console.warn("[parsePreferencesMarkdown] preferences.md exists but uses an unrecognized format — skipping.");
|
|
153
|
+
}
|
|
146
154
|
return null;
|
|
147
155
|
}
|
|
148
156
|
function parseFrontmatterBlock(frontmatter) {
|
|
@@ -21,8 +21,8 @@ Then:
|
|
|
21
21
|
4. Verify each **success criterion** from the milestone definition in `{{roadmapPath}}`. For each criterion, confirm it was met with specific evidence from slice summaries, test results, or observable behavior. List any criterion that was NOT met.
|
|
22
22
|
5. Verify the milestone's **definition of done** — all slices are `[x]`, all slice summaries exist, and any cross-slice integration points work correctly.
|
|
23
23
|
6. Validate **requirement status transitions**. For each requirement that changed status during this milestone, confirm the transition is supported by evidence. Requirements can move between Active, Validated, Deferred, Blocked, or Out of Scope — but only with proof.
|
|
24
|
-
7.
|
|
25
|
-
8. Update `.gsd/REQUIREMENTS.md` if any requirement status transitions were validated in step
|
|
24
|
+
7. **Persist completion through `gsd_complete_milestone`.** Call it with: `milestoneId`, `title`, `oneLiner`, `narrative`, `successCriteriaResults`, `definitionOfDoneResults`, `requirementOutcomes`, `keyDecisions`, `keyFiles`, `lessonsLearned`, `followUps`, `deviations`. The tool updates the milestone status in the DB, renders `{{milestoneSummaryPath}}`, and validates all slices are complete before proceeding.
|
|
25
|
+
8. Update `.gsd/REQUIREMENTS.md` if any requirement status transitions were validated in step 6.
|
|
26
26
|
9. Update `.gsd/PROJECT.md` to reflect milestone completion and current project state.
|
|
27
27
|
10. Review all slice summaries for cross-cutting lessons, patterns, or gotchas that emerged during this milestone. Append any non-obvious, reusable insights to `.gsd/KNOWLEDGE.md`.
|
|
28
28
|
11. Do not commit manually — the system auto-commits your changes after this unit completes.
|
|
@@ -31,6 +31,4 @@ Then:
|
|
|
31
31
|
|
|
32
32
|
**File system safety:** When scanning milestone directories for evidence, use `ls` or `find` to list directory contents first — never pass a directory path (e.g. `tasks/`, `slices/`) directly to the `read` tool. The `read` tool only accepts file paths, not directories.
|
|
33
33
|
|
|
34
|
-
**You MUST write `{{milestoneSummaryPath}}` AND update PROJECT.md before finishing.**
|
|
35
|
-
|
|
36
34
|
When done, say: "Milestone {{milestoneId}} complete."
|
|
@@ -63,7 +63,7 @@ Then:
|
|
|
63
63
|
- a matching task plan file with description, steps, must-haves, verification, inputs, and expected output
|
|
64
64
|
- **Inputs and Expected Output must list concrete backtick-wrapped file paths** (e.g. `` `src/types.ts` ``). These are machine-parsed to derive task dependencies — vague prose without paths breaks parallel execution. Every task must have at least one output file path.
|
|
65
65
|
- Observability Impact section **only if the task touches runtime boundaries, async flows, or error paths** — omit it otherwise
|
|
66
|
-
6. **Persist planning state through
|
|
66
|
+
6. **Persist planning state through `gsd_plan_slice`.** Call it with the full slice planning payload (goal, demo, must-haves, verification, tasks, and metadata). The tool inserts all tasks in the same transaction, writes to the DB, and renders `{{outputPath}}` and `{{slicePath}}/tasks/T##-PLAN.md` files automatically. Do **not** call `gsd_plan_task` separately — `gsd_plan_slice` handles task persistence. Do **not** rely on direct `PLAN.md` writes as the source of truth; the DB-backed tool is the canonical write path for slice and task planning state.
|
|
67
67
|
7. **Self-audit the plan.** Walk through each check — if any fail, fix the plan files before moving on:
|
|
68
68
|
- **Completion semantics:** If every task were completed exactly as written, the slice goal/demo should actually be true.
|
|
69
69
|
- **Requirement coverage:** Every must-have in the slice maps to at least one task. No must-have is orphaned. If `REQUIREMENTS.md` exists, every Active requirement this slice owns maps to at least one task.
|
|
@@ -50,14 +50,14 @@ If all criteria have at least one remaining owning slice, the coverage check pas
|
|
|
50
50
|
|
|
51
51
|
**If the roadmap is still good:**
|
|
52
52
|
|
|
53
|
-
|
|
53
|
+
Use `gsd_reassess_roadmap` with `verdict: "roadmap-confirmed"`, an empty `sliceChanges` object, and the assessment text — the tool writes the assessment to the DB and renders `{{assessmentPath}}`. If requirements exist, explicitly note whether requirement coverage remains sound.
|
|
54
54
|
|
|
55
55
|
**If changes are needed:**
|
|
56
56
|
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
57
|
+
**Persist changes through `gsd_reassess_roadmap`.** Pass: `milestoneId`, `completedSliceId`, `verdict` (e.g. "roadmap-adjusted"), `assessment` (text explaining the decision), and `sliceChanges` with `modified` (array of sliceId, title, risk, depends, demo), `added` (same shape), `removed` (array of slice ID strings). The tool structurally enforces preservation of completed slices, writes the assessment to the DB, re-renders `{{roadmapPath}}`, and renders `{{assessmentPath}}`.
|
|
58
|
+
|
|
59
|
+
If `.gsd/REQUIREMENTS.md` exists and requirement ownership or status changed, update it.
|
|
60
|
+
|
|
61
|
+
{{commitInstruction}}
|
|
62
62
|
|
|
63
63
|
When done, say: "Roadmap reassessed."
|
|
@@ -32,19 +32,8 @@ Consider these captures when rewriting the remaining tasks — they represent th
|
|
|
32
32
|
|
|
33
33
|
1. Read the blocker task summary carefully. Understand exactly what was discovered and why it blocks the current plan.
|
|
34
34
|
2. Analyze the remaining `[ ]` tasks in the slice plan. Determine which are still valid, which need modification, and which should be replaced.
|
|
35
|
-
3. **Persist replan state through `gsd_replan_slice`.** Call it with
|
|
36
|
-
4.
|
|
37
|
-
|
|
38
|
-
- What changed in the plan and why
|
|
39
|
-
- Which incomplete tasks were modified, added, or removed
|
|
40
|
-
- Any new risks or considerations introduced by the replan
|
|
41
|
-
5. If using the degraded fallback, rewrite `{{planPath}}` with the updated slice plan:
|
|
42
|
-
- Keep all `[x]` tasks exactly as they were (same IDs, same descriptions, same checkmarks)
|
|
43
|
-
- Update the `[ ]` tasks to address the blocker
|
|
44
|
-
- Ensure the slice Goal and Demo sections are still achievable with the new tasks, or update them if the blocker fundamentally changes what the slice can deliver
|
|
45
|
-
- Update the Files Likely Touched section if the replan changes which files are affected
|
|
46
|
-
- If a DB-backed planning tool exists for this phase, use it as the source of truth and make any rewritten `PLAN.md` reflect that persisted state rather than bypassing it
|
|
47
|
-
6. If any incomplete task had a `T0x-PLAN.md`, remove or rewrite it to match the new task description.
|
|
48
|
-
7. Do not commit manually — the system auto-commits your changes after this unit completes.
|
|
35
|
+
3. **Persist replan state through `gsd_replan_slice`.** Call it with: `milestoneId`, `sliceId`, `blockerTaskId`, `blockerDescription`, `whatChanged`, `updatedTasks` (array of task objects with taskId, title, description, estimate, files, verify, inputs, expectedOutput), `removedTaskIds` (array of task ID strings). The tool structurally enforces preservation of completed tasks, writes replan history to the DB, re-renders `{{planPath}}`, and renders `{{replanPath}}`.
|
|
36
|
+
4. If any incomplete task had a `T0x-PLAN.md`, remove or rewrite it to match the new task description.
|
|
37
|
+
5. Do not commit manually — the system auto-commits your changes after this unit completes.
|
|
49
38
|
|
|
50
39
|
When done, say: "Slice {{sliceId}} replanned."
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
You are a project reorganization assistant for a GSD (Get Shit Done) project. The user wants to rethink their milestone plan — reorder priorities, remove work that's no longer needed, add new milestones, or restructure dependencies.
|
|
2
|
+
|
|
3
|
+
## Current Milestone Landscape
|
|
4
|
+
|
|
5
|
+
{{rethinkData}}
|
|
6
|
+
|
|
7
|
+
## Detailed Milestone Context
|
|
8
|
+
|
|
9
|
+
{{existingMilestonesContext}}
|
|
10
|
+
|
|
11
|
+
## Your Role
|
|
12
|
+
|
|
13
|
+
1. Present the current milestone order as a clear numbered list with status indicators (e.g. ✅ complete, ▶ active, ⏳ pending, ⏸ parked)
|
|
14
|
+
2. Ask: **"What would you like to change?"**
|
|
15
|
+
3. Execute changes conversationally, confirming destructive operations before proceeding
|
|
16
|
+
|
|
17
|
+
## Supported Operations
|
|
18
|
+
|
|
19
|
+
### Reorder milestones
|
|
20
|
+
Change execution order of pending/active milestones. Write `.gsd/QUEUE-ORDER.json`:
|
|
21
|
+
```json
|
|
22
|
+
{ "order": ["M003", "M001", "M002"], "updatedAt": "<ISO timestamp>" }
|
|
23
|
+
```
|
|
24
|
+
Only include non-complete milestone IDs. Validate dependency constraints before saving.
|
|
25
|
+
|
|
26
|
+
### Park a milestone
|
|
27
|
+
Temporarily shelve a milestone (reversible). Create a `{ID}-PARKED.md` file in the milestone directory:
|
|
28
|
+
```markdown
|
|
29
|
+
---
|
|
30
|
+
parked_at: <ISO timestamp>
|
|
31
|
+
reason: "<reason>"
|
|
32
|
+
---
|
|
33
|
+
|
|
34
|
+
# {ID} — Parked
|
|
35
|
+
|
|
36
|
+
> <reason>
|
|
37
|
+
```
|
|
38
|
+
**Bias toward parking over discarding** when a milestone has any completed slices or tasks.
|
|
39
|
+
|
|
40
|
+
### Unpark a milestone
|
|
41
|
+
Remove the `{ID}-PARKED.md` file from the milestone directory to reactivate it.
|
|
42
|
+
|
|
43
|
+
### Discard a milestone
|
|
44
|
+
**Permanently** delete a milestone directory and prune it from QUEUE-ORDER.json. **Always confirm with the user before discarding.** Warn explicitly if the milestone has completed work.
|
|
45
|
+
|
|
46
|
+
### Add a new milestone
|
|
47
|
+
Use the `gsd_milestone_generate_id` tool to get the next ID, then write a `{ID}-CONTEXT.md` file in `.gsd/milestones/{ID}/` with scope, goals, and success criteria. Update QUEUE-ORDER.json to place it at the desired position.
|
|
48
|
+
|
|
49
|
+
### Update dependencies
|
|
50
|
+
Edit `depends_on` in the YAML frontmatter of a milestone's `{ID}-CONTEXT.md` file. For example:
|
|
51
|
+
```yaml
|
|
52
|
+
depends_on: [M001, M003]
|
|
53
|
+
```
|
|
54
|
+
|
|
55
|
+
## Dependency Validation Rules
|
|
56
|
+
|
|
57
|
+
Before applying any reorder, verify:
|
|
58
|
+
- A milestone **cannot** be scheduled before any milestone in its `depends_on` list (would_block)
|
|
59
|
+
- Circular dependencies are forbidden
|
|
60
|
+
- Dependencies on non-existent milestones are invalid (missing_dep)
|
|
61
|
+
- Completed milestones always satisfy dependencies regardless of position
|
|
62
|
+
|
|
63
|
+
If a proposed order would violate constraints, explain the issue and suggest alternatives (e.g. removing the dependency, reordering differently, or parking the blocker).
|
|
64
|
+
|
|
65
|
+
## After Each Change
|
|
66
|
+
|
|
67
|
+
1. Execute the change (write/delete files, update QUEUE-ORDER.json)
|
|
68
|
+
2. Show the updated milestone order
|
|
69
|
+
3. Note if the active milestone changed as a result
|
|
70
|
+
4. Ask if there's anything else to adjust
|
|
71
|
+
|
|
72
|
+
## Important Constraints
|
|
73
|
+
|
|
74
|
+
- Do NOT modify completed milestones — they're done
|
|
75
|
+
- Do NOT park completed milestones — it would corrupt dependency satisfaction
|
|
76
|
+
- Park is preferred over discard when a milestone has any completed work
|
|
77
|
+
- Always persist queue order changes to `.gsd/QUEUE-ORDER.json`
|
|
78
|
+
- After changes, run `git add .gsd/ && git commit -m "docs: rethink milestone order"` to persist
|
|
@@ -16,6 +16,8 @@ All relevant context has been preloaded below — the roadmap, all slice summari
|
|
|
16
16
|
|
|
17
17
|
{{inlinedContext}}
|
|
18
18
|
|
|
19
|
+
{{skillActivation}}
|
|
20
|
+
|
|
19
21
|
## Validation Steps
|
|
20
22
|
|
|
21
23
|
1. For each **success criterion** in `{{roadmapPath}}`, check whether slice summaries and UAT results provide evidence that it was met. Record pass/fail per criterion.
|
|
@@ -25,47 +27,15 @@ All relevant context has been preloaded below — the roadmap, all slice summari
|
|
|
25
27
|
5. Determine a verdict:
|
|
26
28
|
- `pass` — all criteria met, all slices delivered, no gaps
|
|
27
29
|
- `needs-attention` — minor gaps that do not block completion (document them)
|
|
28
|
-
- `needs-remediation` — material gaps found;
|
|
29
|
-
|
|
30
|
-
## Output
|
|
31
|
-
|
|
32
|
-
Write `{{validationPath}}` with this structure:
|
|
33
|
-
|
|
34
|
-
```markdown
|
|
35
|
-
---
|
|
36
|
-
verdict: <pass|needs-attention|needs-remediation>
|
|
37
|
-
remediation_round: {{remediationRound}}
|
|
38
|
-
---
|
|
39
|
-
|
|
40
|
-
# Milestone Validation: {{milestoneId}}
|
|
30
|
+
- `needs-remediation` — material gaps found; remediation slices must be added to the roadmap
|
|
41
31
|
|
|
42
|
-
##
|
|
43
|
-
- [x] Criterion 1 — evidence: ...
|
|
44
|
-
- [ ] Criterion 2 — gap: ...
|
|
32
|
+
## Persist Validation
|
|
45
33
|
|
|
46
|
-
|
|
47
|
-
| Slice | Claimed | Delivered | Status |
|
|
48
|
-
|-------|---------|-----------|--------|
|
|
49
|
-
| S01 | ... | ... | pass |
|
|
50
|
-
|
|
51
|
-
## Cross-Slice Integration
|
|
52
|
-
(any boundary mismatches)
|
|
53
|
-
|
|
54
|
-
## Requirement Coverage
|
|
55
|
-
(any unaddressed requirements)
|
|
56
|
-
|
|
57
|
-
## Verdict Rationale
|
|
58
|
-
(why this verdict was chosen)
|
|
59
|
-
|
|
60
|
-
## Remediation Plan
|
|
61
|
-
(only if verdict is needs-remediation — list new slices to add to the roadmap)
|
|
62
|
-
```
|
|
34
|
+
**Persist validation results through `gsd_validate_milestone`.** Call it with: `milestoneId`, `verdict`, `remediationRound`, `successCriteriaChecklist`, `sliceDeliveryAudit`, `crossSliceIntegration`, `requirementCoverage`, `verdictRationale`, and `remediationPlan` (if verdict is `needs-remediation`). The tool writes the validation to the DB and renders VALIDATION.md to disk.
|
|
63
35
|
|
|
64
36
|
If verdict is `needs-remediation`:
|
|
65
|
-
-
|
|
66
|
-
- These slices will be planned and executed before validation re-runs
|
|
67
|
-
|
|
68
|
-
**You MUST write `{{validationPath}}` before finishing.**
|
|
37
|
+
- After calling `gsd_validate_milestone`, use `gsd_reassess_roadmap` to add remediation slices. Pass `milestoneId`, a synthetic `completedSliceId` (e.g. "VALIDATION"), `verdict: "roadmap-adjusted"`, `assessment` text, and `sliceChanges` with the new slices in the `added` array. The tool persists the changes to the DB and re-renders ROADMAP.md.
|
|
38
|
+
- These remediation slices will be planned and executed before validation re-runs.
|
|
69
39
|
|
|
70
40
|
**File system safety:** When scanning milestone directories for evidence, use `ls` or `find` to list directory contents first — never pass a directory path (e.g. `tasks/`, `slices/`) directly to the `read` tool. The `read` tool only accepts file paths, not directories.
|
|
71
41
|
|
|
@@ -10,6 +10,10 @@
|
|
|
10
10
|
export function classifyProviderError(errorMsg) {
|
|
11
11
|
const isRateLimit = /rate.?limit|too many requests|429/i.test(errorMsg);
|
|
12
12
|
const isServerError = /internal server error|500|502|503|overloaded|server_error|api_error|service.?unavailable/i.test(errorMsg);
|
|
13
|
+
// Connection/process errors — transient, auto-resume after brief backoff (#2309).
|
|
14
|
+
// These indicate the process was killed, the connection was reset, or a network
|
|
15
|
+
// blip occurred. They are NOT permanent failures.
|
|
16
|
+
const isConnectionError = /terminated|connection.?reset|connection.?refused|other side closed|fetch failed|network.?(?:is\s+)?unavailable|ECONNREFUSED|ECONNRESET|EPIPE/i.test(errorMsg);
|
|
13
17
|
// Permanent errors — never auto-resume
|
|
14
18
|
const isPermanent = /auth|unauthorized|forbidden|invalid.*key|invalid.*api|billing|quota exceeded|account/i.test(errorMsg);
|
|
15
19
|
if (isPermanent && !isRateLimit) {
|
|
@@ -24,6 +28,9 @@ export function classifyProviderError(errorMsg) {
|
|
|
24
28
|
if (isServerError) {
|
|
25
29
|
return { isTransient: true, isRateLimit: false, suggestedDelayMs: 30_000 }; // 30s for server errors
|
|
26
30
|
}
|
|
31
|
+
if (isConnectionError) {
|
|
32
|
+
return { isTransient: true, isRateLimit: false, suggestedDelayMs: 15_000 }; // 15s for connection errors
|
|
33
|
+
}
|
|
27
34
|
// Unknown error — treat as permanent (user reviews)
|
|
28
35
|
return { isTransient: false, isRateLimit: false, suggestedDelayMs: 0 };
|
|
29
36
|
}
|
|
@@ -89,16 +89,17 @@ export function readRepoMeta(externalPath) {
|
|
|
89
89
|
* Returns true when ALL of:
|
|
90
90
|
* 1. basePath is inside a git repo (git rev-parse succeeds)
|
|
91
91
|
* 2. The resolved git root is a proper ancestor of basePath
|
|
92
|
-
* 3. There is no `.gsd` directory at the git root
|
|
93
|
-
* has not been
|
|
92
|
+
* 3. There is no *project* `.gsd` directory at the git root or any
|
|
93
|
+
* intermediate ancestor (the parent project has not been
|
|
94
|
+
* initialised with GSD)
|
|
94
95
|
*
|
|
95
96
|
* When true, the caller should run `git init` at basePath so that
|
|
96
97
|
* `repoIdentity()` produces a hash unique to this directory, preventing
|
|
97
98
|
* cross-project state leaks (#1639).
|
|
98
99
|
*
|
|
99
|
-
* When the git root already has `.gsd`, the directory is a
|
|
100
|
-
* subdirectory of an existing GSD project — `cd src/ && /gsd`
|
|
101
|
-
* still load the parent project's milestones.
|
|
100
|
+
* When the git root already has a project `.gsd`, the directory is a
|
|
101
|
+
* legitimate subdirectory of an existing GSD project — `cd src/ && /gsd`
|
|
102
|
+
* should still load the parent project's milestones.
|
|
102
103
|
*/
|
|
103
104
|
export function isInheritedRepo(basePath) {
|
|
104
105
|
try {
|
|
@@ -109,12 +110,12 @@ export function isInheritedRepo(basePath) {
|
|
|
109
110
|
return false; // basePath IS the root
|
|
110
111
|
// The git root is a proper ancestor. Check whether it already has .gsd
|
|
111
112
|
// (i.e. the parent project was initialised with GSD).
|
|
112
|
-
if (
|
|
113
|
+
if (isProjectGsd(join(root, ".gsd")))
|
|
113
114
|
return false;
|
|
114
115
|
// Also walk up from basePath to the git root checking for .gsd
|
|
115
116
|
let dir = normalizedBase;
|
|
116
117
|
while (dir !== normalizedRoot && dir !== dirname(dir)) {
|
|
117
|
-
if (
|
|
118
|
+
if (isProjectGsd(join(dir, ".gsd")))
|
|
118
119
|
return false;
|
|
119
120
|
dir = dirname(dir);
|
|
120
121
|
}
|
|
@@ -124,6 +125,43 @@ export function isInheritedRepo(basePath) {
|
|
|
124
125
|
return false;
|
|
125
126
|
}
|
|
126
127
|
}
|
|
128
|
+
/**
|
|
129
|
+
* Distinguish a *project* `.gsd` from the global `~/.gsd` state directory.
|
|
130
|
+
*
|
|
131
|
+
* A project `.gsd` is either:
|
|
132
|
+
* - A symlink to an external state directory (normal post-migration layout)
|
|
133
|
+
* - A legacy real directory that is NOT the global GSD home
|
|
134
|
+
*
|
|
135
|
+
* When the user's home directory is itself a git repo (e.g. dotfile managers),
|
|
136
|
+
* `~/.gsd` exists but is the global state directory — not a project `.gsd`.
|
|
137
|
+
* Treating it as a project `.gsd` would cause isInheritedRepo() to wrongly
|
|
138
|
+
* conclude that subdirectories are part of the home "project" (#2393).
|
|
139
|
+
*/
|
|
140
|
+
function isProjectGsd(gsdPath) {
|
|
141
|
+
if (!existsSync(gsdPath))
|
|
142
|
+
return false;
|
|
143
|
+
try {
|
|
144
|
+
const stat = lstatSync(gsdPath);
|
|
145
|
+
// Symlinks are always project .gsd (created by ensureGsdSymlink).
|
|
146
|
+
if (stat.isSymbolicLink())
|
|
147
|
+
return true;
|
|
148
|
+
// For real directories, check that this isn't the global GSD home.
|
|
149
|
+
// Recompute gsdHome dynamically so env overrides (GSD_HOME) are
|
|
150
|
+
// picked up at call time, not just at module load time.
|
|
151
|
+
if (stat.isDirectory()) {
|
|
152
|
+
const currentGsdHome = process.env.GSD_HOME || join(homedir(), ".gsd");
|
|
153
|
+
const normalizedGsdPath = canonicalizeExistingPath(gsdPath);
|
|
154
|
+
const normalizedGsdHome = canonicalizeExistingPath(currentGsdHome);
|
|
155
|
+
if (normalizedGsdPath === normalizedGsdHome)
|
|
156
|
+
return false;
|
|
157
|
+
return true;
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
catch {
|
|
161
|
+
// lstat failed — treat as no .gsd present
|
|
162
|
+
}
|
|
163
|
+
return false;
|
|
164
|
+
}
|
|
127
165
|
// ─── Repo Identity ──────────────────────────────────────────────────────────
|
|
128
166
|
/**
|
|
129
167
|
* Get the git remote URL for "origin", or "" if no remote is configured.
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* GSD Rethink — Conversational project reorganization.
|
|
3
|
+
*
|
|
4
|
+
* Collects a snapshot of all milestones (status, dependencies, slice progress,
|
|
5
|
+
* queue order) and dispatches a prompt that turns Claude into a reorganization
|
|
6
|
+
* assistant. Claude can then reorder, park, unpark, discard, or add milestones
|
|
7
|
+
* through conversation.
|
|
8
|
+
*/
|
|
9
|
+
import { existsSync } from "node:fs";
|
|
10
|
+
import { isAutoActive } from "./auto.js";
|
|
11
|
+
import { deriveState } from "./state.js";
|
|
12
|
+
import { gsdRoot } from "./paths.js";
|
|
13
|
+
import { findMilestoneIds } from "./milestone-ids.js";
|
|
14
|
+
import { loadQueueOrder, validateQueueOrder } from "./queue-order.js";
|
|
15
|
+
import { getParkedReason } from "./milestone-actions.js";
|
|
16
|
+
import { getMilestoneSlices, isDbAvailable } from "./gsd-db.js";
|
|
17
|
+
import { buildExistingMilestonesContext } from "./guided-flow-queue.js";
|
|
18
|
+
import { loadPrompt } from "./prompt-loader.js";
|
|
19
|
+
// ─── Entry Point ──────────────────────────────────────────────────────────────
|
|
20
|
+
export async function handleRethink(_args, ctx, pi) {
|
|
21
|
+
if (isAutoActive()) {
|
|
22
|
+
ctx.ui.notify("Cannot rethink while auto-mode is active. Stop auto-mode first.", "error");
|
|
23
|
+
return;
|
|
24
|
+
}
|
|
25
|
+
const basePath = process.cwd();
|
|
26
|
+
const root = gsdRoot(basePath);
|
|
27
|
+
if (!existsSync(root)) {
|
|
28
|
+
ctx.ui.notify("No GSD project found. Run /gsd init first.", "warning");
|
|
29
|
+
return;
|
|
30
|
+
}
|
|
31
|
+
ctx.ui.notify("Building project snapshot for rethink...", "info");
|
|
32
|
+
const state = await deriveState(basePath);
|
|
33
|
+
const milestoneIds = findMilestoneIds(basePath);
|
|
34
|
+
if (milestoneIds.length === 0) {
|
|
35
|
+
ctx.ui.notify("No milestones exist yet. Nothing to rethink.", "warning");
|
|
36
|
+
return;
|
|
37
|
+
}
|
|
38
|
+
const queueOrder = loadQueueOrder(basePath);
|
|
39
|
+
const rethinkData = buildRethinkData(basePath, milestoneIds, state, queueOrder);
|
|
40
|
+
const existingMilestonesContext = await buildExistingMilestonesContext(basePath, milestoneIds, state);
|
|
41
|
+
const content = loadPrompt("rethink", {
|
|
42
|
+
rethinkData,
|
|
43
|
+
existingMilestonesContext,
|
|
44
|
+
});
|
|
45
|
+
pi.sendMessage({ customType: "gsd-rethink", content, display: false }, { triggerTurn: true });
|
|
46
|
+
}
|
|
47
|
+
// ─── Data Builder ─────────────────────────────────────────────────────────────
|
|
48
|
+
function buildRethinkData(basePath, milestoneIds, state, queueOrder) {
|
|
49
|
+
const lines = [];
|
|
50
|
+
const dbAvailable = isDbAvailable();
|
|
51
|
+
// ── Summary stats ───────────────────────────────────────────────────
|
|
52
|
+
const counts = { complete: 0, active: 0, pending: 0, parked: 0 };
|
|
53
|
+
for (const entry of state.registry) {
|
|
54
|
+
if (entry.status in counts)
|
|
55
|
+
counts[entry.status]++;
|
|
56
|
+
}
|
|
57
|
+
lines.push("### Summary");
|
|
58
|
+
lines.push(`${counts.complete} complete, ${counts.active} active, ${counts.pending} pending, ${counts.parked} parked — ${milestoneIds.length} total`);
|
|
59
|
+
lines.push(`Queue order source: ${queueOrder ? "explicit QUEUE-ORDER.json" : "default numeric (by ID)"}`);
|
|
60
|
+
if (state.activeMilestone) {
|
|
61
|
+
lines.push(`Active milestone: ${state.activeMilestone}`);
|
|
62
|
+
}
|
|
63
|
+
lines.push("");
|
|
64
|
+
// ── Milestone table ─────────────────────────────────────────────────
|
|
65
|
+
lines.push("### Execution Order");
|
|
66
|
+
lines.push("");
|
|
67
|
+
lines.push("| # | ID | Title | Status | Dependencies | Slices |");
|
|
68
|
+
lines.push("|---|-----|-------|--------|--------------|--------|");
|
|
69
|
+
for (let i = 0; i < milestoneIds.length; i++) {
|
|
70
|
+
const mid = milestoneIds[i];
|
|
71
|
+
const entry = state.registry.find(m => m.id === mid);
|
|
72
|
+
const title = entry?.title ?? mid;
|
|
73
|
+
const status = entry?.status ?? "unknown";
|
|
74
|
+
const deps = entry?.dependsOn?.length ? entry.dependsOn.join(", ") : "—";
|
|
75
|
+
let sliceInfo = "—";
|
|
76
|
+
if (dbAvailable && status !== "complete") {
|
|
77
|
+
const slices = getMilestoneSlices(mid);
|
|
78
|
+
if (slices.length > 0) {
|
|
79
|
+
const done = slices.filter(s => s.status === "complete").length;
|
|
80
|
+
sliceInfo = `${done}/${slices.length} complete`;
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
// Add parked reason if applicable
|
|
84
|
+
let statusDisplay = status;
|
|
85
|
+
if (status === "parked") {
|
|
86
|
+
const reason = getParkedReason(basePath, mid);
|
|
87
|
+
if (reason)
|
|
88
|
+
statusDisplay = `parked (${reason})`;
|
|
89
|
+
}
|
|
90
|
+
lines.push(`| ${i + 1} | ${mid} | ${title} | ${statusDisplay} | ${deps} | ${sliceInfo} |`);
|
|
91
|
+
}
|
|
92
|
+
// ── Dependency validation ───────────────────────────────────────────
|
|
93
|
+
const pendingIds = milestoneIds.filter(mid => {
|
|
94
|
+
const entry = state.registry.find(m => m.id === mid);
|
|
95
|
+
return entry?.status !== "complete";
|
|
96
|
+
});
|
|
97
|
+
const completedIds = new Set(state.registry.filter(m => m.status === "complete").map(m => m.id));
|
|
98
|
+
const depsMap = new Map();
|
|
99
|
+
for (const entry of state.registry) {
|
|
100
|
+
if (entry.dependsOn?.length) {
|
|
101
|
+
depsMap.set(entry.id, entry.dependsOn);
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
if (pendingIds.length > 0 && depsMap.size > 0) {
|
|
105
|
+
const validation = validateQueueOrder(pendingIds, depsMap, completedIds);
|
|
106
|
+
if (validation.violations.length > 0) {
|
|
107
|
+
lines.push("");
|
|
108
|
+
lines.push("### Dependency Issues");
|
|
109
|
+
for (const v of validation.violations) {
|
|
110
|
+
lines.push(`- **${v.type}**: ${v.message}`);
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
return lines.join("\n");
|
|
115
|
+
}
|
|
@@ -4,12 +4,12 @@
|
|
|
4
4
|
import { parseRoadmap, parsePlan, } from './parsers-legacy.js';
|
|
5
5
|
import { parseSummary, loadFile, parseRequirementCounts, parseContextDependsOn, } from './files.js';
|
|
6
6
|
import { resolveMilestoneFile, resolveSlicePath, resolveSliceFile, resolveTaskFile, resolveTasksDir, resolveGsdRootFile, gsdRoot, } from './paths.js';
|
|
7
|
-
import { findMilestoneIds } from './milestone-ids.js';
|
|
7
|
+
import { milestoneIdSort, findMilestoneIds } from './milestone-ids.js';
|
|
8
8
|
import { nativeBatchParseGsdFiles } from './native-parser-bridge.js';
|
|
9
9
|
import { join, resolve } from 'path';
|
|
10
10
|
import { existsSync, readdirSync } from 'node:fs';
|
|
11
11
|
import { debugCount, debugTime } from './debug-logger.js';
|
|
12
|
-
import { isDbAvailable, getAllMilestones, getMilestoneSlices, getSliceTasks, getReplanHistory, getSlice, } from './gsd-db.js';
|
|
12
|
+
import { isDbAvailable, getAllMilestones, getMilestoneSlices, getSliceTasks, getReplanHistory, getSlice, insertMilestone, } from './gsd-db.js';
|
|
13
13
|
/**
|
|
14
14
|
* A "ghost" milestone directory contains only META.json (and no substantive
|
|
15
15
|
* files like CONTEXT, CONTEXT-DRAFT, ROADMAP, or SUMMARY). These appear when
|
|
@@ -192,7 +192,45 @@ function isStatusDone(status) {
|
|
|
192
192
|
*/
|
|
193
193
|
export async function deriveStateFromDb(basePath) {
|
|
194
194
|
const requirements = parseRequirementCounts(await loadFile(resolveGsdRootFile(basePath, "REQUIREMENTS")));
|
|
195
|
-
|
|
195
|
+
let allMilestones = getAllMilestones();
|
|
196
|
+
// Incremental disk→DB sync: milestone directories created outside the DB
|
|
197
|
+
// write path (via /gsd queue, manual mkdir, or complete-milestone writing the
|
|
198
|
+
// next CONTEXT.md) are never inserted by the initial migration guard in
|
|
199
|
+
// auto-start.ts because that guard only runs when gsd.db doesn't exist yet.
|
|
200
|
+
// Reconcile here so deriveStateFromDb never silently misses queued milestones.
|
|
201
|
+
// insertMilestone uses INSERT OR IGNORE, so this is safe to call every time.
|
|
202
|
+
const dbIdSet = new Set(allMilestones.map(m => m.id));
|
|
203
|
+
const diskIds = findMilestoneIds(basePath);
|
|
204
|
+
let synced = false;
|
|
205
|
+
for (const diskId of diskIds) {
|
|
206
|
+
if (!dbIdSet.has(diskId) && !isGhostMilestone(basePath, diskId)) {
|
|
207
|
+
insertMilestone({ id: diskId, status: 'active' });
|
|
208
|
+
synced = true;
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
if (synced)
|
|
212
|
+
allMilestones = getAllMilestones();
|
|
213
|
+
// Reconcile: discover milestones that exist on disk but are missing from
|
|
214
|
+
// the DB. This happens when milestones were created before the DB migration
|
|
215
|
+
// or were manually added to the filesystem. Without this, disk-only
|
|
216
|
+
// milestones are invisible after migration (#2416).
|
|
217
|
+
const dbMilestoneIds = new Set(allMilestones.map(m => m.id));
|
|
218
|
+
const diskMilestoneIds = findMilestoneIds(basePath);
|
|
219
|
+
for (const diskId of diskMilestoneIds) {
|
|
220
|
+
if (!dbMilestoneIds.has(diskId)) {
|
|
221
|
+
// Synthesize a minimal MilestoneRow for the disk-only milestone.
|
|
222
|
+
// Title and status will be resolved from disk files in the loop below.
|
|
223
|
+
allMilestones.push({
|
|
224
|
+
id: diskId,
|
|
225
|
+
title: diskId,
|
|
226
|
+
status: 'active',
|
|
227
|
+
depends_on: [],
|
|
228
|
+
created_at: new Date().toISOString(),
|
|
229
|
+
});
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
// Re-sort so milestones are in canonical order after injection
|
|
233
|
+
allMilestones.sort((a, b) => milestoneIdSort(a.id, b.id));
|
|
196
234
|
// Parallel worker isolation: when locked, filter to just the locked milestone
|
|
197
235
|
const milestoneLock = process.env.GSD_MILESTONE_LOCK;
|
|
198
236
|
const milestones = milestoneLock
|