@event4u/agent-config 6.0.0 → 7.0.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/.claude-plugin/marketplace.json +39 -7
- package/AGENTS.md +8 -7
- package/CHANGELOG.md +557 -422
- package/CONTRIBUTING.md +1 -1
- package/README.md +18 -16
- package/dist/agent-src/commands/agent-handoff.md +5 -4
- package/dist/agent-src/commands/agent-status.md +3 -2
- package/dist/agent-src/commands/agents/audit.md +4 -3
- package/dist/agent-src/commands/agents/init.md +4 -1
- package/dist/agent-src/commands/agents/optimize.md +5 -4
- package/dist/agent-src/commands/agents/user/accept.md +1 -0
- package/dist/agent-src/commands/agents/user/init.md +1 -0
- package/dist/agent-src/commands/agents/user/review.md +1 -0
- package/dist/agent-src/commands/agents/user/show.md +1 -0
- package/dist/agent-src/commands/agents/user/update.md +1 -0
- package/dist/agent-src/commands/agents/user.md +1 -0
- package/dist/agent-src/commands/agents.md +1 -0
- package/dist/agent-src/commands/analytics/prune.md +3 -2
- package/dist/agent-src/commands/analytics/show.md +3 -2
- package/dist/agent-src/commands/analytics.md +3 -2
- package/dist/agent-src/commands/analyze/decision.md +108 -0
- package/dist/agent-src/commands/analyze/incident.md +120 -0
- package/dist/agent-src/commands/analyze/near-miss.md +113 -0
- package/dist/agent-src/commands/analyze/postmortem.md +130 -0
- package/dist/agent-src/commands/analyze/premortem.md +104 -0
- package/dist/agent-src/commands/analyze-reference-repo.md +1 -0
- package/dist/agent-src/commands/analyze.md +124 -0
- package/dist/agent-src/commands/brand/identity.md +27 -0
- package/dist/agent-src/commands/brand/review.md +27 -0
- package/dist/agent-src/commands/brand/strategy.md +27 -0
- package/dist/agent-src/commands/brand/tokens.md +28 -0
- package/dist/agent-src/commands/brand/voice.md +27 -0
- package/dist/agent-src/commands/brand.md +58 -0
- package/dist/agent-src/commands/bug-fix.md +1 -0
- package/dist/agent-src/commands/bug-investigate.md +1 -0
- package/dist/agent-src/commands/challenge-me/vision.md +3 -2
- package/dist/agent-src/commands/challenge-me/with-docs.md +3 -2
- package/dist/agent-src/commands/challenge-me.md +3 -2
- package/dist/agent-src/commands/chat-history/import.md +9 -9
- package/dist/agent-src/commands/chat-history.md +32 -30
- package/dist/agent-src/commands/check-current-md.md +4 -3
- package/dist/agent-src/commands/commit/in-chunks.md +1 -0
- package/dist/agent-src/commands/commit.md +1 -0
- package/dist/agent-src/commands/condense.md +3 -2
- package/dist/agent-src/commands/context/create.md +1 -0
- package/dist/agent-src/commands/context/refactor.md +1 -0
- package/dist/agent-src/commands/context.md +1 -0
- package/dist/agent-src/commands/cost-report.md +5 -4
- package/dist/agent-src/commands/council/analysis.md +3 -2
- package/dist/agent-src/commands/council/debate.md +7 -6
- package/dist/agent-src/commands/council/default.md +48 -20
- package/dist/agent-src/commands/council/design.md +3 -2
- package/dist/agent-src/commands/council/optimize.md +3 -2
- package/dist/agent-src/commands/council/pr.md +3 -2
- package/dist/agent-src/commands/council.md +4 -3
- package/dist/agent-src/commands/e2e-heal.md +1 -0
- package/dist/agent-src/commands/e2e-plan.md +1 -0
- package/dist/agent-src/commands/estimate-ticket.md +1 -0
- package/dist/agent-src/commands/feature/dev.md +1 -0
- package/dist/agent-src/commands/feature/explore.md +1 -0
- package/dist/agent-src/commands/feature/plan.md +6 -6
- package/dist/agent-src/commands/feature/refactor.md +1 -0
- package/dist/agent-src/commands/feature/roadmap.md +1 -0
- package/dist/agent-src/commands/feature.md +1 -0
- package/dist/agent-src/commands/fix/ci.md +1 -0
- package/dist/agent-src/commands/fix/portability.md +4 -3
- package/dist/agent-src/commands/fix/pr-comments.md +147 -15
- package/dist/agent-src/commands/fix/refs.md +4 -3
- package/dist/agent-src/commands/fix/seeder.md +1 -0
- package/dist/agent-src/commands/fix.md +8 -8
- package/dist/agent-src/commands/ghostwriter/delete.md +1 -0
- package/dist/agent-src/commands/ghostwriter/fetch.md +1 -0
- package/dist/agent-src/commands/ghostwriter/list.md +1 -0
- package/dist/agent-src/commands/ghostwriter/show.md +1 -0
- package/dist/agent-src/commands/ghostwriter/write.md +1 -0
- package/dist/agent-src/commands/ghostwriter.md +1 -0
- package/dist/agent-src/commands/grill-me.md +3 -2
- package/dist/agent-src/commands/image/analyse.md +1 -0
- package/dist/agent-src/commands/image/create.md +1 -0
- package/dist/agent-src/commands/image/verify.md +1 -0
- package/dist/agent-src/commands/image.md +1 -0
- package/dist/agent-src/commands/implement-ticket.md +37 -6
- package/dist/agent-src/commands/jira-ticket.md +1 -0
- package/dist/agent-src/commands/judge/on-diff.md +1 -0
- package/dist/agent-src/commands/judge/solo.md +1 -0
- package/dist/agent-src/commands/judge/steps.md +1 -0
- package/dist/agent-src/commands/judge.md +1 -0
- package/dist/agent-src/commands/knowledge/cross-repo.md +2 -1
- package/dist/agent-src/commands/knowledge/forget.md +1 -0
- package/dist/agent-src/commands/knowledge/ingest.md +1 -0
- package/dist/agent-src/commands/knowledge/list.md +1 -0
- package/dist/agent-src/commands/knowledge.md +1 -0
- package/dist/agent-src/commands/memory/add.md +9 -7
- package/dist/agent-src/commands/memory/learn-low-impact.md +3 -2
- package/dist/agent-src/commands/memory/load.md +7 -7
- package/dist/agent-src/commands/memory/mine-session.md +39 -12
- package/dist/agent-src/commands/memory/promote.md +3 -2
- package/dist/agent-src/commands/memory/propose.md +7 -6
- package/dist/agent-src/commands/memory.md +3 -2
- package/dist/agent-src/commands/mission/upgrade.md +182 -0
- package/dist/agent-src/commands/mode.md +1 -0
- package/dist/agent-src/commands/module/create.md +1 -0
- package/dist/agent-src/commands/module/explore.md +1 -0
- package/dist/agent-src/commands/module.md +1 -0
- package/dist/agent-src/commands/optimize/agents-dir.md +1 -0
- package/dist/agent-src/commands/optimize/augmentignore.md +1 -0
- package/dist/agent-src/commands/optimize/rtk.md +1 -0
- package/dist/agent-src/commands/optimize/skills.md +3 -2
- package/dist/agent-src/commands/optimize-prompt.md +1 -0
- package/dist/agent-src/commands/optimize.md +1 -0
- package/dist/agent-src/commands/orchestrate.md +2 -1
- package/dist/agent-src/commands/override/create.md +1 -0
- package/dist/agent-src/commands/override/manage.md +1 -0
- package/dist/agent-src/commands/override.md +1 -0
- package/dist/agent-src/commands/package-reset.md +1 -0
- package/dist/agent-src/commands/package-test.md +1 -0
- package/dist/agent-src/commands/post-as/ghostwriter.md +1 -0
- package/dist/agent-src/commands/post-as/me.md +1 -0
- package/dist/agent-src/commands/post-as.md +1 -0
- package/dist/agent-src/commands/pr/create/description-only.md +1 -0
- package/dist/agent-src/commands/pr/create.md +31 -4
- package/dist/agent-src/commands/prediction-pool.md +1 -0
- package/dist/agent-src/commands/prepare-for-review.md +1 -0
- package/dist/agent-src/commands/profile/activate.md +1 -0
- package/dist/agent-src/commands/profile/deactivate.md +1 -0
- package/dist/agent-src/commands/profile/show.md +1 -0
- package/dist/agent-src/commands/profile.md +1 -0
- package/dist/agent-src/commands/project-analyze.md +1 -0
- package/dist/agent-src/commands/project-health.md +1 -0
- package/dist/agent-src/commands/quality-fix.md +1 -0
- package/dist/agent-src/commands/refine-ticket.md +1 -0
- package/dist/agent-src/commands/research/deep.md +1 -0
- package/dist/agent-src/commands/research/report.md +1 -0
- package/dist/agent-src/commands/research.md +1 -0
- package/dist/agent-src/commands/review-changes.md +9 -0
- package/dist/agent-src/commands/review-routing.md +1 -0
- package/dist/agent-src/commands/roadmap/ai-council.md +1 -0
- package/dist/agent-src/commands/roadmap/create.md +1 -0
- package/dist/agent-src/commands/roadmap/materialize.md +73 -0
- package/dist/agent-src/commands/roadmap/process-full.md +1 -0
- package/dist/agent-src/commands/roadmap/process-phase.md +1 -0
- package/dist/agent-src/commands/roadmap/process-step.md +1 -0
- package/dist/agent-src/commands/roadmap.md +1 -0
- package/dist/agent-src/commands/rule-compliance-audit.md +1 -0
- package/dist/agent-src/commands/security-audit-config.md +84 -0
- package/dist/agent-src/commands/set-cost-profile.md +1 -0
- package/dist/agent-src/commands/skill/preview.md +2 -1
- package/dist/agent-src/commands/skill.md +1 -0
- package/dist/agent-src/commands/skills/discover.md +2 -1
- package/dist/agent-src/commands/skills.md +1 -0
- package/dist/agent-src/commands/sync-agent-settings.md +1 -0
- package/dist/agent-src/commands/sync-gitignore/fix.md +1 -0
- package/dist/agent-src/commands/sync-gitignore.md +1 -0
- package/dist/agent-src/commands/tests/create.md +1 -0
- package/dist/agent-src/commands/tests/execute.md +1 -0
- package/dist/agent-src/commands/tests.md +1 -0
- package/dist/agent-src/commands/threat-model.md +5 -4
- package/dist/agent-src/commands/update-form-request-messages.md +1 -0
- package/dist/agent-src/commands/upstream-contribute.md +4 -3
- package/dist/agent-src/commands/video/from-script.md +3 -2
- package/dist/agent-src/commands/video/from-song.md +4 -3
- package/dist/agent-src/commands/video/scene.md +2 -1
- package/dist/agent-src/commands/video/stitch.md +1 -0
- package/dist/agent-src/commands/video/storyboard.md +2 -1
- package/dist/agent-src/commands/video.md +4 -3
- package/dist/agent-src/commands/work.md +1 -0
- package/dist/agent-src/contexts/augment-infrastructure.md +1 -1
- package/dist/agent-src/contexts/communication/rules-auto/skill-quality-mechanics.md +1 -1
- package/dist/agent-src/contexts/communication/rules-auto/slash-command-routing-policy-mechanics.md +2 -2
- package/dist/agent-src/contexts/communication/rules-auto/source-of-truth-mechanics.md +3 -3
- package/dist/agent-src/contexts/communication/rules-auto/think-before-action-mechanics.md +6 -6
- package/dist/agent-src/contexts/communication/rules-auto/user-interaction-mechanics.md +1 -1
- package/dist/agent-src/contexts/contracts/consumer-agents-md-guide.md +2 -2
- package/dist/agent-src/contexts/execution/evidence-discipline.md +153 -0
- package/dist/agent-src/contexts/execution/project-intelligence.md +264 -0
- package/dist/agent-src/contexts/execution/rdp-gate.md +75 -0
- package/dist/agent-src/contexts/execution/roadmap-process-loop.md +2 -1
- package/dist/agent-src/contexts/subagent-configuration.md +1 -0
- package/dist/agent-src/personas/advisors/contrarian.md +1 -1
- package/dist/agent-src/personas/advisors/executor.md +1 -1
- package/dist/agent-src/personas/advisors/expansionist.md +1 -1
- package/dist/agent-src/personas/advisors/first-principles.md +1 -1
- package/dist/agent-src/personas/advisors/outsider.md +1 -1
- package/dist/agent-src/personas/ai-video-technical-director.md +1 -1
- package/dist/agent-src/personas/brand-strategist.md +74 -0
- package/dist/agent-src/personas/design-director.md +74 -0
- package/dist/agent-src/rules/autonomous-execution.md +12 -0
- package/dist/agent-src/rules/brand-consistency.md +77 -0
- package/dist/agent-src/rules/brand-source-of-truth.md +57 -0
- package/dist/agent-src/rules/direct-answers.md +2 -0
- package/dist/agent-src/rules/domain-safety-disclaimer.md +2 -0
- package/dist/agent-src/rules/external-reference-deep-dive.md +1 -1
- package/dist/agent-src/rules/git-history-discipline.md +48 -1
- package/dist/agent-src/rules/icon-consistency.md +53 -0
- package/dist/agent-src/rules/image-likeness-and-rights.md +67 -0
- package/dist/agent-src/rules/improve-before-implement.md +12 -0
- package/dist/agent-src/rules/lethal-trifecta-guard.md +80 -0
- package/dist/agent-src/rules/no-pr-progress-comments.md +3 -4
- package/dist/agent-src/rules/notes-first-reasoning.md +71 -0
- package/dist/agent-src/rules/persona-governance.md +2 -2
- package/dist/agent-src/rules/provider-lifecycle-discipline.md +3 -1
- package/dist/agent-src/rules/roadmap-progress-sync.md +58 -31
- package/dist/agent-src/rules/security-sensitive-stop.md +22 -3
- package/dist/agent-src/rules/size-enforcement.md +1 -1
- package/dist/agent-src/rules/source-confidentiality.md +97 -0
- package/dist/agent-src/rules/source-discovery-gate.md +98 -0
- package/dist/agent-src/rules/think-before-action.md +10 -1
- package/dist/agent-src/rules/ui-audit-gate.md +2 -0
- package/dist/agent-src/rules/untrusted-input-defense.md +76 -0
- package/dist/agent-src/rules/user-interaction.md +1 -1
- package/dist/agent-src/scripts/archive_completed_roadmaps.ts +392 -0
- package/dist/agent-src/scripts/update_roadmap_progress.ts +824 -0
- package/dist/agent-src/skills/adr-create/SKILL.md +5 -5
- package/dist/agent-src/skills/adversarial-review/SKILL.md +14 -0
- package/dist/agent-src/skills/agent-security-review/SKILL.md +113 -0
- package/dist/agent-src/skills/agent-security-review/evals/triggers.json +52 -0
- package/dist/agent-src/skills/agents-md-thin-root/SKILL.md +1 -1
- package/dist/agent-src/skills/ai-council/SKILL.md +4 -4
- package/dist/agent-src/skills/analysis-autonomous-mode/SKILL.md +9 -13
- package/dist/agent-src/skills/async-python-patterns/SKILL.md +1 -1
- package/dist/agent-src/skills/blade-ui/SKILL.md +12 -5
- package/dist/agent-src/skills/blameless-post-mortem/SKILL.md +199 -0
- package/dist/agent-src/skills/blast-radius-analyzer/SKILL.md +12 -11
- package/dist/agent-src/skills/brand/ATTRIBUTION.md +38 -0
- package/dist/agent-src/skills/brand/SKILL.md +115 -0
- package/dist/agent-src/skills/brand/data/archetypes.csv +13 -0
- package/dist/agent-src/skills/brand/data/color-psychology.csv +14 -0
- package/dist/agent-src/skills/brand/data/logo-style-fit.csv +13 -0
- package/dist/agent-src/skills/brand/data/manifest.json +226 -0
- package/dist/agent-src/skills/brand/data/messaging-frameworks.csv +13 -0
- package/dist/agent-src/skills/brand/data/naming-patterns.csv +13 -0
- package/dist/agent-src/skills/brand/data/typography-principles.csv +13 -0
- package/dist/agent-src/skills/brand/data/voice-tone.csv +13 -0
- package/dist/agent-src/skills/brand/evals/triggers.json +17 -0
- package/dist/agent-src/skills/brand-asset-generation/SKILL.md +89 -0
- package/dist/agent-src/skills/brand-asset-generation/evals/triggers.json +17 -0
- package/dist/agent-src/skills/brand-audit/SKILL.md +67 -0
- package/dist/agent-src/skills/brand-audit/evals/triggers.json +17 -0
- package/dist/agent-src/skills/brand-identity/SKILL.md +101 -0
- package/dist/agent-src/skills/brand-identity/evals/triggers.json +17 -0
- package/dist/agent-src/skills/brand-strategy/SKILL.md +83 -0
- package/dist/agent-src/skills/brand-strategy/evals/triggers.json +17 -0
- package/dist/agent-src/skills/brand-to-tokens/SKILL.md +102 -0
- package/dist/agent-src/skills/brand-to-tokens/evals/triggers.json +17 -0
- package/dist/agent-src/skills/brand-to-tokens/templates/marp-brand-deck.md.example +46 -0
- package/dist/agent-src/skills/brand-to-tokens/templates/reveal-brand-deck.yaml +32 -0
- package/dist/agent-src/skills/canvas-design/evals/triggers.json +1 -0
- package/dist/agent-src/skills/check-refs/SKILL.md +5 -5
- package/dist/agent-src/skills/code-review/SKILL.md +6 -15
- package/dist/agent-src/skills/command-routing/SKILL.md +1 -1
- package/dist/agent-src/skills/command-writing/SKILL.md +2 -2
- package/dist/agent-src/skills/complexity-first-planning/SKILL.md +96 -0
- package/dist/agent-src/skills/complexity-first-planning/evals/triggers.json +17 -0
- package/dist/agent-src/skills/context-authoring/SKILL.md +2 -2
- package/dist/agent-src/skills/context-document/SKILL.md +35 -2
- package/dist/agent-src/skills/copilot-config/SKILL.md +3 -4
- package/dist/agent-src/skills/corpus-grounding/evals/triggers.json +1 -0
- package/dist/agent-src/skills/corpus-grounding/scripts/bm25_search.ts +482 -0
- package/dist/agent-src/skills/corpus-grounding/scripts/decision_engine.ts +803 -0
- package/dist/agent-src/skills/corpus-grounding/scripts/ground.ts +541 -0
- package/dist/agent-src/skills/corpus-grounding/scripts/schema_validator.ts +309 -0
- package/dist/agent-src/skills/database/SKILL.md +26 -4
- package/dist/agent-src/skills/decision-record/SKILL.md +1 -1
- package/dist/agent-src/skills/decision-record/evals/triggers.json +17 -0
- package/dist/agent-src/skills/decision-review/SKILL.md +179 -0
- package/dist/agent-src/skills/defense-in-depth/SKILL.md +1 -1
- package/dist/agent-src/skills/description-assist/SKILL.md +1 -1
- package/dist/agent-src/skills/design-intelligence/SKILL.md +1 -1
- package/dist/agent-src/skills/design-intelligence/data/manifest.json +23 -6
- package/dist/agent-src/skills/design-intelligence/evals/triggers.json +1 -0
- package/dist/agent-src/skills/design-tokens/evals/triggers.json +1 -0
- package/dist/agent-src/skills/design-tokens/scripts/tokens.ts +888 -0
- package/dist/agent-src/skills/developer-like-execution/SKILL.md +5 -4
- package/dist/agent-src/skills/doc-coauthoring/evals/triggers.json +1 -0
- package/dist/agent-src/skills/eloquent/evals/triggers.json +1 -0
- package/dist/agent-src/skills/emit-tickets/SKILL.md +198 -0
- package/dist/agent-src/skills/error-handling-patterns/SKILL.md +1 -1
- package/dist/agent-src/skills/estimate-ticket/evals/triggers.json +1 -0
- package/dist/agent-src/skills/feature-planning/SKILL.md +2 -2
- package/dist/agent-src/skills/git-workflow/SKILL.md +33 -0
- package/dist/agent-src/skills/guideline-writing/SKILL.md +2 -2
- package/dist/agent-src/skills/iconography/SKILL.md +88 -0
- package/dist/agent-src/skills/iconography/evals/triggers.json +17 -0
- package/dist/agent-src/skills/image-analyser/evals/triggers.json +1 -0
- package/dist/agent-src/skills/image-creator/evals/triggers.json +1 -0
- package/dist/agent-src/skills/image-editing/SKILL.md +100 -0
- package/dist/agent-src/skills/image-editing/evals/triggers.json +17 -0
- package/dist/agent-src/skills/image-generation/SKILL.md +95 -0
- package/dist/agent-src/skills/image-generation/evals/triggers.json +17 -0
- package/dist/agent-src/skills/image-provider-routing/SKILL.md +96 -0
- package/dist/agent-src/skills/image-provider-routing/evals/triggers.json +17 -0
- package/dist/agent-src/skills/launch-readiness/SKILL.md +21 -0
- package/dist/agent-src/skills/learning-to-rule-or-skill/SKILL.md +12 -8
- package/dist/agent-src/skills/lint-skills/SKILL.md +5 -5
- package/dist/agent-src/skills/logo-generation/SKILL.md +98 -0
- package/dist/agent-src/skills/logo-generation/evals/triggers.json +17 -0
- package/dist/agent-src/skills/markitdown/SKILL.md +1 -1
- package/dist/agent-src/skills/mcp-builder/SKILL.md +1 -1
- package/dist/agent-src/skills/md-language-check/SKILL.md +1 -1
- package/dist/agent-src/skills/memory-consolidation/SKILL.md +63 -17
- package/dist/agent-src/skills/motion-choreographer/SKILL.md +1 -1
- package/dist/agent-src/skills/php-coder/evals/triggers.json +1 -0
- package/dist/agent-src/skills/prediction-pool-optimizer/evals/triggers.json +1 -0
- package/dist/agent-src/skills/premortem/SKILL.md +137 -0
- package/dist/agent-src/skills/prompt-engineering-image/SKILL.md +115 -0
- package/dist/agent-src/skills/prompt-engineering-image/evals/triggers.json +17 -0
- package/dist/agent-src/skills/prompt-engineering-patterns/SKILL.md +1 -1
- package/dist/agent-src/skills/prompt-validator/evals/triggers.json +1 -0
- package/dist/agent-src/skills/react-shadcn-ui/SKILL.md +12 -5
- package/dist/agent-src/skills/react-shadcn-ui/scripts/shadcn_add.ts +388 -0
- package/dist/agent-src/skills/readme-writing-package/SKILL.md +1 -1
- package/dist/agent-src/skills/reasoning-orchestrator/SKILL.md +119 -0
- package/dist/agent-src/skills/reasoning-orchestrator/evals/triggers.json +17 -0
- package/dist/agent-src/skills/receiving-code-review/SKILL.md +6 -6
- package/dist/agent-src/skills/refine-prompt/SKILL.md +1 -1
- package/dist/agent-src/skills/refine-ticket/SKILL.md +1 -1
- package/dist/agent-src/skills/refine-ticket/evals/triggers.json +1 -0
- package/dist/agent-src/skills/repomix-packer/SKILL.md +1 -1
- package/dist/agent-src/skills/roadmap-management/SKILL.md +16 -3
- package/dist/agent-src/skills/roadmap-writing/SKILL.md +76 -0
- package/dist/agent-src/skills/root-cause-frameworks/SKILL.md +146 -0
- package/dist/agent-src/skills/rule-refactor/SKILL.md +9 -9
- package/dist/agent-src/skills/rule-writing/SKILL.md +7 -7
- package/dist/agent-src/skills/script-writing/SKILL.md +2 -2
- package/dist/agent-src/skills/secrets-management/SKILL.md +1 -1
- package/dist/agent-src/skills/security-audit/SKILL.md +5 -0
- package/dist/agent-src/skills/skill-improvement-pipeline/SKILL.md +19 -3
- package/dist/agent-src/skills/skill-management/SKILL.md +3 -3
- package/dist/agent-src/skills/skill-reviewer/SKILL.md +1 -1
- package/dist/agent-src/skills/skill-writing/SKILL.md +5 -5
- package/dist/agent-src/skills/skill-writing/evals/triggers.json +1 -0
- package/dist/agent-src/skills/source-discovery/SKILL.md +182 -0
- package/dist/agent-src/skills/standards-from-config/SKILL.md +93 -0
- package/dist/agent-src/skills/subagent-orchestration/SKILL.md +10 -3
- package/dist/agent-src/skills/systematic-debugging/SKILL.md +7 -0
- package/dist/agent-src/skills/tailwind-engineer/scripts/tailwind_config_gen.ts +561 -0
- package/dist/agent-src/skills/testing-anti-patterns/SKILL.md +1 -1
- package/dist/agent-src/skills/testing-anti-patterns/process-anti-patterns.md +1 -1
- package/dist/agent-src/skills/threat-modeling/SKILL.md +1 -0
- package/dist/agent-src/skills/token-optimizer/SKILL.md +1 -1
- package/dist/agent-src/skills/typography-system/SKILL.md +138 -0
- package/dist/agent-src/skills/typography-system/evals/triggers.json +17 -0
- package/dist/agent-src/skills/upstream-contribute/SKILL.md +3 -3
- package/dist/agent-src/skills/verify-repair-loop/SKILL.md +209 -0
- package/dist/agent-src/skills/verify-repair-loop/evals/output-schema.yml +20 -0
- package/dist/agent-src/skills/verify-repair-loop/evals/triggers.json +17 -0
- package/dist/agent-src/templates/agent-settings.md +7 -0
- package/dist/agent-src/templates/agents/.gitattributes.fragment +0 -1
- package/dist/agent-src/templates/agents/agent-project-settings.example.yml +4 -4
- package/dist/agent-src/templates/contexts/knowledge-card.md +69 -0
- package/dist/agent-src/templates/contexts/lesson-card.md +73 -0
- package/dist/agent-src/templates/roadmaps.md +29 -1
- package/dist/agent-src/templates/scripts/README.md +6 -6
- package/dist/agent-src/templates/scripts/check_memory.ts +640 -0
- package/dist/agent-src/templates/scripts/check_memory_proposal.ts +351 -0
- package/dist/agent-src/templates/scripts/implement_ticket/__main__.ts +27 -0
- package/dist/agent-src/templates/scripts/memory_hash.ts +333 -0
- package/dist/agent-src/templates/scripts/memory_lookup.ts +1067 -0
- package/dist/agent-src/templates/scripts/memory_report.ts +846 -0
- package/dist/agent-src/templates/scripts/memory_signal.ts +422 -0
- package/dist/agent-src/templates/scripts/memory_status.ts +191 -0
- package/dist/agent-src/templates/scripts/pr_review_routing.ts +523 -0
- package/dist/agent-src/templates/scripts/pr_risk_review.ts +0 -0
- package/dist/agent-src/templates/scripts/telemetry/aggregator.ts +0 -0
- package/dist/agent-src/templates/scripts/telemetry/boundary.ts +164 -0
- package/dist/agent-src/templates/scripts/telemetry/engagement.ts +479 -0
- package/dist/agent-src/templates/scripts/telemetry/report_renderer.ts +394 -0
- package/dist/agent-src/templates/scripts/telemetry/settings.ts +210 -0
- package/dist/agent-src/templates/scripts/telemetry_record.ts +255 -0
- package/dist/agent-src/templates/scripts/telemetry_report.ts +189 -0
- package/dist/agent-src/templates/scripts/telemetry_status.ts +312 -0
- package/dist/agent-src/templates/scripts/tier_usage_report.ts +597 -0
- package/dist/agent-src/templates/scripts/work_engine/__main__.ts +14 -0
- package/dist/agent-src/templates/scripts/work_engine/_lib/agent_settings.ts +1118 -0
- package/dist/agent-src/templates/scripts/work_engine/_lib/user_global_paths.ts +329 -0
- package/dist/agent-src/templates/scripts/work_engine/cli.ts +206 -0
- package/dist/agent-src/templates/scripts/work_engine/cli_args.ts +249 -0
- package/dist/agent-src/templates/scripts/work_engine/delivery_state.ts +225 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/analyze.ts +125 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/implement.ts +189 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/index.ts +94 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/memory.ts +193 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/plan.ts +267 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/refine.ts +518 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/report.ts +379 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/test.ts +268 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/verify.ts +258 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/index.ts +32 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/mixed/contract.ts +243 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/mixed/index.ts +108 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/mixed/stitch.ts +259 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/mixed/ui.ts +216 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/_passthrough.ts +40 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/app_spec.ts +241 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/apply.ts +216 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/audit.ts +506 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/design.ts +325 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/index.ts +102 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/polish.ts +462 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/review.ts +474 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/scaffold.ts +352 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/_skipped.ts +33 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/apply.ts +213 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/index.ts +111 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/refine.ts +126 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/report.ts +112 -0
- package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/test.ts +164 -0
- package/dist/agent-src/templates/scripts/work_engine/dispatcher.ts +515 -0
- package/dist/agent-src/templates/scripts/work_engine/emitters.ts +119 -0
- package/dist/agent-src/templates/scripts/work_engine/errors.ts +24 -0
- package/dist/agent-src/templates/scripts/work_engine/hook_bootstrap.ts +104 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/_chat_history_base.ts +176 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/chat_history_append.ts +41 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/chat_history_halt_append.ts +89 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/decision_gate.ts +193 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/decision_trace.ts +304 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/directive_set_guard.ts +110 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/halt_surface_audit.ts +118 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/index.ts +17 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/memory_visibility.ts +161 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/state_shape_validation.ts +45 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/trace.ts +134 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/context.ts +94 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/events.ts +58 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/exceptions.ts +85 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/index.ts +27 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/registry.ts +66 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/runner.ts +90 -0
- package/dist/agent-src/templates/scripts/work_engine/hooks/settings.ts +260 -0
- package/dist/agent-src/templates/scripts/work_engine/input_builders.ts +260 -0
- package/dist/agent-src/templates/scripts/work_engine/intent/classify.ts +466 -0
- package/dist/agent-src/templates/scripts/work_engine/migration/v0_to_v1.ts +531 -0
- package/dist/agent-src/templates/scripts/work_engine/orchestration.ts +366 -0
- package/dist/agent-src/templates/scripts/work_engine/persona_policy.ts +97 -0
- package/dist/agent-src/templates/scripts/work_engine/resolvers/diff.ts +135 -0
- package/dist/agent-src/templates/scripts/work_engine/resolvers/file.ts +175 -0
- package/dist/agent-src/templates/scripts/work_engine/resolvers/prompt.ts +115 -0
- package/dist/agent-src/templates/scripts/work_engine/scoring/confidence.ts +415 -0
- package/dist/agent-src/templates/scripts/work_engine/scoring/decision_engine.ts +466 -0
- package/dist/agent-src/templates/scripts/work_engine/scoring/decision_trace.ts +298 -0
- package/dist/agent-src/templates/scripts/work_engine/scoring/memory_visibility.ts +444 -0
- package/dist/agent-src/templates/scripts/work_engine/stack/detect.ts +252 -0
- package/dist/agent-src/templates/scripts/work_engine/stack/runner.ts +745 -0
- package/dist/agent-src/templates/scripts/work_engine/state.ts +1151 -0
- package/dist/agent-src/templates/scripts/work_engine/state_io.ts +413 -0
- package/dist/agent-src/templates/tickets.md +120 -0
- package/dist/cli/agent-config.js +31 -300
- package/dist/cli/agent-config.js.map +1 -1
- package/dist/cli/commands/commands.js +11 -6
- package/dist/cli/commands/commands.js.map +1 -1
- package/dist/cli/commands/doctorShell.js +4 -22
- package/dist/cli/commands/doctorShell.js.map +1 -1
- package/dist/cli/commands/packs.js +1 -1
- package/dist/cli/commands/packs.js.map +1 -1
- package/dist/cli/commands/recordTriggerEval.js +179 -0
- package/dist/cli/commands/recordTriggerEval.js.map +1 -0
- package/dist/cli/commands/recordTriggerEval.test.js +113 -0
- package/dist/cli/commands/recordTriggerEval.test.js.map +1 -0
- package/dist/cli/commands/workspaces.js +1 -1
- package/dist/cli/commands/workspaces.js.map +1 -1
- package/dist/cli/discovery/loadManifest.js.map +1 -1
- package/dist/cli/main.js +330 -0
- package/dist/cli/main.js.map +1 -0
- package/dist/cli/python/knowledge_ingest.js +1048 -0
- package/dist/cli/python/knowledge_ingest.js.map +1 -0
- package/dist/cli/python/workspace_analytics.js +1085 -0
- package/dist/cli/python/workspace_analytics.js.map +1 -0
- package/dist/cli/python/workspace_crypto.js +544 -0
- package/dist/cli/python/workspace_crypto.js.map +1 -0
- package/dist/cli/python/workspace_documents.js +1216 -0
- package/dist/cli/python/workspace_documents.js.map +1 -0
- package/dist/cli/python/workspace_drive.js +574 -0
- package/dist/cli/python/workspace_drive.js.map +1 -0
- package/dist/cli/python/workspace_drive_health.js +628 -0
- package/dist/cli/python/workspace_drive_health.js.map +1 -0
- package/dist/cli/python/workspace_explain.js +765 -0
- package/dist/cli/python/workspace_explain.js.map +1 -0
- package/dist/cli/python/workspace_hosts.js +349 -0
- package/dist/cli/python/workspace_hosts.js.map +1 -0
- package/dist/cli/python/workspace_inbox.js +692 -0
- package/dist/cli/python/workspace_inbox.js.map +1 -0
- package/dist/cli/python/workspace_render.js +816 -0
- package/dist/cli/python/workspace_render.js.map +1 -0
- package/dist/cli/python/workspace_roles.js +487 -0
- package/dist/cli/python/workspace_roles.js.map +1 -0
- package/dist/cli/python/workspace_secrets.js +180 -0
- package/dist/cli/python/workspace_secrets.js.map +1 -0
- package/dist/cli/python/workspace_sessions.js +1079 -0
- package/dist/cli/python/workspace_sessions.js.map +1 -0
- package/dist/cli/python/workspace_skills.js +417 -0
- package/dist/cli/python/workspace_skills.js.map +1 -0
- package/dist/cli/registry.js +2 -0
- package/dist/cli/registry.js.map +1 -1
- package/dist/discovery/deprecation-report.md +1 -1
- package/dist/discovery/discovery-manifest.json +1802 -448
- package/dist/discovery/discovery-manifest.json.sha256 +1 -1
- package/dist/discovery/discovery-manifest.summary.md +12 -6
- package/dist/discovery/orphan-report.md +1 -1
- package/dist/discovery/packs.json +303 -43
- package/dist/discovery/trust-report.md +4 -4
- package/dist/discovery/workspaces.json +127 -41
- package/dist/install/install.mjs +13934 -0
- package/dist/mcp/registry-manifest.json +4 -4
- package/dist/router.json +1 -1
- package/dist/server/routes/wizard.js +54 -24
- package/dist/server/routes/wizard.js.map +1 -1
- package/dist/server/routes/workspace.js +44 -25
- package/dist/server/routes/workspace.js.map +1 -1
- package/dist/server/schemas/settings.js +33 -0
- package/dist/server/schemas/settings.js.map +1 -1
- package/docs/MIGRATION.md +1 -1
- package/docs/SKILL_CENSUS.md +344 -0
- package/docs/adrs/cost/0001-hard-stop-hook.md +5 -5
- package/docs/adrs/memory/0001-consumer-side-snapshot.md +15 -7
- package/docs/adrs/memory/README.md +6 -5
- package/docs/adrs/router/0001-three-tier-routing.md +2 -2
- package/docs/adrs/schema/0001-json-schema-frontmatter.md +2 -2
- package/docs/adrs/smoke/0001-per-tier-smoke-scripts.md +5 -5
- package/docs/adrs/telegraph/0001-default-off-until-bench.md +3 -3
- package/docs/architecture/augment-projection.md +1 -1
- package/docs/architecture/multi-tool-projection.md +3 -3
- package/docs/architecture.md +42 -11
- package/docs/archive/CHANGELOG-pre-2.2.0.md +30 -30
- package/docs/archive/CHANGELOG-pre-2.25.0.md +1 -1
- package/docs/archive/CHANGELOG-pre-4.5.0.md +1 -1
- package/docs/archive/CHANGELOG-pre-6.0.0.md +473 -0
- package/docs/benchmark.md +51 -53
- package/docs/benchmarks.md +2 -2
- package/docs/capability-matrix.md +32 -0
- package/docs/case-studies/frontend-design-positioning.md +86 -0
- package/docs/catalog.md +63 -15
- package/docs/command-flows.md +90 -92
- package/docs/command-naming-audit.md +60 -0
- package/docs/contracts/STABILITY.md +32 -0
- package/docs/contracts/adr-layout.md +2 -3
- package/docs/contracts/adr-level-6-productization.md +1 -1
- package/docs/contracts/agents-md-tech-stack.md +1 -1
- package/docs/contracts/ai-council-config.md +64 -29
- package/docs/contracts/analysis-memory-loop.md +149 -0
- package/docs/contracts/benchmark-ab-contract.md +3 -3
- package/docs/contracts/branch-protection-policy.md +27 -0
- package/docs/contracts/brand-token-consumption.md +69 -0
- package/docs/contracts/command-clusters.md +3 -2
- package/docs/contracts/command-surface-tiers.md +13 -0
- package/docs/contracts/cost-enforcement.md +1 -1
- package/docs/contracts/cost-summary-schema.md +1 -1
- package/docs/contracts/daily-workspace.md +1 -0
- package/docs/contracts/discovery-manifest.schema.json +25 -4
- package/docs/contracts/explain-modes.md +1 -1
- package/docs/contracts/implement-ticket-flow.md +15 -16
- package/docs/contracts/install-layout.md +249 -0
- package/docs/contracts/kernel-membership.md +1 -1
- package/docs/contracts/linear-ai-rules-inclusion.md +2 -2
- package/docs/contracts/linter-structural-model.md +1 -1
- package/docs/contracts/mcp-discovery-phase-notice.md +1 -1
- package/docs/contracts/mcp-tool-inventory.md +10 -10
- package/docs/contracts/measurement-baseline.md +1 -1
- package/docs/contracts/memory-visibility-v1.md +1 -5
- package/docs/contracts/multi-tool-projection-fidelity.md +1 -1
- package/docs/contracts/namespace.md +3 -3
- package/docs/contracts/no-runtime-boundary.md +56 -0
- package/docs/contracts/package-self-orientation.md +24 -0
- package/docs/contracts/persona-schema.md +1 -1
- package/docs/contracts/provider-lifecycle.md +3 -3
- package/docs/contracts/reasoning-discipline-protocol.md +83 -0
- package/docs/contracts/rule-classification.md +3 -3
- package/docs/contracts/rule-interactions.md +1 -1
- package/docs/contracts/skill-domains.md +1 -1
- package/docs/contracts/smoke-contracts.md +2 -2
- package/docs/contracts/surface-tiers.md +81 -0
- package/docs/contracts/ticket-bundle-format.md +228 -0
- package/docs/contracts/universal-skills.md +0 -1
- package/docs/contracts/workspace-boundary.md +84 -0
- package/docs/cookbook.md +152 -0
- package/docs/customization.md +15 -4
- package/docs/decisions/ADR-009-event4u-namespace.md +1 -1
- package/docs/decisions/ADR-013-discovery-frontmatter-contract.md +17 -1
- package/docs/decisions/ADR-026-explain-mode-translation.md +1 -1
- package/docs/decisions/ADR-056-unvalidated-video-adapters-disposition.md +1 -1
- package/docs/decisions/ADR-059-render-resume-filesystem-as-state.md +1 -1
- package/docs/decisions/ADR-060-comfyui-sandbox-model.md +1 -1
- package/docs/decisions/ADR-061-corpus-grounding-layer.md +48 -1
- package/docs/decisions/ADR-088-no-external-runtime-federation.md +26 -27
- package/docs/decisions/ADR-090-visibility-command-frontmatter-field.md +95 -0
- package/docs/decisions/ADR-091-split-meta-capability-packs.md +113 -0
- package/docs/decisions/ADR-092-defer-command-tier-alias-removal.md +93 -0
- package/docs/decisions/ADR-093-ai-council-config-user-global.md +111 -0
- package/docs/decisions/ADR-094-agent-memory-layer-removal.md +94 -0
- package/docs/decisions/ADR-095-workspace-boundary-contract.md +108 -0
- package/docs/decisions/ADR-096-analysis-workbench.md +110 -0
- package/docs/decisions/ADR-097-mission-recipe-privilege-boundary.md +121 -0
- package/docs/decisions/ADR-098-evidence-first-structure-discovery.md +154 -0
- package/docs/decisions/ADR-099-file-first-pattern-library.md +87 -0
- package/docs/decisions/ADR-100-global-knowledge-card-sharing.md +133 -0
- package/docs/decisions/ADR-101-ticket-bundle-emission.md +109 -0
- package/docs/decisions/ADR-102-ticket-handoff-paste-and-mcp.md +72 -0
- package/docs/decisions/ADR-103-global-knowledge-default-off-until-measured.md +92 -0
- package/docs/decisions/ADR-200-python-to-typescript-migration.md +193 -0
- package/docs/decisions/INDEX.md +15 -0
- package/docs/development.md +5 -7
- package/docs/distribution/mcp-submission-checklist.md +3 -3
- package/docs/featured-commands.md +1 -1
- package/docs/featured-skills.md +1 -1
- package/docs/getting-started-by-role.md +2 -0
- package/docs/getting-started.md +4 -4
- package/docs/guidelines/agent-infra/5w2h-analysis.md +1 -1
- package/docs/guidelines/agent-infra/comparison-matrix.md +1 -1
- package/docs/guidelines/agent-infra/corpus-grounding-authoring.md +1 -1
- package/docs/guidelines/agent-infra/critical-thinking.md +1 -1
- package/docs/guidelines/agent-infra/engineering-memory-data-format.md +1 -5
- package/docs/guidelines/agent-infra/failure-signatures.md +35 -0
- package/docs/guidelines/agent-infra/first-principles.md +1 -1
- package/docs/guidelines/agent-infra/frontier-reasoning-operating-profile.md +169 -0
- package/docs/guidelines/agent-infra/inversion-thinking.md +1 -1
- package/docs/guidelines/agent-infra/ios-simulator-guide.md +9 -14
- package/docs/guidelines/agent-infra/mcp-request-signing.md +19 -22
- package/docs/guidelines/agent-infra/memory-access.md +25 -31
- package/docs/guidelines/agent-infra/mental-models.md +1 -1
- package/docs/guidelines/agent-infra/model-recommendation.md +29 -0
- package/docs/guidelines/agent-infra/scqa-framework.md +3 -3
- package/docs/guidelines/agent-infra/security-lint-containment.md +81 -0
- package/docs/guidelines/agent-infra/six-hats.md +1 -1
- package/docs/guidelines/agent-infra/size-and-scope.md +17 -0
- package/docs/guidelines/agent-infra/skill-quality-checklist.md +2 -2
- package/docs/guidelines/agent-infra/systems-thinking.md +1 -1
- package/docs/guidelines/agent-infra/untrusted-input-spotlighting.md +72 -0
- package/docs/guides/frontend-design-corpus-refresh.md +83 -0
- package/docs/guides/skill-preview.md +1 -1
- package/docs/hook-payload-capture.md +4 -4
- package/docs/installation.md +1 -1
- package/docs/mcp.md +3 -3
- package/docs/migration/consumer-template-consumption-model.md +145 -0
- package/docs/migration/divergences/README.md +55 -0
- package/docs/migration/divergences/_template.md +50 -0
- package/docs/migration/divergences/bench-stats-float-precision.md +72 -0
- package/docs/migration/divergences/mcp-telemetry-node-sqlite.md +61 -0
- package/docs/migration/divergences/pack-mcp-content-gzip-body.md +53 -0
- package/docs/migration/divergences/src-scripts-build_cloud_bundle.md +63 -0
- package/docs/migration/divergences/src-scripts-check_memory.md +91 -0
- package/docs/migration/divergences/src-scripts-inventory_abstraction_budget.md +65 -0
- package/docs/migration/divergences/src-scripts-lint_marketplace.md +57 -0
- package/docs/migration/divergences/src-scripts-lint_mcp_registry_manifest.md +70 -0
- package/docs/migration/divergences/src-scripts-spotcheck_thin_root.md +60 -0
- package/docs/migration/divergences/src-scripts-validate_agent_settings.md +58 -0
- package/docs/migration/node-floor.md +86 -0
- package/docs/migration/yaml-roundtrip-spike.md +163 -0
- package/docs/parity/bench-external.json +58 -0
- package/docs/parity/external-runtime.md +46 -0
- package/docs/personas.md +6 -1
- package/docs/quality.md +3 -3
- package/docs/role-experiences.md +19 -0
- package/docs/safety.md +3 -3
- package/docs/setup/per-ide/windsurf.md +1 -1
- package/docs/skills-catalog.md +26 -2
- package/docs/threat-model.md +28 -0
- package/llms.txt +25 -1
- package/package.json +10 -15
- package/src/config/agent-settings.template.yml +128 -3
- package/src/config/discovery/packs.yml +60 -0
- package/src/config/discovery/unassigned-artefacts.yml +6 -0
- package/src/config/discovery/workspaces.yml +5 -3
- package/src/config/gitignore-block.txt +13 -0
- package/src/scripts/_cli/cmd_doctor.ts +2306 -0
- package/src/scripts/_cli/cmd_explain.ts +748 -0
- package/src/scripts/_cli/cmd_export.ts +375 -0
- package/src/scripts/_cli/cmd_migrate.ts +951 -0
- package/src/scripts/_cli/cmd_prune.ts +610 -0
- package/src/scripts/_cli/cmd_refresh.ts +530 -0
- package/src/scripts/_cli/cmd_settings_check.ts +407 -0
- package/src/scripts/_cli/cmd_settings_migrate.ts +344 -0
- package/src/scripts/_cli/cmd_sync.ts +381 -0
- package/src/scripts/_cli/cmd_uninstall.ts +833 -0
- package/src/scripts/_cli/cmd_update.ts +585 -0
- package/src/scripts/_cli/cmd_upgrade.ts +390 -0
- package/src/scripts/_cli/cmd_validate.ts +394 -0
- package/src/scripts/_cli/cmd_versions.ts +492 -0
- package/src/scripts/_cli/explain_last/assumptions.ts +114 -0
- package/src/scripts/_cli/explain_last/council.ts +197 -0
- package/src/scripts/_cli/explain_last/halt.ts +73 -0
- package/src/scripts/_cli/explain_last/index.ts +155 -0
- package/src/scripts/_cli/explain_last/inputs.ts +211 -0
- package/src/scripts/_cli/explain_last/memory.ts +231 -0
- package/src/scripts/_cli/explain_last/provider.ts +82 -0
- package/src/scripts/_cli/explain_last/render.ts +54 -0
- package/src/scripts/_cli/explain_last/route.ts +70 -0
- package/src/scripts/_cli/explain_last/scrubber.ts +138 -0
- package/src/scripts/_cli/explain_last/sections/assumptions.ts +51 -0
- package/src/scripts/_cli/explain_last/sections/council.ts +56 -0
- package/src/scripts/_cli/explain_last/sections/halt.ts +60 -0
- package/src/scripts/_cli/explain_last/sections/header.ts +50 -0
- package/src/scripts/_cli/explain_last/sections/index.ts +21 -0
- package/src/scripts/_cli/explain_last/sections/inputs.ts +63 -0
- package/src/scripts/_cli/explain_last/sections/memory.ts +124 -0
- package/src/scripts/_cli/explain_last/sections/pack.ts +42 -0
- package/src/scripts/_cli/explain_last/sections/provider.ts +51 -0
- package/src/scripts/_cli/explain_last/sections/route.ts +48 -0
- package/src/scripts/_cli/explain_last/state_loader.ts +119 -0
- package/src/scripts/_dispatch.bash +179 -163
- package/src/scripts/_lib/agent_settings.ts +1123 -0
- package/src/scripts/_lib/agent_src.ts +654 -0
- package/src/scripts/_lib/agents_overlay.ts +183 -0
- package/src/scripts/_lib/bench_ab_cache.ts +399 -0
- package/src/scripts/_lib/bench_ab_scoring.ts +352 -0
- package/src/scripts/_lib/bench_ab_scoring_v2.ts +751 -0
- package/src/scripts/_lib/bench_cost.ts +396 -0
- package/src/scripts/_lib/bench_quality.ts +237 -0
- package/src/scripts/_lib/bench_report.ts +255 -0
- package/src/scripts/_lib/bench_telegraph.ts +516 -0
- package/src/scripts/_lib/bench_telegraph_report.ts +272 -0
- package/src/scripts/_lib/changelog_eras.ts +398 -0
- package/src/scripts/_lib/claude_desktop_bundler.ts +324 -0
- package/src/scripts/_lib/cli_wrapper.ts +89 -0
- package/src/scripts/_lib/fs_atomic.ts +172 -0
- package/src/scripts/_lib/global_deploy_inventory.ts +639 -0
- package/src/scripts/_lib/install_layout.ts +87 -0
- package/src/scripts/_lib/install_regenerator.ts +157 -0
- package/src/scripts/_lib/installed_lock.ts +451 -0
- package/src/scripts/_lib/installed_tools.ts +518 -0
- package/src/scripts/_lib/json_pointers.ts +388 -0
- package/src/scripts/_lib/knowledge_global.ts +770 -0
- package/src/scripts/_lib/knowledge_global_promote.ts +453 -0
- package/src/scripts/_lib/knowledge_global_redaction.ts +448 -0
- package/src/scripts/_lib/link_crypto.ts +325 -0
- package/src/scripts/_lib/linked_projects.ts +613 -0
- package/src/scripts/_lib/model_tier.ts +65 -0
- package/src/scripts/_lib/module_detection.ts +275 -0
- package/src/scripts/_lib/node_sqlite.d.ts +32 -0
- package/src/scripts/_lib/pin_resolver.ts +264 -0
- package/src/scripts/_lib/py_random.ts +212 -0
- package/src/scripts/_lib/script_output.ts +147 -0
- package/src/scripts/_lib/security_lint.ts +623 -0
- package/src/scripts/_lib/surface_tiers.ts +127 -0
- package/src/scripts/_lib/token_count.ts +126 -0
- package/src/scripts/_lib/update_check.ts +297 -0
- package/src/scripts/_lib/user_global_paths.ts +329 -0
- package/src/scripts/_lib/value_ladder.ts +882 -0
- package/src/scripts/_lib/value_report.ts +617 -0
- package/src/scripts/_lib/zip_min.ts +175 -0
- package/src/scripts/adoption_report.ts +357 -0
- package/src/scripts/adoption_snapshot.ts +392 -0
- package/src/scripts/adoption_status.ts +424 -0
- package/src/scripts/adr/regenerate_index.ts +257 -0
- package/src/scripts/ai-image/adapters/flux.sh +45 -0
- package/src/scripts/ai-image/adapters/gemini-image.sh +45 -0
- package/src/scripts/ai-image/adapters/ideogram.sh +45 -0
- package/src/scripts/ai-image/adapters/recraft.sh +47 -0
- package/src/scripts/ai-video/adapters/comfyui.sh +3 -3
- package/src/scripts/ai-video/adapters/fal.sh +3 -3
- package/src/scripts/ai-video/adapters/gemini-veo.sh +3 -3
- package/src/scripts/ai-video/adapters/higgsfield.sh +3 -3
- package/src/scripts/ai-video/adapters/kling.sh +3 -3
- package/src/scripts/ai-video/adapters/musetalk.sh +2 -2
- package/src/scripts/ai-video/adapters/openai-images.sh +3 -3
- package/src/scripts/ai-video/adapters/replicate.sh +3 -3
- package/src/scripts/ai-video/adapters/sora.sh +3 -3
- package/src/scripts/ai-video/adapters/syncso.sh +3 -3
- package/src/scripts/ai-video/audio-adapters/allin1.sh +2 -2
- package/src/scripts/ai-video/audio-adapters/whisperx.sh +2 -2
- package/src/scripts/ai-video/lib/audio-adapter-contract.md +1 -1
- package/src/scripts/ai-video/lib/embed-provenance.sh +2 -2
- package/src/scripts/ai-video/lib/ingest-song.sh +2 -2
- package/src/scripts/ai-video/lib/parse-blueprint.sh +1 -1
- package/src/scripts/ai-video/lib/resume-scan.sh +2 -2
- package/src/scripts/ai-video/smoke-trace.sh +16 -7
- package/src/scripts/ai-video/stitch.sh +2 -2
- package/src/scripts/ai_council/_default_prices.ts +73 -0
- package/src/scripts/ai_council/advisors.ts +244 -0
- package/src/scripts/ai_council/airgap.ts +249 -0
- package/src/scripts/ai_council/budget_guard.ts +492 -0
- package/src/scripts/ai_council/bundler.ts +376 -0
- package/src/scripts/ai_council/cli_hints.ts +120 -0
- package/src/scripts/ai_council/clients.ts +2214 -0
- package/src/scripts/ai_council/compile_corpus.ts +681 -0
- package/src/scripts/ai_council/confidence_gate.ts +230 -0
- package/src/scripts/ai_council/config.ts +1729 -0
- package/src/scripts/ai_council/consensus.ts +551 -0
- package/src/scripts/ai_council/events_log.ts +327 -0
- package/src/scripts/ai_council/learn_low_impact_preview.ts +317 -0
- package/src/scripts/ai_council/low_impact.ts +1069 -0
- package/src/scripts/ai_council/low_impact_corpus.ts +662 -0
- package/src/scripts/ai_council/low_impact_intake.ts +222 -0
- package/src/scripts/ai_council/modes.ts +169 -0
- package/src/scripts/ai_council/necessity.ts +933 -0
- package/src/scripts/ai_council/orchestrator.ts +1689 -0
- package/src/scripts/ai_council/pricing.ts +267 -0
- package/src/scripts/ai_council/probation_gate.ts +282 -0
- package/src/scripts/ai_council/project_context.ts +308 -0
- package/src/scripts/ai_council/prompts.ts +600 -0
- package/src/scripts/ai_council/redact_low_impact_entry.ts +291 -0
- package/src/scripts/ai_council/replay.ts +314 -0
- package/src/scripts/ai_council/session.ts +558 -0
- package/src/scripts/ai_council/shadow_dispatch.ts +509 -0
- package/src/scripts/ai_council/solo_dispatch.ts +281 -0
- package/src/scripts/analysis_freshness.ts +343 -0
- package/src/scripts/annotate_discovery.ts +288 -0
- package/src/scripts/apply_modules_config.ts +537 -0
- package/src/scripts/audit_adr_coverage.ts +357 -0
- package/src/scripts/audit_auto_rules.ts +415 -0
- package/src/scripts/audit_cloud_compatibility.ts +608 -0
- package/src/scripts/audit_command_surface.ts +1227 -0
- package/src/scripts/audit_initial_context.ts +694 -0
- package/src/scripts/audit_likelihood.ts +434 -0
- package/src/scripts/audit_mcp_tools.ts +252 -0
- package/src/scripts/audit_overlap.ts +421 -0
- package/src/scripts/audit_skill_descriptions.ts +402 -0
- package/src/scripts/audit_skill_overlap.ts +576 -0
- package/src/scripts/audit_user_type_axis.ts +264 -0
- package/src/scripts/backfill_model_tier.ts +349 -0
- package/src/scripts/bench_ab_cache_dispatch.ts +126 -0
- package/src/scripts/bench_ab_clone.ts +610 -0
- package/src/scripts/bench_ab_diff.ts +609 -0
- package/src/scripts/bench_ab_integrity.ts +261 -0
- package/src/scripts/bench_ab_run.ts +417 -0
- package/src/scripts/bench_ab_task_runner.ts +1382 -0
- package/src/scripts/bench_ab_tracka_run.ts +436 -0
- package/src/scripts/bench_ab_v2_run.ts +585 -0
- package/src/scripts/bench_ab_v2_stats.ts +1018 -0
- package/src/scripts/bench_baseline_ready.ts +326 -0
- package/src/scripts/bench_condense_memory.ts +479 -0
- package/src/scripts/bench_drift_check.ts +503 -0
- package/src/scripts/bench_per_tool.ts +591 -0
- package/src/scripts/bench_rtk_savings.ts +710 -0
- package/src/scripts/bench_run.ts +509 -0
- package/src/scripts/bench_runner.ts +519 -0
- package/src/scripts/build_cloud_bundle.ts +692 -0
- package/src/scripts/build_discovery_manifest.ts +1371 -0
- package/src/scripts/build_linear_digest.ts +368 -0
- package/src/scripts/build_mcp_registry_manifest.ts +351 -0
- package/src/scripts/build_rule_trigger_matrix.ts +469 -0
- package/src/scripts/capture_showcase_session.ts +735 -0
- package/src/scripts/chat_history.ts +2301 -0
- package/src/scripts/check_always_budget.ts +694 -0
- package/src/scripts/check_artefact_checksums.ts +281 -0
- package/src/scripts/check_augment_description_cap.ts +133 -0
- package/src/scripts/check_augmentignore.ts +108 -0
- package/src/scripts/check_beta_review_markers.ts +234 -0
- package/src/scripts/check_bite_sized_granularity.ts +116 -0
- package/src/scripts/check_cluster_patterns.ts +285 -0
- package/src/scripts/check_command_count_messaging.ts +224 -0
- package/src/scripts/check_condensation.ts +900 -0
- package/src/scripts/check_condensed_paths.ts +414 -0
- package/src/scripts/check_context_paths.ts +388 -0
- package/src/scripts/check_council_config_location.ts +260 -0
- package/src/scripts/check_council_layout.ts +180 -0
- package/src/scripts/check_council_references.ts +345 -0
- package/src/scripts/check_discovery_determinism.ts +124 -0
- package/src/scripts/check_gate_paths.ts +230 -0
- package/src/scripts/check_iron_law_prominence.ts +298 -0
- package/src/scripts/check_kernel_rule_bundle.ts +242 -0
- package/src/scripts/check_knowledge_cards.ts +759 -0
- package/src/scripts/check_md_language.ts +291 -0
- package/src/scripts/check_memory.ts +845 -0
- package/src/scripts/check_memory_proposal.ts +351 -0
- package/src/scripts/check_module_management_neutral.ts +238 -0
- package/src/scripts/check_no_conflict_markers.ts +298 -0
- package/src/scripts/check_no_conflict_markers_allowlist.json +4 -0
- package/src/scripts/check_no_external_sources.ts +351 -0
- package/src/scripts/check_no_local_settings_committed.ts +69 -0
- package/src/scripts/check_no_new_legacy_path.ts +188 -0
- package/src/scripts/check_no_roadmap_refs.ts +304 -0
- package/src/scripts/check_one_off_location.ts +165 -0
- package/src/scripts/check_overlay_cascade_subdirs.ts +188 -0
- package/src/scripts/check_portability.ts +860 -0
- package/src/scripts/check_proposal.ts +0 -0
- package/src/scripts/check_public_catalog_links.ts +204 -0
- package/src/scripts/check_public_links.ts +357 -0
- package/src/scripts/check_references.ts +963 -0
- package/src/scripts/check_release_includes_discovery.ts +94 -0
- package/src/scripts/check_release_pr_shape.ts +222 -0
- package/src/scripts/check_release_published.ts +235 -0
- package/src/scripts/check_release_trunk_sync.ts +203 -0
- package/src/scripts/check_reply_consistency.ts +359 -0
- package/src/scripts/check_roadmap_trackable.ts +268 -0
- package/src/scripts/check_role_doc_links.ts +187 -0
- package/src/scripts/check_safety_floor_untouched.ts +160 -0
- package/src/scripts/check_skill_requires.ts +205 -0
- package/src/scripts/check_structural_breaking.ts +170 -0
- package/src/scripts/check_surface_tiers.ts +567 -0
- package/src/scripts/check_template_pin_drift.ts +222 -0
- package/src/scripts/check_test_coverage_diff.ts +235 -0
- package/src/scripts/check_token_optimizer_freshness.ts +183 -0
- package/src/scripts/check_trigger_evals.ts +375 -0
- package/src/scripts/check_update_banner.ts +143 -0
- package/src/scripts/ci_status.ts +0 -0
- package/src/scripts/ci_summary.ts +235 -0
- package/src/scripts/ci_time_ratio.ts +526 -0
- package/src/scripts/command_suggester/cooldown.ts +176 -0
- package/src/scripts/command_suggester/index.ts +41 -0
- package/src/scripts/command_suggester/loader.ts +205 -0
- package/src/scripts/command_suggester/match.ts +294 -0
- package/src/scripts/command_suggester/rank.ts +201 -0
- package/src/scripts/command_suggester/render.ts +122 -0
- package/src/scripts/command_suggester/sanitize.ts +114 -0
- package/src/scripts/command_suggester/settings.ts +186 -0
- package/src/scripts/command_suggester/types.ts +0 -0
- package/src/scripts/compile_router.ts +297 -0
- package/src/scripts/condense.sh +7 -1
- package/src/scripts/condense.ts +2035 -0
- package/src/scripts/condense_memory.ts +334 -0
- package/src/scripts/config/index.ts +15 -0
- package/src/scripts/config/packs.ts +310 -0
- package/src/scripts/config/presets.ts +369 -0
- package/src/scripts/config/profile_explain.ts +114 -0
- package/src/scripts/config/profiles.ts +277 -0
- package/src/scripts/config/session_profiles.ts +1064 -0
- package/src/scripts/context_hygiene_hook.ts +272 -0
- package/src/scripts/cost_by_conversation.ts +444 -0
- package/src/scripts/cost_summary.ts +407 -0
- package/src/scripts/council_cli.ts +2827 -0
- package/src/scripts/council_prune.ts +153 -0
- package/src/scripts/cross_repo_retrieve.ts +694 -0
- package/src/scripts/discovery_stats.ts +218 -0
- package/src/scripts/evidence_report.ts +580 -0
- package/src/scripts/external_sources_denylist.json +92 -0
- package/src/scripts/extract_audit_patterns.ts +394 -0
- package/src/scripts/first_run_gate_hook.ts +246 -0
- package/src/scripts/gen_discovery_baseline.ts +297 -0
- package/src/scripts/generate_capabilities_index.ts +496 -0
- package/src/scripts/generate_capability_matrix.ts +430 -0
- package/src/scripts/generate_catalog.ts +178 -0
- package/src/scripts/generate_command_flows.ts +316 -0
- package/src/scripts/generate_cookbook.ts +302 -0
- package/src/scripts/generate_index.ts +500 -0
- package/src/scripts/generate_ownership_matrix.ts +646 -0
- package/src/scripts/generate_pack_manifests.ts +1025 -0
- package/src/scripts/generate_role_experiences_catalog.ts +265 -0
- package/src/scripts/hermetic-install.sh +22 -11
- package/src/scripts/hook_manifest.yaml +37 -15
- package/src/scripts/hooks/augment-chat-history.sh +3 -10
- package/src/scripts/hooks/augment-context-hygiene.sh +3 -10
- package/src/scripts/hooks/augment-dispatcher.sh +3 -10
- package/src/scripts/hooks/augment-onboarding-gate.sh +3 -10
- package/src/scripts/hooks/augment-roadmap-progress.sh +3 -10
- package/src/scripts/hooks/block_no_verify.ts +413 -0
- package/src/scripts/hooks/cline-dispatcher.sh +3 -10
- package/src/scripts/hooks/cowork-dispatcher.sh +3 -14
- package/src/scripts/hooks/cursor-dispatcher.sh +3 -10
- package/src/scripts/hooks/dispatch_hook.ts +851 -0
- package/src/scripts/hooks/dispatch_issues.ts +226 -0
- package/src/scripts/hooks/envelope.ts +140 -0
- package/src/scripts/hooks/gemini-dispatcher.sh +3 -8
- package/src/scripts/hooks/replay_hook.ts +364 -0
- package/src/scripts/hooks/state_io.ts +293 -0
- package/src/scripts/hooks/windsurf-dispatcher.sh +3 -9
- package/src/scripts/hooks_doctor.ts +418 -0
- package/src/scripts/hooks_status.ts +292 -0
- package/src/scripts/injection_scan_hook.ts +285 -0
- package/src/scripts/install +36 -22
- package/src/scripts/install-hooks.sh +29 -12
- package/src/scripts/install.sh +38 -14
- package/src/scripts/install.ts +4515 -0
- package/src/scripts/inventory_abstraction_budget.ts +1104 -0
- package/src/scripts/inventory_frontmatter.ts +320 -0
- package/src/scripts/inventory_meta_layers.ts +516 -0
- package/src/scripts/iron_law_sha.ts +233 -0
- package/src/scripts/knowledge_global_cli.ts +1105 -0
- package/src/scripts/linked_projects_list.ts +310 -0
- package/src/scripts/lint_agent_security.ts +224 -0
- package/src/scripts/lint_agent_skill_names.ts +241 -0
- package/src/scripts/lint_agents_layout.ts +205 -0
- package/src/scripts/lint_agents_md.ts +294 -0
- package/src/scripts/lint_archived_skills.ts +309 -0
- package/src/scripts/lint_artefact_frontmatter.ts +359 -0
- package/src/scripts/lint_bench_ab.ts +319 -0
- package/src/scripts/lint_bench_corpus.ts +421 -0
- package/src/scripts/lint_command_flow_coverage.ts +231 -0
- package/src/scripts/lint_command_routing.ts +377 -0
- package/src/scripts/lint_command_tiers.ts +345 -0
- package/src/scripts/lint_command_verbs.ts +379 -0
- package/src/scripts/lint_commit_subjects.ts +243 -0
- package/src/scripts/lint_context_spine_usage.ts +198 -0
- package/src/scripts/lint_discovery_manifest.ts +540 -0
- package/src/scripts/lint_discovery_vocabulary.ts +393 -0
- package/src/scripts/lint_empty_roadmaps.ts +147 -0
- package/src/scripts/lint_eval_freshness.ts +335 -0
- package/src/scripts/lint_examples.ts +183 -0
- package/src/scripts/lint_explain_trace.ts +381 -0
- package/src/scripts/lint_featured_skills.ts +0 -0
- package/src/scripts/lint_flows.ts +701 -0
- package/src/scripts/lint_framework_leakage.ts +497 -0
- package/src/scripts/lint_framework_leakage_allowlist.json +8 -1
- package/src/scripts/lint_frontmatter_boilerplate.ts +356 -0
- package/src/scripts/lint_ghostwriter_source.ts +389 -0
- package/src/scripts/lint_global_paths.ts +420 -0
- package/src/scripts/lint_handoffs.ts +362 -0
- package/src/scripts/lint_hidden_unicode.ts +350 -0
- package/src/scripts/lint_hook_concern_budget.ts +319 -0
- package/src/scripts/lint_hook_manifest.ts +354 -0
- package/src/scripts/lint_instruction_smuggling.ts +173 -0
- package/src/scripts/lint_load_context.ts +371 -0
- package/src/scripts/lint_marketplace.ts +286 -0
- package/src/scripts/lint_marketplace_install_completeness.ts +309 -0
- package/src/scripts/lint_mcp_config_security.ts +225 -0
- package/src/scripts/lint_mcp_registry_manifest.ts +350 -0
- package/src/scripts/lint_media_policy_linkage.ts +224 -0
- package/src/scripts/lint_missions.ts +774 -0
- package/src/scripts/lint_model_tier_coverage.ts +151 -0
- package/src/scripts/lint_namespace.ts +295 -0
- package/src/scripts/lint_namespace_collisions.ts +203 -0
- package/src/scripts/lint_new_skill_gate.ts +462 -0
- package/src/scripts/lint_no_new_atomic_commands.ts +342 -0
- package/src/scripts/lint_one_off_age.ts +348 -0
- package/src/scripts/lint_orchestration_dsl.ts +377 -0
- package/src/scripts/lint_orchestrator_auto_detect.ts +177 -0
- package/src/scripts/lint_pack_boundaries.ts +366 -0
- package/src/scripts/lint_pack_dependencies.ts +541 -0
- package/src/scripts/lint_pack_first_win.ts +202 -0
- package/src/scripts/lint_persona_governance.ts +292 -0
- package/src/scripts/lint_positioning.ts +257 -0
- package/src/scripts/lint_profile_overlay_set_only.ts +324 -0
- package/src/scripts/lint_readme_jargon.ts +189 -0
- package/src/scripts/lint_readme_size.ts +73 -0
- package/src/scripts/lint_regression.ts +497 -0
- package/src/scripts/lint_roadmap_ci_steps.ts +252 -0
- package/src/scripts/lint_roadmap_complexity.ts +295 -0
- package/src/scripts/lint_roadmap_later_disposition.ts +357 -0
- package/src/scripts/lint_role_experiences.ts +410 -0
- package/src/scripts/lint_rule_interactions.ts +281 -0
- package/src/scripts/lint_rule_tiers.ts +169 -0
- package/src/scripts/lint_showcase_sessions.ts +254 -0
- package/src/scripts/lint_skill_frontmatter_safety.ts +279 -0
- package/src/scripts/lint_skill_originality.ts +586 -0
- package/src/scripts/lint_skill_originality_allowlist.json +20 -0
- package/src/scripts/lint_skill_tools.ts +320 -0
- package/src/scripts/lint_ticket_buildable.ts +1027 -0
- package/src/scripts/lint_topics_yaml.ts +203 -0
- package/src/scripts/lint_trust_coherence.ts +377 -0
- package/src/scripts/lint_value_dashboard.ts +314 -0
- package/src/scripts/lint_workflow_security.ts +637 -0
- package/src/scripts/lint_workflow_security_allowlist.json +20 -0
- package/src/scripts/lint_workspace_boundary.ts +248 -0
- package/src/scripts/mcp_parity_smoke.ts +638 -0
- package/src/scripts/mcp_render.ts +346 -0
- package/src/scripts/mcp_server/__main__.ts +28 -0
- package/src/scripts/mcp_server/catalog.ts +154 -0
- package/src/scripts/mcp_server/consumer_tool_catalog.json +2 -3
- package/src/scripts/mcp_server/index.ts +24 -0
- package/src/scripts/mcp_server/metadata.ts +83 -0
- package/src/scripts/mcp_server/prompts.ts +711 -0
- package/src/scripts/mcp_server/resources.ts +343 -0
- package/src/scripts/mcp_server/server.ts +439 -0
- package/src/scripts/mcp_server/telemetry.ts +154 -0
- package/src/scripts/mcp_server/tools.ts +1031 -0
- package/src/scripts/mcp_setup.sh +25 -52
- package/src/scripts/mcp_telemetry_health.ts +362 -0
- package/src/scripts/mcp_telemetry_query.ts +371 -0
- package/src/scripts/mcp_telemetry_store.ts +422 -0
- package/src/scripts/measure_augment_budget.ts +453 -0
- package/src/scripts/measure_density.ts +618 -0
- package/src/scripts/measure_frugality_savings.ts +353 -0
- package/src/scripts/measure_markitdown_lift.ts +299 -0
- package/src/scripts/measure_patterns.ts +682 -0
- package/src/scripts/measure_projection_bytes.ts +425 -0
- package/src/scripts/measure_rule_budget.ts +627 -0
- package/src/scripts/measure_skill_reduction.ts +442 -0
- package/src/scripts/media/lib/adapter-common.sh +247 -0
- package/src/scripts/media/lib/adapter-contract.md +329 -0
- package/src/scripts/media/lib/fixtures/comfyui/result.json +1 -0
- package/src/scripts/media/lib/fixtures/fal/result.json +1 -0
- package/src/scripts/media/lib/fixtures/flux/asset-0001.png +0 -0
- package/src/scripts/media/lib/fixtures/flux/result.json +1 -0
- package/src/scripts/media/lib/fixtures/gemini-image/asset-0001.png +0 -0
- package/src/scripts/media/lib/fixtures/gemini-image/result.json +1 -0
- package/src/scripts/media/lib/fixtures/gemini-veo/result.json +1 -0
- package/src/scripts/media/lib/fixtures/higgsfield/result.json +1 -0
- package/src/scripts/media/lib/fixtures/ideogram/asset-0001.png +0 -0
- package/src/scripts/media/lib/fixtures/ideogram/result.json +1 -0
- package/src/scripts/media/lib/fixtures/kling/result.json +1 -0
- package/src/scripts/media/lib/fixtures/musetalk/result.json +1 -0
- package/src/scripts/media/lib/fixtures/openai-images/result.json +1 -0
- package/src/scripts/media/lib/fixtures/recraft/asset-0001.svg +1 -0
- package/src/scripts/media/lib/fixtures/recraft/result.json +1 -0
- package/src/scripts/media/lib/fixtures/replicate/result.json +1 -0
- package/src/scripts/media/lib/fixtures/sora/result.json +1 -0
- package/src/scripts/media/lib/fixtures/syncso/result.json +1 -0
- package/src/scripts/media/lib/load-config.sh +180 -0
- package/src/scripts/media/lib/redact.sh +85 -0
- package/src/scripts/memory_hash.ts +331 -0
- package/src/scripts/memory_lookup.ts +1278 -0
- package/src/scripts/memory_report.ts +845 -0
- package/src/scripts/memory_signal.ts +417 -0
- package/src/scripts/memory_status.ts +189 -0
- package/src/scripts/migrate_command_suggestions.ts +341 -0
- package/src/scripts/migrate_frontmatter_defaults.ts +539 -0
- package/src/scripts/migration_status.ts +301 -0
- package/src/scripts/mine_session.ts +645 -0
- package/src/scripts/minimal_safe_diff_hook.ts +355 -0
- package/src/scripts/move_artefact.ts +869 -0
- package/src/scripts/new_skill.ts +404 -0
- package/src/scripts/onboarding_gate_hook.ts +224 -0
- package/src/scripts/pack_dependency_allowlist.json +2 -2
- package/src/scripts/pack_mcp_content.ts +552 -0
- package/src/scripts/parity/README.md +140 -0
- package/src/scripts/parity/compare.ts +189 -0
- package/src/scripts/parity/coverage_diff.ts +199 -0
- package/src/scripts/parity/phase-manifest.json +93 -0
- package/src/scripts/parity/phase_gate.ts +270 -0
- package/src/scripts/parity/replay.ts +320 -0
- package/src/scripts/pattern_share.ts +363 -0
- package/src/scripts/plan_physical_move.ts +605 -0
- package/src/scripts/prediction-pool/poisson_sim.ts +537 -0
- package/src/scripts/prediction-pool/pool_winsim.ts +677 -0
- package/src/scripts/prediction-pool/score_ev.ts +546 -0
- package/src/scripts/print_required_checks.ts +249 -0
- package/src/scripts/probe_projection_fidelity.ts +468 -0
- package/src/scripts/probe_skill_registration.ts +787 -0
- package/src/scripts/profile_staleness_hook.ts +169 -0
- package/src/scripts/profile_use.ts +227 -0
- package/src/scripts/project_thin_rules.ts +387 -0
- package/src/scripts/propose_modules_config.ts +311 -0
- package/src/scripts/prototype_lint_contradictions.ts +414 -0
- package/src/scripts/prove_pack_extractable.ts +388 -0
- package/src/scripts/readme_linter.ts +913 -0
- package/src/scripts/redact_hook_capture.ts +325 -0
- package/src/scripts/refine_ticket_detect.ts +703 -0
- package/src/scripts/release.ts +1697 -0
- package/src/scripts/render_benchmark_md.ts +664 -0
- package/src/scripts/render_value_md.ts +506 -0
- package/src/scripts/repro/repro_marketplace_install_gap.sh +1 -1
- package/src/scripts/roadmap_progress_hook.ts +410 -0
- package/src/scripts/router_telemetry.ts +972 -0
- package/src/scripts/run.ts +98 -0
- package/src/scripts/run_skill_evals.ts +477 -0
- package/src/scripts/runtime_dispatcher.ts +586 -0
- package/src/scripts/runtime_handler.ts +231 -0
- package/src/scripts/runtime_registry.ts +394 -0
- package/src/scripts/schemas/command.schema.json +7 -1
- package/src/scripts/schemas/mission-catalog.schema.json +112 -0
- package/src/scripts/schemas/mission.schema.json +87 -0
- package/src/scripts/schemas/pack.schema.json +6 -0
- package/src/scripts/schemas/rule.schema.json +1 -0
- package/src/scripts/schemas/skill.schema.json +1 -0
- package/src/scripts/schemas/ticket-manifest.schema.json +35 -0
- package/src/scripts/schemas/ticket.schema.json +60 -0
- package/src/scripts/score_skill_selection.ts +570 -0
- package/src/scripts/security_audit_config.ts +423 -0
- package/src/scripts/skill_collision_clusters.ts +448 -0
- package/src/scripts/skill_discovery.ts +690 -0
- package/src/scripts/skill_linter.ts +4276 -0
- package/src/scripts/skill_overlap.ts +414 -0
- package/src/scripts/skill_preview.ts +548 -0
- package/src/scripts/skill_tools/audit_persona_coverage.ts +427 -0
- package/src/scripts/skill_tools/audit_user_type_coverage.ts +507 -0
- package/src/scripts/skill_tools/index.ts +28 -0
- package/src/scripts/skill_tools/run_block_d_eval.ts +373 -0
- package/src/scripts/skill_tools/score_skill_relevance.ts +475 -0
- package/src/scripts/skill_tools/suggest_skill_for_task.ts +288 -0
- package/src/scripts/skill_trigger_eval.ts +1046 -0
- package/src/scripts/skill_usage_collect.ts +465 -0
- package/src/scripts/skill_usage_report.ts +364 -0
- package/src/scripts/smoke/kernel.sh +4 -5
- package/src/scripts/smoke/router.sh +76 -76
- package/src/scripts/smoke/schema.sh +2 -2
- package/src/scripts/smoke/skills.sh +73 -52
- package/src/scripts/smoke_path_resolution.ts +194 -0
- package/src/scripts/smoke_quickstart.ts +224 -0
- package/src/scripts/snapshot_agent_outputs.ts +375 -0
- package/src/scripts/spotcheck_thin_root.ts +247 -0
- package/src/scripts/surface-tiers.yml +68 -0
- package/src/scripts/sync_agent_settings.ts +763 -0
- package/src/scripts/sync_github_metadata.ts +550 -0
- package/src/scripts/sync_gitignore.ts +630 -0
- package/src/scripts/sync_yaml_rt.ts +910 -0
- package/src/scripts/telegraph_stats.ts +447 -0
- package/src/scripts/tool_registry.ts +330 -0
- package/src/scripts/tools/adapter_errors.ts +93 -0
- package/src/scripts/tools/base_adapter.ts +147 -0
- package/src/scripts/tools/github_adapter.ts +229 -0
- package/src/scripts/tools/jira_adapter.ts +196 -0
- package/src/scripts/trigger_coverage.ts +251 -0
- package/src/scripts/update_counts.ts +284 -0
- package/src/scripts/update_prices.ts +219 -0
- package/src/scripts/validate_agent_settings.ts +265 -0
- package/src/scripts/validate_decision_engine.ts +366 -0
- package/src/scripts/validate_discovery_manifest.ts +160 -0
- package/src/scripts/validate_frontmatter.ts +1030 -0
- package/src/scripts/validate_pack_yaml.ts +0 -0
- package/src/scripts/validate_safe_paths.ts +164 -0
- package/src/scripts/validate_telegraph_carveouts.ts +485 -0
- package/src/scripts/verify_before_complete_hook.ts +306 -0
- package/src/scripts/verify_physical_move.ts +411 -0
- package/src/scripts/wrapper_freshness_hook.ts +179 -0
- package/dist/agent-src/commands/chat-history/learn.md +0 -184
- package/dist/agent-src/commands/chat-history/show.md +0 -113
- package/dist/agent-src/commands/fix/pr-bot-comments.md +0 -157
- package/dist/agent-src/commands/fix/pr-developer-comments.md +0 -163
- package/dist/agent-src/scripts/update_roadmap_progress.py +0 -537
- package/dist/agent-src/skills/corpus-grounding/scripts/bm25_search.py +0 -212
- package/dist/agent-src/skills/corpus-grounding/scripts/decision_engine.py +0 -438
- package/dist/agent-src/skills/corpus-grounding/scripts/ground.py +0 -166
- package/dist/agent-src/skills/corpus-grounding/scripts/schema_validator.py +0 -160
- package/dist/agent-src/skills/design-tokens/scripts/tokens.py +0 -296
- package/dist/agent-src/skills/react-shadcn-ui/scripts/shadcn_add.py +0 -299
- package/dist/agent-src/skills/tailwind-engineer/scripts/tailwind_config_gen.py +0 -463
- package/dist/agent-src/templates/agents/memory/architecture-decisions.example.yml +0 -95
- package/dist/agent-src/templates/scripts/check_memory.py +0 -283
- package/dist/agent-src/templates/scripts/check_memory_proposal.py +0 -180
- package/dist/agent-src/templates/scripts/implement_ticket/__init__.py +0 -94
- package/dist/agent-src/templates/scripts/implement_ticket/__main__.py +0 -15
- package/dist/agent-src/templates/scripts/memory_hash.py +0 -75
- package/dist/agent-src/templates/scripts/memory_lookup.py +0 -577
- package/dist/agent-src/templates/scripts/memory_report.py +0 -184
- package/dist/agent-src/templates/scripts/memory_signal.py +0 -167
- package/dist/agent-src/templates/scripts/memory_status.py +0 -257
- package/dist/agent-src/templates/scripts/pr_review_routing.py +0 -340
- package/dist/agent-src/templates/scripts/pr_risk_review.py +0 -211
- package/dist/agent-src/templates/scripts/telemetry/__init__.py +0 -42
- package/dist/agent-src/templates/scripts/telemetry/aggregator.py +0 -169
- package/dist/agent-src/templates/scripts/telemetry/boundary.py +0 -171
- package/dist/agent-src/templates/scripts/telemetry/engagement.py +0 -297
- package/dist/agent-src/templates/scripts/telemetry/report_renderer.py +0 -197
- package/dist/agent-src/templates/scripts/telemetry/settings.py +0 -177
- package/dist/agent-src/templates/scripts/telemetry_record.py +0 -179
- package/dist/agent-src/templates/scripts/telemetry_report.py +0 -161
- package/dist/agent-src/templates/scripts/telemetry_status.py +0 -142
- package/dist/agent-src/templates/scripts/tier_usage_report.py +0 -183
- package/dist/agent-src/templates/scripts/work_engine/__init__.py +0 -58
- package/dist/agent-src/templates/scripts/work_engine/__main__.py +0 -9
- package/dist/agent-src/templates/scripts/work_engine/_lib/__init__.py +0 -7
- package/dist/agent-src/templates/scripts/work_engine/_lib/agent_settings.py +0 -840
- package/dist/agent-src/templates/scripts/work_engine/_lib/user_global_paths.py +0 -249
- package/dist/agent-src/templates/scripts/work_engine/cli.py +0 -195
- package/dist/agent-src/templates/scripts/work_engine/cli_args.py +0 -116
- package/dist/agent-src/templates/scripts/work_engine/delivery_state.py +0 -137
- package/dist/agent-src/templates/scripts/work_engine/directives/__init__.py +0 -33
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/__init__.py +0 -98
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/analyze.py +0 -98
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/implement.py +0 -145
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/memory.py +0 -136
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/plan.py +0 -175
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/refine.py +0 -396
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/report.py +0 -227
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/test.py +0 -180
- package/dist/agent-src/templates/scripts/work_engine/directives/backend/verify.py +0 -170
- package/dist/agent-src/templates/scripts/work_engine/directives/mixed/__init__.py +0 -116
- package/dist/agent-src/templates/scripts/work_engine/directives/mixed/contract.py +0 -254
- package/dist/agent-src/templates/scripts/work_engine/directives/mixed/stitch.py +0 -229
- package/dist/agent-src/templates/scripts/work_engine/directives/mixed/ui.py +0 -231
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/__init__.py +0 -113
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/_passthrough.py +0 -44
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/apply.py +0 -241
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/audit.py +0 -414
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/design.py +0 -335
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/polish.py +0 -513
- package/dist/agent-src/templates/scripts/work_engine/directives/ui/review.py +0 -471
- package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/__init__.py +0 -119
- package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/_skipped.py +0 -37
- package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/apply.py +0 -165
- package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/refine.py +0 -66
- package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/report.py +0 -62
- package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/test.py +0 -115
- package/dist/agent-src/templates/scripts/work_engine/dispatcher.py +0 -331
- package/dist/agent-src/templates/scripts/work_engine/emitters.py +0 -68
- package/dist/agent-src/templates/scripts/work_engine/errors.py +0 -19
- package/dist/agent-src/templates/scripts/work_engine/hook_bootstrap.py +0 -91
- package/dist/agent-src/templates/scripts/work_engine/hooks/__init__.py +0 -54
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/__init__.py +0 -35
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/_chat_history_base.py +0 -59
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/chat_history_append.py +0 -43
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/chat_history_halt_append.py +0 -41
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/decision_gate.py +0 -162
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/decision_trace.py +0 -163
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/directive_set_guard.py +0 -53
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/halt_surface_audit.py +0 -50
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/memory_visibility.py +0 -141
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/state_shape_validation.py +0 -52
- package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/trace.py +0 -84
- package/dist/agent-src/templates/scripts/work_engine/hooks/context.py +0 -66
- package/dist/agent-src/templates/scripts/work_engine/hooks/events.py +0 -44
- package/dist/agent-src/templates/scripts/work_engine/hooks/exceptions.py +0 -79
- package/dist/agent-src/templates/scripts/work_engine/hooks/registry.py +0 -60
- package/dist/agent-src/templates/scripts/work_engine/hooks/runner.py +0 -73
- package/dist/agent-src/templates/scripts/work_engine/hooks/settings.py +0 -196
- package/dist/agent-src/templates/scripts/work_engine/input_builders.py +0 -163
- package/dist/agent-src/templates/scripts/work_engine/intent/__init__.py +0 -47
- package/dist/agent-src/templates/scripts/work_engine/intent/classify.py +0 -280
- package/dist/agent-src/templates/scripts/work_engine/migration/__init__.py +0 -8
- package/dist/agent-src/templates/scripts/work_engine/migration/v0_to_v1.py +0 -231
- package/dist/agent-src/templates/scripts/work_engine/orchestration.py +0 -193
- package/dist/agent-src/templates/scripts/work_engine/persona_policy.py +0 -85
- package/dist/agent-src/templates/scripts/work_engine/resolvers/__init__.py +0 -22
- package/dist/agent-src/templates/scripts/work_engine/resolvers/diff.py +0 -106
- package/dist/agent-src/templates/scripts/work_engine/resolvers/file.py +0 -113
- package/dist/agent-src/templates/scripts/work_engine/resolvers/prompt.py +0 -90
- package/dist/agent-src/templates/scripts/work_engine/scoring/__init__.py +0 -14
- package/dist/agent-src/templates/scripts/work_engine/scoring/confidence.py +0 -300
- package/dist/agent-src/templates/scripts/work_engine/scoring/decision_engine.py +0 -351
- package/dist/agent-src/templates/scripts/work_engine/scoring/decision_trace.py +0 -141
- package/dist/agent-src/templates/scripts/work_engine/scoring/memory_visibility.py +0 -284
- package/dist/agent-src/templates/scripts/work_engine/stack/__init__.py +0 -31
- package/dist/agent-src/templates/scripts/work_engine/stack/detect.py +0 -187
- package/dist/agent-src/templates/scripts/work_engine/stack/runner.py +0 -481
- package/dist/agent-src/templates/scripts/work_engine/state.py +0 -694
- package/dist/agent-src/templates/scripts/work_engine/state_io.py +0 -202
- package/dist/cli/python/resolvePython.js +0 -38
- package/dist/cli/python/resolvePython.js.map +0 -1
- package/docs/case-studies/frontend-design-vs-ui-ux-pro-max.md +0 -86
- package/docs/contracts/agent-memory-contract.md +0 -159
- package/docs/parity/bench-ruflo.json +0 -58
- package/docs/parity/ruflo.md +0 -46
- package/src/scripts/__pycache__/validate_frontmatter.cpython-312.pyc +0 -0
- package/src/scripts/_archive/_backfill_skill_domains.py +0 -140
- package/src/scripts/_archive/_bootstrap_tier_frontmatter.py +0 -151
- package/src/scripts/_archive/_p43_bodies.py +0 -235
- package/src/scripts/_archive/_p43_condense.py +0 -118
- package/src/scripts/_archive/_p4_migrate.py +0 -199
- package/src/scripts/_archive/_phase2_shim_helper.py +0 -109
- package/src/scripts/_archive/_pilot_council_question.py +0 -57
- package/src/scripts/_cli/__init__.py +0 -0
- package/src/scripts/_cli/cmd_doctor.py +0 -1583
- package/src/scripts/_cli/cmd_explain.py +0 -355
- package/src/scripts/_cli/cmd_export.py +0 -157
- package/src/scripts/_cli/cmd_migrate.py +0 -524
- package/src/scripts/_cli/cmd_prune.py +0 -322
- package/src/scripts/_cli/cmd_refresh.py +0 -179
- package/src/scripts/_cli/cmd_settings_check.py +0 -171
- package/src/scripts/_cli/cmd_settings_migrate.py +0 -147
- package/src/scripts/_cli/cmd_sync.py +0 -166
- package/src/scripts/_cli/cmd_uninstall.py +0 -476
- package/src/scripts/_cli/cmd_update.py +0 -279
- package/src/scripts/_cli/cmd_upgrade.py +0 -172
- package/src/scripts/_cli/cmd_validate.py +0 -177
- package/src/scripts/_cli/cmd_versions.py +0 -160
- package/src/scripts/_cli/explain_last/__init__.py +0 -122
- package/src/scripts/_cli/explain_last/assumptions.py +0 -59
- package/src/scripts/_cli/explain_last/council.py +0 -105
- package/src/scripts/_cli/explain_last/halt.py +0 -44
- package/src/scripts/_cli/explain_last/inputs.py +0 -128
- package/src/scripts/_cli/explain_last/memory.py +0 -94
- package/src/scripts/_cli/explain_last/provider.py +0 -52
- package/src/scripts/_cli/explain_last/render.py +0 -52
- package/src/scripts/_cli/explain_last/route.py +0 -59
- package/src/scripts/_cli/explain_last/scrubber.py +0 -105
- package/src/scripts/_cli/explain_last/sections/__init__.py +0 -35
- package/src/scripts/_cli/explain_last/sections/assumptions.py +0 -21
- package/src/scripts/_cli/explain_last/sections/council.py +0 -27
- package/src/scripts/_cli/explain_last/sections/halt.py +0 -31
- package/src/scripts/_cli/explain_last/sections/header.py +0 -24
- package/src/scripts/_cli/explain_last/sections/inputs.py +0 -27
- package/src/scripts/_cli/explain_last/sections/memory.py +0 -21
- package/src/scripts/_cli/explain_last/sections/pack.py +0 -16
- package/src/scripts/_cli/explain_last/sections/provider.py +0 -26
- package/src/scripts/_cli/explain_last/sections/route.py +0 -22
- package/src/scripts/_cli/explain_last/state_loader.py +0 -76
- package/src/scripts/_emit_domain_table.py +0 -35
- package/src/scripts/_lib/__init__.py +0 -5
- package/src/scripts/_lib/__pycache__/__init__.cpython-312.pyc +0 -0
- package/src/scripts/_lib/__pycache__/agent_src.cpython-312.pyc +0 -0
- package/src/scripts/_lib/agent_settings.py +0 -840
- package/src/scripts/_lib/agent_src.py +0 -491
- package/src/scripts/_lib/agents_overlay.py +0 -120
- package/src/scripts/_lib/bench_ab_cache.py +0 -162
- package/src/scripts/_lib/bench_ab_scoring.py +0 -209
- package/src/scripts/_lib/bench_cost.py +0 -138
- package/src/scripts/_lib/bench_quality.py +0 -118
- package/src/scripts/_lib/bench_report.py +0 -149
- package/src/scripts/_lib/bench_telegraph.py +0 -273
- package/src/scripts/_lib/bench_telegraph_report.py +0 -151
- package/src/scripts/_lib/changelog_eras.py +0 -330
- package/src/scripts/_lib/claude_desktop_bundler.py +0 -238
- package/src/scripts/_lib/cli_wrapper.py +0 -64
- package/src/scripts/_lib/fs_atomic.py +0 -116
- package/src/scripts/_lib/global_deploy_inventory.py +0 -282
- package/src/scripts/_lib/install_regenerator.py +0 -134
- package/src/scripts/_lib/installed_lock.py +0 -256
- package/src/scripts/_lib/installed_tools.py +0 -381
- package/src/scripts/_lib/json_pointers.py +0 -260
- package/src/scripts/_lib/linked_projects.py +0 -238
- package/src/scripts/_lib/model_tier.py +0 -52
- package/src/scripts/_lib/module_detection.py +0 -223
- package/src/scripts/_lib/pin_resolver.py +0 -152
- package/src/scripts/_lib/script_output.py +0 -144
- package/src/scripts/_lib/token_count.py +0 -95
- package/src/scripts/_lib/update_check.py +0 -207
- package/src/scripts/_lib/user_global_paths.py +0 -249
- package/src/scripts/_lib/value_ladder.py +0 -696
- package/src/scripts/_lib/value_report.py +0 -455
- package/src/scripts/_phase4_bucket.py +0 -210
- package/src/scripts/_pilot_measure.py +0 -53
- package/src/scripts/_tmp_scan_framework_leakage.py +0 -119
- package/src/scripts/adoption_report.py +0 -195
- package/src/scripts/adoption_snapshot.py +0 -219
- package/src/scripts/adoption_status.py +0 -166
- package/src/scripts/adr/regenerate_index.py +0 -79
- package/src/scripts/ai-video/lib/adapter-common.sh +0 -231
- package/src/scripts/ai-video/lib/adapter-contract.md +0 -329
- package/src/scripts/ai-video/lib/fixtures/comfyui/result.json +0 -1
- package/src/scripts/ai-video/lib/fixtures/fal/result.json +0 -1
- package/src/scripts/ai-video/lib/fixtures/gemini-veo/result.json +0 -1
- package/src/scripts/ai-video/lib/fixtures/higgsfield/result.json +0 -1
- package/src/scripts/ai-video/lib/fixtures/kling/result.json +0 -1
- package/src/scripts/ai-video/lib/fixtures/musetalk/result.json +0 -1
- package/src/scripts/ai-video/lib/fixtures/openai-images/result.json +0 -1
- package/src/scripts/ai-video/lib/fixtures/replicate/result.json +0 -1
- package/src/scripts/ai-video/lib/fixtures/sora/result.json +0 -1
- package/src/scripts/ai-video/lib/fixtures/syncso/result.json +0 -1
- package/src/scripts/ai-video/lib/load-config.sh +0 -180
- package/src/scripts/ai-video/lib/redact.sh +0 -85
- package/src/scripts/ai_council/__init__.py +0 -40
- package/src/scripts/ai_council/_default_prices.py +0 -50
- package/src/scripts/ai_council/advisors.py +0 -148
- package/src/scripts/ai_council/airgap.py +0 -165
- package/src/scripts/ai_council/budget_guard.py +0 -202
- package/src/scripts/ai_council/bundler.py +0 -263
- package/src/scripts/ai_council/cli_hints.py +0 -123
- package/src/scripts/ai_council/clients.py +0 -1385
- package/src/scripts/ai_council/compile_corpus.py +0 -179
- package/src/scripts/ai_council/confidence_gate.py +0 -156
- package/src/scripts/ai_council/config.py +0 -1364
- package/src/scripts/ai_council/consensus.py +0 -329
- package/src/scripts/ai_council/events_log.py +0 -141
- package/src/scripts/ai_council/learn_low_impact_preview.py +0 -252
- package/src/scripts/ai_council/low_impact.py +0 -714
- package/src/scripts/ai_council/low_impact_corpus.py +0 -466
- package/src/scripts/ai_council/low_impact_intake.py +0 -163
- package/src/scripts/ai_council/modes.py +0 -131
- package/src/scripts/ai_council/necessity.py +0 -782
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_2a4_acceptance.py +0 -208
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_add_quiet.py +0 -149
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_budget_v2_audit.py +0 -206
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_context_layer_v1_estimate.py +0 -67
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_context_layer_v1_review.py +0 -292
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_followups_review.py +0 -259
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_inject_quiet_flag.py +0 -33
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_measure_v2.sh +0 -36
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_measure_verbosity.sh +0 -26
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_nondestructive_inline_audit.py +0 -209
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_per_task.sh +0 -41
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_phase4_dispatch_latency.py +0 -108
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_phase6_trigger_jaccard.py +0 -92
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_phase_2a_budget_rebalance.py +0 -257
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_phase_2a_post_revert.py +0 -197
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_rebalancing_audit.py +0 -149
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_roundtrip.py +0 -111
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_rule_hardening_v1.py +0 -251
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_silent_taskfiles.py +0 -98
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_structural_open_questions.py +0 -232
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_structural_optimization.py +0 -144
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_structural_v3_gaps.py +0 -252
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_structural_v3_review.py +0 -240
- package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_tier_retrofit.py +0 -180
- package/src/scripts/ai_council/orchestrator.py +0 -1206
- package/src/scripts/ai_council/pricing.py +0 -215
- package/src/scripts/ai_council/probation_gate.py +0 -152
- package/src/scripts/ai_council/project_context.py +0 -159
- package/src/scripts/ai_council/prompts.py +0 -567
- package/src/scripts/ai_council/redact_low_impact_entry.py +0 -155
- package/src/scripts/ai_council/replay.py +0 -155
- package/src/scripts/ai_council/session.py +0 -366
- package/src/scripts/ai_council/shadow_dispatch.py +0 -235
- package/src/scripts/ai_council/solo_dispatch.py +0 -226
- package/src/scripts/annotate_discovery.py +0 -149
- package/src/scripts/apply_modules_config.py +0 -290
- package/src/scripts/audit_adr_coverage.py +0 -175
- package/src/scripts/audit_auto_rules.py +0 -175
- package/src/scripts/audit_cloud_compatibility.py +0 -362
- package/src/scripts/audit_command_surface.py +0 -669
- package/src/scripts/audit_initial_context.py +0 -237
- package/src/scripts/audit_likelihood.py +0 -148
- package/src/scripts/audit_mcp_tools.py +0 -146
- package/src/scripts/audit_overlap.py +0 -145
- package/src/scripts/audit_skill_descriptions.py +0 -180
- package/src/scripts/audit_skill_overlap.py +0 -207
- package/src/scripts/audit_user_type_axis.py +0 -140
- package/src/scripts/backfill_model_tier.py +0 -184
- package/src/scripts/bench_ab_cache_dispatch.py +0 -68
- package/src/scripts/bench_ab_clone.py +0 -170
- package/src/scripts/bench_ab_diff.py +0 -220
- package/src/scripts/bench_ab_integrity.py +0 -143
- package/src/scripts/bench_ab_run.py +0 -235
- package/src/scripts/bench_ab_task_runner.py +0 -369
- package/src/scripts/bench_ab_tracka_run.py +0 -202
- package/src/scripts/bench_baseline_ready.py +0 -108
- package/src/scripts/bench_condense_memory.py +0 -168
- package/src/scripts/bench_drift_check.py +0 -151
- package/src/scripts/bench_per_tool.py +0 -216
- package/src/scripts/bench_rtk_savings.py +0 -320
- package/src/scripts/bench_run.py +0 -272
- package/src/scripts/bench_runner.py +0 -158
- package/src/scripts/build_cloud_bundle.py +0 -458
- package/src/scripts/build_discovery_manifest.py +0 -747
- package/src/scripts/build_linear_digest.py +0 -260
- package/src/scripts/build_mcp_registry_manifest.py +0 -181
- package/src/scripts/build_rule_trigger_matrix.py +0 -350
- package/src/scripts/capture_showcase_session.py +0 -361
- package/src/scripts/chat_history.py +0 -1799
- package/src/scripts/check_always_budget.py +0 -532
- package/src/scripts/check_artefact_checksums.py +0 -111
- package/src/scripts/check_augment_description_cap.py +0 -79
- package/src/scripts/check_augmentignore.py +0 -72
- package/src/scripts/check_beta_review_markers.py +0 -127
- package/src/scripts/check_bite_sized_granularity.py +0 -99
- package/src/scripts/check_cluster_patterns.py +0 -206
- package/src/scripts/check_command_count_messaging.py +0 -152
- package/src/scripts/check_condensation.py +0 -375
- package/src/scripts/check_condensed_paths.py +0 -231
- package/src/scripts/check_context_paths.py +0 -202
- package/src/scripts/check_council_layout.py +0 -125
- package/src/scripts/check_council_references.py +0 -228
- package/src/scripts/check_discovery_determinism.py +0 -70
- package/src/scripts/check_gate_paths.py +0 -128
- package/src/scripts/check_iron_law_prominence.py +0 -145
- package/src/scripts/check_kernel_rule_bundle.py +0 -151
- package/src/scripts/check_md_language.py +0 -161
- package/src/scripts/check_memory.py +0 -443
- package/src/scripts/check_memory_proposal.py +0 -182
- package/src/scripts/check_module_management_neutral.py +0 -147
- package/src/scripts/check_no_local_settings_committed.py +0 -51
- package/src/scripts/check_no_new_legacy_path.py +0 -100
- package/src/scripts/check_no_roadmap_refs.py +0 -155
- package/src/scripts/check_one_off_location.py +0 -81
- package/src/scripts/check_overlay_cascade_subdirs.py +0 -129
- package/src/scripts/check_portability.py +0 -574
- package/src/scripts/check_proposal.py +0 -269
- package/src/scripts/check_public_catalog_links.py +0 -125
- package/src/scripts/check_public_links.py +0 -185
- package/src/scripts/check_references.py +0 -557
- package/src/scripts/check_release_includes_discovery.py +0 -61
- package/src/scripts/check_release_pr_shape.py +0 -123
- package/src/scripts/check_release_published.py +0 -145
- package/src/scripts/check_release_trunk_sync.py +0 -152
- package/src/scripts/check_reply_consistency.py +0 -169
- package/src/scripts/check_roadmap_trackable.py +0 -114
- package/src/scripts/check_role_doc_links.py +0 -110
- package/src/scripts/check_safety_floor_untouched.py +0 -125
- package/src/scripts/check_skill_requires.py +0 -147
- package/src/scripts/check_template_pin_drift.py +0 -129
- package/src/scripts/check_test_coverage_diff.py +0 -180
- package/src/scripts/check_token_optimizer_freshness.py +0 -146
- package/src/scripts/check_update_banner.py +0 -86
- package/src/scripts/ci_status.py +0 -301
- package/src/scripts/ci_summary.py +0 -131
- package/src/scripts/ci_time_ratio.py +0 -168
- package/src/scripts/command_suggester/__init__.py +0 -51
- package/src/scripts/command_suggester/cooldown.py +0 -132
- package/src/scripts/command_suggester/loader.py +0 -73
- package/src/scripts/command_suggester/match.py +0 -180
- package/src/scripts/command_suggester/rank.py +0 -120
- package/src/scripts/command_suggester/render.py +0 -86
- package/src/scripts/command_suggester/sanitize.py +0 -113
- package/src/scripts/command_suggester/settings.py +0 -127
- package/src/scripts/command_suggester/types.py +0 -78
- package/src/scripts/compile_router.py +0 -232
- package/src/scripts/condense.py +0 -1919
- package/src/scripts/condense_memory.py +0 -178
- package/src/scripts/config/__init__.py +0 -9
- package/src/scripts/config/packs.py +0 -157
- package/src/scripts/config/presets.py +0 -224
- package/src/scripts/config/profile_explain.py +0 -89
- package/src/scripts/config/profiles.py +0 -191
- package/src/scripts/config/session_profiles.py +0 -542
- package/src/scripts/context_hygiene_hook.py +0 -181
- package/src/scripts/cost_by_conversation.py +0 -78
- package/src/scripts/cost_summary.py +0 -97
- package/src/scripts/council_cli.py +0 -2557
- package/src/scripts/council_prune.py +0 -81
- package/src/scripts/cross_repo_retrieve.py +0 -172
- package/src/scripts/discovery_stats.py +0 -70
- package/src/scripts/extract_audit_patterns.py +0 -202
- package/src/scripts/first_run_gate_hook.py +0 -179
- package/src/scripts/gen_discovery_baseline.py +0 -127
- package/src/scripts/generate_catalog.py +0 -116
- package/src/scripts/generate_command_flows.py +0 -191
- package/src/scripts/generate_index.py +0 -303
- package/src/scripts/generate_ownership_matrix.py +0 -378
- package/src/scripts/generate_pack_manifests.py +0 -340
- package/src/scripts/hooks/__init__.py +0 -1
- package/src/scripts/hooks/dispatch_hook.py +0 -461
- package/src/scripts/hooks/dispatch_issues.py +0 -136
- package/src/scripts/hooks/envelope.py +0 -98
- package/src/scripts/hooks/replay_hook.py +0 -144
- package/src/scripts/hooks/state_io.py +0 -145
- package/src/scripts/hooks_doctor.py +0 -223
- package/src/scripts/hooks_status.py +0 -157
- package/src/scripts/install.py +0 -5183
- package/src/scripts/inventory_abstraction_budget.py +0 -622
- package/src/scripts/inventory_frontmatter.py +0 -164
- package/src/scripts/inventory_meta_layers.py +0 -288
- package/src/scripts/iron_law_sha.py +0 -107
- package/src/scripts/linked_projects_list.py +0 -91
- package/src/scripts/lint_agent_skill_names.py +0 -150
- package/src/scripts/lint_agents_layout.py +0 -197
- package/src/scripts/lint_agents_md.py +0 -210
- package/src/scripts/lint_archived_skills.py +0 -159
- package/src/scripts/lint_artefact_frontmatter.py +0 -188
- package/src/scripts/lint_bench_ab.py +0 -172
- package/src/scripts/lint_bench_corpus.py +0 -255
- package/src/scripts/lint_command_flow_coverage.py +0 -132
- package/src/scripts/lint_command_routing.py +0 -160
- package/src/scripts/lint_command_tiers.py +0 -175
- package/src/scripts/lint_command_verbs.py +0 -206
- package/src/scripts/lint_commit_subjects.py +0 -139
- package/src/scripts/lint_context_spine_usage.py +0 -137
- package/src/scripts/lint_discovery_manifest.py +0 -176
- package/src/scripts/lint_discovery_vocabulary.py +0 -220
- package/src/scripts/lint_examples.py +0 -102
- package/src/scripts/lint_explain_trace.py +0 -80
- package/src/scripts/lint_featured_skills.py +0 -144
- package/src/scripts/lint_flows.py +0 -215
- package/src/scripts/lint_framework_leakage.py +0 -375
- package/src/scripts/lint_frontmatter_boilerplate.py +0 -77
- package/src/scripts/lint_ghostwriter_source.py +0 -242
- package/src/scripts/lint_global_paths.py +0 -148
- package/src/scripts/lint_handoffs.py +0 -217
- package/src/scripts/lint_hook_concern_budget.py +0 -207
- package/src/scripts/lint_hook_manifest.py +0 -217
- package/src/scripts/lint_load_context.py +0 -196
- package/src/scripts/lint_marketplace.py +0 -180
- package/src/scripts/lint_marketplace_install_completeness.py +0 -198
- package/src/scripts/lint_mcp_registry_manifest.py +0 -69
- package/src/scripts/lint_media_policy_linkage.py +0 -140
- package/src/scripts/lint_model_tier_coverage.py +0 -73
- package/src/scripts/lint_namespace.py +0 -135
- package/src/scripts/lint_namespace_collisions.py +0 -103
- package/src/scripts/lint_new_skill_gate.py +0 -144
- package/src/scripts/lint_no_new_atomic_commands.py +0 -180
- package/src/scripts/lint_one_off_age.py +0 -184
- package/src/scripts/lint_orchestration_dsl.py +0 -217
- package/src/scripts/lint_orchestrator_auto_detect.py +0 -111
- package/src/scripts/lint_pack_boundaries.py +0 -147
- package/src/scripts/lint_pack_dependencies.py +0 -137
- package/src/scripts/lint_pack_first_win.py +0 -121
- package/src/scripts/lint_persona_governance.py +0 -164
- package/src/scripts/lint_positioning.py +0 -143
- package/src/scripts/lint_profile_overlay_set_only.py +0 -179
- package/src/scripts/lint_readme_jargon.py +0 -131
- package/src/scripts/lint_readme_size.py +0 -33
- package/src/scripts/lint_regression.py +0 -251
- package/src/scripts/lint_roadmap_ci_steps.py +0 -186
- package/src/scripts/lint_roadmap_complexity.py +0 -220
- package/src/scripts/lint_role_experiences.py +0 -255
- package/src/scripts/lint_rule_interactions.py +0 -170
- package/src/scripts/lint_rule_tiers.py +0 -90
- package/src/scripts/lint_showcase_sessions.py +0 -148
- package/src/scripts/lint_skill_tools.py +0 -168
- package/src/scripts/lint_topics_yaml.py +0 -89
- package/src/scripts/lint_trust_coherence.py +0 -212
- package/src/scripts/lint_value_dashboard.py +0 -218
- package/src/scripts/mcp_parity_smoke.py +0 -316
- package/src/scripts/mcp_render.py +0 -173
- package/src/scripts/mcp_server/__init__.py +0 -19
- package/src/scripts/mcp_server/__main__.py +0 -12
- package/src/scripts/mcp_server/catalog.py +0 -125
- package/src/scripts/mcp_server/metadata.py +0 -75
- package/src/scripts/mcp_server/prompts.py +0 -442
- package/src/scripts/mcp_server/requirements.txt +0 -4
- package/src/scripts/mcp_server/resources.py +0 -201
- package/src/scripts/mcp_server/server.py +0 -270
- package/src/scripts/mcp_server/telemetry.py +0 -128
- package/src/scripts/mcp_server/tools.py +0 -950
- package/src/scripts/mcp_telemetry_health.py +0 -214
- package/src/scripts/mcp_telemetry_query.py +0 -203
- package/src/scripts/mcp_telemetry_store.py +0 -211
- package/src/scripts/measure_augment_budget.py +0 -214
- package/src/scripts/measure_density.py +0 -232
- package/src/scripts/measure_frugality_savings.py +0 -164
- package/src/scripts/measure_markitdown_lift.py +0 -127
- package/src/scripts/measure_patterns.py +0 -376
- package/src/scripts/measure_projection_bytes.py +0 -159
- package/src/scripts/measure_rule_budget.py +0 -347
- package/src/scripts/measure_skill_reduction.py +0 -102
- package/src/scripts/memory_hash.py +0 -75
- package/src/scripts/memory_lookup.py +0 -705
- package/src/scripts/memory_report.py +0 -336
- package/src/scripts/memory_signal.py +0 -212
- package/src/scripts/memory_status.py +0 -257
- package/src/scripts/migrate_command_suggestions.py +0 -151
- package/src/scripts/migrate_frontmatter_defaults.py +0 -245
- package/src/scripts/mine_session.py +0 -279
- package/src/scripts/minimal_safe_diff_hook.py +0 -245
- package/src/scripts/move_artefact.py +0 -143
- package/src/scripts/new_skill.py +0 -148
- package/src/scripts/onboarding_gate_hook.py +0 -142
- package/src/scripts/pack_mcp_content.py +0 -293
- package/src/scripts/plan_physical_move.py +0 -353
- package/src/scripts/prediction-pool/poisson_sim.py +0 -167
- package/src/scripts/prediction-pool/pool_winsim.py +0 -236
- package/src/scripts/prediction-pool/score_ev.py +0 -188
- package/src/scripts/print_required_checks.py +0 -196
- package/src/scripts/probe_projection_fidelity.py +0 -202
- package/src/scripts/probe_skill_registration.py +0 -413
- package/src/scripts/profile_staleness_hook.py +0 -69
- package/src/scripts/profile_use.py +0 -164
- package/src/scripts/project_thin_rules.py +0 -168
- package/src/scripts/propose_modules_config.py +0 -145
- package/src/scripts/prototype_lint_contradictions.py +0 -150
- package/src/scripts/prove_pack_extractable.py +0 -187
- package/src/scripts/readme_linter.py +0 -589
- package/src/scripts/redact_hook_capture.py +0 -148
- package/src/scripts/refine_ticket_detect.py +0 -646
- package/src/scripts/release.py +0 -1091
- package/src/scripts/render_benchmark_md.py +0 -312
- package/src/scripts/render_value_md.py +0 -347
- package/src/scripts/requirements-evals.txt +0 -8
- package/src/scripts/roadmap_progress_hook.py +0 -274
- package/src/scripts/router_telemetry.py +0 -470
- package/src/scripts/run_skill_evals.py +0 -185
- package/src/scripts/runtime_dispatcher.py +0 -276
- package/src/scripts/runtime_handler.py +0 -148
- package/src/scripts/runtime_registry.py +0 -166
- package/src/scripts/score_skill_selection.py +0 -198
- package/src/scripts/setup_eval_venv.sh +0 -58
- package/src/scripts/skill_collision_clusters.py +0 -162
- package/src/scripts/skill_discovery.py +0 -254
- package/src/scripts/skill_linter.py +0 -3694
- package/src/scripts/skill_overlap.py +0 -204
- package/src/scripts/skill_preview.py +0 -179
- package/src/scripts/skill_tools/__init__.py +0 -22
- package/src/scripts/skill_tools/audit_persona_coverage.py +0 -147
- package/src/scripts/skill_tools/audit_user_type_coverage.py +0 -148
- package/src/scripts/skill_tools/run_block_d_eval.py +0 -129
- package/src/scripts/skill_tools/score_skill_relevance.py +0 -169
- package/src/scripts/skill_tools/suggest_skill_for_task.py +0 -113
- package/src/scripts/skill_trigger_eval.py +0 -682
- package/src/scripts/skill_usage_collect.py +0 -191
- package/src/scripts/skill_usage_report.py +0 -162
- package/src/scripts/smoke_path_resolution.py +0 -93
- package/src/scripts/smoke_quickstart.py +0 -144
- package/src/scripts/snapshot_agent_outputs.py +0 -144
- package/src/scripts/spotcheck_thin_root.py +0 -134
- package/src/scripts/sync_agent_settings.py +0 -180
- package/src/scripts/sync_github_metadata.py +0 -147
- package/src/scripts/sync_gitignore.py +0 -291
- package/src/scripts/sync_yaml_rt.py +0 -734
- package/src/scripts/telegraph_stats.py +0 -119
- package/src/scripts/tool_registry.py +0 -146
- package/src/scripts/tools/__init__.py +0 -1
- package/src/scripts/tools/adapter_errors.py +0 -63
- package/src/scripts/tools/base_adapter.py +0 -91
- package/src/scripts/tools/github_adapter.py +0 -128
- package/src/scripts/tools/jira_adapter.py +0 -115
- package/src/scripts/trigger_coverage.py +0 -129
- package/src/scripts/update_counts.py +0 -199
- package/src/scripts/update_prices.py +0 -125
- package/src/scripts/validate_agent_settings.py +0 -124
- package/src/scripts/validate_decision_engine.py +0 -136
- package/src/scripts/validate_discovery_manifest.py +0 -94
- package/src/scripts/validate_frontmatter.py +0 -647
- package/src/scripts/validate_pack_yaml.py +0 -179
- package/src/scripts/validate_safe_paths.py +0 -118
- package/src/scripts/validate_telegraph_carveouts.py +0 -129
- package/src/scripts/verify_before_complete_hook.py +0 -216
- package/src/scripts/verify_physical_move.py +0 -185
- package/src/scripts/wrapper_freshness_hook.py +0 -86
- /package/dist/agent-src/skills/design-intelligence/data/{typography.csv → font-pairings-reference.csv} +0 -0
- /package/src/scripts/{ai-video → media}/lib/fixtures/allin1/analysis.json +0 -0
- /package/src/scripts/{ai-video → media}/lib/fixtures/comfyui/scene-0001.mp4 +0 -0
- /package/src/scripts/{ai-video → media}/lib/fixtures/fal/scene-0001.mp4 +0 -0
- /package/src/scripts/{ai-video → media}/lib/fixtures/gemini-veo/scene-0001.mp4 +0 -0
- /package/src/scripts/{ai-video → media}/lib/fixtures/higgsfield/scene-0001.mp4 +0 -0
- /package/src/scripts/{ai-video → media}/lib/fixtures/kling/scene-0001.mp4 +0 -0
- /package/src/scripts/{ai-video → media}/lib/fixtures/musetalk/lipsync-0001.mp4 +0 -0
- /package/src/scripts/{ai-video → media}/lib/fixtures/openai-images/scene-0001.png +0 -0
- /package/src/scripts/{ai-video → media}/lib/fixtures/replicate/scene-0001.mp4 +0 -0
- /package/src/scripts/{ai-video → media}/lib/fixtures/sora/scene-0001.mp4 +0 -0
- /package/src/scripts/{ai-video → media}/lib/fixtures/syncso/lipsync-0001.mp4 +0 -0
- /package/src/scripts/{ai-video → media}/lib/fixtures/whisperx/transcript.json +0 -0
- /package/src/scripts/{ai-video → media}/lib/telemetry.sh +0 -0
|
@@ -0,0 +1,1697 @@
|
|
|
1
|
+
#!/usr/bin/env tsx
|
|
2
|
+
/**
|
|
3
|
+
* End-to-end release automation for `event4u/agent-config` (TypeScript twin).
|
|
4
|
+
*
|
|
5
|
+
* TypeScript twin of `src/scripts/release.py` (ADR-200, py2ts migration).
|
|
6
|
+
* The CLI contract mirrors the Python original EXACTLY — same flags, same
|
|
7
|
+
* exit codes, same stdout/stderr split, byte-identical emitted output, same
|
|
8
|
+
* subprocess argv/cwd/env. No behaviour changes — latent quirks are
|
|
9
|
+
* replicated and flagged inline, not fixed.
|
|
10
|
+
*
|
|
11
|
+
* Invoked via `task release`. The bump level (major/minor/patch) is
|
|
12
|
+
* auto-detected from Conventional Commits since the last tag; pass
|
|
13
|
+
* `--as {major,minor,patch}` to force, or `--version X.Y.Z` to pin.
|
|
14
|
+
*
|
|
15
|
+
* Pipeline:
|
|
16
|
+
* 1. Preflight — on main, clean tree, origin in sync, gh available,
|
|
17
|
+
* target tag doesn't exist yet.
|
|
18
|
+
* 2. Plan — compute new version, parse Conventional Commits
|
|
19
|
+
* since the last tag, render CHANGELOG section.
|
|
20
|
+
* 3. Confirm — show preview, ask once (skippable with --yes).
|
|
21
|
+
* 4. Branch + bump — create `release/X.Y.Z`, update package.json,
|
|
22
|
+
* .claude-plugin/marketplace.json, CHANGELOG.md,
|
|
23
|
+
* then run `task release-prepare` so pack
|
|
24
|
+
* manifests and tool projections pick up the
|
|
25
|
+
* new version (otherwise the PR's own consistency
|
|
26
|
+
* check fails — see PR #226 post-mortem).
|
|
27
|
+
* 5. Commit + push — `release: X.Y.Z`, push branch, open PR.
|
|
28
|
+
* 6. Wait for CI — `gh pr checks --watch` (skippable with --no-wait).
|
|
29
|
+
* 7. Merge — `gh pr merge --merge --delete-branch`.
|
|
30
|
+
* 8. Tag main — fast-forward main, tag the merge commit,
|
|
31
|
+
* push the tag (this triggers publish-npm.yml).
|
|
32
|
+
* 9. GitHub Release — `gh release create X.Y.Z --notes <changelog>`.
|
|
33
|
+
*
|
|
34
|
+
* Idempotency: pass `--resume` to recover from a partial failure. Each
|
|
35
|
+
* step then probes existing state (branch, commit, PR, tag, GitHub
|
|
36
|
+
* Release) and skips work that is already done, instead of erroring out.
|
|
37
|
+
* Without `--resume` the pipeline still mutates git/network state, so
|
|
38
|
+
* re-running on a dirty tree needs `--resume` (or a manual cleanup).
|
|
39
|
+
* Each step prints what it's about to do before doing it, so a crash
|
|
40
|
+
* leaves a recoverable trail.
|
|
41
|
+
*
|
|
42
|
+
* Stdlib-only (Python 3.10+). No third-party runtime dependencies.
|
|
43
|
+
*
|
|
44
|
+
* See also:
|
|
45
|
+
* - docs/contracts/release-pr-gating.md — release-PR shape, cut surface,
|
|
46
|
+
* kept surface, fail-closed contract.
|
|
47
|
+
* - docs/contracts/branch-protection-policy.md — per-PR-shape
|
|
48
|
+
* required-check matrix; `task ci:required-checks` previews it.
|
|
49
|
+
* - docs/contracts/ci-cost-budget.md — measured baselines + quarterly
|
|
50
|
+
* review cadence.
|
|
51
|
+
* - .github/workflows/release-validation.yml — the tight release-PR
|
|
52
|
+
* validation jobs (release-shape, changelog-entry, version-consistency).
|
|
53
|
+
*
|
|
54
|
+
* --- Parity notes (ADR-200) ---
|
|
55
|
+
*
|
|
56
|
+
* - `die(msg, code=2)` mirrors Python's `print(error) + sys.exit(code)`: it
|
|
57
|
+
* prints `error: {msg}` to stderr then throws a `SystemExitError(code)`
|
|
58
|
+
* sentinel caught at the CLI entry guard, which sets `process.exitCode`. We
|
|
59
|
+
* never call `process.exit()` (per the migration contract).
|
|
60
|
+
* - argparse usage errors (unknown flag, bad `--as` choice) throw
|
|
61
|
+
* `ArgparseExit(2)`; `-h`/`--help` throws `ArgparseExit(0)` after printing
|
|
62
|
+
* usage. The argparse `--help` BODY (per-flag descriptions) is a documented
|
|
63
|
+
* divergence — argparse re-wraps it to the live terminal width; the tests
|
|
64
|
+
* assert the `usage:` token + exit code, not the body prose.
|
|
65
|
+
* - JSON byte-parity: `set_package_version` uses `json.dumps(data, indent=4)`,
|
|
66
|
+
* `set_marketplace_version` uses `json.dumps(data, indent=2)`, both `+ "\n"`.
|
|
67
|
+
* release.py does NOT pass `ensure_ascii`, so CPython defaults to True
|
|
68
|
+
* (ASCII-escaped non-ASCII). The reused `jsonDumpsIndent` helper (mirrored
|
|
69
|
+
* from install.ts) is the `ensure_ascii=False` variant. The package's
|
|
70
|
+
* package.json / marketplace.json are pure-ASCII, so the two are
|
|
71
|
+
* byte-identical in practice; if a non-ASCII string ever lands in those
|
|
72
|
+
* files this would diverge (release.py would `\uXXXX`-escape, this twin
|
|
73
|
+
* would emit the raw codepoint). Documented, not fixed.
|
|
74
|
+
* - subprocess → `spawnSync` with identical argv/cwd/env. `run()` mirrors
|
|
75
|
+
* `subprocess.run(check, cwd=cwd or REPO_ROOT, text=True, capture_output)`.
|
|
76
|
+
* On `check && capture && non-zero` → `die("command failed (...)")`. On
|
|
77
|
+
* `check && !capture && non-zero` → throw `CalledProcessError` (NOT caught
|
|
78
|
+
* by the entry guard, so it propagates — matches Python letting it raise).
|
|
79
|
+
* `_count_tests_current` catches FileNotFoundError / TimeoutExpired →
|
|
80
|
+
* modelled via `res.error` (ENOENT) and the `timeout` option.
|
|
81
|
+
* - `time.sleep(5)` in `watch_pr_checks` ports faithfully (blocking). It is
|
|
82
|
+
* never reached on any test path (tests never call execute()).
|
|
83
|
+
* - `_date.today().isoformat()` → a `YYYY-MM-DD` local-date helper.
|
|
84
|
+
* - `_lib.changelog_eras` imports resolve to the `.ts` twin, never a `.py`.
|
|
85
|
+
*/
|
|
86
|
+
|
|
87
|
+
import { spawnSync } from 'node:child_process';
|
|
88
|
+
import * as fs from 'node:fs';
|
|
89
|
+
import * as path from 'node:path';
|
|
90
|
+
import process from 'node:process';
|
|
91
|
+
import { fileURLToPath, pathToFileURL } from 'node:url';
|
|
92
|
+
|
|
93
|
+
import type {
|
|
94
|
+
SplitPlan} from './_lib/changelog_eras.js';
|
|
95
|
+
import {
|
|
96
|
+
CURRENT_ERA_BODY_CAP,
|
|
97
|
+
current_era_body_size,
|
|
98
|
+
current_era_insertion_point,
|
|
99
|
+
perform_split,
|
|
100
|
+
plan_split,
|
|
101
|
+
} from './_lib/changelog_eras.js';
|
|
102
|
+
|
|
103
|
+
// `__doc__.splitlines()[0]` in `_parse_args` — the argparse description. Kept
|
|
104
|
+
// as a referenceable constant so the first docstring line is preserved exactly.
|
|
105
|
+
const MODULE_DOC_FIRST_LINE = 'End-to-end release automation for `event4u/agent-config`.';
|
|
106
|
+
|
|
107
|
+
const _HERE = fileURLToPath(import.meta.url);
|
|
108
|
+
|
|
109
|
+
// ---------------------------------------------------------------------------
|
|
110
|
+
// Python-runtime parity helpers
|
|
111
|
+
// ---------------------------------------------------------------------------
|
|
112
|
+
|
|
113
|
+
/** Mirror of Python `sys.exit(code)` raised by `die()`. Caught at the CLI entry. */
|
|
114
|
+
class SystemExitError extends Error {
|
|
115
|
+
constructor(public readonly code: number) {
|
|
116
|
+
super(`system-exit-${code}`);
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
/** argparse usage-error / help exit (code 2 / 0). */
|
|
121
|
+
class ArgparseExit extends Error {
|
|
122
|
+
constructor(public readonly code: number) {
|
|
123
|
+
super(`argparse-exit-${code}`);
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
/**
|
|
128
|
+
* Mirror of `subprocess.CalledProcessError`. Thrown by `run()` when a command
|
|
129
|
+
* fails with `check=True` and output is NOT captured — the Python original
|
|
130
|
+
* lets `CalledProcessError` propagate in that path. The CLI entry guard does
|
|
131
|
+
* NOT catch this, so it surfaces (non-zero exit + traceback), matching Python.
|
|
132
|
+
*/
|
|
133
|
+
class CalledProcessError extends Error {
|
|
134
|
+
constructor(
|
|
135
|
+
public readonly returncode: number,
|
|
136
|
+
public readonly cmd: readonly string[],
|
|
137
|
+
) {
|
|
138
|
+
super(`Command '${cmd.join(' ')}' returned non-zero exit status ${returncode}.`);
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
// ---------------------------------------------------------------------------
|
|
143
|
+
// Module-level constants (release.py:70-84)
|
|
144
|
+
// ---------------------------------------------------------------------------
|
|
145
|
+
|
|
146
|
+
// REPO_ROOT: release.py is at src/scripts/release.py;
|
|
147
|
+
// Path(__file__).resolve().parent.parent.parent == src/scripts → src → repo
|
|
148
|
+
// root. release.ts lives in the same dir, so _HERE-dir is src/scripts and
|
|
149
|
+
// `..`/`..` reaches the repo root (matches changelog_eras.ts's 3-up from
|
|
150
|
+
// _lib/ and install.ts's REPO_ROOT computation).
|
|
151
|
+
const REPO_ROOT = path.resolve(path.dirname(_HERE), '..', '..');
|
|
152
|
+
const PACKAGE_JSON = path.join(REPO_ROOT, 'package.json');
|
|
153
|
+
const MARKETPLACE_JSON = path.join(REPO_ROOT, '.claude-plugin', 'marketplace.json');
|
|
154
|
+
const CHANGELOG = path.join(REPO_ROOT, 'CHANGELOG.md');
|
|
155
|
+
const MAIN_BRANCH = 'main';
|
|
156
|
+
const REMOTE = 'origin';
|
|
157
|
+
const REPO_SLUG = 'event4u-app/agent-config';
|
|
158
|
+
|
|
159
|
+
// GitHub rejects bodies over these limits with a GraphQL "Body is too
|
|
160
|
+
// long" error. The full entry always lands in CHANGELOG.md (committed in
|
|
161
|
+
// the PR diff and attached to the tag), so an oversized body is capped
|
|
162
|
+
// with a pointer rather than failing the release — a major bump can
|
|
163
|
+
// render hundreds of commit bullets, well past the 65 536 PR-body limit.
|
|
164
|
+
const GH_PR_BODY_LIMIT = 65_536; // createPullRequest mutation hard limit
|
|
165
|
+
const GH_RELEASE_NOTES_LIMIT = 125_000; // release-notes body limit
|
|
166
|
+
|
|
167
|
+
// ---------------------------------------------------------------------------
|
|
168
|
+
// Parity helpers — code-point length, comma grouping, regex escape, JSON
|
|
169
|
+
// ---------------------------------------------------------------------------
|
|
170
|
+
|
|
171
|
+
/** Python `len(str)` — number of Unicode code points (not UTF-16 units). */
|
|
172
|
+
function pyLen(s: string): number {
|
|
173
|
+
return [...s].length;
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
/** Python `text[:n]` on a string — first `n` code points. */
|
|
177
|
+
function pySlice(s: string, n: number): string {
|
|
178
|
+
return [...s].slice(0, n).join('');
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
/** Python f-string `{n:,}` — thousands grouping with commas (en-US grouping). */
|
|
182
|
+
function commaGroup(n: number): string {
|
|
183
|
+
const neg = n < 0;
|
|
184
|
+
const digits = Math.abs(n).toString();
|
|
185
|
+
let out = '';
|
|
186
|
+
for (let i = 0; i < digits.length; i += 1) {
|
|
187
|
+
if (i > 0 && (digits.length - i) % 3 === 0) out += ',';
|
|
188
|
+
out += digits[i];
|
|
189
|
+
}
|
|
190
|
+
return neg ? '-' + out : out;
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
/** Python `re.escape` — escape regex-special chars in a literal string. */
|
|
194
|
+
function reEscape(s: string): string {
|
|
195
|
+
return s.replace(/[.*+?^${}()|[\]\\#\-]/g, '\\$&');
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
// --- JSON byte-parity (mirrors install.ts; see Parity notes for ensure_ascii). ---
|
|
199
|
+
|
|
200
|
+
function _jsonStrNoAscii(s: string): string {
|
|
201
|
+
// json.dumps(ensure_ascii=False): escape control chars + " + \, keep >=0x20.
|
|
202
|
+
let out = '"';
|
|
203
|
+
for (const ch of s) {
|
|
204
|
+
const code = ch.codePointAt(0) as number;
|
|
205
|
+
switch (ch) {
|
|
206
|
+
case '"':
|
|
207
|
+
out += '\\"';
|
|
208
|
+
break;
|
|
209
|
+
case '\\':
|
|
210
|
+
out += '\\\\';
|
|
211
|
+
break;
|
|
212
|
+
case '\n':
|
|
213
|
+
out += '\\n';
|
|
214
|
+
break;
|
|
215
|
+
case '\r':
|
|
216
|
+
out += '\\r';
|
|
217
|
+
break;
|
|
218
|
+
case '\t':
|
|
219
|
+
out += '\\t';
|
|
220
|
+
break;
|
|
221
|
+
case '\b':
|
|
222
|
+
out += '\\b';
|
|
223
|
+
break;
|
|
224
|
+
case '\f':
|
|
225
|
+
out += '\\f';
|
|
226
|
+
break;
|
|
227
|
+
default:
|
|
228
|
+
if (code < 0x20) {
|
|
229
|
+
out += '\\u' + code.toString(16).padStart(4, '0');
|
|
230
|
+
} else {
|
|
231
|
+
out += ch;
|
|
232
|
+
}
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
return out + '"';
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
function _jsonScalar(value: unknown): string | null {
|
|
239
|
+
if (value === null || value === undefined) return 'null';
|
|
240
|
+
if (typeof value === 'boolean') return value ? 'true' : 'false';
|
|
241
|
+
if (typeof value === 'number') {
|
|
242
|
+
if (!Number.isFinite(value)) {
|
|
243
|
+
if (Number.isNaN(value)) return 'NaN';
|
|
244
|
+
return value > 0 ? 'Infinity' : '-Infinity';
|
|
245
|
+
}
|
|
246
|
+
return String(value);
|
|
247
|
+
}
|
|
248
|
+
if (typeof value === 'string') return _jsonStrNoAscii(value);
|
|
249
|
+
return null;
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
function _dumpIndent(value: unknown, indent: number, depth: number): string {
|
|
253
|
+
const scalar = _jsonScalar(value);
|
|
254
|
+
if (scalar !== null) return scalar;
|
|
255
|
+
const pad = ' '.repeat(indent * (depth + 1));
|
|
256
|
+
const closePad = ' '.repeat(indent * depth);
|
|
257
|
+
if (Array.isArray(value)) {
|
|
258
|
+
if (value.length === 0) return '[]';
|
|
259
|
+
const items = value.map((v) => pad + _dumpIndent(v, indent, depth + 1));
|
|
260
|
+
return `[\n${items.join(',\n')}\n${closePad}]`;
|
|
261
|
+
}
|
|
262
|
+
if (typeof value === 'object' && value !== null) {
|
|
263
|
+
const obj = value as Record<string, unknown>;
|
|
264
|
+
const keys = Object.keys(obj);
|
|
265
|
+
if (keys.length === 0) return '{}';
|
|
266
|
+
const items = keys.map(
|
|
267
|
+
(k) => `${pad}${_jsonStrNoAscii(k)}: ${_dumpIndent(obj[k], indent, depth + 1)}`,
|
|
268
|
+
);
|
|
269
|
+
return `{\n${items.join(',\n')}\n${closePad}}`;
|
|
270
|
+
}
|
|
271
|
+
return _jsonStrNoAscii(String(value));
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
/** `json.dumps(data, indent=N)` (sort_keys=False; ensure_ascii — see Parity notes). */
|
|
275
|
+
function jsonDumpsIndent(value: unknown, indent: number): string {
|
|
276
|
+
return _dumpIndent(value, indent, 0);
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
/** `_date.today().isoformat()` — local date as `YYYY-MM-DD`. */
|
|
280
|
+
function todayIso(now?: Date): string {
|
|
281
|
+
const d = now ?? new Date();
|
|
282
|
+
const pad = (n: number) => String(n).padStart(2, '0');
|
|
283
|
+
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}`;
|
|
284
|
+
}
|
|
285
|
+
|
|
286
|
+
// ---------------------------------------------------------------------------
|
|
287
|
+
// _cap_body (release.py:87-101)
|
|
288
|
+
// ---------------------------------------------------------------------------
|
|
289
|
+
|
|
290
|
+
/**
|
|
291
|
+
* Return `text` unchanged when within `limit` chars; otherwise truncate at
|
|
292
|
+
* the last line boundary that fits and append a pointer to `full_ref` so
|
|
293
|
+
* nothing is silently lost.
|
|
294
|
+
*/
|
|
295
|
+
function _cap_body(text: string, limit: number, full_ref: string): string {
|
|
296
|
+
if (pyLen(text) <= limit) {
|
|
297
|
+
return text;
|
|
298
|
+
}
|
|
299
|
+
const notice =
|
|
300
|
+
`\n\n> _Changelog truncated to fit GitHub's ` +
|
|
301
|
+
`${commaGroup(limit)}-character body limit — full entry in ${full_ref}._`;
|
|
302
|
+
let head = pySlice(text, limit - pyLen(notice));
|
|
303
|
+
const nl = head.lastIndexOf('\n');
|
|
304
|
+
if (nl > 0) {
|
|
305
|
+
head = head.slice(0, nl);
|
|
306
|
+
}
|
|
307
|
+
return head + notice;
|
|
308
|
+
}
|
|
309
|
+
|
|
310
|
+
// ---------------------------------------------------------------------------
|
|
311
|
+
// SECTIONS + commit regexes (release.py:103-120)
|
|
312
|
+
// ---------------------------------------------------------------------------
|
|
313
|
+
|
|
314
|
+
// Conventional Commit types and how they map into CHANGELOG sections.
|
|
315
|
+
// Order in this tuple determines order in the rendered entry.
|
|
316
|
+
const SECTIONS: ReadonlyArray<readonly [string, string | null, readonly string[]]> = [
|
|
317
|
+
['Features', 'minor', ['feat']],
|
|
318
|
+
['Bug Fixes', 'patch', ['fix']],
|
|
319
|
+
['Performance', 'patch', ['perf']],
|
|
320
|
+
['Reverts', 'patch', ['revert']],
|
|
321
|
+
['Documentation', null, ['docs']],
|
|
322
|
+
['Refactoring', null, ['refactor']],
|
|
323
|
+
['Tests', null, ['test']],
|
|
324
|
+
['Build', null, ['build']],
|
|
325
|
+
['CI', null, ['ci']],
|
|
326
|
+
['Chores', null, ['chore']],
|
|
327
|
+
];
|
|
328
|
+
|
|
329
|
+
const BREAKING_RE = /^([a-z]+)(\([^)]+\))?!:/;
|
|
330
|
+
const CONVENTIONAL_RE = /^(?<type>[a-z]+)(?:\((?<scope>[^)]+)\))?(?<bang>!)?: (?<subject>.+)$/;
|
|
331
|
+
|
|
332
|
+
// ─── dataclasses ──────────────────────────────────────────────────────────────
|
|
333
|
+
|
|
334
|
+
/** `@dataclass(frozen=True) class Commit`. */
|
|
335
|
+
class Commit {
|
|
336
|
+
constructor(
|
|
337
|
+
public readonly sha: string,
|
|
338
|
+
public readonly type: string,
|
|
339
|
+
public readonly scope: string | null,
|
|
340
|
+
public readonly subject: string,
|
|
341
|
+
public readonly breaking: boolean,
|
|
342
|
+
) {}
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
/** `@dataclass(frozen=True) class Plan`. */
|
|
346
|
+
class Plan {
|
|
347
|
+
constructor(
|
|
348
|
+
public readonly current: string,
|
|
349
|
+
public readonly target: string,
|
|
350
|
+
public readonly bump: string, // "major" | "minor" | "patch"
|
|
351
|
+
public readonly commits: readonly Commit[],
|
|
352
|
+
public readonly last_tag: string | null,
|
|
353
|
+
public readonly changelog_body: string, // rendered body (without the heading)
|
|
354
|
+
public readonly changelog_entry: string, // full entry including heading, for CHANGELOG.md
|
|
355
|
+
// Populated only when the release crosses an era boundary AND the
|
|
356
|
+
// current era body has grown past CURRENT_ERA_BODY_CAP. null for
|
|
357
|
+
// patch releases and for minor/major bumps where the era still fits.
|
|
358
|
+
public readonly split_plan: SplitPlan | null = null,
|
|
359
|
+
) {}
|
|
360
|
+
}
|
|
361
|
+
|
|
362
|
+
// ─── utilities ────────────────────────────────────────────────────────────────
|
|
363
|
+
|
|
364
|
+
function die(msg: string, code = 2): never {
|
|
365
|
+
process.stderr.write(`error: ${msg}\n`);
|
|
366
|
+
throw new SystemExitError(code);
|
|
367
|
+
}
|
|
368
|
+
|
|
369
|
+
interface RunResult {
|
|
370
|
+
returncode: number;
|
|
371
|
+
stdout: string;
|
|
372
|
+
stderr: string;
|
|
373
|
+
}
|
|
374
|
+
|
|
375
|
+
/**
|
|
376
|
+
* Thin subprocess wrapper with sane defaults.
|
|
377
|
+
*
|
|
378
|
+
* When `check` and `capture` are both True and the command fails, Python's
|
|
379
|
+
* default behaviour swallows stderr — callers only see a CalledProcessError
|
|
380
|
+
* with no hint of what went wrong. We catch that path and die with the actual
|
|
381
|
+
* stderr so release preflight failures are diagnosable without re-running with
|
|
382
|
+
* a debugger.
|
|
383
|
+
*/
|
|
384
|
+
function run(
|
|
385
|
+
args: readonly string[],
|
|
386
|
+
opts: { check?: boolean; capture?: boolean; cwd?: string | null } = {},
|
|
387
|
+
): RunResult {
|
|
388
|
+
const check = opts.check ?? true;
|
|
389
|
+
const capture = opts.capture ?? false;
|
|
390
|
+
const cwd = opts.cwd ?? REPO_ROOT;
|
|
391
|
+
|
|
392
|
+
const [cmd, ...rest] = args;
|
|
393
|
+
const res = spawnSync(cmd as string, rest, {
|
|
394
|
+
cwd,
|
|
395
|
+
encoding: 'utf-8',
|
|
396
|
+
// capture_output=True → pipe; else inherit so child writes straight to
|
|
397
|
+
// this process's stdout/stderr (text mode, matching subprocess text=True).
|
|
398
|
+
stdio: capture ? ['ignore', 'pipe', 'pipe'] : ['inherit', 'inherit', 'inherit'],
|
|
399
|
+
});
|
|
400
|
+
|
|
401
|
+
if (res.error) {
|
|
402
|
+
// FileNotFoundError analogue (ENOENT) and other spawn failures — Python
|
|
403
|
+
// would raise here; only the explicit catchers (have / _count_tests_current)
|
|
404
|
+
// handle it. Surface as a thrown error.
|
|
405
|
+
throw res.error;
|
|
406
|
+
}
|
|
407
|
+
|
|
408
|
+
const returncode = res.status ?? 0;
|
|
409
|
+
const stdout = capture ? res.stdout ?? '' : '';
|
|
410
|
+
const stderr = capture ? res.stderr ?? '' : '';
|
|
411
|
+
|
|
412
|
+
if (check && returncode !== 0) {
|
|
413
|
+
if (capture) {
|
|
414
|
+
const cmdStr = args.join(' ');
|
|
415
|
+
const out = (stderr || stdout || '').trim();
|
|
416
|
+
die(`command failed (${returncode}): ${cmdStr}\n${out}`);
|
|
417
|
+
}
|
|
418
|
+
// check && !capture → Python re-raises CalledProcessError; replicate.
|
|
419
|
+
throw new CalledProcessError(returncode, args);
|
|
420
|
+
}
|
|
421
|
+
|
|
422
|
+
return { returncode, stdout, stderr };
|
|
423
|
+
}
|
|
424
|
+
|
|
425
|
+
function git(args: readonly string[], opts: { capture?: boolean } = {}): string {
|
|
426
|
+
const capture = opts.capture ?? false;
|
|
427
|
+
const r = run(['git', ...args], { capture });
|
|
428
|
+
return capture ? r.stdout.trim() : '';
|
|
429
|
+
}
|
|
430
|
+
|
|
431
|
+
/**
|
|
432
|
+
* Watch PR checks and tolerate the 'no checks' case.
|
|
433
|
+
*
|
|
434
|
+
* `gh pr checks --watch` exits 1 both on real failures and when no checks are
|
|
435
|
+
* reported at all (no workflow triggered, no required checks configured in
|
|
436
|
+
* branch protection). The latter must not block the release — we warn and
|
|
437
|
+
* continue. Real failures still die.
|
|
438
|
+
*
|
|
439
|
+
* A short grace period gives GitHub time to register workflow runs on a
|
|
440
|
+
* freshly-pushed branch.
|
|
441
|
+
*/
|
|
442
|
+
function watch_pr_checks(): void {
|
|
443
|
+
// time.sleep(5) — blocking grace period. Never reached on any test path.
|
|
444
|
+
const until = Date.now() + 5000;
|
|
445
|
+
while (Date.now() < until) {
|
|
446
|
+
// busy-wait stand-in for time.sleep(5) without an event-loop yield.
|
|
447
|
+
}
|
|
448
|
+
const proc = spawnSync('gh', ['pr', 'checks', '--watch'], {
|
|
449
|
+
cwd: REPO_ROOT,
|
|
450
|
+
encoding: 'utf-8',
|
|
451
|
+
stdio: ['ignore', 'pipe', 'pipe'],
|
|
452
|
+
});
|
|
453
|
+
if (proc.error) {
|
|
454
|
+
throw proc.error;
|
|
455
|
+
}
|
|
456
|
+
const output = ((proc.stdout || '') + (proc.stderr || '')).trim();
|
|
457
|
+
const returncode = proc.status ?? 0;
|
|
458
|
+
if (returncode === 0) {
|
|
459
|
+
if (output) {
|
|
460
|
+
process.stdout.write(output + '\n');
|
|
461
|
+
}
|
|
462
|
+
return;
|
|
463
|
+
}
|
|
464
|
+
if (output.toLowerCase().includes('no checks reported')) {
|
|
465
|
+
process.stdout.write(`⚠️ ${output}\n`);
|
|
466
|
+
process.stdout.write(
|
|
467
|
+
' Continuing without check validation — configure required ' +
|
|
468
|
+
'checks in branch protection to enforce this gate.\n',
|
|
469
|
+
);
|
|
470
|
+
return;
|
|
471
|
+
}
|
|
472
|
+
if (output) {
|
|
473
|
+
process.stderr.write(output + '\n');
|
|
474
|
+
}
|
|
475
|
+
die(`PR checks failed (exit ${returncode})`);
|
|
476
|
+
}
|
|
477
|
+
|
|
478
|
+
function have(bin: string): boolean {
|
|
479
|
+
const res = spawnSync('which', [bin], { encoding: 'utf-8', stdio: ['ignore', 'pipe', 'pipe'] });
|
|
480
|
+
if (res.error) {
|
|
481
|
+
// FileNotFoundError on `which` itself → treat as not found (returncode != 0).
|
|
482
|
+
return false;
|
|
483
|
+
}
|
|
484
|
+
return (res.status ?? 1) === 0;
|
|
485
|
+
}
|
|
486
|
+
|
|
487
|
+
// ─── resume-mode state probes ────────────────────────────────────────────────
|
|
488
|
+
|
|
489
|
+
function _branch_exists_local(branch: string): boolean {
|
|
490
|
+
const r = run(['git', 'rev-parse', '--verify', '--quiet', `refs/heads/${branch}`], {
|
|
491
|
+
check: false,
|
|
492
|
+
capture: true,
|
|
493
|
+
});
|
|
494
|
+
return r.returncode === 0;
|
|
495
|
+
}
|
|
496
|
+
|
|
497
|
+
function _branch_exists_remote(branch: string): boolean {
|
|
498
|
+
const r = run(['git', 'ls-remote', '--exit-code', '--heads', REMOTE, branch], {
|
|
499
|
+
check: false,
|
|
500
|
+
capture: true,
|
|
501
|
+
});
|
|
502
|
+
return r.returncode === 0;
|
|
503
|
+
}
|
|
504
|
+
|
|
505
|
+
function _tag_exists_local(tag: string): boolean {
|
|
506
|
+
return git(['tag', '-l', tag], { capture: true }).split('\n').includes(tag);
|
|
507
|
+
}
|
|
508
|
+
|
|
509
|
+
function _tag_exists_remote(tag: string): boolean {
|
|
510
|
+
const r = run(['git', 'ls-remote', '--exit-code', '--tags', REMOTE, tag], {
|
|
511
|
+
check: false,
|
|
512
|
+
capture: true,
|
|
513
|
+
});
|
|
514
|
+
return r.returncode === 0;
|
|
515
|
+
}
|
|
516
|
+
|
|
517
|
+
/** Most recent PR (any state) with `release/X.Y.Z` as head, or null. */
|
|
518
|
+
function _pr_for_branch(branch: string): Record<string, unknown> | null {
|
|
519
|
+
const r = run(
|
|
520
|
+
[
|
|
521
|
+
'gh',
|
|
522
|
+
'pr',
|
|
523
|
+
'list',
|
|
524
|
+
'--head',
|
|
525
|
+
branch,
|
|
526
|
+
'--state',
|
|
527
|
+
'all',
|
|
528
|
+
'--json',
|
|
529
|
+
'number,state,url',
|
|
530
|
+
'--limit',
|
|
531
|
+
'1',
|
|
532
|
+
],
|
|
533
|
+
{ check: false, capture: true },
|
|
534
|
+
);
|
|
535
|
+
if (r.returncode !== 0) {
|
|
536
|
+
return null;
|
|
537
|
+
}
|
|
538
|
+
let items: unknown;
|
|
539
|
+
try {
|
|
540
|
+
items = JSON.parse(r.stdout || '[]');
|
|
541
|
+
} catch {
|
|
542
|
+
return null;
|
|
543
|
+
}
|
|
544
|
+
return Array.isArray(items) && items.length > 0 ? (items[0] as Record<string, unknown>) : null;
|
|
545
|
+
}
|
|
546
|
+
|
|
547
|
+
function _release_exists(tag: string): boolean {
|
|
548
|
+
const r = run(['gh', 'release', 'view', tag], { check: false, capture: true });
|
|
549
|
+
return r.returncode === 0;
|
|
550
|
+
}
|
|
551
|
+
|
|
552
|
+
// ─── version math ─────────────────────────────────────────────────────────────
|
|
553
|
+
|
|
554
|
+
const SEMVER_RE = /^(\d+)\.(\d+)\.(\d+)$/;
|
|
555
|
+
|
|
556
|
+
function parse_version(s: string): [number, number, number] {
|
|
557
|
+
const m = SEMVER_RE.exec(s.trim());
|
|
558
|
+
if (!m) {
|
|
559
|
+
// {s!r} → Python repr of a str (single-quoted).
|
|
560
|
+
die(`not a bare semver (X.Y.Z): '${s}'`);
|
|
561
|
+
}
|
|
562
|
+
return [
|
|
563
|
+
Number.parseInt(m[1] as string, 10),
|
|
564
|
+
Number.parseInt(m[2] as string, 10),
|
|
565
|
+
Number.parseInt(m[3] as string, 10),
|
|
566
|
+
];
|
|
567
|
+
}
|
|
568
|
+
|
|
569
|
+
function bump_version(current: string, kind: string): string {
|
|
570
|
+
const [major, minor, patch] = parse_version(current);
|
|
571
|
+
if (kind === 'major') {
|
|
572
|
+
return `${major + 1}.0.0`;
|
|
573
|
+
}
|
|
574
|
+
if (kind === 'minor') {
|
|
575
|
+
return `${major}.${minor + 1}.0`;
|
|
576
|
+
}
|
|
577
|
+
if (kind === 'patch') {
|
|
578
|
+
return `${major}.${minor}.${patch + 1}`;
|
|
579
|
+
}
|
|
580
|
+
die(`unknown bump kind: ${kind}`);
|
|
581
|
+
return ''; // unreachable
|
|
582
|
+
}
|
|
583
|
+
|
|
584
|
+
// ─── commit parsing + changelog rendering ────────────────────────────────────
|
|
585
|
+
|
|
586
|
+
/** Return non-merge commits after `tag` (or all of history if tag is null). */
|
|
587
|
+
function commits_since(tag: string | null): Commit[] {
|
|
588
|
+
const rev = tag ? `${tag}..HEAD` : 'HEAD';
|
|
589
|
+
const raw = git(['log', rev, '--no-merges', '--format=%H%x1f%s'], { capture: true });
|
|
590
|
+
const out: Commit[] = [];
|
|
591
|
+
for (const line of raw.split('\n')) {
|
|
592
|
+
if (!line.includes('\x1f')) {
|
|
593
|
+
continue;
|
|
594
|
+
}
|
|
595
|
+
// str.split("\x1f", 1) — maxsplit=1: head + remainder.
|
|
596
|
+
const idx = line.indexOf('\x1f');
|
|
597
|
+
const sha = line.slice(0, idx);
|
|
598
|
+
const subject = line.slice(idx + 1);
|
|
599
|
+
const m = CONVENTIONAL_RE.exec(subject);
|
|
600
|
+
if (!m) {
|
|
601
|
+
out.push(new Commit(sha, 'other', null, subject, false));
|
|
602
|
+
continue;
|
|
603
|
+
}
|
|
604
|
+
const breaking = Boolean(m.groups!['bang']) || subject.includes('BREAKING CHANGE');
|
|
605
|
+
out.push(
|
|
606
|
+
new Commit(
|
|
607
|
+
sha,
|
|
608
|
+
m.groups!['type'] as string,
|
|
609
|
+
(m.groups!['scope'] ?? null) as string | null,
|
|
610
|
+
m.groups!['subject'] as string,
|
|
611
|
+
breaking,
|
|
612
|
+
),
|
|
613
|
+
);
|
|
614
|
+
}
|
|
615
|
+
return out;
|
|
616
|
+
}
|
|
617
|
+
|
|
618
|
+
/** Derive the semver bump from commit types (for preview only). */
|
|
619
|
+
function infer_bump(commits: readonly Commit[]): string {
|
|
620
|
+
if (commits.some((c) => c.breaking)) {
|
|
621
|
+
return 'major';
|
|
622
|
+
}
|
|
623
|
+
for (const [, level, types] of SECTIONS) {
|
|
624
|
+
if (level === 'minor' && commits.some((c) => types.includes(c.type))) {
|
|
625
|
+
return 'minor';
|
|
626
|
+
}
|
|
627
|
+
}
|
|
628
|
+
return 'patch';
|
|
629
|
+
}
|
|
630
|
+
|
|
631
|
+
function latest_tag(): string | null {
|
|
632
|
+
const r = run(
|
|
633
|
+
[
|
|
634
|
+
'git',
|
|
635
|
+
'describe',
|
|
636
|
+
'--tags',
|
|
637
|
+
'--abbrev=0',
|
|
638
|
+
'--match',
|
|
639
|
+
'[0-9]*.[0-9]*.[0-9]*',
|
|
640
|
+
],
|
|
641
|
+
{ check: false, capture: true },
|
|
642
|
+
);
|
|
643
|
+
const tag = r.stdout.trim();
|
|
644
|
+
return tag || null;
|
|
645
|
+
}
|
|
646
|
+
|
|
647
|
+
/**
|
|
648
|
+
* Return [heading-aware full entry, body-only for GitHub Release notes].
|
|
649
|
+
*
|
|
650
|
+
* `test_trend_line` — optional pre-computed `Tests: N (+M …)` footer
|
|
651
|
+
* (road-to-feedback-followups P3.2). Computed by the caller so tests don't
|
|
652
|
+
* trigger a recursive pytest collection.
|
|
653
|
+
*/
|
|
654
|
+
function render_changelog_entry(
|
|
655
|
+
version: string,
|
|
656
|
+
prev: string | null,
|
|
657
|
+
commits: readonly Commit[],
|
|
658
|
+
today: string,
|
|
659
|
+
opts: { test_trend_line?: string | null } = {},
|
|
660
|
+
): [string, string] {
|
|
661
|
+
const test_trend_line = opts.test_trend_line ?? null;
|
|
662
|
+
let heading: string;
|
|
663
|
+
if (prev) {
|
|
664
|
+
heading =
|
|
665
|
+
`## [${version}](https://github.com/${REPO_SLUG}/compare/` +
|
|
666
|
+
`${prev}...${version}) (${today})`;
|
|
667
|
+
} else {
|
|
668
|
+
heading = `## ${version} (${today})`;
|
|
669
|
+
}
|
|
670
|
+
|
|
671
|
+
// Group by section; commits of unknown type drop into "Other".
|
|
672
|
+
const grouped: Record<string, Commit[]> = {};
|
|
673
|
+
for (const [label] of SECTIONS) {
|
|
674
|
+
grouped[label] = [];
|
|
675
|
+
}
|
|
676
|
+
grouped['BREAKING CHANGES'] = [];
|
|
677
|
+
const other: Commit[] = [];
|
|
678
|
+
for (const c of commits) {
|
|
679
|
+
if (c.breaking) {
|
|
680
|
+
grouped['BREAKING CHANGES']!.push(c);
|
|
681
|
+
continue;
|
|
682
|
+
}
|
|
683
|
+
let placed = false;
|
|
684
|
+
for (const [label, , types] of SECTIONS) {
|
|
685
|
+
if (types.includes(c.type)) {
|
|
686
|
+
grouped[label]!.push(c);
|
|
687
|
+
placed = true;
|
|
688
|
+
break;
|
|
689
|
+
}
|
|
690
|
+
}
|
|
691
|
+
if (!placed) {
|
|
692
|
+
other.push(c);
|
|
693
|
+
}
|
|
694
|
+
}
|
|
695
|
+
|
|
696
|
+
const body_lines: string[] = [];
|
|
697
|
+
const ordered_labels = ['BREAKING CHANGES', ...SECTIONS.map(([label]) => label)];
|
|
698
|
+
for (const label of ordered_labels) {
|
|
699
|
+
const bucket = grouped[label] ?? [];
|
|
700
|
+
if (bucket.length === 0) {
|
|
701
|
+
continue;
|
|
702
|
+
}
|
|
703
|
+
body_lines.push('');
|
|
704
|
+
body_lines.push(`### ${label}`);
|
|
705
|
+
body_lines.push('');
|
|
706
|
+
for (const c of bucket) {
|
|
707
|
+
body_lines.push(_changelog_line(c));
|
|
708
|
+
}
|
|
709
|
+
}
|
|
710
|
+
if (other.length > 0) {
|
|
711
|
+
body_lines.push('');
|
|
712
|
+
body_lines.push('### Other');
|
|
713
|
+
body_lines.push('');
|
|
714
|
+
for (const c of other) {
|
|
715
|
+
body_lines.push(_changelog_line(c));
|
|
716
|
+
}
|
|
717
|
+
}
|
|
718
|
+
|
|
719
|
+
// Test-count trend footer (road-to-feedback-followups P3.2). Silent on
|
|
720
|
+
// errors — never a release blocker.
|
|
721
|
+
if (test_trend_line) {
|
|
722
|
+
body_lines.push('');
|
|
723
|
+
body_lines.push(test_trend_line);
|
|
724
|
+
}
|
|
725
|
+
|
|
726
|
+
// "\n".join(...).lstrip("\n") — strip only leading newlines.
|
|
727
|
+
const body = body_lines.join('\n').replace(/^\n+/u, '');
|
|
728
|
+
const full = heading + '\n\n' + body + '\n';
|
|
729
|
+
return [full, body];
|
|
730
|
+
}
|
|
731
|
+
|
|
732
|
+
function _changelog_line(c: Commit): string {
|
|
733
|
+
const scope = c.scope ? `**${c.scope}:** ` : '';
|
|
734
|
+
const short = c.sha.slice(0, 7);
|
|
735
|
+
const link = `https://github.com/${REPO_SLUG}/commit/${c.sha}`;
|
|
736
|
+
return `* ${scope}${c.subject} ([${short}](${link}))`;
|
|
737
|
+
}
|
|
738
|
+
|
|
739
|
+
// ─── test-count trend (road-to-feedback-followups P3.2) ───────────────────────
|
|
740
|
+
|
|
741
|
+
const _TEST_COUNT_LINE_RE = /^Tests:\s+(\d+)/m;
|
|
742
|
+
const _PYTEST_COLLECTED_RE = /^(\d+)\s+tests?\s+collected/m;
|
|
743
|
+
|
|
744
|
+
/**
|
|
745
|
+
* Return the count from `pytest --collect-only -q` on the current tree.
|
|
746
|
+
* Returns null when pytest isn't available or collection fails — the trend
|
|
747
|
+
* line is informational, never a release blocker.
|
|
748
|
+
*/
|
|
749
|
+
function _count_tests_current(): number | null {
|
|
750
|
+
// Release-time pytest-collection probe: python-test-tooling, not package
|
|
751
|
+
// runtime — there is no tsx equivalent of pytest collection. Degrades to
|
|
752
|
+
// null (informational trend line dropped) when python3/pytest is absent.
|
|
753
|
+
const res = spawnSync('python3', ['-m', 'pytest', '--collect-only', '-q'], {
|
|
754
|
+
cwd: REPO_ROOT,
|
|
755
|
+
encoding: 'utf-8',
|
|
756
|
+
stdio: ['ignore', 'pipe', 'pipe'],
|
|
757
|
+
timeout: 120_000,
|
|
758
|
+
});
|
|
759
|
+
if (res.error) {
|
|
760
|
+
// FileNotFoundError (ENOENT) or TimeoutExpired (ETIMEDOUT) → None.
|
|
761
|
+
return null;
|
|
762
|
+
}
|
|
763
|
+
if ((res.status ?? 1) !== 0) {
|
|
764
|
+
return null;
|
|
765
|
+
}
|
|
766
|
+
const match = _PYTEST_COLLECTED_RE.exec(res.stdout ?? '');
|
|
767
|
+
return match ? Number.parseInt(match[1] as string, 10) : null;
|
|
768
|
+
}
|
|
769
|
+
|
|
770
|
+
/**
|
|
771
|
+
* Read CHANGELOG.md and return the most recent `Tests: N` footer under the
|
|
772
|
+
* `prev_tag` heading, or null when not found.
|
|
773
|
+
*/
|
|
774
|
+
function _previous_test_count_from_changelog(prev_tag: string | null): number | null {
|
|
775
|
+
if (!prev_tag || !fs.existsSync(CHANGELOG)) {
|
|
776
|
+
return null;
|
|
777
|
+
}
|
|
778
|
+
const text = fs.readFileSync(CHANGELOG, 'utf-8');
|
|
779
|
+
const heading_re = new RegExp(`^##\\s+\\[?${reEscape(prev_tag)}\\b`, 'm');
|
|
780
|
+
const m = heading_re.exec(text);
|
|
781
|
+
if (!m) {
|
|
782
|
+
return null;
|
|
783
|
+
}
|
|
784
|
+
const headEnd = m.index + m[0].length;
|
|
785
|
+
const rest = text.slice(headEnd);
|
|
786
|
+
const next_re = /^##\s+\[?\d+\.\d+\.\d+/m;
|
|
787
|
+
const next_heading = next_re.exec(rest);
|
|
788
|
+
const sectionEnd = headEnd + (next_heading ? next_heading.index : rest.length);
|
|
789
|
+
const section = text.slice(headEnd, sectionEnd);
|
|
790
|
+
const count_match = _TEST_COUNT_LINE_RE.exec(section);
|
|
791
|
+
return count_match ? Number.parseInt(count_match[1] as string, 10) : null;
|
|
792
|
+
}
|
|
793
|
+
|
|
794
|
+
/**
|
|
795
|
+
* Return the `Tests: N (+M since X.Y.Z)` footer line, or null when the current
|
|
796
|
+
* count cannot be determined. Silent on collection errors.
|
|
797
|
+
*/
|
|
798
|
+
function _render_test_trend_line(prev_tag: string | null): string | null {
|
|
799
|
+
const current = _count_tests_current();
|
|
800
|
+
if (current === null) {
|
|
801
|
+
return null;
|
|
802
|
+
}
|
|
803
|
+
const previous = _previous_test_count_from_changelog(prev_tag);
|
|
804
|
+
if (previous === null || !prev_tag) {
|
|
805
|
+
return `Tests: ${current}`;
|
|
806
|
+
}
|
|
807
|
+
const delta = current - previous;
|
|
808
|
+
const sign = delta >= 0 ? '+' : '';
|
|
809
|
+
return `Tests: ${current} (${sign}${delta} since ${prev_tag})`;
|
|
810
|
+
}
|
|
811
|
+
|
|
812
|
+
/**
|
|
813
|
+
* Insert `entry` inside the current era block.
|
|
814
|
+
*
|
|
815
|
+
* Strategy delegates to `current_era_insertion_point` so a fresh era (no
|
|
816
|
+
* version headings yet, just the intro blockquote) places the new entry after
|
|
817
|
+
* the intro instead of appended at end-of-file. When no current era header
|
|
818
|
+
* exists, falls back to the legacy "above the most recent ## [" heuristic for
|
|
819
|
+
* safety.
|
|
820
|
+
*/
|
|
821
|
+
function prepend_changelog(p: string, entry: string): void {
|
|
822
|
+
const text = fs.readFileSync(p, 'utf-8');
|
|
823
|
+
const lines = _splitlines(text);
|
|
824
|
+
const insert_at = current_era_insertion_point(lines);
|
|
825
|
+
if (insert_at !== null) {
|
|
826
|
+
const before = lines.slice(0, insert_at).join('\n');
|
|
827
|
+
const after = lines.slice(insert_at).join('\n');
|
|
828
|
+
const head = before + (before ? '\n' : '');
|
|
829
|
+
fs.writeFileSync(p, head + entry + '\n' + after + '\n', 'utf-8');
|
|
830
|
+
return;
|
|
831
|
+
}
|
|
832
|
+
|
|
833
|
+
// Legacy fallback — no era header present at all.
|
|
834
|
+
const marker_re = /^## \[?\d+\.\d+\.\d+/m;
|
|
835
|
+
const m = marker_re.exec(text);
|
|
836
|
+
if (!m) {
|
|
837
|
+
fs.writeFileSync(p, _rstrip(text) + '\n\n' + entry, 'utf-8');
|
|
838
|
+
return;
|
|
839
|
+
}
|
|
840
|
+
const before = text.slice(0, m.index);
|
|
841
|
+
const after = text.slice(m.index);
|
|
842
|
+
fs.writeFileSync(p, before + entry + '\n' + after, 'utf-8');
|
|
843
|
+
}
|
|
844
|
+
|
|
845
|
+
/** Mirror of Python `str.splitlines()` (no trailing empty for a final newline). */
|
|
846
|
+
function _splitlines(text: string): string[] {
|
|
847
|
+
if (text === '') {
|
|
848
|
+
return [];
|
|
849
|
+
}
|
|
850
|
+
const parts = text.split(/\r\n|\r|\n/);
|
|
851
|
+
if (parts.length > 0 && parts[parts.length - 1] === '') {
|
|
852
|
+
parts.pop();
|
|
853
|
+
}
|
|
854
|
+
return parts;
|
|
855
|
+
}
|
|
856
|
+
|
|
857
|
+
/** Mirror of Python `str.rstrip()` (trailing whitespace). */
|
|
858
|
+
function _rstrip(text: string): string {
|
|
859
|
+
return text.replace(/\s+$/u, '');
|
|
860
|
+
}
|
|
861
|
+
|
|
862
|
+
// ─── file mutations ───────────────────────────────────────────────────────────
|
|
863
|
+
|
|
864
|
+
/** Update the top-level `version` field; preserve 4-space indentation. */
|
|
865
|
+
function set_package_version(p: string, version: string): void {
|
|
866
|
+
const data = JSON.parse(fs.readFileSync(p, 'utf-8')) as Record<string, unknown>;
|
|
867
|
+
data['version'] = version;
|
|
868
|
+
fs.writeFileSync(p, jsonDumpsIndent(data, 4) + '\n', 'utf-8');
|
|
869
|
+
}
|
|
870
|
+
|
|
871
|
+
/** Update `metadata.version`; preserve 2-space indentation + UTF-8. */
|
|
872
|
+
function set_marketplace_version(p: string, version: string): void {
|
|
873
|
+
const data = JSON.parse(fs.readFileSync(p, 'utf-8')) as Record<string, unknown>;
|
|
874
|
+
// data.setdefault("metadata", {})["version"] = version — preserve key order.
|
|
875
|
+
if (!(typeof data['metadata'] === 'object' && data['metadata'] !== null && !Array.isArray(data['metadata']))) {
|
|
876
|
+
data['metadata'] = {};
|
|
877
|
+
}
|
|
878
|
+
(data['metadata'] as Record<string, unknown>)['version'] = version;
|
|
879
|
+
fs.writeFileSync(p, jsonDumpsIndent(data, 2) + '\n', 'utf-8');
|
|
880
|
+
}
|
|
881
|
+
|
|
882
|
+
// ─── preflight ────────────────────────────────────────────────────────────────
|
|
883
|
+
|
|
884
|
+
/**
|
|
885
|
+
* Fail fast on conditions that would break the release mid-flight.
|
|
886
|
+
*
|
|
887
|
+
* In `--resume` mode two invariants are relaxed:
|
|
888
|
+
*
|
|
889
|
+
* - The starting branch may be `release/{target}` in addition to `main` —
|
|
890
|
+
* both are valid resume positions (mid-pipeline crash after step 1 leaves
|
|
891
|
+
* you on the release branch).
|
|
892
|
+
* - The target-tag-exists check is dropped — execute() probes for existing
|
|
893
|
+
* tags/releases and skips them.
|
|
894
|
+
*
|
|
895
|
+
* Tree cleanliness, gh auth, and `main` in-sync with origin are still
|
|
896
|
+
* enforced, so resuming has the same starting posture as a fresh run; only
|
|
897
|
+
* step-level outcomes differ.
|
|
898
|
+
*/
|
|
899
|
+
function preflight(target: string, opts: { resume?: boolean } = {}): void {
|
|
900
|
+
const resume = opts.resume ?? false;
|
|
901
|
+
for (const b of ['git', 'gh']) {
|
|
902
|
+
if (!have(b)) {
|
|
903
|
+
die(`'${b}' not found on PATH`);
|
|
904
|
+
}
|
|
905
|
+
}
|
|
906
|
+
|
|
907
|
+
// Probe the active token directly via an authenticated API call. `gh auth
|
|
908
|
+
// status` returns non-zero if *any* account in the keyring is broken, even
|
|
909
|
+
// when the active one is fine — so we'd rather ask "does the token the
|
|
910
|
+
// release will actually use work?" than parse multi-account status output.
|
|
911
|
+
const r = run(['gh', 'api', 'user', '--jq', '.login'], { check: false, capture: true });
|
|
912
|
+
if (r.returncode !== 0) {
|
|
913
|
+
die('gh is not authenticated; run `gh auth login` first');
|
|
914
|
+
}
|
|
915
|
+
|
|
916
|
+
const branch = git(['rev-parse', '--abbrev-ref', 'HEAD'], { capture: true });
|
|
917
|
+
const release_branch = `release/${target}`;
|
|
918
|
+
const allowed = resume ? new Set([MAIN_BRANCH, release_branch]) : new Set([MAIN_BRANCH]);
|
|
919
|
+
if (!allowed.has(branch)) {
|
|
920
|
+
if (resume) {
|
|
921
|
+
die(
|
|
922
|
+
`resume must run from '${MAIN_BRANCH}' or '${release_branch}', ` +
|
|
923
|
+
`currently on '${branch}'`,
|
|
924
|
+
);
|
|
925
|
+
}
|
|
926
|
+
die(`release must run from '${MAIN_BRANCH}', currently on '${branch}'`);
|
|
927
|
+
}
|
|
928
|
+
|
|
929
|
+
const porcelain = git(['status', '--porcelain'], { capture: true });
|
|
930
|
+
if (porcelain) {
|
|
931
|
+
die('working tree is not clean; commit or stash first');
|
|
932
|
+
}
|
|
933
|
+
|
|
934
|
+
// --force lets the remote's tag positions win over stale local tags.
|
|
935
|
+
// The release consumes the remote view as source of truth, and we're
|
|
936
|
+
// about to create a new tag anyway — local drift (e.g. from renamed
|
|
937
|
+
// release-please tags) should not block the fetch.
|
|
938
|
+
run(['git', 'fetch', REMOTE, '--tags', '--prune', '--force'], { capture: true });
|
|
939
|
+
|
|
940
|
+
// The local-in-sync-with-origin check only applies to main; if we're
|
|
941
|
+
// already on the release branch in resume mode, the relevant invariant
|
|
942
|
+
// is "main hasn't moved beyond what release/X.Y.Z branched off", which
|
|
943
|
+
// `git pull --ff-only` enforces in step 8 anyway.
|
|
944
|
+
if (branch === MAIN_BRANCH) {
|
|
945
|
+
const local = git(['rev-parse', 'HEAD'], { capture: true });
|
|
946
|
+
const remote = git(['rev-parse', `${REMOTE}/${MAIN_BRANCH}`], { capture: true });
|
|
947
|
+
if (local !== remote) {
|
|
948
|
+
die(
|
|
949
|
+
`local ${MAIN_BRANCH} is not in sync with ` +
|
|
950
|
+
`${REMOTE}/${MAIN_BRANCH}; pull or push first`,
|
|
951
|
+
);
|
|
952
|
+
}
|
|
953
|
+
}
|
|
954
|
+
|
|
955
|
+
if (!resume) {
|
|
956
|
+
const tags = git(['tag', '-l', target], { capture: true }).split('\n');
|
|
957
|
+
if (tags.includes(target)) {
|
|
958
|
+
die(`tag '${target}' already exists; nothing to release`);
|
|
959
|
+
}
|
|
960
|
+
}
|
|
961
|
+
}
|
|
962
|
+
|
|
963
|
+
// ─── plan ─────────────────────────────────────────────────────────────────────
|
|
964
|
+
|
|
965
|
+
function print_preview(plan: Plan): void {
|
|
966
|
+
process.stdout.write('\n');
|
|
967
|
+
process.stdout.write('═'.repeat(72) + '\n');
|
|
968
|
+
process.stdout.write(` Release preview — ${plan.current} → ${plan.target} (${plan.bump})\n`);
|
|
969
|
+
process.stdout.write('═'.repeat(72) + '\n');
|
|
970
|
+
process.stdout.write('\n');
|
|
971
|
+
process.stdout.write(`Previous tag: ${plan.last_tag || '(none)'}\n`);
|
|
972
|
+
process.stdout.write(`New tag: ${plan.target}\n`);
|
|
973
|
+
process.stdout.write(`Commits: ${plan.commits.length} since ${plan.last_tag || 'start'}\n`);
|
|
974
|
+
const detected = plan.commits.length > 0 ? infer_bump(plan.commits) : 'patch';
|
|
975
|
+
if (detected !== plan.bump) {
|
|
976
|
+
process.stdout.write(
|
|
977
|
+
`NOTE: commits suggest a '${detected}' bump, ` +
|
|
978
|
+
`you picked '${plan.bump}'\n`,
|
|
979
|
+
);
|
|
980
|
+
}
|
|
981
|
+
process.stdout.write('\n');
|
|
982
|
+
process.stdout.write('Files to change:\n');
|
|
983
|
+
process.stdout.write(` · ${path.relative(REPO_ROOT, PACKAGE_JSON)}\n`);
|
|
984
|
+
process.stdout.write(` · ${path.relative(REPO_ROOT, MARKETPLACE_JSON)}\n`);
|
|
985
|
+
process.stdout.write(` · ${path.relative(REPO_ROOT, CHANGELOG)}\n`);
|
|
986
|
+
process.stdout.write(' · regenerated derived files via `task release-prepare`\n');
|
|
987
|
+
process.stdout.write(
|
|
988
|
+
' (src/packs/*/pack.yaml + README.md, dist/agent-src/, tool projections)\n',
|
|
989
|
+
);
|
|
990
|
+
if (plan.split_plan !== null) {
|
|
991
|
+
const sp = plan.split_plan;
|
|
992
|
+
process.stdout.write('\n');
|
|
993
|
+
process.stdout.write('Era split (separate commit, before release commit):\n');
|
|
994
|
+
process.stdout.write(` · archive → ${path.relative(REPO_ROOT, sp.archive_path)}\n`);
|
|
995
|
+
process.stdout.write(` · old era → pre-${sp.boundary} (archived pointer)\n`);
|
|
996
|
+
process.stdout.write(` · new era → ${sp.new_era_label} — current (empty body)\n`);
|
|
997
|
+
process.stdout.write(` · subject → ${sp.commit_subject}\n`);
|
|
998
|
+
}
|
|
999
|
+
process.stdout.write('\n');
|
|
1000
|
+
process.stdout.write('Changelog section:\n');
|
|
1001
|
+
process.stdout.write('─'.repeat(72) + '\n');
|
|
1002
|
+
process.stdout.write(_rstrip(plan.changelog_entry) + '\n');
|
|
1003
|
+
process.stdout.write('─'.repeat(72) + '\n');
|
|
1004
|
+
process.stdout.write('\n');
|
|
1005
|
+
process.stdout.write('Release-PR CI shape (docs/contracts/release-pr-gating.md):\n');
|
|
1006
|
+
process.stdout.write(
|
|
1007
|
+
' will run: Consistency · Smoke Contracts · Migration Dry-Run · ' +
|
|
1008
|
+
'Release Validation · Release Guard (post-tag, ~30 s)\n',
|
|
1009
|
+
);
|
|
1010
|
+
process.stdout.write(
|
|
1011
|
+
' will skip: Tests (install / aux / python / node / windows-lockfile-export) · ' +
|
|
1012
|
+
'Public Install Smoke — heavy install matrices cannot be regressed by a release-shape diff\n',
|
|
1013
|
+
);
|
|
1014
|
+
process.stdout.write('\n');
|
|
1015
|
+
}
|
|
1016
|
+
|
|
1017
|
+
function confirm(prompt: string): boolean {
|
|
1018
|
+
const ans = _input(`${prompt} [y/N] `).trim().toLowerCase();
|
|
1019
|
+
return ans === 'y' || ans === 'yes';
|
|
1020
|
+
}
|
|
1021
|
+
|
|
1022
|
+
/** Can we prompt at all — is fd 0 a TTY, or is a controlling terminal openable? */
|
|
1023
|
+
function _canPrompt(): boolean {
|
|
1024
|
+
if (process.env.CI) return false; // CI is non-interactive by contract → require --yes
|
|
1025
|
+
if (process.stdin.isTTY) return true;
|
|
1026
|
+
try {
|
|
1027
|
+
const fd = fs.openSync('/dev/tty', 'r');
|
|
1028
|
+
fs.closeSync(fd);
|
|
1029
|
+
return true;
|
|
1030
|
+
} catch {
|
|
1031
|
+
return false;
|
|
1032
|
+
}
|
|
1033
|
+
}
|
|
1034
|
+
|
|
1035
|
+
/**
|
|
1036
|
+
* Mirror of Python `input(prompt)` — write prompt, read one line.
|
|
1037
|
+
*
|
|
1038
|
+
* Reads from the controlling terminal (`/dev/tty`) when fd 0 is not itself a
|
|
1039
|
+
* TTY. `task release` / `./scripts-run` spawn this script with stdin detached
|
|
1040
|
+
* from the terminal, so a naive `readSync(0)` hits EOF immediately and the
|
|
1041
|
+
* `[y/N]` prompt "auto-aborts" without ever waiting. Falling back to `/dev/tty`
|
|
1042
|
+
* keeps the prompt interactive regardless of how the script is invoked.
|
|
1043
|
+
*/
|
|
1044
|
+
function _input(prompt: string): string {
|
|
1045
|
+
process.stdout.write(prompt);
|
|
1046
|
+
let fd = 0;
|
|
1047
|
+
let openedTty = false;
|
|
1048
|
+
if (!process.stdin.isTTY) {
|
|
1049
|
+
try {
|
|
1050
|
+
fd = fs.openSync('/dev/tty', 'r');
|
|
1051
|
+
openedTty = true;
|
|
1052
|
+
} catch {
|
|
1053
|
+
return ''; // no controlling terminal (true non-interactive); guarded by _canPrompt upstream.
|
|
1054
|
+
}
|
|
1055
|
+
}
|
|
1056
|
+
try {
|
|
1057
|
+
const buf = Buffer.alloc(1);
|
|
1058
|
+
const chars: number[] = [];
|
|
1059
|
+
while (true) {
|
|
1060
|
+
let bytesRead: number;
|
|
1061
|
+
try {
|
|
1062
|
+
bytesRead = fs.readSync(fd, buf, 0, 1, null);
|
|
1063
|
+
} catch {
|
|
1064
|
+
break; // EOF / error → EOFError analogue; return what we have.
|
|
1065
|
+
}
|
|
1066
|
+
if (bytesRead === 0) break;
|
|
1067
|
+
const b = buf[0] as number;
|
|
1068
|
+
if (b === 0x0a) break; // newline terminates the line (stripped, like input()).
|
|
1069
|
+
chars.push(b);
|
|
1070
|
+
}
|
|
1071
|
+
return Buffer.from(chars).toString('utf-8');
|
|
1072
|
+
} finally {
|
|
1073
|
+
if (openedTty) fs.closeSync(fd);
|
|
1074
|
+
}
|
|
1075
|
+
}
|
|
1076
|
+
|
|
1077
|
+
export interface ConfirmVerdict {
|
|
1078
|
+
proceed: boolean;
|
|
1079
|
+
/** Message to surface when not proceeding (no trailing newline). */
|
|
1080
|
+
message?: string;
|
|
1081
|
+
stream?: 'stdout' | 'stderr';
|
|
1082
|
+
}
|
|
1083
|
+
|
|
1084
|
+
/**
|
|
1085
|
+
* Resolve the pre-execute confirmation as a pure verdict (no I/O, so it is
|
|
1086
|
+
* unit-testable). `--yes` proceeds unprompted; otherwise a terminal must be
|
|
1087
|
+
* available — when none is (CI, detached stdin) we surface actionable `--yes`
|
|
1088
|
+
* guidance instead of silently aborting; with a terminal, the user must answer
|
|
1089
|
+
* `y`. The caller performs the I/O + the `return 1`.
|
|
1090
|
+
*/
|
|
1091
|
+
export function confirmGate(target: string, yes: boolean): ConfirmVerdict {
|
|
1092
|
+
if (yes) return { proceed: true };
|
|
1093
|
+
if (!_canPrompt()) {
|
|
1094
|
+
return {
|
|
1095
|
+
proceed: false,
|
|
1096
|
+
stream: 'stderr',
|
|
1097
|
+
message:
|
|
1098
|
+
'No terminal available for the [y/N] confirmation (non-interactive shell). ' +
|
|
1099
|
+
'Re-run with --yes to confirm, e.g. `task release -- --yes`.',
|
|
1100
|
+
};
|
|
1101
|
+
}
|
|
1102
|
+
if (!confirm(`Proceed with release ${target}?`)) {
|
|
1103
|
+
return { proceed: false, stream: 'stdout', message: 'aborted.' };
|
|
1104
|
+
}
|
|
1105
|
+
return { proceed: true };
|
|
1106
|
+
}
|
|
1107
|
+
|
|
1108
|
+
// ─── orchestration ────────────────────────────────────────────────────────────
|
|
1109
|
+
|
|
1110
|
+
function _step(n: number, total: number, msg: string): void {
|
|
1111
|
+
process.stdout.write(`[${n}/${total}] ${msg}\n`);
|
|
1112
|
+
}
|
|
1113
|
+
|
|
1114
|
+
function execute(
|
|
1115
|
+
plan: Plan,
|
|
1116
|
+
opts: { wait_for_checks: boolean; dry_run: boolean; resume?: boolean },
|
|
1117
|
+
): void {
|
|
1118
|
+
const wait_for_checks = opts.wait_for_checks;
|
|
1119
|
+
const dry_run = opts.dry_run;
|
|
1120
|
+
const resume = opts.resume ?? false;
|
|
1121
|
+
|
|
1122
|
+
const branch = `release/${plan.target}`;
|
|
1123
|
+
const total = 10;
|
|
1124
|
+
|
|
1125
|
+
if (dry_run) {
|
|
1126
|
+
process.stdout.write('(dry-run) no git/gh mutations will be performed.\n');
|
|
1127
|
+
return;
|
|
1128
|
+
}
|
|
1129
|
+
|
|
1130
|
+
// Probe the world once at the top so each step skip-decision is cheap.
|
|
1131
|
+
const pr_info = resume ? _pr_for_branch(branch) : null;
|
|
1132
|
+
const pr_state = pr_info ? pr_info['state'] : undefined;
|
|
1133
|
+
const pr_merged = pr_state === 'MERGED';
|
|
1134
|
+
|
|
1135
|
+
// ─── 1. branch ──────────────────────────────────────────────────────────
|
|
1136
|
+
if (pr_merged) {
|
|
1137
|
+
_step(1, total, `PR for ${branch} already merged — staying on ${MAIN_BRANCH}`);
|
|
1138
|
+
if (git(['rev-parse', '--abbrev-ref', 'HEAD'], { capture: true }) !== MAIN_BRANCH) {
|
|
1139
|
+
run(['git', 'checkout', MAIN_BRANCH]);
|
|
1140
|
+
}
|
|
1141
|
+
run(['git', 'pull', '--ff-only', REMOTE, MAIN_BRANCH]);
|
|
1142
|
+
} else if (resume && _branch_exists_local(branch)) {
|
|
1143
|
+
_step(1, total, `Branch ${branch} exists locally — checkout`);
|
|
1144
|
+
run(['git', 'checkout', branch]);
|
|
1145
|
+
} else if (resume && _branch_exists_remote(branch)) {
|
|
1146
|
+
_step(1, total, `Branch ${branch} exists on ${REMOTE} — fetch + checkout`);
|
|
1147
|
+
run(['git', 'fetch', REMOTE, branch]);
|
|
1148
|
+
run(['git', 'checkout', '-b', branch, `${REMOTE}/${branch}`]);
|
|
1149
|
+
} else {
|
|
1150
|
+
_step(1, total, `Create branch ${branch}`);
|
|
1151
|
+
run(['git', 'checkout', '-b', branch]);
|
|
1152
|
+
}
|
|
1153
|
+
|
|
1154
|
+
// ─── 1b. era split (optional, separate commit) ─────────────────────────
|
|
1155
|
+
// Lands as `chore(changelog): split era ...` BEFORE the release commit
|
|
1156
|
+
// so the split is reviewable on its own and the release commit only
|
|
1157
|
+
// touches the bump + new entry. Idempotent: archive already on disk
|
|
1158
|
+
// OR a prior split commit on the branch is treated as already done.
|
|
1159
|
+
if (plan.split_plan !== null && !pr_merged) {
|
|
1160
|
+
const sp = plan.split_plan;
|
|
1161
|
+
const split_already_committed = git(['log', `${MAIN_BRANCH}..HEAD`, '--format=%s'], {
|
|
1162
|
+
capture: true,
|
|
1163
|
+
})
|
|
1164
|
+
.split('\n')
|
|
1165
|
+
.includes(sp.commit_subject);
|
|
1166
|
+
if (fs.existsSync(sp.archive_path) && split_already_committed) {
|
|
1167
|
+
_step(1, total, `Era split for pre-${sp.boundary} already committed — skip`);
|
|
1168
|
+
} else if (fs.existsSync(sp.archive_path) && !split_already_committed) {
|
|
1169
|
+
die(
|
|
1170
|
+
`era archive ${path.relative(REPO_ROOT, sp.archive_path)} exists ` +
|
|
1171
|
+
'but no matching split commit found on this branch — inspect ' +
|
|
1172
|
+
'manually before resuming',
|
|
1173
|
+
);
|
|
1174
|
+
} else {
|
|
1175
|
+
_step(
|
|
1176
|
+
1,
|
|
1177
|
+
total,
|
|
1178
|
+
`Split era ${sp.old_era_label} → pre-${sp.boundary} ` +
|
|
1179
|
+
`(new era ${sp.new_era_label})`,
|
|
1180
|
+
);
|
|
1181
|
+
perform_split(sp);
|
|
1182
|
+
run(['git', 'add', '-A']);
|
|
1183
|
+
run(['git', 'commit', '-m', sp.commit_subject]);
|
|
1184
|
+
}
|
|
1185
|
+
}
|
|
1186
|
+
|
|
1187
|
+
// ─── 2. file mutations ──────────────────────────────────────────────────
|
|
1188
|
+
if (pr_merged) {
|
|
1189
|
+
_step(2, total, 'PR already merged — skip file bumps');
|
|
1190
|
+
} else {
|
|
1191
|
+
const current_pkg = (
|
|
1192
|
+
JSON.parse(fs.readFileSync(PACKAGE_JSON, 'utf-8')) as Record<string, unknown>
|
|
1193
|
+
)['version'];
|
|
1194
|
+
if (resume && current_pkg === plan.target) {
|
|
1195
|
+
_step(2, total, `Files already at ${plan.target} — skip bump`);
|
|
1196
|
+
} else {
|
|
1197
|
+
_step(2, total, 'Bump package.json + marketplace.json, prepend CHANGELOG');
|
|
1198
|
+
set_package_version(PACKAGE_JSON, plan.target);
|
|
1199
|
+
set_marketplace_version(MARKETPLACE_JSON, plan.target);
|
|
1200
|
+
prepend_changelog(CHANGELOG, plan.changelog_entry);
|
|
1201
|
+
}
|
|
1202
|
+
|
|
1203
|
+
// Regenerate derived files (pack manifests, dist/agent-src/, tool
|
|
1204
|
+
// projections) so the PR's own consistency check passes. Without
|
|
1205
|
+
// this the bump only lands in package.json + marketplace.json and
|
|
1206
|
+
// the Sync + Generate Tools Consistency gate fails on the release
|
|
1207
|
+
// PR itself — exactly the failure mode PR #226 hit. `task
|
|
1208
|
+
// release-prepare` is idempotent, so resume runs are safe.
|
|
1209
|
+
_step(2, total, 'Regenerate derived files (`task release-prepare`)');
|
|
1210
|
+
run(['task', 'release-prepare']);
|
|
1211
|
+
}
|
|
1212
|
+
|
|
1213
|
+
// ─── 3. commit ──────────────────────────────────────────────────────────
|
|
1214
|
+
if (pr_merged) {
|
|
1215
|
+
_step(3, total, 'PR already merged — skip commit');
|
|
1216
|
+
} else {
|
|
1217
|
+
const last_msg = git(['log', '-1', '--format=%s'], { capture: true });
|
|
1218
|
+
const porcelain = git(['status', '--porcelain'], { capture: true });
|
|
1219
|
+
if (resume && last_msg === `release: ${plan.target}` && !porcelain) {
|
|
1220
|
+
_step(3, total, `Last commit already \`release: ${plan.target}\` and tree clean — skip`);
|
|
1221
|
+
} else {
|
|
1222
|
+
// `git add -A` stages the three primary bump files AND every
|
|
1223
|
+
// regenerated derived file (src/packs/*/pack.yaml + README.md,
|
|
1224
|
+
// dist/agent-src/, .augment/, tool projections). Listing them
|
|
1225
|
+
// explicitly would silently drift the moment a new generated
|
|
1226
|
+
// tree is added.
|
|
1227
|
+
run(['git', 'add', '-A']);
|
|
1228
|
+
// On resume the bump + era-split may already be committed (the
|
|
1229
|
+
// era-split lands its own commit, so `last_msg` above no longer
|
|
1230
|
+
// equals "release: X" and the skip guard misses). If nothing is
|
|
1231
|
+
// staged, the release content is already in history — skipping
|
|
1232
|
+
// beats failing `git commit` on an empty index.
|
|
1233
|
+
if (git(['diff', '--cached', '--name-only'], { capture: true }).trim()) {
|
|
1234
|
+
_step(3, total, `Commit \`release: ${plan.target}\``);
|
|
1235
|
+
run(['git', 'commit', '-m', `release: ${plan.target}`]);
|
|
1236
|
+
} else {
|
|
1237
|
+
_step(3, total, 'Release content already committed — skip empty commit');
|
|
1238
|
+
}
|
|
1239
|
+
}
|
|
1240
|
+
}
|
|
1241
|
+
|
|
1242
|
+
// ─── 4. push ────────────────────────────────────────────────────────────
|
|
1243
|
+
if (pr_merged) {
|
|
1244
|
+
_step(4, total, 'PR already merged — skip push');
|
|
1245
|
+
} else {
|
|
1246
|
+
// `git push -u` is naturally idempotent — it prints "Everything
|
|
1247
|
+
// up-to-date" when remote already matches. No probe needed.
|
|
1248
|
+
_step(4, total, `Push ${branch} to ${REMOTE}`);
|
|
1249
|
+
run(['git', 'push', '-u', REMOTE, branch]);
|
|
1250
|
+
}
|
|
1251
|
+
|
|
1252
|
+
// ─── 5. PR ──────────────────────────────────────────────────────────────
|
|
1253
|
+
if (pr_merged) {
|
|
1254
|
+
_step(5, total, `PR #${pr_info!['number']} already merged — skip`);
|
|
1255
|
+
} else if (resume && pr_state === 'OPEN') {
|
|
1256
|
+
_step(5, total, `PR already open: ${pr_info!['url']}`);
|
|
1257
|
+
} else {
|
|
1258
|
+
_step(5, total, 'Open pull request');
|
|
1259
|
+
const pr_changelog = _cap_body(
|
|
1260
|
+
plan.changelog_body,
|
|
1261
|
+
GH_PR_BODY_LIMIT - 200, // leave room for the prefix + footer
|
|
1262
|
+
'`CHANGELOG.md` in this PR',
|
|
1263
|
+
);
|
|
1264
|
+
const pr_body =
|
|
1265
|
+
`Release ${plan.target}.\n\n` +
|
|
1266
|
+
`${pr_changelog}\n\n` +
|
|
1267
|
+
'Created by `scripts/release.py`.';
|
|
1268
|
+
run([
|
|
1269
|
+
'gh',
|
|
1270
|
+
'pr',
|
|
1271
|
+
'create',
|
|
1272
|
+
'--base',
|
|
1273
|
+
MAIN_BRANCH,
|
|
1274
|
+
'--head',
|
|
1275
|
+
branch,
|
|
1276
|
+
'--title',
|
|
1277
|
+
`release: ${plan.target}`,
|
|
1278
|
+
'--body',
|
|
1279
|
+
pr_body,
|
|
1280
|
+
]);
|
|
1281
|
+
}
|
|
1282
|
+
|
|
1283
|
+
// ─── 6. wait for checks ─────────────────────────────────────────────────
|
|
1284
|
+
if (pr_merged) {
|
|
1285
|
+
_step(6, total, 'PR already merged — skip checks wait');
|
|
1286
|
+
} else if (wait_for_checks) {
|
|
1287
|
+
_step(6, total, 'Wait for PR checks');
|
|
1288
|
+
watch_pr_checks();
|
|
1289
|
+
} else {
|
|
1290
|
+
_step(6, total, 'Skip waiting for checks (--no-wait)');
|
|
1291
|
+
}
|
|
1292
|
+
|
|
1293
|
+
// ─── 7. merge ───────────────────────────────────────────────────────────
|
|
1294
|
+
if (pr_merged) {
|
|
1295
|
+
_step(7, total, `PR #${pr_info!['number']} already merged — skip`);
|
|
1296
|
+
} else {
|
|
1297
|
+
_step(7, total, 'Merge pull request (merge commit) and delete branch');
|
|
1298
|
+
run(['gh', 'pr', 'merge', '--merge', '--delete-branch']);
|
|
1299
|
+
}
|
|
1300
|
+
|
|
1301
|
+
// ─── 8. tag main + push tag ─────────────────────────────────────────────
|
|
1302
|
+
// Always idempotent — even outside resume mode this prevents a mid-flight
|
|
1303
|
+
// crash on step 9 from leaving a half-tagged release that subsequent
|
|
1304
|
+
// `task release` invocations can't recover from without `--resume`.
|
|
1305
|
+
if (git(['rev-parse', '--abbrev-ref', 'HEAD'], { capture: true }) !== MAIN_BRANCH) {
|
|
1306
|
+
run(['git', 'checkout', MAIN_BRANCH]);
|
|
1307
|
+
}
|
|
1308
|
+
run(['git', 'pull', '--ff-only', REMOTE, MAIN_BRANCH]);
|
|
1309
|
+
|
|
1310
|
+
if (_tag_exists_local(plan.target)) {
|
|
1311
|
+
if (_tag_exists_remote(plan.target)) {
|
|
1312
|
+
_step(8, total, `Tag ${plan.target} already on ${REMOTE} — skip`);
|
|
1313
|
+
} else {
|
|
1314
|
+
_step(8, total, `Tag ${plan.target} exists locally — push only`);
|
|
1315
|
+
run(['git', 'push', REMOTE, plan.target]);
|
|
1316
|
+
}
|
|
1317
|
+
} else {
|
|
1318
|
+
_step(8, total, `Tag merge commit and push ${plan.target}`);
|
|
1319
|
+
run(['git', 'tag', plan.target]);
|
|
1320
|
+
run(['git', 'push', REMOTE, plan.target]);
|
|
1321
|
+
}
|
|
1322
|
+
|
|
1323
|
+
// ─── 9. GitHub Release ──────────────────────────────────────────────────
|
|
1324
|
+
if (_release_exists(plan.target)) {
|
|
1325
|
+
_step(9, total, `GitHub Release ${plan.target} already exists — skip`);
|
|
1326
|
+
} else {
|
|
1327
|
+
_step(9, total, 'Create GitHub Release (triggers publish-npm on the tag)');
|
|
1328
|
+
const notes = _cap_body(
|
|
1329
|
+
plan.changelog_body || `Release ${plan.target}`,
|
|
1330
|
+
GH_RELEASE_NOTES_LIMIT,
|
|
1331
|
+
'`CHANGELOG.md`',
|
|
1332
|
+
);
|
|
1333
|
+
run([
|
|
1334
|
+
'gh',
|
|
1335
|
+
'release',
|
|
1336
|
+
'create',
|
|
1337
|
+
plan.target,
|
|
1338
|
+
'--title',
|
|
1339
|
+
plan.target,
|
|
1340
|
+
'--notes',
|
|
1341
|
+
notes,
|
|
1342
|
+
]);
|
|
1343
|
+
}
|
|
1344
|
+
|
|
1345
|
+
// ─── 10. delete the merged release branch (local + remote) ───────────────
|
|
1346
|
+
// Branch hygiene: a merged-but-undeleted release/X.Y.Z is what made
|
|
1347
|
+
// `--resume` mis-detect an old version. Delete it now so it can never
|
|
1348
|
+
// accumulate. Idempotent — skips whatever is already gone. Never touches
|
|
1349
|
+
// `main` or any tag.
|
|
1350
|
+
if (dry_run) {
|
|
1351
|
+
_step(10, total, `Would delete merged branch ${branch} (local + remote)`);
|
|
1352
|
+
} else {
|
|
1353
|
+
const deleted: string[] = [];
|
|
1354
|
+
if (
|
|
1355
|
+
_branch_exists_local(branch) &&
|
|
1356
|
+
git(['rev-parse', '--abbrev-ref', 'HEAD'], { capture: true }) !== branch
|
|
1357
|
+
) {
|
|
1358
|
+
run(['git', 'branch', '-D', branch], { check: false });
|
|
1359
|
+
deleted.push('local');
|
|
1360
|
+
}
|
|
1361
|
+
if (_branch_exists_remote(branch)) {
|
|
1362
|
+
run(['git', 'push', REMOTE, '--delete', branch], { check: false });
|
|
1363
|
+
deleted.push('remote');
|
|
1364
|
+
}
|
|
1365
|
+
const where = deleted.length > 0 ? deleted.join(' + ') : 'already gone';
|
|
1366
|
+
_step(10, total, `Delete merged branch ${branch} (${where})`);
|
|
1367
|
+
}
|
|
1368
|
+
|
|
1369
|
+
process.stdout.write('\n');
|
|
1370
|
+
process.stdout.write(`✅ Released ${plan.target}\n`);
|
|
1371
|
+
process.stdout.write(` https://github.com/${REPO_SLUG}/releases/tag/${plan.target}\n`);
|
|
1372
|
+
process.stdout.write(' npm publish runs asynchronously via publish-npm.yml on the tag.\n');
|
|
1373
|
+
}
|
|
1374
|
+
|
|
1375
|
+
// ─── entrypoint ───────────────────────────────────────────────────────────────
|
|
1376
|
+
|
|
1377
|
+
/** Mirror of `argparse.Namespace` for this CLI. */
|
|
1378
|
+
interface Args {
|
|
1379
|
+
bump_override: string | null;
|
|
1380
|
+
explicit: string | null;
|
|
1381
|
+
yes: boolean;
|
|
1382
|
+
dry_run: boolean;
|
|
1383
|
+
no_wait: boolean;
|
|
1384
|
+
resume: boolean;
|
|
1385
|
+
}
|
|
1386
|
+
|
|
1387
|
+
const PROG = 'release.py';
|
|
1388
|
+
// Verbatim argparse usage block (captured shape). The argparse `--help` BODY
|
|
1389
|
+
// (per-flag descriptions) is a documented divergence — argparse re-wraps it to
|
|
1390
|
+
// the live terminal width; the tests assert the `usage:` token + exit code,
|
|
1391
|
+
// not the body prose.
|
|
1392
|
+
const USAGE =
|
|
1393
|
+
`usage: ${PROG} [-h] [--as {major,minor,patch}] [--version EXPLICIT] [--yes]\n` +
|
|
1394
|
+
' [--dry-run] [--no-wait] [--resume]\n';
|
|
1395
|
+
|
|
1396
|
+
function _argError(msg: string): never {
|
|
1397
|
+
process.stderr.write(USAGE);
|
|
1398
|
+
process.stderr.write(`${PROG}: error: ${msg}\n`);
|
|
1399
|
+
throw new ArgparseExit(2);
|
|
1400
|
+
}
|
|
1401
|
+
|
|
1402
|
+
function _parse_args(argv: readonly string[]): Args {
|
|
1403
|
+
const args: Args = {
|
|
1404
|
+
bump_override: null,
|
|
1405
|
+
explicit: null,
|
|
1406
|
+
yes: false,
|
|
1407
|
+
dry_run: false,
|
|
1408
|
+
no_wait: false,
|
|
1409
|
+
resume: false,
|
|
1410
|
+
};
|
|
1411
|
+
|
|
1412
|
+
const positionals: string[] = [];
|
|
1413
|
+
let i = 0;
|
|
1414
|
+
while (i < argv.length) {
|
|
1415
|
+
const a = argv[i] as string;
|
|
1416
|
+
if (a === '-h' || a === '--help') {
|
|
1417
|
+
// argparse prints the full help to stdout; we emit the usage block
|
|
1418
|
+
// (the body is COLUMNS-dependent — documented divergence).
|
|
1419
|
+
process.stdout.write(USAGE);
|
|
1420
|
+
throw new ArgparseExit(0);
|
|
1421
|
+
}
|
|
1422
|
+
|
|
1423
|
+
// --flag=value form.
|
|
1424
|
+
const eq = a.startsWith('--') ? a.indexOf('=') : -1;
|
|
1425
|
+
const flag = eq >= 0 ? a.slice(0, eq) : a;
|
|
1426
|
+
const inlineVal = eq >= 0 ? a.slice(eq + 1) : null;
|
|
1427
|
+
|
|
1428
|
+
if (flag === '--as') {
|
|
1429
|
+
let value: string;
|
|
1430
|
+
if (inlineVal !== null) {
|
|
1431
|
+
value = inlineVal;
|
|
1432
|
+
} else {
|
|
1433
|
+
if (i + 1 >= argv.length) _argError('argument --as: expected one argument');
|
|
1434
|
+
value = argv[i + 1] as string;
|
|
1435
|
+
i += 1;
|
|
1436
|
+
}
|
|
1437
|
+
if (!['major', 'minor', 'patch'].includes(value)) {
|
|
1438
|
+
_argError(
|
|
1439
|
+
`argument --as: invalid choice: '${value}' ` +
|
|
1440
|
+
"(choose from 'major', 'minor', 'patch')",
|
|
1441
|
+
);
|
|
1442
|
+
}
|
|
1443
|
+
args.bump_override = value;
|
|
1444
|
+
i += 1;
|
|
1445
|
+
continue;
|
|
1446
|
+
}
|
|
1447
|
+
if (flag === '--version') {
|
|
1448
|
+
let value: string;
|
|
1449
|
+
if (inlineVal !== null) {
|
|
1450
|
+
value = inlineVal;
|
|
1451
|
+
} else {
|
|
1452
|
+
if (i + 1 >= argv.length) _argError('argument --version: expected one argument');
|
|
1453
|
+
value = argv[i + 1] as string;
|
|
1454
|
+
i += 1;
|
|
1455
|
+
}
|
|
1456
|
+
args.explicit = value;
|
|
1457
|
+
i += 1;
|
|
1458
|
+
continue;
|
|
1459
|
+
}
|
|
1460
|
+
if (flag === '--yes' || a === '-y') {
|
|
1461
|
+
if (inlineVal !== null) _argError(`argument --yes/-y: ignored explicit argument '${inlineVal}'`);
|
|
1462
|
+
args.yes = true;
|
|
1463
|
+
i += 1;
|
|
1464
|
+
continue;
|
|
1465
|
+
}
|
|
1466
|
+
if (flag === '--dry-run') {
|
|
1467
|
+
if (inlineVal !== null) _argError(`argument --dry-run: ignored explicit argument '${inlineVal}'`);
|
|
1468
|
+
args.dry_run = true;
|
|
1469
|
+
i += 1;
|
|
1470
|
+
continue;
|
|
1471
|
+
}
|
|
1472
|
+
if (flag === '--no-wait') {
|
|
1473
|
+
if (inlineVal !== null) _argError(`argument --no-wait: ignored explicit argument '${inlineVal}'`);
|
|
1474
|
+
args.no_wait = true;
|
|
1475
|
+
i += 1;
|
|
1476
|
+
continue;
|
|
1477
|
+
}
|
|
1478
|
+
if (flag === '--resume') {
|
|
1479
|
+
if (inlineVal !== null) _argError(`argument --resume: ignored explicit argument '${inlineVal}'`);
|
|
1480
|
+
args.resume = true;
|
|
1481
|
+
i += 1;
|
|
1482
|
+
continue;
|
|
1483
|
+
}
|
|
1484
|
+
if (a.startsWith('-') && a !== '-') {
|
|
1485
|
+
_argError(`unrecognized arguments: ${a}`);
|
|
1486
|
+
}
|
|
1487
|
+
positionals.push(a);
|
|
1488
|
+
i += 1;
|
|
1489
|
+
}
|
|
1490
|
+
if (positionals.length > 0) {
|
|
1491
|
+
_argError(`unrecognized arguments: ${positionals.join(' ')}`);
|
|
1492
|
+
}
|
|
1493
|
+
return args;
|
|
1494
|
+
}
|
|
1495
|
+
|
|
1496
|
+
/** Override wins; otherwise auto-detect from commits (or 'patch' if empty). */
|
|
1497
|
+
function resolve_bump(override: string | null, commits: readonly Commit[]): string {
|
|
1498
|
+
if (override) {
|
|
1499
|
+
return override;
|
|
1500
|
+
}
|
|
1501
|
+
return infer_bump(commits);
|
|
1502
|
+
}
|
|
1503
|
+
|
|
1504
|
+
const _RELEASE_BRANCH_RE = /^release\/(\d+\.\d+\.\d+)$/;
|
|
1505
|
+
|
|
1506
|
+
/**
|
|
1507
|
+
* Find the in-flight release target — the SOURCE OF TRUTH is package.json.
|
|
1508
|
+
*
|
|
1509
|
+
* An "in-flight" release is one whose version was already bumped into `main`'s
|
|
1510
|
+
* `package.json` (and possibly merged) but whose tag has not yet been pushed —
|
|
1511
|
+
* i.e. the publish step never completed. The canonical anchor is therefore
|
|
1512
|
+
* `package.json` version `V` **with no matching tag `V`**, NOT the set of
|
|
1513
|
+
* `release/X.Y.Z` branches.
|
|
1514
|
+
*
|
|
1515
|
+
* Why not the branch set: merged release branches are frequently left
|
|
1516
|
+
* undeleted on the remote, so "highest existing release/* branch" can resolve
|
|
1517
|
+
* to an OLD, already-published version (e.g. picking 5.4.0 while 5.8.0 is the
|
|
1518
|
+
* real in-flight target) and tag a downgrade. The package.json version cannot
|
|
1519
|
+
* lie that way — it is the version main currently claims to be, and an
|
|
1520
|
+
* untagged claim is exactly an incomplete release.
|
|
1521
|
+
*
|
|
1522
|
+
* Resolution order:
|
|
1523
|
+
* 1. If HEAD is on a `release/X.Y.Z` branch, that explicit checkout wins.
|
|
1524
|
+
* 2. Else: read `package.json` version `V`. If tag `V` does not exist
|
|
1525
|
+
* (local or remote), `V` is the in-flight target. If it is already
|
|
1526
|
+
* tagged, the release is complete → return null (regular bump path).
|
|
1527
|
+
*
|
|
1528
|
+
* Stale `release/*` branches are never used for version detection.
|
|
1529
|
+
*/
|
|
1530
|
+
function _detect_in_flight_target(): string | null {
|
|
1531
|
+
const head = git(['rev-parse', '--abbrev-ref', 'HEAD'], { capture: true });
|
|
1532
|
+
const m = _RELEASE_BRANCH_RE.exec(head);
|
|
1533
|
+
if (m) {
|
|
1534
|
+
return m[1] as string;
|
|
1535
|
+
}
|
|
1536
|
+
|
|
1537
|
+
let version: string;
|
|
1538
|
+
try {
|
|
1539
|
+
const data = JSON.parse(fs.readFileSync(PACKAGE_JSON, 'utf-8')) as Record<string, unknown>;
|
|
1540
|
+
if (!('version' in data)) {
|
|
1541
|
+
return null; // KeyError analogue.
|
|
1542
|
+
}
|
|
1543
|
+
version = data['version'] as string;
|
|
1544
|
+
} catch {
|
|
1545
|
+
return null; // OSError / JSONDecodeError analogue.
|
|
1546
|
+
}
|
|
1547
|
+
try {
|
|
1548
|
+
parse_version(version);
|
|
1549
|
+
} catch {
|
|
1550
|
+
return null;
|
|
1551
|
+
}
|
|
1552
|
+
|
|
1553
|
+
// An already-tagged version is a completed release, not in-flight.
|
|
1554
|
+
if (_tag_exists_local(version) || _tag_exists_remote(version)) {
|
|
1555
|
+
return null;
|
|
1556
|
+
}
|
|
1557
|
+
return version;
|
|
1558
|
+
}
|
|
1559
|
+
|
|
1560
|
+
function main(argv: readonly string[] | null = null): number {
|
|
1561
|
+
const args = _parse_args(argv === null ? process.argv.slice(2) : argv);
|
|
1562
|
+
|
|
1563
|
+
const current = (
|
|
1564
|
+
JSON.parse(fs.readFileSync(PACKAGE_JSON, 'utf-8')) as Record<string, unknown>
|
|
1565
|
+
)['version'] as string;
|
|
1566
|
+
parse_version(current);
|
|
1567
|
+
|
|
1568
|
+
const prev = latest_tag();
|
|
1569
|
+
const commits = commits_since(prev);
|
|
1570
|
+
const bump = resolve_bump(args.bump_override, commits);
|
|
1571
|
+
|
|
1572
|
+
// Resume mode: prefer an existing `release/X.Y.Z` over computed bump,
|
|
1573
|
+
// so we don't accidentally start a 1.16.0 release while 1.15.0 is
|
|
1574
|
+
// still in flight. Explicit --version still wins.
|
|
1575
|
+
const in_flight = args.resume ? _detect_in_flight_target() : null;
|
|
1576
|
+
let target: string;
|
|
1577
|
+
if (args.explicit) {
|
|
1578
|
+
target = args.explicit;
|
|
1579
|
+
} else if (in_flight) {
|
|
1580
|
+
target = in_flight;
|
|
1581
|
+
process.stdout.write(
|
|
1582
|
+
`(resume) in-flight target ${in_flight} (package.json version with no tag yet)\n`,
|
|
1583
|
+
);
|
|
1584
|
+
} else {
|
|
1585
|
+
target = bump_version(current, bump);
|
|
1586
|
+
}
|
|
1587
|
+
parse_version(target);
|
|
1588
|
+
|
|
1589
|
+
if (!args.dry_run) {
|
|
1590
|
+
preflight(target, { resume: args.resume });
|
|
1591
|
+
}
|
|
1592
|
+
|
|
1593
|
+
const today = todayIso();
|
|
1594
|
+
const test_trend_line = _render_test_trend_line(prev);
|
|
1595
|
+
const [full, body] = render_changelog_entry(target, prev, commits, today, {
|
|
1596
|
+
test_trend_line,
|
|
1597
|
+
});
|
|
1598
|
+
|
|
1599
|
+
// Era-split planning: only crosses the gate when the current era body
|
|
1600
|
+
// has grown past the drift cap AND the release crosses a minor/major
|
|
1601
|
+
// boundary. Patch overflow is caught by the drift test (red CI), not
|
|
1602
|
+
// by an auto-split into a nonsensical "pre-X.Y.Z" archive.
|
|
1603
|
+
let split: SplitPlan | null = null;
|
|
1604
|
+
const body_size = current_era_body_size();
|
|
1605
|
+
if (body_size > CURRENT_ERA_BODY_CAP) {
|
|
1606
|
+
const candidate = plan_split(target);
|
|
1607
|
+
if (candidate === null) {
|
|
1608
|
+
die(
|
|
1609
|
+
`current era body is ${body_size} lines (cap ` +
|
|
1610
|
+
`${CURRENT_ERA_BODY_CAP}) but release ${target} is a patch ` +
|
|
1611
|
+
`within the same era — split needs a minor/major bump. ` +
|
|
1612
|
+
'Cut a minor release or split CHANGELOG.md manually first.',
|
|
1613
|
+
);
|
|
1614
|
+
}
|
|
1615
|
+
split = candidate;
|
|
1616
|
+
}
|
|
1617
|
+
|
|
1618
|
+
const plan = new Plan(current, target, bump, commits, prev, body, full, split);
|
|
1619
|
+
print_preview(plan);
|
|
1620
|
+
if (args.resume) {
|
|
1621
|
+
process.stdout.write('(resume) probing existing state — completed steps will be skipped.\n');
|
|
1622
|
+
}
|
|
1623
|
+
|
|
1624
|
+
if (args.dry_run) {
|
|
1625
|
+
return 0;
|
|
1626
|
+
}
|
|
1627
|
+
|
|
1628
|
+
const verdict = confirmGate(plan.target, args.yes);
|
|
1629
|
+
if (!verdict.proceed) {
|
|
1630
|
+
if (verdict.message) {
|
|
1631
|
+
const sink = verdict.stream === 'stderr' ? process.stderr : process.stdout;
|
|
1632
|
+
sink.write(`${verdict.message}\n`);
|
|
1633
|
+
}
|
|
1634
|
+
return 1;
|
|
1635
|
+
}
|
|
1636
|
+
|
|
1637
|
+
execute(plan, {
|
|
1638
|
+
wait_for_checks: !args.no_wait,
|
|
1639
|
+
dry_run: false,
|
|
1640
|
+
resume: args.resume,
|
|
1641
|
+
});
|
|
1642
|
+
return 0;
|
|
1643
|
+
}
|
|
1644
|
+
|
|
1645
|
+
// ─── CLI entry (release.py:1090-1091 `raise SystemExit(main())`) ──────────────
|
|
1646
|
+
|
|
1647
|
+
const _isCliEntry =
|
|
1648
|
+
process.argv[1] !== undefined &&
|
|
1649
|
+
import.meta.url === pathToFileURL(path.resolve(process.argv[1])).href;
|
|
1650
|
+
if (_isCliEntry || process.argv[1] === _HERE) {
|
|
1651
|
+
try {
|
|
1652
|
+
// main() can return an int directly OR throw SystemExitError via die();
|
|
1653
|
+
// both flow into process.exitCode. CalledProcessError is intentionally
|
|
1654
|
+
// NOT caught here, so it propagates (non-zero + traceback), matching the
|
|
1655
|
+
// Python original letting subprocess.CalledProcessError raise.
|
|
1656
|
+
process.exitCode = main(process.argv.slice(2));
|
|
1657
|
+
} catch (e) {
|
|
1658
|
+
if (e instanceof SystemExitError || e instanceof ArgparseExit) {
|
|
1659
|
+
process.exitCode = e.code;
|
|
1660
|
+
} else {
|
|
1661
|
+
throw e;
|
|
1662
|
+
}
|
|
1663
|
+
}
|
|
1664
|
+
}
|
|
1665
|
+
|
|
1666
|
+
export {
|
|
1667
|
+
main,
|
|
1668
|
+
_parse_args,
|
|
1669
|
+
parse_version,
|
|
1670
|
+
bump_version,
|
|
1671
|
+
commits_since,
|
|
1672
|
+
infer_bump,
|
|
1673
|
+
render_changelog_entry,
|
|
1674
|
+
_changelog_line,
|
|
1675
|
+
_cap_body,
|
|
1676
|
+
prepend_changelog,
|
|
1677
|
+
set_package_version,
|
|
1678
|
+
set_marketplace_version,
|
|
1679
|
+
resolve_bump,
|
|
1680
|
+
_detect_in_flight_target,
|
|
1681
|
+
print_preview,
|
|
1682
|
+
latest_tag,
|
|
1683
|
+
_render_test_trend_line,
|
|
1684
|
+
_previous_test_count_from_changelog,
|
|
1685
|
+
Commit,
|
|
1686
|
+
Plan,
|
|
1687
|
+
SystemExitError,
|
|
1688
|
+
ArgparseExit,
|
|
1689
|
+
CalledProcessError,
|
|
1690
|
+
die,
|
|
1691
|
+
BREAKING_RE,
|
|
1692
|
+
CONVENTIONAL_RE,
|
|
1693
|
+
SEMVER_RE,
|
|
1694
|
+
_RELEASE_BRANCH_RE,
|
|
1695
|
+
MODULE_DOC_FIRST_LINE,
|
|
1696
|
+
};
|
|
1697
|
+
export type { Args };
|