@sienklogic/plan-build-run 2.18.1 → 2.19.1
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/CHANGELOG.md +1265 -296
- package/CLAUDE.md +56 -34
- package/LICENSE +2 -1
- package/README.md +412 -177
- package/bin/install.js +2752 -0
- package/dashboard/bin/cli.cjs +96 -0
- package/dashboard/bin/stop.cjs +129 -0
- package/dashboard/eslint.config.js +37 -0
- package/dashboard/index.html +20 -0
- package/dashboard/package.json +27 -25
- package/dashboard/server/index.js +142 -0
- package/dashboard/server/lib/frontmatter.js +92 -0
- package/dashboard/server/middleware/static.js +35 -0
- package/dashboard/server/package.json +16 -0
- package/dashboard/server/routes/agents.js +213 -0
- package/dashboard/server/routes/config.js +64 -0
- package/dashboard/server/routes/health.js +95 -0
- package/dashboard/server/routes/incidents.js +78 -0
- package/dashboard/server/routes/intel.js +69 -0
- package/dashboard/server/routes/memory.js +107 -0
- package/dashboard/server/routes/planning.js +234 -0
- package/dashboard/server/routes/progress.js +77 -0
- package/dashboard/server/routes/projects.js +36 -0
- package/dashboard/server/routes/requirements.js +40 -0
- package/dashboard/server/routes/roadmap.js +69 -0
- package/dashboard/server/routes/sessions.js +70 -0
- package/dashboard/server/routes/status.js +25 -0
- package/dashboard/server/routes/telemetry.js +214 -0
- package/dashboard/server/services/file-watcher.js +105 -0
- package/dashboard/server/services/planning-reader.js +741 -0
- package/dashboard/server/test/cli.test.js +34 -0
- package/dashboard/server/test/frontmatter.test.js +104 -0
- package/dashboard/server/test/isolation.test.js +32 -0
- package/dashboard/server/test/planning-reader.test.js +151 -0
- package/dashboard/server/test/routes.test.js +91 -0
- package/dashboard/server/test/ws.test.js +81 -0
- package/dashboard/server/ws.js +96 -0
- package/dashboard/src/App.jsx +165 -0
- package/dashboard/src/components/charts/BudgetBars.jsx +42 -0
- package/dashboard/src/components/charts/ContextRadar.jsx +34 -0
- package/dashboard/src/components/charts/PhaseDonut.jsx +66 -0
- package/dashboard/src/components/charts/SuccessTrend.jsx +45 -0
- package/dashboard/src/components/charts/TokenChart.jsx +55 -0
- package/dashboard/src/components/charts/index.js +5 -0
- package/dashboard/src/components/config/CfgSection.jsx +93 -0
- package/dashboard/src/components/layout/Header.jsx +89 -0
- package/dashboard/src/components/layout/ProjectSwitcher.jsx +160 -0
- package/dashboard/src/components/layout/Sidebar.jsx +161 -0
- package/dashboard/src/components/ui/AutoModeBanner.jsx +138 -0
- package/dashboard/src/components/ui/BackButton.jsx +27 -0
- package/dashboard/src/components/ui/Badge.jsx +27 -0
- package/dashboard/src/components/ui/Card.jsx +23 -0
- package/dashboard/src/components/ui/ChartTooltip.jsx +48 -0
- package/dashboard/src/components/ui/CheckpointBox.jsx +110 -0
- package/dashboard/src/components/ui/CodeBlock.jsx +27 -0
- package/dashboard/src/components/ui/ConfidenceBadge.jsx +20 -0
- package/dashboard/src/components/ui/ConfirmModal.jsx +161 -0
- package/dashboard/src/components/ui/ConnectionBanner.jsx +60 -0
- package/dashboard/src/components/ui/ErrorBoundary.jsx +106 -0
- package/dashboard/src/components/ui/ErrorBox.jsx +107 -0
- package/dashboard/src/components/ui/KeyValue.jsx +33 -0
- package/dashboard/src/components/ui/LoadingSkeleton.jsx +84 -0
- package/dashboard/src/components/ui/MetricCard.jsx +58 -0
- package/dashboard/src/components/ui/NextUpBlock.jsx +92 -0
- package/dashboard/src/components/ui/NumberInput.jsx +44 -0
- package/dashboard/src/components/ui/PBRBanner.jsx +47 -0
- package/dashboard/src/components/ui/PipelineView.jsx +130 -0
- package/dashboard/src/components/ui/ProgressBar.jsx +28 -0
- package/dashboard/src/components/ui/ProgressDisplay.jsx +47 -0
- package/dashboard/src/components/ui/QualityGateBadge.jsx +15 -0
- package/dashboard/src/components/ui/SectionTitle.jsx +35 -0
- package/dashboard/src/components/ui/SelectInput.jsx +45 -0
- package/dashboard/src/components/ui/StatusDot.jsx +51 -0
- package/dashboard/src/components/ui/StatusSymbol.jsx +49 -0
- package/dashboard/src/components/ui/TabBar.jsx +41 -0
- package/dashboard/src/components/ui/TextInput.jsx +42 -0
- package/dashboard/src/components/ui/Toast.jsx +117 -0
- package/dashboard/src/components/ui/Toggle.jsx +70 -0
- package/dashboard/src/components/ui/index.js +29 -0
- package/dashboard/src/hooks/useDocumentTitle.js +16 -0
- package/dashboard/src/hooks/useFetch.js +50 -0
- package/dashboard/src/hooks/useToast.jsx +43 -0
- package/dashboard/src/hooks/useWebSocket.js +103 -0
- package/dashboard/src/lib/api.js +112 -0
- package/dashboard/src/lib/configSchema.js +189 -0
- package/dashboard/src/lib/constants.js +22 -0
- package/dashboard/src/main.jsx +15 -0
- package/dashboard/src/pages/AgentsPage.jsx +191 -0
- package/dashboard/src/pages/ConfigPage.jsx +298 -0
- package/dashboard/src/pages/HooksPage.jsx +412 -0
- package/dashboard/src/pages/IncidentsPage.jsx +135 -0
- package/dashboard/src/pages/IntelPage.jsx +193 -0
- package/dashboard/src/pages/LiveFeed.jsx +274 -0
- package/dashboard/src/pages/MemoryPage.jsx +107 -0
- package/dashboard/src/pages/OnboardingPage.jsx +117 -0
- package/dashboard/src/pages/Overview.jsx +360 -0
- package/dashboard/src/pages/PhaseDetailView.jsx +216 -0
- package/dashboard/src/pages/PlanningPage.jsx +181 -0
- package/dashboard/src/pages/ProgressPage.jsx +249 -0
- package/dashboard/src/pages/ResearchPage.jsx +129 -0
- package/dashboard/src/pages/RoadmapPage.jsx +251 -0
- package/dashboard/src/pages/SessionsPage.jsx +117 -0
- package/dashboard/src/pages/Telemetry.jsx +166 -0
- package/dashboard/src/pages/planning/DecisionsTab.jsx +153 -0
- package/dashboard/src/pages/planning/FilesTab.jsx +420 -0
- package/dashboard/src/pages/planning/MilestoneDetail.jsx +319 -0
- package/dashboard/src/pages/planning/MilestonesTab.jsx +151 -0
- package/dashboard/src/pages/planning/NotesTab.jsx +251 -0
- package/dashboard/src/pages/planning/PhasesTab.jsx +218 -0
- package/dashboard/src/pages/planning/QuickTab.jsx +50 -0
- package/dashboard/src/pages/planning/ResearchTab.jsx +103 -0
- package/dashboard/src/pages/planning/TodosTab.jsx +297 -0
- package/dashboard/src/theme/ThemeProvider.jsx +38 -0
- package/dashboard/src/theme/tokens.js +17 -0
- package/dashboard/tests/components/ConfirmModal.test.jsx +179 -0
- package/dashboard/tests/components/ConnectionBanner.test.jsx +37 -0
- package/dashboard/tests/components/ErrorBoundary.test.jsx +59 -0
- package/dashboard/tests/components/LoadingSkeleton.test.jsx +46 -0
- package/dashboard/tests/components/ToastContainer.test.jsx +47 -0
- package/dashboard/tests/components/Toggle.test.jsx +61 -0
- package/dashboard/tests/hooks/useFetch.test.jsx +77 -0
- package/dashboard/tests/hooks/useToast.test.jsx +78 -0
- package/dashboard/tests/hooks/useWebSocket.test.jsx +128 -0
- package/dashboard/tests/pages/ConfigPage.test.jsx +199 -0
- package/dashboard/tests/pages/PlanningPage.test.jsx +119 -0
- package/dashboard/tests/pages/planning/FilesTab.test.jsx +198 -0
- package/dashboard/tests/pages/planning/NotesTab.test.jsx +178 -0
- package/dashboard/tests/pages/planning/TodosTab.test.jsx +188 -0
- package/dashboard/tests/performance.test.jsx +46 -0
- package/dashboard/tests/routes/config.test.js +98 -0
- package/dashboard/tests/routes/health.test.js +40 -0
- package/dashboard/tests/routes/planning.test.js +112 -0
- package/dashboard/tests/routes/roadmap.test.js +91 -0
- package/dashboard/tests/routes/status.test.js +131 -0
- package/dashboard/tests/server/planning-reader.test.js +153 -0
- package/dashboard/tests/setup.js +7 -0
- package/dashboard/vite.config.js +41 -0
- package/package.json +55 -40
- package/plan-build-run/bin/config-schema.json +1420 -0
- package/plugins/pbr/.claude-plugin/plugin.json +1 -1
- package/plugins/pbr/CLAUDE.md +19 -0
- package/plugins/pbr/UI-CONSISTENCY-GAPS.md +1 -1
- package/plugins/pbr/agents/advisor-researcher.md +100 -0
- package/plugins/pbr/agents/audit.md +205 -89
- package/plugins/pbr/agents/codebase-mapper.md +158 -23
- package/plugins/pbr/agents/debugger.md +222 -30
- package/plugins/pbr/agents/dev-sync.md +206 -0
- package/plugins/pbr/agents/executor.md +717 -39
- package/plugins/pbr/agents/general.md +71 -6
- package/plugins/pbr/agents/integration-checker.md +146 -30
- package/plugins/pbr/agents/intel-updater.md +332 -0
- package/plugins/pbr/agents/nyquist-auditor.md +253 -0
- package/plugins/pbr/agents/plan-checker.md +270 -61
- package/plugins/pbr/agents/planner.md +440 -42
- package/plugins/pbr/agents/researcher.md +223 -36
- package/plugins/pbr/agents/roadmapper.md +397 -0
- package/plugins/pbr/agents/synthesizer.md +170 -26
- package/plugins/pbr/agents/ui-checker.md +203 -0
- package/plugins/pbr/agents/ui-researcher.md +224 -0
- package/plugins/pbr/agents/verifier.md +495 -47
- package/plugins/pbr/commands/add-phase.md +75 -0
- package/plugins/pbr/commands/add-todo.md +8 -0
- package/plugins/pbr/commands/audit-fix.md +5 -0
- package/plugins/pbr/commands/audit-milestone.md +8 -0
- package/plugins/pbr/commands/autonomous.md +5 -0
- package/plugins/pbr/commands/backlog.md +6 -0
- package/plugins/pbr/commands/check-todos.md +8 -0
- package/plugins/pbr/commands/complete-milestone.md +8 -0
- package/plugins/pbr/commands/config.md +1 -1
- package/plugins/pbr/commands/discuss-phase.md +6 -0
- package/plugins/pbr/commands/do.md +5 -0
- package/plugins/pbr/commands/execute-phase.md +6 -0
- package/plugins/pbr/commands/fast.md +6 -0
- package/plugins/pbr/commands/forensics.md +6 -0
- package/plugins/pbr/commands/import.md +1 -1
- package/plugins/pbr/commands/insert-phase.md +65 -0
- package/plugins/pbr/commands/intel.md +5 -0
- package/plugins/pbr/commands/join-discord.md +11 -0
- package/plugins/pbr/commands/list-phase-assumptions.md +5 -0
- package/plugins/pbr/commands/map-codebase.md +6 -0
- package/plugins/pbr/commands/milestone-summary.md +6 -0
- package/plugins/pbr/commands/new-milestone.md +8 -0
- package/plugins/pbr/commands/new-project.md +6 -0
- package/plugins/pbr/commands/pause-work.md +5 -0
- package/plugins/pbr/commands/plan-milestone-gaps.md +7 -0
- package/plugins/pbr/commands/plan-phase.md +6 -0
- package/plugins/pbr/commands/plant-seed.md +6 -0
- package/plugins/pbr/commands/profile-user.md +5 -0
- package/plugins/pbr/commands/profile.md +5 -0
- package/plugins/pbr/commands/progress.md +6 -0
- package/plugins/pbr/commands/quick.md +1 -1
- package/plugins/pbr/commands/reapply-patches.md +47 -0
- package/plugins/pbr/commands/release.md +6 -0
- package/plugins/pbr/commands/remove-phase.md +66 -0
- package/plugins/pbr/commands/research-phase.md +59 -0
- package/plugins/pbr/commands/resume-work.md +5 -0
- package/plugins/pbr/commands/seed.md +6 -0
- package/plugins/pbr/commands/session-report.md +5 -0
- package/plugins/pbr/commands/set-profile.md +6 -0
- package/plugins/pbr/commands/settings.md +5 -0
- package/plugins/pbr/commands/setup.md +1 -1
- package/plugins/pbr/commands/ship.md +5 -0
- package/plugins/pbr/commands/stats.md +6 -0
- package/plugins/pbr/commands/test.md +5 -0
- package/plugins/pbr/commands/thread.md +6 -0
- package/plugins/pbr/commands/todo.md +1 -1
- package/plugins/pbr/commands/ui-phase.md +5 -0
- package/plugins/pbr/commands/ui-review.md +5 -0
- package/plugins/pbr/commands/undo.md +5 -0
- package/plugins/pbr/commands/update.md +37 -0
- package/plugins/pbr/commands/validate-phase.md +5 -0
- package/plugins/pbr/commands/verify-work.md +6 -0
- package/plugins/pbr/dashboard/package-lock.json +6 -0
- package/plugins/pbr/dist/architecture-guard.js +59 -0
- package/plugins/pbr/dist/audit-dimensions.js +556 -0
- package/plugins/pbr/dist/auto-continue.js +277 -0
- package/plugins/pbr/dist/block-skill-self-read.js +124 -0
- package/plugins/pbr/dist/check-agent-state-write.js +63 -0
- package/plugins/pbr/dist/check-config-change.js +162 -0
- package/plugins/pbr/dist/check-cross-plugin-sync.js +93 -0
- package/plugins/pbr/dist/check-dangerous-commands.js +193 -0
- package/plugins/pbr/dist/check-direct-state-write.js +37 -0
- package/plugins/pbr/dist/check-doc-sprawl.js +102 -0
- package/plugins/pbr/dist/check-phase-boundary.js +191 -0
- package/plugins/pbr/dist/check-plan-format.js +241 -0
- package/plugins/pbr/dist/check-read-first.js +345 -0
- package/plugins/pbr/dist/check-roadmap-sync.js +503 -0
- package/plugins/pbr/dist/check-skill-workflow.js +354 -0
- package/plugins/pbr/dist/check-state-sync.js +658 -0
- package/plugins/pbr/dist/check-subagent-output.js +452 -0
- package/plugins/pbr/dist/check-summary-gate.js +199 -0
- package/plugins/pbr/dist/context-bridge.js +425 -0
- package/plugins/pbr/dist/context-budget-check.js +442 -0
- package/plugins/pbr/dist/context-quality.js +271 -0
- package/plugins/pbr/dist/enforce-context-budget.js +138 -0
- package/plugins/pbr/dist/enforce-pbr-workflow.js +277 -0
- package/plugins/pbr/dist/event-handler.js +202 -0
- package/plugins/pbr/dist/event-logger.js +125 -0
- package/plugins/pbr/dist/feedback-loop.js +172 -0
- package/plugins/pbr/dist/graph-update.js +422 -0
- package/plugins/pbr/dist/hook-logger.js +114 -0
- package/plugins/pbr/dist/hook-server-client.js +361 -0
- package/plugins/pbr/dist/hook-server.js +606 -0
- package/plugins/pbr/dist/hooks-schema.json +87 -0
- package/plugins/pbr/dist/instructions-loaded.js +173 -0
- package/plugins/pbr/dist/intercept-plan-mode.js +81 -0
- package/plugins/pbr/dist/log-notification.js +131 -0
- package/plugins/pbr/dist/log-subagent.js +349 -0
- package/plugins/pbr/dist/log-tool-failure.js +140 -0
- package/plugins/pbr/dist/milestone-learnings.js +569 -0
- package/plugins/pbr/dist/pbr-tools.js +2044 -0
- package/plugins/pbr/dist/post-bash-triage.js +154 -0
- package/plugins/pbr/dist/post-compact.js +135 -0
- package/plugins/pbr/dist/post-hoc.js +286 -0
- package/plugins/pbr/dist/post-write-dispatch.js +279 -0
- package/plugins/pbr/dist/post-write-quality.js +208 -0
- package/plugins/pbr/dist/pre-bash-dispatch.js +218 -0
- package/plugins/pbr/dist/pre-skill-dispatch.js +114 -0
- package/plugins/pbr/dist/pre-task-dispatch.js +297 -0
- package/plugins/pbr/dist/pre-write-dispatch.js +234 -0
- package/plugins/pbr/dist/progress-tracker.js +198 -0
- package/plugins/pbr/dist/prompt-guard.js +114 -0
- package/plugins/pbr/dist/prompt-routing.js +209 -0
- package/plugins/pbr/dist/quick-status.js +179 -0
- package/plugins/pbr/dist/record-incident.js +37 -0
- package/plugins/pbr/dist/run-hook.js +144 -0
- package/plugins/pbr/dist/session-cleanup.js +683 -0
- package/plugins/pbr/dist/session-tracker.js +124 -0
- package/plugins/pbr/dist/status-line.js +847 -0
- package/plugins/pbr/dist/suggest-compact.js +315 -0
- package/plugins/pbr/dist/sync-context-to-claude.js +100 -0
- package/plugins/pbr/dist/task-completed.js +206 -0
- package/plugins/pbr/dist/track-context-budget.js +432 -0
- package/plugins/pbr/dist/track-user-gates.js +88 -0
- package/plugins/pbr/dist/trust-tracker.js +193 -0
- package/plugins/pbr/dist/validate-commit.js +271 -0
- package/plugins/pbr/dist/validate-skill-args.js +222 -0
- package/plugins/pbr/dist/validate-task.js +301 -0
- package/plugins/pbr/dist/worktree-create.js +144 -0
- package/plugins/pbr/dist/worktree-remove.js +147 -0
- package/plugins/pbr/hooks/hooks.json +143 -60
- package/plugins/pbr/references/agent-contracts.md +40 -9
- package/plugins/pbr/references/agent-teams.md +3 -3
- package/plugins/pbr/references/archive/checkpoints.md +189 -0
- package/plugins/pbr/references/archive/context-quality-tiers.md +45 -0
- package/plugins/pbr/references/archive/hook-ordering.md +89 -0
- package/plugins/pbr/references/archive/limitations.md +106 -0
- package/plugins/pbr/references/archive/pbr-rules.md +194 -0
- package/plugins/pbr/references/archive/pbr-tools-cli.md +415 -0
- package/plugins/pbr/references/archive/pretooluse-jsonl-behavior.md +58 -0
- package/plugins/pbr/references/archive/signal-files.md +41 -0
- package/plugins/pbr/references/archive/tmux-setup.md +288 -0
- package/plugins/pbr/references/archive/verification-matrix.md +34 -0
- package/plugins/pbr/references/archive/verification-patterns.md +277 -0
- package/plugins/pbr/references/archive/worktree-sparse-checkout.md +86 -0
- package/plugins/pbr/references/behavioral-contexts.md +53 -0
- package/plugins/pbr/references/checkpoints.md +723 -104
- package/plugins/pbr/references/config-reference.md +472 -10
- package/plugins/pbr/references/continuation-format.md +1 -0
- package/plugins/pbr/references/decimal-phase-calculation.md +65 -0
- package/plugins/pbr/references/deviation-rules.md +12 -0
- package/plugins/pbr/references/git-integration.md +110 -27
- package/plugins/pbr/references/git-planning-commit.md +35 -0
- package/plugins/pbr/references/model-profile-resolution.md +34 -0
- package/plugins/pbr/references/model-profiles.md +90 -7
- package/plugins/pbr/references/model-selection.md +1 -1
- package/plugins/pbr/references/node-repair.md +48 -0
- package/plugins/pbr/references/plan-authoring.md +65 -0
- package/plugins/pbr/references/plan-format.md +161 -10
- package/plugins/pbr/references/questioning.md +138 -49
- package/plugins/pbr/references/reading-verification.md +4 -4
- package/plugins/pbr/references/tdd.md +263 -0
- package/plugins/pbr/references/ui-brand.md +449 -0
- package/plugins/pbr/references/verification-overrides.md +39 -0
- package/plugins/pbr/references/verification-patterns.md +529 -113
- package/plugins/pbr/scripts/architecture-guard.js +59 -0
- package/plugins/pbr/scripts/audit-checks/behavioral-compliance.js +2098 -0
- package/plugins/pbr/scripts/audit-checks/error-analysis.js +989 -0
- package/plugins/pbr/scripts/audit-checks/feature-verification.js +723 -0
- package/plugins/pbr/scripts/audit-checks/index.js +433 -0
- package/plugins/pbr/scripts/audit-checks/infrastructure.js +816 -0
- package/plugins/pbr/scripts/audit-checks/quality-metrics.js +452 -0
- package/plugins/pbr/scripts/audit-checks/session-quality.js +980 -0
- package/plugins/pbr/scripts/audit-checks/si-agent-hook-config-checks.js +466 -0
- package/plugins/pbr/scripts/audit-checks/si-cross-cutting-checks.js +272 -0
- package/plugins/pbr/scripts/audit-checks/si-skill-checks.js +424 -0
- package/plugins/pbr/scripts/audit-checks/workflow-compliance.js +1211 -0
- package/plugins/pbr/scripts/audit-dimensions.js +556 -0
- package/plugins/pbr/scripts/auto-continue.js +198 -31
- package/plugins/pbr/scripts/block-skill-self-read.js +124 -0
- package/plugins/pbr/scripts/check-agent-state-write.js +63 -0
- package/plugins/pbr/scripts/check-config-change.js +162 -0
- package/plugins/pbr/scripts/check-cross-plugin-sync.js +93 -0
- package/plugins/pbr/scripts/check-dangerous-commands.js +18 -5
- package/plugins/pbr/scripts/check-direct-state-write.js +37 -0
- package/plugins/pbr/scripts/check-phase-boundary.js +3 -8
- package/plugins/pbr/scripts/check-plan-format.js +166 -277
- package/plugins/pbr/scripts/check-read-first.js +345 -0
- package/plugins/pbr/scripts/check-roadmap-sync.js +167 -10
- package/plugins/pbr/scripts/check-skill-workflow.js +24 -27
- package/plugins/pbr/scripts/check-state-sync.js +339 -215
- package/plugins/pbr/scripts/check-subagent-output.js +338 -276
- package/plugins/pbr/scripts/check-summary-gate.js +2 -1
- package/plugins/pbr/scripts/config-schema.json +1247 -95
- package/plugins/pbr/scripts/context-bridge.js +425 -0
- package/plugins/pbr/scripts/context-budget-check.js +169 -14
- package/plugins/pbr/scripts/context-quality.js +271 -0
- package/plugins/pbr/scripts/enforce-context-budget.js +138 -0
- package/plugins/pbr/scripts/enforce-pbr-workflow.js +277 -0
- package/plugins/pbr/scripts/event-handler.js +128 -77
- package/plugins/pbr/scripts/event-logger.js +58 -25
- package/plugins/pbr/scripts/feedback-loop.js +172 -0
- package/plugins/pbr/scripts/graph-update.js +422 -0
- package/plugins/pbr/scripts/hook-logger.js +69 -35
- package/plugins/pbr/scripts/hook-server-client.js +361 -0
- package/plugins/pbr/scripts/hook-server.js +606 -0
- package/plugins/pbr/scripts/hooks-schema.json +13 -5
- package/plugins/pbr/scripts/instructions-loaded.js +173 -0
- package/plugins/pbr/scripts/intent-router.cjs +147 -0
- package/plugins/pbr/scripts/intercept-plan-mode.js +52 -18
- package/plugins/pbr/scripts/lib/alternatives.js +203 -0
- package/plugins/pbr/scripts/lib/audit.js +65 -0
- package/plugins/pbr/scripts/lib/auto-cleanup.js +221 -0
- package/plugins/pbr/scripts/lib/auto-verify.js +103 -0
- package/plugins/pbr/scripts/lib/autonomy.js +91 -0
- package/plugins/pbr/scripts/lib/build.js +719 -0
- package/plugins/pbr/scripts/lib/ci-fix-loop.js +228 -0
- package/plugins/pbr/scripts/lib/circuit-state.js +133 -0
- package/plugins/pbr/scripts/lib/commands.js +483 -0
- package/plugins/pbr/scripts/lib/completion.js +377 -0
- package/plugins/pbr/scripts/lib/compound.js +216 -0
- package/plugins/pbr/scripts/lib/config.js +1315 -0
- package/plugins/pbr/scripts/lib/context.js +254 -0
- package/plugins/pbr/scripts/lib/contextual-help.js +207 -0
- package/plugins/pbr/scripts/lib/convention-detector.js +413 -0
- package/plugins/pbr/scripts/lib/core.js +1569 -0
- package/plugins/pbr/scripts/lib/dashboard-launch.js +364 -0
- package/plugins/pbr/scripts/lib/data-hygiene.js +179 -0
- package/plugins/pbr/scripts/lib/decision-extraction.js +183 -0
- package/plugins/pbr/scripts/lib/decisions.js +194 -0
- package/plugins/pbr/scripts/lib/dependency-break.js +147 -0
- package/plugins/pbr/scripts/lib/format-validators.js +1050 -0
- package/plugins/pbr/scripts/lib/frontmatter.js +302 -0
- package/plugins/pbr/scripts/lib/gates/advisories.js +129 -0
- package/plugins/pbr/scripts/lib/gates/build-dependency.js +115 -0
- package/plugins/pbr/scripts/lib/gates/build-executor.js +104 -0
- package/plugins/pbr/scripts/lib/gates/doc-existence.js +46 -0
- package/plugins/pbr/scripts/lib/gates/helpers.js +93 -0
- package/plugins/pbr/scripts/lib/gates/inline-execution.js +185 -0
- package/plugins/pbr/scripts/lib/gates/milestone-complete.js +136 -0
- package/plugins/pbr/scripts/lib/gates/milestone-summary.js +119 -0
- package/plugins/pbr/scripts/lib/gates/multi-phase-loader.js +147 -0
- package/plugins/pbr/scripts/lib/gates/plan-executor.js +36 -0
- package/plugins/pbr/scripts/lib/gates/plan-validation.js +114 -0
- package/plugins/pbr/scripts/lib/gates/quick-executor.js +76 -0
- package/plugins/pbr/scripts/lib/gates/review-planner.js +61 -0
- package/plugins/pbr/scripts/lib/gates/review-verifier.js +69 -0
- package/plugins/pbr/scripts/lib/gates/rich-agent-context.js +143 -0
- package/plugins/pbr/scripts/lib/gates/user-confirmation.js +93 -0
- package/plugins/pbr/scripts/lib/graph-cli.js +89 -0
- package/plugins/pbr/scripts/lib/graph.js +553 -0
- package/plugins/pbr/scripts/lib/health-checks.js +107 -0
- package/plugins/pbr/scripts/lib/health-phase06.js +120 -0
- package/plugins/pbr/scripts/lib/health.js +133 -0
- package/plugins/pbr/scripts/lib/help.js +151 -0
- package/plugins/pbr/scripts/lib/history.js +150 -0
- package/plugins/pbr/scripts/lib/hypothesis-runner.js +127 -0
- package/plugins/pbr/scripts/lib/impact-analysis.js +319 -0
- package/plugins/pbr/scripts/lib/incidents.js +190 -0
- package/plugins/pbr/scripts/lib/init.js +643 -0
- package/plugins/pbr/scripts/lib/insights-parser.js +320 -0
- package/plugins/pbr/scripts/lib/intel.js +653 -0
- package/plugins/pbr/scripts/lib/learnings.js +511 -0
- package/plugins/pbr/scripts/lib/local-llm/client.js +237 -0
- package/plugins/pbr/scripts/lib/local-llm/health.js +12 -0
- package/plugins/pbr/scripts/lib/local-llm/index.js +89 -0
- package/plugins/pbr/scripts/lib/local-llm/metrics.js +20 -0
- package/plugins/pbr/scripts/lib/local-llm/operations/classify-artifact.js +4 -0
- package/plugins/pbr/scripts/lib/local-llm/operations/classify-commit.js +4 -0
- package/plugins/pbr/scripts/lib/local-llm/operations/classify-error.js +4 -0
- package/plugins/pbr/scripts/lib/local-llm/operations/classify-file-intent.js +4 -0
- package/plugins/pbr/scripts/lib/local-llm/operations/score-source.js +72 -0
- package/plugins/pbr/scripts/lib/local-llm/operations/summarize-context.js +62 -0
- package/plugins/pbr/scripts/lib/local-llm/operations/triage-test-output.js +12 -0
- package/plugins/pbr/scripts/lib/local-llm/operations/validate-task.js +4 -0
- package/plugins/pbr/scripts/lib/local-llm/router.js +101 -0
- package/plugins/pbr/scripts/lib/local-llm/shadow.js +60 -0
- package/plugins/pbr/scripts/lib/local-llm/threshold-tuner.js +118 -0
- package/plugins/pbr/scripts/lib/migrate.js +298 -0
- package/plugins/pbr/scripts/lib/milestone.js +306 -0
- package/plugins/pbr/scripts/lib/negative-knowledge.js +194 -0
- package/plugins/pbr/scripts/lib/notification-throttle.js +141 -0
- package/plugins/pbr/scripts/lib/onboarding-generator.js +288 -0
- package/plugins/pbr/scripts/lib/parse-args.js +134 -0
- package/plugins/pbr/scripts/lib/pattern-routing.js +55 -0
- package/plugins/pbr/scripts/lib/patterns.js +272 -0
- package/plugins/pbr/scripts/lib/perf.js +190 -0
- package/plugins/pbr/scripts/lib/phase.js +1027 -0
- package/plugins/pbr/scripts/lib/pid-lock.js +154 -0
- package/plugins/pbr/scripts/lib/post-hoc.js +160 -0
- package/plugins/pbr/scripts/lib/pre-commit-checks.js +220 -0
- package/plugins/pbr/scripts/lib/pre-research.js +133 -0
- package/plugins/pbr/scripts/lib/preview.js +174 -0
- package/plugins/pbr/scripts/lib/progress-visualization.js +296 -0
- package/plugins/pbr/scripts/lib/quick-init.js +131 -0
- package/plugins/pbr/scripts/lib/reference.js +236 -0
- package/plugins/pbr/scripts/lib/requirements.js +153 -0
- package/plugins/pbr/scripts/lib/resolve-root.js +66 -0
- package/plugins/pbr/scripts/lib/reverse-spec.js +259 -0
- package/plugins/pbr/scripts/lib/roadmap.js +1113 -0
- package/plugins/pbr/scripts/lib/security-scan.js +200 -0
- package/plugins/pbr/scripts/lib/session-briefing.js +895 -0
- package/plugins/pbr/scripts/lib/skill-section.js +99 -0
- package/plugins/pbr/scripts/lib/smart-next-task.js +207 -0
- package/plugins/pbr/scripts/lib/snapshot-manager.js +232 -0
- package/plugins/pbr/scripts/lib/spec-diff.js +209 -0
- package/plugins/pbr/scripts/lib/spec-engine.js +189 -0
- package/plugins/pbr/scripts/lib/spot-check.js +642 -0
- package/plugins/pbr/scripts/lib/state-queue.js +171 -0
- package/plugins/pbr/scripts/lib/state.js +1187 -0
- package/plugins/pbr/scripts/lib/status-render.js +511 -0
- package/plugins/pbr/scripts/lib/step-verify.js +149 -0
- package/plugins/pbr/scripts/lib/subagent-validators.js +1059 -0
- package/plugins/pbr/scripts/lib/suggest-next.js +435 -0
- package/plugins/pbr/scripts/lib/team-composer.js +87 -0
- package/plugins/pbr/scripts/lib/team-coordinator.js +153 -0
- package/plugins/pbr/scripts/lib/tech-debt-scanner.js +116 -0
- package/plugins/pbr/scripts/lib/template.js +222 -0
- package/plugins/pbr/scripts/lib/templates.js +362 -0
- package/plugins/pbr/scripts/lib/test-cache.js +54 -0
- package/plugins/pbr/scripts/lib/test-selection.js +163 -0
- package/plugins/pbr/scripts/lib/todo.js +300 -0
- package/plugins/pbr/scripts/lib/trust-gate.js +84 -0
- package/plugins/pbr/scripts/lib/verify.js +1473 -0
- package/plugins/pbr/scripts/lib/wiring-check.js +196 -0
- package/plugins/pbr/scripts/log-notification.js +131 -0
- package/plugins/pbr/scripts/log-subagent.js +203 -18
- package/plugins/pbr/scripts/log-tool-failure.js +60 -8
- package/plugins/pbr/scripts/milestone-learnings.js +569 -0
- package/plugins/pbr/scripts/package.json +1 -0
- package/plugins/pbr/scripts/pbr-tools.js +1833 -1167
- package/plugins/pbr/scripts/post-bash-triage.js +154 -0
- package/plugins/pbr/scripts/post-compact.js +135 -0
- package/plugins/pbr/scripts/post-hoc.js +286 -0
- package/plugins/pbr/scripts/post-write-dispatch.js +237 -31
- package/plugins/pbr/scripts/post-write-quality.js +4 -3
- package/plugins/pbr/scripts/pre-bash-dispatch.js +155 -23
- package/plugins/pbr/scripts/pre-skill-dispatch.js +114 -0
- package/plugins/pbr/scripts/pre-task-dispatch.js +297 -0
- package/plugins/pbr/scripts/pre-write-dispatch.js +171 -47
- package/plugins/pbr/scripts/progress-tracker.js +144 -307
- package/plugins/pbr/scripts/prompt-guard.js +114 -0
- package/plugins/pbr/scripts/prompt-routing.js +209 -0
- package/plugins/pbr/scripts/quick-status.js +179 -0
- package/plugins/pbr/scripts/record-incident.js +37 -0
- package/plugins/pbr/scripts/risk-classifier.cjs +123 -0
- package/plugins/pbr/scripts/run-hook.js +62 -10
- package/plugins/pbr/scripts/session-cleanup.js +458 -29
- package/plugins/pbr/scripts/session-tracker.js +124 -0
- package/plugins/pbr/scripts/status-line.js +591 -32
- package/plugins/pbr/scripts/suggest-compact.js +203 -7
- package/plugins/pbr/scripts/sync-context-to-claude.js +100 -0
- package/plugins/pbr/scripts/task-completed.js +165 -4
- package/plugins/pbr/scripts/test/config.test.js +126 -0
- package/plugins/pbr/scripts/test/cross-platform.test.js +131 -0
- package/plugins/pbr/scripts/test/fixtures/config.json +20 -0
- package/plugins/pbr/scripts/test/fixtures/plan.md +54 -0
- package/plugins/pbr/scripts/test/fixtures/project.md +30 -0
- package/plugins/pbr/scripts/test/fixtures/roadmap.md +55 -0
- package/plugins/pbr/scripts/test/fixtures/state.md +60 -0
- package/plugins/pbr/scripts/test/fixtures/summary.md +35 -0
- package/plugins/pbr/scripts/test/fixtures.test.js +184 -0
- package/plugins/pbr/scripts/test/phase.test.js +142 -0
- package/plugins/pbr/scripts/test/roadmap.test.js +96 -0
- package/plugins/pbr/scripts/test/state.test.js +163 -0
- package/plugins/pbr/scripts/track-context-budget.js +368 -99
- package/plugins/pbr/scripts/track-user-gates.js +88 -0
- package/plugins/pbr/scripts/trust-tracker.js +193 -0
- package/plugins/pbr/scripts/validate-commit.js +97 -26
- package/plugins/pbr/scripts/validate-skill-args.js +87 -15
- package/plugins/pbr/scripts/validate-task.js +112 -626
- package/plugins/pbr/scripts/worktree-create.js +144 -0
- package/plugins/pbr/scripts/worktree-remove.js +147 -0
- package/plugins/pbr/skills/audit/SKILL.md +195 -24
- package/plugins/pbr/skills/audit-fix/SKILL.md +326 -0
- package/plugins/pbr/skills/autonomous/SKILL.md +545 -0
- package/plugins/pbr/skills/backlog/SKILL.md +56 -0
- package/plugins/pbr/skills/begin/SKILL.md +507 -153
- package/plugins/pbr/skills/begin/templates/STATE.md.tmpl +1 -2
- package/plugins/pbr/skills/begin/templates/config.json.tmpl +415 -36
- package/plugins/pbr/skills/begin/templates/researcher-prompt.md.tmpl +28 -0
- package/plugins/pbr/skills/begin/templates/roadmap-prompt.md.tmpl +28 -3
- package/plugins/pbr/skills/begin/templates/synthesis-prompt.md.tmpl +33 -5
- package/plugins/pbr/skills/build/SKILL.md +1040 -354
- package/plugins/pbr/skills/build/templates/continuation-prompt.md.tmpl +26 -0
- package/plugins/pbr/skills/build/templates/executor-prompt.md.tmpl +77 -0
- package/plugins/pbr/skills/build/templates/inline-verifier-prompt.md.tmpl +33 -0
- package/plugins/pbr/skills/config/SKILL.md +111 -9
- package/plugins/pbr/skills/continue/SKILL.md +113 -33
- package/plugins/pbr/skills/dashboard/SKILL.md +21 -9
- package/plugins/pbr/skills/debug/SKILL.md +70 -12
- package/plugins/pbr/skills/debug/templates/continuation-prompt.md.tmpl +12 -1
- package/plugins/pbr/skills/debug/templates/initial-investigation-prompt.md.tmpl +12 -5
- package/plugins/pbr/skills/discuss/SKILL.md +206 -25
- package/plugins/pbr/skills/discuss/templates/CONTEXT.md.tmpl +21 -1
- package/plugins/pbr/skills/do/SKILL.md +119 -24
- package/plugins/pbr/skills/explore/SKILL.md +95 -20
- package/plugins/pbr/skills/fast/SKILL.md +94 -0
- package/plugins/pbr/skills/forensics/SKILL.md +144 -0
- package/plugins/pbr/skills/health/SKILL.md +44 -117
- package/plugins/pbr/skills/help/SKILL.md +83 -101
- package/plugins/pbr/skills/import/SKILL.md +332 -13
- package/plugins/pbr/skills/intel/SKILL.md +131 -0
- package/plugins/pbr/skills/list-phase-assumptions/SKILL.md +231 -0
- package/plugins/pbr/skills/milestone/SKILL.md +421 -263
- package/plugins/pbr/skills/milestone/templates/audit-output.md.tmpl +76 -0
- package/plugins/pbr/skills/milestone/templates/complete-output.md.tmpl +32 -0
- package/plugins/pbr/skills/milestone/templates/edge-cases.md +54 -0
- package/plugins/pbr/skills/milestone/templates/gaps-output.md.tmpl +25 -0
- package/plugins/pbr/skills/milestone/templates/integration-checker-prompt.md.tmpl +25 -0
- package/plugins/pbr/skills/milestone/templates/new-output.md.tmpl +29 -0
- package/plugins/pbr/skills/milestone-summary/SKILL.md +86 -0
- package/plugins/pbr/skills/note/SKILL.md +20 -4
- package/plugins/pbr/skills/pause/SKILL.md +53 -14
- package/plugins/pbr/skills/pause/templates/continue-here.md.tmpl +33 -52
- package/plugins/pbr/skills/plan/SKILL.md +526 -280
- package/plugins/pbr/skills/plan/templates/checker-prompt.md.tmpl +5 -2
- package/plugins/pbr/skills/plan/templates/completion-output.md.tmpl +27 -0
- package/plugins/pbr/skills/plan/templates/planner-prompt.md.tmpl +27 -1
- package/plugins/pbr/skills/plan/templates/revision-prompt.md.tmpl +21 -5
- package/plugins/pbr/skills/profile/SKILL.md +185 -0
- package/plugins/pbr/skills/profile-user/SKILL.md +226 -0
- package/plugins/pbr/skills/quick/SKILL.md +434 -100
- package/plugins/pbr/skills/release/SKILL.md +206 -0
- package/plugins/pbr/skills/resume/SKILL.md +169 -46
- package/plugins/pbr/skills/review/SKILL.md +217 -164
- package/plugins/pbr/skills/review/templates/verifier-prompt.md.tmpl +7 -0
- package/plugins/pbr/skills/scan/SKILL.md +151 -106
- package/plugins/pbr/skills/scan/templates/mapper-prompt.md.tmpl +5 -56
- package/plugins/pbr/skills/seed/SKILL.md +87 -0
- package/plugins/pbr/skills/session-report/SKILL.md +130 -0
- package/plugins/pbr/skills/setup/SKILL.md +149 -202
- package/plugins/pbr/skills/shared/agent-context-enrichment.md +21 -0
- package/plugins/pbr/skills/shared/agent-type-resolution.md +32 -0
- package/plugins/pbr/skills/shared/commit-planning-docs.md +8 -0
- package/plugins/pbr/skills/shared/context-budget.md +66 -1
- package/plugins/pbr/skills/shared/context-loader-task.md +18 -11
- package/plugins/pbr/skills/shared/digest-select.md +2 -2
- package/plugins/pbr/skills/shared/domain-probes.md +1 -1
- package/plugins/pbr/skills/shared/error-reporting.md +38 -60
- package/plugins/pbr/skills/shared/gate-prompts.md +4 -2
- package/plugins/pbr/skills/shared/memory-capture.md +48 -0
- package/plugins/pbr/skills/shared/phase-argument-parsing.md +4 -4
- package/plugins/pbr/skills/shared/revision-loop.md +24 -6
- package/plugins/pbr/skills/shared/state-update.md +47 -54
- package/plugins/pbr/skills/shared/universal-anti-patterns.md +27 -4
- package/plugins/pbr/skills/ship/SKILL.md +155 -0
- package/plugins/pbr/skills/stats/SKILL.md +80 -0
- package/plugins/pbr/skills/status/SKILL.md +184 -119
- package/plugins/pbr/skills/test/SKILL.md +254 -0
- package/plugins/pbr/skills/thread/SKILL.md +73 -0
- package/plugins/pbr/skills/todo/SKILL.md +28 -72
- package/plugins/pbr/skills/ui-phase/SKILL.md +180 -0
- package/plugins/pbr/skills/ui-review/SKILL.md +206 -0
- package/plugins/pbr/skills/undo/SKILL.md +221 -0
- package/plugins/pbr/skills/validate-phase/SKILL.md +362 -0
- package/plugins/pbr/templates/CONTEXT.md.tmpl +45 -20
- package/plugins/pbr/templates/DISCOVERY.md.tmpl +29 -0
- package/plugins/pbr/templates/DISCUSSION-LOG.md.tmpl +49 -0
- package/plugins/pbr/templates/HANDOFF.json.tmpl +30 -0
- package/plugins/pbr/templates/INTEGRATION-REPORT.md.tmpl +18 -2
- package/plugins/pbr/templates/MILESTONE-AUDIT.md.tmpl +44 -0
- package/plugins/pbr/templates/PROJECT.md.tmpl +126 -0
- package/plugins/pbr/templates/REQUIREMENTS.md.tmpl +96 -0
- package/plugins/pbr/templates/RETROSPECTIVE.md.tmpl +43 -0
- package/plugins/pbr/templates/ROADMAP.md.tmpl +108 -14
- package/plugins/pbr/templates/SUMMARY-complex.md.tmpl +133 -0
- package/plugins/pbr/templates/SUMMARY-minimal.md.tmpl +55 -0
- package/plugins/pbr/templates/SUMMARY.md.tmpl +21 -0
- package/plugins/pbr/templates/UAT.md.tmpl +94 -0
- package/plugins/pbr/templates/UI-SPEC.md.tmpl +144 -0
- package/plugins/pbr/templates/VALIDATION.md.tmpl +94 -0
- package/plugins/pbr/templates/VERIFICATION-DETAIL.md.tmpl +49 -13
- package/plugins/pbr/templates/project-CONTEXT.md.tmpl +59 -0
- package/plugins/pbr/templates/research-outputs/ARCHITECTURE.md.tmpl +91 -0
- package/plugins/pbr/templates/research-outputs/FEATURES.md.tmpl +64 -0
- package/plugins/pbr/templates/research-outputs/PITFALLS.md.tmpl +50 -0
- package/plugins/pbr/templates/research-outputs/STACK.md.tmpl +63 -0
- package/plugins/pbr/templates/research-outputs/SUMMARY.md.tmpl +98 -0
- package/scripts/build-hooks.js +61 -0
- package/scripts/check-ci.js +100 -0
- package/scripts/clean-changelog.js +364 -0
- package/scripts/generate-derivatives.js +581 -0
- package/scripts/posttest.js +93 -0
- package/scripts/release.js +262 -0
- package/scripts/run-tests.cjs +29 -0
- package/scripts/test-wrapper.js +43 -0
- package/dashboard/bin/cli.js +0 -25
- package/dashboard/public/css/layout.css +0 -636
- package/dashboard/public/css/status-colors.css +0 -98
- package/dashboard/public/css/tokens.css +0 -59
- package/dashboard/public/js/htmx-title.js +0 -5
- package/dashboard/public/js/sidebar-toggle.js +0 -34
- package/dashboard/public/js/sse-client.js +0 -100
- package/dashboard/public/js/theme-toggle.js +0 -46
- package/dashboard/src/app.js +0 -91
- package/dashboard/src/middleware/current-phase.js +0 -24
- package/dashboard/src/middleware/errorHandler.js +0 -52
- package/dashboard/src/middleware/notFoundHandler.js +0 -9
- package/dashboard/src/repositories/planning.repository.js +0 -130
- package/dashboard/src/routes/events.routes.js +0 -45
- package/dashboard/src/routes/index.routes.js +0 -35
- package/dashboard/src/routes/pages.routes.js +0 -426
- package/dashboard/src/server.js +0 -42
- package/dashboard/src/services/analytics.service.js +0 -141
- package/dashboard/src/services/dashboard.service.js +0 -309
- package/dashboard/src/services/milestone.service.js +0 -222
- package/dashboard/src/services/notes.service.js +0 -50
- package/dashboard/src/services/phase.service.js +0 -232
- package/dashboard/src/services/project.service.js +0 -57
- package/dashboard/src/services/roadmap.service.js +0 -244
- package/dashboard/src/services/sse.service.js +0 -58
- package/dashboard/src/services/todo.service.js +0 -263
- package/dashboard/src/services/watcher.service.js +0 -48
- package/dashboard/src/utils/cache.js +0 -55
- package/dashboard/src/views/analytics.ejs +0 -5
- package/dashboard/src/views/coming-soon.ejs +0 -11
- package/dashboard/src/views/dependencies.ejs +0 -5
- package/dashboard/src/views/error.ejs +0 -20
- package/dashboard/src/views/index.ejs +0 -5
- package/dashboard/src/views/milestone-detail.ejs +0 -5
- package/dashboard/src/views/milestones.ejs +0 -5
- package/dashboard/src/views/notes.ejs +0 -5
- package/dashboard/src/views/partials/analytics-content.ejs +0 -90
- package/dashboard/src/views/partials/breadcrumbs.ejs +0 -14
- package/dashboard/src/views/partials/dashboard-content.ejs +0 -84
- package/dashboard/src/views/partials/dependencies-content.ejs +0 -48
- package/dashboard/src/views/partials/empty-state.ejs +0 -7
- package/dashboard/src/views/partials/footer.ejs +0 -3
- package/dashboard/src/views/partials/head.ejs +0 -30
- package/dashboard/src/views/partials/header.ejs +0 -21
- package/dashboard/src/views/partials/layout-bottom.ejs +0 -6
- package/dashboard/src/views/partials/layout-top.ejs +0 -15
- package/dashboard/src/views/partials/milestone-detail-content.ejs +0 -20
- package/dashboard/src/views/partials/milestones-content.ejs +0 -88
- package/dashboard/src/views/partials/notes-content.ejs +0 -23
- package/dashboard/src/views/partials/phase-content.ejs +0 -193
- package/dashboard/src/views/partials/phase-doc-content.ejs +0 -38
- package/dashboard/src/views/partials/phases-content.ejs +0 -124
- package/dashboard/src/views/partials/roadmap-content.ejs +0 -155
- package/dashboard/src/views/partials/sidebar.ejs +0 -99
- package/dashboard/src/views/partials/todo-create-content.ejs +0 -54
- package/dashboard/src/views/partials/todo-detail-content.ejs +0 -42
- package/dashboard/src/views/partials/todos-content.ejs +0 -94
- package/dashboard/src/views/phase-detail.ejs +0 -5
- package/dashboard/src/views/phase-doc.ejs +0 -5
- package/dashboard/src/views/phases.ejs +0 -5
- package/dashboard/src/views/roadmap.ejs +0 -5
- package/dashboard/src/views/todo-create.ejs +0 -5
- package/dashboard/src/views/todo-detail.ejs +0 -5
- package/dashboard/src/views/todos.ejs +0 -5
- package/plugins/copilot-pbr/CHANGELOG.md +0 -19
- package/plugins/copilot-pbr/README.md +0 -139
- package/plugins/copilot-pbr/agents/audit.agent.md +0 -113
- package/plugins/copilot-pbr/agents/codebase-mapper.agent.md +0 -151
- package/plugins/copilot-pbr/agents/debugger.agent.md +0 -172
- package/plugins/copilot-pbr/agents/executor.agent.md +0 -267
- package/plugins/copilot-pbr/agents/general.agent.md +0 -88
- package/plugins/copilot-pbr/agents/integration-checker.agent.md +0 -119
- package/plugins/copilot-pbr/agents/plan-checker.agent.md +0 -199
- package/plugins/copilot-pbr/agents/planner.agent.md +0 -238
- package/plugins/copilot-pbr/agents/researcher.agent.md +0 -186
- package/plugins/copilot-pbr/agents/synthesizer.agent.md +0 -126
- package/plugins/copilot-pbr/agents/verifier.agent.md +0 -228
- package/plugins/copilot-pbr/hooks/hooks.json +0 -156
- package/plugins/copilot-pbr/plugin.json +0 -30
- package/plugins/copilot-pbr/references/agent-anti-patterns.md +0 -25
- package/plugins/copilot-pbr/references/agent-contracts.md +0 -297
- package/plugins/copilot-pbr/references/agent-interactions.md +0 -135
- package/plugins/copilot-pbr/references/agent-teams.md +0 -55
- package/plugins/copilot-pbr/references/checkpoints.md +0 -158
- package/plugins/copilot-pbr/references/common-bug-patterns.md +0 -14
- package/plugins/copilot-pbr/references/config-reference.md +0 -442
- package/plugins/copilot-pbr/references/continuation-format.md +0 -213
- package/plugins/copilot-pbr/references/deviation-rules.md +0 -113
- package/plugins/copilot-pbr/references/git-integration.md +0 -227
- package/plugins/copilot-pbr/references/integration-patterns.md +0 -118
- package/plugins/copilot-pbr/references/model-profiles.md +0 -100
- package/plugins/copilot-pbr/references/model-selection.md +0 -32
- package/plugins/copilot-pbr/references/pbr-rules.md +0 -195
- package/plugins/copilot-pbr/references/pbr-tools-cli.md +0 -285
- package/plugins/copilot-pbr/references/plan-authoring.md +0 -182
- package/plugins/copilot-pbr/references/plan-format.md +0 -288
- package/plugins/copilot-pbr/references/planning-config.md +0 -214
- package/plugins/copilot-pbr/references/questioning.md +0 -215
- package/plugins/copilot-pbr/references/reading-verification.md +0 -128
- package/plugins/copilot-pbr/references/stub-patterns.md +0 -161
- package/plugins/copilot-pbr/references/subagent-coordination.md +0 -120
- package/plugins/copilot-pbr/references/ui-formatting.md +0 -444
- package/plugins/copilot-pbr/references/verification-patterns.md +0 -199
- package/plugins/copilot-pbr/references/wave-execution.md +0 -96
- package/plugins/copilot-pbr/rules/pbr-workflow.mdc +0 -48
- package/plugins/copilot-pbr/setup.ps1 +0 -93
- package/plugins/copilot-pbr/setup.sh +0 -93
- package/plugins/copilot-pbr/skills/audit/SKILL.md +0 -330
- package/plugins/copilot-pbr/skills/begin/SKILL.md +0 -589
- package/plugins/copilot-pbr/skills/begin/templates/PROJECT.md.tmpl +0 -34
- package/plugins/copilot-pbr/skills/begin/templates/REQUIREMENTS.md.tmpl +0 -19
- package/plugins/copilot-pbr/skills/begin/templates/STATE.md.tmpl +0 -50
- package/plugins/copilot-pbr/skills/begin/templates/config.json.tmpl +0 -64
- package/plugins/copilot-pbr/skills/begin/templates/researcher-prompt.md.tmpl +0 -20
- package/plugins/copilot-pbr/skills/begin/templates/roadmap-prompt.md.tmpl +0 -31
- package/plugins/copilot-pbr/skills/begin/templates/synthesis-prompt.md.tmpl +0 -17
- package/plugins/copilot-pbr/skills/build/SKILL.md +0 -960
- package/plugins/copilot-pbr/skills/config/SKILL.md +0 -250
- package/plugins/copilot-pbr/skills/continue/SKILL.md +0 -159
- package/plugins/copilot-pbr/skills/dashboard/SKILL.md +0 -43
- package/plugins/copilot-pbr/skills/debug/SKILL.md +0 -508
- package/plugins/copilot-pbr/skills/debug/templates/continuation-prompt.md.tmpl +0 -17
- package/plugins/copilot-pbr/skills/debug/templates/initial-investigation-prompt.md.tmpl +0 -28
- package/plugins/copilot-pbr/skills/discuss/SKILL.md +0 -353
- package/plugins/copilot-pbr/skills/discuss/templates/CONTEXT.md.tmpl +0 -62
- package/plugins/copilot-pbr/skills/discuss/templates/decision-categories.md +0 -10
- package/plugins/copilot-pbr/skills/do/SKILL.md +0 -66
- package/plugins/copilot-pbr/skills/explore/SKILL.md +0 -373
- package/plugins/copilot-pbr/skills/health/SKILL.md +0 -274
- package/plugins/copilot-pbr/skills/health/templates/check-pattern.md.tmpl +0 -31
- package/plugins/copilot-pbr/skills/health/templates/output-format.md.tmpl +0 -64
- package/plugins/copilot-pbr/skills/help/SKILL.md +0 -170
- package/plugins/copilot-pbr/skills/import/SKILL.md +0 -502
- package/plugins/copilot-pbr/skills/milestone/SKILL.md +0 -745
- package/plugins/copilot-pbr/skills/milestone/templates/audit-report.md.tmpl +0 -49
- package/plugins/copilot-pbr/skills/milestone/templates/stats-file.md.tmpl +0 -31
- package/plugins/copilot-pbr/skills/note/SKILL.md +0 -213
- package/plugins/copilot-pbr/skills/pause/SKILL.md +0 -247
- package/plugins/copilot-pbr/skills/pause/templates/continue-here.md.tmpl +0 -72
- package/plugins/copilot-pbr/skills/plan/SKILL.md +0 -662
- package/plugins/copilot-pbr/skills/plan/templates/checker-prompt.md.tmpl +0 -22
- package/plugins/copilot-pbr/skills/plan/templates/gap-closure-prompt.md.tmpl +0 -33
- package/plugins/copilot-pbr/skills/plan/templates/planner-prompt.md.tmpl +0 -39
- package/plugins/copilot-pbr/skills/plan/templates/researcher-prompt.md.tmpl +0 -20
- package/plugins/copilot-pbr/skills/plan/templates/revision-prompt.md.tmpl +0 -24
- package/plugins/copilot-pbr/skills/quick/SKILL.md +0 -376
- package/plugins/copilot-pbr/skills/resume/SKILL.md +0 -399
- package/plugins/copilot-pbr/skills/review/SKILL.md +0 -653
- package/plugins/copilot-pbr/skills/review/templates/debugger-prompt.md.tmpl +0 -61
- package/plugins/copilot-pbr/skills/review/templates/gap-planner-prompt.md.tmpl +0 -41
- package/plugins/copilot-pbr/skills/review/templates/verifier-prompt.md.tmpl +0 -116
- package/plugins/copilot-pbr/skills/scan/SKILL.md +0 -299
- package/plugins/copilot-pbr/skills/scan/templates/mapper-prompt.md.tmpl +0 -202
- package/plugins/copilot-pbr/skills/setup/SKILL.md +0 -296
- package/plugins/copilot-pbr/skills/shared/commit-planning-docs.md +0 -36
- package/plugins/copilot-pbr/skills/shared/config-loading.md +0 -103
- package/plugins/copilot-pbr/skills/shared/context-budget.md +0 -41
- package/plugins/copilot-pbr/skills/shared/context-loader-task.md +0 -87
- package/plugins/copilot-pbr/skills/shared/digest-select.md +0 -80
- package/plugins/copilot-pbr/skills/shared/domain-probes.md +0 -126
- package/plugins/copilot-pbr/skills/shared/error-reporting.md +0 -81
- package/plugins/copilot-pbr/skills/shared/gate-prompts.md +0 -389
- package/plugins/copilot-pbr/skills/shared/phase-argument-parsing.md +0 -46
- package/plugins/copilot-pbr/skills/shared/progress-display.md +0 -53
- package/plugins/copilot-pbr/skills/shared/revision-loop.md +0 -82
- package/plugins/copilot-pbr/skills/shared/state-loading.md +0 -63
- package/plugins/copilot-pbr/skills/shared/state-update.md +0 -162
- package/plugins/copilot-pbr/skills/shared/universal-anti-patterns.md +0 -38
- package/plugins/copilot-pbr/skills/status/SKILL.md +0 -362
- package/plugins/copilot-pbr/skills/statusline/SKILL.md +0 -149
- package/plugins/copilot-pbr/skills/todo/SKILL.md +0 -279
- package/plugins/copilot-pbr/templates/CONTEXT.md.tmpl +0 -53
- package/plugins/copilot-pbr/templates/INTEGRATION-REPORT.md.tmpl +0 -152
- package/plugins/copilot-pbr/templates/RESEARCH-SUMMARY.md.tmpl +0 -98
- package/plugins/copilot-pbr/templates/ROADMAP.md.tmpl +0 -41
- package/plugins/copilot-pbr/templates/SUMMARY.md.tmpl +0 -82
- package/plugins/copilot-pbr/templates/VERIFICATION-DETAIL.md.tmpl +0 -117
- package/plugins/copilot-pbr/templates/codebase/ARCHITECTURE.md.tmpl +0 -98
- package/plugins/copilot-pbr/templates/codebase/CONCERNS.md.tmpl +0 -93
- package/plugins/copilot-pbr/templates/codebase/CONVENTIONS.md.tmpl +0 -104
- package/plugins/copilot-pbr/templates/codebase/INTEGRATIONS.md.tmpl +0 -78
- package/plugins/copilot-pbr/templates/codebase/STACK.md.tmpl +0 -78
- package/plugins/copilot-pbr/templates/codebase/STRUCTURE.md.tmpl +0 -80
- package/plugins/copilot-pbr/templates/codebase/TESTING.md.tmpl +0 -107
- package/plugins/copilot-pbr/templates/continue-here.md.tmpl +0 -74
- package/plugins/copilot-pbr/templates/prompt-partials/phase-project-context.md.tmpl +0 -38
- package/plugins/copilot-pbr/templates/research/ARCHITECTURE.md.tmpl +0 -124
- package/plugins/copilot-pbr/templates/research/STACK.md.tmpl +0 -71
- package/plugins/copilot-pbr/templates/research/SUMMARY.md.tmpl +0 -112
- package/plugins/copilot-pbr/templates/research-outputs/phase-research.md.tmpl +0 -81
- package/plugins/copilot-pbr/templates/research-outputs/project-research.md.tmpl +0 -99
- package/plugins/copilot-pbr/templates/research-outputs/synthesis.md.tmpl +0 -36
- package/plugins/cursor-pbr/.cursor-plugin/plugin.json +0 -32
- package/plugins/cursor-pbr/CHANGELOG.md +0 -15
- package/plugins/cursor-pbr/README.md +0 -123
- package/plugins/cursor-pbr/agents/audit.md +0 -178
- package/plugins/cursor-pbr/agents/codebase-mapper.md +0 -150
- package/plugins/cursor-pbr/agents/debugger.md +0 -171
- package/plugins/cursor-pbr/agents/executor.md +0 -266
- package/plugins/cursor-pbr/agents/general.md +0 -87
- package/plugins/cursor-pbr/agents/integration-checker.md +0 -118
- package/plugins/cursor-pbr/agents/plan-checker.md +0 -198
- package/plugins/cursor-pbr/agents/planner.md +0 -237
- package/plugins/cursor-pbr/agents/researcher.md +0 -185
- package/plugins/cursor-pbr/agents/synthesizer.md +0 -125
- package/plugins/cursor-pbr/agents/verifier.md +0 -227
- package/plugins/cursor-pbr/assets/.gitkeep +0 -0
- package/plugins/cursor-pbr/assets/logo.svg +0 -21
- package/plugins/cursor-pbr/hooks/hooks.json +0 -213
- package/plugins/cursor-pbr/references/agent-anti-patterns.md +0 -25
- package/plugins/cursor-pbr/references/agent-contracts.md +0 -297
- package/plugins/cursor-pbr/references/agent-interactions.md +0 -135
- package/plugins/cursor-pbr/references/agent-teams.md +0 -55
- package/plugins/cursor-pbr/references/checkpoints.md +0 -158
- package/plugins/cursor-pbr/references/common-bug-patterns.md +0 -14
- package/plugins/cursor-pbr/references/config-reference.md +0 -442
- package/plugins/cursor-pbr/references/continuation-format.md +0 -213
- package/plugins/cursor-pbr/references/deviation-rules.md +0 -113
- package/plugins/cursor-pbr/references/git-integration.md +0 -227
- package/plugins/cursor-pbr/references/integration-patterns.md +0 -118
- package/plugins/cursor-pbr/references/model-profiles.md +0 -100
- package/plugins/cursor-pbr/references/model-selection.md +0 -32
- package/plugins/cursor-pbr/references/pbr-rules.md +0 -195
- package/plugins/cursor-pbr/references/pbr-tools-cli.md +0 -285
- package/plugins/cursor-pbr/references/plan-authoring.md +0 -182
- package/plugins/cursor-pbr/references/plan-format.md +0 -288
- package/plugins/cursor-pbr/references/planning-config.md +0 -214
- package/plugins/cursor-pbr/references/questioning.md +0 -215
- package/plugins/cursor-pbr/references/reading-verification.md +0 -128
- package/plugins/cursor-pbr/references/stub-patterns.md +0 -161
- package/plugins/cursor-pbr/references/subagent-coordination.md +0 -120
- package/plugins/cursor-pbr/references/ui-formatting.md +0 -444
- package/plugins/cursor-pbr/references/verification-patterns.md +0 -199
- package/plugins/cursor-pbr/references/wave-execution.md +0 -96
- package/plugins/cursor-pbr/rules/pbr-workflow.mdc +0 -48
- package/plugins/cursor-pbr/setup.ps1 +0 -78
- package/plugins/cursor-pbr/setup.sh +0 -83
- package/plugins/cursor-pbr/skills/audit/SKILL.md +0 -331
- package/plugins/cursor-pbr/skills/begin/SKILL.md +0 -589
- package/plugins/cursor-pbr/skills/begin/templates/PROJECT.md.tmpl +0 -34
- package/plugins/cursor-pbr/skills/begin/templates/REQUIREMENTS.md.tmpl +0 -19
- package/plugins/cursor-pbr/skills/begin/templates/STATE.md.tmpl +0 -50
- package/plugins/cursor-pbr/skills/begin/templates/config.json.tmpl +0 -64
- package/plugins/cursor-pbr/skills/begin/templates/researcher-prompt.md.tmpl +0 -20
- package/plugins/cursor-pbr/skills/begin/templates/roadmap-prompt.md.tmpl +0 -31
- package/plugins/cursor-pbr/skills/begin/templates/synthesis-prompt.md.tmpl +0 -17
- package/plugins/cursor-pbr/skills/build/SKILL.md +0 -961
- package/plugins/cursor-pbr/skills/config/SKILL.md +0 -252
- package/plugins/cursor-pbr/skills/continue/SKILL.md +0 -159
- package/plugins/cursor-pbr/skills/dashboard/SKILL.md +0 -44
- package/plugins/cursor-pbr/skills/debug/SKILL.md +0 -512
- package/plugins/cursor-pbr/skills/debug/templates/continuation-prompt.md.tmpl +0 -17
- package/plugins/cursor-pbr/skills/debug/templates/initial-investigation-prompt.md.tmpl +0 -28
- package/plugins/cursor-pbr/skills/discuss/SKILL.md +0 -354
- package/plugins/cursor-pbr/skills/discuss/templates/CONTEXT.md.tmpl +0 -62
- package/plugins/cursor-pbr/skills/discuss/templates/decision-categories.md +0 -10
- package/plugins/cursor-pbr/skills/do/SKILL.md +0 -67
- package/plugins/cursor-pbr/skills/explore/SKILL.md +0 -376
- package/plugins/cursor-pbr/skills/health/SKILL.md +0 -274
- package/plugins/cursor-pbr/skills/health/templates/check-pattern.md.tmpl +0 -31
- package/plugins/cursor-pbr/skills/health/templates/output-format.md.tmpl +0 -64
- package/plugins/cursor-pbr/skills/help/SKILL.md +0 -170
- package/plugins/cursor-pbr/skills/import/SKILL.md +0 -505
- package/plugins/cursor-pbr/skills/milestone/SKILL.md +0 -746
- package/plugins/cursor-pbr/skills/milestone/templates/audit-report.md.tmpl +0 -49
- package/plugins/cursor-pbr/skills/milestone/templates/stats-file.md.tmpl +0 -31
- package/plugins/cursor-pbr/skills/note/SKILL.md +0 -214
- package/plugins/cursor-pbr/skills/pause/SKILL.md +0 -248
- package/plugins/cursor-pbr/skills/pause/templates/continue-here.md.tmpl +0 -72
- package/plugins/cursor-pbr/skills/plan/SKILL.md +0 -663
- package/plugins/cursor-pbr/skills/plan/templates/checker-prompt.md.tmpl +0 -22
- package/plugins/cursor-pbr/skills/plan/templates/gap-closure-prompt.md.tmpl +0 -33
- package/plugins/cursor-pbr/skills/plan/templates/planner-prompt.md.tmpl +0 -39
- package/plugins/cursor-pbr/skills/plan/templates/researcher-prompt.md.tmpl +0 -20
- package/plugins/cursor-pbr/skills/plan/templates/revision-prompt.md.tmpl +0 -24
- package/plugins/cursor-pbr/skills/quick/SKILL.md +0 -376
- package/plugins/cursor-pbr/skills/resume/SKILL.md +0 -399
- package/plugins/cursor-pbr/skills/review/SKILL.md +0 -654
- package/plugins/cursor-pbr/skills/review/templates/debugger-prompt.md.tmpl +0 -61
- package/plugins/cursor-pbr/skills/review/templates/gap-planner-prompt.md.tmpl +0 -41
- package/plugins/cursor-pbr/skills/review/templates/verifier-prompt.md.tmpl +0 -116
- package/plugins/cursor-pbr/skills/scan/SKILL.md +0 -300
- package/plugins/cursor-pbr/skills/scan/templates/mapper-prompt.md.tmpl +0 -202
- package/plugins/cursor-pbr/skills/setup/SKILL.md +0 -296
- package/plugins/cursor-pbr/skills/shared/commit-planning-docs.md +0 -36
- package/plugins/cursor-pbr/skills/shared/config-loading.md +0 -103
- package/plugins/cursor-pbr/skills/shared/context-budget.md +0 -41
- package/plugins/cursor-pbr/skills/shared/context-loader-task.md +0 -87
- package/plugins/cursor-pbr/skills/shared/digest-select.md +0 -80
- package/plugins/cursor-pbr/skills/shared/domain-probes.md +0 -126
- package/plugins/cursor-pbr/skills/shared/error-reporting.md +0 -81
- package/plugins/cursor-pbr/skills/shared/gate-prompts.md +0 -389
- package/plugins/cursor-pbr/skills/shared/phase-argument-parsing.md +0 -46
- package/plugins/cursor-pbr/skills/shared/progress-display.md +0 -53
- package/plugins/cursor-pbr/skills/shared/revision-loop.md +0 -82
- package/plugins/cursor-pbr/skills/shared/state-loading.md +0 -63
- package/plugins/cursor-pbr/skills/shared/state-update.md +0 -162
- package/plugins/cursor-pbr/skills/shared/universal-anti-patterns.md +0 -38
- package/plugins/cursor-pbr/skills/status/SKILL.md +0 -362
- package/plugins/cursor-pbr/skills/statusline/SKILL.md +0 -150
- package/plugins/cursor-pbr/skills/todo/SKILL.md +0 -280
- package/plugins/cursor-pbr/templates/CONTEXT.md.tmpl +0 -53
- package/plugins/cursor-pbr/templates/INTEGRATION-REPORT.md.tmpl +0 -152
- package/plugins/cursor-pbr/templates/RESEARCH-SUMMARY.md.tmpl +0 -98
- package/plugins/cursor-pbr/templates/ROADMAP.md.tmpl +0 -41
- package/plugins/cursor-pbr/templates/SUMMARY.md.tmpl +0 -82
- package/plugins/cursor-pbr/templates/VERIFICATION-DETAIL.md.tmpl +0 -117
- package/plugins/cursor-pbr/templates/codebase/ARCHITECTURE.md.tmpl +0 -98
- package/plugins/cursor-pbr/templates/codebase/CONCERNS.md.tmpl +0 -93
- package/plugins/cursor-pbr/templates/codebase/CONVENTIONS.md.tmpl +0 -104
- package/plugins/cursor-pbr/templates/codebase/INTEGRATIONS.md.tmpl +0 -78
- package/plugins/cursor-pbr/templates/codebase/STACK.md.tmpl +0 -78
- package/plugins/cursor-pbr/templates/codebase/STRUCTURE.md.tmpl +0 -80
- package/plugins/cursor-pbr/templates/codebase/TESTING.md.tmpl +0 -107
- package/plugins/cursor-pbr/templates/continue-here.md.tmpl +0 -74
- package/plugins/cursor-pbr/templates/prompt-partials/phase-project-context.md.tmpl +0 -38
- package/plugins/cursor-pbr/templates/research/ARCHITECTURE.md.tmpl +0 -124
- package/plugins/cursor-pbr/templates/research/STACK.md.tmpl +0 -71
- package/plugins/cursor-pbr/templates/research/SUMMARY.md.tmpl +0 -112
- package/plugins/cursor-pbr/templates/research-outputs/phase-research.md.tmpl +0 -81
- package/plugins/cursor-pbr/templates/research-outputs/project-research.md.tmpl +0 -99
- package/plugins/cursor-pbr/templates/research-outputs/synthesis.md.tmpl +0 -36
- package/plugins/pbr/references/agent-interactions.md +0 -134
- package/plugins/pbr/references/pbr-rules.md +0 -194
- package/plugins/pbr/references/pbr-tools-cli.md +0 -285
- package/plugins/pbr/references/planning-config.md +0 -213
- package/plugins/pbr/references/subagent-coordination.md +0 -119
- package/plugins/pbr/references/ui-formatting.md +0 -444
- package/plugins/pbr/scripts/validate-plugin-structure.js +0 -183
- package/plugins/pbr/skills/milestone/templates/audit-report.md.tmpl +0 -48
- package/plugins/pbr/skills/shared/progress-display.md +0 -53
- package/plugins/pbr/skills/shared/state-loading.md +0 -62
- package/plugins/pbr/templates/RESEARCH-SUMMARY.md.tmpl +0 -97
- package/plugins/pbr/templates/research/ARCHITECTURE.md.tmpl +0 -124
- package/plugins/pbr/templates/research/STACK.md.tmpl +0 -71
- package/plugins/pbr/templates/research/SUMMARY.md.tmpl +0 -112
- package/plugins/pbr/templates/research-outputs/phase-research.md.tmpl +0 -81
- package/plugins/pbr/templates/research-outputs/project-research.md.tmpl +0 -99
- package/plugins/pbr/templates/research-outputs/synthesis.md.tmpl +0 -36
- /package/plugins/pbr/references/{agent-anti-patterns.md → archive/agent-anti-patterns.md} +0 -0
|
@@ -0,0 +1,1473 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Verify — Verification suite, consistency, and health validation
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
const fs = require('fs');
|
|
6
|
+
const path = require('path');
|
|
7
|
+
const { safeReadFile, normalizePhaseName, execGit, findPhaseInternal, getMilestoneInfo, output, error } = require('./core');
|
|
8
|
+
const { extractFrontmatter, parseMustHavesBlock } = require('./frontmatter');
|
|
9
|
+
// writeStateMd was never exported from state.cjs — use fs.writeFileSync directly
|
|
10
|
+
|
|
11
|
+
function cmdVerifySummary(cwd, summaryPath, checkFileCount, raw) {
|
|
12
|
+
if (!summaryPath) {
|
|
13
|
+
error('summary-path required');
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
const fullPath = path.join(cwd, summaryPath);
|
|
17
|
+
const checkCount = checkFileCount || 2;
|
|
18
|
+
|
|
19
|
+
// Check 1: Summary exists
|
|
20
|
+
if (!fs.existsSync(fullPath)) {
|
|
21
|
+
const result = {
|
|
22
|
+
passed: false,
|
|
23
|
+
checks: {
|
|
24
|
+
summary_exists: false,
|
|
25
|
+
files_created: { checked: 0, found: 0, missing: [] },
|
|
26
|
+
commits_exist: false,
|
|
27
|
+
self_check: 'not_found',
|
|
28
|
+
},
|
|
29
|
+
errors: ['SUMMARY.md not found'],
|
|
30
|
+
};
|
|
31
|
+
output(result, raw, 'failed');
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
const content = fs.readFileSync(fullPath, 'utf-8');
|
|
36
|
+
const errors = [];
|
|
37
|
+
|
|
38
|
+
// Check 2: Spot-check files mentioned in summary
|
|
39
|
+
const mentionedFiles = new Set();
|
|
40
|
+
const patterns = [
|
|
41
|
+
/`([^`]+\.[a-zA-Z]+)`/g,
|
|
42
|
+
/(?:Created|Modified|Added|Updated|Edited):\s*`?([^\s`]+\.[a-zA-Z]+)`?/gi,
|
|
43
|
+
];
|
|
44
|
+
|
|
45
|
+
for (const pattern of patterns) {
|
|
46
|
+
let m;
|
|
47
|
+
while ((m = pattern.exec(content)) !== null) {
|
|
48
|
+
const filePath = m[1];
|
|
49
|
+
if (filePath && !filePath.startsWith('http') && filePath.includes('/')) {
|
|
50
|
+
mentionedFiles.add(filePath);
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
const filesToCheck = Array.from(mentionedFiles).slice(0, checkCount);
|
|
56
|
+
const missing = [];
|
|
57
|
+
for (const file of filesToCheck) {
|
|
58
|
+
if (!fs.existsSync(path.join(cwd, file))) {
|
|
59
|
+
missing.push(file);
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
// Check 3: Commits exist
|
|
64
|
+
const commitHashPattern = /\b[0-9a-f]{7,40}\b/g;
|
|
65
|
+
const hashes = content.match(commitHashPattern) || [];
|
|
66
|
+
let commitsExist = false;
|
|
67
|
+
if (hashes.length > 0) {
|
|
68
|
+
for (const hash of hashes.slice(0, 3)) {
|
|
69
|
+
const result = execGit(cwd, ['cat-file', '-t', hash]);
|
|
70
|
+
if (result.exitCode === 0 && result.stdout === 'commit') {
|
|
71
|
+
commitsExist = true;
|
|
72
|
+
break;
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
// Check 4: Self-check section
|
|
78
|
+
let selfCheck = 'not_found';
|
|
79
|
+
const selfCheckPattern = /##\s*(?:Self[- ]?Check|Verification|Quality Check)/i;
|
|
80
|
+
if (selfCheckPattern.test(content)) {
|
|
81
|
+
const passPattern = /(?:all\s+)?(?:pass|✓|✅|complete|succeeded)/i;
|
|
82
|
+
const failPattern = /(?:fail|✗|❌|incomplete|blocked)/i;
|
|
83
|
+
const checkSection = content.slice(content.search(selfCheckPattern));
|
|
84
|
+
if (failPattern.test(checkSection)) {
|
|
85
|
+
selfCheck = 'failed';
|
|
86
|
+
} else if (passPattern.test(checkSection)) {
|
|
87
|
+
selfCheck = 'passed';
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
if (missing.length > 0) errors.push('Missing files: ' + missing.join(', '));
|
|
92
|
+
if (!commitsExist && hashes.length > 0) errors.push('Referenced commit hashes not found in git history');
|
|
93
|
+
if (selfCheck === 'failed') errors.push('Self-check section indicates failure');
|
|
94
|
+
|
|
95
|
+
const checks = {
|
|
96
|
+
summary_exists: true,
|
|
97
|
+
files_created: { checked: filesToCheck.length, found: filesToCheck.length - missing.length, missing },
|
|
98
|
+
commits_exist: commitsExist,
|
|
99
|
+
self_check: selfCheck,
|
|
100
|
+
};
|
|
101
|
+
|
|
102
|
+
const passed = missing.length === 0 && selfCheck !== 'failed';
|
|
103
|
+
const result = { passed, checks, errors };
|
|
104
|
+
output(result, raw, passed ? 'passed' : 'failed');
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
function cmdVerifyPlanStructure(cwd, filePath, raw) {
|
|
108
|
+
if (!filePath) { error('file path required'); }
|
|
109
|
+
const fullPath = path.isAbsolute(filePath) ? filePath : path.join(cwd, filePath);
|
|
110
|
+
const content = safeReadFile(fullPath);
|
|
111
|
+
if (!content) { output({ error: 'File not found', path: filePath }, raw); return; }
|
|
112
|
+
|
|
113
|
+
const fm = extractFrontmatter(content);
|
|
114
|
+
const errors = [];
|
|
115
|
+
const warnings = [];
|
|
116
|
+
|
|
117
|
+
// Check required frontmatter fields
|
|
118
|
+
const required = ['phase', 'plan', 'type', 'wave', 'depends_on', 'files_modified', 'autonomous', 'must_haves'];
|
|
119
|
+
for (const field of required) {
|
|
120
|
+
if (fm[field] === undefined) errors.push(`Missing required frontmatter field: ${field}`);
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
// Parse and check task elements
|
|
124
|
+
const taskPattern = /<task[^>]*>([\s\S]*?)<\/task>/g;
|
|
125
|
+
const tasks = [];
|
|
126
|
+
let taskMatch;
|
|
127
|
+
while ((taskMatch = taskPattern.exec(content)) !== null) {
|
|
128
|
+
const taskContent = taskMatch[1];
|
|
129
|
+
const nameMatch = taskContent.match(/<name>([\s\S]*?)<\/name>/);
|
|
130
|
+
const taskName = nameMatch ? nameMatch[1].trim() : 'unnamed';
|
|
131
|
+
const hasReadFirst = /<read_first>/.test(taskContent);
|
|
132
|
+
const hasFiles = /<files>/.test(taskContent);
|
|
133
|
+
const hasAction = /<action>/.test(taskContent);
|
|
134
|
+
const hasAcceptanceCriteria = /<acceptance_criteria>/.test(taskContent);
|
|
135
|
+
const hasVerify = /<verify>/.test(taskContent);
|
|
136
|
+
const hasDone = /<done>/.test(taskContent);
|
|
137
|
+
|
|
138
|
+
if (!nameMatch) errors.push('Task missing <name> element');
|
|
139
|
+
if (!hasAction) errors.push(`Task '${taskName}' missing <action>`);
|
|
140
|
+
if (!hasVerify) errors.push(`Task '${taskName}' missing <verify>`);
|
|
141
|
+
if (!hasDone) errors.push(`Task '${taskName}' missing <done>`);
|
|
142
|
+
if (!hasFiles) warnings.push(`Task '${taskName}' missing <files>`);
|
|
143
|
+
if (!hasReadFirst) warnings.push(`Task '${taskName}' missing <read_first>`);
|
|
144
|
+
if (!hasAcceptanceCriteria) warnings.push(`Task '${taskName}' missing <acceptance_criteria>`);
|
|
145
|
+
|
|
146
|
+
tasks.push({ name: taskName, hasReadFirst, hasFiles, hasAction, hasAcceptanceCriteria, hasVerify, hasDone });
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
if (tasks.length === 0) warnings.push('No <task> elements found');
|
|
150
|
+
|
|
151
|
+
// Wave/depends_on consistency
|
|
152
|
+
if (fm.wave && parseInt(fm.wave) > 1 && (!fm.depends_on || (Array.isArray(fm.depends_on) && fm.depends_on.length === 0))) {
|
|
153
|
+
warnings.push('Wave > 1 but depends_on is empty');
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
// Autonomous/checkpoint consistency
|
|
157
|
+
const hasCheckpoints = /<task\s+type=["']?checkpoint/.test(content);
|
|
158
|
+
if (hasCheckpoints && fm.autonomous !== 'false' && fm.autonomous !== false) {
|
|
159
|
+
errors.push('Has checkpoint tasks but autonomous is not false');
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
output({
|
|
163
|
+
valid: errors.length === 0,
|
|
164
|
+
errors,
|
|
165
|
+
warnings,
|
|
166
|
+
task_count: tasks.length,
|
|
167
|
+
tasks,
|
|
168
|
+
frontmatter_fields: Object.keys(fm),
|
|
169
|
+
}, raw, errors.length === 0 ? 'valid' : 'invalid');
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
function cmdVerifyPhaseCompleteness(cwd, phase, raw) {
|
|
173
|
+
if (!phase) { error('phase required'); }
|
|
174
|
+
const phaseInfo = findPhaseInternal(cwd, phase);
|
|
175
|
+
if (!phaseInfo || !phaseInfo.found) {
|
|
176
|
+
output({ error: 'Phase not found', phase }, raw);
|
|
177
|
+
return;
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
const errors = [];
|
|
181
|
+
const warnings = [];
|
|
182
|
+
const phaseDir = path.join(cwd, phaseInfo.directory);
|
|
183
|
+
|
|
184
|
+
// List plans and summaries
|
|
185
|
+
let files;
|
|
186
|
+
try { files = fs.readdirSync(phaseDir); } catch { output({ error: 'Cannot read phase directory' }, raw); return; }
|
|
187
|
+
|
|
188
|
+
const plans = files.filter(f => f.match(/-PLAN\.md$/i));
|
|
189
|
+
const summaries = files.filter(f => f.match(/-SUMMARY\.md$/i));
|
|
190
|
+
|
|
191
|
+
// Extract plan IDs (everything before -PLAN.md)
|
|
192
|
+
const planIds = new Set(plans.map(p => p.replace(/-PLAN\.md$/i, '')));
|
|
193
|
+
const summaryIds = new Set(summaries.map(s => s.replace(/-SUMMARY\.md$/i, '')));
|
|
194
|
+
|
|
195
|
+
// Plans without summaries
|
|
196
|
+
const incompletePlans = [...planIds].filter(id => !summaryIds.has(id));
|
|
197
|
+
if (incompletePlans.length > 0) {
|
|
198
|
+
errors.push(`Plans without summaries: ${incompletePlans.join(', ')}`);
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
// Summaries without plans (orphans)
|
|
202
|
+
const orphanSummaries = [...summaryIds].filter(id => !planIds.has(id));
|
|
203
|
+
if (orphanSummaries.length > 0) {
|
|
204
|
+
warnings.push(`Summaries without plans: ${orphanSummaries.join(', ')}`);
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
output({
|
|
208
|
+
complete: errors.length === 0,
|
|
209
|
+
phase: phaseInfo.phase_number,
|
|
210
|
+
plan_count: plans.length,
|
|
211
|
+
summary_count: summaries.length,
|
|
212
|
+
incomplete_plans: incompletePlans,
|
|
213
|
+
orphan_summaries: orphanSummaries,
|
|
214
|
+
errors,
|
|
215
|
+
warnings,
|
|
216
|
+
}, raw, errors.length === 0 ? 'complete' : 'incomplete');
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
function cmdVerifyReferences(cwd, filePath, raw) {
|
|
220
|
+
if (!filePath) { error('file path required'); }
|
|
221
|
+
const fullPath = path.isAbsolute(filePath) ? filePath : path.join(cwd, filePath);
|
|
222
|
+
const content = safeReadFile(fullPath);
|
|
223
|
+
if (!content) { output({ error: 'File not found', path: filePath }, raw); return; }
|
|
224
|
+
|
|
225
|
+
const found = [];
|
|
226
|
+
const missing = [];
|
|
227
|
+
|
|
228
|
+
// Find @-references: @path/to/file (must contain / to be a file path)
|
|
229
|
+
const atRefs = content.match(/@([^\s\n,)]+\/[^\s\n,)]+)/g) || [];
|
|
230
|
+
for (const ref of atRefs) {
|
|
231
|
+
const cleanRef = ref.slice(1); // remove @
|
|
232
|
+
const resolved = cleanRef.startsWith('~/')
|
|
233
|
+
? path.join(process.env.HOME || '', cleanRef.slice(2))
|
|
234
|
+
: path.join(cwd, cleanRef);
|
|
235
|
+
if (fs.existsSync(resolved)) {
|
|
236
|
+
found.push(cleanRef);
|
|
237
|
+
} else {
|
|
238
|
+
missing.push(cleanRef);
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
// Find backtick file paths that look like real paths (contain / and have extension)
|
|
243
|
+
const backtickRefs = content.match(/`([^`]+\/[^`]+\.[a-zA-Z]{1,10})`/g) || [];
|
|
244
|
+
for (const ref of backtickRefs) {
|
|
245
|
+
const cleanRef = ref.slice(1, -1); // remove backticks
|
|
246
|
+
if (cleanRef.startsWith('http') || cleanRef.includes('${') || cleanRef.includes('{{')) continue;
|
|
247
|
+
if (found.includes(cleanRef) || missing.includes(cleanRef)) continue; // dedup
|
|
248
|
+
const resolved = path.join(cwd, cleanRef);
|
|
249
|
+
if (fs.existsSync(resolved)) {
|
|
250
|
+
found.push(cleanRef);
|
|
251
|
+
} else {
|
|
252
|
+
missing.push(cleanRef);
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
output({
|
|
257
|
+
valid: missing.length === 0,
|
|
258
|
+
found: found.length,
|
|
259
|
+
missing,
|
|
260
|
+
total: found.length + missing.length,
|
|
261
|
+
}, raw, missing.length === 0 ? 'valid' : 'invalid');
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
function cmdVerifyCommits(cwd, hashes, raw) {
|
|
265
|
+
if (!hashes || hashes.length === 0) { error('At least one commit hash required'); }
|
|
266
|
+
|
|
267
|
+
const valid = [];
|
|
268
|
+
const invalid = [];
|
|
269
|
+
for (const hash of hashes) {
|
|
270
|
+
const result = execGit(cwd, ['cat-file', '-t', hash]);
|
|
271
|
+
if (result.exitCode === 0 && result.stdout.trim() === 'commit') {
|
|
272
|
+
valid.push(hash);
|
|
273
|
+
} else {
|
|
274
|
+
invalid.push(hash);
|
|
275
|
+
}
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
output({
|
|
279
|
+
all_valid: invalid.length === 0,
|
|
280
|
+
valid,
|
|
281
|
+
invalid,
|
|
282
|
+
total: hashes.length,
|
|
283
|
+
}, raw, invalid.length === 0 ? 'valid' : 'invalid');
|
|
284
|
+
}
|
|
285
|
+
|
|
286
|
+
/**
|
|
287
|
+
* Parse a string-format artifact into { path, min_lines } or null if descriptive text.
|
|
288
|
+
* Handles: "path/to/file.ext: >N lines", "path/to/file.ext", "descriptive text"
|
|
289
|
+
*/
|
|
290
|
+
function parseStringArtifact(str) {
|
|
291
|
+
const pathLineMatch = str.match(/^([^\s:]+\.\w+)(?::\s*>(\d+)\s*lines?)?/);
|
|
292
|
+
if (pathLineMatch) {
|
|
293
|
+
return { path: pathLineMatch[1], min_lines: pathLineMatch[2] ? parseInt(pathLineMatch[2]) : null };
|
|
294
|
+
}
|
|
295
|
+
return null; // Descriptive text, skip
|
|
296
|
+
}
|
|
297
|
+
|
|
298
|
+
function cmdVerifyArtifacts(cwd, planFilePath, raw) {
|
|
299
|
+
if (!planFilePath) { error('plan file path required'); }
|
|
300
|
+
const fullPath = path.isAbsolute(planFilePath) ? planFilePath : path.join(cwd, planFilePath);
|
|
301
|
+
const content = safeReadFile(fullPath);
|
|
302
|
+
if (!content) { output({ error: 'File not found', path: planFilePath }, raw); return; }
|
|
303
|
+
|
|
304
|
+
const artifacts = parseMustHavesBlock(content, 'artifacts');
|
|
305
|
+
if (artifacts.length === 0) {
|
|
306
|
+
output({ error: 'No must_haves.artifacts found in frontmatter', path: planFilePath }, raw);
|
|
307
|
+
return;
|
|
308
|
+
}
|
|
309
|
+
|
|
310
|
+
const results = [];
|
|
311
|
+
for (const artifact of artifacts) {
|
|
312
|
+
let artObj = artifact;
|
|
313
|
+
if (typeof artifact === 'string') {
|
|
314
|
+
artObj = parseStringArtifact(artifact);
|
|
315
|
+
if (!artObj) continue; // genuinely descriptive, skip
|
|
316
|
+
}
|
|
317
|
+
const artPath = artObj.path;
|
|
318
|
+
if (!artPath) continue;
|
|
319
|
+
|
|
320
|
+
const artFullPath = path.join(cwd, artPath);
|
|
321
|
+
const exists = fs.existsSync(artFullPath);
|
|
322
|
+
const check = { path: artPath, exists, issues: [], passed: false };
|
|
323
|
+
|
|
324
|
+
if (exists) {
|
|
325
|
+
const fileContent = safeReadFile(artFullPath) || '';
|
|
326
|
+
const lineCount = fileContent.split('\n').length;
|
|
327
|
+
|
|
328
|
+
if (artObj.min_lines && lineCount < artObj.min_lines) {
|
|
329
|
+
check.issues.push(`Only ${lineCount} lines, need ${artObj.min_lines}`);
|
|
330
|
+
}
|
|
331
|
+
if (artObj.contains && !fileContent.includes(artObj.contains)) {
|
|
332
|
+
check.issues.push(`Missing pattern: ${artObj.contains}`);
|
|
333
|
+
}
|
|
334
|
+
if (artObj.exports) {
|
|
335
|
+
const exports = Array.isArray(artObj.exports) ? artObj.exports : [artObj.exports];
|
|
336
|
+
for (const exp of exports) {
|
|
337
|
+
if (!fileContent.includes(exp)) check.issues.push(`Missing export: ${exp}`);
|
|
338
|
+
}
|
|
339
|
+
}
|
|
340
|
+
check.passed = check.issues.length === 0;
|
|
341
|
+
} else {
|
|
342
|
+
check.issues.push('File not found');
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
results.push(check);
|
|
346
|
+
}
|
|
347
|
+
|
|
348
|
+
const passed = results.filter(r => r.passed).length;
|
|
349
|
+
output({
|
|
350
|
+
all_passed: passed === results.length,
|
|
351
|
+
passed,
|
|
352
|
+
total: results.length,
|
|
353
|
+
artifacts: results,
|
|
354
|
+
}, raw, passed === results.length ? 'valid' : 'invalid');
|
|
355
|
+
}
|
|
356
|
+
|
|
357
|
+
function cmdVerifyKeyLinks(cwd, planFilePath, raw) {
|
|
358
|
+
if (!planFilePath) { error('plan file path required'); }
|
|
359
|
+
const fullPath = path.isAbsolute(planFilePath) ? planFilePath : path.join(cwd, planFilePath);
|
|
360
|
+
const content = safeReadFile(fullPath);
|
|
361
|
+
if (!content) { output({ error: 'File not found', path: planFilePath }, raw); return; }
|
|
362
|
+
|
|
363
|
+
const keyLinks = parseMustHavesBlock(content, 'key_links');
|
|
364
|
+
if (keyLinks.length === 0) {
|
|
365
|
+
output({ error: 'No must_haves.key_links found in frontmatter', path: planFilePath }, raw);
|
|
366
|
+
return;
|
|
367
|
+
}
|
|
368
|
+
|
|
369
|
+
const results = [];
|
|
370
|
+
for (const link of keyLinks) {
|
|
371
|
+
if (typeof link === 'string') {
|
|
372
|
+
// String-format key_link: attempt to extract file paths, otherwise treat as descriptive
|
|
373
|
+
const pathMatch = link.match(/([^\s]+\.\w+)/g);
|
|
374
|
+
if (pathMatch && pathMatch.length >= 1) {
|
|
375
|
+
// Try to verify the first file path mentioned exists
|
|
376
|
+
const firstPath = pathMatch[0];
|
|
377
|
+
const exists = fs.existsSync(path.join(cwd, firstPath));
|
|
378
|
+
results.push({
|
|
379
|
+
description: link,
|
|
380
|
+
verified: exists ? 'partial' : false,
|
|
381
|
+
detail: exists ? 'Referenced file exists — manual wiring check recommended' : `Referenced file not found: ${firstPath}`,
|
|
382
|
+
});
|
|
383
|
+
} else {
|
|
384
|
+
results.push({
|
|
385
|
+
description: link,
|
|
386
|
+
verified: 'manual',
|
|
387
|
+
detail: 'Descriptive key_link — requires manual verification',
|
|
388
|
+
});
|
|
389
|
+
}
|
|
390
|
+
continue;
|
|
391
|
+
}
|
|
392
|
+
const check = { from: link.from, to: link.to, via: link.via || '', verified: false, detail: '' };
|
|
393
|
+
|
|
394
|
+
const sourceContent = safeReadFile(path.join(cwd, link.from || ''));
|
|
395
|
+
if (!sourceContent) {
|
|
396
|
+
check.detail = 'Source file not found';
|
|
397
|
+
} else if (link.pattern) {
|
|
398
|
+
try {
|
|
399
|
+
const regex = new RegExp(link.pattern);
|
|
400
|
+
if (regex.test(sourceContent)) {
|
|
401
|
+
check.verified = true;
|
|
402
|
+
check.detail = 'Pattern found in source';
|
|
403
|
+
} else {
|
|
404
|
+
const targetContent = safeReadFile(path.join(cwd, link.to || ''));
|
|
405
|
+
if (targetContent && regex.test(targetContent)) {
|
|
406
|
+
check.verified = true;
|
|
407
|
+
check.detail = 'Pattern found in target';
|
|
408
|
+
} else {
|
|
409
|
+
check.detail = `Pattern "${link.pattern}" not found in source or target`;
|
|
410
|
+
}
|
|
411
|
+
}
|
|
412
|
+
} catch {
|
|
413
|
+
check.detail = `Invalid regex pattern: ${link.pattern}`;
|
|
414
|
+
}
|
|
415
|
+
} else {
|
|
416
|
+
// No pattern: just check source references target
|
|
417
|
+
if (sourceContent.includes(link.to || '')) {
|
|
418
|
+
check.verified = true;
|
|
419
|
+
check.detail = 'Target referenced in source';
|
|
420
|
+
} else {
|
|
421
|
+
check.detail = 'Target not referenced in source';
|
|
422
|
+
}
|
|
423
|
+
}
|
|
424
|
+
|
|
425
|
+
results.push(check);
|
|
426
|
+
}
|
|
427
|
+
|
|
428
|
+
const verified = results.filter(r => r.verified).length;
|
|
429
|
+
output({
|
|
430
|
+
all_verified: verified === results.length,
|
|
431
|
+
verified,
|
|
432
|
+
total: results.length,
|
|
433
|
+
links: results,
|
|
434
|
+
}, raw, verified === results.length ? 'valid' : 'invalid');
|
|
435
|
+
}
|
|
436
|
+
|
|
437
|
+
function cmdValidateConsistency(cwd, raw) {
|
|
438
|
+
const roadmapPath = path.join(cwd, '.planning', 'ROADMAP.md');
|
|
439
|
+
const phasesDir = path.join(cwd, '.planning', 'phases');
|
|
440
|
+
const errors = [];
|
|
441
|
+
const warnings = [];
|
|
442
|
+
|
|
443
|
+
// Check for ROADMAP
|
|
444
|
+
if (!fs.existsSync(roadmapPath)) {
|
|
445
|
+
errors.push('ROADMAP.md not found');
|
|
446
|
+
output({ passed: false, errors, warnings }, raw, 'failed');
|
|
447
|
+
return;
|
|
448
|
+
}
|
|
449
|
+
|
|
450
|
+
const roadmapContent = fs.readFileSync(roadmapPath, 'utf-8');
|
|
451
|
+
|
|
452
|
+
// Extract phases from ROADMAP
|
|
453
|
+
const roadmapPhases = new Set();
|
|
454
|
+
const phasePattern = /#{2,4}\s*Phase\s+(\d+[A-Z]?(?:\.\d+)*)\s*:/gi;
|
|
455
|
+
let m;
|
|
456
|
+
while ((m = phasePattern.exec(roadmapContent)) !== null) {
|
|
457
|
+
roadmapPhases.add(m[1]);
|
|
458
|
+
}
|
|
459
|
+
|
|
460
|
+
// Get phases on disk
|
|
461
|
+
const diskPhases = new Set();
|
|
462
|
+
try {
|
|
463
|
+
const entries = fs.readdirSync(phasesDir, { withFileTypes: true });
|
|
464
|
+
const dirs = entries.filter(e => e.isDirectory()).map(e => e.name);
|
|
465
|
+
for (const dir of dirs) {
|
|
466
|
+
const dm = dir.match(/^(\d+[A-Z]?(?:\.\d+)*)/i);
|
|
467
|
+
if (dm) diskPhases.add(dm[1]);
|
|
468
|
+
}
|
|
469
|
+
} catch {}
|
|
470
|
+
|
|
471
|
+
// Check: phases in ROADMAP but not on disk
|
|
472
|
+
for (const p of roadmapPhases) {
|
|
473
|
+
if (!diskPhases.has(p) && !diskPhases.has(normalizePhaseName(p))) {
|
|
474
|
+
warnings.push(`Phase ${p} in ROADMAP.md but no directory on disk`);
|
|
475
|
+
}
|
|
476
|
+
}
|
|
477
|
+
|
|
478
|
+
// Check: phases on disk but not in ROADMAP
|
|
479
|
+
for (const p of diskPhases) {
|
|
480
|
+
const unpadded = String(parseInt(p, 10));
|
|
481
|
+
if (!roadmapPhases.has(p) && !roadmapPhases.has(unpadded)) {
|
|
482
|
+
warnings.push(`Phase ${p} exists on disk but not in ROADMAP.md`);
|
|
483
|
+
}
|
|
484
|
+
}
|
|
485
|
+
|
|
486
|
+
// Check: sequential phase numbers (integers only)
|
|
487
|
+
const integerPhases = [...diskPhases]
|
|
488
|
+
.filter(p => !p.includes('.'))
|
|
489
|
+
.map(p => parseInt(p, 10))
|
|
490
|
+
.sort((a, b) => a - b);
|
|
491
|
+
|
|
492
|
+
for (let i = 1; i < integerPhases.length; i++) {
|
|
493
|
+
if (integerPhases[i] !== integerPhases[i - 1] + 1) {
|
|
494
|
+
warnings.push(`Gap in phase numbering: ${integerPhases[i - 1]} → ${integerPhases[i]}`);
|
|
495
|
+
}
|
|
496
|
+
}
|
|
497
|
+
|
|
498
|
+
// Check: plan numbering within phases
|
|
499
|
+
try {
|
|
500
|
+
const entries = fs.readdirSync(phasesDir, { withFileTypes: true });
|
|
501
|
+
const dirs = entries.filter(e => e.isDirectory()).map(e => e.name).sort();
|
|
502
|
+
|
|
503
|
+
for (const dir of dirs) {
|
|
504
|
+
const phaseFiles = fs.readdirSync(path.join(phasesDir, dir));
|
|
505
|
+
const plans = phaseFiles.filter(f => f.endsWith('-PLAN.md')).sort();
|
|
506
|
+
|
|
507
|
+
// Extract plan numbers
|
|
508
|
+
const planNums = plans.map(p => {
|
|
509
|
+
const pm = p.match(/-(\d{2})-PLAN\.md$/);
|
|
510
|
+
return pm ? parseInt(pm[1], 10) : null;
|
|
511
|
+
}).filter(n => n !== null);
|
|
512
|
+
|
|
513
|
+
for (let i = 1; i < planNums.length; i++) {
|
|
514
|
+
if (planNums[i] !== planNums[i - 1] + 1) {
|
|
515
|
+
warnings.push(`Gap in plan numbering in ${dir}: plan ${planNums[i - 1]} → ${planNums[i]}`);
|
|
516
|
+
}
|
|
517
|
+
}
|
|
518
|
+
|
|
519
|
+
// Check: plans without summaries (completed plans)
|
|
520
|
+
const summaries = phaseFiles.filter(f => f.endsWith('-SUMMARY.md'));
|
|
521
|
+
const planIds = new Set(plans.map(p => p.replace('-PLAN.md', '')));
|
|
522
|
+
const summaryIds = new Set(summaries.map(s => s.replace('-SUMMARY.md', '')));
|
|
523
|
+
|
|
524
|
+
// Summary without matching plan is suspicious
|
|
525
|
+
for (const sid of summaryIds) {
|
|
526
|
+
if (!planIds.has(sid)) {
|
|
527
|
+
warnings.push(`Summary ${sid}-SUMMARY.md in ${dir} has no matching PLAN.md`);
|
|
528
|
+
}
|
|
529
|
+
}
|
|
530
|
+
}
|
|
531
|
+
} catch {}
|
|
532
|
+
|
|
533
|
+
// Check: frontmatter in plans has required fields
|
|
534
|
+
try {
|
|
535
|
+
const entries = fs.readdirSync(phasesDir, { withFileTypes: true });
|
|
536
|
+
const dirs = entries.filter(e => e.isDirectory()).map(e => e.name);
|
|
537
|
+
|
|
538
|
+
for (const dir of dirs) {
|
|
539
|
+
const phaseFiles = fs.readdirSync(path.join(phasesDir, dir));
|
|
540
|
+
const plans = phaseFiles.filter(f => f.endsWith('-PLAN.md'));
|
|
541
|
+
|
|
542
|
+
for (const plan of plans) {
|
|
543
|
+
const content = fs.readFileSync(path.join(phasesDir, dir, plan), 'utf-8');
|
|
544
|
+
const fm = extractFrontmatter(content);
|
|
545
|
+
|
|
546
|
+
if (!fm.wave) {
|
|
547
|
+
warnings.push(`${dir}/${plan}: missing 'wave' in frontmatter`);
|
|
548
|
+
}
|
|
549
|
+
}
|
|
550
|
+
}
|
|
551
|
+
} catch {}
|
|
552
|
+
|
|
553
|
+
const passed = errors.length === 0;
|
|
554
|
+
output({ passed, errors, warnings, warning_count: warnings.length }, raw, passed ? 'passed' : 'failed');
|
|
555
|
+
}
|
|
556
|
+
|
|
557
|
+
/**
|
|
558
|
+
* Check Phase 05 features: decision_journal, negative_knowledge, living_requirements.
|
|
559
|
+
* @param {string} planningDir - Path to .planning directory
|
|
560
|
+
* @param {object} config - Parsed config.json
|
|
561
|
+
* @returns {object} Per-feature status object
|
|
562
|
+
*/
|
|
563
|
+
function checkPhase05Features(planningDir, config) {
|
|
564
|
+
const features = config.features || {};
|
|
565
|
+
const result = {};
|
|
566
|
+
|
|
567
|
+
// decision_journal
|
|
568
|
+
if (features.decision_journal === false) {
|
|
569
|
+
result.decision_journal = { enabled: false, status: 'disabled' };
|
|
570
|
+
} else if (features.decision_journal) {
|
|
571
|
+
const decisionsDir = path.join(planningDir, 'decisions');
|
|
572
|
+
if (fs.existsSync(decisionsDir)) {
|
|
573
|
+
result.decision_journal = { enabled: true, status: 'healthy' };
|
|
574
|
+
} else {
|
|
575
|
+
result.decision_journal = { enabled: true, status: 'degraded', reason: 'decisions directory not found' };
|
|
576
|
+
}
|
|
577
|
+
}
|
|
578
|
+
|
|
579
|
+
// negative_knowledge
|
|
580
|
+
if (features.negative_knowledge === false) {
|
|
581
|
+
result.negative_knowledge = { enabled: false, status: 'disabled' };
|
|
582
|
+
} else if (features.negative_knowledge) {
|
|
583
|
+
const nkDir = path.join(planningDir, 'negative-knowledge');
|
|
584
|
+
if (fs.existsSync(nkDir)) {
|
|
585
|
+
result.negative_knowledge = { enabled: true, status: 'healthy' };
|
|
586
|
+
} else {
|
|
587
|
+
result.negative_knowledge = { enabled: true, status: 'degraded', reason: 'negative-knowledge directory not found' };
|
|
588
|
+
}
|
|
589
|
+
}
|
|
590
|
+
|
|
591
|
+
// living_requirements
|
|
592
|
+
if (features.living_requirements === false) {
|
|
593
|
+
result.living_requirements = { enabled: false, status: 'disabled' };
|
|
594
|
+
} else if (features.living_requirements) {
|
|
595
|
+
const reqPath = path.join(planningDir, 'REQUIREMENTS.md');
|
|
596
|
+
if (fs.existsSync(reqPath)) {
|
|
597
|
+
const content = fs.readFileSync(reqPath, 'utf-8');
|
|
598
|
+
if (/REQ-/.test(content)) {
|
|
599
|
+
result.living_requirements = { enabled: true, status: 'healthy' };
|
|
600
|
+
} else {
|
|
601
|
+
result.living_requirements = { enabled: true, status: 'degraded', reason: 'REQUIREMENTS.md not found or has no REQ-IDs' };
|
|
602
|
+
}
|
|
603
|
+
} else {
|
|
604
|
+
result.living_requirements = { enabled: true, status: 'degraded', reason: 'REQUIREMENTS.md not found or has no REQ-IDs' };
|
|
605
|
+
}
|
|
606
|
+
}
|
|
607
|
+
|
|
608
|
+
return Object.keys(result).length > 0 ? result : null;
|
|
609
|
+
}
|
|
610
|
+
|
|
611
|
+
function cmdValidateHealth(cwd, options, raw) {
|
|
612
|
+
const planningDir = path.join(cwd, '.planning');
|
|
613
|
+
const projectPath = path.join(planningDir, 'PROJECT.md');
|
|
614
|
+
const roadmapPath = path.join(planningDir, 'ROADMAP.md');
|
|
615
|
+
const statePath = path.join(planningDir, 'STATE.md');
|
|
616
|
+
const configPath = path.join(planningDir, 'config.json');
|
|
617
|
+
const phasesDir = path.join(planningDir, 'phases');
|
|
618
|
+
|
|
619
|
+
const errors = [];
|
|
620
|
+
const warnings = [];
|
|
621
|
+
const info = [];
|
|
622
|
+
const repairs = [];
|
|
623
|
+
|
|
624
|
+
// Helper to add issue
|
|
625
|
+
const addIssue = (severity, code, message, fix, repairable = false) => {
|
|
626
|
+
const issue = { code, message, fix, repairable };
|
|
627
|
+
if (severity === 'error') errors.push(issue);
|
|
628
|
+
else if (severity === 'warning') warnings.push(issue);
|
|
629
|
+
else info.push(issue);
|
|
630
|
+
};
|
|
631
|
+
|
|
632
|
+
// ─── Check 1: .planning/ exists ───────────────────────────────────────────
|
|
633
|
+
if (!fs.existsSync(planningDir)) {
|
|
634
|
+
addIssue('error', 'E001', '.planning/ directory not found', 'Run /pbr:new-project to initialize');
|
|
635
|
+
output({
|
|
636
|
+
status: 'broken',
|
|
637
|
+
errors,
|
|
638
|
+
warnings,
|
|
639
|
+
info,
|
|
640
|
+
repairable_count: 0,
|
|
641
|
+
}, raw);
|
|
642
|
+
return;
|
|
643
|
+
}
|
|
644
|
+
|
|
645
|
+
// ─── Check 2: PROJECT.md exists and has required sections ─────────────────
|
|
646
|
+
if (!fs.existsSync(projectPath)) {
|
|
647
|
+
addIssue('error', 'E002', 'PROJECT.md not found', 'Run /pbr:new-project to create');
|
|
648
|
+
} else {
|
|
649
|
+
const content = fs.readFileSync(projectPath, 'utf-8');
|
|
650
|
+
const requiredSections = ['## What This Is', '## Core Value', '## Requirements'];
|
|
651
|
+
for (const section of requiredSections) {
|
|
652
|
+
if (!content.includes(section)) {
|
|
653
|
+
addIssue('warning', 'W001', `PROJECT.md missing section: ${section}`, 'Add section manually');
|
|
654
|
+
}
|
|
655
|
+
}
|
|
656
|
+
}
|
|
657
|
+
|
|
658
|
+
// ─── Check 3: ROADMAP.md exists ───────────────────────────────────────────
|
|
659
|
+
if (!fs.existsSync(roadmapPath)) {
|
|
660
|
+
addIssue('error', 'E003', 'ROADMAP.md not found', 'Run /pbr:new-milestone to create roadmap');
|
|
661
|
+
}
|
|
662
|
+
|
|
663
|
+
// ─── Check 4: STATE.md exists and references valid phases ─────────────────
|
|
664
|
+
if (!fs.existsSync(statePath)) {
|
|
665
|
+
addIssue('error', 'E004', 'STATE.md not found', 'Run /pbr:health --repair to regenerate', true);
|
|
666
|
+
repairs.push('regenerateState');
|
|
667
|
+
} else {
|
|
668
|
+
const stateContent = fs.readFileSync(statePath, 'utf-8');
|
|
669
|
+
// Extract phase references from STATE.md
|
|
670
|
+
const phaseRefs = [...stateContent.matchAll(/[Pp]hase\s+(\d+(?:\.\d+)*)/g)].map(m => m[1]);
|
|
671
|
+
// Get disk phases
|
|
672
|
+
const diskPhases = new Set();
|
|
673
|
+
try {
|
|
674
|
+
const entries = fs.readdirSync(phasesDir, { withFileTypes: true });
|
|
675
|
+
for (const e of entries) {
|
|
676
|
+
if (e.isDirectory()) {
|
|
677
|
+
const m = e.name.match(/^(\d+(?:\.\d+)*)/);
|
|
678
|
+
if (m) diskPhases.add(m[1]);
|
|
679
|
+
}
|
|
680
|
+
}
|
|
681
|
+
} catch {}
|
|
682
|
+
// Check for invalid references
|
|
683
|
+
for (const ref of phaseRefs) {
|
|
684
|
+
const normalizedRef = String(parseInt(ref, 10)).padStart(2, '0');
|
|
685
|
+
if (!diskPhases.has(ref) && !diskPhases.has(normalizedRef) && !diskPhases.has(String(parseInt(ref, 10)))) {
|
|
686
|
+
// Only warn if phases dir has any content (not just an empty project)
|
|
687
|
+
if (diskPhases.size > 0) {
|
|
688
|
+
addIssue('warning', 'W002', `STATE.md references phase ${ref}, but only phases ${[...diskPhases].sort().join(', ')} exist`, 'Run /pbr:health --repair to regenerate STATE.md', true);
|
|
689
|
+
if (!repairs.includes('regenerateState')) repairs.push('regenerateState');
|
|
690
|
+
}
|
|
691
|
+
}
|
|
692
|
+
}
|
|
693
|
+
}
|
|
694
|
+
|
|
695
|
+
// ─── Check 5: config.json valid JSON + valid schema ───────────────────────
|
|
696
|
+
if (!fs.existsSync(configPath)) {
|
|
697
|
+
addIssue('warning', 'W003', 'config.json not found', 'Run /pbr:health --repair to create with defaults', true);
|
|
698
|
+
repairs.push('createConfig');
|
|
699
|
+
} else {
|
|
700
|
+
try {
|
|
701
|
+
const raw = fs.readFileSync(configPath, 'utf-8');
|
|
702
|
+
const parsed = JSON.parse(raw);
|
|
703
|
+
// Validate known fields
|
|
704
|
+
const validProfiles = ['quality', 'balanced', 'budget'];
|
|
705
|
+
if (parsed.model_profile && !validProfiles.includes(parsed.model_profile)) {
|
|
706
|
+
addIssue('warning', 'W004', `config.json: invalid model_profile "${parsed.model_profile}"`, `Valid values: ${validProfiles.join(', ')}`);
|
|
707
|
+
}
|
|
708
|
+
} catch (err) {
|
|
709
|
+
addIssue('error', 'E005', `config.json: JSON parse error - ${err.message}`, 'Run /pbr:health --repair to reset to defaults', true);
|
|
710
|
+
repairs.push('resetConfig');
|
|
711
|
+
}
|
|
712
|
+
}
|
|
713
|
+
|
|
714
|
+
// ─── Check 5b: Nyquist validation key presence ──────────────────────────
|
|
715
|
+
if (fs.existsSync(configPath)) {
|
|
716
|
+
try {
|
|
717
|
+
const configRaw = fs.readFileSync(configPath, 'utf-8');
|
|
718
|
+
const configParsed = JSON.parse(configRaw);
|
|
719
|
+
if (configParsed.workflow && configParsed.workflow.nyquist_validation === undefined) {
|
|
720
|
+
addIssue('warning', 'W008', 'config.json: workflow.nyquist_validation absent (defaults to enabled but agents may skip)', 'Run /pbr:health --repair to add key', true);
|
|
721
|
+
if (!repairs.includes('addNyquistKey')) repairs.push('addNyquistKey');
|
|
722
|
+
}
|
|
723
|
+
} catch {}
|
|
724
|
+
}
|
|
725
|
+
|
|
726
|
+
// ─── Check 6: Phase directory naming (NN-name format) ─────────────────────
|
|
727
|
+
try {
|
|
728
|
+
const entries = fs.readdirSync(phasesDir, { withFileTypes: true });
|
|
729
|
+
for (const e of entries) {
|
|
730
|
+
if (e.isDirectory() && !e.name.match(/^\d{2}(?:\.\d+)*-[\w-]+$/)) {
|
|
731
|
+
addIssue('warning', 'W005', `Phase directory "${e.name}" doesn't follow NN-name format`, 'Rename to match pattern (e.g., 01-setup)');
|
|
732
|
+
}
|
|
733
|
+
}
|
|
734
|
+
} catch {}
|
|
735
|
+
|
|
736
|
+
// ─── Check 7: Orphaned plans (PLAN without SUMMARY) ───────────────────────
|
|
737
|
+
try {
|
|
738
|
+
const entries = fs.readdirSync(phasesDir, { withFileTypes: true });
|
|
739
|
+
for (const e of entries) {
|
|
740
|
+
if (!e.isDirectory()) continue;
|
|
741
|
+
const phaseFiles = fs.readdirSync(path.join(phasesDir, e.name));
|
|
742
|
+
const plans = phaseFiles.filter(f => f.endsWith('-PLAN.md') || f === 'PLAN.md');
|
|
743
|
+
const summaries = phaseFiles.filter(f => f.endsWith('-SUMMARY.md') || f === 'SUMMARY.md');
|
|
744
|
+
const summaryBases = new Set(summaries.map(s => s.replace('-SUMMARY.md', '').replace('SUMMARY.md', '')));
|
|
745
|
+
|
|
746
|
+
for (const plan of plans) {
|
|
747
|
+
const planBase = plan.replace('-PLAN.md', '').replace('PLAN.md', '');
|
|
748
|
+
if (!summaryBases.has(planBase)) {
|
|
749
|
+
addIssue('info', 'I001', `${e.name}/${plan} has no SUMMARY.md`, 'May be in progress');
|
|
750
|
+
}
|
|
751
|
+
}
|
|
752
|
+
}
|
|
753
|
+
} catch {}
|
|
754
|
+
|
|
755
|
+
// ─── Check 7b: Nyquist VALIDATION.md consistency ────────────────────────
|
|
756
|
+
try {
|
|
757
|
+
const phaseEntries = fs.readdirSync(phasesDir, { withFileTypes: true });
|
|
758
|
+
for (const e of phaseEntries) {
|
|
759
|
+
if (!e.isDirectory()) continue;
|
|
760
|
+
const phaseFiles = fs.readdirSync(path.join(phasesDir, e.name));
|
|
761
|
+
const hasResearch = phaseFiles.some(f => f.endsWith('-RESEARCH.md'));
|
|
762
|
+
const hasValidation = phaseFiles.some(f => f.endsWith('-VALIDATION.md'));
|
|
763
|
+
if (hasResearch && !hasValidation) {
|
|
764
|
+
const researchFile = phaseFiles.find(f => f.endsWith('-RESEARCH.md'));
|
|
765
|
+
const researchContent = fs.readFileSync(path.join(phasesDir, e.name, researchFile), 'utf-8');
|
|
766
|
+
if (researchContent.includes('## Validation Architecture')) {
|
|
767
|
+
addIssue('warning', 'W009', `Phase ${e.name}: has Validation Architecture in RESEARCH.md but no VALIDATION.md`, 'Re-run /pbr:plan-phase with --research to regenerate');
|
|
768
|
+
}
|
|
769
|
+
}
|
|
770
|
+
}
|
|
771
|
+
} catch {}
|
|
772
|
+
|
|
773
|
+
// ─── Check 8: Run existing consistency checks ─────────────────────────────
|
|
774
|
+
// Inline subset of cmdValidateConsistency
|
|
775
|
+
if (fs.existsSync(roadmapPath)) {
|
|
776
|
+
const roadmapContent = fs.readFileSync(roadmapPath, 'utf-8');
|
|
777
|
+
const roadmapPhases = new Set();
|
|
778
|
+
const phasePattern = /#{2,4}\s*Phase\s+(\d+[A-Z]?(?:\.\d+)*)\s*:/gi;
|
|
779
|
+
let m;
|
|
780
|
+
while ((m = phasePattern.exec(roadmapContent)) !== null) {
|
|
781
|
+
roadmapPhases.add(m[1]);
|
|
782
|
+
}
|
|
783
|
+
|
|
784
|
+
const diskPhases = new Set();
|
|
785
|
+
try {
|
|
786
|
+
const entries = fs.readdirSync(phasesDir, { withFileTypes: true });
|
|
787
|
+
for (const e of entries) {
|
|
788
|
+
if (e.isDirectory()) {
|
|
789
|
+
const dm = e.name.match(/^(\d+[A-Z]?(?:\.\d+)*)/i);
|
|
790
|
+
if (dm) diskPhases.add(dm[1]);
|
|
791
|
+
}
|
|
792
|
+
}
|
|
793
|
+
} catch {}
|
|
794
|
+
|
|
795
|
+
// Phases in ROADMAP but not on disk
|
|
796
|
+
for (const p of roadmapPhases) {
|
|
797
|
+
const padded = String(parseInt(p, 10)).padStart(2, '0');
|
|
798
|
+
if (!diskPhases.has(p) && !diskPhases.has(padded)) {
|
|
799
|
+
addIssue('warning', 'W006', `Phase ${p} in ROADMAP.md but no directory on disk`, 'Create phase directory or remove from roadmap');
|
|
800
|
+
}
|
|
801
|
+
}
|
|
802
|
+
|
|
803
|
+
// Phases on disk but not in ROADMAP
|
|
804
|
+
for (const p of diskPhases) {
|
|
805
|
+
const unpadded = String(parseInt(p, 10));
|
|
806
|
+
if (!roadmapPhases.has(p) && !roadmapPhases.has(unpadded)) {
|
|
807
|
+
addIssue('warning', 'W007', `Phase ${p} exists on disk but not in ROADMAP.md`, 'Add to roadmap or remove directory');
|
|
808
|
+
}
|
|
809
|
+
}
|
|
810
|
+
}
|
|
811
|
+
|
|
812
|
+
// ─── Check 9: Phase 1 feature status ────────────────────────────────────────
|
|
813
|
+
const feature_status = {};
|
|
814
|
+
if (fs.existsSync(configPath)) {
|
|
815
|
+
try {
|
|
816
|
+
const configRaw = fs.readFileSync(configPath, 'utf-8');
|
|
817
|
+
const configParsed = JSON.parse(configRaw);
|
|
818
|
+
const features = configParsed.features || {};
|
|
819
|
+
const workflow = configParsed.workflow || {};
|
|
820
|
+
|
|
821
|
+
// enhanced_session_start: default true (enabled unless explicitly false)
|
|
822
|
+
const essEnabled = features.enhanced_session_start !== false;
|
|
823
|
+
feature_status.enhanced_session_start = { enabled: essEnabled, status: essEnabled ? 'healthy' : 'disabled' };
|
|
824
|
+
|
|
825
|
+
// context_quality_scoring: default true (enabled unless explicitly false)
|
|
826
|
+
const cqsEnabled = features.context_quality_scoring !== false;
|
|
827
|
+
feature_status.context_quality_scoring = { enabled: cqsEnabled, status: cqsEnabled ? 'healthy' : 'disabled' };
|
|
828
|
+
|
|
829
|
+
// skip_rag: default false (enabled only if explicitly true)
|
|
830
|
+
const srEnabled = features.skip_rag === true;
|
|
831
|
+
feature_status.skip_rag = { enabled: srEnabled, status: srEnabled ? 'healthy' : 'disabled' };
|
|
832
|
+
|
|
833
|
+
// ─── Phase 8 feature checks ──────────────────────────────────────────
|
|
834
|
+
|
|
835
|
+
// graduated_verification: default true; degraded if enabled but no trust data
|
|
836
|
+
const gvEnabled = features.graduated_verification !== false;
|
|
837
|
+
if (!gvEnabled) {
|
|
838
|
+
feature_status.graduated_verification = { enabled: false, status: 'disabled' };
|
|
839
|
+
} else {
|
|
840
|
+
const trustScoresPath = path.join(planningDir, 'trust', 'scores.json');
|
|
841
|
+
const hasTrustData = fs.existsSync(trustScoresPath);
|
|
842
|
+
feature_status.graduated_verification = {
|
|
843
|
+
enabled: true,
|
|
844
|
+
status: hasTrustData ? 'healthy' : 'degraded'
|
|
845
|
+
};
|
|
846
|
+
}
|
|
847
|
+
|
|
848
|
+
// self_verification: default true; healthy if enabled, disabled otherwise
|
|
849
|
+
const svEnabled = features.self_verification !== false;
|
|
850
|
+
feature_status.self_verification = {
|
|
851
|
+
enabled: svEnabled,
|
|
852
|
+
status: svEnabled ? 'healthy' : 'disabled'
|
|
853
|
+
};
|
|
854
|
+
|
|
855
|
+
// autonomy: check autonomy.level config property
|
|
856
|
+
const autonomyConfig = configParsed.autonomy || {};
|
|
857
|
+
const autonomyLevel = autonomyConfig.level || 'supervised';
|
|
858
|
+
const autonomyExplicit = !!(configParsed.autonomy && configParsed.autonomy.level);
|
|
859
|
+
feature_status.autonomy = {
|
|
860
|
+
enabled: true,
|
|
861
|
+
status: autonomyExplicit ? 'healthy' : 'degraded',
|
|
862
|
+
level: autonomyLevel
|
|
863
|
+
};
|
|
864
|
+
|
|
865
|
+
// Validate orchestrator_budget_pct range (15-50)
|
|
866
|
+
const budgetPct = configParsed.orchestrator_budget_pct;
|
|
867
|
+
if (budgetPct !== undefined) {
|
|
868
|
+
if (budgetPct < 15 || budgetPct > 50) {
|
|
869
|
+
addIssue('warning', 'W010', `orchestrator_budget_pct is ${budgetPct}, outside valid range 15-50`, 'Set to a value between 15 and 50 in config.json');
|
|
870
|
+
}
|
|
871
|
+
}
|
|
872
|
+
|
|
873
|
+
// ─── Check 10: Phase 2 feature status ──────────────────────────────────────
|
|
874
|
+
// inline_simple_tasks: default true, degraded if enabled but inline_max_files/inline_max_lines missing
|
|
875
|
+
const istEnabled = features.inline_simple_tasks !== false;
|
|
876
|
+
if (istEnabled) {
|
|
877
|
+
const hasMaxFiles = workflow.inline_max_files !== undefined &&
|
|
878
|
+
typeof workflow.inline_max_files === 'number' &&
|
|
879
|
+
workflow.inline_max_files >= 1 && workflow.inline_max_files <= 20;
|
|
880
|
+
const hasMaxLines = workflow.inline_max_lines !== undefined &&
|
|
881
|
+
typeof workflow.inline_max_lines === 'number' &&
|
|
882
|
+
workflow.inline_max_lines >= 10 && workflow.inline_max_lines <= 500;
|
|
883
|
+
if (!hasMaxFiles || !hasMaxLines) {
|
|
884
|
+
feature_status.inline_simple_tasks = { enabled: true, status: 'degraded' };
|
|
885
|
+
addIssue('warning', 'W012', 'inline_simple_tasks enabled but inline_max_files/inline_max_lines not configured or invalid', 'Add workflow.inline_max_files (1-20) and workflow.inline_max_lines (10-500) to config.json');
|
|
886
|
+
} else {
|
|
887
|
+
feature_status.inline_simple_tasks = { enabled: true, status: 'enabled' };
|
|
888
|
+
}
|
|
889
|
+
} else {
|
|
890
|
+
feature_status.inline_simple_tasks = { enabled: false, status: 'disabled' };
|
|
891
|
+
}
|
|
892
|
+
|
|
893
|
+
// rich_agent_prompts: default true, no extra validation needed
|
|
894
|
+
const rapEnabled = features.rich_agent_prompts !== false;
|
|
895
|
+
feature_status.rich_agent_prompts = { enabled: rapEnabled, status: rapEnabled ? 'enabled' : 'disabled' };
|
|
896
|
+
|
|
897
|
+
// multi_phase_awareness: default true, degraded if enabled but max_phases_in_context missing
|
|
898
|
+
const mpaEnabled = features.multi_phase_awareness !== false;
|
|
899
|
+
if (mpaEnabled) {
|
|
900
|
+
const hasMaxPhases = workflow.max_phases_in_context !== undefined &&
|
|
901
|
+
typeof workflow.max_phases_in_context === 'number' &&
|
|
902
|
+
workflow.max_phases_in_context >= 1 && workflow.max_phases_in_context <= 10;
|
|
903
|
+
if (!hasMaxPhases) {
|
|
904
|
+
feature_status.multi_phase_awareness = { enabled: true, status: 'degraded' };
|
|
905
|
+
addIssue('warning', 'W013', 'multi_phase_awareness enabled but max_phases_in_context not configured or invalid', 'Add workflow.max_phases_in_context (1-10) to config.json');
|
|
906
|
+
} else {
|
|
907
|
+
feature_status.multi_phase_awareness = { enabled: true, status: 'enabled' };
|
|
908
|
+
}
|
|
909
|
+
} else {
|
|
910
|
+
feature_status.multi_phase_awareness = { enabled: false, status: 'disabled' };
|
|
911
|
+
}
|
|
912
|
+
} catch (_e) { /* config parse errors handled in Check 5 */ }
|
|
913
|
+
}
|
|
914
|
+
|
|
915
|
+
// ─── Check 11: Phase 05 feature status ────────────────────────────────────
|
|
916
|
+
{
|
|
917
|
+
let p05Config = {};
|
|
918
|
+
try { p05Config = JSON.parse(fs.readFileSync(configPath, 'utf-8')); } catch (_) {}
|
|
919
|
+
var phase05_features = checkPhase05Features(planningDir, p05Config);
|
|
920
|
+
}
|
|
921
|
+
|
|
922
|
+
// ─── Check 12: Trust tracking health ──────────────────────────────────────
|
|
923
|
+
{
|
|
924
|
+
let config = {};
|
|
925
|
+
try { config = JSON.parse(fs.readFileSync(configPath, 'utf-8')); } catch (_) {}
|
|
926
|
+
|
|
927
|
+
if (config.features && config.features.trust_tracking === false) {
|
|
928
|
+
addIssue('info', 'I-TRUST-DISABLED', 'trust_tracking is disabled in config', 'Enable features.trust_tracking in config.json if desired');
|
|
929
|
+
} else {
|
|
930
|
+
const trustFile = path.join(planningDir, 'trust', 'agent-scores.json');
|
|
931
|
+
if (!fs.existsSync(trustFile)) {
|
|
932
|
+
addIssue('info', 'I-TRUST-DEGRADED', 'trust_tracking: degraded — agent-scores.json missing. Will populate after first verification.', 'Run a build cycle to generate trust data');
|
|
933
|
+
} else {
|
|
934
|
+
try {
|
|
935
|
+
const scores = JSON.parse(fs.readFileSync(trustFile, 'utf-8'));
|
|
936
|
+
const agents = Object.keys(scores);
|
|
937
|
+
let totalOutcomes = 0;
|
|
938
|
+
for (const agent of agents) {
|
|
939
|
+
for (const cat of Object.values(scores[agent])) {
|
|
940
|
+
totalOutcomes += (cat.pass || 0) + (cat.fail || 0);
|
|
941
|
+
}
|
|
942
|
+
}
|
|
943
|
+
addIssue('info', 'I-TRUST-HEALTHY', `trust_tracking: healthy — ${agents.length} agents, ${totalOutcomes} outcomes tracked`, '');
|
|
944
|
+
} catch (_) {
|
|
945
|
+
addIssue('info', 'I-TRUST-DEGRADED', 'trust_tracking: degraded — agent-scores.json exists but is malformed', 'Delete .planning/trust/agent-scores.json to reset');
|
|
946
|
+
}
|
|
947
|
+
}
|
|
948
|
+
}
|
|
949
|
+
|
|
950
|
+
if (config.features && config.features.confidence_calibration === false) {
|
|
951
|
+
addIssue('info', 'I-CONFIDENCE-DISABLED', 'confidence_calibration is disabled in config', 'Enable features.confidence_calibration in config.json if desired');
|
|
952
|
+
} else {
|
|
953
|
+
const trustFile = path.join(planningDir, 'trust', 'agent-scores.json');
|
|
954
|
+
if (!fs.existsSync(trustFile)) {
|
|
955
|
+
addIssue('info', 'I-CONFIDENCE-DEGRADED', 'confidence_calibration: degraded — no trust data available', 'Run a build cycle to generate trust data');
|
|
956
|
+
} else {
|
|
957
|
+
try {
|
|
958
|
+
JSON.parse(fs.readFileSync(trustFile, 'utf-8'));
|
|
959
|
+
addIssue('info', 'I-CONFIDENCE-HEALTHY', 'confidence_calibration: healthy — trust data available for calibration', '');
|
|
960
|
+
} catch (_) {
|
|
961
|
+
addIssue('info', 'I-CONFIDENCE-DEGRADED', 'confidence_calibration: degraded — trust data malformed', 'Delete .planning/trust/agent-scores.json to reset');
|
|
962
|
+
}
|
|
963
|
+
}
|
|
964
|
+
}
|
|
965
|
+
}
|
|
966
|
+
|
|
967
|
+
// ─── Check 13: Architecture graph feature health ──────────────────────────
|
|
968
|
+
try {
|
|
969
|
+
const graph = require('./graph');
|
|
970
|
+
const graphHealth = graph.graphHealthCheck(planningDir);
|
|
971
|
+
const guardHealth = graph.guardHealthCheck(planningDir);
|
|
972
|
+
feature_status.architecture_graph = graphHealth;
|
|
973
|
+
feature_status.architecture_guard = guardHealth;
|
|
974
|
+
} catch (_e) { /* graph module not available — skip */ }
|
|
975
|
+
|
|
976
|
+
// ─── Check 14: Phase 15 DX feature health ────────────────────────────────
|
|
977
|
+
{
|
|
978
|
+
let p15Config = {};
|
|
979
|
+
try { p15Config = JSON.parse(fs.readFileSync(configPath, 'utf-8')); } catch (_) {}
|
|
980
|
+
const p15Features = (p15Config && p15Config.features) || {};
|
|
981
|
+
|
|
982
|
+
const checkDxFeature = (featureName, modulePath, exportName) => {
|
|
983
|
+
const enabled = p15Features[featureName] !== false; // default true
|
|
984
|
+
if (!enabled) {
|
|
985
|
+
return { enabled: false, status: 'disabled', detail: 'Feature disabled in config' };
|
|
986
|
+
}
|
|
987
|
+
try {
|
|
988
|
+
const mod = require(modulePath);
|
|
989
|
+
if (typeof mod[exportName] === 'function') {
|
|
990
|
+
// Attempt a lightweight invocation to confirm operational
|
|
991
|
+
const result = mod[exportName](planningDir, p15Config);
|
|
992
|
+
const operational = result && result.enabled !== false;
|
|
993
|
+
return {
|
|
994
|
+
enabled: true,
|
|
995
|
+
status: operational ? 'healthy' : 'degraded',
|
|
996
|
+
detail: operational ? `${exportName} returned data` : 'Module returned disabled stub',
|
|
997
|
+
};
|
|
998
|
+
}
|
|
999
|
+
return { enabled: true, status: 'degraded', detail: `${exportName} not a function` };
|
|
1000
|
+
} catch (err) {
|
|
1001
|
+
return { enabled: true, status: 'degraded', detail: `Error: ${err.message}` };
|
|
1002
|
+
}
|
|
1003
|
+
};
|
|
1004
|
+
|
|
1005
|
+
feature_status.progress_visualization = checkDxFeature(
|
|
1006
|
+
'progress_visualization',
|
|
1007
|
+
path.join(__dirname, 'progress-visualization.cjs'),
|
|
1008
|
+
'getProgressData'
|
|
1009
|
+
);
|
|
1010
|
+
feature_status.contextual_help = checkDxFeature(
|
|
1011
|
+
'contextual_help',
|
|
1012
|
+
path.join(__dirname, 'contextual-help.cjs'),
|
|
1013
|
+
'getContextualHelp'
|
|
1014
|
+
);
|
|
1015
|
+
feature_status.team_onboarding = checkDxFeature(
|
|
1016
|
+
'team_onboarding',
|
|
1017
|
+
path.join(__dirname, 'onboarding-generator.cjs'),
|
|
1018
|
+
'generateOnboardingGuide'
|
|
1019
|
+
);
|
|
1020
|
+
}
|
|
1021
|
+
|
|
1022
|
+
// ─── Check 15: Phase 14 Quality & Safety feature health ───────────────────
|
|
1023
|
+
{
|
|
1024
|
+
let p14Config = {};
|
|
1025
|
+
try { p14Config = JSON.parse(fs.readFileSync(configPath, 'utf-8')); } catch (_) {}
|
|
1026
|
+
|
|
1027
|
+
const p14Features = p14Config.features || {};
|
|
1028
|
+
|
|
1029
|
+
// Helper: check a feature module loads correctly
|
|
1030
|
+
const checkFeatureHealth = (featureName, configEnabled, modulePath, validationFn) => {
|
|
1031
|
+
if (!configEnabled) {
|
|
1032
|
+
addIssue('info', `I-${featureName.toUpperCase()}-DISABLED`, `${featureName}: disabled`, `Enable features.${featureName} in config.json if desired`);
|
|
1033
|
+
return { enabled: false, status: 'disabled' };
|
|
1034
|
+
}
|
|
1035
|
+
try {
|
|
1036
|
+
const mod = require(modulePath);
|
|
1037
|
+
const isValid = validationFn(mod);
|
|
1038
|
+
if (isValid) {
|
|
1039
|
+
return { enabled: true, status: 'healthy' };
|
|
1040
|
+
}
|
|
1041
|
+
addIssue('warning', `W-${featureName.toUpperCase()}-DEGRADED`, `${featureName}: degraded (module validation failed)`, `Check ${modulePath} exports`);
|
|
1042
|
+
return { enabled: true, status: 'degraded' };
|
|
1043
|
+
} catch (_e) {
|
|
1044
|
+
addIssue('warning', `W-${featureName.toUpperCase()}-DEGRADED`, `${featureName}: degraded (module load failed)`, `Ensure ${modulePath} exists and is valid`);
|
|
1045
|
+
return { enabled: true, status: 'degraded' };
|
|
1046
|
+
}
|
|
1047
|
+
};
|
|
1048
|
+
|
|
1049
|
+
// regression_prevention: default true
|
|
1050
|
+
const rpEnabled = p14Features.regression_prevention !== false;
|
|
1051
|
+
const rpModPath = path.join(__dirname, 'test-selection.cjs');
|
|
1052
|
+
const rpHealth = checkFeatureHealth(
|
|
1053
|
+
'regression_prevention',
|
|
1054
|
+
rpEnabled,
|
|
1055
|
+
rpModPath,
|
|
1056
|
+
(mod) => typeof mod.selectTests === 'function'
|
|
1057
|
+
);
|
|
1058
|
+
if (rpHealth.status === 'healthy') {
|
|
1059
|
+
addIssue('info', 'I-RP-HEALTHY', 'regression_prevention: healthy', '');
|
|
1060
|
+
}
|
|
1061
|
+
feature_status.regression_prevention = rpHealth;
|
|
1062
|
+
|
|
1063
|
+
// security_scanning: default true
|
|
1064
|
+
const ssEnabled = p14Features.security_scanning !== false;
|
|
1065
|
+
const ssModPath = path.join(__dirname, 'security-scan.cjs');
|
|
1066
|
+
const ssHealth = checkFeatureHealth(
|
|
1067
|
+
'security_scanning',
|
|
1068
|
+
ssEnabled,
|
|
1069
|
+
ssModPath,
|
|
1070
|
+
(mod) => Array.isArray(mod.SECURITY_RULES) && mod.SECURITY_RULES.length > 0
|
|
1071
|
+
);
|
|
1072
|
+
if (ssHealth.status === 'healthy') {
|
|
1073
|
+
try {
|
|
1074
|
+
const ssMod = require(ssModPath);
|
|
1075
|
+
const ruleCount = ssMod.SECURITY_RULES.length;
|
|
1076
|
+
addIssue('info', 'I-SS-HEALTHY', `security_scanning: healthy (${ruleCount} rules loaded)`, '');
|
|
1077
|
+
} catch (_) {}
|
|
1078
|
+
}
|
|
1079
|
+
feature_status.security_scanning = ssHealth;
|
|
1080
|
+
}
|
|
1081
|
+
|
|
1082
|
+
// ─── Check 16: Phase 11 Spec-Driven Development feature health ───────────
|
|
1083
|
+
{
|
|
1084
|
+
let p11Config = {};
|
|
1085
|
+
try { p11Config = JSON.parse(fs.readFileSync(configPath, 'utf-8')); } catch (_) {}
|
|
1086
|
+
const p11Features = p11Config.features || {};
|
|
1087
|
+
|
|
1088
|
+
const checkSpecFeatureHealth = (featureName, defaultEnabled, modulePath, exportName) => {
|
|
1089
|
+
const isEnabled = p11Features[featureName] === undefined ? defaultEnabled : p11Features[featureName];
|
|
1090
|
+
if (!isEnabled) {
|
|
1091
|
+
return { status: 'disabled', enabled: false, details: 'Feature toggle off' };
|
|
1092
|
+
}
|
|
1093
|
+
try {
|
|
1094
|
+
const mod = require(modulePath);
|
|
1095
|
+
if (typeof mod[exportName] === 'function') {
|
|
1096
|
+
return { status: 'healthy', enabled: true, details: `${exportName} loaded` };
|
|
1097
|
+
}
|
|
1098
|
+
return { status: 'degraded', enabled: true, details: `${exportName} not a function` };
|
|
1099
|
+
} catch (_e) {
|
|
1100
|
+
return { status: 'degraded', enabled: true, details: `Cannot load module: ${_e.message}` };
|
|
1101
|
+
}
|
|
1102
|
+
};
|
|
1103
|
+
|
|
1104
|
+
feature_status.machine_executable_plans = checkSpecFeatureHealth(
|
|
1105
|
+
'machine_executable_plans', false, path.join(__dirname, 'spec-engine.cjs'), 'parsePlanToSpec'
|
|
1106
|
+
);
|
|
1107
|
+
feature_status.spec_diffing = checkSpecFeatureHealth(
|
|
1108
|
+
'spec_diffing', true, path.join(__dirname, 'spec-diff.cjs'), 'diffSpecs'
|
|
1109
|
+
);
|
|
1110
|
+
feature_status.reverse_spec = checkSpecFeatureHealth(
|
|
1111
|
+
'reverse_spec', true, path.join(__dirname, 'reverse-spec.cjs'), 'generateReverseSpec'
|
|
1112
|
+
);
|
|
1113
|
+
feature_status.predictive_impact = checkSpecFeatureHealth(
|
|
1114
|
+
'predictive_impact', true, path.join(__dirname, 'impact-analysis.cjs'), 'analyzeImpact'
|
|
1115
|
+
);
|
|
1116
|
+
}
|
|
1117
|
+
|
|
1118
|
+
// ─── Check 17: Phase 16 cross-project intelligence feature health ────────
|
|
1119
|
+
{
|
|
1120
|
+
let p16Config = {};
|
|
1121
|
+
try { p16Config = JSON.parse(fs.readFileSync(configPath, 'utf-8')); } catch (_) {}
|
|
1122
|
+
const p16Features = p16Config.features || {};
|
|
1123
|
+
|
|
1124
|
+
// Helper: check if a feature is enabled (default true unless explicitly false)
|
|
1125
|
+
const checkFeatureToggle = (key, healthCheck) => {
|
|
1126
|
+
const enabled = p16Features[key] !== false;
|
|
1127
|
+
if (!enabled) {
|
|
1128
|
+
return { enabled: false, status: 'disabled' };
|
|
1129
|
+
}
|
|
1130
|
+
return { enabled: true, status: healthCheck() ? 'healthy' : 'degraded' };
|
|
1131
|
+
};
|
|
1132
|
+
|
|
1133
|
+
// cross_project_patterns: healthy if ~/.claude/patterns/ exists and has .json files
|
|
1134
|
+
feature_status.cross_project_patterns = checkFeatureToggle(
|
|
1135
|
+
'cross_project_patterns',
|
|
1136
|
+
() => {
|
|
1137
|
+
try {
|
|
1138
|
+
const patternsDir = require('path').join(require('os').homedir(), '.claude', 'patterns');
|
|
1139
|
+
return fs.existsSync(patternsDir) &&
|
|
1140
|
+
fs.readdirSync(patternsDir).some(f => f.endsWith('.json'));
|
|
1141
|
+
} catch (_) { return false; }
|
|
1142
|
+
}
|
|
1143
|
+
);
|
|
1144
|
+
|
|
1145
|
+
// spec_templates: always healthy when enabled (built-in templates always available)
|
|
1146
|
+
feature_status.spec_templates = checkFeatureToggle(
|
|
1147
|
+
'spec_templates',
|
|
1148
|
+
() => true
|
|
1149
|
+
);
|
|
1150
|
+
|
|
1151
|
+
// global_learnings: healthy if ~/.claude/learnings.jsonl exists
|
|
1152
|
+
feature_status.global_learnings = checkFeatureToggle(
|
|
1153
|
+
'global_learnings',
|
|
1154
|
+
() => {
|
|
1155
|
+
try {
|
|
1156
|
+
const learningsPath = require('path').join(require('os').homedir(), '.claude', 'learnings.jsonl');
|
|
1157
|
+
return fs.existsSync(learningsPath);
|
|
1158
|
+
} catch (_) { return false; }
|
|
1159
|
+
}
|
|
1160
|
+
);
|
|
1161
|
+
}
|
|
1162
|
+
|
|
1163
|
+
// ─── Check 18: Phase 3 (zero-friction) feature status ────────────────────
|
|
1164
|
+
{
|
|
1165
|
+
let p3Config = {};
|
|
1166
|
+
try { p3Config = JSON.parse(fs.readFileSync(configPath, 'utf-8')); } catch (_) {}
|
|
1167
|
+
const p3Features = p3Config.features || {};
|
|
1168
|
+
|
|
1169
|
+
const zfqEnabled = p3Features.zero_friction_quick !== false;
|
|
1170
|
+
feature_status.zero_friction_quick = {
|
|
1171
|
+
enabled: zfqEnabled,
|
|
1172
|
+
status: zfqEnabled ? 'healthy' : 'disabled',
|
|
1173
|
+
};
|
|
1174
|
+
}
|
|
1175
|
+
|
|
1176
|
+
// ─── Check 19: Phase 4 (NL routing) feature status ───────────────────────
|
|
1177
|
+
{
|
|
1178
|
+
let p4Config = {};
|
|
1179
|
+
try { p4Config = JSON.parse(fs.readFileSync(configPath, 'utf-8')); } catch (_) {}
|
|
1180
|
+
const p4Features = p4Config.features || {};
|
|
1181
|
+
const pluginRoot = path.resolve(__dirname, '..', '..', '..', 'plugins', 'pbr');
|
|
1182
|
+
|
|
1183
|
+
// natural_language_routing: default true; try to load module
|
|
1184
|
+
const nlrEnabled = p4Features.natural_language_routing !== false;
|
|
1185
|
+
if (!nlrEnabled) {
|
|
1186
|
+
feature_status.natural_language_routing = { enabled: false, status: 'disabled' };
|
|
1187
|
+
} else {
|
|
1188
|
+
try {
|
|
1189
|
+
require(path.join(pluginRoot, 'scripts', 'lib', 'alternatives.js'));
|
|
1190
|
+
feature_status.natural_language_routing = { enabled: true, status: 'healthy' };
|
|
1191
|
+
} catch (_e) {
|
|
1192
|
+
feature_status.natural_language_routing = { enabled: true, status: 'degraded' };
|
|
1193
|
+
}
|
|
1194
|
+
}
|
|
1195
|
+
|
|
1196
|
+
// adaptive_ceremony: default true; workflow-only feature, no module
|
|
1197
|
+
const acEnabled = p4Features.adaptive_ceremony !== false;
|
|
1198
|
+
feature_status.adaptive_ceremony = {
|
|
1199
|
+
enabled: acEnabled,
|
|
1200
|
+
status: acEnabled ? 'healthy' : 'disabled',
|
|
1201
|
+
};
|
|
1202
|
+
}
|
|
1203
|
+
|
|
1204
|
+
// ─── Check 20: Phase 6 (convention memory) feature status ────────────────
|
|
1205
|
+
{
|
|
1206
|
+
let p6Config = {};
|
|
1207
|
+
try { p6Config = JSON.parse(fs.readFileSync(configPath, 'utf-8')); } catch (_) {}
|
|
1208
|
+
const p6Features = p6Config.features || {};
|
|
1209
|
+
const pluginRoot = path.resolve(__dirname, '..', '..', '..', 'plugins', 'pbr');
|
|
1210
|
+
|
|
1211
|
+
// convention_memory: default true; try to load module
|
|
1212
|
+
const cmEnabled = p6Features.convention_memory !== false;
|
|
1213
|
+
if (!cmEnabled) {
|
|
1214
|
+
feature_status.convention_memory = { enabled: false, status: 'disabled' };
|
|
1215
|
+
} else {
|
|
1216
|
+
try {
|
|
1217
|
+
require(path.join(pluginRoot, 'scripts', 'lib', 'convention-detector.js'));
|
|
1218
|
+
feature_status.convention_memory = { enabled: true, status: 'healthy' };
|
|
1219
|
+
} catch (_e) {
|
|
1220
|
+
feature_status.convention_memory = { enabled: true, status: 'degraded' };
|
|
1221
|
+
}
|
|
1222
|
+
}
|
|
1223
|
+
|
|
1224
|
+
// mental_model_snapshots: default true; try to load module
|
|
1225
|
+
const mmsEnabled = p6Features.mental_model_snapshots !== false;
|
|
1226
|
+
if (!mmsEnabled) {
|
|
1227
|
+
feature_status.mental_model_snapshots = { enabled: false, status: 'disabled' };
|
|
1228
|
+
} else {
|
|
1229
|
+
try {
|
|
1230
|
+
require(path.join(pluginRoot, 'scripts', 'lib', 'snapshot-manager.js'));
|
|
1231
|
+
feature_status.mental_model_snapshots = { enabled: true, status: 'healthy' };
|
|
1232
|
+
} catch (_e) {
|
|
1233
|
+
feature_status.mental_model_snapshots = { enabled: true, status: 'degraded' };
|
|
1234
|
+
}
|
|
1235
|
+
}
|
|
1236
|
+
}
|
|
1237
|
+
|
|
1238
|
+
// ─── Check 21: Phase 9 (proactive intelligence) feature status ───────────
|
|
1239
|
+
{
|
|
1240
|
+
let p9Config = {};
|
|
1241
|
+
try { p9Config = JSON.parse(fs.readFileSync(configPath, 'utf-8')); } catch (_) {}
|
|
1242
|
+
const scriptsDir = path.resolve(__dirname, '..', '..', '..', 'plugins', 'pbr', 'scripts');
|
|
1243
|
+
const { checkFeatureHealth: checkPhase9FeatureHealth } = require('./health');
|
|
1244
|
+
const phase9Features = [
|
|
1245
|
+
'smart_next_task',
|
|
1246
|
+
'dependency_break_detection',
|
|
1247
|
+
'pre_research',
|
|
1248
|
+
'pattern_routing',
|
|
1249
|
+
'tech_debt_surfacing',
|
|
1250
|
+
];
|
|
1251
|
+
for (const name of phase9Features) {
|
|
1252
|
+
const result = checkPhase9FeatureHealth(name, p9Config, scriptsDir);
|
|
1253
|
+
feature_status[name] = { enabled: result.status !== 'disabled', status: result.status };
|
|
1254
|
+
}
|
|
1255
|
+
}
|
|
1256
|
+
|
|
1257
|
+
// ─── Check 22: Phase 10 (post-hoc) feature status ────────────────────────
|
|
1258
|
+
{
|
|
1259
|
+
const p10ModPath = path.resolve(__dirname, '..', '..', '..', 'plugins', 'pbr', 'scripts', 'lib', 'health-checks.js');
|
|
1260
|
+
try {
|
|
1261
|
+
const p10Checks = require(p10ModPath);
|
|
1262
|
+
const postHocResult = p10Checks.checkPostHocArtifacts(planningDir);
|
|
1263
|
+
feature_status.post_hoc_artifacts = {
|
|
1264
|
+
enabled: postHocResult.enabled,
|
|
1265
|
+
status: postHocResult.status,
|
|
1266
|
+
};
|
|
1267
|
+
const feedbackResult = p10Checks.checkAgentFeedbackLoop(planningDir);
|
|
1268
|
+
feature_status.agent_feedback_loop = {
|
|
1269
|
+
enabled: feedbackResult.enabled,
|
|
1270
|
+
status: feedbackResult.status,
|
|
1271
|
+
};
|
|
1272
|
+
const metricsResult = p10Checks.checkSessionMetrics(planningDir);
|
|
1273
|
+
feature_status.session_metrics = {
|
|
1274
|
+
enabled: metricsResult.enabled,
|
|
1275
|
+
status: metricsResult.status,
|
|
1276
|
+
};
|
|
1277
|
+
} catch (_e) {
|
|
1278
|
+
// health-checks.js not available — mark all as degraded
|
|
1279
|
+
feature_status.post_hoc_artifacts = { enabled: true, status: 'degraded' };
|
|
1280
|
+
feature_status.agent_feedback_loop = { enabled: true, status: 'degraded' };
|
|
1281
|
+
feature_status.session_metrics = { enabled: true, status: 'degraded' };
|
|
1282
|
+
}
|
|
1283
|
+
}
|
|
1284
|
+
|
|
1285
|
+
// ─── Check 23: Phase 13 (multi-agent) feature status ─────────────────────
|
|
1286
|
+
{
|
|
1287
|
+
let p13Config = {};
|
|
1288
|
+
try { p13Config = JSON.parse(fs.readFileSync(configPath, 'utf-8')); } catch (_) {}
|
|
1289
|
+
const { checkMultiAgentHealth } = require('./health');
|
|
1290
|
+
const multiAgentResults = checkMultiAgentHealth(p13Config);
|
|
1291
|
+
for (const result of multiAgentResults) {
|
|
1292
|
+
feature_status[result.feature] = {
|
|
1293
|
+
enabled: result.status !== 'disabled',
|
|
1294
|
+
status: result.status,
|
|
1295
|
+
};
|
|
1296
|
+
}
|
|
1297
|
+
}
|
|
1298
|
+
|
|
1299
|
+
// ─── Perform repairs if requested ─────────────────────────────────────────
|
|
1300
|
+
const repairActions = [];
|
|
1301
|
+
if (options.repair && repairs.length > 0) {
|
|
1302
|
+
for (const repair of repairs) {
|
|
1303
|
+
try {
|
|
1304
|
+
switch (repair) {
|
|
1305
|
+
case 'createConfig':
|
|
1306
|
+
case 'resetConfig': {
|
|
1307
|
+
const defaults = {
|
|
1308
|
+
model_profile: 'balanced',
|
|
1309
|
+
commit_docs: true,
|
|
1310
|
+
search_gitignored: false,
|
|
1311
|
+
branching_strategy: 'none',
|
|
1312
|
+
research: true,
|
|
1313
|
+
plan_checker: true,
|
|
1314
|
+
verifier: true,
|
|
1315
|
+
parallelization: true,
|
|
1316
|
+
};
|
|
1317
|
+
fs.writeFileSync(configPath, JSON.stringify(defaults, null, 2), 'utf-8');
|
|
1318
|
+
repairActions.push({ action: repair, success: true, path: 'config.json' });
|
|
1319
|
+
break;
|
|
1320
|
+
}
|
|
1321
|
+
case 'regenerateState': {
|
|
1322
|
+
// Create timestamped backup before overwriting
|
|
1323
|
+
if (fs.existsSync(statePath)) {
|
|
1324
|
+
const timestamp = new Date().toISOString().replace(/[:.]/g, '-').slice(0, 19);
|
|
1325
|
+
const backupPath = `${statePath}.bak-${timestamp}`;
|
|
1326
|
+
fs.copyFileSync(statePath, backupPath);
|
|
1327
|
+
repairActions.push({ action: 'backupState', success: true, path: backupPath });
|
|
1328
|
+
}
|
|
1329
|
+
// Generate minimal STATE.md from ROADMAP.md structure
|
|
1330
|
+
const milestone = getMilestoneInfo(cwd);
|
|
1331
|
+
let stateContent = `# Session State\n\n`;
|
|
1332
|
+
stateContent += `## Project Reference\n\n`;
|
|
1333
|
+
stateContent += `See: .planning/PROJECT.md\n\n`;
|
|
1334
|
+
stateContent += `## Position\n\n`;
|
|
1335
|
+
stateContent += `**Milestone:** ${milestone.version} ${milestone.name}\n`;
|
|
1336
|
+
stateContent += `**Current phase:** (determining...)\n`;
|
|
1337
|
+
stateContent += `**Status:** Resuming\n\n`;
|
|
1338
|
+
stateContent += `## Session Log\n\n`;
|
|
1339
|
+
stateContent += `- ${new Date().toISOString().split('T')[0]}: STATE.md regenerated by /pbr:health --repair\n`;
|
|
1340
|
+
fs.writeFileSync(statePath, stateContent, 'utf-8');
|
|
1341
|
+
repairActions.push({ action: repair, success: true, path: 'STATE.md' });
|
|
1342
|
+
break;
|
|
1343
|
+
}
|
|
1344
|
+
case 'addNyquistKey': {
|
|
1345
|
+
if (fs.existsSync(configPath)) {
|
|
1346
|
+
try {
|
|
1347
|
+
const configRaw = fs.readFileSync(configPath, 'utf-8');
|
|
1348
|
+
const configParsed = JSON.parse(configRaw);
|
|
1349
|
+
if (!configParsed.workflow) configParsed.workflow = {};
|
|
1350
|
+
if (configParsed.workflow.nyquist_validation === undefined) {
|
|
1351
|
+
configParsed.workflow.nyquist_validation = true;
|
|
1352
|
+
fs.writeFileSync(configPath, JSON.stringify(configParsed, null, 2), 'utf-8');
|
|
1353
|
+
}
|
|
1354
|
+
repairActions.push({ action: repair, success: true, path: 'config.json' });
|
|
1355
|
+
} catch (err) {
|
|
1356
|
+
repairActions.push({ action: repair, success: false, error: err.message });
|
|
1357
|
+
}
|
|
1358
|
+
}
|
|
1359
|
+
break;
|
|
1360
|
+
}
|
|
1361
|
+
}
|
|
1362
|
+
} catch (err) {
|
|
1363
|
+
repairActions.push({ action: repair, success: false, error: err.message });
|
|
1364
|
+
}
|
|
1365
|
+
}
|
|
1366
|
+
}
|
|
1367
|
+
|
|
1368
|
+
// ─── Determine overall status ─────────────────────────────────────────────
|
|
1369
|
+
let status;
|
|
1370
|
+
if (errors.length > 0) {
|
|
1371
|
+
status = 'broken';
|
|
1372
|
+
} else if (warnings.length > 0) {
|
|
1373
|
+
status = 'degraded';
|
|
1374
|
+
} else {
|
|
1375
|
+
status = 'healthy';
|
|
1376
|
+
}
|
|
1377
|
+
|
|
1378
|
+
const repairableCount = errors.filter(e => e.repairable).length +
|
|
1379
|
+
warnings.filter(w => w.repairable).length;
|
|
1380
|
+
|
|
1381
|
+
output({
|
|
1382
|
+
status,
|
|
1383
|
+
errors,
|
|
1384
|
+
warnings,
|
|
1385
|
+
info,
|
|
1386
|
+
repairable_count: repairableCount,
|
|
1387
|
+
repairs_performed: repairActions.length > 0 ? repairActions : undefined,
|
|
1388
|
+
feature_status: Object.keys(feature_status).length > 0 ? feature_status : undefined,
|
|
1389
|
+
phase05_features: phase05_features || undefined,
|
|
1390
|
+
}, raw);
|
|
1391
|
+
}
|
|
1392
|
+
|
|
1393
|
+
/**
|
|
1394
|
+
* Generic health check for a feature backed by a loadable module.
|
|
1395
|
+
* @param {string} featureName - Feature name (e.g. 'natural_language_routing')
|
|
1396
|
+
* @param {string} planningDir - Path to .planning directory
|
|
1397
|
+
* @param {string} pluginRoot - Path to plugin root (plugins/pbr)
|
|
1398
|
+
* @param {string} togglePath - Dot path in config.features (same as featureName)
|
|
1399
|
+
* @param {string} modulePath - Relative path under pluginRoot to the module
|
|
1400
|
+
* @param {string} exportName - Name of the expected export function
|
|
1401
|
+
* @returns {{ feature: string, status: string, details: string }}
|
|
1402
|
+
*/
|
|
1403
|
+
function checkFeatureModuleHealth(featureName, planningDir, pluginRoot, modulePath, exportName) {
|
|
1404
|
+
const configPath = path.join(planningDir, 'config.json');
|
|
1405
|
+
let config = {};
|
|
1406
|
+
try {
|
|
1407
|
+
if (fs.existsSync(configPath)) {
|
|
1408
|
+
config = JSON.parse(fs.readFileSync(configPath, 'utf-8'));
|
|
1409
|
+
}
|
|
1410
|
+
} catch (_e) {
|
|
1411
|
+
// Config unreadable — treat as defaults
|
|
1412
|
+
}
|
|
1413
|
+
|
|
1414
|
+
const features = config.features || {};
|
|
1415
|
+
|
|
1416
|
+
// Check toggle (default true)
|
|
1417
|
+
if (features[featureName] === false) {
|
|
1418
|
+
return { feature: featureName, status: 'disabled', details: 'Feature toggle off' };
|
|
1419
|
+
}
|
|
1420
|
+
|
|
1421
|
+
// Try to load the module
|
|
1422
|
+
const fullModulePath = path.join(pluginRoot, modulePath);
|
|
1423
|
+
try {
|
|
1424
|
+
const mod = require(fullModulePath);
|
|
1425
|
+
if (typeof mod[exportName] === 'function') {
|
|
1426
|
+
return { feature: featureName, status: 'healthy', details: `${exportName} loaded from ${modulePath}` };
|
|
1427
|
+
}
|
|
1428
|
+
return { feature: featureName, status: 'degraded', details: `${exportName} not a function in ${modulePath}` };
|
|
1429
|
+
} catch (err) {
|
|
1430
|
+
return { feature: featureName, status: 'degraded', details: `Cannot load ${modulePath}: ${err.message}` };
|
|
1431
|
+
}
|
|
1432
|
+
}
|
|
1433
|
+
|
|
1434
|
+
/**
|
|
1435
|
+
* Health check for natural_language_routing feature.
|
|
1436
|
+
* @param {string} planningDir - Path to .planning directory
|
|
1437
|
+
* @param {string} pluginRoot - Path to plugin root (plugins/pbr)
|
|
1438
|
+
* @returns {{ feature: string, status: string, details: string }}
|
|
1439
|
+
*/
|
|
1440
|
+
function checkNLRoutingHealth(planningDir, pluginRoot) {
|
|
1441
|
+
return checkFeatureModuleHealth(
|
|
1442
|
+
'natural_language_routing', planningDir, pluginRoot,
|
|
1443
|
+
path.join('scripts', 'intent-router.cjs'), 'classifyIntent'
|
|
1444
|
+
);
|
|
1445
|
+
}
|
|
1446
|
+
|
|
1447
|
+
/**
|
|
1448
|
+
* Health check for adaptive_ceremony feature.
|
|
1449
|
+
* @param {string} planningDir - Path to .planning directory
|
|
1450
|
+
* @param {string} pluginRoot - Path to plugin root (plugins/pbr)
|
|
1451
|
+
* @returns {{ feature: string, status: string, details: string }}
|
|
1452
|
+
*/
|
|
1453
|
+
function checkAdaptiveCeremonyHealth(planningDir, pluginRoot) {
|
|
1454
|
+
return checkFeatureModuleHealth(
|
|
1455
|
+
'adaptive_ceremony', planningDir, pluginRoot,
|
|
1456
|
+
path.join('scripts', 'risk-classifier.cjs'), 'classifyRisk'
|
|
1457
|
+
);
|
|
1458
|
+
}
|
|
1459
|
+
|
|
1460
|
+
module.exports = {
|
|
1461
|
+
cmdVerifySummary,
|
|
1462
|
+
cmdVerifyPlanStructure,
|
|
1463
|
+
cmdVerifyPhaseCompleteness,
|
|
1464
|
+
cmdVerifyReferences,
|
|
1465
|
+
cmdVerifyCommits,
|
|
1466
|
+
cmdVerifyArtifacts,
|
|
1467
|
+
cmdVerifyKeyLinks,
|
|
1468
|
+
cmdValidateConsistency,
|
|
1469
|
+
cmdValidateHealth,
|
|
1470
|
+
checkNLRoutingHealth,
|
|
1471
|
+
checkAdaptiveCeremonyHealth,
|
|
1472
|
+
checkFeatureModuleHealth,
|
|
1473
|
+
};
|