gsd-pi 2.50.0 → 2.51.0-dev.7d435fe
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 +6 -4
- package/dist/cli.js +26 -0
- package/dist/headless-events.d.ts +18 -0
- package/dist/headless-events.js +36 -0
- package/dist/headless-types.d.ts +28 -0
- package/dist/headless-types.js +7 -0
- package/dist/headless.d.ts +8 -3
- package/dist/headless.js +47 -16
- package/dist/help-text.js +16 -5
- package/dist/loader.js +4 -0
- package/dist/onboarding.js +5 -4
- package/dist/remote-questions-config.js +1 -1
- package/dist/resource-loader.d.ts +4 -1
- package/dist/resource-loader.js +138 -3
- package/dist/resources/extensions/async-jobs/async-bash-tool.js +32 -18
- package/dist/resources/extensions/bg-shell/interaction.js +3 -1
- package/dist/resources/extensions/bg-shell/process-manager.js +4 -1
- package/dist/resources/extensions/claude-code-cli/partial-builder.js +5 -0
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +18 -19
- package/dist/resources/extensions/gsd/auto/phases.js +16 -1
- package/dist/resources/extensions/gsd/auto/session.js +5 -0
- package/dist/resources/extensions/gsd/auto-dashboard.js +22 -3
- package/dist/resources/extensions/gsd/auto-dispatch.js +98 -53
- package/dist/resources/extensions/gsd/auto-observability.js +54 -0
- package/dist/resources/extensions/gsd/auto-post-unit.js +37 -76
- package/dist/resources/extensions/gsd/auto-prompts.js +57 -0
- package/dist/resources/extensions/gsd/auto-recovery.js +21 -25
- package/dist/resources/extensions/gsd/auto-start.js +2 -0
- package/dist/resources/extensions/gsd/auto-timers.js +24 -2
- package/dist/resources/extensions/gsd/auto-tool-tracking.js +25 -7
- package/dist/resources/extensions/gsd/auto-utils.js +20 -0
- package/dist/resources/extensions/gsd/auto-worktree.js +21 -0
- package/dist/resources/extensions/gsd/auto.js +19 -7
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +95 -69
- package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +12 -2
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +17 -1
- package/dist/resources/extensions/gsd/claude-import.js +60 -9
- package/dist/resources/extensions/gsd/commands/handlers/auto.js +69 -6
- package/dist/resources/extensions/gsd/commands-config.js +10 -5
- package/dist/resources/extensions/gsd/commands-prefs-wizard.js +1 -1
- package/dist/resources/extensions/gsd/crash-recovery.js +6 -2
- package/dist/resources/extensions/gsd/custom-execution-policy.js +3 -2
- package/dist/resources/extensions/gsd/custom-verification.js +3 -1
- package/dist/resources/extensions/gsd/custom-workflow-engine.js +3 -2
- package/dist/resources/extensions/gsd/dashboard-overlay.js +4 -0
- package/dist/resources/extensions/gsd/detection.js +595 -9
- package/dist/resources/extensions/gsd/dispatch-guard.js +2 -1
- package/dist/resources/extensions/gsd/docs/preferences-reference.md +18 -3
- package/dist/resources/extensions/gsd/error-classifier.js +105 -0
- package/dist/resources/extensions/gsd/files.js +5 -1
- package/dist/resources/extensions/gsd/gitignore.js +7 -7
- package/dist/resources/extensions/gsd/gsd-db.js +298 -45
- package/dist/resources/extensions/gsd/guided-flow.js +10 -0
- package/dist/resources/extensions/gsd/init-wizard.js +11 -3
- package/dist/resources/extensions/gsd/key-manager.js +7 -16
- package/dist/resources/extensions/gsd/memory-store.js +28 -13
- package/dist/resources/extensions/gsd/milestone-actions.js +19 -0
- package/dist/resources/extensions/gsd/model-router.js +25 -0
- package/dist/resources/extensions/gsd/notifications.js +23 -0
- package/dist/resources/extensions/gsd/observability-validator.js +422 -0
- package/dist/resources/extensions/gsd/preferences-models.js +1 -13
- package/dist/resources/extensions/gsd/preferences-skills.js +11 -5
- package/dist/resources/extensions/gsd/preferences-types.js +1 -0
- package/dist/resources/extensions/gsd/preferences-validation.js +25 -0
- package/dist/resources/extensions/gsd/preferences.js +16 -13
- package/dist/resources/extensions/gsd/prompts/forensics.md +11 -7
- package/dist/resources/extensions/gsd/prompts/system.md +1 -1
- package/dist/resources/extensions/gsd/prompts/validate-milestone.md +6 -1
- package/dist/resources/extensions/gsd/provider-error-pause.js +0 -36
- package/dist/resources/extensions/gsd/roadmap-mutations.js +110 -0
- package/dist/resources/extensions/gsd/rtk-status.js +43 -0
- package/dist/resources/extensions/gsd/rule-registry.js +11 -12
- package/dist/resources/extensions/gsd/service-tier.js +13 -2
- package/dist/resources/extensions/gsd/session-forensics.js +7 -3
- package/dist/resources/extensions/gsd/skill-catalog.js +1026 -0
- package/dist/resources/extensions/gsd/skill-discovery.js +3 -2
- package/dist/resources/extensions/gsd/skill-health.js +2 -2
- package/dist/resources/extensions/gsd/skill-telemetry.js +15 -5
- package/dist/resources/extensions/gsd/state.js +38 -7
- package/dist/resources/extensions/gsd/templates/{preferences.md → PREFERENCES.md} +2 -0
- package/dist/resources/extensions/gsd/templates/milestone-validation.md +12 -0
- package/dist/resources/extensions/gsd/tests/dist-redirect.mjs +28 -9
- package/dist/resources/extensions/gsd/tools/complete-milestone.js +3 -10
- package/dist/resources/extensions/gsd/tools/complete-slice.js +3 -17
- package/dist/resources/extensions/gsd/tools/complete-task.js +7 -18
- package/dist/resources/extensions/gsd/tools/plan-milestone.js +26 -17
- package/dist/resources/extensions/gsd/tools/plan-slice.js +25 -14
- package/dist/resources/extensions/gsd/tools/plan-task.js +21 -11
- package/dist/resources/extensions/gsd/tools/reassess-roadmap.js +47 -37
- package/dist/resources/extensions/gsd/tools/replan-slice.js +49 -38
- package/dist/resources/extensions/gsd/tools/validate-milestone.js +23 -16
- package/dist/resources/extensions/gsd/undo.js +8 -7
- package/dist/resources/extensions/gsd/unit-runtime.js +2 -1
- package/dist/resources/extensions/gsd/verification-gate.js +3 -1
- package/dist/resources/extensions/gsd/workflow-logger.js +0 -1
- package/dist/resources/extensions/remote-questions/config.js +1 -1
- package/dist/resources/extensions/remote-questions/remote-command.js +1 -1
- package/dist/resources/extensions/search-the-web/native-search.js +1 -1
- package/dist/resources/extensions/search-the-web/provider.js +1 -1
- package/dist/resources/extensions/shared/rtk-session-stats.js +189 -0
- package/dist/resources/extensions/shared/rtk.js +100 -0
- package/dist/rtk.d.ts +52 -0
- package/dist/rtk.js +332 -0
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +18 -17
- package/dist/web/standalone/.next/build-manifest.json +4 -4
- 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/required-server-files.json +3 -3
- package/dist/web/standalone/.next/routes-manifest.json +6 -0
- package/dist/web/standalone/.next/server/app/_global-error/page.js +3 -3
- package/dist/web/standalone/.next/server/app/_global-error/page.js.nft.json +1 -1
- 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 +2 -2
- package/dist/web/standalone/.next/server/app/_not-found/page.js.nft.json +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 +4 -4
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +4 -4
- 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 +4 -4
- 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.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/boot/route_client-reference-manifest.js +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/input/route_client-reference-manifest.js +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/resize/route_client-reference-manifest.js +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/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/browse-directories/route.js.nft.json +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.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/captures/route_client-reference-manifest.js +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/cleanup/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/dev-mode/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/dev-mode/route.js.nft.json +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.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/doctor/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/experimental/route.js +3 -0
- package/dist/web/standalone/.next/server/app/api/experimental/route.js.nft.json +1 -0
- package/dist/web/standalone/.next/server/app/api/experimental/route_client-reference-manifest.js +1 -0
- 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/export-data/route_client-reference-manifest.js +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/files/route_client-reference-manifest.js +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/forensics/route_client-reference-manifest.js +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/git/route_client-reference-manifest.js +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/history/route_client-reference-manifest.js +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/hooks/route_client-reference-manifest.js +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/inspect/route_client-reference-manifest.js +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/knowledge/route_client-reference-manifest.js +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/live-state/route_client-reference-manifest.js +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/onboarding/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/preferences/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/preferences/route.js.nft.json +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.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/projects/route_client-reference-manifest.js +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/recovery/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/remote-questions/route.js +2 -26
- package/dist/web/standalone/.next/server/app/api/remote-questions/route.js.nft.json +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.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/browser/route_client-reference-manifest.js +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/command/route_client-reference-manifest.js +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/events/route_client-reference-manifest.js +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/session/manage/route_client-reference-manifest.js +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/settings-data/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/shutdown/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/shutdown/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +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/skill-health/route_client-reference-manifest.js +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/steer/route_client-reference-manifest.js +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/switch-root/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +2 -2
- package/dist/web/standalone/.next/server/app/api/terminal/input/route.js.nft.json +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.js +2 -2
- package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js.nft.json +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.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/sessions/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +4 -4
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +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.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/upload/route.js.nft.json +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.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/undo/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/update/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +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/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 +5 -5
- 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 +5 -5
- 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 +4 -4
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/page.js +2 -2
- package/dist/web/standalone/.next/server/app/page.js.nft.json +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 +18 -17
- package/dist/web/standalone/.next/server/chunks/2229.js +12 -0
- package/dist/web/standalone/.next/server/chunks/2331.js +25 -0
- package/dist/web/standalone/.next/server/chunks/{741.js → 4741.js} +1 -1
- package/dist/web/standalone/.next/server/chunks/5822.js +2 -0
- package/dist/web/standalone/.next/server/chunks/7471.js +13 -0
- package/dist/web/standalone/.next/server/functions-config-manifest.json +1 -0
- 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/middleware.js +3 -3
- package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
- package/dist/web/standalone/.next/server/next-font-manifest.json +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/server/webpack-runtime.js +1 -1
- package/dist/web/standalone/.next/static/RqOU-jOv9uZ1Q03P6L6nn/_buildManifest.js +1 -0
- package/dist/web/standalone/.next/static/chunks/{485.243af25f0cdf50d6.js → 2008.817d0885545aaea9.js} +6 -6
- package/dist/web/standalone/.next/static/chunks/4024.21054f459af5cc78.js +9 -0
- package/dist/web/standalone/.next/static/chunks/app/_global-error/page-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/_not-found/{page-2f24283c162b6ab3.js → page-f2a7482d42a5614b.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/app/api/boot/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/bridge-terminal/input/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/bridge-terminal/resize/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/bridge-terminal/stream/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/browse-directories/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/captures/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/cleanup/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/dev-mode/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/doctor/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/experimental/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/export-data/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/files/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/forensics/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/git/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/history/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/hooks/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/inspect/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/knowledge/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/live-state/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/onboarding/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/preferences/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/projects/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/recovery/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/remote-questions/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/session/browser/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/session/command/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/session/events/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/session/manage/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/settings-data/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/shutdown/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/skill-health/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/steer/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/switch-root/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/terminal/input/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/terminal/resize/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/terminal/sessions/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/terminal/stream/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/terminal/upload/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/undo/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/update/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/api/visualizer/route-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/app/{layout-9ecfd95f343793f0.js → layout-a16c7a7ecdf0c2cf.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/app/page-b950e4e384cc62b3.js +1 -0
- package/dist/web/standalone/.next/static/chunks/main-app-fdab67f7802d7832.js +1 -0
- package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/app-error-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/forbidden-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-459824ffb8c323dd.js +1 -0
- package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/not-found-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/unauthorized-c4cc189e7b117ea2.js +1 -0
- package/dist/web/standalone/.next/static/chunks/webpack-024d82be84800e52.js +1 -0
- package/dist/web/standalone/.next/static/css/a58ef8a151aa0493.css +1 -0
- package/dist/web/standalone/node_modules/node-pty/build/Makefile +2 -2
- package/dist/web/standalone/node_modules/node-pty/build/Release/pty.node +0 -0
- package/dist/web/standalone/node_modules/node-pty/build/pty.target.mk +14 -14
- package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api.target.mk +14 -14
- package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_except.target.mk +14 -14
- package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_maybe.target.mk +14 -14
- package/dist/web/standalone/server.js +1 -1
- package/dist/wizard.js +4 -1
- package/package.json +2 -2
- package/packages/pi-ai/dist/models.d.ts +14 -3
- package/packages/pi-ai/dist/models.d.ts.map +1 -1
- package/packages/pi-ai/dist/models.js +53 -10
- package/packages/pi-ai/dist/models.js.map +1 -1
- package/packages/pi-ai/dist/models.test.js +102 -1
- package/packages/pi-ai/dist/models.test.js.map +1 -1
- package/packages/pi-ai/dist/types.d.ts +31 -0
- 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/models.test.ts +114 -1
- package/packages/pi-ai/src/models.ts +70 -13
- package/packages/pi-ai/src/types.ts +32 -1
- package/packages/pi-coding-agent/dist/cli/args.d.ts +2 -0
- package/packages/pi-coding-agent/dist/cli/args.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/cli/args.js +3 -0
- package/packages/pi-coding-agent/dist/cli/args.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session.d.ts +1 -0
- package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session.js +1 -0
- package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/bash-executor.d.ts +3 -1
- package/packages/pi-coding-agent/dist/core/bash-executor.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/bash-executor.js +15 -2
- package/packages/pi-coding-agent/dist/core/bash-executor.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/index.d.ts +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/index.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/index.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts +1 -0
- package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/runner.js +13 -0
- package/packages/pi-coding-agent/dist/core/extensions/runner.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +19 -1
- 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/extensions/wrapper.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/wrapper.js +8 -0
- package/packages/pi-coding-agent/dist/core/extensions/wrapper.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry.js +9 -4
- package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/package-manager.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/package-manager.js +8 -1
- package/packages/pi-coding-agent/dist/core/package-manager.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/skills.d.ts +11 -1
- package/packages/pi-coding-agent/dist/core/skills.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/skills.js +30 -8
- package/packages/pi-coding-agent/dist/core/skills.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/slash-commands.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/slash-commands.js +1 -0
- package/packages/pi-coding-agent/dist/core/slash-commands.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/tools/bash-spawn-windows.test.d.ts +19 -0
- package/packages/pi-coding-agent/dist/core/tools/bash-spawn-windows.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/tools/bash-spawn-windows.test.js +83 -0
- package/packages/pi-coding-agent/dist/core/tools/bash-spawn-windows.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/tools/bash.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/tools/bash.js +5 -1
- package/packages/pi-coding-agent/dist/core/tools/bash.js.map +1 -1
- package/packages/pi-coding-agent/dist/index.d.ts +3 -3
- package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/index.js +1 -1
- package/packages/pi-coding-agent/dist/index.js.map +1 -1
- package/packages/pi-coding-agent/dist/main.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/main.js +5 -3
- package/packages/pi-coding-agent/dist/main.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/index.d.ts +1 -1
- package/packages/pi-coding-agent/dist/modes/index.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/index.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/footer.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js +3 -2
- package/packages/pi-coding-agent/dist/modes/interactive/components/footer.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 +0 -2
- 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 +6 -4
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.d.ts +5 -0
- package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.js +12 -0
- package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.js +2 -1
- package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-client.d.ts +28 -1
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-client.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-client.js +49 -0
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-client.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.d.ts +1 -1
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js +114 -6
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-protocol-v2.test.d.ts +9 -0
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-protocol-v2.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-protocol-v2.test.js +831 -0
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-protocol-v2.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.d.ts +66 -0
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.js.map +1 -1
- package/packages/pi-coding-agent/dist/utils/shell.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/utils/shell.js +0 -1
- package/packages/pi-coding-agent/dist/utils/shell.js.map +1 -1
- package/packages/pi-coding-agent/package.json +1 -1
- package/packages/pi-coding-agent/src/cli/args.ts +4 -0
- package/packages/pi-coding-agent/src/core/agent-session.ts +2 -1
- package/packages/pi-coding-agent/src/core/bash-executor.ts +15 -3
- package/packages/pi-coding-agent/src/core/extensions/index.ts +2 -0
- package/packages/pi-coding-agent/src/core/extensions/runner.ts +18 -0
- package/packages/pi-coding-agent/src/core/extensions/types.ts +21 -0
- package/packages/pi-coding-agent/src/core/extensions/wrapper.ts +9 -0
- package/packages/pi-coding-agent/src/core/model-registry.ts +10 -3
- package/packages/pi-coding-agent/src/core/package-manager.ts +10 -1
- package/packages/pi-coding-agent/src/core/skills.ts +35 -10
- package/packages/pi-coding-agent/src/core/slash-commands.ts +1 -0
- package/packages/pi-coding-agent/src/core/tools/bash-spawn-windows.test.ts +101 -0
- package/packages/pi-coding-agent/src/core/tools/bash.ts +5 -1
- package/packages/pi-coding-agent/src/index.ts +7 -0
- package/packages/pi-coding-agent/src/main.ts +5 -3
- package/packages/pi-coding-agent/src/modes/index.ts +8 -1
- package/packages/pi-coding-agent/src/modes/interactive/components/footer.ts +3 -2
- package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +0 -2
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +6 -4
- package/packages/pi-coding-agent/src/modes/interactive/slash-command-handlers.ts +15 -0
- package/packages/pi-coding-agent/src/modes/interactive/theme/themes.ts +2 -1
- package/packages/pi-coding-agent/src/modes/rpc/rpc-client.ts +54 -1
- package/packages/pi-coding-agent/src/modes/rpc/rpc-mode.ts +124 -6
- package/packages/pi-coding-agent/src/modes/rpc/rpc-protocol-v2.test.ts +971 -0
- package/packages/pi-coding-agent/src/modes/rpc/rpc-types.ts +61 -4
- package/packages/pi-coding-agent/src/utils/shell.ts +0 -1
- package/pkg/dist/modes/interactive/theme/themes.d.ts.map +1 -1
- package/pkg/dist/modes/interactive/theme/themes.js +2 -1
- package/pkg/dist/modes/interactive/theme/themes.js.map +1 -1
- package/pkg/package.json +1 -1
- package/scripts/postinstall.js +163 -6
- package/src/resources/extensions/async-jobs/async-bash-tool.ts +25 -12
- package/src/resources/extensions/bg-shell/interaction.ts +3 -1
- package/src/resources/extensions/bg-shell/process-manager.ts +4 -1
- package/src/resources/extensions/claude-code-cli/partial-builder.ts +5 -0
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +19 -20
- package/src/resources/extensions/claude-code-cli/tests/partial-builder.test.ts +105 -0
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +21 -0
- package/src/resources/extensions/gsd/auto/phases.ts +15 -1
- package/src/resources/extensions/gsd/auto/session.ts +6 -0
- package/src/resources/extensions/gsd/auto-dashboard.ts +31 -3
- package/src/resources/extensions/gsd/auto-dispatch.ts +102 -50
- package/src/resources/extensions/gsd/auto-observability.ts +72 -0
- package/src/resources/extensions/gsd/auto-post-unit.ts +56 -83
- package/src/resources/extensions/gsd/auto-prompts.ts +48 -0
- package/src/resources/extensions/gsd/auto-recovery.ts +19 -26
- package/src/resources/extensions/gsd/auto-start.ts +2 -0
- package/src/resources/extensions/gsd/auto-timers.ts +25 -1
- package/src/resources/extensions/gsd/auto-tool-tracking.ts +30 -6
- package/src/resources/extensions/gsd/auto-utils.ts +25 -0
- package/src/resources/extensions/gsd/auto-worktree.ts +21 -0
- package/src/resources/extensions/gsd/auto.ts +20 -7
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +115 -72
- package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +11 -2
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +18 -1
- package/src/resources/extensions/gsd/claude-import.ts +58 -9
- package/src/resources/extensions/gsd/commands/handlers/auto.ts +73 -6
- package/src/resources/extensions/gsd/commands-config.ts +11 -5
- package/src/resources/extensions/gsd/commands-prefs-wizard.ts +1 -1
- package/src/resources/extensions/gsd/crash-recovery.ts +6 -2
- package/src/resources/extensions/gsd/custom-execution-policy.ts +3 -2
- package/src/resources/extensions/gsd/custom-verification.ts +3 -1
- package/src/resources/extensions/gsd/custom-workflow-engine.ts +3 -2
- package/src/resources/extensions/gsd/dashboard-overlay.ts +7 -0
- package/src/resources/extensions/gsd/detection.ts +668 -9
- package/src/resources/extensions/gsd/dispatch-guard.ts +2 -1
- package/src/resources/extensions/gsd/docs/preferences-reference.md +18 -3
- package/src/resources/extensions/gsd/error-classifier.ts +139 -0
- package/src/resources/extensions/gsd/files.ts +6 -1
- package/src/resources/extensions/gsd/gitignore.ts +7 -7
- package/src/resources/extensions/gsd/gsd-db.ts +355 -63
- package/src/resources/extensions/gsd/guided-flow.ts +11 -0
- package/src/resources/extensions/gsd/init-wizard.ts +11 -3
- package/src/resources/extensions/gsd/key-manager.ts +7 -16
- package/src/resources/extensions/gsd/memory-store.ts +29 -18
- package/src/resources/extensions/gsd/milestone-actions.ts +17 -0
- package/src/resources/extensions/gsd/model-router.ts +25 -0
- package/src/resources/extensions/gsd/notifications.ts +23 -0
- package/src/resources/extensions/gsd/observability-validator.ts +456 -0
- package/src/resources/extensions/gsd/preferences-models.ts +1 -13
- package/src/resources/extensions/gsd/preferences-skills.ts +11 -5
- package/src/resources/extensions/gsd/preferences-types.ts +20 -0
- package/src/resources/extensions/gsd/preferences-validation.ts +26 -0
- package/src/resources/extensions/gsd/preferences.ts +15 -13
- package/src/resources/extensions/gsd/prompts/forensics.md +11 -7
- package/src/resources/extensions/gsd/prompts/system.md +1 -1
- package/src/resources/extensions/gsd/prompts/validate-milestone.md +6 -1
- package/src/resources/extensions/gsd/provider-error-pause.ts +0 -48
- package/src/resources/extensions/gsd/roadmap-mutations.ts +134 -0
- package/src/resources/extensions/gsd/rtk-status.ts +53 -0
- package/src/resources/extensions/gsd/rule-registry.ts +11 -12
- package/src/resources/extensions/gsd/service-tier.ts +14 -2
- package/src/resources/extensions/gsd/session-forensics.ts +7 -3
- package/src/resources/extensions/gsd/skill-catalog.ts +1085 -0
- package/src/resources/extensions/gsd/skill-discovery.ts +3 -2
- package/src/resources/extensions/gsd/skill-health.ts +2 -2
- package/src/resources/extensions/gsd/skill-telemetry.ts +15 -5
- package/src/resources/extensions/gsd/state.ts +38 -7
- package/src/resources/extensions/gsd/templates/{preferences.md → PREFERENCES.md} +2 -0
- package/src/resources/extensions/gsd/templates/milestone-validation.md +12 -0
- package/src/resources/extensions/gsd/tests/auto-dashboard.test.ts +16 -0
- package/src/resources/extensions/gsd/tests/auto-lock-creation.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +68 -0
- package/src/resources/extensions/gsd/tests/auto-milestone-target.test.ts +61 -0
- package/src/resources/extensions/gsd/tests/claude-import-marketplace-discovery.test.ts +191 -0
- package/src/resources/extensions/gsd/tests/claude-import-tui.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/commands-config.test.ts +24 -0
- package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/complete-task-rollback-evidence.test.ts +106 -0
- package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/crash-recovery.test.ts +5 -3
- package/src/resources/extensions/gsd/tests/custom-verification.test.ts +33 -0
- package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +39 -10
- package/src/resources/extensions/gsd/tests/detection.test.ts +839 -1
- package/src/resources/extensions/gsd/tests/dist-redirect.mjs +28 -9
- package/src/resources/extensions/gsd/tests/doctor-git.test.ts +4 -4
- package/src/resources/extensions/gsd/tests/doctor-proactive.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/double-merge-guard.test.ts +97 -0
- package/src/resources/extensions/gsd/tests/empty-db-reconciliation.test.ts +79 -0
- package/src/resources/extensions/gsd/tests/git-service.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/gsd-db.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/idle-recovery.test.ts +63 -0
- package/src/resources/extensions/gsd/tests/idle-watchdog-stall-override.test.ts +125 -0
- package/src/resources/extensions/gsd/tests/init-wizard.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/interactive-tool-idle-exemption.test.ts +119 -0
- package/src/resources/extensions/gsd/tests/key-manager.test.ts +16 -1
- package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/model-router.test.ts +40 -0
- package/src/resources/extensions/gsd/tests/none-mode-gates.test.ts +7 -7
- package/src/resources/extensions/gsd/tests/notifications.test.ts +28 -6
- package/src/resources/extensions/gsd/tests/park-db-sync.test.ts +85 -0
- package/src/resources/extensions/gsd/tests/plan-quality-validator.test.ts +474 -0
- package/src/resources/extensions/gsd/tests/preferences-worktree-sync.test.ts +91 -0
- package/src/resources/extensions/gsd/tests/preferences.test.ts +51 -1
- package/src/resources/extensions/gsd/tests/provider-errors.test.ts +77 -70
- package/src/resources/extensions/gsd/tests/queue-reorder-e2e.test.ts +40 -0
- package/src/resources/extensions/gsd/tests/reactive-executor.test.ts +6 -6
- package/src/resources/extensions/gsd/tests/remediation-completion-guard.test.ts +110 -0
- package/src/resources/extensions/gsd/tests/remote-questions.test.ts +29 -0
- package/src/resources/extensions/gsd/tests/retry-diagnostic-reasoning.test.ts +161 -0
- package/src/resources/extensions/gsd/tests/retry-state-reset.test.ts +4 -5
- package/src/resources/extensions/gsd/tests/rewrite-count-persist.test.ts +82 -0
- package/src/resources/extensions/gsd/tests/run-uat.test.ts +146 -0
- package/src/resources/extensions/gsd/tests/sidecar-queue.test.ts +12 -12
- package/src/resources/extensions/gsd/tests/skill-catalog.test.ts +193 -0
- package/src/resources/extensions/gsd/tests/terminated-transient.test.ts +56 -21
- package/src/resources/extensions/gsd/tests/token-cost-display.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/triage-dispatch.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/vacuous-truth-slices.test.ts +115 -0
- package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +90 -0
- package/src/resources/extensions/gsd/tests/workflow-logger.test.ts +81 -1
- package/src/resources/extensions/gsd/tests/worktree-preferences-sync.test.ts +130 -0
- package/src/resources/extensions/gsd/tools/complete-milestone.ts +3 -14
- package/src/resources/extensions/gsd/tools/complete-slice.ts +3 -21
- package/src/resources/extensions/gsd/tools/complete-task.ts +9 -22
- package/src/resources/extensions/gsd/tools/plan-milestone.ts +28 -18
- package/src/resources/extensions/gsd/tools/plan-slice.ts +28 -16
- package/src/resources/extensions/gsd/tools/plan-task.ts +24 -12
- package/src/resources/extensions/gsd/tools/reassess-roadmap.ts +54 -42
- package/src/resources/extensions/gsd/tools/replan-slice.ts +53 -40
- package/src/resources/extensions/gsd/tools/validate-milestone.ts +26 -20
- package/src/resources/extensions/gsd/types.ts +2 -0
- package/src/resources/extensions/gsd/undo.ts +8 -7
- package/src/resources/extensions/gsd/unit-runtime.ts +2 -1
- package/src/resources/extensions/gsd/verification-gate.ts +3 -1
- package/src/resources/extensions/gsd/workflow-logger.ts +0 -1
- package/src/resources/extensions/remote-questions/config.ts +1 -1
- package/src/resources/extensions/remote-questions/remote-command.ts +1 -1
- package/src/resources/extensions/search-the-web/native-search.ts +1 -1
- package/src/resources/extensions/search-the-web/provider.ts +1 -1
- package/src/resources/extensions/shared/rtk-session-stats.ts +249 -0
- package/src/resources/extensions/shared/rtk.ts +120 -0
- package/dist/web/standalone/.next/server/chunks/229.js +0 -12
- package/dist/web/standalone/.next/server/chunks/441.js +0 -2
- package/dist/web/standalone/.next/server/chunks/471.js +0 -13
- package/dist/web/standalone/.next/static/MQOhBnkC_TLtNn_JvZWDj/_buildManifest.js +0 -1
- package/dist/web/standalone/.next/static/chunks/4024.7c75ac378de0f2b5.js +0 -9
- package/dist/web/standalone/.next/static/chunks/app/_global-error/page-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/boot/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/bridge-terminal/input/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/bridge-terminal/resize/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/bridge-terminal/stream/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/browse-directories/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/captures/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/cleanup/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/dev-mode/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/doctor/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/export-data/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/files/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/forensics/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/git/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/history/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/hooks/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/inspect/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/knowledge/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/live-state/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/onboarding/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/preferences/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/projects/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/recovery/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/remote-questions/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/session/browser/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/session/command/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/session/events/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/session/manage/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/settings-data/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/shutdown/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/skill-health/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/steer/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/switch-root/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/terminal/input/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/terminal/resize/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/terminal/sessions/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/terminal/stream/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/terminal/upload/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/undo/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/update/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/api/visualizer/route-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/app/page-12dd5ece0df4badc.js +0 -1
- package/dist/web/standalone/.next/static/chunks/main-app-d3d4c336195465f9.js +0 -1
- package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/app-error-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/forbidden-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-ab5a8926e07ec673.js +0 -1
- package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/not-found-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/unauthorized-d83ba70a25a85472.js +0 -1
- package/dist/web/standalone/.next/static/chunks/webpack-2473ce2c3879fff4.js +0 -1
- package/dist/web/standalone/.next/static/css/dd4ae3f58ac9b600.css +0 -1
- package/packages/pi-ai/pnpm-lock.yaml +0 -2022
- package/packages/pi-coding-agent/pnpm-lock.yaml +0 -454
- /package/dist/web/standalone/.next/static/{MQOhBnkC_TLtNn_JvZWDj → RqOU-jOv9uZ1Q03P6L6nn}/_ssgManifest.js +0 -0
|
@@ -0,0 +1,456 @@
|
|
|
1
|
+
import { loadFile } from "./files.js";
|
|
2
|
+
import { resolveSliceFile, resolveTaskFile, resolveTasksDir, resolveTaskFiles } from "./paths.js";
|
|
3
|
+
|
|
4
|
+
export interface ValidationIssue {
|
|
5
|
+
severity: "info" | "warning" | "error";
|
|
6
|
+
scope: "slice-plan" | "task-plan" | "task-summary" | "slice-summary";
|
|
7
|
+
file: string;
|
|
8
|
+
ruleId: string;
|
|
9
|
+
message: string;
|
|
10
|
+
suggestion?: string;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
function getSection(content: string, heading: string, level: number = 2): string | null {
|
|
14
|
+
const prefix = "#".repeat(level) + " ";
|
|
15
|
+
const escaped = heading.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
|
|
16
|
+
const regex = new RegExp(`^${prefix}${escaped}\\s*$`, "m");
|
|
17
|
+
const match = regex.exec(content);
|
|
18
|
+
if (!match) return null;
|
|
19
|
+
|
|
20
|
+
const start = match.index + match[0].length;
|
|
21
|
+
const rest = content.slice(start);
|
|
22
|
+
const nextHeading = rest.match(new RegExp(`^#{1,${level}} `, "m"));
|
|
23
|
+
const end = nextHeading ? nextHeading.index! : rest.length;
|
|
24
|
+
return rest.slice(0, end).trim();
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
function getFrontmatter(content: string): string | null {
|
|
28
|
+
const trimmed = content.trimStart();
|
|
29
|
+
if (!trimmed.startsWith("---")) return null;
|
|
30
|
+
const afterFirst = trimmed.indexOf("\n");
|
|
31
|
+
if (afterFirst === -1) return null;
|
|
32
|
+
const rest = trimmed.slice(afterFirst + 1);
|
|
33
|
+
const endIdx = rest.indexOf("\n---");
|
|
34
|
+
if (endIdx === -1) return null;
|
|
35
|
+
return rest.slice(0, endIdx);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
function hasFrontmatterKey(content: string, key: string): boolean {
|
|
39
|
+
const fm = getFrontmatter(content);
|
|
40
|
+
if (!fm) return false;
|
|
41
|
+
return new RegExp(`^${key}:`, "m").test(fm);
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
function normalizeMeaningfulLines(text: string): string[] {
|
|
45
|
+
return text
|
|
46
|
+
.split("\n")
|
|
47
|
+
.map(line => line.trim())
|
|
48
|
+
.filter(line => line.length > 0)
|
|
49
|
+
.filter(line => !line.startsWith("<!--"))
|
|
50
|
+
.filter(line => !line.endsWith("-->"))
|
|
51
|
+
.filter(line => !/^[-*]\s*\{\{.+\}\}$/.test(line))
|
|
52
|
+
.filter(line => !/^\{\{.+\}\}$/.test(line));
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
function sectionLooksPlaceholderOnly(text: string | null): boolean {
|
|
56
|
+
if (!text) return true;
|
|
57
|
+
const lines = normalizeMeaningfulLines(text)
|
|
58
|
+
.map(line => line.replace(/^[-*]\s+/, "").trim())
|
|
59
|
+
.filter(line => line.length > 0);
|
|
60
|
+
|
|
61
|
+
if (lines.length === 0) return true;
|
|
62
|
+
|
|
63
|
+
return lines.every(line => {
|
|
64
|
+
const lower = line.toLowerCase();
|
|
65
|
+
return lower === "none" ||
|
|
66
|
+
lower.endsWith(": none") ||
|
|
67
|
+
lower.includes("{{") ||
|
|
68
|
+
lower.includes("}}") ||
|
|
69
|
+
lower.startsWith("required for non-trivial") ||
|
|
70
|
+
lower.startsWith("describe how a future agent") ||
|
|
71
|
+
lower.startsWith("prefer:") ||
|
|
72
|
+
lower.startsWith("keep this section concise");
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
function textSuggestsObservabilityRelevant(content: string): boolean {
|
|
77
|
+
const lower = content.toLowerCase();
|
|
78
|
+
const needles = [
|
|
79
|
+
" api", "route", "server", "worker", "queue", "job", "sync", "import",
|
|
80
|
+
"webhook", "auth", "db", "database", "migration", "cache", "background",
|
|
81
|
+
"polling", "realtime", "socket", "stateful", "integration", "ui", "form",
|
|
82
|
+
"submit", "status", "service", "pipeline", "health endpoint", "error path"
|
|
83
|
+
];
|
|
84
|
+
return needles.some(needle => lower.includes(needle));
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
function verificationMentionsDiagnostics(section: string | null): boolean {
|
|
88
|
+
if (!section) return false;
|
|
89
|
+
const lower = section.toLowerCase();
|
|
90
|
+
const needles = [
|
|
91
|
+
"error", "failure", "diagnostic", "status", "health", "inspect", "log",
|
|
92
|
+
"network", "console", "retry", "last error", "correlation", "readiness"
|
|
93
|
+
];
|
|
94
|
+
return needles.some(needle => lower.includes(needle));
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
export function validateSlicePlanContent(file: string, content: string): ValidationIssue[] {
|
|
98
|
+
const issues: ValidationIssue[] = [];
|
|
99
|
+
|
|
100
|
+
// ── Plan quality rules (always run, not gated by runtime relevance) ──
|
|
101
|
+
|
|
102
|
+
const tasksSection = getSection(content, "Tasks", 2);
|
|
103
|
+
if (tasksSection) {
|
|
104
|
+
const lines = tasksSection.split("\n");
|
|
105
|
+
const taskLinePattern = /^- \[[ x]\] \*\*T\d+:/;
|
|
106
|
+
const taskLineIndices: number[] = [];
|
|
107
|
+
for (let i = 0; i < lines.length; i++) {
|
|
108
|
+
if (taskLinePattern.test(lines[i])) taskLineIndices.push(i);
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
for (let t = 0; t < taskLineIndices.length; t++) {
|
|
112
|
+
const start = taskLineIndices[t];
|
|
113
|
+
const end = t + 1 < taskLineIndices.length ? taskLineIndices[t + 1] : lines.length;
|
|
114
|
+
// Check lines between this task header and the next (or section end)
|
|
115
|
+
const bodyLines = lines.slice(start + 1, end);
|
|
116
|
+
const meaningful = bodyLines.filter(l => l.trim().length > 0);
|
|
117
|
+
if (meaningful.length === 0) {
|
|
118
|
+
issues.push({
|
|
119
|
+
severity: "warning",
|
|
120
|
+
scope: "slice-plan",
|
|
121
|
+
file,
|
|
122
|
+
ruleId: "empty_task_entry",
|
|
123
|
+
message: "Inline task entry has no description content beneath the checkbox line.",
|
|
124
|
+
suggestion: "Add at least a Why/Files/Do/Verify summary so the task is self-describing.",
|
|
125
|
+
});
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
// ── Observability rules (gated by runtime relevance) ──
|
|
131
|
+
|
|
132
|
+
const relevant = textSuggestsObservabilityRelevant(content);
|
|
133
|
+
if (!relevant) return issues;
|
|
134
|
+
|
|
135
|
+
const obs = getSection(content, "Observability / Diagnostics", 2);
|
|
136
|
+
const verification = getSection(content, "Verification", 2);
|
|
137
|
+
|
|
138
|
+
if (!obs) {
|
|
139
|
+
issues.push({
|
|
140
|
+
severity: "warning",
|
|
141
|
+
scope: "slice-plan",
|
|
142
|
+
file,
|
|
143
|
+
ruleId: "missing_observability_section",
|
|
144
|
+
message: "Slice plan appears non-trivial but is missing `## Observability / Diagnostics`.",
|
|
145
|
+
suggestion: "Add runtime signals, inspection surfaces, failure visibility, and redaction constraints.",
|
|
146
|
+
});
|
|
147
|
+
} else if (sectionLooksPlaceholderOnly(obs)) {
|
|
148
|
+
issues.push({
|
|
149
|
+
severity: "warning",
|
|
150
|
+
scope: "slice-plan",
|
|
151
|
+
file,
|
|
152
|
+
ruleId: "observability_section_placeholder_only",
|
|
153
|
+
message: "Slice plan has `## Observability / Diagnostics` but it still looks like placeholder text.",
|
|
154
|
+
suggestion: "Replace placeholders with concrete signals and inspection surfaces a future agent should trust.",
|
|
155
|
+
});
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
if (!verificationMentionsDiagnostics(verification)) {
|
|
159
|
+
issues.push({
|
|
160
|
+
severity: "warning",
|
|
161
|
+
scope: "slice-plan",
|
|
162
|
+
file,
|
|
163
|
+
ruleId: "verification_missing_diagnostic_check",
|
|
164
|
+
message: "Slice verification does not appear to include any diagnostic or failure-path check.",
|
|
165
|
+
suggestion: "Add at least one verification step for inspectable failure state, structured error output, status surface, or equivalent.",
|
|
166
|
+
});
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
return issues;
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
export function validateTaskPlanContent(file: string, content: string): ValidationIssue[] {
|
|
173
|
+
const issues: ValidationIssue[] = [];
|
|
174
|
+
|
|
175
|
+
// ── Plan quality rules (always run, not gated by runtime relevance) ──
|
|
176
|
+
|
|
177
|
+
// Rule: empty or missing Steps section
|
|
178
|
+
const stepsSection = getSection(content, "Steps", 2);
|
|
179
|
+
if (stepsSection === null || sectionLooksPlaceholderOnly(stepsSection)) {
|
|
180
|
+
issues.push({
|
|
181
|
+
severity: "warning",
|
|
182
|
+
scope: "task-plan",
|
|
183
|
+
file,
|
|
184
|
+
ruleId: "empty_steps_section",
|
|
185
|
+
message: "Task plan has an empty or missing `## Steps` section.",
|
|
186
|
+
suggestion: "Add concrete numbered implementation steps so execution has a clear sequence.",
|
|
187
|
+
});
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
// Rule: placeholder-only Verification section
|
|
191
|
+
const verificationSection = getSection(content, "Verification", 2);
|
|
192
|
+
if (verificationSection !== null && sectionLooksPlaceholderOnly(verificationSection)) {
|
|
193
|
+
issues.push({
|
|
194
|
+
severity: "warning",
|
|
195
|
+
scope: "task-plan",
|
|
196
|
+
file,
|
|
197
|
+
ruleId: "placeholder_verification",
|
|
198
|
+
message: "Task plan has `## Verification` but it still looks like placeholder text.",
|
|
199
|
+
suggestion: "Replace placeholders with concrete verification commands, test runs, or observable checks.",
|
|
200
|
+
});
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
// Rule: scope estimate thresholds
|
|
204
|
+
const fm = getFrontmatter(content);
|
|
205
|
+
if (fm) {
|
|
206
|
+
const stepsMatch = fm.match(/^estimated_steps:\s*(\d+)/m);
|
|
207
|
+
const filesMatch = fm.match(/^estimated_files:\s*(\d+)/m);
|
|
208
|
+
|
|
209
|
+
if (stepsMatch) {
|
|
210
|
+
const estimatedSteps = parseInt(stepsMatch[1], 10);
|
|
211
|
+
if (estimatedSteps >= 10) {
|
|
212
|
+
issues.push({
|
|
213
|
+
severity: "warning",
|
|
214
|
+
scope: "task-plan",
|
|
215
|
+
file,
|
|
216
|
+
ruleId: "scope_estimate_steps_high",
|
|
217
|
+
message: `Task plan estimates ${estimatedSteps} steps (threshold: 10). Consider splitting into smaller tasks.`,
|
|
218
|
+
suggestion: "Break the task into sub-tasks or reduce scope so each task stays focused and completable in one pass.",
|
|
219
|
+
});
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
if (filesMatch) {
|
|
224
|
+
const estimatedFiles = parseInt(filesMatch[1], 10);
|
|
225
|
+
if (estimatedFiles >= 12) {
|
|
226
|
+
issues.push({
|
|
227
|
+
severity: "warning",
|
|
228
|
+
scope: "task-plan",
|
|
229
|
+
file,
|
|
230
|
+
ruleId: "scope_estimate_files_high",
|
|
231
|
+
message: `Task plan estimates ${estimatedFiles} files (threshold: 12). Consider splitting into smaller tasks.`,
|
|
232
|
+
suggestion: "Break the task into sub-tasks or reduce scope to keep the change footprint manageable.",
|
|
233
|
+
});
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
// Rule: Inputs and Expected Output should contain backtick-wrapped file paths
|
|
239
|
+
const inputsSection = getSection(content, "Inputs", 2);
|
|
240
|
+
const outputSection = getSection(content, "Expected Output", 2);
|
|
241
|
+
const backtickPathPattern = /`[^`]*[./][^`]*`/;
|
|
242
|
+
|
|
243
|
+
if (outputSection === null || !backtickPathPattern.test(outputSection)) {
|
|
244
|
+
issues.push({
|
|
245
|
+
severity: "warning",
|
|
246
|
+
scope: "task-plan",
|
|
247
|
+
file,
|
|
248
|
+
ruleId: "missing_output_file_paths",
|
|
249
|
+
message: "Task plan `## Expected Output` is missing or has no backtick-wrapped file paths.",
|
|
250
|
+
suggestion: "List concrete output file paths in backticks (e.g. `src/types.ts`). These are machine-parsed to derive task dependencies.",
|
|
251
|
+
});
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
if (inputsSection !== null && inputsSection.trim().length > 0 && !backtickPathPattern.test(inputsSection)) {
|
|
255
|
+
issues.push({
|
|
256
|
+
severity: "info",
|
|
257
|
+
scope: "task-plan",
|
|
258
|
+
file,
|
|
259
|
+
ruleId: "missing_input_file_paths",
|
|
260
|
+
message: "Task plan `## Inputs` has content but no backtick-wrapped file paths.",
|
|
261
|
+
suggestion: "List input file paths in backticks (e.g. `src/config.json`). These are machine-parsed to derive task dependencies.",
|
|
262
|
+
});
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
// ── Observability rules (gated by runtime relevance) ──
|
|
266
|
+
|
|
267
|
+
const relevant = textSuggestsObservabilityRelevant(content);
|
|
268
|
+
if (!relevant) return issues;
|
|
269
|
+
|
|
270
|
+
const obs = getSection(content, "Observability Impact", 2);
|
|
271
|
+
if (!obs) {
|
|
272
|
+
issues.push({
|
|
273
|
+
severity: "warning",
|
|
274
|
+
scope: "task-plan",
|
|
275
|
+
file,
|
|
276
|
+
ruleId: "missing_observability_impact",
|
|
277
|
+
message: "Task plan appears runtime-relevant but is missing `## Observability Impact`.",
|
|
278
|
+
suggestion: "Explain what signals change, how a future agent inspects this task, and what failure state becomes visible.",
|
|
279
|
+
});
|
|
280
|
+
} else if (sectionLooksPlaceholderOnly(obs)) {
|
|
281
|
+
issues.push({
|
|
282
|
+
severity: "warning",
|
|
283
|
+
scope: "task-plan",
|
|
284
|
+
file,
|
|
285
|
+
ruleId: "observability_impact_placeholder_only",
|
|
286
|
+
message: "Task plan has `## Observability Impact` but it still looks empty or placeholder-only.",
|
|
287
|
+
suggestion: "Fill in concrete inspection surfaces or explicitly justify why observability is not applicable.",
|
|
288
|
+
});
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
return issues;
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
export function validateTaskSummaryContent(file: string, content: string): ValidationIssue[] {
|
|
295
|
+
const issues: ValidationIssue[] = [];
|
|
296
|
+
if (!hasFrontmatterKey(content, "observability_surfaces")) {
|
|
297
|
+
issues.push({
|
|
298
|
+
severity: "warning",
|
|
299
|
+
scope: "task-summary",
|
|
300
|
+
file,
|
|
301
|
+
ruleId: "missing_observability_frontmatter",
|
|
302
|
+
message: "Task summary is missing `observability_surfaces` in frontmatter.",
|
|
303
|
+
suggestion: "List the durable status/log/error surfaces a future agent should use.",
|
|
304
|
+
});
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
const diagnostics = getSection(content, "Diagnostics", 2);
|
|
308
|
+
if (!diagnostics) {
|
|
309
|
+
issues.push({
|
|
310
|
+
severity: "warning",
|
|
311
|
+
scope: "task-summary",
|
|
312
|
+
file,
|
|
313
|
+
ruleId: "missing_diagnostics_section",
|
|
314
|
+
message: "Task summary is missing `## Diagnostics`.",
|
|
315
|
+
suggestion: "Document how to inspect what this task built later.",
|
|
316
|
+
});
|
|
317
|
+
} else if (sectionLooksPlaceholderOnly(diagnostics)) {
|
|
318
|
+
issues.push({
|
|
319
|
+
severity: "warning",
|
|
320
|
+
scope: "task-summary",
|
|
321
|
+
file,
|
|
322
|
+
ruleId: "diagnostics_placeholder_only",
|
|
323
|
+
message: "Task summary diagnostics section still looks like placeholder text.",
|
|
324
|
+
suggestion: "Replace placeholders with concrete commands, endpoints, logs, error shapes, or failure artifacts.",
|
|
325
|
+
});
|
|
326
|
+
}
|
|
327
|
+
|
|
328
|
+
const evidence = getSection(content, "Verification Evidence", 2);
|
|
329
|
+
if (!evidence) {
|
|
330
|
+
issues.push({
|
|
331
|
+
severity: "warning",
|
|
332
|
+
scope: "task-summary",
|
|
333
|
+
file,
|
|
334
|
+
ruleId: "evidence_block_missing",
|
|
335
|
+
message: "Task summary is missing `## Verification Evidence`.",
|
|
336
|
+
suggestion: "Add a verification evidence table showing gate check results (command, exit code, verdict, duration).",
|
|
337
|
+
});
|
|
338
|
+
} else if (sectionLooksPlaceholderOnly(evidence)) {
|
|
339
|
+
issues.push({
|
|
340
|
+
severity: "warning",
|
|
341
|
+
scope: "task-summary",
|
|
342
|
+
file,
|
|
343
|
+
ruleId: "evidence_block_placeholder",
|
|
344
|
+
message: "Task summary verification evidence section still looks like placeholder text.",
|
|
345
|
+
suggestion: "Replace placeholders with actual gate results or note that no verification commands were discovered.",
|
|
346
|
+
});
|
|
347
|
+
}
|
|
348
|
+
|
|
349
|
+
return issues;
|
|
350
|
+
}
|
|
351
|
+
|
|
352
|
+
export function validateSliceSummaryContent(file: string, content: string): ValidationIssue[] {
|
|
353
|
+
const issues: ValidationIssue[] = [];
|
|
354
|
+
if (!hasFrontmatterKey(content, "observability_surfaces")) {
|
|
355
|
+
issues.push({
|
|
356
|
+
severity: "warning",
|
|
357
|
+
scope: "slice-summary",
|
|
358
|
+
file,
|
|
359
|
+
ruleId: "missing_observability_frontmatter",
|
|
360
|
+
message: "Slice summary is missing `observability_surfaces` in frontmatter.",
|
|
361
|
+
suggestion: "List the authoritative diagnostics and durable inspection surfaces for this slice.",
|
|
362
|
+
});
|
|
363
|
+
}
|
|
364
|
+
|
|
365
|
+
const diagnostics = getSection(content, "Authoritative diagnostics", 3);
|
|
366
|
+
if (!diagnostics) {
|
|
367
|
+
issues.push({
|
|
368
|
+
severity: "warning",
|
|
369
|
+
scope: "slice-summary",
|
|
370
|
+
file,
|
|
371
|
+
ruleId: "missing_authoritative_diagnostics",
|
|
372
|
+
message: "Slice summary is missing `### Authoritative diagnostics` in Forward Intelligence.",
|
|
373
|
+
suggestion: "Tell future agents where to look first and why that signal is trustworthy.",
|
|
374
|
+
});
|
|
375
|
+
} else if (sectionLooksPlaceholderOnly(diagnostics)) {
|
|
376
|
+
issues.push({
|
|
377
|
+
severity: "warning",
|
|
378
|
+
scope: "slice-summary",
|
|
379
|
+
file,
|
|
380
|
+
ruleId: "authoritative_diagnostics_placeholder_only",
|
|
381
|
+
message: "Slice summary includes authoritative diagnostics but it still looks like placeholder text.",
|
|
382
|
+
suggestion: "Replace placeholders with the real first-stop diagnostic surface for this slice.",
|
|
383
|
+
});
|
|
384
|
+
}
|
|
385
|
+
|
|
386
|
+
return issues;
|
|
387
|
+
}
|
|
388
|
+
|
|
389
|
+
export async function validatePlanBoundary(basePath: string, milestoneId: string, sliceId: string): Promise<ValidationIssue[]> {
|
|
390
|
+
const issues: ValidationIssue[] = [];
|
|
391
|
+
const slicePlan = resolveSliceFile(basePath, milestoneId, sliceId, "PLAN");
|
|
392
|
+
if (slicePlan) {
|
|
393
|
+
const content = await loadFile(slicePlan);
|
|
394
|
+
if (content) issues.push(...validateSlicePlanContent(slicePlan, content));
|
|
395
|
+
}
|
|
396
|
+
|
|
397
|
+
const tasksDir = resolveTasksDir(basePath, milestoneId, sliceId);
|
|
398
|
+
const taskPlans = tasksDir ? resolveTaskFiles(tasksDir, "PLAN") : [];
|
|
399
|
+
for (const file of taskPlans) {
|
|
400
|
+
const taskId = file.split("-")[0];
|
|
401
|
+
const taskPlan = resolveTaskFile(basePath, milestoneId, sliceId, taskId, "PLAN");
|
|
402
|
+
if (!taskPlan) continue;
|
|
403
|
+
const content = await loadFile(taskPlan);
|
|
404
|
+
if (content) issues.push(...validateTaskPlanContent(taskPlan, content));
|
|
405
|
+
}
|
|
406
|
+
|
|
407
|
+
return issues;
|
|
408
|
+
}
|
|
409
|
+
|
|
410
|
+
export async function validateExecuteBoundary(basePath: string, milestoneId: string, sliceId: string, taskId: string): Promise<ValidationIssue[]> {
|
|
411
|
+
const issues: ValidationIssue[] = [];
|
|
412
|
+
const slicePlan = resolveSliceFile(basePath, milestoneId, sliceId, "PLAN");
|
|
413
|
+
if (slicePlan) {
|
|
414
|
+
const content = await loadFile(slicePlan);
|
|
415
|
+
if (content) issues.push(...validateSlicePlanContent(slicePlan, content));
|
|
416
|
+
}
|
|
417
|
+
|
|
418
|
+
const taskPlan = resolveTaskFile(basePath, milestoneId, sliceId, taskId, "PLAN");
|
|
419
|
+
if (taskPlan) {
|
|
420
|
+
const content = await loadFile(taskPlan);
|
|
421
|
+
if (content) issues.push(...validateTaskPlanContent(taskPlan, content));
|
|
422
|
+
}
|
|
423
|
+
|
|
424
|
+
return issues;
|
|
425
|
+
}
|
|
426
|
+
|
|
427
|
+
export async function validateCompleteBoundary(basePath: string, milestoneId: string, sliceId: string): Promise<ValidationIssue[]> {
|
|
428
|
+
const issues: ValidationIssue[] = [];
|
|
429
|
+
const tasksDir = resolveTasksDir(basePath, milestoneId, sliceId);
|
|
430
|
+
const taskSummaries = tasksDir ? resolveTaskFiles(tasksDir, "SUMMARY") : [];
|
|
431
|
+
for (const file of taskSummaries) {
|
|
432
|
+
const taskId = file.split("-")[0];
|
|
433
|
+
const taskSummary = resolveTaskFile(basePath, milestoneId, sliceId, taskId, "SUMMARY");
|
|
434
|
+
if (!taskSummary) continue;
|
|
435
|
+
const content = await loadFile(taskSummary);
|
|
436
|
+
if (content) issues.push(...validateTaskSummaryContent(taskSummary, content));
|
|
437
|
+
}
|
|
438
|
+
|
|
439
|
+
const sliceSummary = resolveSliceFile(basePath, milestoneId, sliceId, "SUMMARY");
|
|
440
|
+
if (sliceSummary) {
|
|
441
|
+
const content = await loadFile(sliceSummary);
|
|
442
|
+
if (content) issues.push(...validateSliceSummaryContent(sliceSummary, content));
|
|
443
|
+
}
|
|
444
|
+
|
|
445
|
+
return issues;
|
|
446
|
+
}
|
|
447
|
+
|
|
448
|
+
export function formatValidationIssues(issues: ValidationIssue[], limit: number = 4): string {
|
|
449
|
+
if (issues.length === 0) return "";
|
|
450
|
+
const lines = issues.slice(0, limit).map(issue => {
|
|
451
|
+
const fileName = issue.file.split("/").pop() || issue.file;
|
|
452
|
+
return `- ${fileName}: ${issue.message}`;
|
|
453
|
+
});
|
|
454
|
+
if (issues.length > limit) lines.push(`- ...and ${issues.length - limit} more`);
|
|
455
|
+
return lines.join("\n");
|
|
456
|
+
}
|
|
@@ -125,18 +125,6 @@ export function getNextFallbackModel(
|
|
|
125
125
|
}
|
|
126
126
|
}
|
|
127
127
|
|
|
128
|
-
/**
|
|
129
|
-
* Detect whether an error message indicates a transient network error
|
|
130
|
-
* (worth retrying the same model) vs a permanent provider error
|
|
131
|
-
* (auth failure, quota exceeded, etc. -- should fall back immediately).
|
|
132
|
-
*/
|
|
133
|
-
export function isTransientNetworkError(errorMsg: string): boolean {
|
|
134
|
-
if (!errorMsg) return false;
|
|
135
|
-
const hasNetworkSignal = /network|ECONNRESET|ETIMEDOUT|ECONNREFUSED|socket hang up|fetch failed|connection.*reset|dns/i.test(errorMsg);
|
|
136
|
-
const hasPermanentSignal = /auth|unauthorized|forbidden|invalid.*key|quota|billing/i.test(errorMsg);
|
|
137
|
-
return hasNetworkSignal && !hasPermanentSignal;
|
|
138
|
-
}
|
|
139
|
-
|
|
140
128
|
/**
|
|
141
129
|
* Validate a model ID string.
|
|
142
130
|
* Returns true if the ID looks like a valid model identifier.
|
|
@@ -308,7 +296,7 @@ export function resolveContextSelection(): import("./types.js").ContextSelection
|
|
|
308
296
|
}
|
|
309
297
|
|
|
310
298
|
/**
|
|
311
|
-
* Resolve the search provider preference from
|
|
299
|
+
* Resolve the search provider preference from PREFERENCES.md.
|
|
312
300
|
* Returns undefined if not configured (caller falls back to existing behavior).
|
|
313
301
|
*/
|
|
314
302
|
export function resolveSearchProviderFromPreferences(): GSDPreferences["search_provider"] | undefined {
|
|
@@ -8,7 +8,6 @@
|
|
|
8
8
|
import { existsSync, readdirSync } from "node:fs";
|
|
9
9
|
import { homedir } from "node:os";
|
|
10
10
|
import { isAbsolute, join } from "node:path";
|
|
11
|
-
import { getAgentDir } from "@gsd/pi-coding-agent";
|
|
12
11
|
import { statSync } from "node:fs";
|
|
13
12
|
|
|
14
13
|
import type {
|
|
@@ -25,13 +24,20 @@ export type { GSDSkillRule, SkillDiscoveryMode, SkillResolution, SkillResolution
|
|
|
25
24
|
|
|
26
25
|
/**
|
|
27
26
|
* Known skill directories, in priority order.
|
|
28
|
-
*
|
|
27
|
+
* Global skills (~/.agents/skills/) take precedence over project skills.
|
|
28
|
+
* Legacy ~/.gsd/agent/skills/ is included as a fallback for pre-migration installs.
|
|
29
29
|
*/
|
|
30
30
|
export function getSkillSearchDirs(cwd: string): Array<{ dir: string; method: SkillResolution["method"] }> {
|
|
31
|
-
|
|
32
|
-
{ dir: join(
|
|
33
|
-
{ dir: join(cwd, ".
|
|
31
|
+
const dirs: Array<{ dir: string; method: SkillResolution["method"] }> = [
|
|
32
|
+
{ dir: join(homedir(), ".agents", "skills"), method: "user-skill" },
|
|
33
|
+
{ dir: join(cwd, ".agents", "skills"), method: "project-skill" },
|
|
34
34
|
];
|
|
35
|
+
// Legacy fallback — read skills from old GSD directory only if migration hasn't completed
|
|
36
|
+
const legacyDir = join(homedir(), ".gsd", "agent", "skills");
|
|
37
|
+
if (existsSync(legacyDir) && !existsSync(join(legacyDir, ".migrated-to-agents"))) {
|
|
38
|
+
dirs.push({ dir: legacyDir, method: "user-skill" });
|
|
39
|
+
}
|
|
40
|
+
return dirs;
|
|
35
41
|
}
|
|
36
42
|
|
|
37
43
|
/**
|
|
@@ -93,6 +93,7 @@ export const KNOWN_PREFERENCE_KEYS = new Set<string>([
|
|
|
93
93
|
"service_tier",
|
|
94
94
|
"forensics_dedup",
|
|
95
95
|
"show_token_cost",
|
|
96
|
+
"experimental",
|
|
96
97
|
]);
|
|
97
98
|
|
|
98
99
|
/** Canonical list of all dispatch unit types. */
|
|
@@ -182,6 +183,20 @@ export interface CmuxPreferences {
|
|
|
182
183
|
browser?: boolean;
|
|
183
184
|
}
|
|
184
185
|
|
|
186
|
+
/**
|
|
187
|
+
* Opt-in experimental features. All features in this block are disabled by
|
|
188
|
+
* default and must be explicitly enabled. They may change or be removed without
|
|
189
|
+
* a deprecation cycle while in experimental status.
|
|
190
|
+
*/
|
|
191
|
+
export interface ExperimentalPreferences {
|
|
192
|
+
/**
|
|
193
|
+
* Enable RTK (Real-Time Kompression) shell-command compression.
|
|
194
|
+
* RTK wraps shell commands to reduce token usage during command execution.
|
|
195
|
+
* Default: false (opt-in required).
|
|
196
|
+
*/
|
|
197
|
+
rtk?: boolean;
|
|
198
|
+
}
|
|
199
|
+
|
|
185
200
|
export interface GSDPreferences {
|
|
186
201
|
version?: number;
|
|
187
202
|
mode?: WorkflowMode;
|
|
@@ -233,6 +248,11 @@ export interface GSDPreferences {
|
|
|
233
248
|
forensics_dedup?: boolean;
|
|
234
249
|
/** Opt-in: show per-prompt and cumulative session token cost in the footer. Default: false. */
|
|
235
250
|
show_token_cost?: boolean;
|
|
251
|
+
/**
|
|
252
|
+
* Opt-in experimental features. All features here are disabled by default.
|
|
253
|
+
* See the preferences reference for details on each feature.
|
|
254
|
+
*/
|
|
255
|
+
experimental?: ExperimentalPreferences;
|
|
236
256
|
}
|
|
237
257
|
|
|
238
258
|
export interface LoadedGSDPreferences {
|
|
@@ -793,5 +793,31 @@ export function validatePreferences(preferences: GSDPreferences): {
|
|
|
793
793
|
}
|
|
794
794
|
}
|
|
795
795
|
|
|
796
|
+
// ─── Experimental Features ────────────────────────────────────────
|
|
797
|
+
if (preferences.experimental !== undefined) {
|
|
798
|
+
if (typeof preferences.experimental === "object" && preferences.experimental !== null) {
|
|
799
|
+
const exp = preferences.experimental as unknown as Record<string, unknown>;
|
|
800
|
+
const validExp: import("./preferences-types.js").ExperimentalPreferences = {};
|
|
801
|
+
|
|
802
|
+
if (exp.rtk !== undefined) {
|
|
803
|
+
if (typeof exp.rtk === "boolean") validExp.rtk = exp.rtk;
|
|
804
|
+
else errors.push("experimental.rtk must be a boolean");
|
|
805
|
+
}
|
|
806
|
+
|
|
807
|
+
const knownExpKeys = new Set(["rtk"]);
|
|
808
|
+
for (const key of Object.keys(exp)) {
|
|
809
|
+
if (!knownExpKeys.has(key)) {
|
|
810
|
+
warnings.push(`unknown experimental key "${key}" — ignored`);
|
|
811
|
+
}
|
|
812
|
+
}
|
|
813
|
+
|
|
814
|
+
if (Object.keys(validExp).length > 0) {
|
|
815
|
+
validated.experimental = validExp;
|
|
816
|
+
}
|
|
817
|
+
} else {
|
|
818
|
+
errors.push("experimental must be an object");
|
|
819
|
+
}
|
|
820
|
+
}
|
|
821
|
+
|
|
796
822
|
return { preferences: validated, errors, warnings };
|
|
797
823
|
}
|
|
@@ -68,7 +68,6 @@ export {
|
|
|
68
68
|
resolveModelForUnit,
|
|
69
69
|
resolveModelWithFallbacksForUnit,
|
|
70
70
|
getNextFallbackModel,
|
|
71
|
-
isTransientNetworkError,
|
|
72
71
|
validateModelId,
|
|
73
72
|
updatePreferencesModels,
|
|
74
73
|
resolveDynamicRoutingConfig,
|
|
@@ -87,7 +86,7 @@ function gsdHome(): string {
|
|
|
87
86
|
}
|
|
88
87
|
|
|
89
88
|
function globalPreferencesPath(): string {
|
|
90
|
-
return join(gsdHome(), "
|
|
89
|
+
return join(gsdHome(), "PREFERENCES.md");
|
|
91
90
|
}
|
|
92
91
|
|
|
93
92
|
function legacyGlobalPreferencesPath(): string {
|
|
@@ -95,15 +94,15 @@ function legacyGlobalPreferencesPath(): string {
|
|
|
95
94
|
}
|
|
96
95
|
|
|
97
96
|
function projectPreferencesPath(): string {
|
|
98
|
-
return join(gsdRoot(process.cwd()), "
|
|
97
|
+
return join(gsdRoot(process.cwd()), "PREFERENCES.md");
|
|
99
98
|
}
|
|
100
|
-
//
|
|
101
|
-
// Check
|
|
102
|
-
function
|
|
103
|
-
return join(gsdHome(), "
|
|
99
|
+
// Legacy: older versions used lowercase preferences.md.
|
|
100
|
+
// Check lowercase as a fallback so those files aren't silently ignored.
|
|
101
|
+
function globalPreferencesPathLegacy(): string {
|
|
102
|
+
return join(gsdHome(), "preferences.md");
|
|
104
103
|
}
|
|
105
|
-
function
|
|
106
|
-
return join(gsdRoot(process.cwd()), "
|
|
104
|
+
function projectPreferencesPathLegacy(): string {
|
|
105
|
+
return join(gsdRoot(process.cwd()), "preferences.md");
|
|
107
106
|
}
|
|
108
107
|
|
|
109
108
|
export function getGlobalGSDPreferencesPath(): string {
|
|
@@ -122,13 +121,13 @@ export function getProjectGSDPreferencesPath(): string {
|
|
|
122
121
|
|
|
123
122
|
export function loadGlobalGSDPreferences(): LoadedGSDPreferences | null {
|
|
124
123
|
return loadPreferencesFile(globalPreferencesPath(), "global")
|
|
125
|
-
?? loadPreferencesFile(
|
|
124
|
+
?? loadPreferencesFile(globalPreferencesPathLegacy(), "global")
|
|
126
125
|
?? loadPreferencesFile(legacyGlobalPreferencesPath(), "global");
|
|
127
126
|
}
|
|
128
127
|
|
|
129
128
|
export function loadProjectGSDPreferences(): LoadedGSDPreferences | null {
|
|
130
129
|
return loadPreferencesFile(projectPreferencesPath(), "project")
|
|
131
|
-
?? loadPreferencesFile(
|
|
130
|
+
?? loadPreferencesFile(projectPreferencesPathLegacy(), "project");
|
|
132
131
|
}
|
|
133
132
|
|
|
134
133
|
export function loadEffectiveGSDPreferences(): LoadedGSDPreferences | null {
|
|
@@ -223,7 +222,7 @@ export function parsePreferencesMarkdown(content: string): GSDPreferences | null
|
|
|
223
222
|
|
|
224
223
|
if (!_warnedUnrecognizedFormat) {
|
|
225
224
|
_warnedUnrecognizedFormat = true;
|
|
226
|
-
console.warn("[parsePreferencesMarkdown]
|
|
225
|
+
console.warn("[parsePreferencesMarkdown] PREFERENCES.md exists but uses an unrecognized format — skipping.");
|
|
227
226
|
}
|
|
228
227
|
return null;
|
|
229
228
|
}
|
|
@@ -353,6 +352,9 @@ function mergePreferences(base: GSDPreferences, override: GSDPreferences): GSDPr
|
|
|
353
352
|
service_tier: override.service_tier ?? base.service_tier,
|
|
354
353
|
forensics_dedup: override.forensics_dedup ?? base.forensics_dedup,
|
|
355
354
|
show_token_cost: override.show_token_cost ?? base.show_token_cost,
|
|
355
|
+
experimental: (base.experimental || override.experimental)
|
|
356
|
+
? { ...(base.experimental ?? {}), ...(override.experimental ?? {}) }
|
|
357
|
+
: undefined,
|
|
356
358
|
};
|
|
357
359
|
}
|
|
358
360
|
|
|
@@ -499,7 +501,7 @@ export function resolvePreDispatchHooks(): PreDispatchHookConfig[] {
|
|
|
499
501
|
* Resolve the effective git isolation mode from preferences.
|
|
500
502
|
* Returns "none" (default), "worktree", or "branch".
|
|
501
503
|
*
|
|
502
|
-
* Default is "none" so GSD works out of the box without
|
|
504
|
+
* Default is "none" so GSD works out of the box without PREFERENCES.md.
|
|
503
505
|
* Worktree isolation requires explicit opt-in because it depends on git
|
|
504
506
|
* branch infrastructure that must be set up before use.
|
|
505
507
|
*/
|