@luanpdd/kit-mcp 1.33.0 → 1.35.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (379) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +168 -168
  3. package/gates/agent-no-recursive-dispatch.md +84 -84
  4. package/kit/COMANDOS.md +138 -138
  5. package/kit/COMPATIBILITY.md +70 -70
  6. package/kit/README.md +76 -76
  7. package/kit/agents/advisor-researcher.md +109 -109
  8. package/kit/agents/ai-mutation-tester.md +289 -289
  9. package/kit/agents/assumptions-analyzer.md +110 -110
  10. package/kit/agents/audit-log-implementer.md +314 -314
  11. package/kit/agents/auditor-consistencia-isolamento.md +414 -414
  12. package/kit/agents/b2b-saas-architect.md +157 -157
  13. package/kit/agents/burn-rate-forecaster.md +153 -153
  14. package/kit/agents/cascading-failures-auditor.md +299 -299
  15. package/kit/agents/codebase-mapper.md +769 -769
  16. package/kit/agents/crm-pipeline-implementer.md +257 -257
  17. package/kit/agents/debugger.md +814 -814
  18. package/kit/agents/designer-ui.md +216 -216
  19. package/kit/agents/detector-tenant-quente.md +338 -338
  20. package/kit/agents/evolution-go-integrator.md +201 -201
  21. package/kit/agents/example-reviewer.md +22 -22
  22. package/kit/agents/executor.md +565 -565
  23. package/kit/agents/golden-signals-instrumenter.md +232 -232
  24. package/kit/agents/incident-investigator.md +238 -238
  25. package/kit/agents/integration-checker.md +203 -203
  26. package/kit/agents/invite-flow-implementer.md +190 -190
  27. package/kit/agents/legacy-characterizer.md +369 -369
  28. package/kit/agents/lgpd-compliance-auditor.md +296 -296
  29. package/kit/agents/load-shedding-instrumenter.md +290 -290
  30. package/kit/agents/multi-tenant-isolation-auditor.md +254 -254
  31. package/kit/agents/multi-tenant-rls-writer.md +341 -341
  32. package/kit/agents/nyquist-auditor.md +181 -181
  33. package/kit/agents/observability-coverage-auditor.md +316 -316
  34. package/kit/agents/observability-instrumenter.md +191 -191
  35. package/kit/agents/omm-auditor.md +291 -291
  36. package/kit/agents/org-onboarding-implementer.md +224 -224
  37. package/kit/agents/payload-capture-instrumenter.md +274 -274
  38. package/kit/agents/phase-researcher.md +697 -697
  39. package/kit/agents/plan-checker.md +275 -275
  40. package/kit/agents/planner.md +923 -923
  41. package/kit/agents/postmortem-writer.md +273 -273
  42. package/kit/agents/project-researcher.md +653 -653
  43. package/kit/agents/prr-conductor.md +287 -287
  44. package/kit/agents/refactor-safety-auditor.md +405 -405
  45. package/kit/agents/release-pipeline-auditor.md +364 -364
  46. package/kit/agents/research-synthesizer.md +246 -246
  47. package/kit/agents/roadmapper.md +678 -678
  48. package/kit/agents/schema-checker.md +160 -160
  49. package/kit/agents/seam-finder.md +360 -360
  50. package/kit/agents/shotgun-surgery-detector.md +350 -350
  51. package/kit/agents/slo-engineer.md +217 -217
  52. package/kit/agents/storytelling-analyst.md +300 -300
  53. package/kit/agents/supabase-architect.md +249 -249
  54. package/kit/agents/supabase-auth-bootstrapper.md +400 -400
  55. package/kit/agents/supabase-auth-hook-writer.md +418 -418
  56. package/kit/agents/supabase-branching-architect.md +563 -563
  57. package/kit/agents/supabase-cicd-pipeline-implementer.md +778 -778
  58. package/kit/agents/supabase-column-privileges-writer.md +400 -400
  59. package/kit/agents/supabase-edge-fn-tester.md +288 -288
  60. package/kit/agents/supabase-edge-fn-writer.md +341 -341
  61. package/kit/agents/supabase-mfa-implementer.md +439 -439
  62. package/kit/agents/supabase-migration-writer.md +386 -386
  63. package/kit/agents/supabase-oauth-server-implementer.md +507 -507
  64. package/kit/agents/supabase-rbac-implementer.md +393 -393
  65. package/kit/agents/supabase-realtime-implementer.md +364 -364
  66. package/kit/agents/supabase-rls-hardener.md +522 -522
  67. package/kit/agents/supabase-rls-writer.md +324 -324
  68. package/kit/agents/supabase-roles-implementer.md +356 -356
  69. package/kit/agents/supabase-social-auth-implementer.md +451 -451
  70. package/kit/agents/supabase-sso-saml-architect.md +549 -549
  71. package/kit/agents/supabase-storage-implementer.md +407 -407
  72. package/kit/agents/super-admin-implementer.md +282 -282
  73. package/kit/agents/toil-auditor.md +268 -268
  74. package/kit/agents/ui-auditor.md +438 -438
  75. package/kit/agents/ui-checker.md +305 -305
  76. package/kit/agents/ui-researcher.md +356 -356
  77. package/kit/agents/user-profiler.md +176 -176
  78. package/kit/agents/validador-evolucao-schema.md +336 -336
  79. package/kit/agents/verifier.md +729 -729
  80. package/kit/agents/workflow-generator.md +167 -0
  81. package/kit/commands/adicionar-backlog.md +75 -75
  82. package/kit/commands/adicionar-fase.md +42 -42
  83. package/kit/commands/adicionar-tarefa.md +45 -45
  84. package/kit/commands/adicionar-testes.md +41 -41
  85. package/kit/commands/ajuda.md +21 -21
  86. package/kit/commands/atualizar.md +37 -37
  87. package/kit/commands/auditar-cascading.md +111 -111
  88. package/kit/commands/auditar-marco.md +179 -179
  89. package/kit/commands/auditar-observabilidade-cobertura-workflow.md +121 -0
  90. package/kit/commands/auditar-observabilidade-cobertura.md +183 -183
  91. package/kit/commands/auditar-refactor.md +219 -219
  92. package/kit/commands/auditar-release.md +109 -109
  93. package/kit/commands/auditar-uat.md +23 -23
  94. package/kit/commands/autonomo.md +40 -40
  95. package/kit/commands/branch-pr.md +24 -24
  96. package/kit/commands/burn-rate-status.md +408 -408
  97. package/kit/commands/capturar-payloads.md +193 -193
  98. package/kit/commands/caracterizar.md +212 -212
  99. package/kit/commands/concluir-marco.md +247 -247
  100. package/kit/commands/configuracoes.md +36 -36
  101. package/kit/commands/criar-workflow.md +158 -0
  102. package/kit/commands/dados-distribuidos.md +188 -188
  103. package/kit/commands/definir-perfil.md +10 -10
  104. package/kit/commands/depurar.md +190 -190
  105. package/kit/commands/detectar-duplicacao.md +197 -197
  106. package/kit/commands/discutir-fase.md +131 -131
  107. package/kit/commands/encontrar-seams.md +136 -136
  108. package/kit/commands/entrar-discord.md +17 -17
  109. package/kit/commands/estatisticas.md +18 -18
  110. package/kit/commands/example-greeting.md +33 -33
  111. package/kit/commands/executar-fase.md +58 -58
  112. package/kit/commands/expresso.md +56 -56
  113. package/kit/commands/fase-ui.md +34 -34
  114. package/kit/commands/fazer.md +57 -57
  115. package/kit/commands/fio.md +125 -125
  116. package/kit/commands/fluxos-trabalho.md +64 -64
  117. package/kit/commands/forense.md +176 -176
  118. package/kit/commands/gerenciador.md +38 -38
  119. package/kit/commands/inserir-fase.md +31 -31
  120. package/kit/commands/legacy.md +263 -263
  121. package/kit/commands/limpeza.md +17 -17
  122. package/kit/commands/listar-hipoteses-fase.md +45 -45
  123. package/kit/commands/listar-workspaces.md +18 -18
  124. package/kit/commands/load-shedding.md +117 -117
  125. package/kit/commands/mapear-codebase.md +70 -70
  126. package/kit/commands/multi-tenant.md +163 -163
  127. package/kit/commands/nota.md +33 -33
  128. package/kit/commands/novo-marco.md +43 -43
  129. package/kit/commands/novo-projeto.md +41 -41
  130. package/kit/commands/novo-workspace.md +43 -43
  131. package/kit/commands/pausar-trabalho.md +37 -37
  132. package/kit/commands/perfil-usuario.md +45 -45
  133. package/kit/commands/pesquisar-fase.md +195 -195
  134. package/kit/commands/planejar-fase.md +67 -67
  135. package/kit/commands/planejar-lacunas.md +33 -33
  136. package/kit/commands/plantar-ideia.md +25 -25
  137. package/kit/commands/progresso.md +24 -24
  138. package/kit/commands/proximo.md +30 -30
  139. package/kit/commands/publicar.md +490 -490
  140. package/kit/commands/rapido.md +35 -35
  141. package/kit/commands/reaplicar-patches.md +124 -124
  142. package/kit/commands/refactor-seguro.md +321 -321
  143. package/kit/commands/relatorio-sessao.md +19 -19
  144. package/kit/commands/remover-fase.md +31 -31
  145. package/kit/commands/remover-workspace.md +26 -26
  146. package/kit/commands/resumo-marco.md +50 -50
  147. package/kit/commands/retomar-trabalho.md +40 -40
  148. package/kit/commands/revisar-backlog.md +60 -60
  149. package/kit/commands/revisar-ui.md +32 -32
  150. package/kit/commands/revisar.md +37 -37
  151. package/kit/commands/saude.md +21 -21
  152. package/kit/commands/setup-notion.md +93 -93
  153. package/kit/commands/storytelling.md +179 -179
  154. package/kit/commands/supabase.md +238 -238
  155. package/kit/commands/sync-main.md +68 -68
  156. package/kit/commands/validar-fase.md +35 -35
  157. package/kit/commands/verificar-tarefas.md +44 -44
  158. package/kit/commands/verificar-trabalho.md +64 -64
  159. package/kit/file-manifest.json +424 -419
  160. package/kit/framework/bin/lib/commands.cjs +959 -959
  161. package/kit/framework/bin/lib/config.cjs +442 -442
  162. package/kit/framework/bin/lib/core.cjs +1230 -1230
  163. package/kit/framework/bin/lib/frontmatter.cjs +336 -336
  164. package/kit/framework/bin/lib/init.cjs +1442 -1442
  165. package/kit/framework/bin/lib/milestone.cjs +252 -252
  166. package/kit/framework/bin/lib/model-profiles.cjs +68 -68
  167. package/kit/framework/bin/lib/phase.cjs +888 -888
  168. package/kit/framework/bin/lib/profile-output.cjs +952 -952
  169. package/kit/framework/bin/lib/profile-pipeline.cjs +539 -539
  170. package/kit/framework/bin/lib/roadmap.cjs +329 -329
  171. package/kit/framework/bin/lib/security.cjs +382 -382
  172. package/kit/framework/bin/lib/state.cjs +1031 -1031
  173. package/kit/framework/bin/lib/template.cjs +222 -222
  174. package/kit/framework/bin/lib/uat.cjs +282 -282
  175. package/kit/framework/bin/lib/verify.cjs +888 -888
  176. package/kit/framework/bin/lib/workstream.cjs +491 -491
  177. package/kit/framework/bin/tools.cjs +918 -918
  178. package/kit/framework/commands/workstreams.md +63 -63
  179. package/kit/framework/references/checkpoints.md +778 -778
  180. package/kit/framework/references/continuation-format.md +249 -249
  181. package/kit/framework/references/decimal-phase-calculation.md +64 -64
  182. package/kit/framework/references/git-integration.md +295 -295
  183. package/kit/framework/references/git-planning-commit.md +38 -38
  184. package/kit/framework/references/model-profile-resolution.md +36 -36
  185. package/kit/framework/references/model-profiles.md +139 -139
  186. package/kit/framework/references/phase-argument-parsing.md +61 -61
  187. package/kit/framework/references/planning-config.md +202 -202
  188. package/kit/framework/references/questioning.md +162 -162
  189. package/kit/framework/references/tdd.md +263 -263
  190. package/kit/framework/references/ui-brand.md +160 -160
  191. package/kit/framework/references/user-profiling.md +657 -657
  192. package/kit/framework/references/verification-patterns.md +612 -612
  193. package/kit/framework/references/workstream-flag.md +58 -58
  194. package/kit/framework/templates/DEBUG.md +164 -164
  195. package/kit/framework/templates/UAT.md +265 -265
  196. package/kit/framework/templates/UI-SPEC.md +100 -100
  197. package/kit/framework/templates/VALIDATION.md +76 -76
  198. package/kit/framework/templates/claude-md.md +122 -122
  199. package/kit/framework/templates/codebase/architecture.md +185 -185
  200. package/kit/framework/templates/codebase/concerns.md +205 -205
  201. package/kit/framework/templates/codebase/conventions.md +204 -204
  202. package/kit/framework/templates/codebase/integrations.md +192 -192
  203. package/kit/framework/templates/codebase/stack.md +158 -158
  204. package/kit/framework/templates/codebase/structure.md +199 -199
  205. package/kit/framework/templates/codebase/testing.md +301 -301
  206. package/kit/framework/templates/config.json +44 -44
  207. package/kit/framework/templates/context.md +352 -352
  208. package/kit/framework/templates/continue-here.md +78 -78
  209. package/kit/framework/templates/copilot-instructions.md +7 -7
  210. package/kit/framework/templates/debug-subagent-prompt.md +91 -91
  211. package/kit/framework/templates/dev-preferences.md +20 -20
  212. package/kit/framework/templates/discovery.md +146 -146
  213. package/kit/framework/templates/discussion-log.md +63 -63
  214. package/kit/framework/templates/milestone-archive.md +123 -123
  215. package/kit/framework/templates/milestone.md +115 -115
  216. package/kit/framework/templates/phase-prompt.md +610 -610
  217. package/kit/framework/templates/planner-subagent-prompt.md +117 -117
  218. package/kit/framework/templates/project.md +186 -186
  219. package/kit/framework/templates/requirements.md +231 -231
  220. package/kit/framework/templates/research-project/ARCHITECTURE.md +204 -204
  221. package/kit/framework/templates/research-project/FEATURES.md +147 -147
  222. package/kit/framework/templates/research-project/PITFALLS.md +200 -200
  223. package/kit/framework/templates/research-project/STACK.md +120 -120
  224. package/kit/framework/templates/research-project/SUMMARY.md +170 -170
  225. package/kit/framework/templates/research.md +419 -419
  226. package/kit/framework/templates/retrospective.md +54 -54
  227. package/kit/framework/templates/roadmap.md +202 -202
  228. package/kit/framework/templates/state.md +176 -176
  229. package/kit/framework/templates/summary-complex.md +59 -59
  230. package/kit/framework/templates/summary-minimal.md +41 -41
  231. package/kit/framework/templates/summary-standard.md +48 -48
  232. package/kit/framework/templates/summary.md +209 -209
  233. package/kit/framework/templates/user-profile.md +146 -146
  234. package/kit/framework/templates/user-setup.md +256 -256
  235. package/kit/framework/templates/verification-report.md +258 -258
  236. package/kit/framework/workflows/add-phase.md +112 -112
  237. package/kit/framework/workflows/add-tests.md +351 -351
  238. package/kit/framework/workflows/add-todo.md +158 -158
  239. package/kit/framework/workflows/audit-milestone.md +340 -340
  240. package/kit/framework/workflows/audit-uat.md +109 -109
  241. package/kit/framework/workflows/autonomous.md +891 -891
  242. package/kit/framework/workflows/check-todos.md +177 -177
  243. package/kit/framework/workflows/cleanup.md +152 -152
  244. package/kit/framework/workflows/complete-milestone.md +696 -696
  245. package/kit/framework/workflows/diagnose-issues.md +231 -231
  246. package/kit/framework/workflows/discovery-phase.md +289 -289
  247. package/kit/framework/workflows/discuss-phase-assumptions.md +653 -653
  248. package/kit/framework/workflows/discuss-phase.md +784 -784
  249. package/kit/framework/workflows/do.md +104 -104
  250. package/kit/framework/workflows/execute-phase.md +838 -838
  251. package/kit/framework/workflows/execute-plan.md +510 -510
  252. package/kit/framework/workflows/fast.md +102 -102
  253. package/kit/framework/workflows/forensics.md +265 -265
  254. package/kit/framework/workflows/health.md +181 -181
  255. package/kit/framework/workflows/help.md +619 -619
  256. package/kit/framework/workflows/insert-phase.md +130 -130
  257. package/kit/framework/workflows/list-phase-assumptions.md +178 -178
  258. package/kit/framework/workflows/list-workspaces.md +56 -56
  259. package/kit/framework/workflows/manager.md +362 -362
  260. package/kit/framework/workflows/map-codebase.md +377 -377
  261. package/kit/framework/workflows/milestone-summary.md +223 -223
  262. package/kit/framework/workflows/new-milestone.md +486 -486
  263. package/kit/framework/workflows/new-project.md +1159 -1159
  264. package/kit/framework/workflows/new-workspace.md +237 -237
  265. package/kit/framework/workflows/next.md +97 -97
  266. package/kit/framework/workflows/node-repair.md +92 -92
  267. package/kit/framework/workflows/note.md +156 -156
  268. package/kit/framework/workflows/pause-work.md +176 -176
  269. package/kit/framework/workflows/plan-milestone-gaps.md +273 -273
  270. package/kit/framework/workflows/plan-phase.md +765 -765
  271. package/kit/framework/workflows/plant-seed.md +169 -169
  272. package/kit/framework/workflows/pr-branch.md +129 -129
  273. package/kit/framework/workflows/profile-user.md +450 -450
  274. package/kit/framework/workflows/progress.md +507 -507
  275. package/kit/framework/workflows/quick.md +757 -757
  276. package/kit/framework/workflows/remove-phase.md +155 -155
  277. package/kit/framework/workflows/remove-workspace.md +90 -90
  278. package/kit/framework/workflows/research-phase.md +82 -82
  279. package/kit/framework/workflows/resume-project.md +326 -326
  280. package/kit/framework/workflows/review.md +228 -228
  281. package/kit/framework/workflows/session-report.md +146 -146
  282. package/kit/framework/workflows/settings.md +283 -283
  283. package/kit/framework/workflows/ship.md +228 -228
  284. package/kit/framework/workflows/stats.md +60 -60
  285. package/kit/framework/workflows/transition.md +671 -671
  286. package/kit/framework/workflows/ui-phase.md +302 -302
  287. package/kit/framework/workflows/ui-review.md +165 -165
  288. package/kit/framework/workflows/update.md +323 -323
  289. package/kit/framework/workflows/validate-phase.md +174 -174
  290. package/kit/framework/workflows/verify-phase.md +252 -252
  291. package/kit/framework/workflows/verify-work.md +637 -637
  292. package/kit/hooks/check-update.js +118 -118
  293. package/kit/hooks/context-monitor.js +163 -163
  294. package/kit/hooks/kit-attribution-reminder.cjs +92 -92
  295. package/kit/hooks/kit-router.cjs +137 -137
  296. package/kit/hooks/prompt-guard.js +103 -103
  297. package/kit/hooks/statusline.js +125 -125
  298. package/kit/hooks/workflow-guard.js +101 -101
  299. package/kit/settings.json +45 -45
  300. package/kit/skills/ai-prompt-characterization/SKILL.md +335 -335
  301. package/kit/skills/armadilhas-sistemas-distribuidos/SKILL.md +447 -447
  302. package/kit/skills/audit-log-multi-tenant/SKILL.md +340 -340
  303. package/kit/skills/b2b-saas-architecture/SKILL.md +300 -300
  304. package/kit/skills/consistencia-leitura-replica/SKILL.md +385 -385
  305. package/kit/skills/crm-lead-pipeline-patterns/SKILL.md +343 -343
  306. package/kit/skills/dynamic-workflow-authoring/SKILL.md +223 -0
  307. package/kit/skills/escolha-modelo-consistencia/SKILL.md +494 -494
  308. package/kit/skills/evolucao-schema-compativel/SKILL.md +448 -448
  309. package/kit/skills/evolution-go-whatsapp-integration/SKILL.md +322 -322
  310. package/kit/skills/example-skill/SKILL.md +42 -42
  311. package/kit/skills/legacy-api-only-applications/SKILL.md +358 -358
  312. package/kit/skills/legacy-characterization-tests/SKILL.md +330 -330
  313. package/kit/skills/legacy-effect-analysis/SKILL.md +331 -331
  314. package/kit/skills/legacy-extract-class/SKILL.md +203 -203
  315. package/kit/skills/legacy-programming-by-difference/SKILL.md +252 -252
  316. package/kit/skills/legacy-seams-and-test-harness/SKILL.md +460 -460
  317. package/kit/skills/legacy-shotgun-surgery/SKILL.md +286 -286
  318. package/kit/skills/legacy-sprout-wrap-techniques/SKILL.md +434 -434
  319. package/kit/skills/legacy-storytelling-naked-crc/SKILL.md +270 -270
  320. package/kit/skills/lgpd-multi-tenant-compliance/SKILL.md +340 -340
  321. package/kit/skills/member-invite-flow/SKILL.md +305 -305
  322. package/kit/skills/member-management-react-shadcn/SKILL.md +328 -328
  323. package/kit/skills/multi-tenant-performance-scaling/SKILL.md +316 -316
  324. package/kit/skills/multi-tenant-rls-hierarchy/SKILL.md +342 -342
  325. package/kit/skills/org-onboarding-flow/SKILL.md +257 -257
  326. package/kit/skills/org-switcher-react-pattern/SKILL.md +349 -349
  327. package/kit/skills/permission-gate-react-pattern/SKILL.md +271 -271
  328. package/kit/skills/postgres-isolamento-concorrencia/SKILL.md +552 -552
  329. package/kit/skills/pre-refactor-characterization/SKILL.md +421 -421
  330. package/kit/skills/rbac-permissions-matrix-supabase/SKILL.md +338 -338
  331. package/kit/skills/streams-eventos-cdc/SKILL.md +711 -711
  332. package/kit/skills/supabase-auth-hardening/SKILL.md +674 -674
  333. package/kit/skills/supabase-auth-hooks/SKILL.md +875 -875
  334. package/kit/skills/supabase-auth-methods/SKILL.md +486 -486
  335. package/kit/skills/supabase-auth-sessions/SKILL.md +579 -579
  336. package/kit/skills/supabase-auth-ssr/SKILL.md +306 -306
  337. package/kit/skills/supabase-branching-workflow/SKILL.md +544 -544
  338. package/kit/skills/supabase-ci-cd-github-actions/SKILL.md +880 -880
  339. package/kit/skills/supabase-column-level-security/SKILL.md +426 -426
  340. package/kit/skills/supabase-config-toml-remotes/SKILL.md +807 -807
  341. package/kit/skills/supabase-custom-claims-rbac/SKILL.md +472 -472
  342. package/kit/skills/supabase-edge-functions/SKILL.md +330 -330
  343. package/kit/skills/supabase-edge-functions-auth/SKILL.md +309 -309
  344. package/kit/skills/supabase-edge-functions-limits/SKILL.md +302 -302
  345. package/kit/skills/supabase-edge-functions-mcp-server/SKILL.md +279 -279
  346. package/kit/skills/supabase-edge-functions-testing/SKILL.md +277 -277
  347. package/kit/skills/supabase-edge-runtime-builtins/SKILL.md +357 -357
  348. package/kit/skills/supabase-enterprise-sso-saml/SKILL.md +545 -545
  349. package/kit/skills/supabase-jwt-signing-keys/SKILL.md +399 -399
  350. package/kit/skills/supabase-mfa/SKILL.md +488 -488
  351. package/kit/skills/supabase-migration-repair/SKILL.md +823 -823
  352. package/kit/skills/supabase-migrations/SKILL.md +297 -297
  353. package/kit/skills/supabase-oauth-server/SKILL.md +537 -537
  354. package/kit/skills/supabase-pgtap-testing/SKILL.md +1053 -1053
  355. package/kit/skills/supabase-postgres-roles/SKILL.md +392 -392
  356. package/kit/skills/supabase-realtime/SKILL.md +460 -460
  357. package/kit/skills/supabase-rls-defense-in-depth/SKILL.md +418 -418
  358. package/kit/skills/supabase-rls-policies/SKILL.md +635 -635
  359. package/kit/skills/supabase-social-oauth/SKILL.md +480 -480
  360. package/kit/skills/supabase-third-party-auth/SKILL.md +450 -450
  361. package/kit/skills/super-admin-platform-pattern/SKILL.md +326 -326
  362. package/kit/skills/tenant-quente-mitigacao/SKILL.md +605 -605
  363. package/kit/skills/ui-anti-padroes-ia/SKILL.md +261 -261
  364. package/kit/skills/ui-contexto-produto/SKILL.md +248 -248
  365. package/kit/skills/ui-cor-estrategia/SKILL.md +213 -213
  366. package/kit/skills/ui-critica-auditoria/SKILL.md +260 -260
  367. package/kit/skills/ui-motion-funcional/SKILL.md +264 -264
  368. package/kit/skills/ui-ritmo-espacial/SKILL.md +259 -259
  369. package/kit/skills/ui-tipografia/SKILL.md +211 -211
  370. package/kit/skills/whatsapp-conversation-state-machine/SKILL.md +287 -287
  371. package/kit/workflows/auditar-observabilidade-cobertura.workflow.js +250 -0
  372. package/package.json +65 -63
  373. package/src/core/kit.js +333 -216
  374. package/src/core/reflect.js +247 -247
  375. package/src/core/registry.js +123 -112
  376. package/src/core/reverse-sync.js +448 -372
  377. package/src/core/sync.js +477 -437
  378. package/src/core/watch.js +121 -121
  379. package/src/mcp-server/index.js +794 -794
@@ -1,1031 +1,1031 @@
1
- /**
2
- * State — STATE.md operations and progression engine
3
- */
4
-
5
- const fs = require('fs');
6
- const path = require('path');
7
- const { escapeRegex, loadConfig, getMilestoneInfo, getMilestonePhaseFilter, normalizeMd, planningDir, planningPaths, output, error } = require('./core.cjs');
8
- const { extractFrontmatter, reconstructFrontmatter } = require('./frontmatter.cjs');
9
-
10
- /** Shorthand — every state command needs this path */
11
- function getStatePath(cwd) {
12
- return planningPaths(cwd).state;
13
- }
14
-
15
- // Shared helper: extract a field value from STATE.md content.
16
- // Supports both **Field:** bold and plain Field: format.
17
- function stateExtractField(content, fieldName) {
18
- const escaped = escapeRegex(fieldName);
19
- const boldPattern = new RegExp(`\\*\\*${escaped}:\\*\\*\\s*(.+)`, 'i');
20
- const boldMatch = content.match(boldPattern);
21
- if (boldMatch) return boldMatch[1].trim();
22
- const plainPattern = new RegExp(`^${escaped}:\\s*(.+)`, 'im');
23
- const plainMatch = content.match(plainPattern);
24
- return plainMatch ? plainMatch[1].trim() : null;
25
- }
26
-
27
- function cmdStateLoad(cwd, raw) {
28
- const config = loadConfig(cwd);
29
- const planDir = planningPaths(cwd).planning;
30
-
31
- let stateRaw = '';
32
- try {
33
- stateRaw = fs.readFileSync(path.join(planDir, 'STATE.md'), 'utf-8');
34
- } catch { /* intentionally empty */ }
35
-
36
- const configExists = fs.existsSync(path.join(planDir, 'config.json'));
37
- const roadmapExists = fs.existsSync(path.join(planDir, 'ROADMAP.md'));
38
- const stateExists = stateRaw.length > 0;
39
-
40
- const result = {
41
- config,
42
- state_raw: stateRaw,
43
- state_exists: stateExists,
44
- roadmap_exists: roadmapExists,
45
- config_exists: configExists,
46
- };
47
-
48
- // For --raw, output a condensed key=value format
49
- if (raw) {
50
- const c = config;
51
- const lines = [
52
- `model_profile=${c.model_profile}`,
53
- `commit_docs=${c.commit_docs}`,
54
- `branching_strategy=${c.branching_strategy}`,
55
- `phase_branch_template=${c.phase_branch_template}`,
56
- `milestone_branch_template=${c.milestone_branch_template}`,
57
- `parallelization=${c.parallelization}`,
58
- `research=${c.research}`,
59
- `plan_checker=${c.plan_checker}`,
60
- `verifier=${c.verifier}`,
61
- `config_exists=${configExists}`,
62
- `roadmap_exists=${roadmapExists}`,
63
- `state_exists=${stateExists}`,
64
- ];
65
- process.stdout.write(lines.join('\n'));
66
- process.exit(0);
67
- }
68
-
69
- output(result);
70
- }
71
-
72
- function cmdStateGet(cwd, section, raw) {
73
- const statePath = planningPaths(cwd).state;
74
- try {
75
- const content = fs.readFileSync(statePath, 'utf-8');
76
-
77
- if (!section) {
78
- output({ content }, raw, content);
79
- return;
80
- }
81
-
82
- // Try to find markdown section or field
83
- const fieldEscaped = escapeRegex(section);
84
-
85
- // Check for **field:** value (bold format)
86
- const boldPattern = new RegExp(`\\*\\*${fieldEscaped}:\\*\\*\\s*(.*)`, 'i');
87
- const boldMatch = content.match(boldPattern);
88
- if (boldMatch) {
89
- output({ [section]: boldMatch[1].trim() }, raw, boldMatch[1].trim());
90
- return;
91
- }
92
-
93
- // Check for field: value (plain format)
94
- const plainPattern = new RegExp(`^${fieldEscaped}:\\s*(.*)`, 'im');
95
- const plainMatch = content.match(plainPattern);
96
- if (plainMatch) {
97
- output({ [section]: plainMatch[1].trim() }, raw, plainMatch[1].trim());
98
- return;
99
- }
100
-
101
- // Check for ## Section
102
- const sectionPattern = new RegExp(`##\\s*${fieldEscaped}\\s*\n([\\s\\S]*?)(?=\\n##|$)`, 'i');
103
- const sectionMatch = content.match(sectionPattern);
104
- if (sectionMatch) {
105
- output({ [section]: sectionMatch[1].trim() }, raw, sectionMatch[1].trim());
106
- return;
107
- }
108
-
109
- output({ error: `Section or field "${section}" not found` }, raw, '');
110
- } catch {
111
- error('STATE.md not found');
112
- }
113
- }
114
-
115
- function readTextArgOrFile(cwd, value, filePath, label) {
116
- if (!filePath) return value;
117
-
118
- // Path traversal guard: ensure file resolves within project directory
119
- const { validatePath } = require('./security.cjs');
120
- const pathCheck = validatePath(filePath, cwd, { allowAbsolute: true });
121
- if (!pathCheck.safe) {
122
- throw new Error(`${label} path rejected: ${pathCheck.error}`);
123
- }
124
-
125
- try {
126
- return fs.readFileSync(pathCheck.resolved, 'utf-8').trimEnd();
127
- } catch {
128
- throw new Error(`${label} file not found: ${filePath}`);
129
- }
130
- }
131
-
132
- function cmdStatePatch(cwd, patches, raw) {
133
- // Validate all field names before processing
134
- const { validateFieldName } = require('./security.cjs');
135
- for (const field of Object.keys(patches)) {
136
- const fieldCheck = validateFieldName(field);
137
- if (!fieldCheck.valid) {
138
- error(`state patch: ${fieldCheck.error}`);
139
- }
140
- }
141
-
142
- const statePath = planningPaths(cwd).state;
143
- try {
144
- let content = fs.readFileSync(statePath, 'utf-8');
145
- const results = { updated: [], failed: [] };
146
-
147
- for (const [field, value] of Object.entries(patches)) {
148
- const fieldEscaped = escapeRegex(field);
149
- // Try **Field:** bold format first, then plain Field: format
150
- const boldPattern = new RegExp(`(\\*\\*${fieldEscaped}:\\*\\*\\s*)(.*)`, 'i');
151
- const plainPattern = new RegExp(`(^${fieldEscaped}:\\s*)(.*)`, 'im');
152
-
153
- if (boldPattern.test(content)) {
154
- content = content.replace(boldPattern, (_match, prefix) => `${prefix}${value}`);
155
- results.updated.push(field);
156
- } else if (plainPattern.test(content)) {
157
- content = content.replace(plainPattern, (_match, prefix) => `${prefix}${value}`);
158
- results.updated.push(field);
159
- } else {
160
- results.failed.push(field);
161
- }
162
- }
163
-
164
- if (results.updated.length > 0) {
165
- writeStateMd(statePath, content, cwd);
166
- }
167
-
168
- output(results, raw, results.updated.length > 0 ? 'true' : 'false');
169
- } catch {
170
- error('STATE.md not found');
171
- }
172
- }
173
-
174
- function cmdStateUpdate(cwd, field, value) {
175
- if (!field || value === undefined) {
176
- error('field and value required for state update');
177
- }
178
-
179
- // Validate field name to prevent regex injection via crafted field names
180
- const { validateFieldName } = require('./security.cjs');
181
- const fieldCheck = validateFieldName(field);
182
- if (!fieldCheck.valid) {
183
- error(`state update: ${fieldCheck.error}`);
184
- }
185
-
186
- const statePath = planningPaths(cwd).state;
187
- try {
188
- let content = fs.readFileSync(statePath, 'utf-8');
189
- const fieldEscaped = escapeRegex(field);
190
- // Try **Field:** bold format first, then plain Field: format
191
- const boldPattern = new RegExp(`(\\*\\*${fieldEscaped}:\\*\\*\\s*)(.*)`, 'i');
192
- const plainPattern = new RegExp(`(^${fieldEscaped}:\\s*)(.*)`, 'im');
193
- if (boldPattern.test(content)) {
194
- content = content.replace(boldPattern, (_match, prefix) => `${prefix}${value}`);
195
- writeStateMd(statePath, content, cwd);
196
- output({ updated: true });
197
- } else if (plainPattern.test(content)) {
198
- content = content.replace(plainPattern, (_match, prefix) => `${prefix}${value}`);
199
- writeStateMd(statePath, content, cwd);
200
- output({ updated: true });
201
- } else {
202
- output({ updated: false, reason: `Field "${field}" not found in STATE.md` });
203
- }
204
- } catch {
205
- output({ updated: false, reason: 'STATE.md not found' });
206
- }
207
- }
208
-
209
- // ─── State Progression Engine ────────────────────────────────────────────────
210
- // stateExtractField is defined above (shared helper) — do not duplicate.
211
-
212
- function stateReplaceField(content, fieldName, newValue) {
213
- const escaped = escapeRegex(fieldName);
214
- // Try **Field:** bold format first, then plain Field: format
215
- const boldPattern = new RegExp(`(\\*\\*${escaped}:\\*\\*\\s*)(.*)`, 'i');
216
- if (boldPattern.test(content)) {
217
- return content.replace(boldPattern, (_match, prefix) => `${prefix}${newValue}`);
218
- }
219
- const plainPattern = new RegExp(`(^${escaped}:\\s*)(.*)`, 'im');
220
- if (plainPattern.test(content)) {
221
- return content.replace(plainPattern, (_match, prefix) => `${prefix}${newValue}`);
222
- }
223
- return null;
224
- }
225
-
226
- /**
227
- * Replace a STATE.md field with fallback field name support.
228
- * Tries `primary` first, then `fallback` (if provided), returns content unchanged
229
- * if neither matches. This consolidates the replaceWithFallback pattern that was
230
- * previously duplicated inline across phase.cjs, milestone.cjs, and state.cjs.
231
- */
232
- function stateReplaceFieldWithFallback(content, primary, fallback, value) {
233
- let result = stateReplaceField(content, primary, value);
234
- if (result) return result;
235
- if (fallback) {
236
- result = stateReplaceField(content, fallback, value);
237
- if (result) return result;
238
- }
239
- return content;
240
- }
241
-
242
- /**
243
- * Update fields within the ## Current Position section of STATE.md.
244
- * This keeps the Current Position body in sync with the bold frontmatter fields.
245
- * Only updates fields that already exist in the section; does not add new lines.
246
- * Fixes #1365: advance-plan could not update Status/Last activity after begin-phase.
247
- */
248
- function updateCurrentPositionFields(content, fields) {
249
- const posPattern = /(##\s*Current Position\s*\n)([\s\S]*?)(?=\n##|$)/i;
250
- const posMatch = content.match(posPattern);
251
- if (!posMatch) return content;
252
-
253
- let posBody = posMatch[2];
254
-
255
- if (fields.status && /^Status:/m.test(posBody)) {
256
- posBody = posBody.replace(/^Status:.*$/m, `Status: ${fields.status}`);
257
- }
258
- if (fields.lastActivity && /^Last activity:/im.test(posBody)) {
259
- posBody = posBody.replace(/^Last activity:.*$/im, `Last activity: ${fields.lastActivity}`);
260
- }
261
- if (fields.plan && /^Plan:/m.test(posBody)) {
262
- posBody = posBody.replace(/^Plan:.*$/m, `Plan: ${fields.plan}`);
263
- }
264
-
265
- return content.replace(posPattern, `${posMatch[1]}${posBody}`);
266
- }
267
-
268
- function cmdStateAdvancePlan(cwd, raw) {
269
- const statePath = planningPaths(cwd).state;
270
- if (!fs.existsSync(statePath)) { output({ error: 'STATE.md not found' }, raw); return; }
271
-
272
- let content = fs.readFileSync(statePath, 'utf-8');
273
- const today = new Date().toISOString().split('T')[0];
274
-
275
- // Try legacy separate fields first, then compound "Plan: X of Y" format
276
- const legacyPlan = stateExtractField(content, 'Current Plan');
277
- const legacyTotal = stateExtractField(content, 'Total Plans in Phase');
278
- const planField = stateExtractField(content, 'Plan');
279
-
280
- let currentPlan, totalPlans;
281
- let useCompoundFormat = false;
282
-
283
- if (legacyPlan && legacyTotal) {
284
- currentPlan = parseInt(legacyPlan, 10);
285
- totalPlans = parseInt(legacyTotal, 10);
286
- } else if (planField) {
287
- // Compound format: "2 of 6 in current phase" or "2 of 6"
288
- currentPlan = parseInt(planField, 10);
289
- const ofMatch = planField.match(/of\s+(\d+)/);
290
- totalPlans = ofMatch ? parseInt(ofMatch[1], 10) : NaN;
291
- useCompoundFormat = true;
292
- }
293
-
294
- if (isNaN(currentPlan) || isNaN(totalPlans)) {
295
- output({ error: 'Cannot parse Current Plan or Total Plans in Phase from STATE.md' }, raw);
296
- return;
297
- }
298
-
299
- if (currentPlan >= totalPlans) {
300
- content = stateReplaceFieldWithFallback(content, 'Status', null, 'Phase complete — ready for verification');
301
- content = stateReplaceFieldWithFallback(content, 'Last Activity', 'Last activity', today);
302
- content = updateCurrentPositionFields(content, { status: 'Phase complete — ready for verification', lastActivity: today });
303
- writeStateMd(statePath, content, cwd);
304
- output({ advanced: false, reason: 'last_plan', current_plan: currentPlan, total_plans: totalPlans, status: 'ready_for_verification' }, raw, 'false');
305
- } else {
306
- const newPlan = currentPlan + 1;
307
- let planDisplayValue;
308
- if (useCompoundFormat) {
309
- // Preserve compound format: "X of Y in current phase" → replace X only
310
- planDisplayValue = planField.replace(/^\d+/, String(newPlan));
311
- content = stateReplaceField(content, 'Plan', planDisplayValue) || content;
312
- } else {
313
- planDisplayValue = `${newPlan} of ${totalPlans}`;
314
- content = stateReplaceField(content, 'Current Plan', String(newPlan)) || content;
315
- }
316
- content = stateReplaceFieldWithFallback(content, 'Status', null, 'Ready to execute');
317
- content = stateReplaceFieldWithFallback(content, 'Last Activity', 'Last activity', today);
318
- content = updateCurrentPositionFields(content, { status: 'Ready to execute', lastActivity: today, plan: planDisplayValue });
319
- writeStateMd(statePath, content, cwd);
320
- output({ advanced: true, previous_plan: currentPlan, current_plan: newPlan, total_plans: totalPlans }, raw, 'true');
321
- }
322
- }
323
-
324
- function cmdStateRecordMetric(cwd, options, raw) {
325
- const statePath = planningPaths(cwd).state;
326
- if (!fs.existsSync(statePath)) { output({ error: 'STATE.md not found' }, raw); return; }
327
-
328
- let content = fs.readFileSync(statePath, 'utf-8');
329
- const { phase, plan, duration, tasks, files } = options;
330
-
331
- if (!phase || !plan || !duration) {
332
- output({ error: 'phase, plan, and duration required' }, raw);
333
- return;
334
- }
335
-
336
- // Find Performance Metrics section and its table
337
- const metricsPattern = /(##\s*Performance Metrics[\s\S]*?\n\|[^\n]+\n\|[-|\s]+\n)([\s\S]*?)(?=\n##|\n$|$)/i;
338
- const metricsMatch = content.match(metricsPattern);
339
-
340
- if (metricsMatch) {
341
- let tableBody = metricsMatch[2].trimEnd();
342
- const newRow = `| Phase ${phase} P${plan} | ${duration} | ${tasks || '-'} tasks | ${files || '-'} files |`;
343
-
344
- if (tableBody.trim() === '' || tableBody.includes('None yet')) {
345
- tableBody = newRow;
346
- } else {
347
- tableBody = tableBody + '\n' + newRow;
348
- }
349
-
350
- content = content.replace(metricsPattern, (_match, header) => `${header}${tableBody}\n`);
351
- writeStateMd(statePath, content, cwd);
352
- output({ recorded: true, phase, plan, duration }, raw, 'true');
353
- } else {
354
- output({ recorded: false, reason: 'Performance Metrics section not found in STATE.md' }, raw, 'false');
355
- }
356
- }
357
-
358
- function cmdStateUpdateProgress(cwd, raw) {
359
- const statePath = planningPaths(cwd).state;
360
- if (!fs.existsSync(statePath)) { output({ error: 'STATE.md not found' }, raw); return; }
361
-
362
- let content = fs.readFileSync(statePath, 'utf-8');
363
-
364
- // Count summaries across current milestone phases only
365
- const phasesDir = planningPaths(cwd).phases;
366
- let totalPlans = 0;
367
- let totalSummaries = 0;
368
-
369
- if (fs.existsSync(phasesDir)) {
370
- const isDirInMilestone = getMilestonePhaseFilter(cwd);
371
- const phaseDirs = fs.readdirSync(phasesDir, { withFileTypes: true })
372
- .filter(e => e.isDirectory()).map(e => e.name)
373
- .filter(isDirInMilestone);
374
- for (const dir of phaseDirs) {
375
- const files = fs.readdirSync(path.join(phasesDir, dir));
376
- totalPlans += files.filter(f => f.match(/-PLAN\.md$/i)).length;
377
- totalSummaries += files.filter(f => f.match(/-SUMMARY\.md$/i)).length;
378
- }
379
- }
380
-
381
- const percent = totalPlans > 0 ? Math.min(100, Math.round(totalSummaries / totalPlans * 100)) : 0;
382
- const barWidth = 10;
383
- const filled = Math.round(percent / 100 * barWidth);
384
- const bar = '\u2588'.repeat(filled) + '\u2591'.repeat(barWidth - filled);
385
- const progressStr = `[${bar}] ${percent}%`;
386
-
387
- // Try **Progress:** bold format first, then plain Progress: format
388
- const boldProgressPattern = /(\*\*Progress:\*\*\s*).*/i;
389
- const plainProgressPattern = /^(Progress:\s*).*/im;
390
- if (boldProgressPattern.test(content)) {
391
- content = content.replace(boldProgressPattern, (_match, prefix) => `${prefix}${progressStr}`);
392
- writeStateMd(statePath, content, cwd);
393
- output({ updated: true, percent, completed: totalSummaries, total: totalPlans, bar: progressStr }, raw, progressStr);
394
- } else if (plainProgressPattern.test(content)) {
395
- content = content.replace(plainProgressPattern, (_match, prefix) => `${prefix}${progressStr}`);
396
- writeStateMd(statePath, content, cwd);
397
- output({ updated: true, percent, completed: totalSummaries, total: totalPlans, bar: progressStr }, raw, progressStr);
398
- } else {
399
- output({ updated: false, reason: 'Progress field not found in STATE.md' }, raw, 'false');
400
- }
401
- }
402
-
403
- function cmdStateAddDecision(cwd, options, raw) {
404
- const statePath = planningPaths(cwd).state;
405
- if (!fs.existsSync(statePath)) { output({ error: 'STATE.md not found' }, raw); return; }
406
-
407
- const { phase, summary, summary_file, rationale, rationale_file } = options;
408
- let summaryText = null;
409
- let rationaleText = '';
410
-
411
- try {
412
- summaryText = readTextArgOrFile(cwd, summary, summary_file, 'summary');
413
- rationaleText = readTextArgOrFile(cwd, rationale || '', rationale_file, 'rationale');
414
- } catch (err) {
415
- output({ added: false, reason: err.message }, raw, 'false');
416
- return;
417
- }
418
-
419
- if (!summaryText) { output({ error: 'summary required' }, raw); return; }
420
-
421
- let content = fs.readFileSync(statePath, 'utf-8');
422
- const entry = `- [Phase ${phase || '?'}]: ${summaryText}${rationaleText ? ` — ${rationaleText}` : ''}`;
423
-
424
- // Find Decisions section (various heading patterns)
425
- const sectionPattern = /(###?\s*(?:Decisions|Decisions Made|Accumulated.*Decisions)\s*\n)([\s\S]*?)(?=\n###?|\n##[^#]|$)/i;
426
- const match = content.match(sectionPattern);
427
-
428
- if (match) {
429
- let sectionBody = match[2];
430
- // Remove placeholders
431
- sectionBody = sectionBody.replace(/None yet\.?\s*\n?/gi, '').replace(/No decisions yet\.?\s*\n?/gi, '');
432
- sectionBody = sectionBody.trimEnd() + '\n' + entry + '\n';
433
- content = content.replace(sectionPattern, (_match, header) => `${header}${sectionBody}`);
434
- writeStateMd(statePath, content, cwd);
435
- output({ added: true, decision: entry }, raw, 'true');
436
- } else {
437
- output({ added: false, reason: 'Decisions section not found in STATE.md' }, raw, 'false');
438
- }
439
- }
440
-
441
- function cmdStateAddBlocker(cwd, text, raw) {
442
- const statePath = planningPaths(cwd).state;
443
- if (!fs.existsSync(statePath)) { output({ error: 'STATE.md not found' }, raw); return; }
444
- const blockerOptions = typeof text === 'object' && text !== null ? text : { text };
445
- let blockerText = null;
446
-
447
- try {
448
- blockerText = readTextArgOrFile(cwd, blockerOptions.text, blockerOptions.text_file, 'blocker');
449
- } catch (err) {
450
- output({ added: false, reason: err.message }, raw, 'false');
451
- return;
452
- }
453
-
454
- if (!blockerText) { output({ error: 'text required' }, raw); return; }
455
-
456
- let content = fs.readFileSync(statePath, 'utf-8');
457
- const entry = `- ${blockerText}`;
458
-
459
- const sectionPattern = /(###?\s*(?:Blockers|Blockers\/Concerns|Concerns)\s*\n)([\s\S]*?)(?=\n###?|\n##[^#]|$)/i;
460
- const match = content.match(sectionPattern);
461
-
462
- if (match) {
463
- let sectionBody = match[2];
464
- sectionBody = sectionBody.replace(/None\.?\s*\n?/gi, '').replace(/None yet\.?\s*\n?/gi, '');
465
- sectionBody = sectionBody.trimEnd() + '\n' + entry + '\n';
466
- content = content.replace(sectionPattern, (_match, header) => `${header}${sectionBody}`);
467
- writeStateMd(statePath, content, cwd);
468
- output({ added: true, blocker: blockerText }, raw, 'true');
469
- } else {
470
- output({ added: false, reason: 'Blockers section not found in STATE.md' }, raw, 'false');
471
- }
472
- }
473
-
474
- function cmdStateResolveBlocker(cwd, text, raw) {
475
- const statePath = planningPaths(cwd).state;
476
- if (!fs.existsSync(statePath)) { output({ error: 'STATE.md not found' }, raw); return; }
477
- if (!text) { output({ error: 'text required' }, raw); return; }
478
-
479
- let content = fs.readFileSync(statePath, 'utf-8');
480
-
481
- const sectionPattern = /(###?\s*(?:Blockers|Blockers\/Concerns|Concerns)\s*\n)([\s\S]*?)(?=\n###?|\n##[^#]|$)/i;
482
- const match = content.match(sectionPattern);
483
-
484
- if (match) {
485
- const sectionBody = match[2];
486
- const lines = sectionBody.split('\n');
487
- const filtered = lines.filter(line => {
488
- if (!line.startsWith('- ')) return true;
489
- return !line.toLowerCase().includes(text.toLowerCase());
490
- });
491
-
492
- let newBody = filtered.join('\n');
493
- // If section is now empty, add placeholder
494
- if (!newBody.trim() || !newBody.includes('- ')) {
495
- newBody = 'None\n';
496
- }
497
-
498
- content = content.replace(sectionPattern, (_match, header) => `${header}${newBody}`);
499
- writeStateMd(statePath, content, cwd);
500
- output({ resolved: true, blocker: text }, raw, 'true');
501
- } else {
502
- output({ resolved: false, reason: 'Blockers section not found in STATE.md' }, raw, 'false');
503
- }
504
- }
505
-
506
- function cmdStateRecordSession(cwd, options, raw) {
507
- const statePath = planningPaths(cwd).state;
508
- if (!fs.existsSync(statePath)) { output({ error: 'STATE.md not found' }, raw); return; }
509
-
510
- let content = fs.readFileSync(statePath, 'utf-8');
511
- const now = new Date().toISOString();
512
- const updated = [];
513
-
514
- // Update Last session / Last Date
515
- let result = stateReplaceField(content, 'Last session', now);
516
- if (result) { content = result; updated.push('Last session'); }
517
- result = stateReplaceField(content, 'Last Date', now);
518
- if (result) { content = result; updated.push('Last Date'); }
519
-
520
- // Update Stopped at
521
- if (options.stopped_at) {
522
- result = stateReplaceField(content, 'Stopped At', options.stopped_at);
523
- if (!result) result = stateReplaceField(content, 'Stopped at', options.stopped_at);
524
- if (result) { content = result; updated.push('Stopped At'); }
525
- }
526
-
527
- // Update Resume file
528
- const resumeFile = options.resume_file || 'None';
529
- result = stateReplaceField(content, 'Resume File', resumeFile);
530
- if (!result) result = stateReplaceField(content, 'Resume file', resumeFile);
531
- if (result) { content = result; updated.push('Resume File'); }
532
-
533
- if (updated.length > 0) {
534
- writeStateMd(statePath, content, cwd);
535
- output({ recorded: true, updated }, raw, 'true');
536
- } else {
537
- output({ recorded: false, reason: 'No session fields found in STATE.md' }, raw, 'false');
538
- }
539
- }
540
-
541
- function cmdStateSnapshot(cwd, raw) {
542
- const statePath = planningPaths(cwd).state;
543
-
544
- if (!fs.existsSync(statePath)) {
545
- output({ error: 'STATE.md not found' }, raw);
546
- return;
547
- }
548
-
549
- const content = fs.readFileSync(statePath, 'utf-8');
550
-
551
- // Extract basic fields
552
- const currentPhase = stateExtractField(content, 'Current Phase');
553
- const currentPhaseName = stateExtractField(content, 'Current Phase Name');
554
- const totalPhasesRaw = stateExtractField(content, 'Total Phases');
555
- const currentPlan = stateExtractField(content, 'Current Plan');
556
- const totalPlansRaw = stateExtractField(content, 'Total Plans in Phase');
557
- const status = stateExtractField(content, 'Status');
558
- const progressRaw = stateExtractField(content, 'Progress');
559
- const lastActivity = stateExtractField(content, 'Last Activity');
560
- const lastActivityDesc = stateExtractField(content, 'Last Activity Description');
561
- const pausedAt = stateExtractField(content, 'Paused At');
562
-
563
- // Parse numeric fields
564
- const totalPhases = totalPhasesRaw ? parseInt(totalPhasesRaw, 10) : null;
565
- const totalPlansInPhase = totalPlansRaw ? parseInt(totalPlansRaw, 10) : null;
566
- const progressPercent = progressRaw ? parseInt(progressRaw.replace('%', ''), 10) : null;
567
-
568
- // Extract decisions table
569
- const decisions = [];
570
- const decisionsMatch = content.match(/##\s*Decisions Made[\s\S]*?\n\|[^\n]+\n\|[-|\s]+\n([\s\S]*?)(?=\n##|\n$|$)/i);
571
- if (decisionsMatch) {
572
- const tableBody = decisionsMatch[1];
573
- const rows = tableBody.trim().split('\n').filter(r => r.includes('|'));
574
- for (const row of rows) {
575
- const cells = row.split('|').map(c => c.trim()).filter(Boolean);
576
- if (cells.length >= 3) {
577
- decisions.push({
578
- phase: cells[0],
579
- summary: cells[1],
580
- rationale: cells[2],
581
- });
582
- }
583
- }
584
- }
585
-
586
- // Extract blockers list
587
- const blockers = [];
588
- const blockersMatch = content.match(/##\s*Blockers\s*\n([\s\S]*?)(?=\n##|$)/i);
589
- if (blockersMatch) {
590
- const blockersSection = blockersMatch[1];
591
- const items = blockersSection.match(/^-\s+(.+)$/gm) || [];
592
- for (const item of items) {
593
- blockers.push(item.replace(/^-\s+/, '').trim());
594
- }
595
- }
596
-
597
- // Extract session info
598
- const session = {
599
- last_date: null,
600
- stopped_at: null,
601
- resume_file: null,
602
- };
603
-
604
- const sessionMatch = content.match(/##\s*Session\s*\n([\s\S]*?)(?=\n##|$)/i);
605
- if (sessionMatch) {
606
- const sessionSection = sessionMatch[1];
607
- const lastDateMatch = sessionSection.match(/\*\*Last Date:\*\*\s*(.+)/i)
608
- || sessionSection.match(/^Last Date:\s*(.+)/im);
609
- const stoppedAtMatch = sessionSection.match(/\*\*Stopped At:\*\*\s*(.+)/i)
610
- || sessionSection.match(/^Stopped At:\s*(.+)/im);
611
- const resumeFileMatch = sessionSection.match(/\*\*Resume File:\*\*\s*(.+)/i)
612
- || sessionSection.match(/^Resume File:\s*(.+)/im);
613
-
614
- if (lastDateMatch) session.last_date = lastDateMatch[1].trim();
615
- if (stoppedAtMatch) session.stopped_at = stoppedAtMatch[1].trim();
616
- if (resumeFileMatch) session.resume_file = resumeFileMatch[1].trim();
617
- }
618
-
619
- const result = {
620
- current_phase: currentPhase,
621
- current_phase_name: currentPhaseName,
622
- total_phases: totalPhases,
623
- current_plan: currentPlan,
624
- total_plans_in_phase: totalPlansInPhase,
625
- status,
626
- progress_percent: progressPercent,
627
- last_activity: lastActivity,
628
- last_activity_desc: lastActivityDesc,
629
- decisions,
630
- blockers,
631
- paused_at: pausedAt,
632
- session,
633
- };
634
-
635
- output(result, raw);
636
- }
637
-
638
- // ─── State Frontmatter Sync ──────────────────────────────────────────────────
639
-
640
- /**
641
- * Extract machine-readable fields from STATE.md markdown body and build
642
- * a YAML frontmatter object. Allows hooks and scripts to read state
643
- * reliably via `state json` instead of fragile regex parsing.
644
- */
645
- function buildStateFrontmatter(bodyContent, cwd) {
646
- const currentPhase = stateExtractField(bodyContent, 'Current Phase');
647
- const currentPhaseName = stateExtractField(bodyContent, 'Current Phase Name');
648
- const currentPlan = stateExtractField(bodyContent, 'Current Plan');
649
- const totalPhasesRaw = stateExtractField(bodyContent, 'Total Phases');
650
- const totalPlansRaw = stateExtractField(bodyContent, 'Total Plans in Phase');
651
- const status = stateExtractField(bodyContent, 'Status');
652
- const progressRaw = stateExtractField(bodyContent, 'Progress');
653
- const lastActivity = stateExtractField(bodyContent, 'Last Activity');
654
- const stoppedAt = stateExtractField(bodyContent, 'Stopped At') || stateExtractField(bodyContent, 'Stopped at');
655
- const pausedAt = stateExtractField(bodyContent, 'Paused At');
656
-
657
- let milestone = null;
658
- let milestoneName = null;
659
- if (cwd) {
660
- try {
661
- const info = getMilestoneInfo(cwd);
662
- milestone = info.version;
663
- milestoneName = info.name;
664
- } catch { /* intentionally empty */ }
665
- }
666
-
667
- let totalPhases = totalPhasesRaw ? parseInt(totalPhasesRaw, 10) : null;
668
- let completedPhases = null;
669
- let totalPlans = totalPlansRaw ? parseInt(totalPlansRaw, 10) : null;
670
- let completedPlans = null;
671
-
672
- if (cwd) {
673
- try {
674
- const phasesDir = planningPaths(cwd).phases;
675
- if (fs.existsSync(phasesDir)) {
676
- const isDirInMilestone = getMilestonePhaseFilter(cwd);
677
- const phaseDirs = fs.readdirSync(phasesDir, { withFileTypes: true })
678
- .filter(e => e.isDirectory()).map(e => e.name)
679
- .filter(isDirInMilestone);
680
- let diskTotalPlans = 0;
681
- let diskTotalSummaries = 0;
682
- let diskCompletedPhases = 0;
683
-
684
- for (const dir of phaseDirs) {
685
- const files = fs.readdirSync(path.join(phasesDir, dir));
686
- const plans = files.filter(f => f.match(/-PLAN\.md$/i)).length;
687
- const summaries = files.filter(f => f.match(/-SUMMARY\.md$/i)).length;
688
- diskTotalPlans += plans;
689
- diskTotalSummaries += summaries;
690
- if (plans > 0 && summaries >= plans) diskCompletedPhases++;
691
- }
692
- totalPhases = isDirInMilestone.phaseCount > 0
693
- ? Math.max(phaseDirs.length, isDirInMilestone.phaseCount)
694
- : phaseDirs.length;
695
- completedPhases = diskCompletedPhases;
696
- totalPlans = diskTotalPlans;
697
- completedPlans = diskTotalSummaries;
698
- }
699
- } catch { /* intentionally empty */ }
700
- }
701
-
702
- let progressPercent = null;
703
- if (progressRaw) {
704
- const pctMatch = progressRaw.match(/(\d+)%/);
705
- if (pctMatch) progressPercent = parseInt(pctMatch[1], 10);
706
- }
707
-
708
- // Normalize status to one of: planning, discussing, executing, verifying, paused, completed, unknown
709
- let normalizedStatus = status || 'unknown';
710
- const statusLower = (status || '').toLowerCase();
711
- if (statusLower.includes('paused') || statusLower.includes('stopped') || pausedAt) {
712
- normalizedStatus = 'paused';
713
- } else if (statusLower.includes('executing') || statusLower.includes('in progress')) {
714
- normalizedStatus = 'executing';
715
- } else if (statusLower.includes('planning') || statusLower.includes('ready to plan')) {
716
- normalizedStatus = 'planning';
717
- } else if (statusLower.includes('discussing')) {
718
- normalizedStatus = 'discussing';
719
- } else if (statusLower.includes('verif')) {
720
- normalizedStatus = 'verifying';
721
- } else if (statusLower.includes('complete') || statusLower.includes('done')) {
722
- normalizedStatus = 'completed';
723
- } else if (statusLower.includes('ready to execute')) {
724
- normalizedStatus = 'executing';
725
- }
726
-
727
- const fm = { state_version: '1.0' };
728
-
729
- if (milestone) fm.milestone = milestone;
730
- if (milestoneName) fm.milestone_name = milestoneName;
731
- if (currentPhase) fm.current_phase = currentPhase;
732
- if (currentPhaseName) fm.current_phase_name = currentPhaseName;
733
- if (currentPlan) fm.current_plan = currentPlan;
734
- fm.status = normalizedStatus;
735
- if (stoppedAt) fm.stopped_at = stoppedAt;
736
- if (pausedAt) fm.paused_at = pausedAt;
737
- fm.last_updated = new Date().toISOString();
738
- if (lastActivity) fm.last_activity = lastActivity;
739
-
740
- const progress = {};
741
- if (totalPhases !== null) progress.total_phases = totalPhases;
742
- if (completedPhases !== null) progress.completed_phases = completedPhases;
743
- if (totalPlans !== null) progress.total_plans = totalPlans;
744
- if (completedPlans !== null) progress.completed_plans = completedPlans;
745
- if (progressPercent !== null) progress.percent = progressPercent;
746
- if (Object.keys(progress).length > 0) fm.progress = progress;
747
-
748
- return fm;
749
- }
750
-
751
- function stripFrontmatter(content) {
752
- // Strip ALL frontmatter blocks at the start of the file.
753
- // Handles CRLF line endings and multiple stacked blocks (corruption recovery).
754
- // Greedy: keeps stripping ---...--- blocks separated by optional whitespace.
755
- let result = content;
756
- // eslint-disable-next-line no-constant-condition
757
- while (true) {
758
- const stripped = result.replace(/^\s*---\r?\n[\s\S]*?\r?\n---\s*/, '');
759
- if (stripped === result) break;
760
- result = stripped;
761
- }
762
- return result;
763
- }
764
-
765
- function syncStateFrontmatter(content, cwd) {
766
- // Read existing frontmatter BEFORE stripping — it may contain values
767
- // that the body no longer has (e.g., Status field removed by an agent).
768
- const existingFm = extractFrontmatter(content);
769
- const body = stripFrontmatter(content);
770
- const derivedFm = buildStateFrontmatter(body, cwd);
771
-
772
- // Preserve existing frontmatter status when body-derived status is 'unknown'.
773
- // This prevents a missing Status: field in the body from overwriting a
774
- // previously valid status (e.g., 'executing' → 'unknown').
775
- if (derivedFm.status === 'unknown' && existingFm.status && existingFm.status !== 'unknown') {
776
- derivedFm.status = existingFm.status;
777
- }
778
-
779
- const yamlStr = reconstructFrontmatter(derivedFm);
780
- return `---\n${yamlStr}\n---\n\n${body}`;
781
- }
782
-
783
- /**
784
- * Write STATE.md with synchronized YAML frontmatter.
785
- * All STATE.md writes should use this instead of raw writeFileSync.
786
- * Uses a simple lockfile to prevent parallel agents from overwriting
787
- * each other's changes (race condition with read-modify-write cycle).
788
- */
789
- function writeStateMd(statePath, content, cwd) {
790
- const synced = syncStateFrontmatter(content, cwd);
791
- const lockPath = statePath + '.lock';
792
- const maxRetries = 10;
793
- const retryDelay = 200; // ms
794
-
795
- // Acquire lock (spin with backoff)
796
- for (let i = 0; i < maxRetries; i++) {
797
- try {
798
- // O_EXCL fails if file already exists — atomic lock
799
- const fd = fs.openSync(lockPath, fs.constants.O_CREAT | fs.constants.O_EXCL | fs.constants.O_WRONLY);
800
- fs.writeSync(fd, String(process.pid));
801
- fs.closeSync(fd);
802
- break;
803
- } catch (err) {
804
- if (err.code === 'EEXIST') {
805
- // Check for stale lock (> 10s old)
806
- try {
807
- const stat = fs.statSync(lockPath);
808
- if (Date.now() - stat.mtimeMs > 10000) {
809
- fs.unlinkSync(lockPath);
810
- continue; // retry immediately after clearing stale lock
811
- }
812
- } catch { /* lock was released between check — retry */ }
813
-
814
- if (i === maxRetries - 1) {
815
- // Last resort: write anyway rather than losing data
816
- try { fs.unlinkSync(lockPath); } catch {}
817
- break;
818
- }
819
- // Spin-wait with small jitter
820
- const jitter = Math.floor(Math.random() * 50);
821
- const start = Date.now();
822
- while (Date.now() - start < retryDelay + jitter) { /* busy wait */ }
823
- continue;
824
- }
825
- break; // non-EEXIST error — proceed without lock
826
- }
827
- }
828
-
829
- try {
830
- fs.writeFileSync(statePath, normalizeMd(synced), 'utf-8');
831
- } finally {
832
- try { fs.unlinkSync(lockPath); } catch { /* lock already gone */ }
833
- }
834
- }
835
-
836
- function cmdStateJson(cwd, raw) {
837
- const statePath = planningPaths(cwd).state;
838
- if (!fs.existsSync(statePath)) {
839
- output({ error: 'STATE.md not found' }, raw, 'STATE.md not found');
840
- return;
841
- }
842
-
843
- const content = fs.readFileSync(statePath, 'utf-8');
844
- const fm = extractFrontmatter(content);
845
-
846
- if (!fm || Object.keys(fm).length === 0) {
847
- const body = stripFrontmatter(content);
848
- const built = buildStateFrontmatter(body, cwd);
849
- output(built, raw, JSON.stringify(built, null, 2));
850
- return;
851
- }
852
-
853
- output(fm, raw, JSON.stringify(fm, null, 2));
854
- }
855
-
856
- /**
857
- * Update STATE.md when a new phase begins execution.
858
- * Updates body text fields (Current focus, Status, Last Activity, Current Position)
859
- * and synchronizes frontmatter via writeStateMd.
860
- * Fixes: #1102 (plan counts), #1103 (status/last_activity), #1104 (body text).
861
- */
862
- function cmdStateBeginPhase(cwd, phaseNumber, phaseName, planCount, raw) {
863
- const statePath = planningPaths(cwd).state;
864
- if (!fs.existsSync(statePath)) {
865
- output({ error: 'STATE.md not found' }, raw);
866
- return;
867
- }
868
-
869
- let content = fs.readFileSync(statePath, 'utf-8');
870
- const today = new Date().toISOString().split('T')[0];
871
- const updated = [];
872
-
873
- // Update Status field
874
- const statusValue = `Executing Phase ${phaseNumber}`;
875
- let result = stateReplaceField(content, 'Status', statusValue);
876
- if (result) { content = result; updated.push('Status'); }
877
-
878
- // Update Last Activity
879
- result = stateReplaceField(content, 'Last Activity', today);
880
- if (result) { content = result; updated.push('Last Activity'); }
881
-
882
- // Update Last Activity Description if it exists
883
- const activityDesc = `Phase ${phaseNumber} execution started`;
884
- result = stateReplaceField(content, 'Last Activity Description', activityDesc);
885
- if (result) { content = result; updated.push('Last Activity Description'); }
886
-
887
- // Update Current Phase
888
- result = stateReplaceField(content, 'Current Phase', String(phaseNumber));
889
- if (result) { content = result; updated.push('Current Phase'); }
890
-
891
- // Update Current Phase Name
892
- if (phaseName) {
893
- result = stateReplaceField(content, 'Current Phase Name', phaseName);
894
- if (result) { content = result; updated.push('Current Phase Name'); }
895
- }
896
-
897
- // Update Current Plan to 1 (starting from the first plan)
898
- result = stateReplaceField(content, 'Current Plan', '1');
899
- if (result) { content = result; updated.push('Current Plan'); }
900
-
901
- // Update Total Plans in Phase
902
- if (planCount) {
903
- result = stateReplaceField(content, 'Total Plans in Phase', String(planCount));
904
- if (result) { content = result; updated.push('Total Plans in Phase'); }
905
- }
906
-
907
- // Update **Current focus:** body text line (#1104)
908
- const focusLabel = phaseName ? `Phase ${phaseNumber} — ${phaseName}` : `Phase ${phaseNumber}`;
909
- const focusPattern = /(\*\*Current focus:\*\*\s*).*/i;
910
- if (focusPattern.test(content)) {
911
- content = content.replace(focusPattern, (_match, prefix) => `${prefix}${focusLabel}`);
912
- updated.push('Current focus');
913
- }
914
-
915
- // Update ## Current Position section (#1104, #1365)
916
- // Update individual fields within Current Position instead of replacing the
917
- // entire section, so that Status, Last activity, and Progress are preserved.
918
- const positionPattern = /(##\s*Current Position\s*\n)([\s\S]*?)(?=\n##|$)/i;
919
- const positionMatch = content.match(positionPattern);
920
- if (positionMatch) {
921
- const header = positionMatch[1];
922
- let posBody = positionMatch[2];
923
-
924
- // Update or insert Phase line
925
- const newPhase = `Phase: ${phaseNumber}${phaseName ? ` (${phaseName})` : ''} — EXECUTING`;
926
- if (/^Phase:/m.test(posBody)) {
927
- posBody = posBody.replace(/^Phase:.*$/m, newPhase);
928
- } else {
929
- posBody = newPhase + '\n' + posBody;
930
- }
931
-
932
- // Update or insert Plan line
933
- const newPlan = `Plan: 1 of ${planCount || '?'}`;
934
- if (/^Plan:/m.test(posBody)) {
935
- posBody = posBody.replace(/^Plan:.*$/m, newPlan);
936
- } else {
937
- posBody = posBody.replace(/^(Phase:.*$)/m, `$1\n${newPlan}`);
938
- }
939
-
940
- // Update Status line if present
941
- const newStatus = `Status: Executing Phase ${phaseNumber}`;
942
- if (/^Status:/m.test(posBody)) {
943
- posBody = posBody.replace(/^Status:.*$/m, newStatus);
944
- }
945
-
946
- // Update Last activity line if present
947
- const newActivity = `Last activity: ${today} -- Phase ${phaseNumber} execution started`;
948
- if (/^Last activity:/im.test(posBody)) {
949
- posBody = posBody.replace(/^Last activity:.*$/im, newActivity);
950
- }
951
-
952
- content = content.replace(positionPattern, `${header}${posBody}`);
953
- updated.push('Current Position');
954
- }
955
-
956
- if (updated.length > 0) {
957
- writeStateMd(statePath, content, cwd);
958
- }
959
-
960
- output({ updated, phase: phaseNumber, phase_name: phaseName || null, plan_count: planCount || null }, raw, updated.length > 0 ? 'true' : 'false');
961
- }
962
-
963
- /**
964
- * Write a WAITING.json signal file when framework hits a decision point.
965
- * External watchers (fswatch, polling, orchestrators) can detect this.
966
- * File is written to .planning/WAITING.json (or .framework/WAITING.json if .framework exists).
967
- * Fixes #1034.
968
- */
969
- function cmdSignalWaiting(cwd, type, question, options, phase, raw) {
970
- const frameworkDir = fs.existsSync(path.join(cwd, '.framework')) ? path.join(cwd, '.framework') : planningDir(cwd);
971
- const waitingPath = path.join(frameworkDir, 'WAITING.json');
972
-
973
- const signal = {
974
- status: 'waiting',
975
- type: type || 'decision_point',
976
- question: question || null,
977
- options: options ? options.split('|').map(o => o.trim()) : [],
978
- since: new Date().toISOString(),
979
- phase: phase || null,
980
- };
981
-
982
- try {
983
- fs.mkdirSync(frameworkDir, { recursive: true });
984
- fs.writeFileSync(waitingPath, JSON.stringify(signal, null, 2), 'utf-8');
985
- output({ signaled: true, path: waitingPath }, raw, 'true');
986
- } catch (e) {
987
- output({ signaled: false, error: e.message }, raw, 'false');
988
- }
989
- }
990
-
991
- /**
992
- * Remove the WAITING.json signal file when user answers and agent resumes.
993
- */
994
- function cmdSignalResume(cwd, raw) {
995
- const paths = [
996
- path.join(cwd, '.framework', 'WAITING.json'),
997
- path.join(planningDir(cwd), 'WAITING.json'),
998
- ];
999
-
1000
- let removed = false;
1001
- for (const p of paths) {
1002
- if (fs.existsSync(p)) {
1003
- try { fs.unlinkSync(p); removed = true; } catch {}
1004
- }
1005
- }
1006
-
1007
- output({ resumed: true, removed }, raw, removed ? 'true' : 'false');
1008
- }
1009
-
1010
- module.exports = {
1011
- stateExtractField,
1012
- stateReplaceField,
1013
- stateReplaceFieldWithFallback,
1014
- writeStateMd,
1015
- cmdStateLoad,
1016
- cmdStateGet,
1017
- cmdStatePatch,
1018
- cmdStateUpdate,
1019
- cmdStateAdvancePlan,
1020
- cmdStateRecordMetric,
1021
- cmdStateUpdateProgress,
1022
- cmdStateAddDecision,
1023
- cmdStateAddBlocker,
1024
- cmdStateResolveBlocker,
1025
- cmdStateRecordSession,
1026
- cmdStateSnapshot,
1027
- cmdStateJson,
1028
- cmdStateBeginPhase,
1029
- cmdSignalWaiting,
1030
- cmdSignalResume,
1031
- };
1
+ /**
2
+ * State — STATE.md operations and progression engine
3
+ */
4
+
5
+ const fs = require('fs');
6
+ const path = require('path');
7
+ const { escapeRegex, loadConfig, getMilestoneInfo, getMilestonePhaseFilter, normalizeMd, planningDir, planningPaths, output, error } = require('./core.cjs');
8
+ const { extractFrontmatter, reconstructFrontmatter } = require('./frontmatter.cjs');
9
+
10
+ /** Shorthand — every state command needs this path */
11
+ function getStatePath(cwd) {
12
+ return planningPaths(cwd).state;
13
+ }
14
+
15
+ // Shared helper: extract a field value from STATE.md content.
16
+ // Supports both **Field:** bold and plain Field: format.
17
+ function stateExtractField(content, fieldName) {
18
+ const escaped = escapeRegex(fieldName);
19
+ const boldPattern = new RegExp(`\\*\\*${escaped}:\\*\\*\\s*(.+)`, 'i');
20
+ const boldMatch = content.match(boldPattern);
21
+ if (boldMatch) return boldMatch[1].trim();
22
+ const plainPattern = new RegExp(`^${escaped}:\\s*(.+)`, 'im');
23
+ const plainMatch = content.match(plainPattern);
24
+ return plainMatch ? plainMatch[1].trim() : null;
25
+ }
26
+
27
+ function cmdStateLoad(cwd, raw) {
28
+ const config = loadConfig(cwd);
29
+ const planDir = planningPaths(cwd).planning;
30
+
31
+ let stateRaw = '';
32
+ try {
33
+ stateRaw = fs.readFileSync(path.join(planDir, 'STATE.md'), 'utf-8');
34
+ } catch { /* intentionally empty */ }
35
+
36
+ const configExists = fs.existsSync(path.join(planDir, 'config.json'));
37
+ const roadmapExists = fs.existsSync(path.join(planDir, 'ROADMAP.md'));
38
+ const stateExists = stateRaw.length > 0;
39
+
40
+ const result = {
41
+ config,
42
+ state_raw: stateRaw,
43
+ state_exists: stateExists,
44
+ roadmap_exists: roadmapExists,
45
+ config_exists: configExists,
46
+ };
47
+
48
+ // For --raw, output a condensed key=value format
49
+ if (raw) {
50
+ const c = config;
51
+ const lines = [
52
+ `model_profile=${c.model_profile}`,
53
+ `commit_docs=${c.commit_docs}`,
54
+ `branching_strategy=${c.branching_strategy}`,
55
+ `phase_branch_template=${c.phase_branch_template}`,
56
+ `milestone_branch_template=${c.milestone_branch_template}`,
57
+ `parallelization=${c.parallelization}`,
58
+ `research=${c.research}`,
59
+ `plan_checker=${c.plan_checker}`,
60
+ `verifier=${c.verifier}`,
61
+ `config_exists=${configExists}`,
62
+ `roadmap_exists=${roadmapExists}`,
63
+ `state_exists=${stateExists}`,
64
+ ];
65
+ process.stdout.write(lines.join('\n'));
66
+ process.exit(0);
67
+ }
68
+
69
+ output(result);
70
+ }
71
+
72
+ function cmdStateGet(cwd, section, raw) {
73
+ const statePath = planningPaths(cwd).state;
74
+ try {
75
+ const content = fs.readFileSync(statePath, 'utf-8');
76
+
77
+ if (!section) {
78
+ output({ content }, raw, content);
79
+ return;
80
+ }
81
+
82
+ // Try to find markdown section or field
83
+ const fieldEscaped = escapeRegex(section);
84
+
85
+ // Check for **field:** value (bold format)
86
+ const boldPattern = new RegExp(`\\*\\*${fieldEscaped}:\\*\\*\\s*(.*)`, 'i');
87
+ const boldMatch = content.match(boldPattern);
88
+ if (boldMatch) {
89
+ output({ [section]: boldMatch[1].trim() }, raw, boldMatch[1].trim());
90
+ return;
91
+ }
92
+
93
+ // Check for field: value (plain format)
94
+ const plainPattern = new RegExp(`^${fieldEscaped}:\\s*(.*)`, 'im');
95
+ const plainMatch = content.match(plainPattern);
96
+ if (plainMatch) {
97
+ output({ [section]: plainMatch[1].trim() }, raw, plainMatch[1].trim());
98
+ return;
99
+ }
100
+
101
+ // Check for ## Section
102
+ const sectionPattern = new RegExp(`##\\s*${fieldEscaped}\\s*\n([\\s\\S]*?)(?=\\n##|$)`, 'i');
103
+ const sectionMatch = content.match(sectionPattern);
104
+ if (sectionMatch) {
105
+ output({ [section]: sectionMatch[1].trim() }, raw, sectionMatch[1].trim());
106
+ return;
107
+ }
108
+
109
+ output({ error: `Section or field "${section}" not found` }, raw, '');
110
+ } catch {
111
+ error('STATE.md not found');
112
+ }
113
+ }
114
+
115
+ function readTextArgOrFile(cwd, value, filePath, label) {
116
+ if (!filePath) return value;
117
+
118
+ // Path traversal guard: ensure file resolves within project directory
119
+ const { validatePath } = require('./security.cjs');
120
+ const pathCheck = validatePath(filePath, cwd, { allowAbsolute: true });
121
+ if (!pathCheck.safe) {
122
+ throw new Error(`${label} path rejected: ${pathCheck.error}`);
123
+ }
124
+
125
+ try {
126
+ return fs.readFileSync(pathCheck.resolved, 'utf-8').trimEnd();
127
+ } catch {
128
+ throw new Error(`${label} file not found: ${filePath}`);
129
+ }
130
+ }
131
+
132
+ function cmdStatePatch(cwd, patches, raw) {
133
+ // Validate all field names before processing
134
+ const { validateFieldName } = require('./security.cjs');
135
+ for (const field of Object.keys(patches)) {
136
+ const fieldCheck = validateFieldName(field);
137
+ if (!fieldCheck.valid) {
138
+ error(`state patch: ${fieldCheck.error}`);
139
+ }
140
+ }
141
+
142
+ const statePath = planningPaths(cwd).state;
143
+ try {
144
+ let content = fs.readFileSync(statePath, 'utf-8');
145
+ const results = { updated: [], failed: [] };
146
+
147
+ for (const [field, value] of Object.entries(patches)) {
148
+ const fieldEscaped = escapeRegex(field);
149
+ // Try **Field:** bold format first, then plain Field: format
150
+ const boldPattern = new RegExp(`(\\*\\*${fieldEscaped}:\\*\\*\\s*)(.*)`, 'i');
151
+ const plainPattern = new RegExp(`(^${fieldEscaped}:\\s*)(.*)`, 'im');
152
+
153
+ if (boldPattern.test(content)) {
154
+ content = content.replace(boldPattern, (_match, prefix) => `${prefix}${value}`);
155
+ results.updated.push(field);
156
+ } else if (plainPattern.test(content)) {
157
+ content = content.replace(plainPattern, (_match, prefix) => `${prefix}${value}`);
158
+ results.updated.push(field);
159
+ } else {
160
+ results.failed.push(field);
161
+ }
162
+ }
163
+
164
+ if (results.updated.length > 0) {
165
+ writeStateMd(statePath, content, cwd);
166
+ }
167
+
168
+ output(results, raw, results.updated.length > 0 ? 'true' : 'false');
169
+ } catch {
170
+ error('STATE.md not found');
171
+ }
172
+ }
173
+
174
+ function cmdStateUpdate(cwd, field, value) {
175
+ if (!field || value === undefined) {
176
+ error('field and value required for state update');
177
+ }
178
+
179
+ // Validate field name to prevent regex injection via crafted field names
180
+ const { validateFieldName } = require('./security.cjs');
181
+ const fieldCheck = validateFieldName(field);
182
+ if (!fieldCheck.valid) {
183
+ error(`state update: ${fieldCheck.error}`);
184
+ }
185
+
186
+ const statePath = planningPaths(cwd).state;
187
+ try {
188
+ let content = fs.readFileSync(statePath, 'utf-8');
189
+ const fieldEscaped = escapeRegex(field);
190
+ // Try **Field:** bold format first, then plain Field: format
191
+ const boldPattern = new RegExp(`(\\*\\*${fieldEscaped}:\\*\\*\\s*)(.*)`, 'i');
192
+ const plainPattern = new RegExp(`(^${fieldEscaped}:\\s*)(.*)`, 'im');
193
+ if (boldPattern.test(content)) {
194
+ content = content.replace(boldPattern, (_match, prefix) => `${prefix}${value}`);
195
+ writeStateMd(statePath, content, cwd);
196
+ output({ updated: true });
197
+ } else if (plainPattern.test(content)) {
198
+ content = content.replace(plainPattern, (_match, prefix) => `${prefix}${value}`);
199
+ writeStateMd(statePath, content, cwd);
200
+ output({ updated: true });
201
+ } else {
202
+ output({ updated: false, reason: `Field "${field}" not found in STATE.md` });
203
+ }
204
+ } catch {
205
+ output({ updated: false, reason: 'STATE.md not found' });
206
+ }
207
+ }
208
+
209
+ // ─── State Progression Engine ────────────────────────────────────────────────
210
+ // stateExtractField is defined above (shared helper) — do not duplicate.
211
+
212
+ function stateReplaceField(content, fieldName, newValue) {
213
+ const escaped = escapeRegex(fieldName);
214
+ // Try **Field:** bold format first, then plain Field: format
215
+ const boldPattern = new RegExp(`(\\*\\*${escaped}:\\*\\*\\s*)(.*)`, 'i');
216
+ if (boldPattern.test(content)) {
217
+ return content.replace(boldPattern, (_match, prefix) => `${prefix}${newValue}`);
218
+ }
219
+ const plainPattern = new RegExp(`(^${escaped}:\\s*)(.*)`, 'im');
220
+ if (plainPattern.test(content)) {
221
+ return content.replace(plainPattern, (_match, prefix) => `${prefix}${newValue}`);
222
+ }
223
+ return null;
224
+ }
225
+
226
+ /**
227
+ * Replace a STATE.md field with fallback field name support.
228
+ * Tries `primary` first, then `fallback` (if provided), returns content unchanged
229
+ * if neither matches. This consolidates the replaceWithFallback pattern that was
230
+ * previously duplicated inline across phase.cjs, milestone.cjs, and state.cjs.
231
+ */
232
+ function stateReplaceFieldWithFallback(content, primary, fallback, value) {
233
+ let result = stateReplaceField(content, primary, value);
234
+ if (result) return result;
235
+ if (fallback) {
236
+ result = stateReplaceField(content, fallback, value);
237
+ if (result) return result;
238
+ }
239
+ return content;
240
+ }
241
+
242
+ /**
243
+ * Update fields within the ## Current Position section of STATE.md.
244
+ * This keeps the Current Position body in sync with the bold frontmatter fields.
245
+ * Only updates fields that already exist in the section; does not add new lines.
246
+ * Fixes #1365: advance-plan could not update Status/Last activity after begin-phase.
247
+ */
248
+ function updateCurrentPositionFields(content, fields) {
249
+ const posPattern = /(##\s*Current Position\s*\n)([\s\S]*?)(?=\n##|$)/i;
250
+ const posMatch = content.match(posPattern);
251
+ if (!posMatch) return content;
252
+
253
+ let posBody = posMatch[2];
254
+
255
+ if (fields.status && /^Status:/m.test(posBody)) {
256
+ posBody = posBody.replace(/^Status:.*$/m, `Status: ${fields.status}`);
257
+ }
258
+ if (fields.lastActivity && /^Last activity:/im.test(posBody)) {
259
+ posBody = posBody.replace(/^Last activity:.*$/im, `Last activity: ${fields.lastActivity}`);
260
+ }
261
+ if (fields.plan && /^Plan:/m.test(posBody)) {
262
+ posBody = posBody.replace(/^Plan:.*$/m, `Plan: ${fields.plan}`);
263
+ }
264
+
265
+ return content.replace(posPattern, `${posMatch[1]}${posBody}`);
266
+ }
267
+
268
+ function cmdStateAdvancePlan(cwd, raw) {
269
+ const statePath = planningPaths(cwd).state;
270
+ if (!fs.existsSync(statePath)) { output({ error: 'STATE.md not found' }, raw); return; }
271
+
272
+ let content = fs.readFileSync(statePath, 'utf-8');
273
+ const today = new Date().toISOString().split('T')[0];
274
+
275
+ // Try legacy separate fields first, then compound "Plan: X of Y" format
276
+ const legacyPlan = stateExtractField(content, 'Current Plan');
277
+ const legacyTotal = stateExtractField(content, 'Total Plans in Phase');
278
+ const planField = stateExtractField(content, 'Plan');
279
+
280
+ let currentPlan, totalPlans;
281
+ let useCompoundFormat = false;
282
+
283
+ if (legacyPlan && legacyTotal) {
284
+ currentPlan = parseInt(legacyPlan, 10);
285
+ totalPlans = parseInt(legacyTotal, 10);
286
+ } else if (planField) {
287
+ // Compound format: "2 of 6 in current phase" or "2 of 6"
288
+ currentPlan = parseInt(planField, 10);
289
+ const ofMatch = planField.match(/of\s+(\d+)/);
290
+ totalPlans = ofMatch ? parseInt(ofMatch[1], 10) : NaN;
291
+ useCompoundFormat = true;
292
+ }
293
+
294
+ if (isNaN(currentPlan) || isNaN(totalPlans)) {
295
+ output({ error: 'Cannot parse Current Plan or Total Plans in Phase from STATE.md' }, raw);
296
+ return;
297
+ }
298
+
299
+ if (currentPlan >= totalPlans) {
300
+ content = stateReplaceFieldWithFallback(content, 'Status', null, 'Phase complete — ready for verification');
301
+ content = stateReplaceFieldWithFallback(content, 'Last Activity', 'Last activity', today);
302
+ content = updateCurrentPositionFields(content, { status: 'Phase complete — ready for verification', lastActivity: today });
303
+ writeStateMd(statePath, content, cwd);
304
+ output({ advanced: false, reason: 'last_plan', current_plan: currentPlan, total_plans: totalPlans, status: 'ready_for_verification' }, raw, 'false');
305
+ } else {
306
+ const newPlan = currentPlan + 1;
307
+ let planDisplayValue;
308
+ if (useCompoundFormat) {
309
+ // Preserve compound format: "X of Y in current phase" → replace X only
310
+ planDisplayValue = planField.replace(/^\d+/, String(newPlan));
311
+ content = stateReplaceField(content, 'Plan', planDisplayValue) || content;
312
+ } else {
313
+ planDisplayValue = `${newPlan} of ${totalPlans}`;
314
+ content = stateReplaceField(content, 'Current Plan', String(newPlan)) || content;
315
+ }
316
+ content = stateReplaceFieldWithFallback(content, 'Status', null, 'Ready to execute');
317
+ content = stateReplaceFieldWithFallback(content, 'Last Activity', 'Last activity', today);
318
+ content = updateCurrentPositionFields(content, { status: 'Ready to execute', lastActivity: today, plan: planDisplayValue });
319
+ writeStateMd(statePath, content, cwd);
320
+ output({ advanced: true, previous_plan: currentPlan, current_plan: newPlan, total_plans: totalPlans }, raw, 'true');
321
+ }
322
+ }
323
+
324
+ function cmdStateRecordMetric(cwd, options, raw) {
325
+ const statePath = planningPaths(cwd).state;
326
+ if (!fs.existsSync(statePath)) { output({ error: 'STATE.md not found' }, raw); return; }
327
+
328
+ let content = fs.readFileSync(statePath, 'utf-8');
329
+ const { phase, plan, duration, tasks, files } = options;
330
+
331
+ if (!phase || !plan || !duration) {
332
+ output({ error: 'phase, plan, and duration required' }, raw);
333
+ return;
334
+ }
335
+
336
+ // Find Performance Metrics section and its table
337
+ const metricsPattern = /(##\s*Performance Metrics[\s\S]*?\n\|[^\n]+\n\|[-|\s]+\n)([\s\S]*?)(?=\n##|\n$|$)/i;
338
+ const metricsMatch = content.match(metricsPattern);
339
+
340
+ if (metricsMatch) {
341
+ let tableBody = metricsMatch[2].trimEnd();
342
+ const newRow = `| Phase ${phase} P${plan} | ${duration} | ${tasks || '-'} tasks | ${files || '-'} files |`;
343
+
344
+ if (tableBody.trim() === '' || tableBody.includes('None yet')) {
345
+ tableBody = newRow;
346
+ } else {
347
+ tableBody = tableBody + '\n' + newRow;
348
+ }
349
+
350
+ content = content.replace(metricsPattern, (_match, header) => `${header}${tableBody}\n`);
351
+ writeStateMd(statePath, content, cwd);
352
+ output({ recorded: true, phase, plan, duration }, raw, 'true');
353
+ } else {
354
+ output({ recorded: false, reason: 'Performance Metrics section not found in STATE.md' }, raw, 'false');
355
+ }
356
+ }
357
+
358
+ function cmdStateUpdateProgress(cwd, raw) {
359
+ const statePath = planningPaths(cwd).state;
360
+ if (!fs.existsSync(statePath)) { output({ error: 'STATE.md not found' }, raw); return; }
361
+
362
+ let content = fs.readFileSync(statePath, 'utf-8');
363
+
364
+ // Count summaries across current milestone phases only
365
+ const phasesDir = planningPaths(cwd).phases;
366
+ let totalPlans = 0;
367
+ let totalSummaries = 0;
368
+
369
+ if (fs.existsSync(phasesDir)) {
370
+ const isDirInMilestone = getMilestonePhaseFilter(cwd);
371
+ const phaseDirs = fs.readdirSync(phasesDir, { withFileTypes: true })
372
+ .filter(e => e.isDirectory()).map(e => e.name)
373
+ .filter(isDirInMilestone);
374
+ for (const dir of phaseDirs) {
375
+ const files = fs.readdirSync(path.join(phasesDir, dir));
376
+ totalPlans += files.filter(f => f.match(/-PLAN\.md$/i)).length;
377
+ totalSummaries += files.filter(f => f.match(/-SUMMARY\.md$/i)).length;
378
+ }
379
+ }
380
+
381
+ const percent = totalPlans > 0 ? Math.min(100, Math.round(totalSummaries / totalPlans * 100)) : 0;
382
+ const barWidth = 10;
383
+ const filled = Math.round(percent / 100 * barWidth);
384
+ const bar = '\u2588'.repeat(filled) + '\u2591'.repeat(barWidth - filled);
385
+ const progressStr = `[${bar}] ${percent}%`;
386
+
387
+ // Try **Progress:** bold format first, then plain Progress: format
388
+ const boldProgressPattern = /(\*\*Progress:\*\*\s*).*/i;
389
+ const plainProgressPattern = /^(Progress:\s*).*/im;
390
+ if (boldProgressPattern.test(content)) {
391
+ content = content.replace(boldProgressPattern, (_match, prefix) => `${prefix}${progressStr}`);
392
+ writeStateMd(statePath, content, cwd);
393
+ output({ updated: true, percent, completed: totalSummaries, total: totalPlans, bar: progressStr }, raw, progressStr);
394
+ } else if (plainProgressPattern.test(content)) {
395
+ content = content.replace(plainProgressPattern, (_match, prefix) => `${prefix}${progressStr}`);
396
+ writeStateMd(statePath, content, cwd);
397
+ output({ updated: true, percent, completed: totalSummaries, total: totalPlans, bar: progressStr }, raw, progressStr);
398
+ } else {
399
+ output({ updated: false, reason: 'Progress field not found in STATE.md' }, raw, 'false');
400
+ }
401
+ }
402
+
403
+ function cmdStateAddDecision(cwd, options, raw) {
404
+ const statePath = planningPaths(cwd).state;
405
+ if (!fs.existsSync(statePath)) { output({ error: 'STATE.md not found' }, raw); return; }
406
+
407
+ const { phase, summary, summary_file, rationale, rationale_file } = options;
408
+ let summaryText = null;
409
+ let rationaleText = '';
410
+
411
+ try {
412
+ summaryText = readTextArgOrFile(cwd, summary, summary_file, 'summary');
413
+ rationaleText = readTextArgOrFile(cwd, rationale || '', rationale_file, 'rationale');
414
+ } catch (err) {
415
+ output({ added: false, reason: err.message }, raw, 'false');
416
+ return;
417
+ }
418
+
419
+ if (!summaryText) { output({ error: 'summary required' }, raw); return; }
420
+
421
+ let content = fs.readFileSync(statePath, 'utf-8');
422
+ const entry = `- [Phase ${phase || '?'}]: ${summaryText}${rationaleText ? ` — ${rationaleText}` : ''}`;
423
+
424
+ // Find Decisions section (various heading patterns)
425
+ const sectionPattern = /(###?\s*(?:Decisions|Decisions Made|Accumulated.*Decisions)\s*\n)([\s\S]*?)(?=\n###?|\n##[^#]|$)/i;
426
+ const match = content.match(sectionPattern);
427
+
428
+ if (match) {
429
+ let sectionBody = match[2];
430
+ // Remove placeholders
431
+ sectionBody = sectionBody.replace(/None yet\.?\s*\n?/gi, '').replace(/No decisions yet\.?\s*\n?/gi, '');
432
+ sectionBody = sectionBody.trimEnd() + '\n' + entry + '\n';
433
+ content = content.replace(sectionPattern, (_match, header) => `${header}${sectionBody}`);
434
+ writeStateMd(statePath, content, cwd);
435
+ output({ added: true, decision: entry }, raw, 'true');
436
+ } else {
437
+ output({ added: false, reason: 'Decisions section not found in STATE.md' }, raw, 'false');
438
+ }
439
+ }
440
+
441
+ function cmdStateAddBlocker(cwd, text, raw) {
442
+ const statePath = planningPaths(cwd).state;
443
+ if (!fs.existsSync(statePath)) { output({ error: 'STATE.md not found' }, raw); return; }
444
+ const blockerOptions = typeof text === 'object' && text !== null ? text : { text };
445
+ let blockerText = null;
446
+
447
+ try {
448
+ blockerText = readTextArgOrFile(cwd, blockerOptions.text, blockerOptions.text_file, 'blocker');
449
+ } catch (err) {
450
+ output({ added: false, reason: err.message }, raw, 'false');
451
+ return;
452
+ }
453
+
454
+ if (!blockerText) { output({ error: 'text required' }, raw); return; }
455
+
456
+ let content = fs.readFileSync(statePath, 'utf-8');
457
+ const entry = `- ${blockerText}`;
458
+
459
+ const sectionPattern = /(###?\s*(?:Blockers|Blockers\/Concerns|Concerns)\s*\n)([\s\S]*?)(?=\n###?|\n##[^#]|$)/i;
460
+ const match = content.match(sectionPattern);
461
+
462
+ if (match) {
463
+ let sectionBody = match[2];
464
+ sectionBody = sectionBody.replace(/None\.?\s*\n?/gi, '').replace(/None yet\.?\s*\n?/gi, '');
465
+ sectionBody = sectionBody.trimEnd() + '\n' + entry + '\n';
466
+ content = content.replace(sectionPattern, (_match, header) => `${header}${sectionBody}`);
467
+ writeStateMd(statePath, content, cwd);
468
+ output({ added: true, blocker: blockerText }, raw, 'true');
469
+ } else {
470
+ output({ added: false, reason: 'Blockers section not found in STATE.md' }, raw, 'false');
471
+ }
472
+ }
473
+
474
+ function cmdStateResolveBlocker(cwd, text, raw) {
475
+ const statePath = planningPaths(cwd).state;
476
+ if (!fs.existsSync(statePath)) { output({ error: 'STATE.md not found' }, raw); return; }
477
+ if (!text) { output({ error: 'text required' }, raw); return; }
478
+
479
+ let content = fs.readFileSync(statePath, 'utf-8');
480
+
481
+ const sectionPattern = /(###?\s*(?:Blockers|Blockers\/Concerns|Concerns)\s*\n)([\s\S]*?)(?=\n###?|\n##[^#]|$)/i;
482
+ const match = content.match(sectionPattern);
483
+
484
+ if (match) {
485
+ const sectionBody = match[2];
486
+ const lines = sectionBody.split('\n');
487
+ const filtered = lines.filter(line => {
488
+ if (!line.startsWith('- ')) return true;
489
+ return !line.toLowerCase().includes(text.toLowerCase());
490
+ });
491
+
492
+ let newBody = filtered.join('\n');
493
+ // If section is now empty, add placeholder
494
+ if (!newBody.trim() || !newBody.includes('- ')) {
495
+ newBody = 'None\n';
496
+ }
497
+
498
+ content = content.replace(sectionPattern, (_match, header) => `${header}${newBody}`);
499
+ writeStateMd(statePath, content, cwd);
500
+ output({ resolved: true, blocker: text }, raw, 'true');
501
+ } else {
502
+ output({ resolved: false, reason: 'Blockers section not found in STATE.md' }, raw, 'false');
503
+ }
504
+ }
505
+
506
+ function cmdStateRecordSession(cwd, options, raw) {
507
+ const statePath = planningPaths(cwd).state;
508
+ if (!fs.existsSync(statePath)) { output({ error: 'STATE.md not found' }, raw); return; }
509
+
510
+ let content = fs.readFileSync(statePath, 'utf-8');
511
+ const now = new Date().toISOString();
512
+ const updated = [];
513
+
514
+ // Update Last session / Last Date
515
+ let result = stateReplaceField(content, 'Last session', now);
516
+ if (result) { content = result; updated.push('Last session'); }
517
+ result = stateReplaceField(content, 'Last Date', now);
518
+ if (result) { content = result; updated.push('Last Date'); }
519
+
520
+ // Update Stopped at
521
+ if (options.stopped_at) {
522
+ result = stateReplaceField(content, 'Stopped At', options.stopped_at);
523
+ if (!result) result = stateReplaceField(content, 'Stopped at', options.stopped_at);
524
+ if (result) { content = result; updated.push('Stopped At'); }
525
+ }
526
+
527
+ // Update Resume file
528
+ const resumeFile = options.resume_file || 'None';
529
+ result = stateReplaceField(content, 'Resume File', resumeFile);
530
+ if (!result) result = stateReplaceField(content, 'Resume file', resumeFile);
531
+ if (result) { content = result; updated.push('Resume File'); }
532
+
533
+ if (updated.length > 0) {
534
+ writeStateMd(statePath, content, cwd);
535
+ output({ recorded: true, updated }, raw, 'true');
536
+ } else {
537
+ output({ recorded: false, reason: 'No session fields found in STATE.md' }, raw, 'false');
538
+ }
539
+ }
540
+
541
+ function cmdStateSnapshot(cwd, raw) {
542
+ const statePath = planningPaths(cwd).state;
543
+
544
+ if (!fs.existsSync(statePath)) {
545
+ output({ error: 'STATE.md not found' }, raw);
546
+ return;
547
+ }
548
+
549
+ const content = fs.readFileSync(statePath, 'utf-8');
550
+
551
+ // Extract basic fields
552
+ const currentPhase = stateExtractField(content, 'Current Phase');
553
+ const currentPhaseName = stateExtractField(content, 'Current Phase Name');
554
+ const totalPhasesRaw = stateExtractField(content, 'Total Phases');
555
+ const currentPlan = stateExtractField(content, 'Current Plan');
556
+ const totalPlansRaw = stateExtractField(content, 'Total Plans in Phase');
557
+ const status = stateExtractField(content, 'Status');
558
+ const progressRaw = stateExtractField(content, 'Progress');
559
+ const lastActivity = stateExtractField(content, 'Last Activity');
560
+ const lastActivityDesc = stateExtractField(content, 'Last Activity Description');
561
+ const pausedAt = stateExtractField(content, 'Paused At');
562
+
563
+ // Parse numeric fields
564
+ const totalPhases = totalPhasesRaw ? parseInt(totalPhasesRaw, 10) : null;
565
+ const totalPlansInPhase = totalPlansRaw ? parseInt(totalPlansRaw, 10) : null;
566
+ const progressPercent = progressRaw ? parseInt(progressRaw.replace('%', ''), 10) : null;
567
+
568
+ // Extract decisions table
569
+ const decisions = [];
570
+ const decisionsMatch = content.match(/##\s*Decisions Made[\s\S]*?\n\|[^\n]+\n\|[-|\s]+\n([\s\S]*?)(?=\n##|\n$|$)/i);
571
+ if (decisionsMatch) {
572
+ const tableBody = decisionsMatch[1];
573
+ const rows = tableBody.trim().split('\n').filter(r => r.includes('|'));
574
+ for (const row of rows) {
575
+ const cells = row.split('|').map(c => c.trim()).filter(Boolean);
576
+ if (cells.length >= 3) {
577
+ decisions.push({
578
+ phase: cells[0],
579
+ summary: cells[1],
580
+ rationale: cells[2],
581
+ });
582
+ }
583
+ }
584
+ }
585
+
586
+ // Extract blockers list
587
+ const blockers = [];
588
+ const blockersMatch = content.match(/##\s*Blockers\s*\n([\s\S]*?)(?=\n##|$)/i);
589
+ if (blockersMatch) {
590
+ const blockersSection = blockersMatch[1];
591
+ const items = blockersSection.match(/^-\s+(.+)$/gm) || [];
592
+ for (const item of items) {
593
+ blockers.push(item.replace(/^-\s+/, '').trim());
594
+ }
595
+ }
596
+
597
+ // Extract session info
598
+ const session = {
599
+ last_date: null,
600
+ stopped_at: null,
601
+ resume_file: null,
602
+ };
603
+
604
+ const sessionMatch = content.match(/##\s*Session\s*\n([\s\S]*?)(?=\n##|$)/i);
605
+ if (sessionMatch) {
606
+ const sessionSection = sessionMatch[1];
607
+ const lastDateMatch = sessionSection.match(/\*\*Last Date:\*\*\s*(.+)/i)
608
+ || sessionSection.match(/^Last Date:\s*(.+)/im);
609
+ const stoppedAtMatch = sessionSection.match(/\*\*Stopped At:\*\*\s*(.+)/i)
610
+ || sessionSection.match(/^Stopped At:\s*(.+)/im);
611
+ const resumeFileMatch = sessionSection.match(/\*\*Resume File:\*\*\s*(.+)/i)
612
+ || sessionSection.match(/^Resume File:\s*(.+)/im);
613
+
614
+ if (lastDateMatch) session.last_date = lastDateMatch[1].trim();
615
+ if (stoppedAtMatch) session.stopped_at = stoppedAtMatch[1].trim();
616
+ if (resumeFileMatch) session.resume_file = resumeFileMatch[1].trim();
617
+ }
618
+
619
+ const result = {
620
+ current_phase: currentPhase,
621
+ current_phase_name: currentPhaseName,
622
+ total_phases: totalPhases,
623
+ current_plan: currentPlan,
624
+ total_plans_in_phase: totalPlansInPhase,
625
+ status,
626
+ progress_percent: progressPercent,
627
+ last_activity: lastActivity,
628
+ last_activity_desc: lastActivityDesc,
629
+ decisions,
630
+ blockers,
631
+ paused_at: pausedAt,
632
+ session,
633
+ };
634
+
635
+ output(result, raw);
636
+ }
637
+
638
+ // ─── State Frontmatter Sync ──────────────────────────────────────────────────
639
+
640
+ /**
641
+ * Extract machine-readable fields from STATE.md markdown body and build
642
+ * a YAML frontmatter object. Allows hooks and scripts to read state
643
+ * reliably via `state json` instead of fragile regex parsing.
644
+ */
645
+ function buildStateFrontmatter(bodyContent, cwd) {
646
+ const currentPhase = stateExtractField(bodyContent, 'Current Phase');
647
+ const currentPhaseName = stateExtractField(bodyContent, 'Current Phase Name');
648
+ const currentPlan = stateExtractField(bodyContent, 'Current Plan');
649
+ const totalPhasesRaw = stateExtractField(bodyContent, 'Total Phases');
650
+ const totalPlansRaw = stateExtractField(bodyContent, 'Total Plans in Phase');
651
+ const status = stateExtractField(bodyContent, 'Status');
652
+ const progressRaw = stateExtractField(bodyContent, 'Progress');
653
+ const lastActivity = stateExtractField(bodyContent, 'Last Activity');
654
+ const stoppedAt = stateExtractField(bodyContent, 'Stopped At') || stateExtractField(bodyContent, 'Stopped at');
655
+ const pausedAt = stateExtractField(bodyContent, 'Paused At');
656
+
657
+ let milestone = null;
658
+ let milestoneName = null;
659
+ if (cwd) {
660
+ try {
661
+ const info = getMilestoneInfo(cwd);
662
+ milestone = info.version;
663
+ milestoneName = info.name;
664
+ } catch { /* intentionally empty */ }
665
+ }
666
+
667
+ let totalPhases = totalPhasesRaw ? parseInt(totalPhasesRaw, 10) : null;
668
+ let completedPhases = null;
669
+ let totalPlans = totalPlansRaw ? parseInt(totalPlansRaw, 10) : null;
670
+ let completedPlans = null;
671
+
672
+ if (cwd) {
673
+ try {
674
+ const phasesDir = planningPaths(cwd).phases;
675
+ if (fs.existsSync(phasesDir)) {
676
+ const isDirInMilestone = getMilestonePhaseFilter(cwd);
677
+ const phaseDirs = fs.readdirSync(phasesDir, { withFileTypes: true })
678
+ .filter(e => e.isDirectory()).map(e => e.name)
679
+ .filter(isDirInMilestone);
680
+ let diskTotalPlans = 0;
681
+ let diskTotalSummaries = 0;
682
+ let diskCompletedPhases = 0;
683
+
684
+ for (const dir of phaseDirs) {
685
+ const files = fs.readdirSync(path.join(phasesDir, dir));
686
+ const plans = files.filter(f => f.match(/-PLAN\.md$/i)).length;
687
+ const summaries = files.filter(f => f.match(/-SUMMARY\.md$/i)).length;
688
+ diskTotalPlans += plans;
689
+ diskTotalSummaries += summaries;
690
+ if (plans > 0 && summaries >= plans) diskCompletedPhases++;
691
+ }
692
+ totalPhases = isDirInMilestone.phaseCount > 0
693
+ ? Math.max(phaseDirs.length, isDirInMilestone.phaseCount)
694
+ : phaseDirs.length;
695
+ completedPhases = diskCompletedPhases;
696
+ totalPlans = diskTotalPlans;
697
+ completedPlans = diskTotalSummaries;
698
+ }
699
+ } catch { /* intentionally empty */ }
700
+ }
701
+
702
+ let progressPercent = null;
703
+ if (progressRaw) {
704
+ const pctMatch = progressRaw.match(/(\d+)%/);
705
+ if (pctMatch) progressPercent = parseInt(pctMatch[1], 10);
706
+ }
707
+
708
+ // Normalize status to one of: planning, discussing, executing, verifying, paused, completed, unknown
709
+ let normalizedStatus = status || 'unknown';
710
+ const statusLower = (status || '').toLowerCase();
711
+ if (statusLower.includes('paused') || statusLower.includes('stopped') || pausedAt) {
712
+ normalizedStatus = 'paused';
713
+ } else if (statusLower.includes('executing') || statusLower.includes('in progress')) {
714
+ normalizedStatus = 'executing';
715
+ } else if (statusLower.includes('planning') || statusLower.includes('ready to plan')) {
716
+ normalizedStatus = 'planning';
717
+ } else if (statusLower.includes('discussing')) {
718
+ normalizedStatus = 'discussing';
719
+ } else if (statusLower.includes('verif')) {
720
+ normalizedStatus = 'verifying';
721
+ } else if (statusLower.includes('complete') || statusLower.includes('done')) {
722
+ normalizedStatus = 'completed';
723
+ } else if (statusLower.includes('ready to execute')) {
724
+ normalizedStatus = 'executing';
725
+ }
726
+
727
+ const fm = { state_version: '1.0' };
728
+
729
+ if (milestone) fm.milestone = milestone;
730
+ if (milestoneName) fm.milestone_name = milestoneName;
731
+ if (currentPhase) fm.current_phase = currentPhase;
732
+ if (currentPhaseName) fm.current_phase_name = currentPhaseName;
733
+ if (currentPlan) fm.current_plan = currentPlan;
734
+ fm.status = normalizedStatus;
735
+ if (stoppedAt) fm.stopped_at = stoppedAt;
736
+ if (pausedAt) fm.paused_at = pausedAt;
737
+ fm.last_updated = new Date().toISOString();
738
+ if (lastActivity) fm.last_activity = lastActivity;
739
+
740
+ const progress = {};
741
+ if (totalPhases !== null) progress.total_phases = totalPhases;
742
+ if (completedPhases !== null) progress.completed_phases = completedPhases;
743
+ if (totalPlans !== null) progress.total_plans = totalPlans;
744
+ if (completedPlans !== null) progress.completed_plans = completedPlans;
745
+ if (progressPercent !== null) progress.percent = progressPercent;
746
+ if (Object.keys(progress).length > 0) fm.progress = progress;
747
+
748
+ return fm;
749
+ }
750
+
751
+ function stripFrontmatter(content) {
752
+ // Strip ALL frontmatter blocks at the start of the file.
753
+ // Handles CRLF line endings and multiple stacked blocks (corruption recovery).
754
+ // Greedy: keeps stripping ---...--- blocks separated by optional whitespace.
755
+ let result = content;
756
+ // eslint-disable-next-line no-constant-condition
757
+ while (true) {
758
+ const stripped = result.replace(/^\s*---\r?\n[\s\S]*?\r?\n---\s*/, '');
759
+ if (stripped === result) break;
760
+ result = stripped;
761
+ }
762
+ return result;
763
+ }
764
+
765
+ function syncStateFrontmatter(content, cwd) {
766
+ // Read existing frontmatter BEFORE stripping — it may contain values
767
+ // that the body no longer has (e.g., Status field removed by an agent).
768
+ const existingFm = extractFrontmatter(content);
769
+ const body = stripFrontmatter(content);
770
+ const derivedFm = buildStateFrontmatter(body, cwd);
771
+
772
+ // Preserve existing frontmatter status when body-derived status is 'unknown'.
773
+ // This prevents a missing Status: field in the body from overwriting a
774
+ // previously valid status (e.g., 'executing' → 'unknown').
775
+ if (derivedFm.status === 'unknown' && existingFm.status && existingFm.status !== 'unknown') {
776
+ derivedFm.status = existingFm.status;
777
+ }
778
+
779
+ const yamlStr = reconstructFrontmatter(derivedFm);
780
+ return `---\n${yamlStr}\n---\n\n${body}`;
781
+ }
782
+
783
+ /**
784
+ * Write STATE.md with synchronized YAML frontmatter.
785
+ * All STATE.md writes should use this instead of raw writeFileSync.
786
+ * Uses a simple lockfile to prevent parallel agents from overwriting
787
+ * each other's changes (race condition with read-modify-write cycle).
788
+ */
789
+ function writeStateMd(statePath, content, cwd) {
790
+ const synced = syncStateFrontmatter(content, cwd);
791
+ const lockPath = statePath + '.lock';
792
+ const maxRetries = 10;
793
+ const retryDelay = 200; // ms
794
+
795
+ // Acquire lock (spin with backoff)
796
+ for (let i = 0; i < maxRetries; i++) {
797
+ try {
798
+ // O_EXCL fails if file already exists — atomic lock
799
+ const fd = fs.openSync(lockPath, fs.constants.O_CREAT | fs.constants.O_EXCL | fs.constants.O_WRONLY);
800
+ fs.writeSync(fd, String(process.pid));
801
+ fs.closeSync(fd);
802
+ break;
803
+ } catch (err) {
804
+ if (err.code === 'EEXIST') {
805
+ // Check for stale lock (> 10s old)
806
+ try {
807
+ const stat = fs.statSync(lockPath);
808
+ if (Date.now() - stat.mtimeMs > 10000) {
809
+ fs.unlinkSync(lockPath);
810
+ continue; // retry immediately after clearing stale lock
811
+ }
812
+ } catch { /* lock was released between check — retry */ }
813
+
814
+ if (i === maxRetries - 1) {
815
+ // Last resort: write anyway rather than losing data
816
+ try { fs.unlinkSync(lockPath); } catch {}
817
+ break;
818
+ }
819
+ // Spin-wait with small jitter
820
+ const jitter = Math.floor(Math.random() * 50);
821
+ const start = Date.now();
822
+ while (Date.now() - start < retryDelay + jitter) { /* busy wait */ }
823
+ continue;
824
+ }
825
+ break; // non-EEXIST error — proceed without lock
826
+ }
827
+ }
828
+
829
+ try {
830
+ fs.writeFileSync(statePath, normalizeMd(synced), 'utf-8');
831
+ } finally {
832
+ try { fs.unlinkSync(lockPath); } catch { /* lock already gone */ }
833
+ }
834
+ }
835
+
836
+ function cmdStateJson(cwd, raw) {
837
+ const statePath = planningPaths(cwd).state;
838
+ if (!fs.existsSync(statePath)) {
839
+ output({ error: 'STATE.md not found' }, raw, 'STATE.md not found');
840
+ return;
841
+ }
842
+
843
+ const content = fs.readFileSync(statePath, 'utf-8');
844
+ const fm = extractFrontmatter(content);
845
+
846
+ if (!fm || Object.keys(fm).length === 0) {
847
+ const body = stripFrontmatter(content);
848
+ const built = buildStateFrontmatter(body, cwd);
849
+ output(built, raw, JSON.stringify(built, null, 2));
850
+ return;
851
+ }
852
+
853
+ output(fm, raw, JSON.stringify(fm, null, 2));
854
+ }
855
+
856
+ /**
857
+ * Update STATE.md when a new phase begins execution.
858
+ * Updates body text fields (Current focus, Status, Last Activity, Current Position)
859
+ * and synchronizes frontmatter via writeStateMd.
860
+ * Fixes: #1102 (plan counts), #1103 (status/last_activity), #1104 (body text).
861
+ */
862
+ function cmdStateBeginPhase(cwd, phaseNumber, phaseName, planCount, raw) {
863
+ const statePath = planningPaths(cwd).state;
864
+ if (!fs.existsSync(statePath)) {
865
+ output({ error: 'STATE.md not found' }, raw);
866
+ return;
867
+ }
868
+
869
+ let content = fs.readFileSync(statePath, 'utf-8');
870
+ const today = new Date().toISOString().split('T')[0];
871
+ const updated = [];
872
+
873
+ // Update Status field
874
+ const statusValue = `Executing Phase ${phaseNumber}`;
875
+ let result = stateReplaceField(content, 'Status', statusValue);
876
+ if (result) { content = result; updated.push('Status'); }
877
+
878
+ // Update Last Activity
879
+ result = stateReplaceField(content, 'Last Activity', today);
880
+ if (result) { content = result; updated.push('Last Activity'); }
881
+
882
+ // Update Last Activity Description if it exists
883
+ const activityDesc = `Phase ${phaseNumber} execution started`;
884
+ result = stateReplaceField(content, 'Last Activity Description', activityDesc);
885
+ if (result) { content = result; updated.push('Last Activity Description'); }
886
+
887
+ // Update Current Phase
888
+ result = stateReplaceField(content, 'Current Phase', String(phaseNumber));
889
+ if (result) { content = result; updated.push('Current Phase'); }
890
+
891
+ // Update Current Phase Name
892
+ if (phaseName) {
893
+ result = stateReplaceField(content, 'Current Phase Name', phaseName);
894
+ if (result) { content = result; updated.push('Current Phase Name'); }
895
+ }
896
+
897
+ // Update Current Plan to 1 (starting from the first plan)
898
+ result = stateReplaceField(content, 'Current Plan', '1');
899
+ if (result) { content = result; updated.push('Current Plan'); }
900
+
901
+ // Update Total Plans in Phase
902
+ if (planCount) {
903
+ result = stateReplaceField(content, 'Total Plans in Phase', String(planCount));
904
+ if (result) { content = result; updated.push('Total Plans in Phase'); }
905
+ }
906
+
907
+ // Update **Current focus:** body text line (#1104)
908
+ const focusLabel = phaseName ? `Phase ${phaseNumber} — ${phaseName}` : `Phase ${phaseNumber}`;
909
+ const focusPattern = /(\*\*Current focus:\*\*\s*).*/i;
910
+ if (focusPattern.test(content)) {
911
+ content = content.replace(focusPattern, (_match, prefix) => `${prefix}${focusLabel}`);
912
+ updated.push('Current focus');
913
+ }
914
+
915
+ // Update ## Current Position section (#1104, #1365)
916
+ // Update individual fields within Current Position instead of replacing the
917
+ // entire section, so that Status, Last activity, and Progress are preserved.
918
+ const positionPattern = /(##\s*Current Position\s*\n)([\s\S]*?)(?=\n##|$)/i;
919
+ const positionMatch = content.match(positionPattern);
920
+ if (positionMatch) {
921
+ const header = positionMatch[1];
922
+ let posBody = positionMatch[2];
923
+
924
+ // Update or insert Phase line
925
+ const newPhase = `Phase: ${phaseNumber}${phaseName ? ` (${phaseName})` : ''} — EXECUTING`;
926
+ if (/^Phase:/m.test(posBody)) {
927
+ posBody = posBody.replace(/^Phase:.*$/m, newPhase);
928
+ } else {
929
+ posBody = newPhase + '\n' + posBody;
930
+ }
931
+
932
+ // Update or insert Plan line
933
+ const newPlan = `Plan: 1 of ${planCount || '?'}`;
934
+ if (/^Plan:/m.test(posBody)) {
935
+ posBody = posBody.replace(/^Plan:.*$/m, newPlan);
936
+ } else {
937
+ posBody = posBody.replace(/^(Phase:.*$)/m, `$1\n${newPlan}`);
938
+ }
939
+
940
+ // Update Status line if present
941
+ const newStatus = `Status: Executing Phase ${phaseNumber}`;
942
+ if (/^Status:/m.test(posBody)) {
943
+ posBody = posBody.replace(/^Status:.*$/m, newStatus);
944
+ }
945
+
946
+ // Update Last activity line if present
947
+ const newActivity = `Last activity: ${today} -- Phase ${phaseNumber} execution started`;
948
+ if (/^Last activity:/im.test(posBody)) {
949
+ posBody = posBody.replace(/^Last activity:.*$/im, newActivity);
950
+ }
951
+
952
+ content = content.replace(positionPattern, `${header}${posBody}`);
953
+ updated.push('Current Position');
954
+ }
955
+
956
+ if (updated.length > 0) {
957
+ writeStateMd(statePath, content, cwd);
958
+ }
959
+
960
+ output({ updated, phase: phaseNumber, phase_name: phaseName || null, plan_count: planCount || null }, raw, updated.length > 0 ? 'true' : 'false');
961
+ }
962
+
963
+ /**
964
+ * Write a WAITING.json signal file when framework hits a decision point.
965
+ * External watchers (fswatch, polling, orchestrators) can detect this.
966
+ * File is written to .planning/WAITING.json (or .framework/WAITING.json if .framework exists).
967
+ * Fixes #1034.
968
+ */
969
+ function cmdSignalWaiting(cwd, type, question, options, phase, raw) {
970
+ const frameworkDir = fs.existsSync(path.join(cwd, '.framework')) ? path.join(cwd, '.framework') : planningDir(cwd);
971
+ const waitingPath = path.join(frameworkDir, 'WAITING.json');
972
+
973
+ const signal = {
974
+ status: 'waiting',
975
+ type: type || 'decision_point',
976
+ question: question || null,
977
+ options: options ? options.split('|').map(o => o.trim()) : [],
978
+ since: new Date().toISOString(),
979
+ phase: phase || null,
980
+ };
981
+
982
+ try {
983
+ fs.mkdirSync(frameworkDir, { recursive: true });
984
+ fs.writeFileSync(waitingPath, JSON.stringify(signal, null, 2), 'utf-8');
985
+ output({ signaled: true, path: waitingPath }, raw, 'true');
986
+ } catch (e) {
987
+ output({ signaled: false, error: e.message }, raw, 'false');
988
+ }
989
+ }
990
+
991
+ /**
992
+ * Remove the WAITING.json signal file when user answers and agent resumes.
993
+ */
994
+ function cmdSignalResume(cwd, raw) {
995
+ const paths = [
996
+ path.join(cwd, '.framework', 'WAITING.json'),
997
+ path.join(planningDir(cwd), 'WAITING.json'),
998
+ ];
999
+
1000
+ let removed = false;
1001
+ for (const p of paths) {
1002
+ if (fs.existsSync(p)) {
1003
+ try { fs.unlinkSync(p); removed = true; } catch {}
1004
+ }
1005
+ }
1006
+
1007
+ output({ resumed: true, removed }, raw, removed ? 'true' : 'false');
1008
+ }
1009
+
1010
+ module.exports = {
1011
+ stateExtractField,
1012
+ stateReplaceField,
1013
+ stateReplaceFieldWithFallback,
1014
+ writeStateMd,
1015
+ cmdStateLoad,
1016
+ cmdStateGet,
1017
+ cmdStatePatch,
1018
+ cmdStateUpdate,
1019
+ cmdStateAdvancePlan,
1020
+ cmdStateRecordMetric,
1021
+ cmdStateUpdateProgress,
1022
+ cmdStateAddDecision,
1023
+ cmdStateAddBlocker,
1024
+ cmdStateResolveBlocker,
1025
+ cmdStateRecordSession,
1026
+ cmdStateSnapshot,
1027
+ cmdStateJson,
1028
+ cmdStateBeginPhase,
1029
+ cmdSignalWaiting,
1030
+ cmdSignalResume,
1031
+ };