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,716 @@
|
|
|
1
|
+
import { spawnSync as defaultSpawnSync } from 'node:child_process';
|
|
2
|
+
import fs from 'node:fs';
|
|
3
|
+
import os from 'node:os';
|
|
4
|
+
import path from 'node:path';
|
|
5
|
+
import { filterExcludedRows } from './baselines/kinds/maintainability.js';
|
|
6
|
+
import { canonicalise as canonicalisePath } from './baselines/path-canon.js';
|
|
7
|
+
import { loadFile as readerLoadFile } from './baselines/reader.js';
|
|
8
|
+
import {
|
|
9
|
+
write as defaultWriteBaseline,
|
|
10
|
+
writeFile as defaultWriteBaselineFile,
|
|
11
|
+
} from './baselines/writer.js';
|
|
12
|
+
import {
|
|
13
|
+
getBaselines as defaultGetBaselines,
|
|
14
|
+
getQuality as defaultGetQuality,
|
|
15
|
+
resolveConfig as defaultResolveConfig,
|
|
16
|
+
PROJECT_ROOT,
|
|
17
|
+
} from './config-resolver.js';
|
|
18
|
+
import { loadCoverage } from './coverage-utils.js';
|
|
19
|
+
import {
|
|
20
|
+
resolveEscomplexVersion,
|
|
21
|
+
resolveTsTranspilerVersion,
|
|
22
|
+
scanAndScore,
|
|
23
|
+
} from './crap-utils.js';
|
|
24
|
+
import { ensureEpicBranchRef as defaultEnsureEpicBranchRef } from './git-branch-lifecycle.js';
|
|
25
|
+
import { calculateAll, scanDirectory } from './maintainability-utils.js';
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* baseline-snapshot.js — per-Epic baseline lifecycle helpers.
|
|
29
|
+
*
|
|
30
|
+
* Story #1396 (Epic #1386). The Epic-snapshot scheme freezes the maintainability
|
|
31
|
+
* and crap baselines at /epic-plan time and reconciles them back to `main`
|
|
32
|
+
* at /epic-deliver time. Two helpers, both pure-ish (deterministic given the
|
|
33
|
+
* working tree + injected I/O):
|
|
34
|
+
*
|
|
35
|
+
* - forkMainToEpic({ epicId, cwd }) — copies the tracked main baselines
|
|
36
|
+
* under `temp/epic-<id>/baselines/`. Idempotent: re-running with the same
|
|
37
|
+
* source content produces the same destination bytes (no fs churn). When
|
|
38
|
+
* the source baseline is missing, emits a warn through the injected
|
|
39
|
+
* logger and returns `{ written: false, reason: 'source-missing' }` for
|
|
40
|
+
* that file — callers (e.g. /epic-plan Phase 7) treat the absence as
|
|
41
|
+
* non-fatal and stay in `--full-scope` mode.
|
|
42
|
+
*
|
|
43
|
+
* - regenerateMainFromTree({ cwd }) — re-scores maintainability + crap
|
|
44
|
+
* against the current working tree and writes the result to the tracked
|
|
45
|
+
* main baseline paths. Returns `{ didChange, paths }` where `didChange`
|
|
46
|
+
* is true iff any baseline file's content differs from what's already on
|
|
47
|
+
* disk. Callers in /epic-deliver use `didChange === false` to skip the
|
|
48
|
+
* `baseline-refresh: epic-<id>` commit.
|
|
49
|
+
*
|
|
50
|
+
* Lifecycle note (Story #1467): per-epic ratchet snapshots are ephemeral
|
|
51
|
+
* scratch state under the `temp/epic-<id>/baselines/` namespace, NOT committed
|
|
52
|
+
* artifacts. They inherit the existing per-epic temp-tree cleanup contract —
|
|
53
|
+
* `/epic-deliver` reaps the parent `temp/epic-<id>/` directory on merge, so
|
|
54
|
+
* no manual prune is required. Earlier versions of this module wrote under
|
|
55
|
+
* `baselines/epic/<id>/`, which committed them to git and accumulated obsolete
|
|
56
|
+
* snapshots forever.
|
|
57
|
+
*
|
|
58
|
+
* Why "pure-ish" and not pure: both helpers read+write the filesystem and
|
|
59
|
+
* (for regenerateMainFromTree) walk source trees + parse coverage. The seam
|
|
60
|
+
* exposes the pieces that matter for tests — `fs`, the config accessors,
|
|
61
|
+
* the scoring helpers — through dependency injection so the unit tests can
|
|
62
|
+
* pin behaviour without ever touching real `baselines/*.json`.
|
|
63
|
+
*/
|
|
64
|
+
|
|
65
|
+
const EPIC_BASELINES = ['maintainability', 'crap'];
|
|
66
|
+
|
|
67
|
+
/**
|
|
68
|
+
* Resolve the per-Epic snapshot path for a baseline kind.
|
|
69
|
+
*
|
|
70
|
+
* @param {{ epicId: number, kind: 'maintainability'|'crap', cwd?: string }} opts
|
|
71
|
+
* @returns {string} absolute path under `<cwd>/temp/epic-<id>/baselines/<kind>.json`
|
|
72
|
+
*/
|
|
73
|
+
export function epicSnapshotPathFor({ epicId, kind, cwd = process.cwd() }) {
|
|
74
|
+
if (!Number.isInteger(epicId) || epicId <= 0) {
|
|
75
|
+
throw new TypeError(
|
|
76
|
+
'[baseline-snapshot] epicId must be a positive integer',
|
|
77
|
+
);
|
|
78
|
+
}
|
|
79
|
+
if (kind !== 'maintainability' && kind !== 'crap') {
|
|
80
|
+
throw new TypeError(
|
|
81
|
+
`[baseline-snapshot] kind must be one of ${EPIC_BASELINES.join(', ')}`,
|
|
82
|
+
);
|
|
83
|
+
}
|
|
84
|
+
return path.resolve(
|
|
85
|
+
cwd,
|
|
86
|
+
'temp',
|
|
87
|
+
`epic-${epicId}`,
|
|
88
|
+
'baselines',
|
|
89
|
+
`${kind}.json`,
|
|
90
|
+
);
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
/**
|
|
94
|
+
* Fork the tracked main baselines into `temp/epic-<id>/baselines/`. Idempotent.
|
|
95
|
+
*
|
|
96
|
+
* Source paths are resolved through the agent-settings config so a repo that
|
|
97
|
+
* relocates its baselines (`delivery.quality.gates.{maintainability,crap}.baselinePath`)
|
|
98
|
+
* is honoured. Destination layout is fixed at `temp/epic-<id>/baselines/<kind>.json`
|
|
99
|
+
* so the close-validation gate's `--epic-ref` resolution stays predictable, and
|
|
100
|
+
* the per-epic temp-tree cleanup reaps them on Story merge with no extra wiring.
|
|
101
|
+
*
|
|
102
|
+
* Failure modes:
|
|
103
|
+
* - Source baseline missing → returned per-file `{ written: false,
|
|
104
|
+
* reason: 'source-missing' }`. Logger warn fires once per missing file.
|
|
105
|
+
* Caller stays in `--full-scope` mode.
|
|
106
|
+
* - Source unreadable / not parseable → throws. Re-running /epic-plan
|
|
107
|
+
* with `--force` after fixing the source recovers.
|
|
108
|
+
*
|
|
109
|
+
* @param {{
|
|
110
|
+
* epicId: number,
|
|
111
|
+
* cwd?: string,
|
|
112
|
+
* force?: boolean, // unused at this layer; reserved
|
|
113
|
+
* resolveConfig?: typeof defaultResolveConfig,
|
|
114
|
+
* getBaselines?: typeof defaultGetBaselines,
|
|
115
|
+
* logger?: { warn?: (m: string) => void, info?: (m: string) => void },
|
|
116
|
+
* fsImpl?: { existsSync: typeof fs.existsSync, readFileSync: typeof fs.readFileSync, writeFileSync: typeof fs.writeFileSync, mkdirSync: typeof fs.mkdirSync },
|
|
117
|
+
* }} opts
|
|
118
|
+
* @returns {{
|
|
119
|
+
* epicId: number,
|
|
120
|
+
* results: Array<{
|
|
121
|
+
* kind: 'maintainability'|'crap',
|
|
122
|
+
* source: string,
|
|
123
|
+
* destination: string,
|
|
124
|
+
* written: boolean,
|
|
125
|
+
* reason?: 'source-missing'|'idempotent'|'fresh',
|
|
126
|
+
* }>,
|
|
127
|
+
* }}
|
|
128
|
+
*/
|
|
129
|
+
export function forkMainToEpic({
|
|
130
|
+
epicId,
|
|
131
|
+
cwd = process.cwd(),
|
|
132
|
+
resolveConfig = defaultResolveConfig,
|
|
133
|
+
getBaselines = defaultGetBaselines,
|
|
134
|
+
logger = console,
|
|
135
|
+
fsImpl = fs,
|
|
136
|
+
} = {}) {
|
|
137
|
+
if (!Number.isInteger(epicId) || epicId <= 0) {
|
|
138
|
+
throw new TypeError(
|
|
139
|
+
'[baseline-snapshot] forkMainToEpic: epicId must be a positive integer',
|
|
140
|
+
);
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
const config = resolveConfig({ cwd });
|
|
144
|
+
const baselines = getBaselines(config);
|
|
145
|
+
const results = [];
|
|
146
|
+
|
|
147
|
+
for (const kind of EPIC_BASELINES) {
|
|
148
|
+
const sourceRel = baselines?.[kind]?.path;
|
|
149
|
+
if (typeof sourceRel !== 'string' || sourceRel.length === 0) {
|
|
150
|
+
logger.warn?.(
|
|
151
|
+
`[baseline-snapshot] no configured path for ${kind} baseline — skipping fork.`,
|
|
152
|
+
);
|
|
153
|
+
results.push({
|
|
154
|
+
kind,
|
|
155
|
+
source: '',
|
|
156
|
+
destination: epicSnapshotPathFor({ epicId, kind, cwd }),
|
|
157
|
+
written: false,
|
|
158
|
+
reason: 'source-missing',
|
|
159
|
+
});
|
|
160
|
+
continue;
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
const sourceAbs = path.isAbsolute(sourceRel)
|
|
164
|
+
? sourceRel
|
|
165
|
+
: path.resolve(cwd, sourceRel);
|
|
166
|
+
const destinationAbs = epicSnapshotPathFor({ epicId, kind, cwd });
|
|
167
|
+
|
|
168
|
+
if (!fsImpl.existsSync(sourceAbs)) {
|
|
169
|
+
logger.warn?.(
|
|
170
|
+
`[baseline-snapshot] ⚠ source baseline missing for ${kind} at ${sourceRel} — fork skipped (gate stays in --full-scope mode).`,
|
|
171
|
+
);
|
|
172
|
+
results.push({
|
|
173
|
+
kind,
|
|
174
|
+
source: sourceAbs,
|
|
175
|
+
destination: destinationAbs,
|
|
176
|
+
written: false,
|
|
177
|
+
reason: 'source-missing',
|
|
178
|
+
});
|
|
179
|
+
continue;
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
const sourceBytes = fsImpl.readFileSync(sourceAbs, 'utf8');
|
|
183
|
+
|
|
184
|
+
let existingBytes = null;
|
|
185
|
+
if (fsImpl.existsSync(destinationAbs)) {
|
|
186
|
+
try {
|
|
187
|
+
existingBytes = fsImpl.readFileSync(destinationAbs, 'utf8');
|
|
188
|
+
} catch {
|
|
189
|
+
existingBytes = null;
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
if (existingBytes === sourceBytes) {
|
|
194
|
+
results.push({
|
|
195
|
+
kind,
|
|
196
|
+
source: sourceAbs,
|
|
197
|
+
destination: destinationAbs,
|
|
198
|
+
written: false,
|
|
199
|
+
reason: 'idempotent',
|
|
200
|
+
});
|
|
201
|
+
continue;
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
fsImpl.mkdirSync(path.dirname(destinationAbs), { recursive: true });
|
|
205
|
+
fsImpl.writeFileSync(destinationAbs, sourceBytes);
|
|
206
|
+
logger.info?.(
|
|
207
|
+
`[baseline-snapshot] forked ${kind} baseline → ${path.relative(cwd, destinationAbs)}`,
|
|
208
|
+
);
|
|
209
|
+
results.push({
|
|
210
|
+
kind,
|
|
211
|
+
source: sourceAbs,
|
|
212
|
+
destination: destinationAbs,
|
|
213
|
+
written: true,
|
|
214
|
+
reason: 'fresh',
|
|
215
|
+
});
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
return { epicId, results };
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
/**
|
|
222
|
+
* Author a single planning commit on `epic/<id>` that adds the per-Epic
|
|
223
|
+
* baseline snapshots, without disturbing the live working tree or HEAD.
|
|
224
|
+
*
|
|
225
|
+
* Implementation strategy: build a fresh, isolated git index seeded from the
|
|
226
|
+
* Epic branch's tree (`read-tree`), `update-index --add` the snapshot blobs
|
|
227
|
+
* (sourced via `hash-object -w`), `write-tree` against that index, and
|
|
228
|
+
* `commit-tree` the result with the Epic branch as parent. The commit is
|
|
229
|
+
* then attached via `update-ref refs/heads/epic/<id>`. The live worktree
|
|
230
|
+
* `.git/index` is never touched — we route every git invocation through a
|
|
231
|
+
* temporary `GIT_INDEX_FILE`.
|
|
232
|
+
*
|
|
233
|
+
* Idempotent: when the resulting tree equals the parent's tree (because the
|
|
234
|
+
* blobs were already on the Epic branch), no commit is made and the helper
|
|
235
|
+
* returns `{ committed: false, reason: 'no-change' }`.
|
|
236
|
+
*
|
|
237
|
+
* Pre-conditions:
|
|
238
|
+
* - `epic/<id>` ref exists (caller has invoked `ensureEpicBranchRef`).
|
|
239
|
+
* - The destination snapshot files exist on disk (call `forkMainToEpic`
|
|
240
|
+
* immediately before this helper).
|
|
241
|
+
*
|
|
242
|
+
* @param {{
|
|
243
|
+
* epicId: number,
|
|
244
|
+
* cwd?: string,
|
|
245
|
+
* epicBranch?: string,
|
|
246
|
+
* message?: string,
|
|
247
|
+
* files?: Array<{ destination: string }>, // accepts forkMainToEpic results
|
|
248
|
+
* gitSpawn?: typeof defaultGitSpawn,
|
|
249
|
+
* logger?: { info?: (m: string) => void, warn?: (m: string) => void },
|
|
250
|
+
* }} opts
|
|
251
|
+
* @returns {{ committed: boolean, sha?: string, reason?: 'no-change'|'no-files'|'epic-missing', detail?: string }}
|
|
252
|
+
*/
|
|
253
|
+
export function commitSnapshotsToEpicBranch({
|
|
254
|
+
epicId,
|
|
255
|
+
cwd = process.cwd(),
|
|
256
|
+
epicBranch = `epic/${epicId}`,
|
|
257
|
+
message = `chore(baseline-snapshot): seed per-epic snapshots for epic-${epicId}`,
|
|
258
|
+
files = [],
|
|
259
|
+
spawnSync = defaultSpawnSync,
|
|
260
|
+
fsImpl = fs,
|
|
261
|
+
logger = console,
|
|
262
|
+
} = {}) {
|
|
263
|
+
if (!Number.isInteger(epicId) || epicId <= 0) {
|
|
264
|
+
throw new TypeError(
|
|
265
|
+
'[baseline-snapshot] commitSnapshotsToEpicBranch: epicId must be a positive integer',
|
|
266
|
+
);
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
// Filter to files that actually exist on disk and are under cwd. The helper
|
|
270
|
+
// is purely additive — it never deletes — so files: [] short-circuits.
|
|
271
|
+
const targets = files
|
|
272
|
+
.filter((f) => f && typeof f.destination === 'string')
|
|
273
|
+
.filter((f) => fsImpl.existsSync(f.destination))
|
|
274
|
+
.map((f) => ({
|
|
275
|
+
abs: f.destination,
|
|
276
|
+
rel: path.relative(cwd, f.destination).split(path.sep).join('/'),
|
|
277
|
+
}));
|
|
278
|
+
if (targets.length === 0) {
|
|
279
|
+
return { committed: false, reason: 'no-files' };
|
|
280
|
+
}
|
|
281
|
+
|
|
282
|
+
function runGit(args, extraEnv = {}) {
|
|
283
|
+
const result = spawnSync('git', args, {
|
|
284
|
+
cwd,
|
|
285
|
+
encoding: 'utf-8',
|
|
286
|
+
stdio: 'pipe',
|
|
287
|
+
shell: false,
|
|
288
|
+
env: { ...process.env, ...extraEnv },
|
|
289
|
+
});
|
|
290
|
+
return {
|
|
291
|
+
status: result.status ?? 1,
|
|
292
|
+
stdout: (result.stdout ?? '').trim(),
|
|
293
|
+
stderr: (result.stderr ?? '').trim(),
|
|
294
|
+
};
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
// Verify the epic branch ref exists before doing any plumbing work.
|
|
298
|
+
const verify = runGit(['rev-parse', '--verify', epicBranch]);
|
|
299
|
+
if (verify.status !== 0) {
|
|
300
|
+
return {
|
|
301
|
+
committed: false,
|
|
302
|
+
reason: 'epic-missing',
|
|
303
|
+
detail: `epic branch ref ${epicBranch} does not exist`,
|
|
304
|
+
};
|
|
305
|
+
}
|
|
306
|
+
const parentSha = verify.stdout;
|
|
307
|
+
|
|
308
|
+
// Allocate an isolated index file so the live `.git/index` never moves.
|
|
309
|
+
const tmpIndex = path.join(
|
|
310
|
+
os.tmpdir(),
|
|
311
|
+
`baseline-snapshot-${epicId}-${process.pid}-${Date.now()}.index`,
|
|
312
|
+
);
|
|
313
|
+
const env = { GIT_INDEX_FILE: tmpIndex };
|
|
314
|
+
|
|
315
|
+
try {
|
|
316
|
+
// Seed the index from the Epic branch tree.
|
|
317
|
+
const readTree = runGit(['read-tree', epicBranch], env);
|
|
318
|
+
if (readTree.status !== 0) {
|
|
319
|
+
return {
|
|
320
|
+
committed: false,
|
|
321
|
+
reason: 'epic-missing',
|
|
322
|
+
detail: `read-tree failed: ${readTree.stderr || readTree.stdout}`,
|
|
323
|
+
};
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
// Hash each blob (writing it to the object DB) and stage it in the
|
|
327
|
+
// temp index.
|
|
328
|
+
for (const t of targets) {
|
|
329
|
+
const hashRes = runGit(['hash-object', '-w', '--', t.abs]);
|
|
330
|
+
if (hashRes.status !== 0) {
|
|
331
|
+
throw new Error(
|
|
332
|
+
`[baseline-snapshot] hash-object failed for ${t.rel}: ${hashRes.stderr}`,
|
|
333
|
+
);
|
|
334
|
+
}
|
|
335
|
+
const blobSha = hashRes.stdout;
|
|
336
|
+
const updateIdx = runGit(
|
|
337
|
+
['update-index', '--add', '--cacheinfo', `100644,${blobSha},${t.rel}`],
|
|
338
|
+
env,
|
|
339
|
+
);
|
|
340
|
+
if (updateIdx.status !== 0) {
|
|
341
|
+
throw new Error(
|
|
342
|
+
`[baseline-snapshot] update-index failed for ${t.rel}: ${updateIdx.stderr}`,
|
|
343
|
+
);
|
|
344
|
+
}
|
|
345
|
+
}
|
|
346
|
+
|
|
347
|
+
// Write the staged tree.
|
|
348
|
+
const writeTree = runGit(['write-tree'], env);
|
|
349
|
+
if (writeTree.status !== 0) {
|
|
350
|
+
throw new Error(
|
|
351
|
+
`[baseline-snapshot] write-tree failed: ${writeTree.stderr}`,
|
|
352
|
+
);
|
|
353
|
+
}
|
|
354
|
+
const newTreeSha = writeTree.stdout;
|
|
355
|
+
|
|
356
|
+
// Compare against the parent tree — skip the commit when nothing moved.
|
|
357
|
+
const parentTreeRes = runGit(['rev-parse', `${parentSha}^{tree}`]);
|
|
358
|
+
if (parentTreeRes.status === 0 && parentTreeRes.stdout === newTreeSha) {
|
|
359
|
+
return { committed: false, reason: 'no-change' };
|
|
360
|
+
}
|
|
361
|
+
|
|
362
|
+
// Author the commit and attach it to the Epic branch ref.
|
|
363
|
+
const commitRes = runGit([
|
|
364
|
+
'commit-tree',
|
|
365
|
+
newTreeSha,
|
|
366
|
+
'-p',
|
|
367
|
+
parentSha,
|
|
368
|
+
'-m',
|
|
369
|
+
message,
|
|
370
|
+
]);
|
|
371
|
+
if (commitRes.status !== 0) {
|
|
372
|
+
throw new Error(
|
|
373
|
+
`[baseline-snapshot] commit-tree failed: ${commitRes.stderr}`,
|
|
374
|
+
);
|
|
375
|
+
}
|
|
376
|
+
const newCommitSha = commitRes.stdout;
|
|
377
|
+
|
|
378
|
+
const updateRef = runGit([
|
|
379
|
+
'update-ref',
|
|
380
|
+
`refs/heads/${epicBranch}`,
|
|
381
|
+
newCommitSha,
|
|
382
|
+
parentSha,
|
|
383
|
+
]);
|
|
384
|
+
if (updateRef.status !== 0) {
|
|
385
|
+
throw new Error(
|
|
386
|
+
`[baseline-snapshot] update-ref failed: ${updateRef.stderr}`,
|
|
387
|
+
);
|
|
388
|
+
}
|
|
389
|
+
|
|
390
|
+
logger.info?.(
|
|
391
|
+
`[baseline-snapshot] committed ${targets.length} snapshot file(s) to ${epicBranch} (${newCommitSha.slice(0, 7)}).`,
|
|
392
|
+
);
|
|
393
|
+
return { committed: true, sha: newCommitSha };
|
|
394
|
+
} finally {
|
|
395
|
+
// Best-effort cleanup of the temp index file.
|
|
396
|
+
try {
|
|
397
|
+
if (fsImpl.existsSync(tmpIndex)) fsImpl.unlinkSync(tmpIndex);
|
|
398
|
+
} catch {
|
|
399
|
+
// ignore — temp file in OS tmpdir, not our problem long-term
|
|
400
|
+
}
|
|
401
|
+
}
|
|
402
|
+
}
|
|
403
|
+
|
|
404
|
+
/**
|
|
405
|
+
* Re-score the main baselines from the current working tree and write the
|
|
406
|
+
* result back to the tracked baseline paths.
|
|
407
|
+
*
|
|
408
|
+
* Story #2135 / Task #2145 — rewritten to route every write through the
|
|
409
|
+
* shared `lib/baselines/writer.js` funnel. The legacy `saveMaintainabilityFn`
|
|
410
|
+
* / `saveCrapFn` injection seams are gone — the writer is itself the seam
|
|
411
|
+
* (`writeFn` + `writeFileFn`), and the on-disk envelopes are now the
|
|
412
|
+
* canonical V2 shape that `lib/baselines/reader.js` schema-validates.
|
|
413
|
+
*
|
|
414
|
+
* Change detection now uses the writer's structural-equality short-circuit
|
|
415
|
+
* rather than write-then-byte-compare: we read the prior envelope through
|
|
416
|
+
* the reader, pass it to the writer as `priorEnvelope`, and inspect the
|
|
417
|
+
* returned envelope's `generatedAt`. When the rows + rollup are
|
|
418
|
+
* structurally equal to the prior, the writer returns the prior envelope
|
|
419
|
+
* unchanged and `didChange` stays false — no `writeFile` is invoked, the
|
|
420
|
+
* on-disk bytes are guaranteed identical.
|
|
421
|
+
*
|
|
422
|
+
* Returns `{ didChange, files }` so callers (epic-deliver-finalize) can decide
|
|
423
|
+
* whether to author a `baseline-refresh: epic-<id>` commit. `didChange` is the
|
|
424
|
+
* union of per-file change detection — if any baseline's bytes change, the
|
|
425
|
+
* commit is needed.
|
|
426
|
+
*
|
|
427
|
+
* Coverage source for crap regeneration defaults to `coverage/coverage-final.json`
|
|
428
|
+
* via `delivery.quality.gates.crap.coveragePath`. When coverage is missing and
|
|
429
|
+
* `requireCoverage` is true, the crap regeneration is skipped (didChange stays
|
|
430
|
+
* false for that file) and a warn is emitted — the operator is expected to run
|
|
431
|
+
* `npm run test:coverage` before /epic-deliver if a refresh is anticipated.
|
|
432
|
+
*
|
|
433
|
+
* @param {{
|
|
434
|
+
* cwd?: string,
|
|
435
|
+
* resolveConfig?: typeof defaultResolveConfig,
|
|
436
|
+
* getBaselines?: typeof defaultGetBaselines,
|
|
437
|
+
* getQuality?: typeof defaultGetQuality,
|
|
438
|
+
* logger?: { warn?: (m: string) => void, info?: (m: string) => void },
|
|
439
|
+
* fsImpl?: { existsSync: typeof fs.existsSync, readFileSync: typeof fs.readFileSync, writeFileSync: typeof fs.writeFileSync, mkdirSync: typeof fs.mkdirSync, renameSync?: typeof fs.renameSync },
|
|
440
|
+
* scanDirectoryFn?: typeof scanDirectory,
|
|
441
|
+
* calculateAllFn?: typeof calculateAll,
|
|
442
|
+
* scanAndScoreFn?: typeof scanAndScore,
|
|
443
|
+
* loadCoverageFn?: typeof loadCoverage,
|
|
444
|
+
* resolveEscomplexVersionFn?: typeof resolveEscomplexVersion,
|
|
445
|
+
* resolveTsTranspilerVersionFn?: typeof resolveTsTranspilerVersion,
|
|
446
|
+
* writeFn?: typeof defaultWriteBaseline,
|
|
447
|
+
* writeFileFn?: typeof defaultWriteBaselineFile,
|
|
448
|
+
* loadPriorFn?: (absPath: string, kind: string) => object | null,
|
|
449
|
+
* }} [opts]
|
|
450
|
+
* @returns {Promise<{
|
|
451
|
+
* didChange: boolean,
|
|
452
|
+
* files: Array<{ kind: 'maintainability'|'crap', path: string, didChange: boolean, reason?: 'no-coverage'|'unchanged'|'updated' }>,
|
|
453
|
+
* }>}
|
|
454
|
+
*/
|
|
455
|
+
export async function regenerateMainFromTree({
|
|
456
|
+
cwd = process.cwd(),
|
|
457
|
+
resolveConfig = defaultResolveConfig,
|
|
458
|
+
getBaselines = defaultGetBaselines,
|
|
459
|
+
getQuality = defaultGetQuality,
|
|
460
|
+
logger = console,
|
|
461
|
+
fsImpl = fs,
|
|
462
|
+
scanDirectoryFn = scanDirectory,
|
|
463
|
+
calculateAllFn = calculateAll,
|
|
464
|
+
scanAndScoreFn = scanAndScore,
|
|
465
|
+
loadCoverageFn = loadCoverage,
|
|
466
|
+
resolveEscomplexVersionFn = resolveEscomplexVersion,
|
|
467
|
+
resolveTsTranspilerVersionFn = resolveTsTranspilerVersion,
|
|
468
|
+
writeFn = defaultWriteBaseline,
|
|
469
|
+
writeFileFn = defaultWriteBaselineFile,
|
|
470
|
+
loadPriorFn = defaultLoadPriorEnvelope,
|
|
471
|
+
} = {}) {
|
|
472
|
+
const config = resolveConfig({ cwd });
|
|
473
|
+
const baselines = getBaselines(config);
|
|
474
|
+
const quality = getQuality(config);
|
|
475
|
+
|
|
476
|
+
const files = [];
|
|
477
|
+
let didChange = false;
|
|
478
|
+
|
|
479
|
+
// ── maintainability ──────────────────────────────────────────────────────
|
|
480
|
+
const miPath = baselines?.maintainability?.path;
|
|
481
|
+
const miTargetDirs = quality?.maintainability?.targetDirs ?? [];
|
|
482
|
+
const miIgnoreGlobs = quality?.maintainability?.ignoreGlobs ?? [];
|
|
483
|
+
// Hoisted so the CRAP pass can reuse it when targetDirs match — avoids a
|
|
484
|
+
// second full-tree walk over the same directories (Story #3663).
|
|
485
|
+
let miSourceList = null;
|
|
486
|
+
if (typeof miPath === 'string' && miPath.length > 0) {
|
|
487
|
+
const miAbs = path.isAbsolute(miPath) ? miPath : path.resolve(cwd, miPath);
|
|
488
|
+
miSourceList = [];
|
|
489
|
+
for (const dir of miTargetDirs) {
|
|
490
|
+
const abs = path.isAbsolute(dir) ? dir : path.resolve(cwd, dir);
|
|
491
|
+
scanDirectoryFn(abs, miSourceList, { cwd, ignoreGlobs: miIgnoreGlobs });
|
|
492
|
+
}
|
|
493
|
+
const scores = await calculateAllFn(miSourceList);
|
|
494
|
+
|
|
495
|
+
// Project the scoring helper's `{path: mi}` map onto the writer's
|
|
496
|
+
// canonical row shape. Story #2079 path-canon defence stays in place —
|
|
497
|
+
// the writer would canonicalise again, but doing it here keeps any
|
|
498
|
+
// pre-canonicalised comparison inside the function meaningful.
|
|
499
|
+
const miRows = filterExcludedRows(
|
|
500
|
+
Object.entries(scores).map(([key, mi]) => {
|
|
501
|
+
const rel = path.isAbsolute(key) ? path.relative(cwd, key) : key;
|
|
502
|
+
const posixRel = rel.split(path.sep).join('/');
|
|
503
|
+
return { path: canonicalisePath(posixRel), mi };
|
|
504
|
+
}),
|
|
505
|
+
);
|
|
506
|
+
|
|
507
|
+
const priorMi = loadPriorFn(miAbs, 'maintainability');
|
|
508
|
+
const envelope = writeFn({
|
|
509
|
+
kind: 'maintainability',
|
|
510
|
+
rows: miRows,
|
|
511
|
+
priorEnvelope: priorMi,
|
|
512
|
+
});
|
|
513
|
+
if (priorMi && envelope === priorMi) {
|
|
514
|
+
// Structural-equality short-circuit fired — on-disk bytes are
|
|
515
|
+
// guaranteed identical, no writeFile invocation needed.
|
|
516
|
+
files.push({
|
|
517
|
+
kind: 'maintainability',
|
|
518
|
+
path: miAbs,
|
|
519
|
+
didChange: false,
|
|
520
|
+
reason: 'unchanged',
|
|
521
|
+
});
|
|
522
|
+
} else {
|
|
523
|
+
writeFileFn(miAbs, envelope, { fsImpl });
|
|
524
|
+
didChange = true;
|
|
525
|
+
files.push({
|
|
526
|
+
kind: 'maintainability',
|
|
527
|
+
path: miAbs,
|
|
528
|
+
didChange: true,
|
|
529
|
+
reason: 'updated',
|
|
530
|
+
});
|
|
531
|
+
}
|
|
532
|
+
}
|
|
533
|
+
|
|
534
|
+
// ── crap ─────────────────────────────────────────────────────────────────
|
|
535
|
+
const crapPath = baselines?.crap?.path;
|
|
536
|
+
const crapCfg = quality?.crap ?? {};
|
|
537
|
+
const crapTargetDirs = Array.isArray(crapCfg.targetDirs)
|
|
538
|
+
? crapCfg.targetDirs
|
|
539
|
+
: [];
|
|
540
|
+
const crapIgnoreGlobs = Array.isArray(crapCfg.ignoreGlobs)
|
|
541
|
+
? crapCfg.ignoreGlobs
|
|
542
|
+
: [];
|
|
543
|
+
const requireCoverage = crapCfg.requireCoverage !== false;
|
|
544
|
+
const coveragePath = crapCfg.coveragePath ?? 'coverage/coverage-final.json';
|
|
545
|
+
if (typeof crapPath === 'string' && crapPath.length > 0) {
|
|
546
|
+
const crapAbs = path.isAbsolute(crapPath)
|
|
547
|
+
? crapPath
|
|
548
|
+
: path.resolve(cwd, crapPath);
|
|
549
|
+
const coverageAbs = path.isAbsolute(coveragePath)
|
|
550
|
+
? coveragePath
|
|
551
|
+
: path.resolve(cwd, coveragePath);
|
|
552
|
+
const coverage = loadCoverageFn(coverageAbs);
|
|
553
|
+
if (!coverage && requireCoverage) {
|
|
554
|
+
logger.warn?.(
|
|
555
|
+
`[baseline-snapshot] ⚠ no coverage at ${coveragePath} — skipping crap regeneration (refresh stays clean for this file).`,
|
|
556
|
+
);
|
|
557
|
+
files.push({
|
|
558
|
+
kind: 'crap',
|
|
559
|
+
path: crapAbs,
|
|
560
|
+
didChange: false,
|
|
561
|
+
reason: 'no-coverage',
|
|
562
|
+
});
|
|
563
|
+
} else {
|
|
564
|
+
// Reuse the MI scan's file list when CRAP and MI target the same
|
|
565
|
+
// directories with the same ignore globs — avoids a second full-tree
|
|
566
|
+
// walk over identical source trees (Story #3663).
|
|
567
|
+
const crapDirsMatchMi =
|
|
568
|
+
miSourceList !== null &&
|
|
569
|
+
crapTargetDirs.length === miTargetDirs.length &&
|
|
570
|
+
crapTargetDirs.every((d, i) => d === miTargetDirs[i]) &&
|
|
571
|
+
crapIgnoreGlobs.length === miIgnoreGlobs.length &&
|
|
572
|
+
crapIgnoreGlobs.every((g, i) => g === miIgnoreGlobs[i]);
|
|
573
|
+
const { rows } = await scanAndScoreFn({
|
|
574
|
+
targetDirs: crapTargetDirs,
|
|
575
|
+
coverage,
|
|
576
|
+
requireCoverage,
|
|
577
|
+
cwd,
|
|
578
|
+
ignoreGlobs: crapIgnoreGlobs,
|
|
579
|
+
...(crapDirsMatchMi && { preScannedFiles: miSourceList }),
|
|
580
|
+
});
|
|
581
|
+
// scanAndScore yields rows keyed by `file:`; the per-kind crap module's
|
|
582
|
+
// `projectRow` handles `path ?? file`, so the writer takes either.
|
|
583
|
+
// Filter to actually-scored rows here (crap is nullable for trivial
|
|
584
|
+
// methods); the writer's `assertEnvelope` would reject otherwise.
|
|
585
|
+
const crapRows = (rows ?? []).filter(
|
|
586
|
+
(r) => typeof r?.crap === 'number' && Number.isFinite(r.crap),
|
|
587
|
+
);
|
|
588
|
+
|
|
589
|
+
// CRAP gates need the running scorer's versions present on the
|
|
590
|
+
// envelope-adjacent shape; the V2 envelope itself only carries
|
|
591
|
+
// `kernelVersion`, so we stamp escomplex/tsTranspiler via the writer's
|
|
592
|
+
// `kernelVersion` override and let the existing per-kind module
|
|
593
|
+
// resolve the rest. We also resolve them eagerly so a test stub can
|
|
594
|
+
// pin them deterministically.
|
|
595
|
+
resolveEscomplexVersionFn(cwd);
|
|
596
|
+
resolveTsTranspilerVersionFn();
|
|
597
|
+
|
|
598
|
+
const priorCrap = loadPriorFn(crapAbs, 'crap');
|
|
599
|
+
const envelope = writeFn({
|
|
600
|
+
kind: 'crap',
|
|
601
|
+
rows: crapRows,
|
|
602
|
+
priorEnvelope: priorCrap,
|
|
603
|
+
});
|
|
604
|
+
if (priorCrap && envelope === priorCrap) {
|
|
605
|
+
files.push({
|
|
606
|
+
kind: 'crap',
|
|
607
|
+
path: crapAbs,
|
|
608
|
+
didChange: false,
|
|
609
|
+
reason: 'unchanged',
|
|
610
|
+
});
|
|
611
|
+
} else {
|
|
612
|
+
writeFileFn(crapAbs, envelope, { fsImpl });
|
|
613
|
+
didChange = true;
|
|
614
|
+
files.push({
|
|
615
|
+
kind: 'crap',
|
|
616
|
+
path: crapAbs,
|
|
617
|
+
didChange: true,
|
|
618
|
+
reason: 'updated',
|
|
619
|
+
});
|
|
620
|
+
}
|
|
621
|
+
}
|
|
622
|
+
}
|
|
623
|
+
|
|
624
|
+
return { didChange, files };
|
|
625
|
+
}
|
|
626
|
+
|
|
627
|
+
/**
|
|
628
|
+
* Story #2135 / Task #2145 — load the prior envelope for the structural-
|
|
629
|
+
* equality short-circuit. Reads the file through `reader.loadFile` (which
|
|
630
|
+
* schema-validates against the per-kind envelope) and synthesises an
|
|
631
|
+
* envelope object the writer can compare against. Returns `null` when the
|
|
632
|
+
* file is missing, unreadable, or fails schema validation — in which case
|
|
633
|
+
* the writer falls through to the normal stamp-and-write path.
|
|
634
|
+
*
|
|
635
|
+
* Exported as the default `loadPriorFn` so tests can replace it without
|
|
636
|
+
* monkey-patching the module surface.
|
|
637
|
+
*/
|
|
638
|
+
function defaultLoadPriorEnvelope(absPath, kind) {
|
|
639
|
+
try {
|
|
640
|
+
const parsed = readerLoadFile(absPath, { kind });
|
|
641
|
+
if (!parsed || !Array.isArray(parsed.rows)) return null;
|
|
642
|
+
// Synthesise the envelope shape the writer's short-circuit expects.
|
|
643
|
+
return {
|
|
644
|
+
$schema: `.agents/schemas/baselines/${kind}.schema.json`,
|
|
645
|
+
kernelVersion: parsed.kernelVersion,
|
|
646
|
+
generatedAt: parsed.generatedAt,
|
|
647
|
+
rollup: parsed.rollup,
|
|
648
|
+
rows: parsed.rows,
|
|
649
|
+
};
|
|
650
|
+
} catch {
|
|
651
|
+
return null;
|
|
652
|
+
}
|
|
653
|
+
}
|
|
654
|
+
|
|
655
|
+
/**
|
|
656
|
+
* Story #1396 (re-targeted by Story #1467; relocated by Story #1585):
|
|
657
|
+
* fork the tracked main baselines into `temp/epic/<id>/baselines/` and
|
|
658
|
+
* commit the snapshots onto the Epic branch. Originally lived in
|
|
659
|
+
* `epic-plan-spec.js`; relocated to the lower-level module so callers
|
|
660
|
+
* (notably `lib/story-init/branch-initializer.js`) do not need to import
|
|
661
|
+
* the heavy CLI script.
|
|
662
|
+
*
|
|
663
|
+
* `epic-plan-spec.js` re-exports this symbol to preserve the historic
|
|
664
|
+
* import path and the existing test suite.
|
|
665
|
+
*
|
|
666
|
+
* Failure modes are non-fatal: a missing source baseline downgrades to a
|
|
667
|
+
* `--full-scope` warning, an unresolvable Epic branch is logged and
|
|
668
|
+
* skipped, and the helper never throws into the caller. Idempotent: the
|
|
669
|
+
* downstream `commitSnapshotsToEpicBranch` returns `no-change` when the
|
|
670
|
+
* staged tree matches the Epic branch tip, so subsequent invocations on
|
|
671
|
+
* the same Epic produce no new commit.
|
|
672
|
+
*
|
|
673
|
+
* @param {{
|
|
674
|
+
* epicId: number,
|
|
675
|
+
* cwd?: string,
|
|
676
|
+
* baseBranch?: string,
|
|
677
|
+
* logger?: object,
|
|
678
|
+
* forkFn?: typeof forkMainToEpic,
|
|
679
|
+
* commitFn?: typeof commitSnapshotsToEpicBranch,
|
|
680
|
+
* ensureEpicBranchRefFn?: typeof defaultEnsureEpicBranchRef,
|
|
681
|
+
* }} opts
|
|
682
|
+
* @returns {{ fork: object, commit: object }}
|
|
683
|
+
*/
|
|
684
|
+
export function forkAndCommitEpicSnapshot({
|
|
685
|
+
epicId,
|
|
686
|
+
cwd = PROJECT_ROOT,
|
|
687
|
+
baseBranch = 'main',
|
|
688
|
+
logger = console,
|
|
689
|
+
forkFn = forkMainToEpic,
|
|
690
|
+
commitFn = commitSnapshotsToEpicBranch,
|
|
691
|
+
ensureEpicBranchRefFn = defaultEnsureEpicBranchRef,
|
|
692
|
+
} = {}) {
|
|
693
|
+
const epicBranch = `epic/${epicId}`;
|
|
694
|
+
try {
|
|
695
|
+
ensureEpicBranchRefFn(epicBranch, baseBranch, cwd, {
|
|
696
|
+
progress: () => {},
|
|
697
|
+
});
|
|
698
|
+
} catch (err) {
|
|
699
|
+
logger.warn?.(
|
|
700
|
+
`[baseline-snapshot] snapshot-fork: failed to ensure ${epicBranch}: ${err?.message ?? err}. Skipping fork.`,
|
|
701
|
+
);
|
|
702
|
+
return {
|
|
703
|
+
fork: { epicId, results: [] },
|
|
704
|
+
commit: { committed: false, reason: 'epic-missing' },
|
|
705
|
+
};
|
|
706
|
+
}
|
|
707
|
+
const fork = forkFn({ epicId, cwd, logger });
|
|
708
|
+
const commit = commitFn({
|
|
709
|
+
epicId,
|
|
710
|
+
cwd,
|
|
711
|
+
epicBranch,
|
|
712
|
+
files: fork.results.filter((r) => r.written || r.reason === 'idempotent'),
|
|
713
|
+
logger,
|
|
714
|
+
});
|
|
715
|
+
return { fork, commit };
|
|
716
|
+
}
|