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.
- package/.agents/README.md +954 -0
- package/.agents/docs/SDLC.md +1420 -0
- package/.agents/docs/agentrc-reference.json +278 -0
- package/.agents/docs/configuration.md +1040 -0
- package/.agents/docs/workflows.md +59 -0
- package/.agents/instructions.md +384 -0
- package/.agents/personas/architect.md +107 -0
- package/.agents/personas/devops-engineer.md +36 -0
- package/.agents/personas/engineer-mobile.md +119 -0
- package/.agents/personas/engineer-web.md +110 -0
- package/.agents/personas/engineer.md +90 -0
- package/.agents/personas/product.md +88 -0
- package/.agents/personas/project-manager.md +110 -0
- package/.agents/personas/qa-engineer.md +91 -0
- package/.agents/personas/refactorer.md +110 -0
- package/.agents/personas/security-engineer.md +112 -0
- package/.agents/personas/sre.md +86 -0
- package/.agents/personas/technical-writer.md +100 -0
- package/.agents/personas/ux-designer.md +95 -0
- package/.agents/rules/api-conventions.md +75 -0
- package/.agents/rules/changelog-style.md +238 -0
- package/.agents/rules/gherkin-standards.md +146 -0
- package/.agents/rules/git-conventions.md +146 -0
- package/.agents/rules/orchestration-error-handling.md +35 -0
- package/.agents/rules/security-baseline.md +92 -0
- package/.agents/rules/shell-conventions.md +70 -0
- package/.agents/rules/test-seams.md +59 -0
- package/.agents/rules/testing-standards.md +177 -0
- package/.agents/runtime-deps.json +18 -0
- package/.agents/schemas/acceptance-eval-verdict.schema.json +93 -0
- package/.agents/schemas/agentrc.schema.json +1583 -0
- package/.agents/schemas/audit-results.schema.json +69 -0
- package/.agents/schemas/audit-rules.json +134 -0
- package/.agents/schemas/audit-rules.schema.json +69 -0
- package/.agents/schemas/baselines/baseline-envelope.schema.json +44 -0
- package/.agents/schemas/baselines/bundle-size.schema.json +47 -0
- package/.agents/schemas/baselines/coverage.schema.json +50 -0
- package/.agents/schemas/baselines/crap.schema.json +52 -0
- package/.agents/schemas/baselines/duplication.schema.json +62 -0
- package/.agents/schemas/baselines/lighthouse.schema.json +59 -0
- package/.agents/schemas/baselines/lint.schema.json +47 -0
- package/.agents/schemas/baselines/maintainability.schema.json +71 -0
- package/.agents/schemas/baselines/mutation.schema.json +52 -0
- package/.agents/schemas/crap-baseline.schema.json +57 -0
- package/.agents/schemas/crap-report.schema.json +102 -0
- package/.agents/schemas/dispatch-manifest.json +232 -0
- package/.agents/schemas/epic-perf-report.schema.json +89 -0
- package/.agents/schemas/epic-spec.schema.json +183 -0
- package/.agents/schemas/friction-event.schema.json +56 -0
- package/.agents/schemas/lifecycle/README.md +18 -0
- package/.agents/schemas/lifecycle/acceptance.reconcile.failed.schema.json +13 -0
- package/.agents/schemas/lifecycle/acceptance.reconcile.ok.schema.json +13 -0
- package/.agents/schemas/lifecycle/acceptance.reconcile.skipped.schema.json +13 -0
- package/.agents/schemas/lifecycle/acceptance.reconcile.start.schema.json +12 -0
- package/.agents/schemas/lifecycle/acceptance.reconcile.waived.schema.json +13 -0
- package/.agents/schemas/lifecycle/checkpoint.written.schema.json +13 -0
- package/.agents/schemas/lifecycle/close-validate.end.schema.json +18 -0
- package/.agents/schemas/lifecycle/close-validate.start.schema.json +13 -0
- package/.agents/schemas/lifecycle/code-review.end.schema.json +30 -0
- package/.agents/schemas/lifecycle/code-review.start.schema.json +12 -0
- package/.agents/schemas/lifecycle/epic.automerge.end.schema.json +14 -0
- package/.agents/schemas/lifecycle/epic.automerge.start.schema.json +13 -0
- package/.agents/schemas/lifecycle/epic.blocked.schema.json +13 -0
- package/.agents/schemas/lifecycle/epic.cleanup.end.schema.json +12 -0
- package/.agents/schemas/lifecycle/epic.cleanup.start.schema.json +12 -0
- package/.agents/schemas/lifecycle/epic.close.end.schema.json +12 -0
- package/.agents/schemas/lifecycle/epic.complete.schema.json +13 -0
- package/.agents/schemas/lifecycle/epic.finalize.end.schema.json +13 -0
- package/.agents/schemas/lifecycle/epic.finalize.start.schema.json +12 -0
- package/.agents/schemas/lifecycle/epic.merge.armed.schema.json +13 -0
- package/.agents/schemas/lifecycle/epic.merge.blocked.schema.json +14 -0
- package/.agents/schemas/lifecycle/epic.merge.confirmed.schema.json +17 -0
- package/.agents/schemas/lifecycle/epic.merge.ready.schema.json +15 -0
- package/.agents/schemas/lifecycle/epic.plan.end.schema.json +18 -0
- package/.agents/schemas/lifecycle/epic.plan.start.schema.json +12 -0
- package/.agents/schemas/lifecycle/epic.snapshot.end.schema.json +16 -0
- package/.agents/schemas/lifecycle/epic.snapshot.start.schema.json +12 -0
- package/.agents/schemas/lifecycle/epic.watch.end.schema.json +28 -0
- package/.agents/schemas/lifecycle/epic.watch.start.schema.json +16 -0
- package/.agents/schemas/lifecycle/intervention.recorded.schema.json +15 -0
- package/.agents/schemas/lifecycle/ledger-record.schema.json +59 -0
- package/.agents/schemas/lifecycle/notification.emitted.schema.json +18 -0
- package/.agents/schemas/lifecycle/pr.created.schema.json +14 -0
- package/.agents/schemas/lifecycle/retro.end.schema.json +16 -0
- package/.agents/schemas/lifecycle/retro.start.schema.json +12 -0
- package/.agents/schemas/lifecycle/story.blocked.schema.json +13 -0
- package/.agents/schemas/lifecycle/story.dispatch.end.schema.json +17 -0
- package/.agents/schemas/lifecycle/story.dispatch.start.schema.json +15 -0
- package/.agents/schemas/lifecycle/story.heartbeat.schema.json +20 -0
- package/.agents/schemas/lifecycle/story.merged.schema.json +13 -0
- package/.agents/schemas/mi-report.schema.json +58 -0
- package/.agents/schemas/model-attribution.schema.json +49 -0
- package/.agents/schemas/qa-finding.schema.json +133 -0
- package/.agents/schemas/qa-ledger.schema.json +89 -0
- package/.agents/schemas/risk-verdict.schema.json +53 -0
- package/.agents/schemas/signal-event.schema.json +58 -0
- package/.agents/schemas/skill.schema.json +31 -0
- package/.agents/schemas/skills-index.schema.json +81 -0
- package/.agents/schemas/story-perf-summary.schema.json +73 -0
- package/.agents/schemas/validation-evidence.schema.json +78 -0
- package/.agents/scripts/README.md +93 -0
- package/.agents/scripts/acceptance-eval.js +284 -0
- package/.agents/scripts/acceptance-spec-reconciler.js +556 -0
- package/.agents/scripts/agents-bootstrap-github.js +634 -0
- package/.agents/scripts/analyze-execution.js +369 -0
- package/.agents/scripts/assert-branch.js +83 -0
- package/.agents/scripts/audit-labels-bootstrap.js +253 -0
- package/.agents/scripts/audit-to-stories.js +257 -0
- package/.agents/scripts/bootstrap.js +1378 -0
- package/.agents/scripts/check-baselines.js +81 -0
- package/.agents/scripts/check-dead-exports.js +311 -0
- package/.agents/scripts/check-doc-links.js +401 -0
- package/.agents/scripts/check-gherkin-placeholders.js +663 -0
- package/.agents/scripts/check-lifecycle-doc-drift.js +402 -0
- package/.agents/scripts/check-lifecycle-lint.js +379 -0
- package/.agents/scripts/check-prepush-recovery.js +90 -0
- package/.agents/scripts/check-windows-git-perf.js +138 -0
- package/.agents/scripts/cleanup-repo-test-temp.js +67 -0
- package/.agents/scripts/coverage-capture.js +112 -0
- package/.agents/scripts/detect-merges.js +111 -0
- package/.agents/scripts/diagnose-friction.js +257 -0
- package/.agents/scripts/diagnose.js +240 -0
- package/.agents/scripts/dispatcher.js +295 -0
- package/.agents/scripts/drain-pending-cleanup.js +147 -0
- package/.agents/scripts/epic-audit-prepare.js +419 -0
- package/.agents/scripts/epic-audit-recheck.js +241 -0
- package/.agents/scripts/epic-deliver-note-intervention.js +192 -0
- package/.agents/scripts/epic-deliver-preflight.js +407 -0
- package/.agents/scripts/epic-deliver-prepare.js +383 -0
- package/.agents/scripts/epic-execute-record-wave.js +463 -0
- package/.agents/scripts/epic-plan-clarity.js +201 -0
- package/.agents/scripts/epic-plan-decompose.js +79 -0
- package/.agents/scripts/epic-plan-healthcheck.js +363 -0
- package/.agents/scripts/epic-plan-spec-validate.js +111 -0
- package/.agents/scripts/epic-plan-spec.js +198 -0
- package/.agents/scripts/epic-reconcile.js +637 -0
- package/.agents/scripts/evidence-gate.js +235 -0
- package/.agents/scripts/generate-config-docs.js +516 -0
- package/.agents/scripts/generate-lifecycle-docs.js +224 -0
- package/.agents/scripts/generate-skills-index.js +252 -0
- package/.agents/scripts/generate-workflows-doc.js +168 -0
- package/.agents/scripts/git-cleanup.js +124 -0
- package/.agents/scripts/git-pr-quality-gate.js +203 -0
- package/.agents/scripts/git-rebase-and-resolve.js +234 -0
- package/.agents/scripts/hierarchy-gate.js +176 -0
- package/.agents/scripts/hydrate-context.js +179 -0
- package/.agents/scripts/install-matrix-assert.js +282 -0
- package/.agents/scripts/lib/Graph.js +326 -0
- package/.agents/scripts/lib/ITicketingProvider.js +349 -0
- package/.agents/scripts/lib/Logger.js +194 -0
- package/.agents/scripts/lib/audit-suite/cli.js +64 -0
- package/.agents/scripts/lib/audit-suite/findings.js +164 -0
- package/.agents/scripts/lib/audit-suite/frontmatter-lint.js +32 -0
- package/.agents/scripts/lib/audit-suite/frontmatter.js +110 -0
- package/.agents/scripts/lib/audit-suite/index.js +22 -0
- package/.agents/scripts/lib/audit-suite/runner.js +233 -0
- package/.agents/scripts/lib/audit-suite/selector.js +235 -0
- package/.agents/scripts/lib/audit-suite/substitutions.js +124 -0
- package/.agents/scripts/lib/audit-suite/workflow-loader.js +49 -0
- package/.agents/scripts/lib/audit-to-stories/build-story-body.js +130 -0
- package/.agents/scripts/lib/audit-to-stories/dedupe-against-github.js +114 -0
- package/.agents/scripts/lib/audit-to-stories/finding-adapter.js +93 -0
- package/.agents/scripts/lib/audit-to-stories/group-findings.js +265 -0
- package/.agents/scripts/lib/audit-to-stories/parse-audit-md.js +246 -0
- package/.agents/scripts/lib/audit-to-stories/seed-epic-from-findings.js +160 -0
- package/.agents/scripts/lib/auto-refresh-baselines.js +308 -0
- package/.agents/scripts/lib/baseline-loader.js +0 -0
- package/.agents/scripts/lib/baseline-schema-registry.js +69 -0
- package/.agents/scripts/lib/baseline-snapshot.js +716 -0
- package/.agents/scripts/lib/baselines/component-matcher.js +21 -0
- package/.agents/scripts/lib/baselines/components.js +126 -0
- package/.agents/scripts/lib/baselines/diff-scope-cli.js +203 -0
- package/.agents/scripts/lib/baselines/duplication-scanner.js +220 -0
- package/.agents/scripts/lib/baselines/env-overrides.js +129 -0
- package/.agents/scripts/lib/baselines/envelope.js +368 -0
- package/.agents/scripts/lib/baselines/exit-codes.js +89 -0
- package/.agents/scripts/lib/baselines/git-base.js +0 -0
- package/.agents/scripts/lib/baselines/kernel.js +111 -0
- package/.agents/scripts/lib/baselines/kinds/_shared-metric.js +220 -0
- package/.agents/scripts/lib/baselines/kinds/bundle-size.js +157 -0
- package/.agents/scripts/lib/baselines/kinds/coverage.js +194 -0
- package/.agents/scripts/lib/baselines/kinds/crap.js +555 -0
- package/.agents/scripts/lib/baselines/kinds/duplication.js +197 -0
- package/.agents/scripts/lib/baselines/kinds/lighthouse.js +185 -0
- package/.agents/scripts/lib/baselines/kinds/lint.js +172 -0
- package/.agents/scripts/lib/baselines/kinds/maintainability.js +340 -0
- package/.agents/scripts/lib/baselines/kinds/mutation.js +153 -0
- package/.agents/scripts/lib/baselines/path-canon.js +279 -0
- package/.agents/scripts/lib/baselines/preview-gates.js +298 -0
- package/.agents/scripts/lib/baselines/reader.js +321 -0
- package/.agents/scripts/lib/baselines/refresh-service.js +733 -0
- package/.agents/scripts/lib/baselines/scope.js +291 -0
- package/.agents/scripts/lib/baselines/writer.js +312 -0
- package/.agents/scripts/lib/bdd-runner-detect.js +417 -0
- package/.agents/scripts/lib/bdd-scenario-scanner.js +310 -0
- package/.agents/scripts/lib/bootstrap/baselines-layout-migration.js +202 -0
- package/.agents/scripts/lib/bootstrap/branch-protection.js +222 -0
- package/.agents/scripts/lib/bootstrap/ci-workflow-template.js +171 -0
- package/.agents/scripts/lib/bootstrap/commit-push.js +146 -0
- package/.agents/scripts/lib/bootstrap/gh-list.js +153 -0
- package/.agents/scripts/lib/bootstrap/gh-preflight.js +306 -0
- package/.agents/scripts/lib/bootstrap/hitl-confirm.js +89 -0
- package/.agents/scripts/lib/bootstrap/install-ledger.js +174 -0
- package/.agents/scripts/lib/bootstrap/manifest.js +272 -0
- package/.agents/scripts/lib/bootstrap/merge-methods.js +108 -0
- package/.agents/scripts/lib/bootstrap/preflight.js +195 -0
- package/.agents/scripts/lib/bootstrap/project-bootstrap.js +801 -0
- package/.agents/scripts/lib/bootstrap/prompt.js +480 -0
- package/.agents/scripts/lib/bootstrap/quality-bootstrap.js +370 -0
- package/.agents/scripts/lib/bootstrap/summary.js +75 -0
- package/.agents/scripts/lib/bootstrap/workflow-audit.js +256 -0
- package/.agents/scripts/lib/branch-name-guard.js +98 -0
- package/.agents/scripts/lib/c8-cli-path.js +21 -0
- package/.agents/scripts/lib/changed-files.js +184 -0
- package/.agents/scripts/lib/checks/baseline-drift-main-checkout.js +104 -0
- package/.agents/scripts/lib/checks/core-bare-clean.js +48 -0
- package/.agents/scripts/lib/checks/epic-merge-lock-stale.js +54 -0
- package/.agents/scripts/lib/checks/index.js +288 -0
- package/.agents/scripts/lib/checks/push-hook-parity.js +106 -0
- package/.agents/scripts/lib/checks/stale-origin-epic.js +49 -0
- package/.agents/scripts/lib/checks/state.js +558 -0
- package/.agents/scripts/lib/checks/story-init-not-backgrounded.js +186 -0
- package/.agents/scripts/lib/checks/subagent-agent-tool-required.js +182 -0
- package/.agents/scripts/lib/checks/windows-coverage-noise-floor.js +92 -0
- package/.agents/scripts/lib/checks/worktree-bootstrap-env.js +81 -0
- package/.agents/scripts/lib/checks/worktree-residue-biome.js +55 -0
- package/.agents/scripts/lib/cli/parse-numeric.js +60 -0
- package/.agents/scripts/lib/cli/standard-args.js +351 -0
- package/.agents/scripts/lib/cli-args.js +286 -0
- package/.agents/scripts/lib/cli-utils.js +69 -0
- package/.agents/scripts/lib/close-validation/projections/head-sha.js +44 -0
- package/.agents/scripts/lib/close-validation/projections/inputs.js +86 -0
- package/.agents/scripts/lib/close-validation/projections/maintainability.js +286 -0
- package/.agents/scripts/lib/close-validation.js +897 -0
- package/.agents/scripts/lib/codebase-snapshot.js +513 -0
- package/.agents/scripts/lib/command-header.js +33 -0
- package/.agents/scripts/lib/config/acceptance-eval.js +95 -0
- package/.agents/scripts/lib/config/baselines.js +60 -0
- package/.agents/scripts/lib/config/ci.js +30 -0
- package/.agents/scripts/lib/config/commands.js +36 -0
- package/.agents/scripts/lib/config/defaults.js +119 -0
- package/.agents/scripts/lib/config/explain.js +348 -0
- package/.agents/scripts/lib/config/gates/bundle-size.schema.js +23 -0
- package/.agents/scripts/lib/config/gates/coverage.schema.js +18 -0
- package/.agents/scripts/lib/config/gates/crap.schema.js +33 -0
- package/.agents/scripts/lib/config/gates/duplication.schema.js +26 -0
- package/.agents/scripts/lib/config/gates/index.js +36 -0
- package/.agents/scripts/lib/config/gates/lighthouse.schema.js +23 -0
- package/.agents/scripts/lib/config/gates/lint.schema.js +9 -0
- package/.agents/scripts/lib/config/gates/maintainability.schema.js +20 -0
- package/.agents/scripts/lib/config/gates/mutation.schema.js +12 -0
- package/.agents/scripts/lib/config/gates/shared.js +117 -0
- package/.agents/scripts/lib/config/github.js +122 -0
- package/.agents/scripts/lib/config/lifecycle.js +40 -0
- package/.agents/scripts/lib/config/limits.js +211 -0
- package/.agents/scripts/lib/config/paths.js +73 -0
- package/.agents/scripts/lib/config/preflight.js +58 -0
- package/.agents/scripts/lib/config/quality.js +665 -0
- package/.agents/scripts/lib/config/retro.js +77 -0
- package/.agents/scripts/lib/config/runners.js +105 -0
- package/.agents/scripts/lib/config/runtime.js +167 -0
- package/.agents/scripts/lib/config/shared.js +46 -0
- package/.agents/scripts/lib/config/sync-agentrc.js +243 -0
- package/.agents/scripts/lib/config/temp-paths.js +373 -0
- package/.agents/scripts/lib/config/validate-orchestration.js +81 -0
- package/.agents/scripts/lib/config/worktree-isolation.js +80 -0
- package/.agents/scripts/lib/config-resolver.js +298 -0
- package/.agents/scripts/lib/config-schema-shared.js +32 -0
- package/.agents/scripts/lib/config-schema.js +20 -0
- package/.agents/scripts/lib/config-settings-schema-delivery.js +332 -0
- package/.agents/scripts/lib/config-settings-schema-quality.js +165 -0
- package/.agents/scripts/lib/config-settings-schema.js +420 -0
- package/.agents/scripts/lib/coverage-baseline.js +352 -0
- package/.agents/scripts/lib/coverage-capture.js +195 -0
- package/.agents/scripts/lib/coverage-utils.js +239 -0
- package/.agents/scripts/lib/cpu-pool.js +223 -0
- package/.agents/scripts/lib/crap-engine.js +119 -0
- package/.agents/scripts/lib/crap-utils.js +479 -0
- package/.agents/scripts/lib/degraded-mode.js +69 -0
- package/.agents/scripts/lib/dependency-parser.js +129 -0
- package/.agents/scripts/lib/duplicate-search.js +189 -0
- package/.agents/scripts/lib/dynamic-workflow/architecture-report-contract.js +70 -0
- package/.agents/scripts/lib/dynamic-workflow/audit-orchestrator.js +197 -0
- package/.agents/scripts/lib/dynamic-workflow/capability.js +396 -0
- package/.agents/scripts/lib/dynamic-workflow/clean-code-report-contract.js +80 -0
- package/.agents/scripts/lib/dynamic-workflow/performance-report-contract.js +72 -0
- package/.agents/scripts/lib/dynamic-workflow/quality-report-contract.js +90 -0
- package/.agents/scripts/lib/dynamic-workflow/report-contract-core.js +43 -0
- package/.agents/scripts/lib/dynamic-workflow/security-report-contract.js +83 -0
- package/.agents/scripts/lib/env-loader.js +52 -0
- package/.agents/scripts/lib/epic-merge-lock.js +239 -0
- package/.agents/scripts/lib/epic-plan-clarity.js +142 -0
- package/.agents/scripts/lib/epic-plan-ideation.js +228 -0
- package/.agents/scripts/lib/error-redactor.js +125 -0
- package/.agents/scripts/lib/errors/index.js +67 -0
- package/.agents/scripts/lib/feedback-loop/audit-results-graduator.js +230 -0
- package/.agents/scripts/lib/feedback-loop/code-review-graduator.js +207 -0
- package/.agents/scripts/lib/feedback-loop/graduator-core.js +421 -0
- package/.agents/scripts/lib/feedback-loop/memory-freshness.js +480 -0
- package/.agents/scripts/lib/feedback-loop/prior-feedback-fetcher.js +229 -0
- package/.agents/scripts/lib/findings/classify-finding.js +195 -0
- package/.agents/scripts/lib/findings/promote-finding.js +353 -0
- package/.agents/scripts/lib/findings/route-finding.js +283 -0
- package/.agents/scripts/lib/findings/semantic-issue-search.js +179 -0
- package/.agents/scripts/lib/findings/severity.js +102 -0
- package/.agents/scripts/lib/gates/baseline-store.js +106 -0
- package/.agents/scripts/lib/gates/friction.js +43 -0
- package/.agents/scripts/lib/gh-exec.js +553 -0
- package/.agents/scripts/lib/git/cached-fetch.js +0 -0
- package/.agents/scripts/lib/git/sync-from-base.js +162 -0
- package/.agents/scripts/lib/git-branch-cleanup.js +213 -0
- package/.agents/scripts/lib/git-branch-lifecycle.js +353 -0
- package/.agents/scripts/lib/git-merge-orchestrator.js +261 -0
- package/.agents/scripts/lib/git-utils.js +363 -0
- package/.agents/scripts/lib/github-url.js +29 -0
- package/.agents/scripts/lib/install-cmd-parser.js +51 -0
- package/.agents/scripts/lib/issue-link-parser.js +74 -0
- package/.agents/scripts/lib/json-utils.js +60 -0
- package/.agents/scripts/lib/label-constants.js +169 -0
- package/.agents/scripts/lib/label-taxonomy.js +200 -0
- package/.agents/scripts/lib/maintainability-engine.js +164 -0
- package/.agents/scripts/lib/maintainability-utils.js +343 -0
- package/.agents/scripts/lib/mandrel-catalog.js +170 -0
- package/.agents/scripts/lib/mutation/baseline-snapshot.js +238 -0
- package/.agents/scripts/lib/mutation/config-detector.js +119 -0
- package/.agents/scripts/lib/mutation/stryker-runner.js +306 -0
- package/.agents/scripts/lib/mutation/survivor-report.js +160 -0
- package/.agents/scripts/lib/notifications/notifier.js +75 -0
- package/.agents/scripts/lib/observability/active-story-env.js +182 -0
- package/.agents/scripts/lib/observability/baseline-refresh-rate.js +221 -0
- package/.agents/scripts/lib/observability/perf-aggregator.js +887 -0
- package/.agents/scripts/lib/observability/perf-report-readers.js +319 -0
- package/.agents/scripts/lib/observability/perf-report-render.js +182 -0
- package/.agents/scripts/lib/observability/signals-writer.js +296 -0
- package/.agents/scripts/lib/observability/source-classifier.js +103 -0
- package/.agents/scripts/lib/observability/tool-trace-hook.js +417 -0
- package/.agents/scripts/lib/onboard/detect-stack.js +300 -0
- package/.agents/scripts/lib/onboard/scaffold-docs.js +128 -0
- package/.agents/scripts/lib/orchestration/acceptance-eval-decision.js +173 -0
- package/.agents/scripts/lib/orchestration/cascade-grouping.js +275 -0
- package/.agents/scripts/lib/orchestration/check-baselines/phases/compare.js +131 -0
- package/.agents/scripts/lib/orchestration/check-baselines/phases/evaluate.js +80 -0
- package/.agents/scripts/lib/orchestration/check-baselines/phases/floors.js +132 -0
- package/.agents/scripts/lib/orchestration/check-baselines/phases/friction.js +142 -0
- package/.agents/scripts/lib/orchestration/check-baselines/phases/parse-args.js +149 -0
- package/.agents/scripts/lib/orchestration/check-baselines/phases/pipeline.js +158 -0
- package/.agents/scripts/lib/orchestration/check-baselines/phases/report.js +56 -0
- package/.agents/scripts/lib/orchestration/code-review.js +652 -0
- package/.agents/scripts/lib/orchestration/column-sync.js +286 -0
- package/.agents/scripts/lib/orchestration/context-envelope.js +280 -0
- package/.agents/scripts/lib/orchestration/context-hydration-engine.js +581 -0
- package/.agents/scripts/lib/orchestration/dependency-analyzer.js +88 -0
- package/.agents/scripts/lib/orchestration/detectors-phase.js +188 -0
- package/.agents/scripts/lib/orchestration/dispatch-engine.js +144 -0
- package/.agents/scripts/lib/orchestration/dispatch-pipeline.js +206 -0
- package/.agents/scripts/lib/orchestration/doc-reader.js +94 -0
- package/.agents/scripts/lib/orchestration/epic-cleanup.js +473 -0
- package/.agents/scripts/lib/orchestration/epic-deliver-lease-guard.js +310 -0
- package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/cli.js +167 -0
- package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/context.js +151 -0
- package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/creation.js +74 -0
- package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/dag.js +78 -0
- package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/diagnostics.js +72 -0
- package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/persist-helpers.js +155 -0
- package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/persist.js +321 -0
- package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/planning-artifacts.js +75 -0
- package/.agents/scripts/lib/orchestration/epic-plan-decompose/phases/reconcile-spawn.js +86 -0
- package/.agents/scripts/lib/orchestration/epic-plan-lease-guard.js +235 -0
- package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/authoring-context.js +197 -0
- package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/cli-args.js +48 -0
- package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/drain.js +94 -0
- package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/plan-epic.js +414 -0
- package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/prompts.js +55 -0
- package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/risk-verdict.js +105 -0
- package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/run-spec-phase.js +235 -0
- package/.agents/scripts/lib/orchestration/epic-plan-spec/phases/spec-freshness.js +120 -0
- package/.agents/scripts/lib/orchestration/epic-plan-state-store.js +118 -0
- package/.agents/scripts/lib/orchestration/epic-run-state-store.js +295 -0
- package/.agents/scripts/lib/orchestration/epic-runner/concurrency-gate.js +186 -0
- package/.agents/scripts/lib/orchestration/epic-runner/deliver-phases.js +50 -0
- package/.agents/scripts/lib/orchestration/epic-runner/phases/build-wave-dag.js +146 -0
- package/.agents/scripts/lib/orchestration/epic-runner/phases/snapshot.js +110 -0
- package/.agents/scripts/lib/orchestration/epic-runner/progress-reporter/composition.js +392 -0
- package/.agents/scripts/lib/orchestration/epic-runner/progress-reporter/signals.js +217 -0
- package/.agents/scripts/lib/orchestration/epic-runner/progress-reporter/transport.js +235 -0
- package/.agents/scripts/lib/orchestration/epic-runner/progress-reporter.js +69 -0
- package/.agents/scripts/lib/orchestration/epic-runner/progress-signals/_bullet-format.js +32 -0
- package/.agents/scripts/lib/orchestration/epic-runner/progress-signals/crap-drift.js +291 -0
- package/.agents/scripts/lib/orchestration/epic-runner/progress-signals/maintainability-drift.js +175 -0
- package/.agents/scripts/lib/orchestration/epic-runner/progress-signals/stalled-worktree.js +37 -0
- package/.agents/scripts/lib/orchestration/epic-runner/story-launcher.js +127 -0
- package/.agents/scripts/lib/orchestration/epic-runner/story-run-progress-writer.js +400 -0
- package/.agents/scripts/lib/orchestration/epic-runner/sub-agent-return.js +285 -0
- package/.agents/scripts/lib/orchestration/epic-runner/wave-scheduler.js +66 -0
- package/.agents/scripts/lib/orchestration/epic-spec-reconciler-apply.js +797 -0
- package/.agents/scripts/lib/orchestration/epic-spec-reconciler-diff.js +619 -0
- package/.agents/scripts/lib/orchestration/epic-spec-reconciler-discriminator.js +335 -0
- package/.agents/scripts/lib/orchestration/epic-spec-reconciler-format.js +230 -0
- package/.agents/scripts/lib/orchestration/epic-spec-reconciler-ops.js +363 -0
- package/.agents/scripts/lib/orchestration/error-journal.js +139 -0
- package/.agents/scripts/lib/orchestration/file-assumption-enum.js +31 -0
- package/.agents/scripts/lib/orchestration/file-assumptions.js +506 -0
- package/.agents/scripts/lib/orchestration/finalize/close-planning-tickets.js +116 -0
- package/.agents/scripts/lib/orchestration/finalize/open-or-locate-pr.js +241 -0
- package/.agents/scripts/lib/orchestration/finalize/post-handoff-comment.js +489 -0
- package/.agents/scripts/lib/orchestration/finalize/sanitize-skip-ci.js +88 -0
- package/.agents/scripts/lib/orchestration/git-cleanup/phases/branches-reap.js +219 -0
- package/.agents/scripts/lib/orchestration/git-cleanup/phases/branches.js +309 -0
- package/.agents/scripts/lib/orchestration/git-cleanup/phases/cli.js +99 -0
- package/.agents/scripts/lib/orchestration/git-cleanup/phases/fast-forward.js +123 -0
- package/.agents/scripts/lib/orchestration/git-cleanup/phases/filters.js +57 -0
- package/.agents/scripts/lib/orchestration/git-cleanup/phases/git-probes-ff.js +114 -0
- package/.agents/scripts/lib/orchestration/git-cleanup/phases/git-probes.js +426 -0
- package/.agents/scripts/lib/orchestration/git-cleanup/phases/parse-args.js +84 -0
- package/.agents/scripts/lib/orchestration/git-cleanup/phases/phase-drivers.js +365 -0
- package/.agents/scripts/lib/orchestration/git-cleanup/phases/prompts.js +72 -0
- package/.agents/scripts/lib/orchestration/git-cleanup/phases/prune.js +69 -0
- package/.agents/scripts/lib/orchestration/git-cleanup/phases/render.js +214 -0
- package/.agents/scripts/lib/orchestration/git-cleanup/phases/stashes.js +137 -0
- package/.agents/scripts/lib/orchestration/label-transitions.js +43 -0
- package/.agents/scripts/lib/orchestration/lifecycle/bus.js +309 -0
- package/.agents/scripts/lib/orchestration/lifecycle/emit-story-dispatch-end.js +147 -0
- package/.agents/scripts/lib/orchestration/lifecycle/emit-story-heartbeat.js +155 -0
- package/.agents/scripts/lib/orchestration/lifecycle/ledger-writer.js +226 -0
- package/.agents/scripts/lib/orchestration/lifecycle/listeners/README.md +69 -0
- package/.agents/scripts/lib/orchestration/lifecycle/listeners/acceptance-reconciler.js +378 -0
- package/.agents/scripts/lib/orchestration/lifecycle/listeners/automerge-armer.js +248 -0
- package/.agents/scripts/lib/orchestration/lifecycle/listeners/automerge-predicate.js +527 -0
- package/.agents/scripts/lib/orchestration/lifecycle/listeners/branch-cleaner.js +259 -0
- package/.agents/scripts/lib/orchestration/lifecycle/listeners/checkpoint-pointer-writer.js +278 -0
- package/.agents/scripts/lib/orchestration/lifecycle/listeners/cleaner.js +355 -0
- package/.agents/scripts/lib/orchestration/lifecycle/listeners/finalizer.js +647 -0
- package/.agents/scripts/lib/orchestration/lifecycle/listeners/index.js +331 -0
- package/.agents/scripts/lib/orchestration/lifecycle/listeners/intervention-recorder.js +140 -0
- package/.agents/scripts/lib/orchestration/lifecycle/listeners/merge-watcher.js +421 -0
- package/.agents/scripts/lib/orchestration/lifecycle/listeners/notify-dispatcher.js +168 -0
- package/.agents/scripts/lib/orchestration/lifecycle/listeners/watcher.js +668 -0
- package/.agents/scripts/lib/orchestration/lifecycle/trace-logger.js +322 -0
- package/.agents/scripts/lib/orchestration/lint-baseline-service.js +114 -0
- package/.agents/scripts/lib/orchestration/manifest-builder.js +216 -0
- package/.agents/scripts/lib/orchestration/model-attribution.js +390 -0
- package/.agents/scripts/lib/orchestration/parked-follow-ons.js +147 -0
- package/.agents/scripts/lib/orchestration/phase-runner.js +87 -0
- package/.agents/scripts/lib/orchestration/plan-review-routing.js +63 -0
- package/.agents/scripts/lib/orchestration/plan-runner/plan-router.js +86 -0
- package/.agents/scripts/lib/orchestration/plan-runner/worktree-sweep.js +212 -0
- package/.agents/scripts/lib/orchestration/planning-context-budget.js +213 -0
- package/.agents/scripts/lib/orchestration/planning-risk.js +155 -0
- package/.agents/scripts/lib/orchestration/planning-state-manager.js +318 -0
- package/.agents/scripts/lib/orchestration/post-merge/phases/branch-cleanup.js +56 -0
- package/.agents/scripts/lib/orchestration/post-merge/phases/dashboard-refresh.js +33 -0
- package/.agents/scripts/lib/orchestration/post-merge/phases/notification.js +78 -0
- package/.agents/scripts/lib/orchestration/post-merge/phases/temp-cleanup.js +68 -0
- package/.agents/scripts/lib/orchestration/post-merge/phases/ticket-closure.js +118 -0
- package/.agents/scripts/lib/orchestration/post-merge/phases/worktree-reap.js +396 -0
- package/.agents/scripts/lib/orchestration/post-merge-pipeline.js +205 -0
- package/.agents/scripts/lib/orchestration/pr-base-guard.js +47 -0
- package/.agents/scripts/lib/orchestration/preflight-cache.js +164 -0
- package/.agents/scripts/lib/orchestration/reassert-status-column.js +202 -0
- package/.agents/scripts/lib/orchestration/reconciler.js +137 -0
- package/.agents/scripts/lib/orchestration/recurring-failure-detector.js +152 -0
- package/.agents/scripts/lib/orchestration/recut.js +56 -0
- package/.agents/scripts/lib/orchestration/resolves-token.js +127 -0
- package/.agents/scripts/lib/orchestration/retro/phases/checks.js +94 -0
- package/.agents/scripts/lib/orchestration/retro/phases/compose-body.js +448 -0
- package/.agents/scripts/lib/orchestration/retro/phases/gather-signals.js +335 -0
- package/.agents/scripts/lib/orchestration/retro/phases/post-and-mirror.js +133 -0
- package/.agents/scripts/lib/orchestration/retro-heuristics.js +57 -0
- package/.agents/scripts/lib/orchestration/retro-perf-heuristics.js +275 -0
- package/.agents/scripts/lib/orchestration/retro-proposals.js +395 -0
- package/.agents/scripts/lib/orchestration/retro-runner.js +171 -0
- package/.agents/scripts/lib/orchestration/review-depth.js +93 -0
- package/.agents/scripts/lib/orchestration/review-providers/codex.js +363 -0
- package/.agents/scripts/lib/orchestration/review-providers/findings-renderer.js +205 -0
- package/.agents/scripts/lib/orchestration/review-providers/native.js +805 -0
- package/.agents/scripts/lib/orchestration/review-providers/review-depth.js +73 -0
- package/.agents/scripts/lib/orchestration/review-providers/review-provider-factory.js +396 -0
- package/.agents/scripts/lib/orchestration/review-providers/security-review.js +373 -0
- package/.agents/scripts/lib/orchestration/review-providers/types.js +89 -0
- package/.agents/scripts/lib/orchestration/review-providers/ultrareview.js +107 -0
- package/.agents/scripts/lib/orchestration/single-story-close/phases/auto-merge.js +159 -0
- package/.agents/scripts/lib/orchestration/single-story-close/phases/base-sync.js +194 -0
- package/.agents/scripts/lib/orchestration/single-story-close/phases/close-validation.js +81 -0
- package/.agents/scripts/lib/orchestration/single-story-close/phases/code-review.js +190 -0
- package/.agents/scripts/lib/orchestration/single-story-close/phases/options.js +70 -0
- package/.agents/scripts/lib/orchestration/single-story-close/phases/pull-request.js +106 -0
- package/.agents/scripts/lib/orchestration/single-story-close/phases/push.js +42 -0
- package/.agents/scripts/lib/orchestration/single-story-close/phases/worktree-reap.js +73 -0
- package/.agents/scripts/lib/orchestration/single-story-close/phases/wrong-tree-guard.js +225 -0
- package/.agents/scripts/lib/orchestration/single-story-close/runner.js +315 -0
- package/.agents/scripts/lib/orchestration/single-story-lease-guard.js +149 -0
- package/.agents/scripts/lib/orchestration/skill-capsule-loader.js +110 -0
- package/.agents/scripts/lib/orchestration/spec-freshness.js +320 -0
- package/.agents/scripts/lib/orchestration/spec-renderer.js +456 -0
- package/.agents/scripts/lib/orchestration/spec-section-validator.js +80 -0
- package/.agents/scripts/lib/orchestration/story-close/auto-refresh-runner.js +797 -0
- package/.agents/scripts/lib/orchestration/story-close/baseline-attribution/phases/gate-failure.js +163 -0
- package/.agents/scripts/lib/orchestration/story-close/baseline-attribution/phases/pre-merge-attribution.js +152 -0
- package/.agents/scripts/lib/orchestration/story-close/baseline-attribution/phases/refresh-commit.js +387 -0
- package/.agents/scripts/lib/orchestration/story-close/baseline-attribution/phases/regression-projection.js +266 -0
- package/.agents/scripts/lib/orchestration/story-close/baseline-attribution/phases/scope-discovery.js +48 -0
- package/.agents/scripts/lib/orchestration/story-close/baseline-attribution-wiring.js +67 -0
- package/.agents/scripts/lib/orchestration/story-close/baseline-attribution.js +161 -0
- package/.agents/scripts/lib/orchestration/story-close/baseline-friction-body.js +117 -0
- package/.agents/scripts/lib/orchestration/story-close/cd-out-guard.js +86 -0
- package/.agents/scripts/lib/orchestration/story-close/cleanup-reconciler.js +147 -0
- package/.agents/scripts/lib/orchestration/story-close/close-inputs.js +142 -0
- package/.agents/scripts/lib/orchestration/story-close/comment-bodies.js +62 -0
- package/.agents/scripts/lib/orchestration/story-close/format-autofix-scoped.js +221 -0
- package/.agents/scripts/lib/orchestration/story-close/format-autofix-shared.js +123 -0
- package/.agents/scripts/lib/orchestration/story-close/format-autofix.js +216 -0
- package/.agents/scripts/lib/orchestration/story-close/merge-runner.js +636 -0
- package/.agents/scripts/lib/orchestration/story-close/merge-subject.js +198 -0
- package/.agents/scripts/lib/orchestration/story-close/phases/branch-restore.js +105 -0
- package/.agents/scripts/lib/orchestration/story-close/phases/close.js +222 -0
- package/.agents/scripts/lib/orchestration/story-close/phases/code-review.js +220 -0
- package/.agents/scripts/lib/orchestration/story-close/phases/gates.js +291 -0
- package/.agents/scripts/lib/orchestration/story-close/phases/locked-pipeline.js +234 -0
- package/.agents/scripts/lib/orchestration/story-close/phases/preflight.js +110 -0
- package/.agents/scripts/lib/orchestration/story-close/phases/refresh.js +86 -0
- package/.agents/scripts/lib/orchestration/story-close/phases/timeout-blocked-emitter.js +112 -0
- package/.agents/scripts/lib/orchestration/story-close/phases/timeout-blocked.js +157 -0
- package/.agents/scripts/lib/orchestration/story-close/post-merge-close.js +434 -0
- package/.agents/scripts/lib/orchestration/story-close/pre-merge-validation.js +290 -0
- package/.agents/scripts/lib/orchestration/story-close-recovery.js +643 -0
- package/.agents/scripts/lib/orchestration/structured-comment-parser.js +67 -0
- package/.agents/scripts/lib/orchestration/task-body-validator.js +391 -0
- package/.agents/scripts/lib/orchestration/ticket-lease.js +358 -0
- package/.agents/scripts/lib/orchestration/ticket-validator-conflicts.js +783 -0
- package/.agents/scripts/lib/orchestration/ticket-validator-sizing.js +367 -0
- package/.agents/scripts/lib/orchestration/ticket-validator.js +691 -0
- package/.agents/scripts/lib/orchestration/ticketing/bulk.js +723 -0
- package/.agents/scripts/lib/orchestration/ticketing/reads.js +474 -0
- package/.agents/scripts/lib/orchestration/ticketing/state.js +559 -0
- package/.agents/scripts/lib/orchestration/ticketing.js +55 -0
- package/.agents/scripts/lib/orchestration/wave-marker.js +28 -0
- package/.agents/scripts/lib/orchestration/wave-record-io.js +277 -0
- package/.agents/scripts/lib/orchestration/wave-record-notifications.js +189 -0
- package/.agents/scripts/lib/orchestration/wave-record-projection.js +423 -0
- package/.agents/scripts/lib/path-security.js +25 -0
- package/.agents/scripts/lib/plan-phase-cleanup.js +125 -0
- package/.agents/scripts/lib/preflight-runner.js +196 -0
- package/.agents/scripts/lib/presentation/dispatch-manifest-render.js +95 -0
- package/.agents/scripts/lib/presentation/manifest-builder.js +245 -0
- package/.agents/scripts/lib/presentation/manifest-formatter.js +243 -0
- package/.agents/scripts/lib/presentation/manifest-helpers.js +213 -0
- package/.agents/scripts/lib/presentation/manifest-persistence.js +262 -0
- package/.agents/scripts/lib/presentation/manifest-procedures.js +55 -0
- package/.agents/scripts/lib/presentation/manifest-render-waves.js +252 -0
- package/.agents/scripts/lib/presentation/manifest-renderer.js +188 -0
- package/.agents/scripts/lib/presentation/manifest-story-views.js +119 -0
- package/.agents/scripts/lib/provider-factory.js +80 -0
- package/.agents/scripts/lib/push-epic-retry.js +209 -0
- package/.agents/scripts/lib/qa/console-allowlist.js +151 -0
- package/.agents/scripts/lib/qa/coverage-report.js +181 -0
- package/.agents/scripts/lib/qa/coverage-verdict.js +296 -0
- package/.agents/scripts/lib/qa/propose-missing-test.js +95 -0
- package/.agents/scripts/lib/qa/qa-context-hydrator.js +296 -0
- package/.agents/scripts/lib/qa/qa-session.js +197 -0
- package/.agents/scripts/lib/qa/redact-evidence.js +245 -0
- package/.agents/scripts/lib/qa/resolve-qa-contract.js +190 -0
- package/.agents/scripts/lib/qa/resolve-selection.js +373 -0
- package/.agents/scripts/lib/runtime-deps/ensure-installed.js +100 -0
- package/.agents/scripts/lib/runtime-deps/manifest.js +96 -0
- package/.agents/scripts/lib/runtime-deps/preflight.js +78 -0
- package/.agents/scripts/lib/runtime-deps/scan-imports.js +202 -0
- package/.agents/scripts/lib/signals/detectors/common.js +36 -0
- package/.agents/scripts/lib/signals/detectors/hotspot.js +298 -0
- package/.agents/scripts/lib/signals/detectors/index.js +14 -0
- package/.agents/scripts/lib/signals/detectors/retry.js +289 -0
- package/.agents/scripts/lib/signals/detectors/rework.js +204 -0
- package/.agents/scripts/lib/signals/index.js +39 -0
- package/.agents/scripts/lib/signals/read.js +268 -0
- package/.agents/scripts/lib/signals/schema.js +225 -0
- package/.agents/scripts/lib/signals/span-tree.js +290 -0
- package/.agents/scripts/lib/signals/write.js +19 -0
- package/.agents/scripts/lib/single-story/confirm-merge.js +201 -0
- package/.agents/scripts/lib/single-story/story-merged-notify.js +126 -0
- package/.agents/scripts/lib/single-story-sweep/protection.js +274 -0
- package/.agents/scripts/lib/single-story-sweep/sweep-lock.js +169 -0
- package/.agents/scripts/lib/single-story-sweep.js +329 -0
- package/.agents/scripts/lib/skills/parse-skill.js +202 -0
- package/.agents/scripts/lib/skills/walk-skill-files.js +56 -0
- package/.agents/scripts/lib/spec/index.js +36 -0
- package/.agents/scripts/lib/spec/loader.js +425 -0
- package/.agents/scripts/lib/spec/state.js +217 -0
- package/.agents/scripts/lib/story-body/story-body.js +743 -0
- package/.agents/scripts/lib/story-init/blocker-validator.js +68 -0
- package/.agents/scripts/lib/story-init/branch-initializer.js +422 -0
- package/.agents/scripts/lib/story-init/context-resolver.js +92 -0
- package/.agents/scripts/lib/story-init/donor-precheck.js +207 -0
- package/.agents/scripts/lib/story-init/hierarchy-tracer.js +36 -0
- package/.agents/scripts/lib/story-init/state-transitioner.js +80 -0
- package/.agents/scripts/lib/story-init/task-graph-builder.js +114 -0
- package/.agents/scripts/lib/story-init/transition-summary.js +34 -0
- package/.agents/scripts/lib/story-lifecycle.js +186 -0
- package/.agents/scripts/lib/story-plan.js +246 -0
- package/.agents/scripts/lib/task-utils.js +26 -0
- package/.agents/scripts/lib/templates/decomposer-prompts.js +168 -0
- package/.agents/scripts/lib/test-env.js +30 -0
- package/.agents/scripts/lib/test-isolate/env-snapshot-loader.js +52 -0
- package/.agents/scripts/lib/test-isolate/list-files.js +90 -0
- package/.agents/scripts/lib/test-isolate/parse-tap.js +75 -0
- package/.agents/scripts/lib/test-isolate/runner.js +483 -0
- package/.agents/scripts/lib/test-profile/parse-tap.js +136 -0
- package/.agents/scripts/lib/test-profile/render-report.js +45 -0
- package/.agents/scripts/lib/test-reserved-epic-temp-ids.js +35 -0
- package/.agents/scripts/lib/test-tiers.js +94 -0
- package/.agents/scripts/lib/util/concurrent-map.js +59 -0
- package/.agents/scripts/lib/util/phase-timer-state.js +72 -0
- package/.agents/scripts/lib/util/phase-timer.js +163 -0
- package/.agents/scripts/lib/util/poll-loop.js +86 -0
- package/.agents/scripts/lib/util/with-timeout.js +32 -0
- package/.agents/scripts/lib/validation-evidence.js +323 -0
- package/.agents/scripts/lib/wave-runner/tick.js +665 -0
- package/.agents/scripts/lib/wave-runner/wave-checkpoint.js +91 -0
- package/.agents/scripts/lib/wave-runner/wave-runner-error.js +19 -0
- package/.agents/scripts/lib/workers/crap-worker.js +197 -0
- package/.agents/scripts/lib/workers/maintainability-report-worker.js +137 -0
- package/.agents/scripts/lib/workers/maintainability-worker.js +79 -0
- package/.agents/scripts/lib/workspace-provisioner.js +189 -0
- package/.agents/scripts/lib/worktree/bootstrapper.js +48 -0
- package/.agents/scripts/lib/worktree/inspector.js +140 -0
- package/.agents/scripts/lib/worktree/lifecycle/creation.js +118 -0
- package/.agents/scripts/lib/worktree/lifecycle/drift-detection.js +62 -0
- package/.agents/scripts/lib/worktree/lifecycle/force-drain.js +276 -0
- package/.agents/scripts/lib/worktree/lifecycle/gc.js +49 -0
- package/.agents/scripts/lib/worktree/lifecycle/merge-reachability.js +178 -0
- package/.agents/scripts/lib/worktree/lifecycle/pending-cleanup.js +264 -0
- package/.agents/scripts/lib/worktree/lifecycle/precheck.js +100 -0
- package/.agents/scripts/lib/worktree/lifecycle/reap.js +588 -0
- package/.agents/scripts/lib/worktree/lifecycle/registry-sync.js +124 -0
- package/.agents/scripts/lib/worktree/lifecycle/shared.js +26 -0
- package/.agents/scripts/lib/worktree/lifecycle-manager.js +40 -0
- package/.agents/scripts/lib/worktree/node-modules-strategy.js +349 -0
- package/.agents/scripts/lib/worktree-manager.js +243 -0
- package/.agents/scripts/lifecycle-diff.js +206 -0
- package/.agents/scripts/lifecycle-emit-story-dispatch.js +194 -0
- package/.agents/scripts/lifecycle-emit.js +479 -0
- package/.agents/scripts/lint-baseline.js +507 -0
- package/.agents/scripts/lint-label-vocabulary.js +237 -0
- package/.agents/scripts/loc-delta.js +205 -0
- package/.agents/scripts/notify.js +307 -0
- package/.agents/scripts/package.json +3 -0
- package/.agents/scripts/post-structured-comment.js +127 -0
- package/.agents/scripts/pr-watch-with-update.js +152 -0
- package/.agents/scripts/providers/github/auth.js +65 -0
- package/.agents/scripts/providers/github/board-add.js +63 -0
- package/.agents/scripts/providers/github/branch-protection.js +186 -0
- package/.agents/scripts/providers/github/cache.js +72 -0
- package/.agents/scripts/providers/github/comments.js +131 -0
- package/.agents/scripts/providers/github/compose.js +111 -0
- package/.agents/scripts/providers/github/errors.js +242 -0
- package/.agents/scripts/providers/github/issues.js +242 -0
- package/.agents/scripts/providers/github/labels.js +179 -0
- package/.agents/scripts/providers/github/mappers.js +126 -0
- package/.agents/scripts/providers/github/merge-methods.js +82 -0
- package/.agents/scripts/providers/github/project-board.js +47 -0
- package/.agents/scripts/providers/github/projects-v2-graphql.js +472 -0
- package/.agents/scripts/providers/github/prs.js +103 -0
- package/.agents/scripts/providers/github/request-helpers.js +110 -0
- package/.agents/scripts/providers/github/sub-issues.js +369 -0
- package/.agents/scripts/providers/github/tickets.js +381 -0
- package/.agents/scripts/providers/github/transient-retry.js +62 -0
- package/.agents/scripts/providers/github.js +157 -0
- package/.agents/scripts/quality-preview.js +327 -0
- package/.agents/scripts/quality-watch.js +223 -0
- package/.agents/scripts/render-manifest.js +143 -0
- package/.agents/scripts/resync-status-column.js +176 -0
- package/.agents/scripts/retro-run.js +167 -0
- package/.agents/scripts/run-audit-suite.js +97 -0
- package/.agents/scripts/run-coverage.js +103 -0
- package/.agents/scripts/run-lint.js +94 -0
- package/.agents/scripts/run-test-profile.js +126 -0
- package/.agents/scripts/run-tests.js +185 -0
- package/.agents/scripts/run-verify.js +56 -0
- package/.agents/scripts/select-audits.js +155 -0
- package/.agents/scripts/signals-view.js +294 -0
- package/.agents/scripts/single-story-close.js +83 -0
- package/.agents/scripts/single-story-confirm-merge.js +183 -0
- package/.agents/scripts/single-story-init.js +692 -0
- package/.agents/scripts/stories-wave-tick.js +415 -0
- package/.agents/scripts/story-close.js +246 -0
- package/.agents/scripts/story-deliver-prepare.js +267 -0
- package/.agents/scripts/story-init.js +516 -0
- package/.agents/scripts/story-phase.js +327 -0
- package/.agents/scripts/story-plan.js +284 -0
- package/.agents/scripts/sync-agentrc.js +71 -0
- package/.agents/scripts/sync-branch-from-base.js +138 -0
- package/.agents/scripts/sync-claude-commands.js +151 -0
- package/.agents/scripts/test-isolate.js +222 -0
- package/.agents/scripts/test-wrapper.js +108 -0
- package/.agents/scripts/update-coverage-baseline.js +129 -0
- package/.agents/scripts/update-crap-baseline.js +177 -0
- package/.agents/scripts/update-duplication-baseline.js +134 -0
- package/.agents/scripts/update-maintainability-baseline.js +183 -0
- package/.agents/scripts/update-mutation-baseline.js +189 -0
- package/.agents/scripts/update-ticket-state.js +107 -0
- package/.agents/scripts/validate-docs-freshness.js +259 -0
- package/.agents/scripts/validate-skills.js +278 -0
- package/.agents/scripts/wave-tick.js +335 -0
- package/.agents/skills/core/analyze-execution/SKILL.md +98 -0
- package/.agents/skills/core/api-and-interface-design/SKILL.md +327 -0
- package/.agents/skills/core/baseline-refresh/SKILL.md +181 -0
- package/.agents/skills/core/browser-testing-with-devtools/SKILL.md +352 -0
- package/.agents/skills/core/ci-cd-and-automation/SKILL.md +274 -0
- package/.agents/skills/core/ci-cd-and-automation/examples.md +211 -0
- package/.agents/skills/core/code-review-and-quality/SKILL.md +421 -0
- package/.agents/skills/core/code-simplification/SKILL.md +389 -0
- package/.agents/skills/core/context-engineering/SKILL.md +309 -0
- package/.agents/skills/core/context-engineering/examples.md +58 -0
- package/.agents/skills/core/debugging-and-error-recovery/SKILL.md +338 -0
- package/.agents/skills/core/deprecation-and-migration/SKILL.md +250 -0
- package/.agents/skills/core/diagnose-friction/SKILL.md +79 -0
- package/.agents/skills/core/documentation-and-adrs/SKILL.md +323 -0
- package/.agents/skills/core/epic-plan-consolidate/SKILL.md +145 -0
- package/.agents/skills/core/epic-plan-decompose-author/SKILL.md +425 -0
- package/.agents/skills/core/epic-plan-spec-author/SKILL.md +393 -0
- package/.agents/skills/core/frontend-ui-engineering/SKILL.md +357 -0
- package/.agents/skills/core/git-workflow-and-versioning/SKILL.md +352 -0
- package/.agents/skills/core/hydrate-context/SKILL.md +118 -0
- package/.agents/skills/core/idea-refinement/SKILL.md +317 -0
- package/.agents/skills/core/idea-refinement/examples.md +437 -0
- package/.agents/skills/core/idea-refinement/frameworks.md +135 -0
- package/.agents/skills/core/idea-refinement/refinement-criteria.md +155 -0
- package/.agents/skills/core/idea-refinement/scripts/idea-refine.sh +15 -0
- package/.agents/skills/core/incremental-implementation/SKILL.md +271 -0
- package/.agents/skills/core/introducing-a-baseline-gate/SKILL.md +213 -0
- package/.agents/skills/core/knowledge-transfer/SKILL.md +175 -0
- package/.agents/skills/core/mutation-survivor-remediation/SKILL.md +117 -0
- package/.agents/skills/core/performance-optimization/SKILL.md +314 -0
- package/.agents/skills/core/planning-and-task-breakdown/SKILL.md +277 -0
- package/.agents/skills/core/property-based-testing/SKILL.md +148 -0
- package/.agents/skills/core/qa-coverage-mapping/SKILL.md +105 -0
- package/.agents/skills/core/refactoring-discipline/SKILL.md +111 -0
- package/.agents/skills/core/scope-triage/SKILL.md +127 -0
- package/.agents/skills/core/security-and-hardening/SKILL.md +400 -0
- package/.agents/skills/core/shipping-and-launch/SKILL.md +328 -0
- package/.agents/skills/core/spec-driven-development/SKILL.md +252 -0
- package/.agents/skills/core/test-driven-development/SKILL.md +475 -0
- package/.agents/skills/core/using-agent-skills/SKILL.md +232 -0
- package/.agents/skills/skills.index.json +596 -0
- package/.agents/skills/stack/architecture/monorepo-path-strategist/SKILL.md +31 -0
- package/.agents/skills/stack/architecture/structured-output-zod/SKILL.md +51 -0
- package/.agents/skills/stack/architecture/subagent-orchestration/SKILL.md +48 -0
- package/.agents/skills/stack/backend/cloudflare-hono-architect/SKILL.md +31 -0
- package/.agents/skills/stack/backend/cloudflare-hono-architect/examples/route-template.ts +33 -0
- package/.agents/skills/stack/backend/cloudflare-queue-manager/SKILL.md +31 -0
- package/.agents/skills/stack/backend/cloudflare-workers/SKILL.md +51 -0
- package/.agents/skills/stack/backend/highlevel-crm/SKILL.md +54 -0
- package/.agents/skills/stack/backend/sqlite-drizzle-expert/SKILL.md +29 -0
- package/.agents/skills/stack/backend/sqlite-drizzle-expert/examples/schema-template.ts +30 -0
- package/.agents/skills/stack/backend/stripe-integration/SKILL.md +57 -0
- package/.agents/skills/stack/backend/stripe-integration/scripts/listen-stripe.sh +9 -0
- package/.agents/skills/stack/backend/turso-sqlite/SKILL.md +48 -0
- package/.agents/skills/stack/frontend/astro/SKILL.md +62 -0
- package/.agents/skills/stack/frontend/astro-react-island-strategist/SKILL.md +30 -0
- package/.agents/skills/stack/frontend/expo-react-native-developer/SKILL.md +29 -0
- package/.agents/skills/stack/frontend/google-analytics-v4/SKILL.md +50 -0
- package/.agents/skills/stack/frontend/tailwind-v4/SKILL.md +58 -0
- package/.agents/skills/stack/frontend/ui-accessibility-engineer/SKILL.md +34 -0
- package/.agents/skills/stack/qa/audit-accessibility/SKILL.md +51 -0
- package/.agents/skills/stack/qa/gherkin-authoring/SKILL.md +257 -0
- package/.agents/skills/stack/qa/gherkin-authoring/examples/invoice-issue.feature +41 -0
- package/.agents/skills/stack/qa/lighthouse-baseline/SKILL.md +199 -0
- package/.agents/skills/stack/qa/playwright/SKILL.md +50 -0
- package/.agents/skills/stack/qa/playwright-bdd/SKILL.md +188 -0
- package/.agents/skills/stack/qa/qa-explore-driving/SKILL.md +142 -0
- package/.agents/skills/stack/qa/qa-harness/SKILL.md +220 -0
- package/.agents/skills/stack/qa/vitest/SKILL.md +51 -0
- package/.agents/skills/stack/security/backend-security-patterns/SKILL.md +68 -0
- package/.agents/starter-agentrc.json +22 -0
- package/.agents/templates/agent-protocol.md +72 -0
- package/.agents/templates/docs/architecture.md +30 -0
- package/.agents/templates/docs/decisions.md +24 -0
- package/.agents/templates/epic-from-idea.md +21 -0
- package/.agents/templates/single-story-body.md +17 -0
- package/.agents/workflows/agents-update.md +415 -0
- package/.agents/workflows/audit-architecture.md +312 -0
- package/.agents/workflows/audit-clean-code.md +179 -0
- package/.agents/workflows/audit-dependencies.md +91 -0
- package/.agents/workflows/audit-devops.md +110 -0
- package/.agents/workflows/audit-lighthouse.md +260 -0
- package/.agents/workflows/audit-performance.md +161 -0
- package/.agents/workflows/audit-privacy.md +104 -0
- package/.agents/workflows/audit-quality.md +191 -0
- package/.agents/workflows/audit-security.md +156 -0
- package/.agents/workflows/audit-seo.md +118 -0
- package/.agents/workflows/audit-sre.md +139 -0
- package/.agents/workflows/audit-to-stories.md +257 -0
- package/.agents/workflows/audit-ux-ui.md +102 -0
- package/.agents/workflows/epic-deliver.md +864 -0
- package/.agents/workflows/epic-plan.md +998 -0
- package/.agents/workflows/explain.md +118 -0
- package/.agents/workflows/git-cleanup.md +250 -0
- package/.agents/workflows/git-commit-all.md +15 -0
- package/.agents/workflows/git-merge-pr.md +377 -0
- package/.agents/workflows/git-pr-all.md +278 -0
- package/.agents/workflows/git-push.md +60 -0
- package/.agents/workflows/helpers/_merge-conflict-template.md +54 -0
- package/.agents/workflows/helpers/acceptance-self-eval.md +74 -0
- package/.agents/workflows/helpers/agents-sync-config.md +129 -0
- package/.agents/workflows/helpers/code-quality-guardrails.md +101 -0
- package/.agents/workflows/helpers/code-review.md +370 -0
- package/.agents/workflows/helpers/diagnose.md +117 -0
- package/.agents/workflows/helpers/epic-audit.md +295 -0
- package/.agents/workflows/helpers/epic-deliver-story.md +370 -0
- package/.agents/workflows/helpers/epic-plan-decompose.md +199 -0
- package/.agents/workflows/helpers/epic-plan-spec.md +184 -0
- package/.agents/workflows/helpers/epic-testing.md +125 -0
- package/.agents/workflows/helpers/parallel-tooling.md +88 -0
- package/.agents/workflows/helpers/signals.md +112 -0
- package/.agents/workflows/helpers/single-story-deliver.md +636 -0
- package/.agents/workflows/helpers/worktree-lifecycle.md +317 -0
- package/.agents/workflows/onboard.md +207 -0
- package/.agents/workflows/qa-assist.md +293 -0
- package/.agents/workflows/qa-explore.md +350 -0
- package/.agents/workflows/qa-run-harness.md +288 -0
- package/.agents/workflows/story-deliver.md +327 -0
- package/.agents/workflows/story-plan.md +233 -0
- package/LICENSE +21 -0
- package/README.md +193 -0
- package/bin/mandrel.js +56 -0
- package/bin/postinstall.js +195 -0
- package/lib/cli/__tests__/migrate.test.js +268 -0
- package/lib/cli/__tests__/sync-local-zone.test.js +247 -0
- package/lib/cli/__tests__/sync.test.js +372 -0
- package/lib/cli/__tests__/update-major.test.js +217 -0
- package/lib/cli/__tests__/update.test.js +696 -0
- package/lib/cli/__tests__/version-check.test.js +398 -0
- package/lib/cli/doctor.js +124 -0
- package/lib/cli/explain.js +107 -0
- package/lib/cli/migrate.js +260 -0
- package/lib/cli/registry.js +830 -0
- package/lib/cli/sync-commands.js +50 -0
- package/lib/cli/sync.js +200 -0
- package/lib/cli/uninstall.js +795 -0
- package/lib/cli/update.js +854 -0
- package/lib/cli/version-check.js +206 -0
- package/lib/migrations/README.md +69 -0
- package/lib/migrations/__tests__/index.test.js +216 -0
- package/lib/migrations/index.js +164 -0
- 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
|
+
}
|