@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.
Files changed (977) hide show
  1. package/CHANGELOG.md +1265 -296
  2. package/CLAUDE.md +56 -34
  3. package/LICENSE +2 -1
  4. package/README.md +412 -177
  5. package/bin/install.js +2752 -0
  6. package/dashboard/bin/cli.cjs +96 -0
  7. package/dashboard/bin/stop.cjs +129 -0
  8. package/dashboard/eslint.config.js +37 -0
  9. package/dashboard/index.html +20 -0
  10. package/dashboard/package.json +27 -25
  11. package/dashboard/server/index.js +142 -0
  12. package/dashboard/server/lib/frontmatter.js +92 -0
  13. package/dashboard/server/middleware/static.js +35 -0
  14. package/dashboard/server/package.json +16 -0
  15. package/dashboard/server/routes/agents.js +213 -0
  16. package/dashboard/server/routes/config.js +64 -0
  17. package/dashboard/server/routes/health.js +95 -0
  18. package/dashboard/server/routes/incidents.js +78 -0
  19. package/dashboard/server/routes/intel.js +69 -0
  20. package/dashboard/server/routes/memory.js +107 -0
  21. package/dashboard/server/routes/planning.js +234 -0
  22. package/dashboard/server/routes/progress.js +77 -0
  23. package/dashboard/server/routes/projects.js +36 -0
  24. package/dashboard/server/routes/requirements.js +40 -0
  25. package/dashboard/server/routes/roadmap.js +69 -0
  26. package/dashboard/server/routes/sessions.js +70 -0
  27. package/dashboard/server/routes/status.js +25 -0
  28. package/dashboard/server/routes/telemetry.js +214 -0
  29. package/dashboard/server/services/file-watcher.js +105 -0
  30. package/dashboard/server/services/planning-reader.js +741 -0
  31. package/dashboard/server/test/cli.test.js +34 -0
  32. package/dashboard/server/test/frontmatter.test.js +104 -0
  33. package/dashboard/server/test/isolation.test.js +32 -0
  34. package/dashboard/server/test/planning-reader.test.js +151 -0
  35. package/dashboard/server/test/routes.test.js +91 -0
  36. package/dashboard/server/test/ws.test.js +81 -0
  37. package/dashboard/server/ws.js +96 -0
  38. package/dashboard/src/App.jsx +165 -0
  39. package/dashboard/src/components/charts/BudgetBars.jsx +42 -0
  40. package/dashboard/src/components/charts/ContextRadar.jsx +34 -0
  41. package/dashboard/src/components/charts/PhaseDonut.jsx +66 -0
  42. package/dashboard/src/components/charts/SuccessTrend.jsx +45 -0
  43. package/dashboard/src/components/charts/TokenChart.jsx +55 -0
  44. package/dashboard/src/components/charts/index.js +5 -0
  45. package/dashboard/src/components/config/CfgSection.jsx +93 -0
  46. package/dashboard/src/components/layout/Header.jsx +89 -0
  47. package/dashboard/src/components/layout/ProjectSwitcher.jsx +160 -0
  48. package/dashboard/src/components/layout/Sidebar.jsx +161 -0
  49. package/dashboard/src/components/ui/AutoModeBanner.jsx +138 -0
  50. package/dashboard/src/components/ui/BackButton.jsx +27 -0
  51. package/dashboard/src/components/ui/Badge.jsx +27 -0
  52. package/dashboard/src/components/ui/Card.jsx +23 -0
  53. package/dashboard/src/components/ui/ChartTooltip.jsx +48 -0
  54. package/dashboard/src/components/ui/CheckpointBox.jsx +110 -0
  55. package/dashboard/src/components/ui/CodeBlock.jsx +27 -0
  56. package/dashboard/src/components/ui/ConfidenceBadge.jsx +20 -0
  57. package/dashboard/src/components/ui/ConfirmModal.jsx +161 -0
  58. package/dashboard/src/components/ui/ConnectionBanner.jsx +60 -0
  59. package/dashboard/src/components/ui/ErrorBoundary.jsx +106 -0
  60. package/dashboard/src/components/ui/ErrorBox.jsx +107 -0
  61. package/dashboard/src/components/ui/KeyValue.jsx +33 -0
  62. package/dashboard/src/components/ui/LoadingSkeleton.jsx +84 -0
  63. package/dashboard/src/components/ui/MetricCard.jsx +58 -0
  64. package/dashboard/src/components/ui/NextUpBlock.jsx +92 -0
  65. package/dashboard/src/components/ui/NumberInput.jsx +44 -0
  66. package/dashboard/src/components/ui/PBRBanner.jsx +47 -0
  67. package/dashboard/src/components/ui/PipelineView.jsx +130 -0
  68. package/dashboard/src/components/ui/ProgressBar.jsx +28 -0
  69. package/dashboard/src/components/ui/ProgressDisplay.jsx +47 -0
  70. package/dashboard/src/components/ui/QualityGateBadge.jsx +15 -0
  71. package/dashboard/src/components/ui/SectionTitle.jsx +35 -0
  72. package/dashboard/src/components/ui/SelectInput.jsx +45 -0
  73. package/dashboard/src/components/ui/StatusDot.jsx +51 -0
  74. package/dashboard/src/components/ui/StatusSymbol.jsx +49 -0
  75. package/dashboard/src/components/ui/TabBar.jsx +41 -0
  76. package/dashboard/src/components/ui/TextInput.jsx +42 -0
  77. package/dashboard/src/components/ui/Toast.jsx +117 -0
  78. package/dashboard/src/components/ui/Toggle.jsx +70 -0
  79. package/dashboard/src/components/ui/index.js +29 -0
  80. package/dashboard/src/hooks/useDocumentTitle.js +16 -0
  81. package/dashboard/src/hooks/useFetch.js +50 -0
  82. package/dashboard/src/hooks/useToast.jsx +43 -0
  83. package/dashboard/src/hooks/useWebSocket.js +103 -0
  84. package/dashboard/src/lib/api.js +112 -0
  85. package/dashboard/src/lib/configSchema.js +189 -0
  86. package/dashboard/src/lib/constants.js +22 -0
  87. package/dashboard/src/main.jsx +15 -0
  88. package/dashboard/src/pages/AgentsPage.jsx +191 -0
  89. package/dashboard/src/pages/ConfigPage.jsx +298 -0
  90. package/dashboard/src/pages/HooksPage.jsx +412 -0
  91. package/dashboard/src/pages/IncidentsPage.jsx +135 -0
  92. package/dashboard/src/pages/IntelPage.jsx +193 -0
  93. package/dashboard/src/pages/LiveFeed.jsx +274 -0
  94. package/dashboard/src/pages/MemoryPage.jsx +107 -0
  95. package/dashboard/src/pages/OnboardingPage.jsx +117 -0
  96. package/dashboard/src/pages/Overview.jsx +360 -0
  97. package/dashboard/src/pages/PhaseDetailView.jsx +216 -0
  98. package/dashboard/src/pages/PlanningPage.jsx +181 -0
  99. package/dashboard/src/pages/ProgressPage.jsx +249 -0
  100. package/dashboard/src/pages/ResearchPage.jsx +129 -0
  101. package/dashboard/src/pages/RoadmapPage.jsx +251 -0
  102. package/dashboard/src/pages/SessionsPage.jsx +117 -0
  103. package/dashboard/src/pages/Telemetry.jsx +166 -0
  104. package/dashboard/src/pages/planning/DecisionsTab.jsx +153 -0
  105. package/dashboard/src/pages/planning/FilesTab.jsx +420 -0
  106. package/dashboard/src/pages/planning/MilestoneDetail.jsx +319 -0
  107. package/dashboard/src/pages/planning/MilestonesTab.jsx +151 -0
  108. package/dashboard/src/pages/planning/NotesTab.jsx +251 -0
  109. package/dashboard/src/pages/planning/PhasesTab.jsx +218 -0
  110. package/dashboard/src/pages/planning/QuickTab.jsx +50 -0
  111. package/dashboard/src/pages/planning/ResearchTab.jsx +103 -0
  112. package/dashboard/src/pages/planning/TodosTab.jsx +297 -0
  113. package/dashboard/src/theme/ThemeProvider.jsx +38 -0
  114. package/dashboard/src/theme/tokens.js +17 -0
  115. package/dashboard/tests/components/ConfirmModal.test.jsx +179 -0
  116. package/dashboard/tests/components/ConnectionBanner.test.jsx +37 -0
  117. package/dashboard/tests/components/ErrorBoundary.test.jsx +59 -0
  118. package/dashboard/tests/components/LoadingSkeleton.test.jsx +46 -0
  119. package/dashboard/tests/components/ToastContainer.test.jsx +47 -0
  120. package/dashboard/tests/components/Toggle.test.jsx +61 -0
  121. package/dashboard/tests/hooks/useFetch.test.jsx +77 -0
  122. package/dashboard/tests/hooks/useToast.test.jsx +78 -0
  123. package/dashboard/tests/hooks/useWebSocket.test.jsx +128 -0
  124. package/dashboard/tests/pages/ConfigPage.test.jsx +199 -0
  125. package/dashboard/tests/pages/PlanningPage.test.jsx +119 -0
  126. package/dashboard/tests/pages/planning/FilesTab.test.jsx +198 -0
  127. package/dashboard/tests/pages/planning/NotesTab.test.jsx +178 -0
  128. package/dashboard/tests/pages/planning/TodosTab.test.jsx +188 -0
  129. package/dashboard/tests/performance.test.jsx +46 -0
  130. package/dashboard/tests/routes/config.test.js +98 -0
  131. package/dashboard/tests/routes/health.test.js +40 -0
  132. package/dashboard/tests/routes/planning.test.js +112 -0
  133. package/dashboard/tests/routes/roadmap.test.js +91 -0
  134. package/dashboard/tests/routes/status.test.js +131 -0
  135. package/dashboard/tests/server/planning-reader.test.js +153 -0
  136. package/dashboard/tests/setup.js +7 -0
  137. package/dashboard/vite.config.js +41 -0
  138. package/package.json +55 -40
  139. package/plan-build-run/bin/config-schema.json +1420 -0
  140. package/plugins/pbr/.claude-plugin/plugin.json +1 -1
  141. package/plugins/pbr/CLAUDE.md +19 -0
  142. package/plugins/pbr/UI-CONSISTENCY-GAPS.md +1 -1
  143. package/plugins/pbr/agents/advisor-researcher.md +100 -0
  144. package/plugins/pbr/agents/audit.md +205 -89
  145. package/plugins/pbr/agents/codebase-mapper.md +158 -23
  146. package/plugins/pbr/agents/debugger.md +222 -30
  147. package/plugins/pbr/agents/dev-sync.md +206 -0
  148. package/plugins/pbr/agents/executor.md +717 -39
  149. package/plugins/pbr/agents/general.md +71 -6
  150. package/plugins/pbr/agents/integration-checker.md +146 -30
  151. package/plugins/pbr/agents/intel-updater.md +332 -0
  152. package/plugins/pbr/agents/nyquist-auditor.md +253 -0
  153. package/plugins/pbr/agents/plan-checker.md +270 -61
  154. package/plugins/pbr/agents/planner.md +440 -42
  155. package/plugins/pbr/agents/researcher.md +223 -36
  156. package/plugins/pbr/agents/roadmapper.md +397 -0
  157. package/plugins/pbr/agents/synthesizer.md +170 -26
  158. package/plugins/pbr/agents/ui-checker.md +203 -0
  159. package/plugins/pbr/agents/ui-researcher.md +224 -0
  160. package/plugins/pbr/agents/verifier.md +495 -47
  161. package/plugins/pbr/commands/add-phase.md +75 -0
  162. package/plugins/pbr/commands/add-todo.md +8 -0
  163. package/plugins/pbr/commands/audit-fix.md +5 -0
  164. package/plugins/pbr/commands/audit-milestone.md +8 -0
  165. package/plugins/pbr/commands/autonomous.md +5 -0
  166. package/plugins/pbr/commands/backlog.md +6 -0
  167. package/plugins/pbr/commands/check-todos.md +8 -0
  168. package/plugins/pbr/commands/complete-milestone.md +8 -0
  169. package/plugins/pbr/commands/config.md +1 -1
  170. package/plugins/pbr/commands/discuss-phase.md +6 -0
  171. package/plugins/pbr/commands/do.md +5 -0
  172. package/plugins/pbr/commands/execute-phase.md +6 -0
  173. package/plugins/pbr/commands/fast.md +6 -0
  174. package/plugins/pbr/commands/forensics.md +6 -0
  175. package/plugins/pbr/commands/import.md +1 -1
  176. package/plugins/pbr/commands/insert-phase.md +65 -0
  177. package/plugins/pbr/commands/intel.md +5 -0
  178. package/plugins/pbr/commands/join-discord.md +11 -0
  179. package/plugins/pbr/commands/list-phase-assumptions.md +5 -0
  180. package/plugins/pbr/commands/map-codebase.md +6 -0
  181. package/plugins/pbr/commands/milestone-summary.md +6 -0
  182. package/plugins/pbr/commands/new-milestone.md +8 -0
  183. package/plugins/pbr/commands/new-project.md +6 -0
  184. package/plugins/pbr/commands/pause-work.md +5 -0
  185. package/plugins/pbr/commands/plan-milestone-gaps.md +7 -0
  186. package/plugins/pbr/commands/plan-phase.md +6 -0
  187. package/plugins/pbr/commands/plant-seed.md +6 -0
  188. package/plugins/pbr/commands/profile-user.md +5 -0
  189. package/plugins/pbr/commands/profile.md +5 -0
  190. package/plugins/pbr/commands/progress.md +6 -0
  191. package/plugins/pbr/commands/quick.md +1 -1
  192. package/plugins/pbr/commands/reapply-patches.md +47 -0
  193. package/plugins/pbr/commands/release.md +6 -0
  194. package/plugins/pbr/commands/remove-phase.md +66 -0
  195. package/plugins/pbr/commands/research-phase.md +59 -0
  196. package/plugins/pbr/commands/resume-work.md +5 -0
  197. package/plugins/pbr/commands/seed.md +6 -0
  198. package/plugins/pbr/commands/session-report.md +5 -0
  199. package/plugins/pbr/commands/set-profile.md +6 -0
  200. package/plugins/pbr/commands/settings.md +5 -0
  201. package/plugins/pbr/commands/setup.md +1 -1
  202. package/plugins/pbr/commands/ship.md +5 -0
  203. package/plugins/pbr/commands/stats.md +6 -0
  204. package/plugins/pbr/commands/test.md +5 -0
  205. package/plugins/pbr/commands/thread.md +6 -0
  206. package/plugins/pbr/commands/todo.md +1 -1
  207. package/plugins/pbr/commands/ui-phase.md +5 -0
  208. package/plugins/pbr/commands/ui-review.md +5 -0
  209. package/plugins/pbr/commands/undo.md +5 -0
  210. package/plugins/pbr/commands/update.md +37 -0
  211. package/plugins/pbr/commands/validate-phase.md +5 -0
  212. package/plugins/pbr/commands/verify-work.md +6 -0
  213. package/plugins/pbr/dashboard/package-lock.json +6 -0
  214. package/plugins/pbr/dist/architecture-guard.js +59 -0
  215. package/plugins/pbr/dist/audit-dimensions.js +556 -0
  216. package/plugins/pbr/dist/auto-continue.js +277 -0
  217. package/plugins/pbr/dist/block-skill-self-read.js +124 -0
  218. package/plugins/pbr/dist/check-agent-state-write.js +63 -0
  219. package/plugins/pbr/dist/check-config-change.js +162 -0
  220. package/plugins/pbr/dist/check-cross-plugin-sync.js +93 -0
  221. package/plugins/pbr/dist/check-dangerous-commands.js +193 -0
  222. package/plugins/pbr/dist/check-direct-state-write.js +37 -0
  223. package/plugins/pbr/dist/check-doc-sprawl.js +102 -0
  224. package/plugins/pbr/dist/check-phase-boundary.js +191 -0
  225. package/plugins/pbr/dist/check-plan-format.js +241 -0
  226. package/plugins/pbr/dist/check-read-first.js +345 -0
  227. package/plugins/pbr/dist/check-roadmap-sync.js +503 -0
  228. package/plugins/pbr/dist/check-skill-workflow.js +354 -0
  229. package/plugins/pbr/dist/check-state-sync.js +658 -0
  230. package/plugins/pbr/dist/check-subagent-output.js +452 -0
  231. package/plugins/pbr/dist/check-summary-gate.js +199 -0
  232. package/plugins/pbr/dist/context-bridge.js +425 -0
  233. package/plugins/pbr/dist/context-budget-check.js +442 -0
  234. package/plugins/pbr/dist/context-quality.js +271 -0
  235. package/plugins/pbr/dist/enforce-context-budget.js +138 -0
  236. package/plugins/pbr/dist/enforce-pbr-workflow.js +277 -0
  237. package/plugins/pbr/dist/event-handler.js +202 -0
  238. package/plugins/pbr/dist/event-logger.js +125 -0
  239. package/plugins/pbr/dist/feedback-loop.js +172 -0
  240. package/plugins/pbr/dist/graph-update.js +422 -0
  241. package/plugins/pbr/dist/hook-logger.js +114 -0
  242. package/plugins/pbr/dist/hook-server-client.js +361 -0
  243. package/plugins/pbr/dist/hook-server.js +606 -0
  244. package/plugins/pbr/dist/hooks-schema.json +87 -0
  245. package/plugins/pbr/dist/instructions-loaded.js +173 -0
  246. package/plugins/pbr/dist/intercept-plan-mode.js +81 -0
  247. package/plugins/pbr/dist/log-notification.js +131 -0
  248. package/plugins/pbr/dist/log-subagent.js +349 -0
  249. package/plugins/pbr/dist/log-tool-failure.js +140 -0
  250. package/plugins/pbr/dist/milestone-learnings.js +569 -0
  251. package/plugins/pbr/dist/pbr-tools.js +2044 -0
  252. package/plugins/pbr/dist/post-bash-triage.js +154 -0
  253. package/plugins/pbr/dist/post-compact.js +135 -0
  254. package/plugins/pbr/dist/post-hoc.js +286 -0
  255. package/plugins/pbr/dist/post-write-dispatch.js +279 -0
  256. package/plugins/pbr/dist/post-write-quality.js +208 -0
  257. package/plugins/pbr/dist/pre-bash-dispatch.js +218 -0
  258. package/plugins/pbr/dist/pre-skill-dispatch.js +114 -0
  259. package/plugins/pbr/dist/pre-task-dispatch.js +297 -0
  260. package/plugins/pbr/dist/pre-write-dispatch.js +234 -0
  261. package/plugins/pbr/dist/progress-tracker.js +198 -0
  262. package/plugins/pbr/dist/prompt-guard.js +114 -0
  263. package/plugins/pbr/dist/prompt-routing.js +209 -0
  264. package/plugins/pbr/dist/quick-status.js +179 -0
  265. package/plugins/pbr/dist/record-incident.js +37 -0
  266. package/plugins/pbr/dist/run-hook.js +144 -0
  267. package/plugins/pbr/dist/session-cleanup.js +683 -0
  268. package/plugins/pbr/dist/session-tracker.js +124 -0
  269. package/plugins/pbr/dist/status-line.js +847 -0
  270. package/plugins/pbr/dist/suggest-compact.js +315 -0
  271. package/plugins/pbr/dist/sync-context-to-claude.js +100 -0
  272. package/plugins/pbr/dist/task-completed.js +206 -0
  273. package/plugins/pbr/dist/track-context-budget.js +432 -0
  274. package/plugins/pbr/dist/track-user-gates.js +88 -0
  275. package/plugins/pbr/dist/trust-tracker.js +193 -0
  276. package/plugins/pbr/dist/validate-commit.js +271 -0
  277. package/plugins/pbr/dist/validate-skill-args.js +222 -0
  278. package/plugins/pbr/dist/validate-task.js +301 -0
  279. package/plugins/pbr/dist/worktree-create.js +144 -0
  280. package/plugins/pbr/dist/worktree-remove.js +147 -0
  281. package/plugins/pbr/hooks/hooks.json +143 -60
  282. package/plugins/pbr/references/agent-contracts.md +40 -9
  283. package/plugins/pbr/references/agent-teams.md +3 -3
  284. package/plugins/pbr/references/archive/checkpoints.md +189 -0
  285. package/plugins/pbr/references/archive/context-quality-tiers.md +45 -0
  286. package/plugins/pbr/references/archive/hook-ordering.md +89 -0
  287. package/plugins/pbr/references/archive/limitations.md +106 -0
  288. package/plugins/pbr/references/archive/pbr-rules.md +194 -0
  289. package/plugins/pbr/references/archive/pbr-tools-cli.md +415 -0
  290. package/plugins/pbr/references/archive/pretooluse-jsonl-behavior.md +58 -0
  291. package/plugins/pbr/references/archive/signal-files.md +41 -0
  292. package/plugins/pbr/references/archive/tmux-setup.md +288 -0
  293. package/plugins/pbr/references/archive/verification-matrix.md +34 -0
  294. package/plugins/pbr/references/archive/verification-patterns.md +277 -0
  295. package/plugins/pbr/references/archive/worktree-sparse-checkout.md +86 -0
  296. package/plugins/pbr/references/behavioral-contexts.md +53 -0
  297. package/plugins/pbr/references/checkpoints.md +723 -104
  298. package/plugins/pbr/references/config-reference.md +472 -10
  299. package/plugins/pbr/references/continuation-format.md +1 -0
  300. package/plugins/pbr/references/decimal-phase-calculation.md +65 -0
  301. package/plugins/pbr/references/deviation-rules.md +12 -0
  302. package/plugins/pbr/references/git-integration.md +110 -27
  303. package/plugins/pbr/references/git-planning-commit.md +35 -0
  304. package/plugins/pbr/references/model-profile-resolution.md +34 -0
  305. package/plugins/pbr/references/model-profiles.md +90 -7
  306. package/plugins/pbr/references/model-selection.md +1 -1
  307. package/plugins/pbr/references/node-repair.md +48 -0
  308. package/plugins/pbr/references/plan-authoring.md +65 -0
  309. package/plugins/pbr/references/plan-format.md +161 -10
  310. package/plugins/pbr/references/questioning.md +138 -49
  311. package/plugins/pbr/references/reading-verification.md +4 -4
  312. package/plugins/pbr/references/tdd.md +263 -0
  313. package/plugins/pbr/references/ui-brand.md +449 -0
  314. package/plugins/pbr/references/verification-overrides.md +39 -0
  315. package/plugins/pbr/references/verification-patterns.md +529 -113
  316. package/plugins/pbr/scripts/architecture-guard.js +59 -0
  317. package/plugins/pbr/scripts/audit-checks/behavioral-compliance.js +2098 -0
  318. package/plugins/pbr/scripts/audit-checks/error-analysis.js +989 -0
  319. package/plugins/pbr/scripts/audit-checks/feature-verification.js +723 -0
  320. package/plugins/pbr/scripts/audit-checks/index.js +433 -0
  321. package/plugins/pbr/scripts/audit-checks/infrastructure.js +816 -0
  322. package/plugins/pbr/scripts/audit-checks/quality-metrics.js +452 -0
  323. package/plugins/pbr/scripts/audit-checks/session-quality.js +980 -0
  324. package/plugins/pbr/scripts/audit-checks/si-agent-hook-config-checks.js +466 -0
  325. package/plugins/pbr/scripts/audit-checks/si-cross-cutting-checks.js +272 -0
  326. package/plugins/pbr/scripts/audit-checks/si-skill-checks.js +424 -0
  327. package/plugins/pbr/scripts/audit-checks/workflow-compliance.js +1211 -0
  328. package/plugins/pbr/scripts/audit-dimensions.js +556 -0
  329. package/plugins/pbr/scripts/auto-continue.js +198 -31
  330. package/plugins/pbr/scripts/block-skill-self-read.js +124 -0
  331. package/plugins/pbr/scripts/check-agent-state-write.js +63 -0
  332. package/plugins/pbr/scripts/check-config-change.js +162 -0
  333. package/plugins/pbr/scripts/check-cross-plugin-sync.js +93 -0
  334. package/plugins/pbr/scripts/check-dangerous-commands.js +18 -5
  335. package/plugins/pbr/scripts/check-direct-state-write.js +37 -0
  336. package/plugins/pbr/scripts/check-phase-boundary.js +3 -8
  337. package/plugins/pbr/scripts/check-plan-format.js +166 -277
  338. package/plugins/pbr/scripts/check-read-first.js +345 -0
  339. package/plugins/pbr/scripts/check-roadmap-sync.js +167 -10
  340. package/plugins/pbr/scripts/check-skill-workflow.js +24 -27
  341. package/plugins/pbr/scripts/check-state-sync.js +339 -215
  342. package/plugins/pbr/scripts/check-subagent-output.js +338 -276
  343. package/plugins/pbr/scripts/check-summary-gate.js +2 -1
  344. package/plugins/pbr/scripts/config-schema.json +1247 -95
  345. package/plugins/pbr/scripts/context-bridge.js +425 -0
  346. package/plugins/pbr/scripts/context-budget-check.js +169 -14
  347. package/plugins/pbr/scripts/context-quality.js +271 -0
  348. package/plugins/pbr/scripts/enforce-context-budget.js +138 -0
  349. package/plugins/pbr/scripts/enforce-pbr-workflow.js +277 -0
  350. package/plugins/pbr/scripts/event-handler.js +128 -77
  351. package/plugins/pbr/scripts/event-logger.js +58 -25
  352. package/plugins/pbr/scripts/feedback-loop.js +172 -0
  353. package/plugins/pbr/scripts/graph-update.js +422 -0
  354. package/plugins/pbr/scripts/hook-logger.js +69 -35
  355. package/plugins/pbr/scripts/hook-server-client.js +361 -0
  356. package/plugins/pbr/scripts/hook-server.js +606 -0
  357. package/plugins/pbr/scripts/hooks-schema.json +13 -5
  358. package/plugins/pbr/scripts/instructions-loaded.js +173 -0
  359. package/plugins/pbr/scripts/intent-router.cjs +147 -0
  360. package/plugins/pbr/scripts/intercept-plan-mode.js +52 -18
  361. package/plugins/pbr/scripts/lib/alternatives.js +203 -0
  362. package/plugins/pbr/scripts/lib/audit.js +65 -0
  363. package/plugins/pbr/scripts/lib/auto-cleanup.js +221 -0
  364. package/plugins/pbr/scripts/lib/auto-verify.js +103 -0
  365. package/plugins/pbr/scripts/lib/autonomy.js +91 -0
  366. package/plugins/pbr/scripts/lib/build.js +719 -0
  367. package/plugins/pbr/scripts/lib/ci-fix-loop.js +228 -0
  368. package/plugins/pbr/scripts/lib/circuit-state.js +133 -0
  369. package/plugins/pbr/scripts/lib/commands.js +483 -0
  370. package/plugins/pbr/scripts/lib/completion.js +377 -0
  371. package/plugins/pbr/scripts/lib/compound.js +216 -0
  372. package/plugins/pbr/scripts/lib/config.js +1315 -0
  373. package/plugins/pbr/scripts/lib/context.js +254 -0
  374. package/plugins/pbr/scripts/lib/contextual-help.js +207 -0
  375. package/plugins/pbr/scripts/lib/convention-detector.js +413 -0
  376. package/plugins/pbr/scripts/lib/core.js +1569 -0
  377. package/plugins/pbr/scripts/lib/dashboard-launch.js +364 -0
  378. package/plugins/pbr/scripts/lib/data-hygiene.js +179 -0
  379. package/plugins/pbr/scripts/lib/decision-extraction.js +183 -0
  380. package/plugins/pbr/scripts/lib/decisions.js +194 -0
  381. package/plugins/pbr/scripts/lib/dependency-break.js +147 -0
  382. package/plugins/pbr/scripts/lib/format-validators.js +1050 -0
  383. package/plugins/pbr/scripts/lib/frontmatter.js +302 -0
  384. package/plugins/pbr/scripts/lib/gates/advisories.js +129 -0
  385. package/plugins/pbr/scripts/lib/gates/build-dependency.js +115 -0
  386. package/plugins/pbr/scripts/lib/gates/build-executor.js +104 -0
  387. package/plugins/pbr/scripts/lib/gates/doc-existence.js +46 -0
  388. package/plugins/pbr/scripts/lib/gates/helpers.js +93 -0
  389. package/plugins/pbr/scripts/lib/gates/inline-execution.js +185 -0
  390. package/plugins/pbr/scripts/lib/gates/milestone-complete.js +136 -0
  391. package/plugins/pbr/scripts/lib/gates/milestone-summary.js +119 -0
  392. package/plugins/pbr/scripts/lib/gates/multi-phase-loader.js +147 -0
  393. package/plugins/pbr/scripts/lib/gates/plan-executor.js +36 -0
  394. package/plugins/pbr/scripts/lib/gates/plan-validation.js +114 -0
  395. package/plugins/pbr/scripts/lib/gates/quick-executor.js +76 -0
  396. package/plugins/pbr/scripts/lib/gates/review-planner.js +61 -0
  397. package/plugins/pbr/scripts/lib/gates/review-verifier.js +69 -0
  398. package/plugins/pbr/scripts/lib/gates/rich-agent-context.js +143 -0
  399. package/plugins/pbr/scripts/lib/gates/user-confirmation.js +93 -0
  400. package/plugins/pbr/scripts/lib/graph-cli.js +89 -0
  401. package/plugins/pbr/scripts/lib/graph.js +553 -0
  402. package/plugins/pbr/scripts/lib/health-checks.js +107 -0
  403. package/plugins/pbr/scripts/lib/health-phase06.js +120 -0
  404. package/plugins/pbr/scripts/lib/health.js +133 -0
  405. package/plugins/pbr/scripts/lib/help.js +151 -0
  406. package/plugins/pbr/scripts/lib/history.js +150 -0
  407. package/plugins/pbr/scripts/lib/hypothesis-runner.js +127 -0
  408. package/plugins/pbr/scripts/lib/impact-analysis.js +319 -0
  409. package/plugins/pbr/scripts/lib/incidents.js +190 -0
  410. package/plugins/pbr/scripts/lib/init.js +643 -0
  411. package/plugins/pbr/scripts/lib/insights-parser.js +320 -0
  412. package/plugins/pbr/scripts/lib/intel.js +653 -0
  413. package/plugins/pbr/scripts/lib/learnings.js +511 -0
  414. package/plugins/pbr/scripts/lib/local-llm/client.js +237 -0
  415. package/plugins/pbr/scripts/lib/local-llm/health.js +12 -0
  416. package/plugins/pbr/scripts/lib/local-llm/index.js +89 -0
  417. package/plugins/pbr/scripts/lib/local-llm/metrics.js +20 -0
  418. package/plugins/pbr/scripts/lib/local-llm/operations/classify-artifact.js +4 -0
  419. package/plugins/pbr/scripts/lib/local-llm/operations/classify-commit.js +4 -0
  420. package/plugins/pbr/scripts/lib/local-llm/operations/classify-error.js +4 -0
  421. package/plugins/pbr/scripts/lib/local-llm/operations/classify-file-intent.js +4 -0
  422. package/plugins/pbr/scripts/lib/local-llm/operations/score-source.js +72 -0
  423. package/plugins/pbr/scripts/lib/local-llm/operations/summarize-context.js +62 -0
  424. package/plugins/pbr/scripts/lib/local-llm/operations/triage-test-output.js +12 -0
  425. package/plugins/pbr/scripts/lib/local-llm/operations/validate-task.js +4 -0
  426. package/plugins/pbr/scripts/lib/local-llm/router.js +101 -0
  427. package/plugins/pbr/scripts/lib/local-llm/shadow.js +60 -0
  428. package/plugins/pbr/scripts/lib/local-llm/threshold-tuner.js +118 -0
  429. package/plugins/pbr/scripts/lib/migrate.js +298 -0
  430. package/plugins/pbr/scripts/lib/milestone.js +306 -0
  431. package/plugins/pbr/scripts/lib/negative-knowledge.js +194 -0
  432. package/plugins/pbr/scripts/lib/notification-throttle.js +141 -0
  433. package/plugins/pbr/scripts/lib/onboarding-generator.js +288 -0
  434. package/plugins/pbr/scripts/lib/parse-args.js +134 -0
  435. package/plugins/pbr/scripts/lib/pattern-routing.js +55 -0
  436. package/plugins/pbr/scripts/lib/patterns.js +272 -0
  437. package/plugins/pbr/scripts/lib/perf.js +190 -0
  438. package/plugins/pbr/scripts/lib/phase.js +1027 -0
  439. package/plugins/pbr/scripts/lib/pid-lock.js +154 -0
  440. package/plugins/pbr/scripts/lib/post-hoc.js +160 -0
  441. package/plugins/pbr/scripts/lib/pre-commit-checks.js +220 -0
  442. package/plugins/pbr/scripts/lib/pre-research.js +133 -0
  443. package/plugins/pbr/scripts/lib/preview.js +174 -0
  444. package/plugins/pbr/scripts/lib/progress-visualization.js +296 -0
  445. package/plugins/pbr/scripts/lib/quick-init.js +131 -0
  446. package/plugins/pbr/scripts/lib/reference.js +236 -0
  447. package/plugins/pbr/scripts/lib/requirements.js +153 -0
  448. package/plugins/pbr/scripts/lib/resolve-root.js +66 -0
  449. package/plugins/pbr/scripts/lib/reverse-spec.js +259 -0
  450. package/plugins/pbr/scripts/lib/roadmap.js +1113 -0
  451. package/plugins/pbr/scripts/lib/security-scan.js +200 -0
  452. package/plugins/pbr/scripts/lib/session-briefing.js +895 -0
  453. package/plugins/pbr/scripts/lib/skill-section.js +99 -0
  454. package/plugins/pbr/scripts/lib/smart-next-task.js +207 -0
  455. package/plugins/pbr/scripts/lib/snapshot-manager.js +232 -0
  456. package/plugins/pbr/scripts/lib/spec-diff.js +209 -0
  457. package/plugins/pbr/scripts/lib/spec-engine.js +189 -0
  458. package/plugins/pbr/scripts/lib/spot-check.js +642 -0
  459. package/plugins/pbr/scripts/lib/state-queue.js +171 -0
  460. package/plugins/pbr/scripts/lib/state.js +1187 -0
  461. package/plugins/pbr/scripts/lib/status-render.js +511 -0
  462. package/plugins/pbr/scripts/lib/step-verify.js +149 -0
  463. package/plugins/pbr/scripts/lib/subagent-validators.js +1059 -0
  464. package/plugins/pbr/scripts/lib/suggest-next.js +435 -0
  465. package/plugins/pbr/scripts/lib/team-composer.js +87 -0
  466. package/plugins/pbr/scripts/lib/team-coordinator.js +153 -0
  467. package/plugins/pbr/scripts/lib/tech-debt-scanner.js +116 -0
  468. package/plugins/pbr/scripts/lib/template.js +222 -0
  469. package/plugins/pbr/scripts/lib/templates.js +362 -0
  470. package/plugins/pbr/scripts/lib/test-cache.js +54 -0
  471. package/plugins/pbr/scripts/lib/test-selection.js +163 -0
  472. package/plugins/pbr/scripts/lib/todo.js +300 -0
  473. package/plugins/pbr/scripts/lib/trust-gate.js +84 -0
  474. package/plugins/pbr/scripts/lib/verify.js +1473 -0
  475. package/plugins/pbr/scripts/lib/wiring-check.js +196 -0
  476. package/plugins/pbr/scripts/log-notification.js +131 -0
  477. package/plugins/pbr/scripts/log-subagent.js +203 -18
  478. package/plugins/pbr/scripts/log-tool-failure.js +60 -8
  479. package/plugins/pbr/scripts/milestone-learnings.js +569 -0
  480. package/plugins/pbr/scripts/package.json +1 -0
  481. package/plugins/pbr/scripts/pbr-tools.js +1833 -1167
  482. package/plugins/pbr/scripts/post-bash-triage.js +154 -0
  483. package/plugins/pbr/scripts/post-compact.js +135 -0
  484. package/plugins/pbr/scripts/post-hoc.js +286 -0
  485. package/plugins/pbr/scripts/post-write-dispatch.js +237 -31
  486. package/plugins/pbr/scripts/post-write-quality.js +4 -3
  487. package/plugins/pbr/scripts/pre-bash-dispatch.js +155 -23
  488. package/plugins/pbr/scripts/pre-skill-dispatch.js +114 -0
  489. package/plugins/pbr/scripts/pre-task-dispatch.js +297 -0
  490. package/plugins/pbr/scripts/pre-write-dispatch.js +171 -47
  491. package/plugins/pbr/scripts/progress-tracker.js +144 -307
  492. package/plugins/pbr/scripts/prompt-guard.js +114 -0
  493. package/plugins/pbr/scripts/prompt-routing.js +209 -0
  494. package/plugins/pbr/scripts/quick-status.js +179 -0
  495. package/plugins/pbr/scripts/record-incident.js +37 -0
  496. package/plugins/pbr/scripts/risk-classifier.cjs +123 -0
  497. package/plugins/pbr/scripts/run-hook.js +62 -10
  498. package/plugins/pbr/scripts/session-cleanup.js +458 -29
  499. package/plugins/pbr/scripts/session-tracker.js +124 -0
  500. package/plugins/pbr/scripts/status-line.js +591 -32
  501. package/plugins/pbr/scripts/suggest-compact.js +203 -7
  502. package/plugins/pbr/scripts/sync-context-to-claude.js +100 -0
  503. package/plugins/pbr/scripts/task-completed.js +165 -4
  504. package/plugins/pbr/scripts/test/config.test.js +126 -0
  505. package/plugins/pbr/scripts/test/cross-platform.test.js +131 -0
  506. package/plugins/pbr/scripts/test/fixtures/config.json +20 -0
  507. package/plugins/pbr/scripts/test/fixtures/plan.md +54 -0
  508. package/plugins/pbr/scripts/test/fixtures/project.md +30 -0
  509. package/plugins/pbr/scripts/test/fixtures/roadmap.md +55 -0
  510. package/plugins/pbr/scripts/test/fixtures/state.md +60 -0
  511. package/plugins/pbr/scripts/test/fixtures/summary.md +35 -0
  512. package/plugins/pbr/scripts/test/fixtures.test.js +184 -0
  513. package/plugins/pbr/scripts/test/phase.test.js +142 -0
  514. package/plugins/pbr/scripts/test/roadmap.test.js +96 -0
  515. package/plugins/pbr/scripts/test/state.test.js +163 -0
  516. package/plugins/pbr/scripts/track-context-budget.js +368 -99
  517. package/plugins/pbr/scripts/track-user-gates.js +88 -0
  518. package/plugins/pbr/scripts/trust-tracker.js +193 -0
  519. package/plugins/pbr/scripts/validate-commit.js +97 -26
  520. package/plugins/pbr/scripts/validate-skill-args.js +87 -15
  521. package/plugins/pbr/scripts/validate-task.js +112 -626
  522. package/plugins/pbr/scripts/worktree-create.js +144 -0
  523. package/plugins/pbr/scripts/worktree-remove.js +147 -0
  524. package/plugins/pbr/skills/audit/SKILL.md +195 -24
  525. package/plugins/pbr/skills/audit-fix/SKILL.md +326 -0
  526. package/plugins/pbr/skills/autonomous/SKILL.md +545 -0
  527. package/plugins/pbr/skills/backlog/SKILL.md +56 -0
  528. package/plugins/pbr/skills/begin/SKILL.md +507 -153
  529. package/plugins/pbr/skills/begin/templates/STATE.md.tmpl +1 -2
  530. package/plugins/pbr/skills/begin/templates/config.json.tmpl +415 -36
  531. package/plugins/pbr/skills/begin/templates/researcher-prompt.md.tmpl +28 -0
  532. package/plugins/pbr/skills/begin/templates/roadmap-prompt.md.tmpl +28 -3
  533. package/plugins/pbr/skills/begin/templates/synthesis-prompt.md.tmpl +33 -5
  534. package/plugins/pbr/skills/build/SKILL.md +1040 -354
  535. package/plugins/pbr/skills/build/templates/continuation-prompt.md.tmpl +26 -0
  536. package/plugins/pbr/skills/build/templates/executor-prompt.md.tmpl +77 -0
  537. package/plugins/pbr/skills/build/templates/inline-verifier-prompt.md.tmpl +33 -0
  538. package/plugins/pbr/skills/config/SKILL.md +111 -9
  539. package/plugins/pbr/skills/continue/SKILL.md +113 -33
  540. package/plugins/pbr/skills/dashboard/SKILL.md +21 -9
  541. package/plugins/pbr/skills/debug/SKILL.md +70 -12
  542. package/plugins/pbr/skills/debug/templates/continuation-prompt.md.tmpl +12 -1
  543. package/plugins/pbr/skills/debug/templates/initial-investigation-prompt.md.tmpl +12 -5
  544. package/plugins/pbr/skills/discuss/SKILL.md +206 -25
  545. package/plugins/pbr/skills/discuss/templates/CONTEXT.md.tmpl +21 -1
  546. package/plugins/pbr/skills/do/SKILL.md +119 -24
  547. package/plugins/pbr/skills/explore/SKILL.md +95 -20
  548. package/plugins/pbr/skills/fast/SKILL.md +94 -0
  549. package/plugins/pbr/skills/forensics/SKILL.md +144 -0
  550. package/plugins/pbr/skills/health/SKILL.md +44 -117
  551. package/plugins/pbr/skills/help/SKILL.md +83 -101
  552. package/plugins/pbr/skills/import/SKILL.md +332 -13
  553. package/plugins/pbr/skills/intel/SKILL.md +131 -0
  554. package/plugins/pbr/skills/list-phase-assumptions/SKILL.md +231 -0
  555. package/plugins/pbr/skills/milestone/SKILL.md +421 -263
  556. package/plugins/pbr/skills/milestone/templates/audit-output.md.tmpl +76 -0
  557. package/plugins/pbr/skills/milestone/templates/complete-output.md.tmpl +32 -0
  558. package/plugins/pbr/skills/milestone/templates/edge-cases.md +54 -0
  559. package/plugins/pbr/skills/milestone/templates/gaps-output.md.tmpl +25 -0
  560. package/plugins/pbr/skills/milestone/templates/integration-checker-prompt.md.tmpl +25 -0
  561. package/plugins/pbr/skills/milestone/templates/new-output.md.tmpl +29 -0
  562. package/plugins/pbr/skills/milestone-summary/SKILL.md +86 -0
  563. package/plugins/pbr/skills/note/SKILL.md +20 -4
  564. package/plugins/pbr/skills/pause/SKILL.md +53 -14
  565. package/plugins/pbr/skills/pause/templates/continue-here.md.tmpl +33 -52
  566. package/plugins/pbr/skills/plan/SKILL.md +526 -280
  567. package/plugins/pbr/skills/plan/templates/checker-prompt.md.tmpl +5 -2
  568. package/plugins/pbr/skills/plan/templates/completion-output.md.tmpl +27 -0
  569. package/plugins/pbr/skills/plan/templates/planner-prompt.md.tmpl +27 -1
  570. package/plugins/pbr/skills/plan/templates/revision-prompt.md.tmpl +21 -5
  571. package/plugins/pbr/skills/profile/SKILL.md +185 -0
  572. package/plugins/pbr/skills/profile-user/SKILL.md +226 -0
  573. package/plugins/pbr/skills/quick/SKILL.md +434 -100
  574. package/plugins/pbr/skills/release/SKILL.md +206 -0
  575. package/plugins/pbr/skills/resume/SKILL.md +169 -46
  576. package/plugins/pbr/skills/review/SKILL.md +217 -164
  577. package/plugins/pbr/skills/review/templates/verifier-prompt.md.tmpl +7 -0
  578. package/plugins/pbr/skills/scan/SKILL.md +151 -106
  579. package/plugins/pbr/skills/scan/templates/mapper-prompt.md.tmpl +5 -56
  580. package/plugins/pbr/skills/seed/SKILL.md +87 -0
  581. package/plugins/pbr/skills/session-report/SKILL.md +130 -0
  582. package/plugins/pbr/skills/setup/SKILL.md +149 -202
  583. package/plugins/pbr/skills/shared/agent-context-enrichment.md +21 -0
  584. package/plugins/pbr/skills/shared/agent-type-resolution.md +32 -0
  585. package/plugins/pbr/skills/shared/commit-planning-docs.md +8 -0
  586. package/plugins/pbr/skills/shared/context-budget.md +66 -1
  587. package/plugins/pbr/skills/shared/context-loader-task.md +18 -11
  588. package/plugins/pbr/skills/shared/digest-select.md +2 -2
  589. package/plugins/pbr/skills/shared/domain-probes.md +1 -1
  590. package/plugins/pbr/skills/shared/error-reporting.md +38 -60
  591. package/plugins/pbr/skills/shared/gate-prompts.md +4 -2
  592. package/plugins/pbr/skills/shared/memory-capture.md +48 -0
  593. package/plugins/pbr/skills/shared/phase-argument-parsing.md +4 -4
  594. package/plugins/pbr/skills/shared/revision-loop.md +24 -6
  595. package/plugins/pbr/skills/shared/state-update.md +47 -54
  596. package/plugins/pbr/skills/shared/universal-anti-patterns.md +27 -4
  597. package/plugins/pbr/skills/ship/SKILL.md +155 -0
  598. package/plugins/pbr/skills/stats/SKILL.md +80 -0
  599. package/plugins/pbr/skills/status/SKILL.md +184 -119
  600. package/plugins/pbr/skills/test/SKILL.md +254 -0
  601. package/plugins/pbr/skills/thread/SKILL.md +73 -0
  602. package/plugins/pbr/skills/todo/SKILL.md +28 -72
  603. package/plugins/pbr/skills/ui-phase/SKILL.md +180 -0
  604. package/plugins/pbr/skills/ui-review/SKILL.md +206 -0
  605. package/plugins/pbr/skills/undo/SKILL.md +221 -0
  606. package/plugins/pbr/skills/validate-phase/SKILL.md +362 -0
  607. package/plugins/pbr/templates/CONTEXT.md.tmpl +45 -20
  608. package/plugins/pbr/templates/DISCOVERY.md.tmpl +29 -0
  609. package/plugins/pbr/templates/DISCUSSION-LOG.md.tmpl +49 -0
  610. package/plugins/pbr/templates/HANDOFF.json.tmpl +30 -0
  611. package/plugins/pbr/templates/INTEGRATION-REPORT.md.tmpl +18 -2
  612. package/plugins/pbr/templates/MILESTONE-AUDIT.md.tmpl +44 -0
  613. package/plugins/pbr/templates/PROJECT.md.tmpl +126 -0
  614. package/plugins/pbr/templates/REQUIREMENTS.md.tmpl +96 -0
  615. package/plugins/pbr/templates/RETROSPECTIVE.md.tmpl +43 -0
  616. package/plugins/pbr/templates/ROADMAP.md.tmpl +108 -14
  617. package/plugins/pbr/templates/SUMMARY-complex.md.tmpl +133 -0
  618. package/plugins/pbr/templates/SUMMARY-minimal.md.tmpl +55 -0
  619. package/plugins/pbr/templates/SUMMARY.md.tmpl +21 -0
  620. package/plugins/pbr/templates/UAT.md.tmpl +94 -0
  621. package/plugins/pbr/templates/UI-SPEC.md.tmpl +144 -0
  622. package/plugins/pbr/templates/VALIDATION.md.tmpl +94 -0
  623. package/plugins/pbr/templates/VERIFICATION-DETAIL.md.tmpl +49 -13
  624. package/plugins/pbr/templates/project-CONTEXT.md.tmpl +59 -0
  625. package/plugins/pbr/templates/research-outputs/ARCHITECTURE.md.tmpl +91 -0
  626. package/plugins/pbr/templates/research-outputs/FEATURES.md.tmpl +64 -0
  627. package/plugins/pbr/templates/research-outputs/PITFALLS.md.tmpl +50 -0
  628. package/plugins/pbr/templates/research-outputs/STACK.md.tmpl +63 -0
  629. package/plugins/pbr/templates/research-outputs/SUMMARY.md.tmpl +98 -0
  630. package/scripts/build-hooks.js +61 -0
  631. package/scripts/check-ci.js +100 -0
  632. package/scripts/clean-changelog.js +364 -0
  633. package/scripts/generate-derivatives.js +581 -0
  634. package/scripts/posttest.js +93 -0
  635. package/scripts/release.js +262 -0
  636. package/scripts/run-tests.cjs +29 -0
  637. package/scripts/test-wrapper.js +43 -0
  638. package/dashboard/bin/cli.js +0 -25
  639. package/dashboard/public/css/layout.css +0 -636
  640. package/dashboard/public/css/status-colors.css +0 -98
  641. package/dashboard/public/css/tokens.css +0 -59
  642. package/dashboard/public/js/htmx-title.js +0 -5
  643. package/dashboard/public/js/sidebar-toggle.js +0 -34
  644. package/dashboard/public/js/sse-client.js +0 -100
  645. package/dashboard/public/js/theme-toggle.js +0 -46
  646. package/dashboard/src/app.js +0 -91
  647. package/dashboard/src/middleware/current-phase.js +0 -24
  648. package/dashboard/src/middleware/errorHandler.js +0 -52
  649. package/dashboard/src/middleware/notFoundHandler.js +0 -9
  650. package/dashboard/src/repositories/planning.repository.js +0 -130
  651. package/dashboard/src/routes/events.routes.js +0 -45
  652. package/dashboard/src/routes/index.routes.js +0 -35
  653. package/dashboard/src/routes/pages.routes.js +0 -426
  654. package/dashboard/src/server.js +0 -42
  655. package/dashboard/src/services/analytics.service.js +0 -141
  656. package/dashboard/src/services/dashboard.service.js +0 -309
  657. package/dashboard/src/services/milestone.service.js +0 -222
  658. package/dashboard/src/services/notes.service.js +0 -50
  659. package/dashboard/src/services/phase.service.js +0 -232
  660. package/dashboard/src/services/project.service.js +0 -57
  661. package/dashboard/src/services/roadmap.service.js +0 -244
  662. package/dashboard/src/services/sse.service.js +0 -58
  663. package/dashboard/src/services/todo.service.js +0 -263
  664. package/dashboard/src/services/watcher.service.js +0 -48
  665. package/dashboard/src/utils/cache.js +0 -55
  666. package/dashboard/src/views/analytics.ejs +0 -5
  667. package/dashboard/src/views/coming-soon.ejs +0 -11
  668. package/dashboard/src/views/dependencies.ejs +0 -5
  669. package/dashboard/src/views/error.ejs +0 -20
  670. package/dashboard/src/views/index.ejs +0 -5
  671. package/dashboard/src/views/milestone-detail.ejs +0 -5
  672. package/dashboard/src/views/milestones.ejs +0 -5
  673. package/dashboard/src/views/notes.ejs +0 -5
  674. package/dashboard/src/views/partials/analytics-content.ejs +0 -90
  675. package/dashboard/src/views/partials/breadcrumbs.ejs +0 -14
  676. package/dashboard/src/views/partials/dashboard-content.ejs +0 -84
  677. package/dashboard/src/views/partials/dependencies-content.ejs +0 -48
  678. package/dashboard/src/views/partials/empty-state.ejs +0 -7
  679. package/dashboard/src/views/partials/footer.ejs +0 -3
  680. package/dashboard/src/views/partials/head.ejs +0 -30
  681. package/dashboard/src/views/partials/header.ejs +0 -21
  682. package/dashboard/src/views/partials/layout-bottom.ejs +0 -6
  683. package/dashboard/src/views/partials/layout-top.ejs +0 -15
  684. package/dashboard/src/views/partials/milestone-detail-content.ejs +0 -20
  685. package/dashboard/src/views/partials/milestones-content.ejs +0 -88
  686. package/dashboard/src/views/partials/notes-content.ejs +0 -23
  687. package/dashboard/src/views/partials/phase-content.ejs +0 -193
  688. package/dashboard/src/views/partials/phase-doc-content.ejs +0 -38
  689. package/dashboard/src/views/partials/phases-content.ejs +0 -124
  690. package/dashboard/src/views/partials/roadmap-content.ejs +0 -155
  691. package/dashboard/src/views/partials/sidebar.ejs +0 -99
  692. package/dashboard/src/views/partials/todo-create-content.ejs +0 -54
  693. package/dashboard/src/views/partials/todo-detail-content.ejs +0 -42
  694. package/dashboard/src/views/partials/todos-content.ejs +0 -94
  695. package/dashboard/src/views/phase-detail.ejs +0 -5
  696. package/dashboard/src/views/phase-doc.ejs +0 -5
  697. package/dashboard/src/views/phases.ejs +0 -5
  698. package/dashboard/src/views/roadmap.ejs +0 -5
  699. package/dashboard/src/views/todo-create.ejs +0 -5
  700. package/dashboard/src/views/todo-detail.ejs +0 -5
  701. package/dashboard/src/views/todos.ejs +0 -5
  702. package/plugins/copilot-pbr/CHANGELOG.md +0 -19
  703. package/plugins/copilot-pbr/README.md +0 -139
  704. package/plugins/copilot-pbr/agents/audit.agent.md +0 -113
  705. package/plugins/copilot-pbr/agents/codebase-mapper.agent.md +0 -151
  706. package/plugins/copilot-pbr/agents/debugger.agent.md +0 -172
  707. package/plugins/copilot-pbr/agents/executor.agent.md +0 -267
  708. package/plugins/copilot-pbr/agents/general.agent.md +0 -88
  709. package/plugins/copilot-pbr/agents/integration-checker.agent.md +0 -119
  710. package/plugins/copilot-pbr/agents/plan-checker.agent.md +0 -199
  711. package/plugins/copilot-pbr/agents/planner.agent.md +0 -238
  712. package/plugins/copilot-pbr/agents/researcher.agent.md +0 -186
  713. package/plugins/copilot-pbr/agents/synthesizer.agent.md +0 -126
  714. package/plugins/copilot-pbr/agents/verifier.agent.md +0 -228
  715. package/plugins/copilot-pbr/hooks/hooks.json +0 -156
  716. package/plugins/copilot-pbr/plugin.json +0 -30
  717. package/plugins/copilot-pbr/references/agent-anti-patterns.md +0 -25
  718. package/plugins/copilot-pbr/references/agent-contracts.md +0 -297
  719. package/plugins/copilot-pbr/references/agent-interactions.md +0 -135
  720. package/plugins/copilot-pbr/references/agent-teams.md +0 -55
  721. package/plugins/copilot-pbr/references/checkpoints.md +0 -158
  722. package/plugins/copilot-pbr/references/common-bug-patterns.md +0 -14
  723. package/plugins/copilot-pbr/references/config-reference.md +0 -442
  724. package/plugins/copilot-pbr/references/continuation-format.md +0 -213
  725. package/plugins/copilot-pbr/references/deviation-rules.md +0 -113
  726. package/plugins/copilot-pbr/references/git-integration.md +0 -227
  727. package/plugins/copilot-pbr/references/integration-patterns.md +0 -118
  728. package/plugins/copilot-pbr/references/model-profiles.md +0 -100
  729. package/plugins/copilot-pbr/references/model-selection.md +0 -32
  730. package/plugins/copilot-pbr/references/pbr-rules.md +0 -195
  731. package/plugins/copilot-pbr/references/pbr-tools-cli.md +0 -285
  732. package/plugins/copilot-pbr/references/plan-authoring.md +0 -182
  733. package/plugins/copilot-pbr/references/plan-format.md +0 -288
  734. package/plugins/copilot-pbr/references/planning-config.md +0 -214
  735. package/plugins/copilot-pbr/references/questioning.md +0 -215
  736. package/plugins/copilot-pbr/references/reading-verification.md +0 -128
  737. package/plugins/copilot-pbr/references/stub-patterns.md +0 -161
  738. package/plugins/copilot-pbr/references/subagent-coordination.md +0 -120
  739. package/plugins/copilot-pbr/references/ui-formatting.md +0 -444
  740. package/plugins/copilot-pbr/references/verification-patterns.md +0 -199
  741. package/plugins/copilot-pbr/references/wave-execution.md +0 -96
  742. package/plugins/copilot-pbr/rules/pbr-workflow.mdc +0 -48
  743. package/plugins/copilot-pbr/setup.ps1 +0 -93
  744. package/plugins/copilot-pbr/setup.sh +0 -93
  745. package/plugins/copilot-pbr/skills/audit/SKILL.md +0 -330
  746. package/plugins/copilot-pbr/skills/begin/SKILL.md +0 -589
  747. package/plugins/copilot-pbr/skills/begin/templates/PROJECT.md.tmpl +0 -34
  748. package/plugins/copilot-pbr/skills/begin/templates/REQUIREMENTS.md.tmpl +0 -19
  749. package/plugins/copilot-pbr/skills/begin/templates/STATE.md.tmpl +0 -50
  750. package/plugins/copilot-pbr/skills/begin/templates/config.json.tmpl +0 -64
  751. package/plugins/copilot-pbr/skills/begin/templates/researcher-prompt.md.tmpl +0 -20
  752. package/plugins/copilot-pbr/skills/begin/templates/roadmap-prompt.md.tmpl +0 -31
  753. package/plugins/copilot-pbr/skills/begin/templates/synthesis-prompt.md.tmpl +0 -17
  754. package/plugins/copilot-pbr/skills/build/SKILL.md +0 -960
  755. package/plugins/copilot-pbr/skills/config/SKILL.md +0 -250
  756. package/plugins/copilot-pbr/skills/continue/SKILL.md +0 -159
  757. package/plugins/copilot-pbr/skills/dashboard/SKILL.md +0 -43
  758. package/plugins/copilot-pbr/skills/debug/SKILL.md +0 -508
  759. package/plugins/copilot-pbr/skills/debug/templates/continuation-prompt.md.tmpl +0 -17
  760. package/plugins/copilot-pbr/skills/debug/templates/initial-investigation-prompt.md.tmpl +0 -28
  761. package/plugins/copilot-pbr/skills/discuss/SKILL.md +0 -353
  762. package/plugins/copilot-pbr/skills/discuss/templates/CONTEXT.md.tmpl +0 -62
  763. package/plugins/copilot-pbr/skills/discuss/templates/decision-categories.md +0 -10
  764. package/plugins/copilot-pbr/skills/do/SKILL.md +0 -66
  765. package/plugins/copilot-pbr/skills/explore/SKILL.md +0 -373
  766. package/plugins/copilot-pbr/skills/health/SKILL.md +0 -274
  767. package/plugins/copilot-pbr/skills/health/templates/check-pattern.md.tmpl +0 -31
  768. package/plugins/copilot-pbr/skills/health/templates/output-format.md.tmpl +0 -64
  769. package/plugins/copilot-pbr/skills/help/SKILL.md +0 -170
  770. package/plugins/copilot-pbr/skills/import/SKILL.md +0 -502
  771. package/plugins/copilot-pbr/skills/milestone/SKILL.md +0 -745
  772. package/plugins/copilot-pbr/skills/milestone/templates/audit-report.md.tmpl +0 -49
  773. package/plugins/copilot-pbr/skills/milestone/templates/stats-file.md.tmpl +0 -31
  774. package/plugins/copilot-pbr/skills/note/SKILL.md +0 -213
  775. package/plugins/copilot-pbr/skills/pause/SKILL.md +0 -247
  776. package/plugins/copilot-pbr/skills/pause/templates/continue-here.md.tmpl +0 -72
  777. package/plugins/copilot-pbr/skills/plan/SKILL.md +0 -662
  778. package/plugins/copilot-pbr/skills/plan/templates/checker-prompt.md.tmpl +0 -22
  779. package/plugins/copilot-pbr/skills/plan/templates/gap-closure-prompt.md.tmpl +0 -33
  780. package/plugins/copilot-pbr/skills/plan/templates/planner-prompt.md.tmpl +0 -39
  781. package/plugins/copilot-pbr/skills/plan/templates/researcher-prompt.md.tmpl +0 -20
  782. package/plugins/copilot-pbr/skills/plan/templates/revision-prompt.md.tmpl +0 -24
  783. package/plugins/copilot-pbr/skills/quick/SKILL.md +0 -376
  784. package/plugins/copilot-pbr/skills/resume/SKILL.md +0 -399
  785. package/plugins/copilot-pbr/skills/review/SKILL.md +0 -653
  786. package/plugins/copilot-pbr/skills/review/templates/debugger-prompt.md.tmpl +0 -61
  787. package/plugins/copilot-pbr/skills/review/templates/gap-planner-prompt.md.tmpl +0 -41
  788. package/plugins/copilot-pbr/skills/review/templates/verifier-prompt.md.tmpl +0 -116
  789. package/plugins/copilot-pbr/skills/scan/SKILL.md +0 -299
  790. package/plugins/copilot-pbr/skills/scan/templates/mapper-prompt.md.tmpl +0 -202
  791. package/plugins/copilot-pbr/skills/setup/SKILL.md +0 -296
  792. package/plugins/copilot-pbr/skills/shared/commit-planning-docs.md +0 -36
  793. package/plugins/copilot-pbr/skills/shared/config-loading.md +0 -103
  794. package/plugins/copilot-pbr/skills/shared/context-budget.md +0 -41
  795. package/plugins/copilot-pbr/skills/shared/context-loader-task.md +0 -87
  796. package/plugins/copilot-pbr/skills/shared/digest-select.md +0 -80
  797. package/plugins/copilot-pbr/skills/shared/domain-probes.md +0 -126
  798. package/plugins/copilot-pbr/skills/shared/error-reporting.md +0 -81
  799. package/plugins/copilot-pbr/skills/shared/gate-prompts.md +0 -389
  800. package/plugins/copilot-pbr/skills/shared/phase-argument-parsing.md +0 -46
  801. package/plugins/copilot-pbr/skills/shared/progress-display.md +0 -53
  802. package/plugins/copilot-pbr/skills/shared/revision-loop.md +0 -82
  803. package/plugins/copilot-pbr/skills/shared/state-loading.md +0 -63
  804. package/plugins/copilot-pbr/skills/shared/state-update.md +0 -162
  805. package/plugins/copilot-pbr/skills/shared/universal-anti-patterns.md +0 -38
  806. package/plugins/copilot-pbr/skills/status/SKILL.md +0 -362
  807. package/plugins/copilot-pbr/skills/statusline/SKILL.md +0 -149
  808. package/plugins/copilot-pbr/skills/todo/SKILL.md +0 -279
  809. package/plugins/copilot-pbr/templates/CONTEXT.md.tmpl +0 -53
  810. package/plugins/copilot-pbr/templates/INTEGRATION-REPORT.md.tmpl +0 -152
  811. package/plugins/copilot-pbr/templates/RESEARCH-SUMMARY.md.tmpl +0 -98
  812. package/plugins/copilot-pbr/templates/ROADMAP.md.tmpl +0 -41
  813. package/plugins/copilot-pbr/templates/SUMMARY.md.tmpl +0 -82
  814. package/plugins/copilot-pbr/templates/VERIFICATION-DETAIL.md.tmpl +0 -117
  815. package/plugins/copilot-pbr/templates/codebase/ARCHITECTURE.md.tmpl +0 -98
  816. package/plugins/copilot-pbr/templates/codebase/CONCERNS.md.tmpl +0 -93
  817. package/plugins/copilot-pbr/templates/codebase/CONVENTIONS.md.tmpl +0 -104
  818. package/plugins/copilot-pbr/templates/codebase/INTEGRATIONS.md.tmpl +0 -78
  819. package/plugins/copilot-pbr/templates/codebase/STACK.md.tmpl +0 -78
  820. package/plugins/copilot-pbr/templates/codebase/STRUCTURE.md.tmpl +0 -80
  821. package/plugins/copilot-pbr/templates/codebase/TESTING.md.tmpl +0 -107
  822. package/plugins/copilot-pbr/templates/continue-here.md.tmpl +0 -74
  823. package/plugins/copilot-pbr/templates/prompt-partials/phase-project-context.md.tmpl +0 -38
  824. package/plugins/copilot-pbr/templates/research/ARCHITECTURE.md.tmpl +0 -124
  825. package/plugins/copilot-pbr/templates/research/STACK.md.tmpl +0 -71
  826. package/plugins/copilot-pbr/templates/research/SUMMARY.md.tmpl +0 -112
  827. package/plugins/copilot-pbr/templates/research-outputs/phase-research.md.tmpl +0 -81
  828. package/plugins/copilot-pbr/templates/research-outputs/project-research.md.tmpl +0 -99
  829. package/plugins/copilot-pbr/templates/research-outputs/synthesis.md.tmpl +0 -36
  830. package/plugins/cursor-pbr/.cursor-plugin/plugin.json +0 -32
  831. package/plugins/cursor-pbr/CHANGELOG.md +0 -15
  832. package/plugins/cursor-pbr/README.md +0 -123
  833. package/plugins/cursor-pbr/agents/audit.md +0 -178
  834. package/plugins/cursor-pbr/agents/codebase-mapper.md +0 -150
  835. package/plugins/cursor-pbr/agents/debugger.md +0 -171
  836. package/plugins/cursor-pbr/agents/executor.md +0 -266
  837. package/plugins/cursor-pbr/agents/general.md +0 -87
  838. package/plugins/cursor-pbr/agents/integration-checker.md +0 -118
  839. package/plugins/cursor-pbr/agents/plan-checker.md +0 -198
  840. package/plugins/cursor-pbr/agents/planner.md +0 -237
  841. package/plugins/cursor-pbr/agents/researcher.md +0 -185
  842. package/plugins/cursor-pbr/agents/synthesizer.md +0 -125
  843. package/plugins/cursor-pbr/agents/verifier.md +0 -227
  844. package/plugins/cursor-pbr/assets/.gitkeep +0 -0
  845. package/plugins/cursor-pbr/assets/logo.svg +0 -21
  846. package/plugins/cursor-pbr/hooks/hooks.json +0 -213
  847. package/plugins/cursor-pbr/references/agent-anti-patterns.md +0 -25
  848. package/plugins/cursor-pbr/references/agent-contracts.md +0 -297
  849. package/plugins/cursor-pbr/references/agent-interactions.md +0 -135
  850. package/plugins/cursor-pbr/references/agent-teams.md +0 -55
  851. package/plugins/cursor-pbr/references/checkpoints.md +0 -158
  852. package/plugins/cursor-pbr/references/common-bug-patterns.md +0 -14
  853. package/plugins/cursor-pbr/references/config-reference.md +0 -442
  854. package/plugins/cursor-pbr/references/continuation-format.md +0 -213
  855. package/plugins/cursor-pbr/references/deviation-rules.md +0 -113
  856. package/plugins/cursor-pbr/references/git-integration.md +0 -227
  857. package/plugins/cursor-pbr/references/integration-patterns.md +0 -118
  858. package/plugins/cursor-pbr/references/model-profiles.md +0 -100
  859. package/plugins/cursor-pbr/references/model-selection.md +0 -32
  860. package/plugins/cursor-pbr/references/pbr-rules.md +0 -195
  861. package/plugins/cursor-pbr/references/pbr-tools-cli.md +0 -285
  862. package/plugins/cursor-pbr/references/plan-authoring.md +0 -182
  863. package/plugins/cursor-pbr/references/plan-format.md +0 -288
  864. package/plugins/cursor-pbr/references/planning-config.md +0 -214
  865. package/plugins/cursor-pbr/references/questioning.md +0 -215
  866. package/plugins/cursor-pbr/references/reading-verification.md +0 -128
  867. package/plugins/cursor-pbr/references/stub-patterns.md +0 -161
  868. package/plugins/cursor-pbr/references/subagent-coordination.md +0 -120
  869. package/plugins/cursor-pbr/references/ui-formatting.md +0 -444
  870. package/plugins/cursor-pbr/references/verification-patterns.md +0 -199
  871. package/plugins/cursor-pbr/references/wave-execution.md +0 -96
  872. package/plugins/cursor-pbr/rules/pbr-workflow.mdc +0 -48
  873. package/plugins/cursor-pbr/setup.ps1 +0 -78
  874. package/plugins/cursor-pbr/setup.sh +0 -83
  875. package/plugins/cursor-pbr/skills/audit/SKILL.md +0 -331
  876. package/plugins/cursor-pbr/skills/begin/SKILL.md +0 -589
  877. package/plugins/cursor-pbr/skills/begin/templates/PROJECT.md.tmpl +0 -34
  878. package/plugins/cursor-pbr/skills/begin/templates/REQUIREMENTS.md.tmpl +0 -19
  879. package/plugins/cursor-pbr/skills/begin/templates/STATE.md.tmpl +0 -50
  880. package/plugins/cursor-pbr/skills/begin/templates/config.json.tmpl +0 -64
  881. package/plugins/cursor-pbr/skills/begin/templates/researcher-prompt.md.tmpl +0 -20
  882. package/plugins/cursor-pbr/skills/begin/templates/roadmap-prompt.md.tmpl +0 -31
  883. package/plugins/cursor-pbr/skills/begin/templates/synthesis-prompt.md.tmpl +0 -17
  884. package/plugins/cursor-pbr/skills/build/SKILL.md +0 -961
  885. package/plugins/cursor-pbr/skills/config/SKILL.md +0 -252
  886. package/plugins/cursor-pbr/skills/continue/SKILL.md +0 -159
  887. package/plugins/cursor-pbr/skills/dashboard/SKILL.md +0 -44
  888. package/plugins/cursor-pbr/skills/debug/SKILL.md +0 -512
  889. package/plugins/cursor-pbr/skills/debug/templates/continuation-prompt.md.tmpl +0 -17
  890. package/plugins/cursor-pbr/skills/debug/templates/initial-investigation-prompt.md.tmpl +0 -28
  891. package/plugins/cursor-pbr/skills/discuss/SKILL.md +0 -354
  892. package/plugins/cursor-pbr/skills/discuss/templates/CONTEXT.md.tmpl +0 -62
  893. package/plugins/cursor-pbr/skills/discuss/templates/decision-categories.md +0 -10
  894. package/plugins/cursor-pbr/skills/do/SKILL.md +0 -67
  895. package/plugins/cursor-pbr/skills/explore/SKILL.md +0 -376
  896. package/plugins/cursor-pbr/skills/health/SKILL.md +0 -274
  897. package/plugins/cursor-pbr/skills/health/templates/check-pattern.md.tmpl +0 -31
  898. package/plugins/cursor-pbr/skills/health/templates/output-format.md.tmpl +0 -64
  899. package/plugins/cursor-pbr/skills/help/SKILL.md +0 -170
  900. package/plugins/cursor-pbr/skills/import/SKILL.md +0 -505
  901. package/plugins/cursor-pbr/skills/milestone/SKILL.md +0 -746
  902. package/plugins/cursor-pbr/skills/milestone/templates/audit-report.md.tmpl +0 -49
  903. package/plugins/cursor-pbr/skills/milestone/templates/stats-file.md.tmpl +0 -31
  904. package/plugins/cursor-pbr/skills/note/SKILL.md +0 -214
  905. package/plugins/cursor-pbr/skills/pause/SKILL.md +0 -248
  906. package/plugins/cursor-pbr/skills/pause/templates/continue-here.md.tmpl +0 -72
  907. package/plugins/cursor-pbr/skills/plan/SKILL.md +0 -663
  908. package/plugins/cursor-pbr/skills/plan/templates/checker-prompt.md.tmpl +0 -22
  909. package/plugins/cursor-pbr/skills/plan/templates/gap-closure-prompt.md.tmpl +0 -33
  910. package/plugins/cursor-pbr/skills/plan/templates/planner-prompt.md.tmpl +0 -39
  911. package/plugins/cursor-pbr/skills/plan/templates/researcher-prompt.md.tmpl +0 -20
  912. package/plugins/cursor-pbr/skills/plan/templates/revision-prompt.md.tmpl +0 -24
  913. package/plugins/cursor-pbr/skills/quick/SKILL.md +0 -376
  914. package/plugins/cursor-pbr/skills/resume/SKILL.md +0 -399
  915. package/plugins/cursor-pbr/skills/review/SKILL.md +0 -654
  916. package/plugins/cursor-pbr/skills/review/templates/debugger-prompt.md.tmpl +0 -61
  917. package/plugins/cursor-pbr/skills/review/templates/gap-planner-prompt.md.tmpl +0 -41
  918. package/plugins/cursor-pbr/skills/review/templates/verifier-prompt.md.tmpl +0 -116
  919. package/plugins/cursor-pbr/skills/scan/SKILL.md +0 -300
  920. package/plugins/cursor-pbr/skills/scan/templates/mapper-prompt.md.tmpl +0 -202
  921. package/plugins/cursor-pbr/skills/setup/SKILL.md +0 -296
  922. package/plugins/cursor-pbr/skills/shared/commit-planning-docs.md +0 -36
  923. package/plugins/cursor-pbr/skills/shared/config-loading.md +0 -103
  924. package/plugins/cursor-pbr/skills/shared/context-budget.md +0 -41
  925. package/plugins/cursor-pbr/skills/shared/context-loader-task.md +0 -87
  926. package/plugins/cursor-pbr/skills/shared/digest-select.md +0 -80
  927. package/plugins/cursor-pbr/skills/shared/domain-probes.md +0 -126
  928. package/plugins/cursor-pbr/skills/shared/error-reporting.md +0 -81
  929. package/plugins/cursor-pbr/skills/shared/gate-prompts.md +0 -389
  930. package/plugins/cursor-pbr/skills/shared/phase-argument-parsing.md +0 -46
  931. package/plugins/cursor-pbr/skills/shared/progress-display.md +0 -53
  932. package/plugins/cursor-pbr/skills/shared/revision-loop.md +0 -82
  933. package/plugins/cursor-pbr/skills/shared/state-loading.md +0 -63
  934. package/plugins/cursor-pbr/skills/shared/state-update.md +0 -162
  935. package/plugins/cursor-pbr/skills/shared/universal-anti-patterns.md +0 -38
  936. package/plugins/cursor-pbr/skills/status/SKILL.md +0 -362
  937. package/plugins/cursor-pbr/skills/statusline/SKILL.md +0 -150
  938. package/plugins/cursor-pbr/skills/todo/SKILL.md +0 -280
  939. package/plugins/cursor-pbr/templates/CONTEXT.md.tmpl +0 -53
  940. package/plugins/cursor-pbr/templates/INTEGRATION-REPORT.md.tmpl +0 -152
  941. package/plugins/cursor-pbr/templates/RESEARCH-SUMMARY.md.tmpl +0 -98
  942. package/plugins/cursor-pbr/templates/ROADMAP.md.tmpl +0 -41
  943. package/plugins/cursor-pbr/templates/SUMMARY.md.tmpl +0 -82
  944. package/plugins/cursor-pbr/templates/VERIFICATION-DETAIL.md.tmpl +0 -117
  945. package/plugins/cursor-pbr/templates/codebase/ARCHITECTURE.md.tmpl +0 -98
  946. package/plugins/cursor-pbr/templates/codebase/CONCERNS.md.tmpl +0 -93
  947. package/plugins/cursor-pbr/templates/codebase/CONVENTIONS.md.tmpl +0 -104
  948. package/plugins/cursor-pbr/templates/codebase/INTEGRATIONS.md.tmpl +0 -78
  949. package/plugins/cursor-pbr/templates/codebase/STACK.md.tmpl +0 -78
  950. package/plugins/cursor-pbr/templates/codebase/STRUCTURE.md.tmpl +0 -80
  951. package/plugins/cursor-pbr/templates/codebase/TESTING.md.tmpl +0 -107
  952. package/plugins/cursor-pbr/templates/continue-here.md.tmpl +0 -74
  953. package/plugins/cursor-pbr/templates/prompt-partials/phase-project-context.md.tmpl +0 -38
  954. package/plugins/cursor-pbr/templates/research/ARCHITECTURE.md.tmpl +0 -124
  955. package/plugins/cursor-pbr/templates/research/STACK.md.tmpl +0 -71
  956. package/plugins/cursor-pbr/templates/research/SUMMARY.md.tmpl +0 -112
  957. package/plugins/cursor-pbr/templates/research-outputs/phase-research.md.tmpl +0 -81
  958. package/plugins/cursor-pbr/templates/research-outputs/project-research.md.tmpl +0 -99
  959. package/plugins/cursor-pbr/templates/research-outputs/synthesis.md.tmpl +0 -36
  960. package/plugins/pbr/references/agent-interactions.md +0 -134
  961. package/plugins/pbr/references/pbr-rules.md +0 -194
  962. package/plugins/pbr/references/pbr-tools-cli.md +0 -285
  963. package/plugins/pbr/references/planning-config.md +0 -213
  964. package/plugins/pbr/references/subagent-coordination.md +0 -119
  965. package/plugins/pbr/references/ui-formatting.md +0 -444
  966. package/plugins/pbr/scripts/validate-plugin-structure.js +0 -183
  967. package/plugins/pbr/skills/milestone/templates/audit-report.md.tmpl +0 -48
  968. package/plugins/pbr/skills/shared/progress-display.md +0 -53
  969. package/plugins/pbr/skills/shared/state-loading.md +0 -62
  970. package/plugins/pbr/templates/RESEARCH-SUMMARY.md.tmpl +0 -97
  971. package/plugins/pbr/templates/research/ARCHITECTURE.md.tmpl +0 -124
  972. package/plugins/pbr/templates/research/STACK.md.tmpl +0 -71
  973. package/plugins/pbr/templates/research/SUMMARY.md.tmpl +0 -112
  974. package/plugins/pbr/templates/research-outputs/phase-research.md.tmpl +0 -81
  975. package/plugins/pbr/templates/research-outputs/project-research.md.tmpl +0 -99
  976. package/plugins/pbr/templates/research-outputs/synthesis.md.tmpl +0 -36
  977. /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
+ };