mandrel 1.57.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 (843) hide show
  1. package/.agents/README.md +954 -0
  2. package/.agents/docs/SDLC.md +1420 -0
  3. package/.agents/docs/agentrc-reference.json +278 -0
  4. package/.agents/docs/configuration.md +1040 -0
  5. package/.agents/docs/workflows.md +59 -0
  6. package/.agents/instructions.md +384 -0
  7. package/.agents/personas/architect.md +107 -0
  8. package/.agents/personas/devops-engineer.md +36 -0
  9. package/.agents/personas/engineer-mobile.md +119 -0
  10. package/.agents/personas/engineer-web.md +110 -0
  11. package/.agents/personas/engineer.md +90 -0
  12. package/.agents/personas/product.md +88 -0
  13. package/.agents/personas/project-manager.md +110 -0
  14. package/.agents/personas/qa-engineer.md +91 -0
  15. package/.agents/personas/refactorer.md +110 -0
  16. package/.agents/personas/security-engineer.md +112 -0
  17. package/.agents/personas/sre.md +86 -0
  18. package/.agents/personas/technical-writer.md +100 -0
  19. package/.agents/personas/ux-designer.md +95 -0
  20. package/.agents/rules/api-conventions.md +75 -0
  21. package/.agents/rules/changelog-style.md +238 -0
  22. package/.agents/rules/gherkin-standards.md +146 -0
  23. package/.agents/rules/git-conventions.md +146 -0
  24. package/.agents/rules/orchestration-error-handling.md +35 -0
  25. package/.agents/rules/security-baseline.md +92 -0
  26. package/.agents/rules/shell-conventions.md +70 -0
  27. package/.agents/rules/test-seams.md +59 -0
  28. package/.agents/rules/testing-standards.md +177 -0
  29. package/.agents/runtime-deps.json +18 -0
  30. package/.agents/schemas/acceptance-eval-verdict.schema.json +93 -0
  31. package/.agents/schemas/agentrc.schema.json +1583 -0
  32. package/.agents/schemas/audit-results.schema.json +69 -0
  33. package/.agents/schemas/audit-rules.json +134 -0
  34. package/.agents/schemas/audit-rules.schema.json +69 -0
  35. package/.agents/schemas/baselines/baseline-envelope.schema.json +44 -0
  36. package/.agents/schemas/baselines/bundle-size.schema.json +47 -0
  37. package/.agents/schemas/baselines/coverage.schema.json +50 -0
  38. package/.agents/schemas/baselines/crap.schema.json +52 -0
  39. package/.agents/schemas/baselines/duplication.schema.json +62 -0
  40. package/.agents/schemas/baselines/lighthouse.schema.json +59 -0
  41. package/.agents/schemas/baselines/lint.schema.json +47 -0
  42. package/.agents/schemas/baselines/maintainability.schema.json +71 -0
  43. package/.agents/schemas/baselines/mutation.schema.json +52 -0
  44. package/.agents/schemas/crap-baseline.schema.json +57 -0
  45. package/.agents/schemas/crap-report.schema.json +102 -0
  46. package/.agents/schemas/dispatch-manifest.json +232 -0
  47. package/.agents/schemas/epic-perf-report.schema.json +89 -0
  48. package/.agents/schemas/epic-spec.schema.json +183 -0
  49. package/.agents/schemas/friction-event.schema.json +56 -0
  50. package/.agents/schemas/lifecycle/README.md +18 -0
  51. package/.agents/schemas/lifecycle/acceptance.reconcile.failed.schema.json +13 -0
  52. package/.agents/schemas/lifecycle/acceptance.reconcile.ok.schema.json +13 -0
  53. package/.agents/schemas/lifecycle/acceptance.reconcile.skipped.schema.json +13 -0
  54. package/.agents/schemas/lifecycle/acceptance.reconcile.start.schema.json +12 -0
  55. package/.agents/schemas/lifecycle/acceptance.reconcile.waived.schema.json +13 -0
  56. package/.agents/schemas/lifecycle/checkpoint.written.schema.json +13 -0
  57. package/.agents/schemas/lifecycle/close-validate.end.schema.json +18 -0
  58. package/.agents/schemas/lifecycle/close-validate.start.schema.json +13 -0
  59. package/.agents/schemas/lifecycle/code-review.end.schema.json +30 -0
  60. package/.agents/schemas/lifecycle/code-review.start.schema.json +12 -0
  61. package/.agents/schemas/lifecycle/epic.automerge.end.schema.json +14 -0
  62. package/.agents/schemas/lifecycle/epic.automerge.start.schema.json +13 -0
  63. package/.agents/schemas/lifecycle/epic.blocked.schema.json +13 -0
  64. package/.agents/schemas/lifecycle/epic.cleanup.end.schema.json +12 -0
  65. package/.agents/schemas/lifecycle/epic.cleanup.start.schema.json +12 -0
  66. package/.agents/schemas/lifecycle/epic.close.end.schema.json +12 -0
  67. package/.agents/schemas/lifecycle/epic.complete.schema.json +13 -0
  68. package/.agents/schemas/lifecycle/epic.finalize.end.schema.json +13 -0
  69. package/.agents/schemas/lifecycle/epic.finalize.start.schema.json +12 -0
  70. package/.agents/schemas/lifecycle/epic.merge.armed.schema.json +13 -0
  71. package/.agents/schemas/lifecycle/epic.merge.blocked.schema.json +14 -0
  72. package/.agents/schemas/lifecycle/epic.merge.confirmed.schema.json +17 -0
  73. package/.agents/schemas/lifecycle/epic.merge.ready.schema.json +15 -0
  74. package/.agents/schemas/lifecycle/epic.plan.end.schema.json +18 -0
  75. package/.agents/schemas/lifecycle/epic.plan.start.schema.json +12 -0
  76. package/.agents/schemas/lifecycle/epic.snapshot.end.schema.json +16 -0
  77. package/.agents/schemas/lifecycle/epic.snapshot.start.schema.json +12 -0
  78. package/.agents/schemas/lifecycle/epic.watch.end.schema.json +28 -0
  79. package/.agents/schemas/lifecycle/epic.watch.start.schema.json +16 -0
  80. package/.agents/schemas/lifecycle/intervention.recorded.schema.json +15 -0
  81. package/.agents/schemas/lifecycle/ledger-record.schema.json +59 -0
  82. package/.agents/schemas/lifecycle/notification.emitted.schema.json +18 -0
  83. package/.agents/schemas/lifecycle/pr.created.schema.json +14 -0
  84. package/.agents/schemas/lifecycle/retro.end.schema.json +16 -0
  85. package/.agents/schemas/lifecycle/retro.start.schema.json +12 -0
  86. package/.agents/schemas/lifecycle/story.blocked.schema.json +13 -0
  87. package/.agents/schemas/lifecycle/story.dispatch.end.schema.json +17 -0
  88. package/.agents/schemas/lifecycle/story.dispatch.start.schema.json +15 -0
  89. package/.agents/schemas/lifecycle/story.heartbeat.schema.json +20 -0
  90. package/.agents/schemas/lifecycle/story.merged.schema.json +13 -0
  91. package/.agents/schemas/mi-report.schema.json +58 -0
  92. package/.agents/schemas/model-attribution.schema.json +49 -0
  93. package/.agents/schemas/qa-finding.schema.json +133 -0
  94. package/.agents/schemas/qa-ledger.schema.json +89 -0
  95. package/.agents/schemas/risk-verdict.schema.json +53 -0
  96. package/.agents/schemas/signal-event.schema.json +58 -0
  97. package/.agents/schemas/skill.schema.json +31 -0
  98. package/.agents/schemas/skills-index.schema.json +81 -0
  99. package/.agents/schemas/story-perf-summary.schema.json +73 -0
  100. package/.agents/schemas/validation-evidence.schema.json +78 -0
  101. package/.agents/scripts/README.md +93 -0
  102. package/.agents/scripts/acceptance-eval.js +284 -0
  103. package/.agents/scripts/acceptance-spec-reconciler.js +556 -0
  104. package/.agents/scripts/agents-bootstrap-github.js +634 -0
  105. package/.agents/scripts/analyze-execution.js +369 -0
  106. package/.agents/scripts/assert-branch.js +83 -0
  107. package/.agents/scripts/audit-labels-bootstrap.js +253 -0
  108. package/.agents/scripts/audit-to-stories.js +257 -0
  109. package/.agents/scripts/bootstrap.js +1378 -0
  110. package/.agents/scripts/check-baselines.js +81 -0
  111. package/.agents/scripts/check-dead-exports.js +311 -0
  112. package/.agents/scripts/check-doc-links.js +401 -0
  113. package/.agents/scripts/check-gherkin-placeholders.js +663 -0
  114. package/.agents/scripts/check-lifecycle-doc-drift.js +402 -0
  115. package/.agents/scripts/check-lifecycle-lint.js +379 -0
  116. package/.agents/scripts/check-prepush-recovery.js +90 -0
  117. package/.agents/scripts/check-windows-git-perf.js +138 -0
  118. package/.agents/scripts/cleanup-repo-test-temp.js +67 -0
  119. package/.agents/scripts/coverage-capture.js +112 -0
  120. package/.agents/scripts/detect-merges.js +111 -0
  121. package/.agents/scripts/diagnose-friction.js +257 -0
  122. package/.agents/scripts/diagnose.js +240 -0
  123. package/.agents/scripts/dispatcher.js +295 -0
  124. package/.agents/scripts/drain-pending-cleanup.js +147 -0
  125. package/.agents/scripts/epic-audit-prepare.js +419 -0
  126. package/.agents/scripts/epic-audit-recheck.js +241 -0
  127. package/.agents/scripts/epic-deliver-note-intervention.js +192 -0
  128. package/.agents/scripts/epic-deliver-preflight.js +407 -0
  129. package/.agents/scripts/epic-deliver-prepare.js +383 -0
  130. package/.agents/scripts/epic-execute-record-wave.js +463 -0
  131. package/.agents/scripts/epic-plan-clarity.js +201 -0
  132. package/.agents/scripts/epic-plan-decompose.js +79 -0
  133. package/.agents/scripts/epic-plan-healthcheck.js +363 -0
  134. package/.agents/scripts/epic-plan-spec-validate.js +111 -0
  135. package/.agents/scripts/epic-plan-spec.js +198 -0
  136. package/.agents/scripts/epic-reconcile.js +637 -0
  137. package/.agents/scripts/evidence-gate.js +235 -0
  138. package/.agents/scripts/generate-config-docs.js +516 -0
  139. package/.agents/scripts/generate-lifecycle-docs.js +224 -0
  140. package/.agents/scripts/generate-skills-index.js +252 -0
  141. package/.agents/scripts/generate-workflows-doc.js +168 -0
  142. package/.agents/scripts/git-cleanup.js +124 -0
  143. package/.agents/scripts/git-pr-quality-gate.js +203 -0
  144. package/.agents/scripts/git-rebase-and-resolve.js +234 -0
  145. package/.agents/scripts/hierarchy-gate.js +176 -0
  146. package/.agents/scripts/hydrate-context.js +179 -0
  147. package/.agents/scripts/install-matrix-assert.js +282 -0
  148. package/.agents/scripts/lib/Graph.js +326 -0
  149. package/.agents/scripts/lib/ITicketingProvider.js +349 -0
  150. package/.agents/scripts/lib/Logger.js +194 -0
  151. package/.agents/scripts/lib/audit-suite/cli.js +64 -0
  152. package/.agents/scripts/lib/audit-suite/findings.js +164 -0
  153. package/.agents/scripts/lib/audit-suite/frontmatter-lint.js +32 -0
  154. package/.agents/scripts/lib/audit-suite/frontmatter.js +110 -0
  155. package/.agents/scripts/lib/audit-suite/index.js +22 -0
  156. package/.agents/scripts/lib/audit-suite/runner.js +233 -0
  157. package/.agents/scripts/lib/audit-suite/selector.js +235 -0
  158. package/.agents/scripts/lib/audit-suite/substitutions.js +124 -0
  159. package/.agents/scripts/lib/audit-suite/workflow-loader.js +49 -0
  160. package/.agents/scripts/lib/audit-to-stories/build-story-body.js +130 -0
  161. package/.agents/scripts/lib/audit-to-stories/dedupe-against-github.js +114 -0
  162. package/.agents/scripts/lib/audit-to-stories/finding-adapter.js +93 -0
  163. package/.agents/scripts/lib/audit-to-stories/group-findings.js +265 -0
  164. package/.agents/scripts/lib/audit-to-stories/parse-audit-md.js +246 -0
  165. package/.agents/scripts/lib/audit-to-stories/seed-epic-from-findings.js +160 -0
  166. package/.agents/scripts/lib/auto-refresh-baselines.js +308 -0
  167. package/.agents/scripts/lib/baseline-loader.js +0 -0
  168. package/.agents/scripts/lib/baseline-schema-registry.js +69 -0
  169. package/.agents/scripts/lib/baseline-snapshot.js +716 -0
  170. package/.agents/scripts/lib/baselines/component-matcher.js +21 -0
  171. package/.agents/scripts/lib/baselines/components.js +126 -0
  172. package/.agents/scripts/lib/baselines/diff-scope-cli.js +203 -0
  173. package/.agents/scripts/lib/baselines/duplication-scanner.js +220 -0
  174. package/.agents/scripts/lib/baselines/env-overrides.js +129 -0
  175. package/.agents/scripts/lib/baselines/envelope.js +368 -0
  176. package/.agents/scripts/lib/baselines/exit-codes.js +89 -0
  177. package/.agents/scripts/lib/baselines/git-base.js +0 -0
  178. package/.agents/scripts/lib/baselines/kernel.js +111 -0
  179. package/.agents/scripts/lib/baselines/kinds/_shared-metric.js +220 -0
  180. package/.agents/scripts/lib/baselines/kinds/bundle-size.js +157 -0
  181. package/.agents/scripts/lib/baselines/kinds/coverage.js +194 -0
  182. package/.agents/scripts/lib/baselines/kinds/crap.js +555 -0
  183. package/.agents/scripts/lib/baselines/kinds/duplication.js +197 -0
  184. package/.agents/scripts/lib/baselines/kinds/lighthouse.js +185 -0
  185. package/.agents/scripts/lib/baselines/kinds/lint.js +172 -0
  186. package/.agents/scripts/lib/baselines/kinds/maintainability.js +340 -0
  187. package/.agents/scripts/lib/baselines/kinds/mutation.js +153 -0
  188. package/.agents/scripts/lib/baselines/path-canon.js +279 -0
  189. package/.agents/scripts/lib/baselines/preview-gates.js +298 -0
  190. package/.agents/scripts/lib/baselines/reader.js +321 -0
  191. package/.agents/scripts/lib/baselines/refresh-service.js +733 -0
  192. package/.agents/scripts/lib/baselines/scope.js +291 -0
  193. package/.agents/scripts/lib/baselines/writer.js +312 -0
  194. package/.agents/scripts/lib/bdd-runner-detect.js +417 -0
  195. package/.agents/scripts/lib/bdd-scenario-scanner.js +310 -0
  196. package/.agents/scripts/lib/bootstrap/baselines-layout-migration.js +202 -0
  197. package/.agents/scripts/lib/bootstrap/branch-protection.js +222 -0
  198. package/.agents/scripts/lib/bootstrap/ci-workflow-template.js +171 -0
  199. package/.agents/scripts/lib/bootstrap/commit-push.js +146 -0
  200. package/.agents/scripts/lib/bootstrap/gh-list.js +153 -0
  201. package/.agents/scripts/lib/bootstrap/gh-preflight.js +306 -0
  202. package/.agents/scripts/lib/bootstrap/hitl-confirm.js +89 -0
  203. package/.agents/scripts/lib/bootstrap/install-ledger.js +174 -0
  204. package/.agents/scripts/lib/bootstrap/manifest.js +272 -0
  205. package/.agents/scripts/lib/bootstrap/merge-methods.js +108 -0
  206. package/.agents/scripts/lib/bootstrap/preflight.js +195 -0
  207. package/.agents/scripts/lib/bootstrap/project-bootstrap.js +801 -0
  208. package/.agents/scripts/lib/bootstrap/prompt.js +480 -0
  209. package/.agents/scripts/lib/bootstrap/quality-bootstrap.js +370 -0
  210. package/.agents/scripts/lib/bootstrap/summary.js +75 -0
  211. package/.agents/scripts/lib/bootstrap/workflow-audit.js +256 -0
  212. package/.agents/scripts/lib/branch-name-guard.js +98 -0
  213. package/.agents/scripts/lib/c8-cli-path.js +21 -0
  214. package/.agents/scripts/lib/changed-files.js +184 -0
  215. package/.agents/scripts/lib/checks/baseline-drift-main-checkout.js +104 -0
  216. package/.agents/scripts/lib/checks/core-bare-clean.js +48 -0
  217. package/.agents/scripts/lib/checks/epic-merge-lock-stale.js +54 -0
  218. package/.agents/scripts/lib/checks/index.js +288 -0
  219. package/.agents/scripts/lib/checks/push-hook-parity.js +106 -0
  220. package/.agents/scripts/lib/checks/stale-origin-epic.js +49 -0
  221. package/.agents/scripts/lib/checks/state.js +558 -0
  222. package/.agents/scripts/lib/checks/story-init-not-backgrounded.js +186 -0
  223. package/.agents/scripts/lib/checks/subagent-agent-tool-required.js +182 -0
  224. package/.agents/scripts/lib/checks/windows-coverage-noise-floor.js +92 -0
  225. package/.agents/scripts/lib/checks/worktree-bootstrap-env.js +81 -0
  226. package/.agents/scripts/lib/checks/worktree-residue-biome.js +55 -0
  227. package/.agents/scripts/lib/cli/parse-numeric.js +60 -0
  228. package/.agents/scripts/lib/cli/standard-args.js +351 -0
  229. package/.agents/scripts/lib/cli-args.js +286 -0
  230. package/.agents/scripts/lib/cli-utils.js +69 -0
  231. package/.agents/scripts/lib/close-validation/projections/head-sha.js +44 -0
  232. package/.agents/scripts/lib/close-validation/projections/inputs.js +86 -0
  233. package/.agents/scripts/lib/close-validation/projections/maintainability.js +286 -0
  234. package/.agents/scripts/lib/close-validation.js +897 -0
  235. package/.agents/scripts/lib/codebase-snapshot.js +513 -0
  236. package/.agents/scripts/lib/command-header.js +33 -0
  237. package/.agents/scripts/lib/config/acceptance-eval.js +95 -0
  238. package/.agents/scripts/lib/config/baselines.js +60 -0
  239. package/.agents/scripts/lib/config/ci.js +30 -0
  240. package/.agents/scripts/lib/config/commands.js +36 -0
  241. package/.agents/scripts/lib/config/defaults.js +119 -0
  242. package/.agents/scripts/lib/config/explain.js +348 -0
  243. package/.agents/scripts/lib/config/gates/bundle-size.schema.js +23 -0
  244. package/.agents/scripts/lib/config/gates/coverage.schema.js +18 -0
  245. package/.agents/scripts/lib/config/gates/crap.schema.js +33 -0
  246. package/.agents/scripts/lib/config/gates/duplication.schema.js +26 -0
  247. package/.agents/scripts/lib/config/gates/index.js +36 -0
  248. package/.agents/scripts/lib/config/gates/lighthouse.schema.js +23 -0
  249. package/.agents/scripts/lib/config/gates/lint.schema.js +9 -0
  250. package/.agents/scripts/lib/config/gates/maintainability.schema.js +20 -0
  251. package/.agents/scripts/lib/config/gates/mutation.schema.js +12 -0
  252. package/.agents/scripts/lib/config/gates/shared.js +117 -0
  253. package/.agents/scripts/lib/config/github.js +122 -0
  254. package/.agents/scripts/lib/config/lifecycle.js +40 -0
  255. package/.agents/scripts/lib/config/limits.js +211 -0
  256. package/.agents/scripts/lib/config/paths.js +73 -0
  257. package/.agents/scripts/lib/config/preflight.js +58 -0
  258. package/.agents/scripts/lib/config/quality.js +665 -0
  259. package/.agents/scripts/lib/config/retro.js +77 -0
  260. package/.agents/scripts/lib/config/runners.js +105 -0
  261. package/.agents/scripts/lib/config/runtime.js +167 -0
  262. package/.agents/scripts/lib/config/shared.js +46 -0
  263. package/.agents/scripts/lib/config/sync-agentrc.js +243 -0
  264. package/.agents/scripts/lib/config/temp-paths.js +373 -0
  265. package/.agents/scripts/lib/config/validate-orchestration.js +81 -0
  266. package/.agents/scripts/lib/config/worktree-isolation.js +80 -0
  267. package/.agents/scripts/lib/config-resolver.js +298 -0
  268. package/.agents/scripts/lib/config-schema-shared.js +32 -0
  269. package/.agents/scripts/lib/config-schema.js +20 -0
  270. package/.agents/scripts/lib/config-settings-schema-delivery.js +332 -0
  271. package/.agents/scripts/lib/config-settings-schema-quality.js +165 -0
  272. package/.agents/scripts/lib/config-settings-schema.js +420 -0
  273. package/.agents/scripts/lib/coverage-baseline.js +352 -0
  274. package/.agents/scripts/lib/coverage-capture.js +195 -0
  275. package/.agents/scripts/lib/coverage-utils.js +239 -0
  276. package/.agents/scripts/lib/cpu-pool.js +223 -0
  277. package/.agents/scripts/lib/crap-engine.js +119 -0
  278. package/.agents/scripts/lib/crap-utils.js +479 -0
  279. package/.agents/scripts/lib/degraded-mode.js +69 -0
  280. package/.agents/scripts/lib/dependency-parser.js +129 -0
  281. package/.agents/scripts/lib/duplicate-search.js +189 -0
  282. package/.agents/scripts/lib/dynamic-workflow/architecture-report-contract.js +70 -0
  283. package/.agents/scripts/lib/dynamic-workflow/audit-orchestrator.js +197 -0
  284. package/.agents/scripts/lib/dynamic-workflow/capability.js +396 -0
  285. package/.agents/scripts/lib/dynamic-workflow/clean-code-report-contract.js +80 -0
  286. package/.agents/scripts/lib/dynamic-workflow/performance-report-contract.js +72 -0
  287. package/.agents/scripts/lib/dynamic-workflow/quality-report-contract.js +90 -0
  288. package/.agents/scripts/lib/dynamic-workflow/report-contract-core.js +43 -0
  289. package/.agents/scripts/lib/dynamic-workflow/security-report-contract.js +83 -0
  290. package/.agents/scripts/lib/env-loader.js +52 -0
  291. package/.agents/scripts/lib/epic-merge-lock.js +239 -0
  292. package/.agents/scripts/lib/epic-plan-clarity.js +142 -0
  293. package/.agents/scripts/lib/epic-plan-ideation.js +228 -0
  294. package/.agents/scripts/lib/error-redactor.js +125 -0
  295. package/.agents/scripts/lib/errors/index.js +67 -0
  296. package/.agents/scripts/lib/feedback-loop/audit-results-graduator.js +230 -0
  297. package/.agents/scripts/lib/feedback-loop/code-review-graduator.js +207 -0
  298. package/.agents/scripts/lib/feedback-loop/graduator-core.js +421 -0
  299. package/.agents/scripts/lib/feedback-loop/memory-freshness.js +480 -0
  300. package/.agents/scripts/lib/feedback-loop/prior-feedback-fetcher.js +229 -0
  301. package/.agents/scripts/lib/findings/classify-finding.js +195 -0
  302. package/.agents/scripts/lib/findings/promote-finding.js +353 -0
  303. package/.agents/scripts/lib/findings/route-finding.js +283 -0
  304. package/.agents/scripts/lib/findings/semantic-issue-search.js +179 -0
  305. package/.agents/scripts/lib/findings/severity.js +102 -0
  306. package/.agents/scripts/lib/gates/baseline-store.js +106 -0
  307. package/.agents/scripts/lib/gates/friction.js +43 -0
  308. package/.agents/scripts/lib/gh-exec.js +553 -0
  309. package/.agents/scripts/lib/git/cached-fetch.js +0 -0
  310. package/.agents/scripts/lib/git/sync-from-base.js +162 -0
  311. package/.agents/scripts/lib/git-branch-cleanup.js +213 -0
  312. package/.agents/scripts/lib/git-branch-lifecycle.js +353 -0
  313. package/.agents/scripts/lib/git-merge-orchestrator.js +261 -0
  314. package/.agents/scripts/lib/git-utils.js +363 -0
  315. package/.agents/scripts/lib/github-url.js +29 -0
  316. package/.agents/scripts/lib/install-cmd-parser.js +51 -0
  317. package/.agents/scripts/lib/issue-link-parser.js +74 -0
  318. package/.agents/scripts/lib/json-utils.js +60 -0
  319. package/.agents/scripts/lib/label-constants.js +169 -0
  320. package/.agents/scripts/lib/label-taxonomy.js +200 -0
  321. package/.agents/scripts/lib/maintainability-engine.js +164 -0
  322. package/.agents/scripts/lib/maintainability-utils.js +343 -0
  323. package/.agents/scripts/lib/mandrel-catalog.js +170 -0
  324. package/.agents/scripts/lib/mutation/baseline-snapshot.js +238 -0
  325. package/.agents/scripts/lib/mutation/config-detector.js +119 -0
  326. package/.agents/scripts/lib/mutation/stryker-runner.js +306 -0
  327. package/.agents/scripts/lib/mutation/survivor-report.js +160 -0
  328. package/.agents/scripts/lib/notifications/notifier.js +75 -0
  329. package/.agents/scripts/lib/observability/active-story-env.js +182 -0
  330. package/.agents/scripts/lib/observability/baseline-refresh-rate.js +221 -0
  331. package/.agents/scripts/lib/observability/perf-aggregator.js +887 -0
  332. package/.agents/scripts/lib/observability/perf-report-readers.js +319 -0
  333. package/.agents/scripts/lib/observability/perf-report-render.js +182 -0
  334. package/.agents/scripts/lib/observability/signals-writer.js +296 -0
  335. package/.agents/scripts/lib/observability/source-classifier.js +103 -0
  336. package/.agents/scripts/lib/observability/tool-trace-hook.js +417 -0
  337. package/.agents/scripts/lib/onboard/detect-stack.js +300 -0
  338. package/.agents/scripts/lib/onboard/scaffold-docs.js +128 -0
  339. package/.agents/scripts/lib/orchestration/acceptance-eval-decision.js +173 -0
  340. package/.agents/scripts/lib/orchestration/cascade-grouping.js +275 -0
  341. package/.agents/scripts/lib/orchestration/check-baselines/phases/compare.js +131 -0
  342. package/.agents/scripts/lib/orchestration/check-baselines/phases/evaluate.js +80 -0
  343. package/.agents/scripts/lib/orchestration/check-baselines/phases/floors.js +132 -0
  344. package/.agents/scripts/lib/orchestration/check-baselines/phases/friction.js +142 -0
  345. package/.agents/scripts/lib/orchestration/check-baselines/phases/parse-args.js +149 -0
  346. package/.agents/scripts/lib/orchestration/check-baselines/phases/pipeline.js +158 -0
  347. package/.agents/scripts/lib/orchestration/check-baselines/phases/report.js +56 -0
  348. package/.agents/scripts/lib/orchestration/code-review.js +652 -0
  349. package/.agents/scripts/lib/orchestration/column-sync.js +286 -0
  350. package/.agents/scripts/lib/orchestration/context-envelope.js +280 -0
  351. package/.agents/scripts/lib/orchestration/context-hydration-engine.js +581 -0
  352. package/.agents/scripts/lib/orchestration/dependency-analyzer.js +88 -0
  353. package/.agents/scripts/lib/orchestration/detectors-phase.js +188 -0
  354. package/.agents/scripts/lib/orchestration/dispatch-engine.js +144 -0
  355. package/.agents/scripts/lib/orchestration/dispatch-pipeline.js +206 -0
  356. package/.agents/scripts/lib/orchestration/doc-reader.js +94 -0
  357. package/.agents/scripts/lib/orchestration/epic-cleanup.js +473 -0
  358. package/.agents/scripts/lib/orchestration/epic-deliver-lease-guard.js +310 -0
  359. package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/cli.js +167 -0
  360. package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/context.js +151 -0
  361. package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/creation.js +74 -0
  362. package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/dag.js +78 -0
  363. package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/diagnostics.js +72 -0
  364. package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/persist-helpers.js +155 -0
  365. package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/persist.js +321 -0
  366. package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/planning-artifacts.js +75 -0
  367. package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/reconcile-spawn.js +86 -0
  368. package/.agents/scripts/lib/orchestration/epic-plan-lease-guard.js +235 -0
  369. package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/authoring-context.js +197 -0
  370. package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/cli-args.js +48 -0
  371. package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/drain.js +94 -0
  372. package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/plan-epic.js +414 -0
  373. package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/prompts.js +55 -0
  374. package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/risk-verdict.js +105 -0
  375. package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/run-spec-phase.js +235 -0
  376. package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/spec-freshness.js +120 -0
  377. package/.agents/scripts/lib/orchestration/epic-plan-state-store.js +118 -0
  378. package/.agents/scripts/lib/orchestration/epic-run-state-store.js +295 -0
  379. package/.agents/scripts/lib/orchestration/epic-runner/concurrency-gate.js +186 -0
  380. package/.agents/scripts/lib/orchestration/epic-runner/deliver-phases.js +50 -0
  381. package/.agents/scripts/lib/orchestration/epic-runner/phases/build-wave-dag.js +146 -0
  382. package/.agents/scripts/lib/orchestration/epic-runner/phases/snapshot.js +110 -0
  383. package/.agents/scripts/lib/orchestration/epic-runner/progress-reporter/composition.js +392 -0
  384. package/.agents/scripts/lib/orchestration/epic-runner/progress-reporter/signals.js +217 -0
  385. package/.agents/scripts/lib/orchestration/epic-runner/progress-reporter/transport.js +235 -0
  386. package/.agents/scripts/lib/orchestration/epic-runner/progress-reporter.js +69 -0
  387. package/.agents/scripts/lib/orchestration/epic-runner/progress-signals/_bullet-format.js +32 -0
  388. package/.agents/scripts/lib/orchestration/epic-runner/progress-signals/crap-drift.js +291 -0
  389. package/.agents/scripts/lib/orchestration/epic-runner/progress-signals/maintainability-drift.js +175 -0
  390. package/.agents/scripts/lib/orchestration/epic-runner/progress-signals/stalled-worktree.js +37 -0
  391. package/.agents/scripts/lib/orchestration/epic-runner/story-launcher.js +127 -0
  392. package/.agents/scripts/lib/orchestration/epic-runner/story-run-progress-writer.js +400 -0
  393. package/.agents/scripts/lib/orchestration/epic-runner/sub-agent-return.js +285 -0
  394. package/.agents/scripts/lib/orchestration/epic-runner/wave-scheduler.js +66 -0
  395. package/.agents/scripts/lib/orchestration/epic-spec-reconciler-apply.js +797 -0
  396. package/.agents/scripts/lib/orchestration/epic-spec-reconciler-diff.js +619 -0
  397. package/.agents/scripts/lib/orchestration/epic-spec-reconciler-discriminator.js +335 -0
  398. package/.agents/scripts/lib/orchestration/epic-spec-reconciler-format.js +230 -0
  399. package/.agents/scripts/lib/orchestration/epic-spec-reconciler-ops.js +363 -0
  400. package/.agents/scripts/lib/orchestration/error-journal.js +139 -0
  401. package/.agents/scripts/lib/orchestration/file-assumption-enum.js +31 -0
  402. package/.agents/scripts/lib/orchestration/file-assumptions.js +506 -0
  403. package/.agents/scripts/lib/orchestration/finalize/close-planning-tickets.js +116 -0
  404. package/.agents/scripts/lib/orchestration/finalize/open-or-locate-pr.js +241 -0
  405. package/.agents/scripts/lib/orchestration/finalize/post-handoff-comment.js +489 -0
  406. package/.agents/scripts/lib/orchestration/finalize/sanitize-skip-ci.js +88 -0
  407. package/.agents/scripts/lib/orchestration/git-cleanup/phases/branches-reap.js +219 -0
  408. package/.agents/scripts/lib/orchestration/git-cleanup/phases/branches.js +309 -0
  409. package/.agents/scripts/lib/orchestration/git-cleanup/phases/cli.js +99 -0
  410. package/.agents/scripts/lib/orchestration/git-cleanup/phases/fast-forward.js +123 -0
  411. package/.agents/scripts/lib/orchestration/git-cleanup/phases/filters.js +57 -0
  412. package/.agents/scripts/lib/orchestration/git-cleanup/phases/git-probes-ff.js +114 -0
  413. package/.agents/scripts/lib/orchestration/git-cleanup/phases/git-probes.js +426 -0
  414. package/.agents/scripts/lib/orchestration/git-cleanup/phases/parse-args.js +84 -0
  415. package/.agents/scripts/lib/orchestration/git-cleanup/phases/phase-drivers.js +365 -0
  416. package/.agents/scripts/lib/orchestration/git-cleanup/phases/prompts.js +72 -0
  417. package/.agents/scripts/lib/orchestration/git-cleanup/phases/prune.js +69 -0
  418. package/.agents/scripts/lib/orchestration/git-cleanup/phases/render.js +214 -0
  419. package/.agents/scripts/lib/orchestration/git-cleanup/phases/stashes.js +137 -0
  420. package/.agents/scripts/lib/orchestration/label-transitions.js +43 -0
  421. package/.agents/scripts/lib/orchestration/lifecycle/bus.js +309 -0
  422. package/.agents/scripts/lib/orchestration/lifecycle/emit-story-dispatch-end.js +147 -0
  423. package/.agents/scripts/lib/orchestration/lifecycle/emit-story-heartbeat.js +155 -0
  424. package/.agents/scripts/lib/orchestration/lifecycle/ledger-writer.js +226 -0
  425. package/.agents/scripts/lib/orchestration/lifecycle/listeners/README.md +69 -0
  426. package/.agents/scripts/lib/orchestration/lifecycle/listeners/acceptance-reconciler.js +378 -0
  427. package/.agents/scripts/lib/orchestration/lifecycle/listeners/automerge-armer.js +248 -0
  428. package/.agents/scripts/lib/orchestration/lifecycle/listeners/automerge-predicate.js +527 -0
  429. package/.agents/scripts/lib/orchestration/lifecycle/listeners/branch-cleaner.js +259 -0
  430. package/.agents/scripts/lib/orchestration/lifecycle/listeners/checkpoint-pointer-writer.js +278 -0
  431. package/.agents/scripts/lib/orchestration/lifecycle/listeners/cleaner.js +355 -0
  432. package/.agents/scripts/lib/orchestration/lifecycle/listeners/finalizer.js +647 -0
  433. package/.agents/scripts/lib/orchestration/lifecycle/listeners/index.js +331 -0
  434. package/.agents/scripts/lib/orchestration/lifecycle/listeners/intervention-recorder.js +140 -0
  435. package/.agents/scripts/lib/orchestration/lifecycle/listeners/merge-watcher.js +421 -0
  436. package/.agents/scripts/lib/orchestration/lifecycle/listeners/notify-dispatcher.js +168 -0
  437. package/.agents/scripts/lib/orchestration/lifecycle/listeners/watcher.js +668 -0
  438. package/.agents/scripts/lib/orchestration/lifecycle/trace-logger.js +322 -0
  439. package/.agents/scripts/lib/orchestration/lint-baseline-service.js +114 -0
  440. package/.agents/scripts/lib/orchestration/manifest-builder.js +216 -0
  441. package/.agents/scripts/lib/orchestration/model-attribution.js +390 -0
  442. package/.agents/scripts/lib/orchestration/parked-follow-ons.js +147 -0
  443. package/.agents/scripts/lib/orchestration/phase-runner.js +87 -0
  444. package/.agents/scripts/lib/orchestration/plan-review-routing.js +63 -0
  445. package/.agents/scripts/lib/orchestration/plan-runner/plan-router.js +86 -0
  446. package/.agents/scripts/lib/orchestration/plan-runner/worktree-sweep.js +212 -0
  447. package/.agents/scripts/lib/orchestration/planning-context-budget.js +213 -0
  448. package/.agents/scripts/lib/orchestration/planning-risk.js +155 -0
  449. package/.agents/scripts/lib/orchestration/planning-state-manager.js +318 -0
  450. package/.agents/scripts/lib/orchestration/post-merge/phases/branch-cleanup.js +56 -0
  451. package/.agents/scripts/lib/orchestration/post-merge/phases/dashboard-refresh.js +33 -0
  452. package/.agents/scripts/lib/orchestration/post-merge/phases/notification.js +78 -0
  453. package/.agents/scripts/lib/orchestration/post-merge/phases/temp-cleanup.js +68 -0
  454. package/.agents/scripts/lib/orchestration/post-merge/phases/ticket-closure.js +118 -0
  455. package/.agents/scripts/lib/orchestration/post-merge/phases/worktree-reap.js +396 -0
  456. package/.agents/scripts/lib/orchestration/post-merge-pipeline.js +205 -0
  457. package/.agents/scripts/lib/orchestration/pr-base-guard.js +47 -0
  458. package/.agents/scripts/lib/orchestration/preflight-cache.js +164 -0
  459. package/.agents/scripts/lib/orchestration/reassert-status-column.js +202 -0
  460. package/.agents/scripts/lib/orchestration/reconciler.js +137 -0
  461. package/.agents/scripts/lib/orchestration/recurring-failure-detector.js +152 -0
  462. package/.agents/scripts/lib/orchestration/recut.js +56 -0
  463. package/.agents/scripts/lib/orchestration/resolves-token.js +127 -0
  464. package/.agents/scripts/lib/orchestration/retro/phases/checks.js +94 -0
  465. package/.agents/scripts/lib/orchestration/retro/phases/compose-body.js +448 -0
  466. package/.agents/scripts/lib/orchestration/retro/phases/gather-signals.js +335 -0
  467. package/.agents/scripts/lib/orchestration/retro/phases/post-and-mirror.js +133 -0
  468. package/.agents/scripts/lib/orchestration/retro-heuristics.js +57 -0
  469. package/.agents/scripts/lib/orchestration/retro-perf-heuristics.js +275 -0
  470. package/.agents/scripts/lib/orchestration/retro-proposals.js +395 -0
  471. package/.agents/scripts/lib/orchestration/retro-runner.js +171 -0
  472. package/.agents/scripts/lib/orchestration/review-depth.js +93 -0
  473. package/.agents/scripts/lib/orchestration/review-providers/codex.js +363 -0
  474. package/.agents/scripts/lib/orchestration/review-providers/findings-renderer.js +205 -0
  475. package/.agents/scripts/lib/orchestration/review-providers/native.js +805 -0
  476. package/.agents/scripts/lib/orchestration/review-providers/review-depth.js +73 -0
  477. package/.agents/scripts/lib/orchestration/review-providers/review-provider-factory.js +396 -0
  478. package/.agents/scripts/lib/orchestration/review-providers/security-review.js +373 -0
  479. package/.agents/scripts/lib/orchestration/review-providers/types.js +89 -0
  480. package/.agents/scripts/lib/orchestration/review-providers/ultrareview.js +107 -0
  481. package/.agents/scripts/lib/orchestration/single-story-close/phases/auto-merge.js +159 -0
  482. package/.agents/scripts/lib/orchestration/single-story-close/phases/base-sync.js +194 -0
  483. package/.agents/scripts/lib/orchestration/single-story-close/phases/close-validation.js +81 -0
  484. package/.agents/scripts/lib/orchestration/single-story-close/phases/code-review.js +190 -0
  485. package/.agents/scripts/lib/orchestration/single-story-close/phases/options.js +70 -0
  486. package/.agents/scripts/lib/orchestration/single-story-close/phases/pull-request.js +106 -0
  487. package/.agents/scripts/lib/orchestration/single-story-close/phases/push.js +42 -0
  488. package/.agents/scripts/lib/orchestration/single-story-close/phases/worktree-reap.js +73 -0
  489. package/.agents/scripts/lib/orchestration/single-story-close/phases/wrong-tree-guard.js +225 -0
  490. package/.agents/scripts/lib/orchestration/single-story-close/runner.js +315 -0
  491. package/.agents/scripts/lib/orchestration/single-story-lease-guard.js +149 -0
  492. package/.agents/scripts/lib/orchestration/skill-capsule-loader.js +110 -0
  493. package/.agents/scripts/lib/orchestration/spec-freshness.js +320 -0
  494. package/.agents/scripts/lib/orchestration/spec-renderer.js +456 -0
  495. package/.agents/scripts/lib/orchestration/spec-section-validator.js +80 -0
  496. package/.agents/scripts/lib/orchestration/story-close/auto-refresh-runner.js +797 -0
  497. package/.agents/scripts/lib/orchestration/story-close/baseline-attribution/phases/gate-failure.js +163 -0
  498. package/.agents/scripts/lib/orchestration/story-close/baseline-attribution/phases/pre-merge-attribution.js +152 -0
  499. package/.agents/scripts/lib/orchestration/story-close/baseline-attribution/phases/refresh-commit.js +387 -0
  500. package/.agents/scripts/lib/orchestration/story-close/baseline-attribution/phases/regression-projection.js +266 -0
  501. package/.agents/scripts/lib/orchestration/story-close/baseline-attribution/phases/scope-discovery.js +48 -0
  502. package/.agents/scripts/lib/orchestration/story-close/baseline-attribution-wiring.js +67 -0
  503. package/.agents/scripts/lib/orchestration/story-close/baseline-attribution.js +161 -0
  504. package/.agents/scripts/lib/orchestration/story-close/baseline-friction-body.js +117 -0
  505. package/.agents/scripts/lib/orchestration/story-close/cd-out-guard.js +86 -0
  506. package/.agents/scripts/lib/orchestration/story-close/cleanup-reconciler.js +147 -0
  507. package/.agents/scripts/lib/orchestration/story-close/close-inputs.js +142 -0
  508. package/.agents/scripts/lib/orchestration/story-close/comment-bodies.js +62 -0
  509. package/.agents/scripts/lib/orchestration/story-close/format-autofix-scoped.js +221 -0
  510. package/.agents/scripts/lib/orchestration/story-close/format-autofix-shared.js +123 -0
  511. package/.agents/scripts/lib/orchestration/story-close/format-autofix.js +216 -0
  512. package/.agents/scripts/lib/orchestration/story-close/merge-runner.js +636 -0
  513. package/.agents/scripts/lib/orchestration/story-close/merge-subject.js +198 -0
  514. package/.agents/scripts/lib/orchestration/story-close/phases/branch-restore.js +105 -0
  515. package/.agents/scripts/lib/orchestration/story-close/phases/close.js +222 -0
  516. package/.agents/scripts/lib/orchestration/story-close/phases/code-review.js +220 -0
  517. package/.agents/scripts/lib/orchestration/story-close/phases/gates.js +291 -0
  518. package/.agents/scripts/lib/orchestration/story-close/phases/locked-pipeline.js +234 -0
  519. package/.agents/scripts/lib/orchestration/story-close/phases/preflight.js +110 -0
  520. package/.agents/scripts/lib/orchestration/story-close/phases/refresh.js +86 -0
  521. package/.agents/scripts/lib/orchestration/story-close/phases/timeout-blocked-emitter.js +112 -0
  522. package/.agents/scripts/lib/orchestration/story-close/phases/timeout-blocked.js +157 -0
  523. package/.agents/scripts/lib/orchestration/story-close/post-merge-close.js +434 -0
  524. package/.agents/scripts/lib/orchestration/story-close/pre-merge-validation.js +290 -0
  525. package/.agents/scripts/lib/orchestration/story-close-recovery.js +643 -0
  526. package/.agents/scripts/lib/orchestration/structured-comment-parser.js +67 -0
  527. package/.agents/scripts/lib/orchestration/task-body-validator.js +391 -0
  528. package/.agents/scripts/lib/orchestration/ticket-lease.js +358 -0
  529. package/.agents/scripts/lib/orchestration/ticket-validator-conflicts.js +783 -0
  530. package/.agents/scripts/lib/orchestration/ticket-validator-sizing.js +367 -0
  531. package/.agents/scripts/lib/orchestration/ticket-validator.js +691 -0
  532. package/.agents/scripts/lib/orchestration/ticketing/bulk.js +723 -0
  533. package/.agents/scripts/lib/orchestration/ticketing/reads.js +474 -0
  534. package/.agents/scripts/lib/orchestration/ticketing/state.js +559 -0
  535. package/.agents/scripts/lib/orchestration/ticketing.js +55 -0
  536. package/.agents/scripts/lib/orchestration/wave-marker.js +28 -0
  537. package/.agents/scripts/lib/orchestration/wave-record-io.js +277 -0
  538. package/.agents/scripts/lib/orchestration/wave-record-notifications.js +189 -0
  539. package/.agents/scripts/lib/orchestration/wave-record-projection.js +423 -0
  540. package/.agents/scripts/lib/path-security.js +25 -0
  541. package/.agents/scripts/lib/plan-phase-cleanup.js +125 -0
  542. package/.agents/scripts/lib/preflight-runner.js +196 -0
  543. package/.agents/scripts/lib/presentation/dispatch-manifest-render.js +95 -0
  544. package/.agents/scripts/lib/presentation/manifest-builder.js +245 -0
  545. package/.agents/scripts/lib/presentation/manifest-formatter.js +243 -0
  546. package/.agents/scripts/lib/presentation/manifest-helpers.js +213 -0
  547. package/.agents/scripts/lib/presentation/manifest-persistence.js +262 -0
  548. package/.agents/scripts/lib/presentation/manifest-procedures.js +55 -0
  549. package/.agents/scripts/lib/presentation/manifest-render-waves.js +252 -0
  550. package/.agents/scripts/lib/presentation/manifest-renderer.js +188 -0
  551. package/.agents/scripts/lib/presentation/manifest-story-views.js +119 -0
  552. package/.agents/scripts/lib/provider-factory.js +80 -0
  553. package/.agents/scripts/lib/push-epic-retry.js +209 -0
  554. package/.agents/scripts/lib/qa/console-allowlist.js +151 -0
  555. package/.agents/scripts/lib/qa/coverage-report.js +181 -0
  556. package/.agents/scripts/lib/qa/coverage-verdict.js +296 -0
  557. package/.agents/scripts/lib/qa/propose-missing-test.js +95 -0
  558. package/.agents/scripts/lib/qa/qa-context-hydrator.js +296 -0
  559. package/.agents/scripts/lib/qa/qa-session.js +197 -0
  560. package/.agents/scripts/lib/qa/redact-evidence.js +245 -0
  561. package/.agents/scripts/lib/qa/resolve-qa-contract.js +190 -0
  562. package/.agents/scripts/lib/qa/resolve-selection.js +373 -0
  563. package/.agents/scripts/lib/runtime-deps/ensure-installed.js +100 -0
  564. package/.agents/scripts/lib/runtime-deps/manifest.js +96 -0
  565. package/.agents/scripts/lib/runtime-deps/preflight.js +78 -0
  566. package/.agents/scripts/lib/runtime-deps/scan-imports.js +202 -0
  567. package/.agents/scripts/lib/signals/detectors/common.js +36 -0
  568. package/.agents/scripts/lib/signals/detectors/hotspot.js +298 -0
  569. package/.agents/scripts/lib/signals/detectors/index.js +14 -0
  570. package/.agents/scripts/lib/signals/detectors/retry.js +289 -0
  571. package/.agents/scripts/lib/signals/detectors/rework.js +204 -0
  572. package/.agents/scripts/lib/signals/index.js +39 -0
  573. package/.agents/scripts/lib/signals/read.js +268 -0
  574. package/.agents/scripts/lib/signals/schema.js +225 -0
  575. package/.agents/scripts/lib/signals/span-tree.js +290 -0
  576. package/.agents/scripts/lib/signals/write.js +19 -0
  577. package/.agents/scripts/lib/single-story/confirm-merge.js +201 -0
  578. package/.agents/scripts/lib/single-story/story-merged-notify.js +126 -0
  579. package/.agents/scripts/lib/single-story-sweep/protection.js +274 -0
  580. package/.agents/scripts/lib/single-story-sweep/sweep-lock.js +169 -0
  581. package/.agents/scripts/lib/single-story-sweep.js +329 -0
  582. package/.agents/scripts/lib/skills/parse-skill.js +202 -0
  583. package/.agents/scripts/lib/skills/walk-skill-files.js +56 -0
  584. package/.agents/scripts/lib/spec/index.js +36 -0
  585. package/.agents/scripts/lib/spec/loader.js +425 -0
  586. package/.agents/scripts/lib/spec/state.js +217 -0
  587. package/.agents/scripts/lib/story-body/story-body.js +743 -0
  588. package/.agents/scripts/lib/story-init/blocker-validator.js +68 -0
  589. package/.agents/scripts/lib/story-init/branch-initializer.js +422 -0
  590. package/.agents/scripts/lib/story-init/context-resolver.js +92 -0
  591. package/.agents/scripts/lib/story-init/donor-precheck.js +207 -0
  592. package/.agents/scripts/lib/story-init/hierarchy-tracer.js +36 -0
  593. package/.agents/scripts/lib/story-init/state-transitioner.js +80 -0
  594. package/.agents/scripts/lib/story-init/task-graph-builder.js +114 -0
  595. package/.agents/scripts/lib/story-init/transition-summary.js +34 -0
  596. package/.agents/scripts/lib/story-lifecycle.js +186 -0
  597. package/.agents/scripts/lib/story-plan.js +246 -0
  598. package/.agents/scripts/lib/task-utils.js +26 -0
  599. package/.agents/scripts/lib/templates/decomposer-prompts.js +168 -0
  600. package/.agents/scripts/lib/test-env.js +30 -0
  601. package/.agents/scripts/lib/test-isolate/env-snapshot-loader.js +52 -0
  602. package/.agents/scripts/lib/test-isolate/list-files.js +90 -0
  603. package/.agents/scripts/lib/test-isolate/parse-tap.js +75 -0
  604. package/.agents/scripts/lib/test-isolate/runner.js +483 -0
  605. package/.agents/scripts/lib/test-profile/parse-tap.js +136 -0
  606. package/.agents/scripts/lib/test-profile/render-report.js +45 -0
  607. package/.agents/scripts/lib/test-reserved-epic-temp-ids.js +35 -0
  608. package/.agents/scripts/lib/test-tiers.js +94 -0
  609. package/.agents/scripts/lib/util/concurrent-map.js +59 -0
  610. package/.agents/scripts/lib/util/phase-timer-state.js +72 -0
  611. package/.agents/scripts/lib/util/phase-timer.js +163 -0
  612. package/.agents/scripts/lib/util/poll-loop.js +86 -0
  613. package/.agents/scripts/lib/util/with-timeout.js +32 -0
  614. package/.agents/scripts/lib/validation-evidence.js +323 -0
  615. package/.agents/scripts/lib/wave-runner/tick.js +665 -0
  616. package/.agents/scripts/lib/wave-runner/wave-checkpoint.js +91 -0
  617. package/.agents/scripts/lib/wave-runner/wave-runner-error.js +19 -0
  618. package/.agents/scripts/lib/workers/crap-worker.js +197 -0
  619. package/.agents/scripts/lib/workers/maintainability-report-worker.js +137 -0
  620. package/.agents/scripts/lib/workers/maintainability-worker.js +79 -0
  621. package/.agents/scripts/lib/workspace-provisioner.js +189 -0
  622. package/.agents/scripts/lib/worktree/bootstrapper.js +48 -0
  623. package/.agents/scripts/lib/worktree/inspector.js +140 -0
  624. package/.agents/scripts/lib/worktree/lifecycle/creation.js +118 -0
  625. package/.agents/scripts/lib/worktree/lifecycle/drift-detection.js +62 -0
  626. package/.agents/scripts/lib/worktree/lifecycle/force-drain.js +276 -0
  627. package/.agents/scripts/lib/worktree/lifecycle/gc.js +49 -0
  628. package/.agents/scripts/lib/worktree/lifecycle/merge-reachability.js +178 -0
  629. package/.agents/scripts/lib/worktree/lifecycle/pending-cleanup.js +264 -0
  630. package/.agents/scripts/lib/worktree/lifecycle/precheck.js +100 -0
  631. package/.agents/scripts/lib/worktree/lifecycle/reap.js +588 -0
  632. package/.agents/scripts/lib/worktree/lifecycle/registry-sync.js +124 -0
  633. package/.agents/scripts/lib/worktree/lifecycle/shared.js +26 -0
  634. package/.agents/scripts/lib/worktree/lifecycle-manager.js +40 -0
  635. package/.agents/scripts/lib/worktree/node-modules-strategy.js +349 -0
  636. package/.agents/scripts/lib/worktree-manager.js +243 -0
  637. package/.agents/scripts/lifecycle-diff.js +206 -0
  638. package/.agents/scripts/lifecycle-emit-story-dispatch.js +194 -0
  639. package/.agents/scripts/lifecycle-emit.js +479 -0
  640. package/.agents/scripts/lint-baseline.js +507 -0
  641. package/.agents/scripts/lint-label-vocabulary.js +237 -0
  642. package/.agents/scripts/loc-delta.js +205 -0
  643. package/.agents/scripts/notify.js +307 -0
  644. package/.agents/scripts/package.json +3 -0
  645. package/.agents/scripts/post-structured-comment.js +127 -0
  646. package/.agents/scripts/pr-watch-with-update.js +152 -0
  647. package/.agents/scripts/providers/github/auth.js +65 -0
  648. package/.agents/scripts/providers/github/board-add.js +63 -0
  649. package/.agents/scripts/providers/github/branch-protection.js +186 -0
  650. package/.agents/scripts/providers/github/cache.js +72 -0
  651. package/.agents/scripts/providers/github/comments.js +131 -0
  652. package/.agents/scripts/providers/github/compose.js +111 -0
  653. package/.agents/scripts/providers/github/errors.js +242 -0
  654. package/.agents/scripts/providers/github/issues.js +242 -0
  655. package/.agents/scripts/providers/github/labels.js +179 -0
  656. package/.agents/scripts/providers/github/mappers.js +126 -0
  657. package/.agents/scripts/providers/github/merge-methods.js +82 -0
  658. package/.agents/scripts/providers/github/project-board.js +47 -0
  659. package/.agents/scripts/providers/github/projects-v2-graphql.js +472 -0
  660. package/.agents/scripts/providers/github/prs.js +103 -0
  661. package/.agents/scripts/providers/github/request-helpers.js +110 -0
  662. package/.agents/scripts/providers/github/sub-issues.js +369 -0
  663. package/.agents/scripts/providers/github/tickets.js +381 -0
  664. package/.agents/scripts/providers/github/transient-retry.js +62 -0
  665. package/.agents/scripts/providers/github.js +157 -0
  666. package/.agents/scripts/quality-preview.js +327 -0
  667. package/.agents/scripts/quality-watch.js +223 -0
  668. package/.agents/scripts/render-manifest.js +143 -0
  669. package/.agents/scripts/resync-status-column.js +176 -0
  670. package/.agents/scripts/retro-run.js +167 -0
  671. package/.agents/scripts/run-audit-suite.js +97 -0
  672. package/.agents/scripts/run-coverage.js +103 -0
  673. package/.agents/scripts/run-lint.js +94 -0
  674. package/.agents/scripts/run-test-profile.js +126 -0
  675. package/.agents/scripts/run-tests.js +185 -0
  676. package/.agents/scripts/run-verify.js +56 -0
  677. package/.agents/scripts/select-audits.js +155 -0
  678. package/.agents/scripts/signals-view.js +294 -0
  679. package/.agents/scripts/single-story-close.js +83 -0
  680. package/.agents/scripts/single-story-confirm-merge.js +183 -0
  681. package/.agents/scripts/single-story-init.js +692 -0
  682. package/.agents/scripts/stories-wave-tick.js +415 -0
  683. package/.agents/scripts/story-close.js +246 -0
  684. package/.agents/scripts/story-deliver-prepare.js +267 -0
  685. package/.agents/scripts/story-init.js +516 -0
  686. package/.agents/scripts/story-phase.js +327 -0
  687. package/.agents/scripts/story-plan.js +284 -0
  688. package/.agents/scripts/sync-agentrc.js +71 -0
  689. package/.agents/scripts/sync-branch-from-base.js +138 -0
  690. package/.agents/scripts/sync-claude-commands.js +151 -0
  691. package/.agents/scripts/test-isolate.js +222 -0
  692. package/.agents/scripts/test-wrapper.js +108 -0
  693. package/.agents/scripts/update-coverage-baseline.js +129 -0
  694. package/.agents/scripts/update-crap-baseline.js +177 -0
  695. package/.agents/scripts/update-duplication-baseline.js +134 -0
  696. package/.agents/scripts/update-maintainability-baseline.js +183 -0
  697. package/.agents/scripts/update-mutation-baseline.js +189 -0
  698. package/.agents/scripts/update-ticket-state.js +107 -0
  699. package/.agents/scripts/validate-docs-freshness.js +259 -0
  700. package/.agents/scripts/validate-skills.js +278 -0
  701. package/.agents/scripts/wave-tick.js +335 -0
  702. package/.agents/skills/core/analyze-execution/SKILL.md +98 -0
  703. package/.agents/skills/core/api-and-interface-design/SKILL.md +327 -0
  704. package/.agents/skills/core/baseline-refresh/SKILL.md +181 -0
  705. package/.agents/skills/core/browser-testing-with-devtools/SKILL.md +352 -0
  706. package/.agents/skills/core/ci-cd-and-automation/SKILL.md +274 -0
  707. package/.agents/skills/core/ci-cd-and-automation/examples.md +211 -0
  708. package/.agents/skills/core/code-review-and-quality/SKILL.md +421 -0
  709. package/.agents/skills/core/code-simplification/SKILL.md +389 -0
  710. package/.agents/skills/core/context-engineering/SKILL.md +309 -0
  711. package/.agents/skills/core/context-engineering/examples.md +58 -0
  712. package/.agents/skills/core/debugging-and-error-recovery/SKILL.md +338 -0
  713. package/.agents/skills/core/deprecation-and-migration/SKILL.md +250 -0
  714. package/.agents/skills/core/diagnose-friction/SKILL.md +79 -0
  715. package/.agents/skills/core/documentation-and-adrs/SKILL.md +323 -0
  716. package/.agents/skills/core/epic-plan-consolidate/SKILL.md +145 -0
  717. package/.agents/skills/core/epic-plan-decompose-author/SKILL.md +425 -0
  718. package/.agents/skills/core/epic-plan-spec-author/SKILL.md +393 -0
  719. package/.agents/skills/core/frontend-ui-engineering/SKILL.md +357 -0
  720. package/.agents/skills/core/git-workflow-and-versioning/SKILL.md +352 -0
  721. package/.agents/skills/core/hydrate-context/SKILL.md +118 -0
  722. package/.agents/skills/core/idea-refinement/SKILL.md +317 -0
  723. package/.agents/skills/core/idea-refinement/examples.md +437 -0
  724. package/.agents/skills/core/idea-refinement/frameworks.md +135 -0
  725. package/.agents/skills/core/idea-refinement/refinement-criteria.md +155 -0
  726. package/.agents/skills/core/idea-refinement/scripts/idea-refine.sh +15 -0
  727. package/.agents/skills/core/incremental-implementation/SKILL.md +271 -0
  728. package/.agents/skills/core/introducing-a-baseline-gate/SKILL.md +213 -0
  729. package/.agents/skills/core/knowledge-transfer/SKILL.md +175 -0
  730. package/.agents/skills/core/mutation-survivor-remediation/SKILL.md +117 -0
  731. package/.agents/skills/core/performance-optimization/SKILL.md +314 -0
  732. package/.agents/skills/core/planning-and-task-breakdown/SKILL.md +277 -0
  733. package/.agents/skills/core/property-based-testing/SKILL.md +148 -0
  734. package/.agents/skills/core/qa-coverage-mapping/SKILL.md +105 -0
  735. package/.agents/skills/core/refactoring-discipline/SKILL.md +111 -0
  736. package/.agents/skills/core/scope-triage/SKILL.md +127 -0
  737. package/.agents/skills/core/security-and-hardening/SKILL.md +400 -0
  738. package/.agents/skills/core/shipping-and-launch/SKILL.md +328 -0
  739. package/.agents/skills/core/spec-driven-development/SKILL.md +252 -0
  740. package/.agents/skills/core/test-driven-development/SKILL.md +475 -0
  741. package/.agents/skills/core/using-agent-skills/SKILL.md +232 -0
  742. package/.agents/skills/skills.index.json +596 -0
  743. package/.agents/skills/stack/architecture/monorepo-path-strategist/SKILL.md +31 -0
  744. package/.agents/skills/stack/architecture/structured-output-zod/SKILL.md +51 -0
  745. package/.agents/skills/stack/architecture/subagent-orchestration/SKILL.md +48 -0
  746. package/.agents/skills/stack/backend/cloudflare-hono-architect/SKILL.md +31 -0
  747. package/.agents/skills/stack/backend/cloudflare-hono-architect/examples/route-template.ts +33 -0
  748. package/.agents/skills/stack/backend/cloudflare-queue-manager/SKILL.md +31 -0
  749. package/.agents/skills/stack/backend/cloudflare-workers/SKILL.md +51 -0
  750. package/.agents/skills/stack/backend/highlevel-crm/SKILL.md +54 -0
  751. package/.agents/skills/stack/backend/sqlite-drizzle-expert/SKILL.md +29 -0
  752. package/.agents/skills/stack/backend/sqlite-drizzle-expert/examples/schema-template.ts +30 -0
  753. package/.agents/skills/stack/backend/stripe-integration/SKILL.md +57 -0
  754. package/.agents/skills/stack/backend/stripe-integration/scripts/listen-stripe.sh +9 -0
  755. package/.agents/skills/stack/backend/turso-sqlite/SKILL.md +48 -0
  756. package/.agents/skills/stack/frontend/astro/SKILL.md +62 -0
  757. package/.agents/skills/stack/frontend/astro-react-island-strategist/SKILL.md +30 -0
  758. package/.agents/skills/stack/frontend/expo-react-native-developer/SKILL.md +29 -0
  759. package/.agents/skills/stack/frontend/google-analytics-v4/SKILL.md +50 -0
  760. package/.agents/skills/stack/frontend/tailwind-v4/SKILL.md +58 -0
  761. package/.agents/skills/stack/frontend/ui-accessibility-engineer/SKILL.md +34 -0
  762. package/.agents/skills/stack/qa/audit-accessibility/SKILL.md +51 -0
  763. package/.agents/skills/stack/qa/gherkin-authoring/SKILL.md +257 -0
  764. package/.agents/skills/stack/qa/gherkin-authoring/examples/invoice-issue.feature +41 -0
  765. package/.agents/skills/stack/qa/lighthouse-baseline/SKILL.md +199 -0
  766. package/.agents/skills/stack/qa/playwright/SKILL.md +50 -0
  767. package/.agents/skills/stack/qa/playwright-bdd/SKILL.md +188 -0
  768. package/.agents/skills/stack/qa/qa-explore-driving/SKILL.md +142 -0
  769. package/.agents/skills/stack/qa/qa-harness/SKILL.md +220 -0
  770. package/.agents/skills/stack/qa/vitest/SKILL.md +51 -0
  771. package/.agents/skills/stack/security/backend-security-patterns/SKILL.md +68 -0
  772. package/.agents/starter-agentrc.json +22 -0
  773. package/.agents/templates/agent-protocol.md +72 -0
  774. package/.agents/templates/docs/architecture.md +30 -0
  775. package/.agents/templates/docs/decisions.md +24 -0
  776. package/.agents/templates/epic-from-idea.md +21 -0
  777. package/.agents/templates/single-story-body.md +17 -0
  778. package/.agents/workflows/agents-update.md +415 -0
  779. package/.agents/workflows/audit-architecture.md +312 -0
  780. package/.agents/workflows/audit-clean-code.md +179 -0
  781. package/.agents/workflows/audit-dependencies.md +91 -0
  782. package/.agents/workflows/audit-devops.md +110 -0
  783. package/.agents/workflows/audit-lighthouse.md +260 -0
  784. package/.agents/workflows/audit-performance.md +161 -0
  785. package/.agents/workflows/audit-privacy.md +104 -0
  786. package/.agents/workflows/audit-quality.md +191 -0
  787. package/.agents/workflows/audit-security.md +156 -0
  788. package/.agents/workflows/audit-seo.md +118 -0
  789. package/.agents/workflows/audit-sre.md +139 -0
  790. package/.agents/workflows/audit-to-stories.md +257 -0
  791. package/.agents/workflows/audit-ux-ui.md +102 -0
  792. package/.agents/workflows/epic-deliver.md +864 -0
  793. package/.agents/workflows/epic-plan.md +998 -0
  794. package/.agents/workflows/explain.md +118 -0
  795. package/.agents/workflows/git-cleanup.md +250 -0
  796. package/.agents/workflows/git-commit-all.md +15 -0
  797. package/.agents/workflows/git-merge-pr.md +377 -0
  798. package/.agents/workflows/git-pr-all.md +278 -0
  799. package/.agents/workflows/git-push.md +60 -0
  800. package/.agents/workflows/helpers/_merge-conflict-template.md +54 -0
  801. package/.agents/workflows/helpers/acceptance-self-eval.md +74 -0
  802. package/.agents/workflows/helpers/agents-sync-config.md +129 -0
  803. package/.agents/workflows/helpers/code-quality-guardrails.md +101 -0
  804. package/.agents/workflows/helpers/code-review.md +370 -0
  805. package/.agents/workflows/helpers/diagnose.md +117 -0
  806. package/.agents/workflows/helpers/epic-audit.md +295 -0
  807. package/.agents/workflows/helpers/epic-deliver-story.md +370 -0
  808. package/.agents/workflows/helpers/epic-plan-decompose.md +199 -0
  809. package/.agents/workflows/helpers/epic-plan-spec.md +184 -0
  810. package/.agents/workflows/helpers/epic-testing.md +125 -0
  811. package/.agents/workflows/helpers/parallel-tooling.md +88 -0
  812. package/.agents/workflows/helpers/signals.md +112 -0
  813. package/.agents/workflows/helpers/single-story-deliver.md +636 -0
  814. package/.agents/workflows/helpers/worktree-lifecycle.md +317 -0
  815. package/.agents/workflows/onboard.md +207 -0
  816. package/.agents/workflows/qa-assist.md +293 -0
  817. package/.agents/workflows/qa-explore.md +350 -0
  818. package/.agents/workflows/qa-run-harness.md +288 -0
  819. package/.agents/workflows/story-deliver.md +327 -0
  820. package/.agents/workflows/story-plan.md +233 -0
  821. package/LICENSE +21 -0
  822. package/README.md +193 -0
  823. package/bin/mandrel.js +56 -0
  824. package/bin/postinstall.js +195 -0
  825. package/lib/cli/__tests__/migrate.test.js +268 -0
  826. package/lib/cli/__tests__/sync-local-zone.test.js +247 -0
  827. package/lib/cli/__tests__/sync.test.js +372 -0
  828. package/lib/cli/__tests__/update-major.test.js +217 -0
  829. package/lib/cli/__tests__/update.test.js +696 -0
  830. package/lib/cli/__tests__/version-check.test.js +398 -0
  831. package/lib/cli/doctor.js +124 -0
  832. package/lib/cli/explain.js +107 -0
  833. package/lib/cli/migrate.js +260 -0
  834. package/lib/cli/registry.js +830 -0
  835. package/lib/cli/sync-commands.js +50 -0
  836. package/lib/cli/sync.js +200 -0
  837. package/lib/cli/uninstall.js +795 -0
  838. package/lib/cli/update.js +854 -0
  839. package/lib/cli/version-check.js +206 -0
  840. package/lib/migrations/README.md +69 -0
  841. package/lib/migrations/__tests__/index.test.js +216 -0
  842. package/lib/migrations/index.js +164 -0
  843. package/package.json +105 -0
@@ -0,0 +1,643 @@
1
+ /* node:coverage ignore file -- prior-state detection over live git + filesystem signals; testing requires mocking the entire merge/worktree state machine, asserts only mock structure */
2
+
3
+ /**
4
+ * story-close-recovery.js — prior-state detection for story-close.
5
+ *
6
+ * Reconstructs close-recovery state from git + filesystem signals at invocation
7
+ * time. No on-disk schema — every signal is observable in the checkout.
8
+ *
9
+ * States (priority order, first match wins):
10
+ * - `partial-merge` — a merge is in progress in the main checkout.
11
+ * - `uncommitted-worktree` — the story worktree exists with uncommitted work.
12
+ * - `already-merged` — the story tip is reachable from `origin/epic/<id>`
13
+ * already (typical Windows partial-reap recovery
14
+ * path: merge + push succeeded but worktree reap
15
+ * or ticket transitions stalled), OR the story
16
+ * diff is fully present on `origin/epic/<id>` as
17
+ * rebased commits with different SHAs (manual
18
+ * recovery path, detected via `git cherry`
19
+ * patch-id comparison; Story #3161), OR — when
20
+ * both Story refs have already been reaped — the
21
+ * Epic history carries an integration commit
22
+ * referencing the Story (`(resolves #<id>)` /
23
+ * `(refs #<id>)`), found via `git log --grep`
24
+ * (ref-independent path; Story #3327 / Epic #3316).
25
+ * - `pushed-unmerged` — the story branch is on origin and not yet merged.
26
+ * - `fresh` — no prior close activity detected.
27
+ */
28
+
29
+ import fs from 'node:fs';
30
+ import { resolveWorkingPath } from '../config-resolver.js';
31
+ import { gitSpawn } from '../git-utils.js';
32
+ import { Logger } from '../Logger.js';
33
+ import { resolvesOrRefsGrepArgs } from './resolves-token.js';
34
+
35
+ export const RECOVERY_STATES = Object.freeze({
36
+ FRESH: 'fresh',
37
+ PARTIAL_MERGE: 'partial-merge',
38
+ UNCOMMITTED_WORKTREE: 'uncommitted-worktree',
39
+ ALREADY_MERGED: 'already-merged',
40
+ PUSHED_UNMERGED: 'pushed-unmerged',
41
+ });
42
+
43
+ const DEFAULT_GIT_ADAPTER = {
44
+ status(cwd) {
45
+ return gitSpawn(cwd, 'status', '--porcelain=v1');
46
+ },
47
+ lsRemote(cwd, ref) {
48
+ return gitSpawn(cwd, 'ls-remote', '--heads', 'origin', ref);
49
+ },
50
+ isAncestor(cwd, ancestor, descendant) {
51
+ return gitSpawn(cwd, 'merge-base', '--is-ancestor', ancestor, descendant);
52
+ },
53
+ showRef(cwd, ref) {
54
+ return gitSpawn(cwd, 'show-ref', '--verify', '--quiet', ref);
55
+ },
56
+ fetchOrigin(cwd, ref) {
57
+ return gitSpawn(cwd, 'fetch', '--quiet', 'origin', ref);
58
+ },
59
+ cherry(cwd, upstream, head) {
60
+ return gitSpawn(cwd, 'cherry', upstream, head);
61
+ },
62
+ logGrep(cwd, ref, grepArgs) {
63
+ return gitSpawn(cwd, 'log', ...grepArgs, '--format=%H', ref);
64
+ },
65
+ };
66
+
67
+ const DEFAULT_FS_ADAPTER = {
68
+ existsSync: fs.existsSync,
69
+ };
70
+
71
+ /**
72
+ * Scan the Epic branch history for an integration/merge commit whose subject
73
+ * references this Story via `(resolves #<id>)` or `(refs #<id>)`.
74
+ *
75
+ * This is the **ref-independent** already-merged signal: it survives the case
76
+ * where BOTH the local `story-<id>` branch and the remote `origin/story-<id>`
77
+ * ref have already been deleted by a prior partial close run, leaving no Story
78
+ * ref to anchor the ancestor / cherry probes (branches a–c). Because the search
79
+ * is scoped to the Epic ref's reachable history (`git log <epicRef> --grep`),
80
+ * any match is by definition already an ancestor of the Epic tip — no separate
81
+ * ancestor check is required.
82
+ *
83
+ * The closing paren in the pattern disambiguates `#<id>` from a longer id that
84
+ * shares the same prefix (e.g. `#3327` must not match `(resolves #33270)`).
85
+ *
86
+ * **origin-only scope (Story #3907).** The probe is restricted to the
87
+ * **remote** `origin/epic/<id>` ref. The earlier implementation also searched
88
+ * the **local** `epic/<id>` ref, which mis-classifies the
89
+ * "merged locally, push failed" recovery state: when a prior close created the
90
+ * integration merge commit on the local Epic branch but the `git push` failed,
91
+ * the local-ref scan finds that commit and reports ALREADY_MERGED. The resumed
92
+ * close then skips the push, flips the ticket done, and reaps the branch — so
93
+ * the merge survives in one clone only, and a sibling's `pull --rebase` can
94
+ * linearize away the `(resolves #<id>)` commit that four subsystems depend on.
95
+ * "Already merged" means "merged on the remote integration branch"; a
96
+ * local-only merge is unpushed work, which the `pushed-unmerged` /
97
+ * resume-from-post-merge paths re-push instead.
98
+ *
99
+ * @returns {{ sha: string, epicRef: string } | null}
100
+ */
101
+ function findMergeCommitForStory({ cwd, storyId, epicId, git }) {
102
+ if (!git.logGrep) return null;
103
+ const grepArgs = resolvesOrRefsGrepArgs(storyId);
104
+ const epicRef = `origin/epic/${epicId}`;
105
+ const res = git.logGrep(cwd, epicRef, grepArgs);
106
+ if (!res || res.status !== 0) return null;
107
+ const sha = (res.stdout ?? '').toString().trim().split('\n')[0]?.trim();
108
+ if (sha) return { sha, epicRef };
109
+ return null;
110
+ }
111
+
112
+ function storyWorktreePath(cwd, storyId, worktreeRoot) {
113
+ return resolveWorkingPath({
114
+ worktreeEnabled: true,
115
+ repoRoot: cwd,
116
+ storyId,
117
+ worktreeRoot,
118
+ });
119
+ }
120
+
121
+ /**
122
+ * Return true if `git status --porcelain=v1` output contains an unmerged
123
+ * marker (`UU`, `AA`, `DD`, `AU`, `UA`, `DU`, `UD`). These are the entries
124
+ * git emits while a merge is in progress with unresolved content.
125
+ */
126
+ function hasUnmergedMarkers(porcelainOutput) {
127
+ if (!porcelainOutput) return false;
128
+ return porcelainOutput
129
+ .split('\n')
130
+ .some((line) => /^(UU|AA|DD|AU|UA|DU|UD) /.test(line));
131
+ }
132
+
133
+ /**
134
+ * Return true if the porcelain output has any non-empty entries (i.e. the
135
+ * working tree is not clean).
136
+ */
137
+ function hasAnyUncommittedChanges(porcelainOutput) {
138
+ if (!porcelainOutput) return false;
139
+ return porcelainOutput.split('\n').some((line) => line.trim().length > 0);
140
+ }
141
+
142
+ /**
143
+ * Probe: partial-merge — UU markers in the main checkout.
144
+ *
145
+ * @returns {{ phase: string, detail: object } | null}
146
+ */
147
+ function detectPartialMerge({ cwd, detail, git }) {
148
+ const mainStatus = git.status(cwd);
149
+ const mainStatusOut = (mainStatus?.stdout ?? '').toString();
150
+ if (!hasUnmergedMarkers(mainStatusOut)) return null;
151
+ return {
152
+ phase: RECOVERY_STATES.PARTIAL_MERGE,
153
+ detail: { ...detail, checkout: cwd },
154
+ };
155
+ }
156
+
157
+ /**
158
+ * Probe: uncommitted-worktree — worktree present + dirty.
159
+ *
160
+ * @returns {{ phase: string, detail: object } | null}
161
+ */
162
+ function detectUncommittedWorktree({
163
+ cwd,
164
+ storyId,
165
+ worktreeRoot,
166
+ detail,
167
+ git,
168
+ fs: fsAdapter,
169
+ }) {
170
+ const wtPath = storyWorktreePath(cwd, storyId, worktreeRoot);
171
+ if (!fsAdapter.existsSync(wtPath)) return null;
172
+ const wtStatus = git.status(wtPath);
173
+ const wtStatusOut = (wtStatus?.stdout ?? '').toString();
174
+ if (!hasAnyUncommittedChanges(wtStatusOut)) return null;
175
+ return {
176
+ phase: RECOVERY_STATES.UNCOMMITTED_WORKTREE,
177
+ detail: { ...detail, worktreePath: wtPath },
178
+ };
179
+ }
180
+
181
+ /**
182
+ * Probe: already-merged — story tip is reachable from `origin/epic/<id>`.
183
+ *
184
+ * Triggered when a prior close pushed the merge but stalled before the
185
+ * ticket transitions / cascade / dashboard regen finished — typical
186
+ * Windows partial-reap recovery
187
+ * "merge/close succeed but branchDeleted: false"). Detected from either:
188
+ * a) the local `story-<id>` branch still exists and is an ancestor of
189
+ * `origin/epic/<id>`; or
190
+ * b) the remote `origin/story-<id>` ref is present and merged; or
191
+ * c) every commit on the Story branch is patch-equivalent (`git cherry`)
192
+ * to a commit already on the Epic (rebased-equivalents, Story #3161).
193
+ * Any one signal is enough — the local branch may have been reaped while
194
+ * the remote one survived, or vice versa.
195
+ *
196
+ * Returns `null` when `epicId` is absent or no merge signal matches.
197
+ *
198
+ * @returns {{ phase: string, detail: object } | null}
199
+ */
200
+ function detectAlreadyMerged({ cwd, storyId, epicId, lsrOut, detail, git }) {
201
+ if (!epicId) return null;
202
+
203
+ const storyBranch = `story-${storyId}`;
204
+ const epicRefs = ['origin/epic', `origin/epic/${epicId}`];
205
+ const probeAncestor = (storyRef, epicRef) =>
206
+ git.isAncestor(cwd, storyRef, epicRef)?.status === 0;
207
+
208
+ let resolvedDetail = null;
209
+
210
+ // a) local story branch still exists.
211
+ const localStoryRef = `refs/heads/${storyBranch}`;
212
+ if (git.showRef && git.showRef(cwd, localStoryRef)?.status === 0) {
213
+ const remoteEpicRef = `origin/epic/${epicId}`;
214
+ if (probeAncestor(storyBranch, remoteEpicRef)) {
215
+ resolvedDetail = { localStoryRef: storyBranch, remoteEpicRef };
216
+ }
217
+ }
218
+
219
+ // b) remote story branch present and merged.
220
+ if (!resolvedDetail && lsrOut.length > 0) {
221
+ for (const epicRef of epicRefs) {
222
+ if (probeAncestor(`origin/${storyBranch}`, epicRef)) {
223
+ resolvedDetail = {
224
+ remoteStoryRef: `origin/${storyBranch}`,
225
+ remoteEpicRef: epicRef,
226
+ };
227
+ break;
228
+ }
229
+ }
230
+ }
231
+
232
+ // c) Rebased equivalents (Story #3161). Story tip is not an ancestor
233
+ // of `origin/epic/<id>`, but every commit on the Story branch is
234
+ // patch-equivalent (`git cherry`) to a commit already on the Epic.
235
+ // Surfaces the manual-recovery case where the operator rebased
236
+ // Story content directly onto `epic/<id>` so the diff is present
237
+ // as commits with different SHAs and no `(resolves #<id>)` merge
238
+ // commit. Without this branch, `assertMergeReachable` throws at
239
+ // resume time and strands close at `agent::closing`.
240
+ if (!resolvedDetail && git.cherry) {
241
+ const candidates = [];
242
+ const localStoryRefName = `refs/heads/${storyBranch}`;
243
+ if (git.showRef && git.showRef(cwd, localStoryRefName)?.status === 0) {
244
+ candidates.push({ ref: storyBranch, kind: 'local' });
245
+ }
246
+ if (lsrOut.length > 0) {
247
+ candidates.push({ ref: `origin/${storyBranch}`, kind: 'remote' });
248
+ }
249
+ const remoteEpicRef = `origin/epic/${epicId}`;
250
+ for (const cand of candidates) {
251
+ const cherry = git.cherry(cwd, remoteEpicRef, cand.ref);
252
+ if (!cherry || cherry.status !== 0) continue;
253
+ const lines = (cherry.stdout ?? '')
254
+ .toString()
255
+ .split('\n')
256
+ .map((l) => l.trim())
257
+ .filter(Boolean);
258
+ if (lines.length === 0) continue;
259
+ if (lines.every((l) => l.startsWith('- '))) {
260
+ resolvedDetail = {
261
+ [cand.kind === 'local' ? 'localStoryRef' : 'remoteStoryRef']:
262
+ cand.ref,
263
+ remoteEpicRef,
264
+ via: 'rebased-equivalents',
265
+ equivalents: lines.length,
266
+ };
267
+ break;
268
+ }
269
+ }
270
+ }
271
+
272
+ // d) Merge-commit-message scan (ref-independent; Story #3327 / Epic #3316).
273
+ // Both the local `story-<id>` branch and the remote `origin/story-<id>`
274
+ // ref were deleted by a prior partial close run, so branches a–c have no
275
+ // ref to anchor on and fall through. Recover the already-merged signal
276
+ // from the Epic history itself: locate the integration commit whose
277
+ // subject carries `(resolves #<id>)` / `(refs #<id>)`. Without this
278
+ // branch, detection falls to FRESH and the resumed close re-enters the
279
+ // pre-merge gate chain, which crashes in `format-autofix-scoped.js` on
280
+ // `git diff <epicBranch>...story-<id>` because the Story ref is gone.
281
+ if (!resolvedDetail) {
282
+ const mc = findMergeCommitForStory({ cwd, storyId, epicId, git });
283
+ if (mc) {
284
+ resolvedDetail = {
285
+ via: 'merge-commit-message',
286
+ mergeCommit: mc.sha,
287
+ remoteEpicRef: mc.epicRef,
288
+ };
289
+ }
290
+ }
291
+
292
+ if (!resolvedDetail) return null;
293
+ return {
294
+ phase: RECOVERY_STATES.ALREADY_MERGED,
295
+ detail: { ...detail, ...resolvedDetail },
296
+ };
297
+ }
298
+
299
+ /**
300
+ * Probe: pushed-unmerged — remote story branch exists and not yet merged.
301
+ *
302
+ * @returns {{ phase: string, detail: object } | null}
303
+ */
304
+ function detectPushedUnmerged({ lsrOut, detail }) {
305
+ if (lsrOut.length === 0) return null;
306
+ return {
307
+ phase: RECOVERY_STATES.PUSHED_UNMERGED,
308
+ detail: { ...detail, remoteRef: lsrOut.split('\n')[0] },
309
+ };
310
+ }
311
+
312
+ /**
313
+ * Detect the prior-close state for a Story.
314
+ *
315
+ * Runs the single-purpose probes in priority order and returns the first
316
+ * match, falling back to FRESH when none fire. Probe order is load-bearing:
317
+ * partial-merge > uncommitted-worktree > already-merged > pushed-unmerged.
318
+ *
319
+ * @param {object} opts
320
+ * @param {string} opts.cwd Main checkout root.
321
+ * @param {number|string} opts.storyId
322
+ * @param {number|string} [opts.epicId] Epic id, used to form `origin/epic/<id>`.
323
+ * @param {string} [opts.worktreeRoot] Worktree root relative to cwd. Default `.worktrees`.
324
+ * @param {object} [opts.git] Git adapter. Defaults to real git via gitSpawn.
325
+ * @param {object} [opts.fs] FS adapter with `existsSync`. Defaults to node:fs.
326
+ * @returns {{ phase: string, detail: object }}
327
+ */
328
+ export function detectPriorPhase({
329
+ cwd,
330
+ storyId,
331
+ epicId,
332
+ worktreeRoot,
333
+ git = DEFAULT_GIT_ADAPTER,
334
+ fs: fsAdapter = DEFAULT_FS_ADAPTER,
335
+ } = {}) {
336
+ if (!cwd) throw new Error('detectPriorPhase: cwd is required');
337
+ if (!storyId) throw new Error('detectPriorPhase: storyId is required');
338
+
339
+ const storyBranch = `story-${storyId}`;
340
+ const detail = { storyId, storyBranch };
341
+
342
+ // `lsRemote` is probed once and shared by the already-merged and
343
+ // pushed-unmerged probes (both key off the remote story ref).
344
+ const lsr = git.lsRemote(cwd, storyBranch);
345
+ const lsrOut = (lsr?.stdout ?? '').toString().trim();
346
+
347
+ return (
348
+ detectPartialMerge({ cwd, detail, git }) ??
349
+ detectUncommittedWorktree({
350
+ cwd,
351
+ storyId,
352
+ worktreeRoot,
353
+ detail,
354
+ git,
355
+ fs: fsAdapter,
356
+ }) ??
357
+ detectAlreadyMerged({ cwd, storyId, epicId, lsrOut, detail, git }) ??
358
+ detectPushedUnmerged({ lsrOut, detail }) ?? {
359
+ phase: RECOVERY_STATES.FRESH,
360
+ detail,
361
+ }
362
+ );
363
+ }
364
+
365
+ export const RECOVERY_ACTIONS = Object.freeze({
366
+ PROCEED: 'proceed',
367
+ EXIT_PRIOR_STATE: 'exit-prior-state',
368
+ RESUME_FROM_VALIDATE: 'resume-from-validate',
369
+ RESUME_FROM_MERGE: 'resume-from-merge',
370
+ RESUME_FROM_CONFLICT: 'resume-from-conflict',
371
+ RESUME_FROM_POST_MERGE: 'resume-from-post-merge',
372
+ RESTART: 'restart',
373
+ });
374
+
375
+ /**
376
+ * Decide how to dispatch given a detected prior state and CLI flags.
377
+ *
378
+ * Exactly one of `resume` / `restart` may be truthy. Passing both throws.
379
+ *
380
+ * @param {object} opts
381
+ * @param {string} opts.state One of RECOVERY_STATES.
382
+ * @param {boolean} [opts.resume]
383
+ * @param {boolean} [opts.restart]
384
+ * @returns {{ action: string, exitCode?: number, reason?: string }}
385
+ */
386
+ export function computeRecoveryMode({ state, resume, restart } = {}) {
387
+ if (resume && restart) {
388
+ throw new Error(
389
+ 'computeRecoveryMode: --resume and --restart are mutually exclusive',
390
+ );
391
+ }
392
+
393
+ if (state === RECOVERY_STATES.FRESH) {
394
+ // Flags are no-ops on fresh state — proceed normally.
395
+ return { action: RECOVERY_ACTIONS.PROCEED };
396
+ }
397
+
398
+ // ALREADY_MERGED short-circuits the prior-state gate: the merge has already
399
+ // landed on `origin/epic/<id>`, so re-running close should pick up exactly
400
+ // the post-merge work (ticket transitions, cascade, health, dashboard) that
401
+ // stalled the first time. No `--resume` flag required — re-running close on
402
+ // a successfully merged Story is the canonical recovery path for partial
403
+ // reaps and is safe to perform automatically.
404
+ if (state === RECOVERY_STATES.ALREADY_MERGED) {
405
+ if (restart) return { action: RECOVERY_ACTIONS.RESTART };
406
+ return { action: RECOVERY_ACTIONS.RESUME_FROM_POST_MERGE };
407
+ }
408
+
409
+ if (restart) {
410
+ return { action: RECOVERY_ACTIONS.RESTART };
411
+ }
412
+
413
+ if (resume) {
414
+ switch (state) {
415
+ case RECOVERY_STATES.PARTIAL_MERGE:
416
+ return { action: RECOVERY_ACTIONS.RESUME_FROM_CONFLICT };
417
+ case RECOVERY_STATES.UNCOMMITTED_WORKTREE:
418
+ return { action: RECOVERY_ACTIONS.RESUME_FROM_VALIDATE };
419
+ case RECOVERY_STATES.PUSHED_UNMERGED:
420
+ return { action: RECOVERY_ACTIONS.RESUME_FROM_MERGE };
421
+ default:
422
+ throw new Error(`computeRecoveryMode: unknown state "${state}"`);
423
+ }
424
+ }
425
+
426
+ // Prior state detected + no flag → refuse to silently proceed.
427
+ return {
428
+ action: RECOVERY_ACTIONS.EXIT_PRIOR_STATE,
429
+ exitCode: 2,
430
+ reason: state,
431
+ };
432
+ }
433
+
434
+ function dropWorktreeIfPresent({ cwd, wtPath, progress, logger }) {
435
+ if (!fs.existsSync(wtPath)) return;
436
+ progress('RESTART', `Removing worktree ${wtPath}`);
437
+ const remove = gitSpawn(cwd, 'worktree', 'remove', '--force', wtPath);
438
+ if (remove.status !== 0) {
439
+ logger.error(
440
+ `[story-close] Worktree remove failed: ${remove.stderr || 'unknown'}. ` +
441
+ 'Attempting prune to clean stale registration.',
442
+ );
443
+ }
444
+ gitSpawn(cwd, 'worktree', 'prune');
445
+ }
446
+
447
+ function recreateStoryBranchRef({ cwd, storyBranch, epicBranch, logger }) {
448
+ gitSpawn(cwd, 'branch', '-D', storyBranch);
449
+ const create = gitSpawn(cwd, 'branch', storyBranch, epicBranch);
450
+ if (create.status !== 0) {
451
+ logger.fatal(
452
+ `Failed to recreate ${storyBranch} from ${epicBranch}: ${create.stderr || 'unknown'}`,
453
+ );
454
+ }
455
+ }
456
+
457
+ function reseedWorktreeIfNeeded({
458
+ cwd,
459
+ wtConfig,
460
+ storyId,
461
+ storyBranch,
462
+ progress,
463
+ logger,
464
+ }) {
465
+ if (!wtConfig?.enabled) return;
466
+ const wtPath = storyWorktreePath(cwd, storyId, wtConfig.root);
467
+ const add = gitSpawn(cwd, 'worktree', 'add', wtPath, storyBranch);
468
+ if (add.status !== 0) {
469
+ logger.fatal(
470
+ `Failed to re-seed worktree at ${wtPath}: ${add.stderr || 'unknown'}`,
471
+ );
472
+ }
473
+ progress('RESTART', `✅ Re-seeded worktree at ${wtPath}`);
474
+ }
475
+
476
+ /**
477
+ * Restart path: abort any in-progress merge, drop the worktree, delete the
478
+ * story branch ref, and re-seed branch + worktree from the Epic branch. The
479
+ * caller then falls through to the normal fresh-close flow.
480
+ */
481
+ function restartStoryState({
482
+ cwd,
483
+ orchestration,
484
+ storyId,
485
+ epicBranch,
486
+ storyBranch,
487
+ progress = () => {},
488
+ logger = Logger,
489
+ } = {}) {
490
+ progress('RESTART', `Resetting prior state for Story #${storyId}...`);
491
+ gitSpawn(cwd, 'merge', '--abort');
492
+
493
+ const wtConfig = orchestration?.worktreeIsolation;
494
+ if (wtConfig?.enabled) {
495
+ dropWorktreeIfPresent({
496
+ cwd,
497
+ wtPath: storyWorktreePath(cwd, storyId, wtConfig.root),
498
+ progress,
499
+ logger,
500
+ });
501
+ }
502
+
503
+ recreateStoryBranchRef({ cwd, storyBranch, epicBranch, logger });
504
+ reseedWorktreeIfNeeded({
505
+ cwd,
506
+ wtConfig,
507
+ storyId,
508
+ storyBranch,
509
+ progress,
510
+ logger,
511
+ });
512
+ }
513
+
514
+ /**
515
+ * Single-call front door for the prior-state machine inside
516
+ * `runStoryClose`. Detects the prior phase, computes the recovery mode for
517
+ * the supplied flags, and:
518
+ *
519
+ * - throws an `Error` with `exitCode: 2` (preserving the existing
520
+ * contract for the CLI wrapper) when no flag was supplied for a
521
+ * non-fresh state;
522
+ * - invokes `restartStoryState` (or the supplied `restartFn`) when
523
+ * `--restart` was passed;
524
+ * - emits the matching progress line for any `--resume` action.
525
+ *
526
+ * Returns a small dispatch summary the caller uses to branch into the
527
+ * conflict-resume vs fresh-merge path and to decide whether to skip the
528
+ * pre-merge validation gates.
529
+ *
530
+ * @param {object} opts
531
+ * @param {string} opts.cwd
532
+ * @param {number|string} opts.storyId
533
+ * @param {number|string} opts.epicId
534
+ * @param {string} opts.epicBranch
535
+ * @param {string} opts.storyBranch
536
+ * @param {object} opts.orchestration
537
+ * @param {boolean} [opts.resume]
538
+ * @param {boolean} [opts.restart]
539
+ * @param {Function} [opts.progress]
540
+ * @param {object} [opts.logger]
541
+ * @param {Function} [opts.detectFn] Override for `detectPriorPhase` (tests).
542
+ * @param {Function} [opts.restartFn] Override for `restartStoryState` (tests).
543
+ * @returns {{
544
+ * action: string,
545
+ * priorPhase: { phase: string, detail: object },
546
+ * resumeFromConflict: boolean,
547
+ * resumeFromMerge: boolean,
548
+ * resumeFromValidate: boolean,
549
+ * }}
550
+ */
551
+ export function dispatchRecovery({
552
+ cwd,
553
+ storyId,
554
+ epicId,
555
+ epicBranch,
556
+ storyBranch,
557
+ orchestration,
558
+ resume = false,
559
+ restart = false,
560
+ progress = () => {},
561
+ logger = Logger,
562
+ detectFn = detectPriorPhase,
563
+ restartFn = restartStoryState,
564
+ } = {}) {
565
+ if (resume && restart) {
566
+ logger.fatal('--resume and --restart are mutually exclusive');
567
+ }
568
+
569
+ const priorPhase = detectFn({ cwd, storyId, epicId });
570
+ const mode = computeRecoveryMode({
571
+ state: priorPhase.phase,
572
+ resume,
573
+ restart,
574
+ });
575
+
576
+ if (mode.action === RECOVERY_ACTIONS.EXIT_PRIOR_STATE) {
577
+ logger.error(
578
+ `[phase=prior-state]\nPrior close state detected: ${priorPhase.phase}\n` +
579
+ `${JSON.stringify(priorPhase.detail, null, 2)}\n\n` +
580
+ 'Re-run with --resume to continue from the detected state, or ' +
581
+ '--restart to abort prior state and re-init.',
582
+ );
583
+ const err = new Error(`prior-state:${priorPhase.phase}`);
584
+ err.exitCode = mode.exitCode ?? 2;
585
+ throw err;
586
+ }
587
+
588
+ if (mode.action === RECOVERY_ACTIONS.RESTART) {
589
+ progress(
590
+ 'RESTART',
591
+ `--restart: aborting prior state (${priorPhase.phase}) and re-initializing`,
592
+ );
593
+ restartFn({
594
+ cwd,
595
+ orchestration,
596
+ storyId,
597
+ epicBranch,
598
+ storyBranch,
599
+ progress,
600
+ logger,
601
+ });
602
+ }
603
+
604
+ const resumeFromConflict =
605
+ mode.action === RECOVERY_ACTIONS.RESUME_FROM_CONFLICT;
606
+ const resumeFromMerge = mode.action === RECOVERY_ACTIONS.RESUME_FROM_MERGE;
607
+ const resumeFromValidate =
608
+ mode.action === RECOVERY_ACTIONS.RESUME_FROM_VALIDATE;
609
+ const resumeFromPostMerge =
610
+ mode.action === RECOVERY_ACTIONS.RESUME_FROM_POST_MERGE;
611
+
612
+ if (resumeFromConflict) {
613
+ progress(
614
+ 'RESUME',
615
+ `--resume: resuming from conflict resolution (phase=${priorPhase.phase})`,
616
+ );
617
+ } else if (resumeFromMerge) {
618
+ progress(
619
+ 'RESUME',
620
+ `--resume: resuming from merge (phase=${priorPhase.phase})`,
621
+ );
622
+ } else if (resumeFromValidate) {
623
+ progress(
624
+ 'RESUME',
625
+ `--resume: resuming from validate (phase=${priorPhase.phase})`,
626
+ );
627
+ } else if (resumeFromPostMerge) {
628
+ progress(
629
+ 'RESUME',
630
+ `prior merge already landed on epic — skipping rebase + merge, ` +
631
+ `running post-merge close work only (phase=${priorPhase.phase})`,
632
+ );
633
+ }
634
+
635
+ return {
636
+ action: mode.action,
637
+ priorPhase,
638
+ resumeFromConflict,
639
+ resumeFromMerge,
640
+ resumeFromValidate,
641
+ resumeFromPostMerge,
642
+ };
643
+ }
@@ -0,0 +1,67 @@
1
+ /**
2
+ * structured-comment-parser.js — shared parser for fenced-JSON structured
3
+ * comments.
4
+ *
5
+ * Several callers (epic-runner Checkpointer, ProgressReporter
6
+ * `parseStoryRunProgressComment` / `parsePhaseTimingsComment`, wave-gate's
7
+ * local `extractJsonBlock`) need to extract the `{...}` payload from the
8
+ * fenced ```json``` block of a structured comment body. They had each open-
9
+ * coded the same `/```json\s*\n([\s\S]*?)\n```/` regex + `JSON.parse` +
10
+ * try/catch dance — small enough to copy, large enough to drift.
11
+ *
12
+ * This helper centralizes that single regex so a future change to the
13
+ * fence format (e.g. tolerating CRLF, surrounding whitespace, alternate
14
+ * fence languages) lives in exactly one place. The functions are
15
+ * deliberately permissive about input shape: anything that isn't the
16
+ * expected type returns `null` rather than throwing, matching what the
17
+ * existing callers already do — they all treat parse failure as "no
18
+ * payload available" and fall back to a default state.
19
+ *
20
+ * Acceptance contract for this helper:
21
+ * - returns the parsed JSON value when the body contains a valid fenced
22
+ * ```json``` block;
23
+ * - returns `null` for missing comment, missing body, missing fence, or
24
+ * malformed JSON inside the fence;
25
+ * - never throws.
26
+ */
27
+
28
+ const JSON_FENCE_RE = /```json\s*\n([\s\S]*?)\n```/;
29
+
30
+ /**
31
+ * Extract the parsed JSON object from the first fenced ```json``` block in
32
+ * a raw string. Returns `null` for any malformed or missing input — callers
33
+ * treat that as "no payload" and fall back.
34
+ *
35
+ * Use this variant when the caller already holds the raw comment body as a
36
+ * string (e.g. after extracting `.body` themselves). Use
37
+ * `parseFencedJsonComment` when working with a comment-like object.
38
+ *
39
+ * @param {unknown} text — the raw string to scan.
40
+ * @returns {unknown | null} the parsed JSON value (typically an object),
41
+ * or `null` when extraction fails.
42
+ */
43
+ export function parseFencedJson(text) {
44
+ if (typeof text !== 'string') return null;
45
+ const match = text.match(JSON_FENCE_RE);
46
+ if (!match) return null;
47
+ try {
48
+ return JSON.parse(match[1]);
49
+ } catch {
50
+ return null;
51
+ }
52
+ }
53
+
54
+ /**
55
+ * Extract the parsed JSON object from the first fenced ```json``` block in
56
+ * a structured comment's `body`. Returns `null` for any malformed or
57
+ * missing input — callers treat that as "no payload" and fall back.
58
+ *
59
+ * @param {{ body?: unknown } | null | undefined} comment — a comment-like
60
+ * object. Only the string `body` field is consulted.
61
+ * @returns {unknown | null} the parsed JSON value (typically an object),
62
+ * or `null` when extraction fails.
63
+ */
64
+ export function parseFencedJsonComment(comment) {
65
+ if (!comment || typeof comment.body !== 'string') return null;
66
+ return parseFencedJson(comment.body);
67
+ }