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,266 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* regression-projection.js — phase 2 of baseline-attribution.
|
|
3
|
+
*
|
|
4
|
+
* For each baseline-style gate (`check-maintainability`, `check-crap`),
|
|
5
|
+
* re-derive the per-file/per-method regression rows that `runPreMergeGates`
|
|
6
|
+
* would have surfaced on failure. These rows feed the attribution
|
|
7
|
+
* classifier downstream so attributable rows can trigger a refresh and
|
|
8
|
+
* non-attributable rows can route to a friction comment.
|
|
9
|
+
*
|
|
10
|
+
* Two projectors live here:
|
|
11
|
+
*
|
|
12
|
+
* - `projectMaintainabilityForGate` — reuses the MI projection from
|
|
13
|
+
* `close-validation.js` (Story #874).
|
|
14
|
+
* - `projectCrapForGate` / `projectCrapRegressions` — diff CRAP envelopes
|
|
15
|
+
* at `origin/<epicBranch>` vs `storyBranch`, optionally scoped to the
|
|
16
|
+
* Story's touched files (Story #1124).
|
|
17
|
+
*/
|
|
18
|
+
|
|
19
|
+
import { readBaselineAtRef as defaultReadBaselineAtRef } from '../../../../baseline-loader.js';
|
|
20
|
+
import { projectMaintainabilityRegressions as defaultProjectMaintainabilityRegressions } from '../../../../close-validation.js';
|
|
21
|
+
import { getBaselines as defaultGetBaselines } from '../../../../config-resolver.js';
|
|
22
|
+
import {
|
|
23
|
+
computeStoryDiffPaths,
|
|
24
|
+
validateProjectionContext,
|
|
25
|
+
} from './scope-discovery.js';
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* Default CRAP regression tolerance — mirrors `check-crap.js`. Score noise
|
|
29
|
+
* floor is ~0.01 from coverage rounding shifts across Node/V8 builds; a
|
|
30
|
+
* 0.05 tolerance clears that without admitting real regressions (those
|
|
31
|
+
* cross whole-integer thresholds and clear 0.05 trivially).
|
|
32
|
+
*/
|
|
33
|
+
const DEFAULT_CRAP_TOLERANCE = 0.05;
|
|
34
|
+
|
|
35
|
+
function coerceScopeSet(touchedFiles) {
|
|
36
|
+
if (touchedFiles == null) return null;
|
|
37
|
+
if (touchedFiles instanceof Set) return touchedFiles;
|
|
38
|
+
return new Set(touchedFiles);
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
// Story #1895: rows from the canonical envelope key by `path`; legacy
|
|
42
|
+
// rows key by `file`. Accept either so this attribution layer keeps
|
|
43
|
+
// working while the Epic migrates consumers off the legacy shape.
|
|
44
|
+
function rowFileKey(row) {
|
|
45
|
+
if (!row) return null;
|
|
46
|
+
if (typeof row.file === 'string') return row.file;
|
|
47
|
+
if (typeof row.path === 'string') return row.path;
|
|
48
|
+
return null;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
function indexCrapBaselineByMethod(baselineRows) {
|
|
52
|
+
const byMethod = new Map();
|
|
53
|
+
for (const b of baselineRows) {
|
|
54
|
+
const f = rowFileKey(b);
|
|
55
|
+
if (!f || typeof b.method !== 'string') continue;
|
|
56
|
+
const key = `${f}::${b.method}`;
|
|
57
|
+
if (!byMethod.has(key)) byMethod.set(key, []);
|
|
58
|
+
byMethod.get(key).push(b);
|
|
59
|
+
}
|
|
60
|
+
return byMethod;
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
function pickClosestUnseen(candidates, headStartLine, seen) {
|
|
64
|
+
let pick = null;
|
|
65
|
+
let bestDist = Number.POSITIVE_INFINITY;
|
|
66
|
+
for (const c of candidates) {
|
|
67
|
+
const f = rowFileKey(c);
|
|
68
|
+
const k = `${f}::${c.method}@${c.startLine}`;
|
|
69
|
+
if (seen.has(k)) continue;
|
|
70
|
+
const d = Math.abs((c.startLine ?? 0) - (headStartLine ?? 0));
|
|
71
|
+
if (d < bestDist) {
|
|
72
|
+
bestDist = d;
|
|
73
|
+
pick = c;
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
return pick;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
function isValidHeadRow(row) {
|
|
80
|
+
return Boolean(row && rowFileKey(row) && typeof row.method === 'string');
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
function buildCrapRegression(row, pick) {
|
|
84
|
+
const headCrap = typeof row.crap === 'number' ? row.crap : 0;
|
|
85
|
+
const baseCrap = typeof pick.crap === 'number' ? pick.crap : 0;
|
|
86
|
+
const f = rowFileKey(row);
|
|
87
|
+
return {
|
|
88
|
+
file: f,
|
|
89
|
+
path: f,
|
|
90
|
+
method: row.method,
|
|
91
|
+
startLine: row.startLine,
|
|
92
|
+
crap: headCrap,
|
|
93
|
+
projected: headCrap,
|
|
94
|
+
baseline: baseCrap,
|
|
95
|
+
drop: headCrap - baseCrap,
|
|
96
|
+
headCrap,
|
|
97
|
+
baseCrap,
|
|
98
|
+
};
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
/**
|
|
102
|
+
* Pure helper — given two CRAP baseline envelopes (`{ rows: [...] }`), produce
|
|
103
|
+
* the regression rows for methods whose `crap` score increased beyond
|
|
104
|
+
* `tolerance` between `baselineRows` and `headRows`. When `touchedFiles` is
|
|
105
|
+
* supplied (as a Set or array of repo-relative POSIX paths), rows are filtered
|
|
106
|
+
* to functions inside files the Story changed — sibling drift outside the
|
|
107
|
+
* Story's diff is excluded by construction, matching the maintainability
|
|
108
|
+
* projector's "touched-only" contract.
|
|
109
|
+
*
|
|
110
|
+
* Row shape mirrors the maintainability projector — `{ file, method,
|
|
111
|
+
* startLine, crap, baseline, drop, projected }` — so downstream attribution
|
|
112
|
+
* + refresh-commit logic (`classifyBaselineDrift`,
|
|
113
|
+
* `renderBaselineFrictionBody`) can read either projector's output with the
|
|
114
|
+
* same field accessors. `projected` is an alias for `crap` retained for
|
|
115
|
+
* shape compatibility with maintainability rows.
|
|
116
|
+
*
|
|
117
|
+
* Exported so unit tests can pin the diff math against a fixture pair of
|
|
118
|
+
* baseline envelopes without spawning `git`.
|
|
119
|
+
*/
|
|
120
|
+
export function diffCrapBaselines({
|
|
121
|
+
baselineRows,
|
|
122
|
+
headRows,
|
|
123
|
+
touchedFiles = null,
|
|
124
|
+
tolerance = DEFAULT_CRAP_TOLERANCE,
|
|
125
|
+
} = {}) {
|
|
126
|
+
if (!Array.isArray(baselineRows) || !Array.isArray(headRows)) return [];
|
|
127
|
+
const scope = coerceScopeSet(touchedFiles);
|
|
128
|
+
const byMethod = indexCrapBaselineByMethod(baselineRows);
|
|
129
|
+
const seen = new Set();
|
|
130
|
+
const regressions = [];
|
|
131
|
+
|
|
132
|
+
for (const row of headRows) {
|
|
133
|
+
if (!isValidHeadRow(row)) continue;
|
|
134
|
+
const rowFile = rowFileKey(row);
|
|
135
|
+
if (scope && !scope.has(rowFile)) continue;
|
|
136
|
+
const candidates = byMethod.get(`${rowFile}::${row.method}`);
|
|
137
|
+
if (!Array.isArray(candidates) || candidates.length === 0) continue;
|
|
138
|
+
const pick = pickClosestUnseen(candidates, row.startLine, seen);
|
|
139
|
+
if (!pick) continue;
|
|
140
|
+
seen.add(`${rowFileKey(pick)}::${pick.method}@${pick.startLine}`);
|
|
141
|
+
const entry = buildCrapRegression(row, pick);
|
|
142
|
+
if (entry.headCrap <= entry.baseCrap + tolerance) continue;
|
|
143
|
+
const { headCrap: _h, baseCrap: _b, ...publicEntry } = entry;
|
|
144
|
+
regressions.push(publicEntry);
|
|
145
|
+
}
|
|
146
|
+
return regressions;
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
export function projectCrapRegressions({
|
|
150
|
+
touchedFiles,
|
|
151
|
+
baselineRef,
|
|
152
|
+
headRef,
|
|
153
|
+
cwd,
|
|
154
|
+
baselinePath,
|
|
155
|
+
tolerance = DEFAULT_CRAP_TOLERANCE,
|
|
156
|
+
readBaselineAtRef = defaultReadBaselineAtRef,
|
|
157
|
+
getBaselines = defaultGetBaselines,
|
|
158
|
+
config,
|
|
159
|
+
} = {}) {
|
|
160
|
+
if (!baselineRef || !headRef) return [];
|
|
161
|
+
const resolvedPath = baselinePath ?? getBaselines(config)?.crap?.path;
|
|
162
|
+
if (!resolvedPath) return [];
|
|
163
|
+
|
|
164
|
+
let baselineEnv;
|
|
165
|
+
let headEnv;
|
|
166
|
+
try {
|
|
167
|
+
baselineEnv = readBaselineAtRef(baselineRef, resolvedPath, { cwd });
|
|
168
|
+
} catch {
|
|
169
|
+
return [];
|
|
170
|
+
}
|
|
171
|
+
try {
|
|
172
|
+
headEnv = readBaselineAtRef(headRef, resolvedPath, { cwd });
|
|
173
|
+
} catch {
|
|
174
|
+
return [];
|
|
175
|
+
}
|
|
176
|
+
const baselineRows = Array.isArray(baselineEnv?.rows) ? baselineEnv.rows : [];
|
|
177
|
+
const headRows = Array.isArray(headEnv?.rows) ? headEnv.rows : [];
|
|
178
|
+
return diffCrapBaselines({
|
|
179
|
+
baselineRows,
|
|
180
|
+
headRows,
|
|
181
|
+
touchedFiles,
|
|
182
|
+
tolerance,
|
|
183
|
+
});
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
/**
|
|
187
|
+
* Maintainability projector — extracts the same regression rows
|
|
188
|
+
* `runPreMergeGates` would have surfaced for `check-maintainability` by
|
|
189
|
+
* re-running the per-file MI ceiling projection against `origin/<epicBranch>`.
|
|
190
|
+
*
|
|
191
|
+
* Behaviour is preserved byte-for-byte from the pre-refactor early-return
|
|
192
|
+
* branch of `projectRegressionsForGate`: missing baseline path → `[]`, and
|
|
193
|
+
* the underlying `projectMaintainabilityRegressions` decides what counts as
|
|
194
|
+
* a regression row.
|
|
195
|
+
*
|
|
196
|
+
* @returns {Array<{ path?: string, file?: string }>}
|
|
197
|
+
*/
|
|
198
|
+
function projectMaintainabilityForGate({
|
|
199
|
+
cwd,
|
|
200
|
+
epicBranch,
|
|
201
|
+
storyBranch,
|
|
202
|
+
config,
|
|
203
|
+
projectMaintainability = defaultProjectMaintainabilityRegressions,
|
|
204
|
+
getBaselines = defaultGetBaselines,
|
|
205
|
+
}) {
|
|
206
|
+
const baselinePath = getBaselines(config)?.maintainability?.path;
|
|
207
|
+
if (!baselinePath) return [];
|
|
208
|
+
const projection = projectMaintainability({
|
|
209
|
+
cwd,
|
|
210
|
+
epicBranch,
|
|
211
|
+
storyBranch,
|
|
212
|
+
baselinePath,
|
|
213
|
+
});
|
|
214
|
+
return projection?.regressions ?? [];
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
function projectCrapForGate({
|
|
218
|
+
cwd,
|
|
219
|
+
epicBranch,
|
|
220
|
+
storyBranch,
|
|
221
|
+
config,
|
|
222
|
+
getBaselines = defaultGetBaselines,
|
|
223
|
+
readBaselineAtRef = defaultReadBaselineAtRef,
|
|
224
|
+
computeTouched = computeStoryDiffPaths,
|
|
225
|
+
projectCrap = projectCrapRegressions,
|
|
226
|
+
} = {}) {
|
|
227
|
+
if (!validateProjectionContext({ cwd, epicBranch, storyBranch })) return [];
|
|
228
|
+
const touchedFiles = new Set(
|
|
229
|
+
computeTouched({ cwd, epicBranch, storyBranch }),
|
|
230
|
+
);
|
|
231
|
+
return projectCrap({
|
|
232
|
+
touchedFiles,
|
|
233
|
+
baselineRef: `origin/${epicBranch}`,
|
|
234
|
+
headRef: storyBranch,
|
|
235
|
+
cwd,
|
|
236
|
+
config,
|
|
237
|
+
readBaselineAtRef,
|
|
238
|
+
getBaselines,
|
|
239
|
+
});
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
export const PROJECTORS = {
|
|
243
|
+
'check-maintainability': projectMaintainabilityForGate,
|
|
244
|
+
'check-crap': projectCrapForGate,
|
|
245
|
+
};
|
|
246
|
+
|
|
247
|
+
export function projectRegressionsForGate({
|
|
248
|
+
gateName,
|
|
249
|
+
cwd,
|
|
250
|
+
epicBranch,
|
|
251
|
+
storyBranch,
|
|
252
|
+
config,
|
|
253
|
+
projectMaintainability = defaultProjectMaintainabilityRegressions,
|
|
254
|
+
getBaselines = defaultGetBaselines,
|
|
255
|
+
}) {
|
|
256
|
+
const project = PROJECTORS[gateName];
|
|
257
|
+
if (!project) return [];
|
|
258
|
+
return project({
|
|
259
|
+
cwd,
|
|
260
|
+
epicBranch,
|
|
261
|
+
storyBranch,
|
|
262
|
+
config,
|
|
263
|
+
projectMaintainability,
|
|
264
|
+
getBaselines,
|
|
265
|
+
});
|
|
266
|
+
}
|
package/.agents/scripts/lib/orchestration/story-close/baseline-attribution/phases/scope-discovery.js
ADDED
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* scope-discovery.js — phase 1 of baseline-attribution.
|
|
3
|
+
*
|
|
4
|
+
* Resolve the Story branch's diff scope vs `origin/<epicBranch>` so the
|
|
5
|
+
* downstream projection + attribution layers can split "this Story's
|
|
6
|
+
* regressions" from "sibling drift inherited from the Epic branch".
|
|
7
|
+
*
|
|
8
|
+
* Pure helpers — every git invocation goes through the injected
|
|
9
|
+
* `gitRunner` so tests can pin the diff math without spawning git.
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
import { diffNameOnly } from '../../../../changed-files.js';
|
|
13
|
+
import { gitSpawn as defaultGitSpawn } from '../../../../git-utils.js';
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Compute repo-relative paths the Story branch changed vs `origin/<epicBranch>`.
|
|
17
|
+
* Best-effort: a non-zero diff exit returns an empty array so the caller
|
|
18
|
+
* conservatively treats every regression as non-attributable (the safer
|
|
19
|
+
* default — the close blocks rather than absorbs sibling drift).
|
|
20
|
+
*/
|
|
21
|
+
export function computeStoryDiffPaths({
|
|
22
|
+
cwd,
|
|
23
|
+
epicBranch,
|
|
24
|
+
storyBranch,
|
|
25
|
+
gitRunner = { gitSpawn: defaultGitSpawn },
|
|
26
|
+
}) {
|
|
27
|
+
if (!cwd || !epicBranch || !storyBranch) return [];
|
|
28
|
+
try {
|
|
29
|
+
return diffNameOnly({
|
|
30
|
+
range: `origin/${epicBranch}...${storyBranch}`,
|
|
31
|
+
cwd,
|
|
32
|
+
gitSpawn: gitRunner.gitSpawn,
|
|
33
|
+
});
|
|
34
|
+
} catch {
|
|
35
|
+
return [];
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Guard for the projection phase — the projectors require a worktree + the
|
|
41
|
+
* pair of branch refs to compute a meaningful diff.
|
|
42
|
+
*/
|
|
43
|
+
export function validateProjectionContext({ cwd, epicBranch, storyBranch }) {
|
|
44
|
+
if (!cwd) return false;
|
|
45
|
+
if (!epicBranch) return false;
|
|
46
|
+
if (!storyBranch) return false;
|
|
47
|
+
return true;
|
|
48
|
+
}
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* baseline-attribution-wiring.js — glue between the close-validation gate
|
|
3
|
+
* chain and the diff-based attribution classifier (Story #1124).
|
|
4
|
+
*
|
|
5
|
+
* `runPreMergeGates` throws on the first failed gate but does not surface
|
|
6
|
+
* regression rows. For baseline gates (`check-maintainability`, `check-crap`)
|
|
7
|
+
* the post-#1120 contract is to:
|
|
8
|
+
*
|
|
9
|
+
* 1. Compute the regressions list ourselves (the pre-merge MI projection
|
|
10
|
+
* already knows how — Story #874).
|
|
11
|
+
* 2. Compute the Story's diff vs `epic/<id>` so the classifier can split
|
|
12
|
+
* attributable from non-attributable rows.
|
|
13
|
+
* 3. If every regression is attributable, refresh the kind's baseline via
|
|
14
|
+
* `refreshBaseline()` (Story #2197), stage the changed baseline file,
|
|
15
|
+
* and commit on the Story branch with a `chore(baselines): refresh
|
|
16
|
+
* <kind> for story-<id>` subject. The caller then re-runs the gate
|
|
17
|
+
* chain — drift is now committed, gate passes.
|
|
18
|
+
* 4. If any regression is non-attributable, render the friction body
|
|
19
|
+
* (`renderBaselineFrictionBody`) and upsert it via
|
|
20
|
+
* `upsertStructuredComment`. Return a status that signals story-close
|
|
21
|
+
* to short-circuit with `{ status: 'blocked', phase: 'closing' }`.
|
|
22
|
+
*
|
|
23
|
+
* Story #2205 — refresh path now flows through `refreshBaseline()` from
|
|
24
|
+
* `.agents/scripts/lib/baselines/refresh-service.js`. The `--amend` / `--allow-empty`
|
|
25
|
+
* shortcuts and the legacy `npm run <kind>:update` shell-outs are gone.
|
|
26
|
+
* Post-refresh hygiene is: stage the baseline file, run `git diff --cached
|
|
27
|
+
* --exit-code`, and either skip (empty diff → log "no baseline drift to
|
|
28
|
+
* fold in") or emit one canonical `chore(baselines): refresh <kind> for
|
|
29
|
+
* story-<id>` commit. The retry loop is gated by an idempotency token
|
|
30
|
+
* (`cycleState.refreshedKinds`) so a fail-then-pass sequence still emits
|
|
31
|
+
* at most one baseline-refresh commit per close cycle (AC-9, #2176-fixture).
|
|
32
|
+
*
|
|
33
|
+
* Story #3002 — the module body was decomposed into
|
|
34
|
+
* `./baseline-attribution/phases/` (one file per functional phase)
|
|
35
|
+
* following the established pattern at `./phases/`, `../post-merge/phases/`,
|
|
36
|
+
* and `../retro/phases/`. This file is now a thin re-export sequencer; the
|
|
37
|
+
* phase implementations live under:
|
|
38
|
+
*
|
|
39
|
+
* - phases/scope-discovery.js — Story-diff scope + projection guard.
|
|
40
|
+
* - phases/regression-projection.js — per-gate regression projectors.
|
|
41
|
+
* - phases/refresh-commit.js — in-process refresh + commit.
|
|
42
|
+
* - phases/gate-failure.js — per-failure classifier
|
|
43
|
+
* (handleBaselineGateFailure).
|
|
44
|
+
* - phases/pre-merge-attribution.js — bounded retry driver
|
|
45
|
+
* (runPreMergeGatesWithAttribution).
|
|
46
|
+
*/
|
|
47
|
+
|
|
48
|
+
export {
|
|
49
|
+
DEFAULT_GATE_REGISTRY,
|
|
50
|
+
handleBaselineGateFailure,
|
|
51
|
+
} from './baseline-attribution/phases/gate-failure.js';
|
|
52
|
+
export { runPreMergeGatesWithAttribution } from './baseline-attribution/phases/pre-merge-attribution.js';
|
|
53
|
+
export {
|
|
54
|
+
buildKindScorer,
|
|
55
|
+
runRefreshCommit,
|
|
56
|
+
stageAndCheckBaselineDrift,
|
|
57
|
+
} from './baseline-attribution/phases/refresh-commit.js';
|
|
58
|
+
export {
|
|
59
|
+
diffCrapBaselines,
|
|
60
|
+
PROJECTORS,
|
|
61
|
+
projectCrapRegressions,
|
|
62
|
+
projectRegressionsForGate,
|
|
63
|
+
} from './baseline-attribution/phases/regression-projection.js';
|
|
64
|
+
export {
|
|
65
|
+
computeStoryDiffPaths,
|
|
66
|
+
validateProjectionContext,
|
|
67
|
+
} from './baseline-attribution/phases/scope-discovery.js';
|
|
@@ -0,0 +1,161 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* baseline-attribution.js — diff-based attribution of baseline regressions.
|
|
3
|
+
*
|
|
4
|
+
* Story #1124 (Tech Spec #902, "s-baseline-refresh-discipline") closes a
|
|
5
|
+
* subtle race in the post-#1120 worktree-local close path: when a Story's
|
|
6
|
+
* close run trips a baseline gate (maintainability, crap, etc.) the failing
|
|
7
|
+
* regression rows may include paths the Story never touched — drift carried
|
|
8
|
+
* in from a sibling Story that already merged into `epic/<id>` between this
|
|
9
|
+
* Story's branch cut and its close attempt. Auto-refreshing those baselines
|
|
10
|
+
* here would silently absorb the sibling's debt onto this Story's PR and
|
|
11
|
+
* destroy the audit trail. Doing nothing locks the close behind drift the
|
|
12
|
+
* operator did not cause.
|
|
13
|
+
*
|
|
14
|
+
* The split this module computes is:
|
|
15
|
+
*
|
|
16
|
+
* - `attributable` — regressions on paths intersecting the Story's diff
|
|
17
|
+
* vs `epic/<id>`. Safe to auto-refresh on the Story
|
|
18
|
+
* branch as `baseline-refresh: ...`.
|
|
19
|
+
* - `nonAttributable` — regressions on paths the Story never touched. Each
|
|
20
|
+
* row is annotated with the most recent commit on
|
|
21
|
+
* `epic/<id>` that touched the path (`suspectSha`)
|
|
22
|
+
* and the `(resolves #N)` token parsed out of that
|
|
23
|
+
* commit subject (`suspectStoryNumber`, may be null).
|
|
24
|
+
* The caller surfaces these as friction so the
|
|
25
|
+
* operator can route the refresh back to the sibling
|
|
26
|
+
* that caused the drift.
|
|
27
|
+
*
|
|
28
|
+
* The classifier itself does no IO when every regression is attributable —
|
|
29
|
+
* the `git log` lookup runs only on non-attributable rows so the hot path
|
|
30
|
+
* (the common case where the operator's Story is the sole writer) costs
|
|
31
|
+
* one `Set` membership check per row.
|
|
32
|
+
*/
|
|
33
|
+
|
|
34
|
+
import { gitSpawn as defaultGitSpawn } from '../../git-utils.js';
|
|
35
|
+
import { parseResolvesStoryId } from '../resolves-token.js';
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Normalize a path for set-based intersection. `git diff --name-only`
|
|
39
|
+
* returns POSIX paths on every OS, but a caller passing
|
|
40
|
+
* `storyDiffPaths` from a Windows tooling boundary may have `\` in the
|
|
41
|
+
* path. Round both sides through `/` so the intersection is honest.
|
|
42
|
+
*/
|
|
43
|
+
function normalize(p) {
|
|
44
|
+
return typeof p === 'string' ? p.replace(/\\/g, '/') : p;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* Look up the most recent commit on `epicRef` that touched `path` and
|
|
49
|
+
* extract the `(resolves #N)` trailer if present. Returns
|
|
50
|
+
* `{ suspectSha, suspectStoryNumber }`; either field may be `null` when
|
|
51
|
+
* the lookup yields no commit (e.g. file added on this branch but
|
|
52
|
+
* still flagged as a regression by the gate against a stale baseline)
|
|
53
|
+
* or the most recent commit subject does not carry the trailer.
|
|
54
|
+
*
|
|
55
|
+
* Failure of the spawn is treated as "no information" — the caller
|
|
56
|
+
* still surfaces the row as non-attributable, just without a suspect.
|
|
57
|
+
*/
|
|
58
|
+
function lookupSuspect({ gitRunner, cwd, epicRef, path }) {
|
|
59
|
+
const res = gitRunner.gitSpawn(
|
|
60
|
+
cwd,
|
|
61
|
+
'log',
|
|
62
|
+
'--oneline',
|
|
63
|
+
'-n',
|
|
64
|
+
'1',
|
|
65
|
+
epicRef,
|
|
66
|
+
'--',
|
|
67
|
+
path,
|
|
68
|
+
);
|
|
69
|
+
if (res.status !== 0) {
|
|
70
|
+
return { suspectSha: null, suspectStoryNumber: null };
|
|
71
|
+
}
|
|
72
|
+
const line = (res.stdout || '').split('\n')[0]?.trim() ?? '';
|
|
73
|
+
if (!line) return { suspectSha: null, suspectStoryNumber: null };
|
|
74
|
+
|
|
75
|
+
// `--oneline` format is `<short-sha> <subject…>`; split on the first
|
|
76
|
+
// run of whitespace so multi-space subjects survive.
|
|
77
|
+
const sepIdx = line.search(/\s/);
|
|
78
|
+
const suspectSha = sepIdx === -1 ? line : line.slice(0, sepIdx);
|
|
79
|
+
const subject = sepIdx === -1 ? '' : line.slice(sepIdx + 1);
|
|
80
|
+
const suspectStoryNumber = parseResolvesStoryId(subject);
|
|
81
|
+
return { suspectSha: suspectSha || null, suspectStoryNumber };
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
/**
|
|
85
|
+
* Classify the supplied baseline regressions into rows the running Story
|
|
86
|
+
* should auto-refresh vs rows it should surface as friction.
|
|
87
|
+
*
|
|
88
|
+
* @param {object} opts
|
|
89
|
+
* @param {Array<{ path?: string, file?: string }>} opts.regressions
|
|
90
|
+
* Regression rows from a baseline gate. Either `path` or `file` is read
|
|
91
|
+
* for the file key — different gate emitters use different names and
|
|
92
|
+
* normalizing them here keeps the wiring at the call site trivial.
|
|
93
|
+
* @param {Iterable<string>} opts.storyDiffPaths
|
|
94
|
+
* Repo-relative paths the Story branch changed vs `epic/<id>` (typically
|
|
95
|
+
* the output of `git diff --name-only origin/<epicBranch>...storyBranch`).
|
|
96
|
+
* @param {string} opts.epicRef Git ref the suspect lookup walks (e.g.
|
|
97
|
+
* `origin/epic/1114` or `epic/1114`).
|
|
98
|
+
* @param {string} [opts.cwd] CWD for `git log`. Defaults to `process.cwd()`.
|
|
99
|
+
* @param {{ gitSpawn: typeof defaultGitSpawn }} [opts.gitRunner]
|
|
100
|
+
* Injected git seam. Tests pass a fake recorder.
|
|
101
|
+
* @returns {{
|
|
102
|
+
* attributable: Array<{ path: string }>,
|
|
103
|
+
* nonAttributable: Array<{
|
|
104
|
+
* path: string,
|
|
105
|
+
* suspectSha: string|null,
|
|
106
|
+
* suspectStoryNumber: number|null,
|
|
107
|
+
* }>,
|
|
108
|
+
* }}
|
|
109
|
+
*/
|
|
110
|
+
export function classifyBaselineDrift({
|
|
111
|
+
regressions,
|
|
112
|
+
storyDiffPaths,
|
|
113
|
+
epicRef,
|
|
114
|
+
cwd,
|
|
115
|
+
gitRunner,
|
|
116
|
+
} = {}) {
|
|
117
|
+
const rows = Array.isArray(regressions) ? regressions : [];
|
|
118
|
+
const touched = new Set(
|
|
119
|
+
Array.from(storyDiffPaths ?? [], normalize).filter(Boolean),
|
|
120
|
+
);
|
|
121
|
+
|
|
122
|
+
const attributable = [];
|
|
123
|
+
const nonAttributable = [];
|
|
124
|
+
|
|
125
|
+
// Defer git seam construction until at least one non-attributable row
|
|
126
|
+
// appears — the all-attributable hot path costs zero spawns.
|
|
127
|
+
let runner = null;
|
|
128
|
+
const ensureRunner = () => {
|
|
129
|
+
if (runner) return runner;
|
|
130
|
+
runner = gitRunner ?? { gitSpawn: defaultGitSpawn };
|
|
131
|
+
return runner;
|
|
132
|
+
};
|
|
133
|
+
const lookupCwd =
|
|
134
|
+
cwd ?? (typeof process !== 'undefined' ? process.cwd() : '.');
|
|
135
|
+
|
|
136
|
+
for (const row of rows) {
|
|
137
|
+
const rawPath = row?.path ?? row?.file;
|
|
138
|
+
if (typeof rawPath !== 'string' || rawPath.length === 0) continue;
|
|
139
|
+
const p = normalize(rawPath);
|
|
140
|
+
if (touched.has(p)) {
|
|
141
|
+
attributable.push({ ...row, path: p });
|
|
142
|
+
continue;
|
|
143
|
+
}
|
|
144
|
+
const suspect = epicRef
|
|
145
|
+
? lookupSuspect({
|
|
146
|
+
gitRunner: ensureRunner(),
|
|
147
|
+
cwd: lookupCwd,
|
|
148
|
+
epicRef,
|
|
149
|
+
path: p,
|
|
150
|
+
})
|
|
151
|
+
: { suspectSha: null, suspectStoryNumber: null };
|
|
152
|
+
nonAttributable.push({
|
|
153
|
+
...row,
|
|
154
|
+
path: p,
|
|
155
|
+
suspectSha: suspect.suspectSha,
|
|
156
|
+
suspectStoryNumber: suspect.suspectStoryNumber,
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
return { attributable, nonAttributable };
|
|
161
|
+
}
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* baseline-friction-body.js — render the friction-comment body for
|
|
3
|
+
* non-attributable baseline drift surfaced by the close-validation chain.
|
|
4
|
+
*
|
|
5
|
+
* Story #1124 (Tech Spec #902, "s-baseline-refresh-discipline"). When
|
|
6
|
+
* `classifyBaselineDrift` reports rows on paths the running Story never
|
|
7
|
+
* touched, story-close upserts a `friction`-typed structured comment on
|
|
8
|
+
* the Story so the operator can route the refresh back to the sibling
|
|
9
|
+
* Story whose merge introduced the drift. The body shape is stable so
|
|
10
|
+
* the `diagnose-friction` analyzer can pattern-match the section.
|
|
11
|
+
*
|
|
12
|
+
* Output contract (markdown):
|
|
13
|
+
*
|
|
14
|
+
* ### Baseline drift not attributable to this Story
|
|
15
|
+
*
|
|
16
|
+
* The close-validation chain flagged the following file(s) as breaching
|
|
17
|
+
* a committed baseline, but none of the paths intersect this Story's
|
|
18
|
+
* diff vs `epic/<epicId>`. Refreshing them here would silently absorb a
|
|
19
|
+
* sibling Story's debt onto this Story's PR, so the close is blocked
|
|
20
|
+
* until an operator routes the refresh to the suspect Story below.
|
|
21
|
+
*
|
|
22
|
+
* | Path | Suspect Story | Suspect Commit |
|
|
23
|
+
* | --- | --- | --- |
|
|
24
|
+
* | lib/foo.js | #777 | `deadbee1` |
|
|
25
|
+
* | lib/orphan.js | _unknown_ | `beef0001` |
|
|
26
|
+
*
|
|
27
|
+
* <triage instructions>
|
|
28
|
+
*
|
|
29
|
+
* The renderer is pure — no IO, no provider calls — so unit tests can pin
|
|
30
|
+
* the exact body string against fixed inputs.
|
|
31
|
+
*/
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* Format a single suspect cell: `#<num>` when present, italic `_unknown_`
|
|
35
|
+
* otherwise. Italic so a Markdown-aware reader can scan the column for
|
|
36
|
+
* holes without parsing the table semantics.
|
|
37
|
+
*/
|
|
38
|
+
function formatSuspectStory(num) {
|
|
39
|
+
return typeof num === 'number' && Number.isFinite(num) && num > 0
|
|
40
|
+
? `#${num}`
|
|
41
|
+
: '_unknown_';
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* Format the suspect-commit cell as an inline-code SHA, or em-dash when
|
|
46
|
+
* the lookup yielded nothing (the row was non-attributable but git log
|
|
47
|
+
* returned no commit on `epicRef` for the path — typically a renamed
|
|
48
|
+
* file or a stale baseline against an empty path).
|
|
49
|
+
*/
|
|
50
|
+
function formatSuspectSha(sha) {
|
|
51
|
+
return typeof sha === 'string' && sha.length > 0 ? `\`${sha}\`` : '—';
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* Render the friction-comment body. The caller passes the result to
|
|
56
|
+
* `upsertStructuredComment(provider, ticketId, 'friction', body)` which
|
|
57
|
+
* applies the canonical structured-comment marker.
|
|
58
|
+
*
|
|
59
|
+
* @param {object} input
|
|
60
|
+
* @param {Array<{
|
|
61
|
+
* path: string,
|
|
62
|
+
* suspectStoryNumber: number|null,
|
|
63
|
+
* suspectSha: string|null,
|
|
64
|
+
* }>} input.rows
|
|
65
|
+
* The `nonAttributable` slice from `classifyBaselineDrift`. Empty input
|
|
66
|
+
* yields a defensive "no rows" body so a misuse never produces a
|
|
67
|
+
* blank comment — that comment shape is asserted by the unit test.
|
|
68
|
+
* @param {number|string} input.epicId
|
|
69
|
+
* @param {number|string} input.storyId
|
|
70
|
+
* @returns {string} markdown body (no trailing newline)
|
|
71
|
+
*/
|
|
72
|
+
export function renderBaselineFrictionBody({ rows, epicId, storyId } = {}) {
|
|
73
|
+
const tableRows = Array.isArray(rows) ? rows : [];
|
|
74
|
+
const heading = `### Baseline drift not attributable to Story #${storyId}`;
|
|
75
|
+
const intro =
|
|
76
|
+
`The close-validation chain flagged the file(s) below as breaching a ` +
|
|
77
|
+
`committed baseline, but none of the paths intersect this Story's diff vs ` +
|
|
78
|
+
`\`epic/${epicId}\`. Auto-refreshing them on Story #${storyId} would silently ` +
|
|
79
|
+
`absorb a sibling Story's debt onto this Story's PR, so the close is ` +
|
|
80
|
+
`blocked until an operator routes the refresh to the suspect Story below.`;
|
|
81
|
+
|
|
82
|
+
if (tableRows.length === 0) {
|
|
83
|
+
return [
|
|
84
|
+
heading,
|
|
85
|
+
'',
|
|
86
|
+
intro,
|
|
87
|
+
'',
|
|
88
|
+
'_No non-attributable rows were supplied — this comment was emitted defensively._',
|
|
89
|
+
].join('\n');
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
const tableHeader = [
|
|
93
|
+
'| Path | Suspect Story | Suspect Commit |',
|
|
94
|
+
'| --- | --- | --- |',
|
|
95
|
+
];
|
|
96
|
+
const tableBody = tableRows.map(
|
|
97
|
+
(r) =>
|
|
98
|
+
`| \`${r.path}\` | ${formatSuspectStory(r.suspectStoryNumber)} | ${formatSuspectSha(r.suspectSha)} |`,
|
|
99
|
+
);
|
|
100
|
+
const triage = [
|
|
101
|
+
'**Triage:**',
|
|
102
|
+
`1. Open each suspect Story above and run \`npm run maintainability:update\` (or \`npm run crap:update\`) on **its** branch, then commit with a \`baseline-refresh:\` subject and re-close it.`,
|
|
103
|
+
`2. Re-run \`/story-deliver ${storyId}\` once the suspect Story's refresh has merged into \`epic/${epicId}\`.`,
|
|
104
|
+
`3. If the suspect column reads \`_unknown_\`, the path has no commit on \`epic/${epicId}\` — investigate the baseline file directly before refreshing.`,
|
|
105
|
+
];
|
|
106
|
+
|
|
107
|
+
return [
|
|
108
|
+
heading,
|
|
109
|
+
'',
|
|
110
|
+
intro,
|
|
111
|
+
'',
|
|
112
|
+
...tableHeader,
|
|
113
|
+
...tableBody,
|
|
114
|
+
'',
|
|
115
|
+
...triage,
|
|
116
|
+
].join('\n');
|
|
117
|
+
}
|