@event4u/agent-config 6.1.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 +35 -3
- package/AGENTS.md +8 -7
- package/CHANGELOG.md +408 -0
- package/CONTRIBUTING.md +1 -1
- package/README.md +17 -15
- package/dist/agent-src/commands/agent-status.md +2 -2
- package/dist/agent-src/commands/agents/audit.md +3 -3
- package/dist/agent-src/commands/agents/init.md +1 -1
- package/dist/agent-src/commands/agents/optimize.md +4 -4
- 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.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/check-current-md.md +3 -3
- package/dist/agent-src/commands/condense.md +2 -2
- package/dist/agent-src/commands/council/debate.md +2 -2
- package/dist/agent-src/commands/council/default.md +45 -18
- package/dist/agent-src/commands/fix/portability.md +3 -3
- package/dist/agent-src/commands/fix/refs.md +3 -3
- package/dist/agent-src/commands/implement-ticket.md +36 -6
- package/dist/agent-src/commands/knowledge/cross-repo.md +1 -1
- package/dist/agent-src/commands/memory/add.md +1 -1
- package/dist/agent-src/commands/mission/upgrade.md +182 -0
- package/dist/agent-src/commands/optimize/skills.md +2 -2
- package/dist/agent-src/commands/orchestrate.md +1 -1
- package/dist/agent-src/commands/pr/create.md +6 -4
- package/dist/agent-src/commands/review-changes.md +8 -0
- package/dist/agent-src/commands/roadmap/materialize.md +73 -0
- package/dist/agent-src/commands/skill/preview.md +1 -1
- package/dist/agent-src/commands/skills/discover.md +1 -1
- package/dist/agent-src/commands/threat-model.md +4 -4
- package/dist/agent-src/commands/upstream-contribute.md +3 -3
- package/dist/agent-src/commands/video/from-script.md +2 -2
- package/dist/agent-src/commands/video/from-song.md +3 -3
- package/dist/agent-src/commands/video/scene.md +1 -1
- package/dist/agent-src/commands/video/storyboard.md +1 -1
- package/dist/agent-src/commands/video.md +3 -3
- package/dist/agent-src/contexts/communication/rules-auto/source-of-truth-mechanics.md +3 -3
- package/dist/agent-src/contexts/communication/rules-auto/user-interaction-mechanics.md +1 -1
- 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/roadmap-process-loop.md +2 -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/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/git-history-discipline.md +1 -0
- 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/lethal-trifecta-guard.md +1 -1
- 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 +10 -0
- package/dist/agent-src/rules/security-sensitive-stop.md +9 -3
- package/dist/agent-src/rules/size-enforcement.md +1 -1
- package/dist/agent-src/rules/source-confidentiality.md +3 -3
- package/dist/agent-src/rules/source-discovery-gate.md +98 -0
- package/dist/agent-src/rules/think-before-action.md +1 -0
- package/dist/agent-src/rules/ui-audit-gate.md +2 -0
- package/dist/agent-src/rules/untrusted-input-defense.md +1 -1
- 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/agent-security-review/evals/triggers.json +1 -0
- package/dist/agent-src/skills/agents-md-thin-root/SKILL.md +1 -1
- package/dist/agent-src/skills/ai-council/SKILL.md +1 -1
- package/dist/agent-src/skills/analysis-autonomous-mode/SKILL.md +9 -13
- 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/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-writing/SKILL.md +2 -2
- package/dist/agent-src/skills/complexity-first-planning/evals/triggers.json +1 -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/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/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/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/estimate-ticket/evals/triggers.json +1 -0
- 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/md-language-check/SKILL.md +1 -1
- 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-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/reasoning-orchestrator/SKILL.md +1 -1
- package/dist/agent-src/skills/reasoning-orchestrator/evals/triggers.json +1 -0
- package/dist/agent-src/skills/refine-ticket/evals/triggers.json +1 -0
- 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/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/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/threat-modeling/SKILL.md +1 -0
- 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/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/commands/commands.js +2 -2
- 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/main.js +22 -1
- package/dist/cli/main.js.map +1 -1
- 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 +1174 -123
- package/dist/discovery/discovery-manifest.json.sha256 +1 -1
- package/dist/discovery/discovery-manifest.summary.md +9 -6
- package/dist/discovery/orphan-report.md +1 -1
- package/dist/discovery/packs.json +163 -15
- package/dist/discovery/trust-report.md +4 -4
- package/dist/discovery/workspaces.json +73 -12
- 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 +50 -21
- 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 +15 -0
- package/dist/server/schemas/settings.js.map +1 -1
- package/docs/SKILL_CENSUS.md +344 -0
- package/docs/architecture/augment-projection.md +1 -1
- package/docs/architecture/multi-tool-projection.md +3 -3
- package/docs/architecture.md +37 -6
- package/docs/benchmark.md +24 -27
- package/docs/capability-matrix.md +32 -0
- package/docs/catalog.md +50 -9
- package/docs/command-naming-audit.md +60 -0
- package/docs/contracts/STABILITY.md +32 -0
- package/docs/contracts/agents-md-tech-stack.md +1 -1
- package/docs/contracts/ai-council-config.md +22 -22
- 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 +2 -1
- package/docs/contracts/command-surface-tiers.md +13 -0
- package/docs/contracts/discovery-manifest.schema.json +24 -5
- package/docs/contracts/implement-ticket-flow.md +9 -9
- 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/multi-tool-projection-fidelity.md +1 -1
- package/docs/contracts/namespace.md +2 -2
- package/docs/contracts/no-runtime-boundary.md +56 -0
- package/docs/contracts/package-self-orientation.md +24 -0
- 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/skill-domains.md +1 -1
- package/docs/contracts/smoke-contracts.md +1 -1
- package/docs/contracts/surface-tiers.md +81 -0
- package/docs/contracts/ticket-bundle-format.md +228 -0
- package/docs/cookbook.md +152 -0
- package/docs/customization.md +12 -1
- package/docs/decisions/ADR-013-discovery-frontmatter-contract.md +16 -0
- 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-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 +9 -0
- 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 +2 -2
- package/docs/guidelines/agent-infra/failure-signatures.md +35 -0
- package/docs/guidelines/agent-infra/frontier-reasoning-operating-profile.md +5 -0
- package/docs/guidelines/agent-infra/size-and-scope.md +17 -0
- package/docs/guidelines/agent-infra/skill-quality-checklist.md +2 -2
- 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/mcp.md +1 -1
- 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/personas.md +6 -1
- package/docs/role-experiences.md +19 -0
- package/docs/setup/per-ide/windsurf.md +1 -1
- package/docs/skills-catalog.md +24 -3
- package/docs/threat-model.md +28 -0
- package/llms.txt +23 -2
- package/package.json +10 -15
- package/src/config/agent-settings.template.yml +64 -1
- package/src/config/discovery/packs.yml +31 -0
- package/src/config/discovery/unassigned-artefacts.yml +6 -0
- package/src/config/discovery/workspaces.yml +2 -2
- package/src/config/gitignore-block.txt +7 -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 +1 -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 +24 -10
- 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 +20 -14
- 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/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 +1 -1
- 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 +3 -2
- 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/scripts/archive_completed_roadmaps.py +0 -171
- 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/scripts/check_memory.py +0 -282
- 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 -436
- package/dist/agent-src/templates/scripts/memory_report.py +0 -314
- package/dist/agent-src/templates/scripts/memory_signal.py +0 -165
- package/dist/agent-src/templates/scripts/memory_status.py +0 -76
- 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 -283
- 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/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 -1669
- 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_ab_scoring_v2.py +0 -227
- 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 -312
- 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/link_crypto.py +0 -206
- 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/security_lint.py +0 -228
- 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 -1419
- 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 -173
- 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 -682
- 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 -220
- 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 -814
- package/src/scripts/bench_ab_tracka_run.py +0 -202
- package/src/scripts/bench_ab_v2_run.py +0 -247
- package/src/scripts/bench_ab_v2_stats.py +0 -347
- 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 -757
- 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 -98
- 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 -429
- package/src/scripts/check_memory_proposal.py +0 -182
- package/src/scripts/check_module_management_neutral.py +0 -147
- package/src/scripts/check_no_external_sources.py +0 -101
- 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 -559
- 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 -2571
- 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/injection_scan_hook.py +0 -145
- package/src/scripts/install.py +0 -5258
- 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_security.py +0 -112
- 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 -173
- 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 -216
- 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 -222
- package/src/scripts/lint_empty_roadmaps.py +0 -80
- 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_hidden_unicode.py +0 -132
- package/src/scripts/lint_hook_concern_budget.py +0 -207
- package/src/scripts/lint_hook_manifest.py +0 -217
- package/src/scripts/lint_instruction_smuggling.py +0 -107
- 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_config_security.py +0 -124
- 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_frontmatter_safety.py +0 -144
- 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/lint_workspace_boundary.py +0 -122
- 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 -926
- 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 -436
- package/src/scripts/memory_report.py +0 -314
- package/src/scripts/memory_signal.py +0 -165
- package/src/scripts/memory_status.py +0 -76
- 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 -356
- 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 -401
- 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/security_audit_config.py +0 -153
- 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
package/src/scripts/condense.py
DELETED
|
@@ -1,1919 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env python3
|
|
2
|
-
"""
|
|
3
|
-
Agent-config sync ā condense .agent-src.uncondensed/ ā dist/agent-src/
|
|
4
|
-
and project dist/agent-src/ ā .augment/ (copies for rules by default,
|
|
5
|
-
symlinks for the rest; opt into rule symlinks via
|
|
6
|
-
augment.rules_use_symlinks in .agent-settings.yml).
|
|
7
|
-
|
|
8
|
-
Copies non-.md files as-is. Lists .md files that need condensation (done by the
|
|
9
|
-
Augment agent interactively). Tracks SHA-256 hashes of source files to detect
|
|
10
|
-
changes since last condensation.
|
|
11
|
-
|
|
12
|
-
Usage:
|
|
13
|
-
python scripts/condense.py # sync all non-.md files + cleanup + project
|
|
14
|
-
python scripts/condense.py --list # list .md files needing condensation
|
|
15
|
-
python scripts/condense.py --changed # list only .md files changed since last condensation
|
|
16
|
-
python scripts/condense.py --check # check if directories are in sync
|
|
17
|
-
python scripts/condense.py --mark-done <relative-path> # mark file as condensed (update hash)
|
|
18
|
-
python scripts/condense.py --mark-all-done # mark ALL .md files as condensed
|
|
19
|
-
python scripts/condense.py --project-augment # rebuild .augment/ projection
|
|
20
|
-
"""
|
|
21
|
-
|
|
22
|
-
from __future__ import annotations
|
|
23
|
-
|
|
24
|
-
import hashlib
|
|
25
|
-
import json
|
|
26
|
-
import re
|
|
27
|
-
import shutil
|
|
28
|
-
import sys
|
|
29
|
-
from pathlib import Path
|
|
30
|
-
try: # invocation-agnostic import (repo-root-on-path vs scripts-on-path)
|
|
31
|
-
from scripts._lib.agent_settings import project_settings_path
|
|
32
|
-
except ModuleNotFoundError: # pragma: no cover
|
|
33
|
-
from _lib.agent_settings import project_settings_path
|
|
34
|
-
|
|
35
|
-
import yaml
|
|
36
|
-
|
|
37
|
-
sys.path.insert(0, str(Path(__file__).resolve().parent))
|
|
38
|
-
from _lib.script_output import info, success, flush_summary, resolve_level # noqa: E402
|
|
39
|
-
from _lib.agent_src import ( # noqa: E402
|
|
40
|
-
_SLUG_PREFIX_RE,
|
|
41
|
-
artefact_roots,
|
|
42
|
-
iter_all_sources,
|
|
43
|
-
resolve_logical,
|
|
44
|
-
strip_source_prefix,
|
|
45
|
-
)
|
|
46
|
-
|
|
47
|
-
PROJECT_ROOT = Path(__file__).resolve().parent.parent.parent
|
|
48
|
-
# Legacy single-root anchor ā kept for backward compatibility with callers
|
|
49
|
-
# that pass it explicitly. Multi-root iteration (post-ADR-017 physical
|
|
50
|
-
# move) goes through `_lib.agent_src` helpers.
|
|
51
|
-
SOURCE_DIR = PROJECT_ROOT / ".agent-src.uncondensed"
|
|
52
|
-
TARGET_DIR = PROJECT_ROOT / "dist/agent-src"
|
|
53
|
-
AUGMENT_DIR = PROJECT_ROOT / ".augment"
|
|
54
|
-
HASH_FILE = PROJECT_ROOT / "internal" / ".condensation-hashes.json"
|
|
55
|
-
SETTINGS_FILE = project_settings_path(PROJECT_ROOT)
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
def _iter_sources():
|
|
59
|
-
"""Yield (physical_path, logical_relpath) for every source artefact.
|
|
60
|
-
|
|
61
|
-
Wraps `_lib.agent_src.iter_all_sources` so the condenseor walks every
|
|
62
|
-
active source root (legacy `.agent-src.uncondensed/` plus any
|
|
63
|
-
`packages/*/.agent-src.uncondensed/`) and keys outputs by the
|
|
64
|
-
logical relative path that survives the physical move (ADR-017).
|
|
65
|
-
"""
|
|
66
|
-
yield from iter_all_sources()
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
def _resolve_source(relative: str) -> Path | None:
|
|
70
|
-
"""Find the physical path that backs a logical relative path."""
|
|
71
|
-
return resolve_logical(relative)
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
def _any_source_root_exists() -> bool:
|
|
75
|
-
"""True if at least one artefact source root contains files."""
|
|
76
|
-
return bool(artefact_roots())
|
|
77
|
-
|
|
78
|
-
# Self-projection tool toggle ā see agents/.agent-tools.yml. When the file is
|
|
79
|
-
# absent (e.g. tests run in tmp dirs, consumer projects), `_active_tools`
|
|
80
|
-
# returns ``None`` which is treated as "emit every tool".
|
|
81
|
-
_ALL_TOOLS = frozenset({
|
|
82
|
-
"claude-code", "claude-desktop", "augment", "copilot",
|
|
83
|
-
"cursor", "windsurf", "cline", "gemini",
|
|
84
|
-
})
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
def _active_tools() -> frozenset[str] | None:
|
|
88
|
-
"""Return the set of active self-projection tools, or None for "all".
|
|
89
|
-
|
|
90
|
-
Reads `agents/.agent-tools.yml` relative to the current `PROJECT_ROOT`
|
|
91
|
-
so test fixtures that monkey-patch `condense.PROJECT_ROOT` see their
|
|
92
|
-
own (empty) project root and get the default "all tools" behaviour.
|
|
93
|
-
"""
|
|
94
|
-
tools_file = PROJECT_ROOT / "agents" / ".agent-tools.yml"
|
|
95
|
-
if not tools_file.exists():
|
|
96
|
-
return None
|
|
97
|
-
try:
|
|
98
|
-
data = yaml.safe_load(tools_file.read_text()) or {}
|
|
99
|
-
except yaml.YAMLError:
|
|
100
|
-
return None
|
|
101
|
-
tools = data.get("tools") if isinstance(data, dict) else None
|
|
102
|
-
if not isinstance(tools, list):
|
|
103
|
-
return None
|
|
104
|
-
return frozenset(str(t) for t in tools if isinstance(t, str))
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
def _tool_active(tool_id: str) -> bool:
|
|
108
|
-
"""True when ``tool_id`` should be emitted by self-projection."""
|
|
109
|
-
active = _active_tools()
|
|
110
|
-
return True if active is None else tool_id in active
|
|
111
|
-
|
|
112
|
-
# Files to copy as-is even if .md (not condensed by agent)
|
|
113
|
-
COPY_AS_IS = {"README.md"}
|
|
114
|
-
|
|
115
|
-
# Directories (relative to SOURCE_DIR) whose .md content is data, not prose,
|
|
116
|
-
# and must be copied verbatim. Ghostwriter fixtures carry voice_samples that
|
|
117
|
-
# would be destroyed by telegraph condensation.
|
|
118
|
-
COPY_AS_IS_DIRS = frozenset({"ghostwriter"})
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
def _read_augment_rules_use_symlinks() -> bool:
|
|
122
|
-
"""Read augment.rules_use_symlinks from .agent-settings.yml.
|
|
123
|
-
|
|
124
|
-
Returns True only when the setting is present under the top-level
|
|
125
|
-
``augment:`` block and resolves to a truthy value. Missing file,
|
|
126
|
-
missing block, or any other value ā False (preserve copy default).
|
|
127
|
-
|
|
128
|
-
Centralized loader (road-to-portable-dev-preferences P3): tolerance
|
|
129
|
-
contract handles missing file / malformed YAML / no PyYAML uniformly.
|
|
130
|
-
"""
|
|
131
|
-
try:
|
|
132
|
-
from scripts._lib.agent_settings import load_agent_settings
|
|
133
|
-
except ImportError: # pragma: no cover ā script-style invocation
|
|
134
|
-
import sys as _sys
|
|
135
|
-
from pathlib import Path as _Path
|
|
136
|
-
_sys.path.insert(0, str(_Path(__file__).resolve().parent))
|
|
137
|
-
from _lib.agent_settings import load_agent_settings # type: ignore[import-not-found]
|
|
138
|
-
|
|
139
|
-
data = load_agent_settings(project_path=SETTINGS_FILE)
|
|
140
|
-
augment = data.get("augment")
|
|
141
|
-
if not isinstance(augment, dict):
|
|
142
|
-
return False
|
|
143
|
-
value = augment.get("rules_use_symlinks")
|
|
144
|
-
if isinstance(value, bool):
|
|
145
|
-
return value
|
|
146
|
-
if isinstance(value, str):
|
|
147
|
-
return value.strip().lower() in ("true", "yes", "on", "1")
|
|
148
|
-
if isinstance(value, int):
|
|
149
|
-
return value == 1
|
|
150
|
-
return False
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
def _lean_projection_mode() -> str:
|
|
154
|
-
"""Read lean_projection.mode from .agent-settings.yml.
|
|
155
|
-
|
|
156
|
-
`eager-all` (default) ā every rule body inlined into every projection
|
|
157
|
-
(today's behaviour). `thin` ā kernel full-bodied + non-kernel rules as
|
|
158
|
-
router-resolved pointers (lean-initial-context Phase 3.1; ~36k GPT tok
|
|
159
|
-
lighter, measured). Missing / malformed ā `eager-all`, so the thin path
|
|
160
|
-
is strictly opt-in and one-flip-revertible (see docs/contracts/rule-router.md
|
|
161
|
-
§ Kill-switch). The flip MUST be live-A/B-validated before it ships as the
|
|
162
|
-
default ā a thin projection only holds behaviour if the agent resolves the
|
|
163
|
-
pointer on trigger-match.
|
|
164
|
-
"""
|
|
165
|
-
try:
|
|
166
|
-
from scripts._lib.agent_settings import load_agent_settings
|
|
167
|
-
except ImportError: # pragma: no cover ā script-style invocation
|
|
168
|
-
import sys as _sys
|
|
169
|
-
from pathlib import Path as _Path
|
|
170
|
-
_sys.path.insert(0, str(_Path(__file__).resolve().parent))
|
|
171
|
-
from _lib.agent_settings import load_agent_settings # type: ignore[import-not-found]
|
|
172
|
-
|
|
173
|
-
data = load_agent_settings(project_path=SETTINGS_FILE)
|
|
174
|
-
lean = data.get("lean_projection")
|
|
175
|
-
if isinstance(lean, dict) and str(lean.get("mode", "")).strip().lower() == "thin":
|
|
176
|
-
return "thin"
|
|
177
|
-
return "eager-all"
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
def file_hash(filepath: Path) -> str:
|
|
181
|
-
"""Return SHA-256 hex digest of a file."""
|
|
182
|
-
h = hashlib.sha256()
|
|
183
|
-
h.update(filepath.read_bytes())
|
|
184
|
-
return h.hexdigest()
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
# ---------------------------------------------------------------------------
|
|
188
|
-
# Transitive include hashing (road-to-6.0.0-D Phase 0 Step 1)
|
|
189
|
-
#
|
|
190
|
-
# An artefact declares the skills / rules it includes via its frontmatter
|
|
191
|
-
# `skills:` / `rules:` lists. The condensation gate must treat that artefact
|
|
192
|
-
# as stale when an included skill/rule changes ā otherwise a moved or edited
|
|
193
|
-
# dependency slips past the hash check silently (the single biggest risk the
|
|
194
|
-
# 6.0.0-D council named). `effective_hash` folds the content hash of every
|
|
195
|
-
# transitive include into the artefact's own content hash.
|
|
196
|
-
#
|
|
197
|
-
# Leaf artefacts (no declared includes) hash exactly like `file_hash`, so the
|
|
198
|
-
# stored hash table only changes for artefacts that actually declare deps ā
|
|
199
|
-
# the migration stays contained.
|
|
200
|
-
# ---------------------------------------------------------------------------
|
|
201
|
-
|
|
202
|
-
_DEP_FRONTMATTER_KEYS = ("skills", "rules")
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
def _slug_to_logical(slug: str) -> str | None:
|
|
206
|
-
"""Map a skill / rule slug to its logical relative path, if it resolves.
|
|
207
|
-
|
|
208
|
-
Skill slug ``foo`` ā ``skills/foo/SKILL.md``; rule slug ``bar`` ā
|
|
209
|
-
``rules/bar.md``. Returns the first candidate that backs a real source
|
|
210
|
-
file, else ``None`` (a typo'd / external slug contributes no edge).
|
|
211
|
-
"""
|
|
212
|
-
for cand in (f"skills/{slug}/SKILL.md", f"rules/{slug}.md"):
|
|
213
|
-
if _resolve_source(cand) is not None:
|
|
214
|
-
return cand
|
|
215
|
-
return None
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
def _direct_includes(relative: str) -> list[str]:
|
|
219
|
-
"""Logical relpaths an artefact directly includes via frontmatter.
|
|
220
|
-
|
|
221
|
-
Reads the artefact's ``skills:`` / ``rules:`` frontmatter lists and
|
|
222
|
-
resolves each slug to a logical path. Non-resolving slugs are skipped.
|
|
223
|
-
"""
|
|
224
|
-
source = _resolve_source(relative)
|
|
225
|
-
if source is None:
|
|
226
|
-
return []
|
|
227
|
-
try:
|
|
228
|
-
meta, _ = _parse_frontmatter(source.read_text(encoding="utf-8"))
|
|
229
|
-
except (OSError, UnicodeDecodeError):
|
|
230
|
-
return []
|
|
231
|
-
deps: list[str] = []
|
|
232
|
-
for key in _DEP_FRONTMATTER_KEYS:
|
|
233
|
-
value = meta.get(key)
|
|
234
|
-
if not isinstance(value, list):
|
|
235
|
-
continue
|
|
236
|
-
for item in value:
|
|
237
|
-
if not isinstance(item, str):
|
|
238
|
-
continue
|
|
239
|
-
logical = _slug_to_logical(item.strip())
|
|
240
|
-
if logical is not None and logical != relative:
|
|
241
|
-
deps.append(logical)
|
|
242
|
-
return deps
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
def effective_hash(relative: str, _seen: frozenset | None = None) -> str:
|
|
246
|
-
"""Content hash of an artefact folded with its transitive include hashes.
|
|
247
|
-
|
|
248
|
-
Changes whenever the artefact's own bytes change OR any skill/rule it
|
|
249
|
-
transitively includes changes. Cycle-safe: a slug that re-enters the
|
|
250
|
-
current resolution chain folds only its own content hash, breaking the
|
|
251
|
-
loop. Leaf artefacts (no includes) return exactly ``file_hash`` so the
|
|
252
|
-
stored-hash migration is limited to artefacts that declare dependencies.
|
|
253
|
-
"""
|
|
254
|
-
source = _resolve_source(relative)
|
|
255
|
-
if source is None:
|
|
256
|
-
return ""
|
|
257
|
-
own = file_hash(source)
|
|
258
|
-
if _seen is not None and relative in _seen:
|
|
259
|
-
return own # cycle ā fold own content only
|
|
260
|
-
deps = sorted(set(_direct_includes(relative)))
|
|
261
|
-
if not deps:
|
|
262
|
-
return own # leaf ā identical to plain file_hash
|
|
263
|
-
seen_next = (_seen or frozenset()) | {relative}
|
|
264
|
-
parts = [own] + [effective_hash(dep, seen_next) for dep in deps]
|
|
265
|
-
return hashlib.sha256("\n".join(parts).encode("utf-8")).hexdigest()
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
def load_hashes() -> dict:
|
|
269
|
-
"""Load stored condensation hashes from JSON file."""
|
|
270
|
-
if HASH_FILE.exists():
|
|
271
|
-
try:
|
|
272
|
-
return json.loads(HASH_FILE.read_text())
|
|
273
|
-
except (json.JSONDecodeError, OSError):
|
|
274
|
-
return {}
|
|
275
|
-
return {}
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
def save_hashes(hashes: dict) -> None:
|
|
279
|
-
"""Save condensation hashes to JSON file."""
|
|
280
|
-
HASH_FILE.parent.mkdir(parents=True, exist_ok=True)
|
|
281
|
-
HASH_FILE.write_text(json.dumps(hashes, indent=2, sort_keys=True) + "\n")
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
def mark_done(relative_path: str) -> None:
|
|
285
|
-
"""Mark a single file as condensed by storing its current source hash.
|
|
286
|
-
|
|
287
|
-
Also runs the path rewriter on the just-written `dist/agent-src/<path>` so
|
|
288
|
-
logical names from the source frontmatter resolve to deployment-correct
|
|
289
|
-
relative paths in the shipped layer (P1 of road-to-path-fixes.md).
|
|
290
|
-
Idempotent ā re-running is a no-op.
|
|
291
|
-
"""
|
|
292
|
-
source_file = _resolve_source(relative_path)
|
|
293
|
-
if source_file is None or not source_file.exists():
|
|
294
|
-
print(f"ā Source file not found: {relative_path}")
|
|
295
|
-
sys.exit(1)
|
|
296
|
-
apply_path_rewriter(relative_path)
|
|
297
|
-
hashes = load_hashes()
|
|
298
|
-
hashes[relative_path] = effective_hash(relative_path)
|
|
299
|
-
save_hashes(hashes)
|
|
300
|
-
print(f"ā
Marked as condensed: {relative_path}")
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
def apply_path_rewriter(relative_path: str) -> bool:
|
|
304
|
-
"""Apply `_rewrite_paths` to `dist/agent-src/<relative_path>` in-place.
|
|
305
|
-
|
|
306
|
-
Returns True if the file was modified, False otherwise. Silently
|
|
307
|
-
returns False if the target doesn't exist (condensation hasn't run
|
|
308
|
-
yet) ā `--mark-done` is also valid before content exists.
|
|
309
|
-
"""
|
|
310
|
-
target = TARGET_DIR / relative_path
|
|
311
|
-
if not target.exists() or not relative_path.endswith(".md"):
|
|
312
|
-
return False
|
|
313
|
-
original = target.read_text(encoding="utf-8")
|
|
314
|
-
rewritten = _rewrite_paths(original, relative_path)
|
|
315
|
-
if rewritten == original:
|
|
316
|
-
return False
|
|
317
|
-
target.write_text(rewritten, encoding="utf-8")
|
|
318
|
-
return True
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
def mark_all_done() -> None:
|
|
322
|
-
"""Mark ALL .md files as condensed (e.g. after initial full condensation)."""
|
|
323
|
-
hashes = load_hashes()
|
|
324
|
-
count = 0
|
|
325
|
-
for source_file, relative in _iter_sources():
|
|
326
|
-
if not should_condense(source_file):
|
|
327
|
-
continue
|
|
328
|
-
hashes[relative] = effective_hash(relative)
|
|
329
|
-
count += 1
|
|
330
|
-
save_hashes(hashes)
|
|
331
|
-
print(f"ā
Marked {count} files as condensed")
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
def list_changed_md(source_dir: Path) -> list:
|
|
335
|
-
"""List .md files whose source hash differs from stored hash (= need recondensation).
|
|
336
|
-
|
|
337
|
-
The ``source_dir`` parameter is retained for backward compatibility but
|
|
338
|
-
ignored ā iteration walks every active source root (ADR-017).
|
|
339
|
-
"""
|
|
340
|
-
del source_dir # multi-root: ignored, kept for signature stability
|
|
341
|
-
hashes = load_hashes()
|
|
342
|
-
changed = []
|
|
343
|
-
for source_file, relative in _iter_sources():
|
|
344
|
-
if not should_condense(source_file):
|
|
345
|
-
continue
|
|
346
|
-
current_hash = effective_hash(relative)
|
|
347
|
-
stored_hash = hashes.get(relative)
|
|
348
|
-
if stored_hash != current_hash:
|
|
349
|
-
changed.append(relative)
|
|
350
|
-
return changed
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
def find_stale_hashes(source_dir: Path) -> list:
|
|
354
|
-
"""Find hashes stored for source files that no longer exist in any root."""
|
|
355
|
-
del source_dir # multi-root: ignored, kept for signature stability
|
|
356
|
-
hashes = load_hashes()
|
|
357
|
-
stale = []
|
|
358
|
-
for relative in sorted(hashes.keys()):
|
|
359
|
-
if _resolve_source(relative) is None:
|
|
360
|
-
stale.append(relative)
|
|
361
|
-
return stale
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
def clean_stale_hashes(source_dir: Path) -> int:
|
|
365
|
-
"""Remove hashes for source files that no longer exist. Returns count removed."""
|
|
366
|
-
stale = find_stale_hashes(source_dir)
|
|
367
|
-
if not stale:
|
|
368
|
-
return 0
|
|
369
|
-
hashes = load_hashes()
|
|
370
|
-
for relative in stale:
|
|
371
|
-
del hashes[relative]
|
|
372
|
-
save_hashes(hashes)
|
|
373
|
-
return len(stale)
|
|
374
|
-
|
|
375
|
-
|
|
376
|
-
|
|
377
|
-
def should_condense(filepath: Path) -> bool:
|
|
378
|
-
"""Check if file should be condensed (is .md and not in copy-as-is list)."""
|
|
379
|
-
if filepath.suffix != ".md":
|
|
380
|
-
return False
|
|
381
|
-
if filepath.name in COPY_AS_IS:
|
|
382
|
-
return False
|
|
383
|
-
# Determine the logical relative path so the COPY_AS_IS_DIRS check
|
|
384
|
-
# works for both legacy (`.agent-src.uncondensed/`) and post-move
|
|
385
|
-
# (`packages/*/.agent-src.uncondensed/`) source roots.
|
|
386
|
-
rel_parts: tuple[str, ...] = filepath.parts
|
|
387
|
-
for root in artefact_roots():
|
|
388
|
-
try:
|
|
389
|
-
rel_parts = filepath.relative_to(root).parts
|
|
390
|
-
break
|
|
391
|
-
except ValueError:
|
|
392
|
-
continue
|
|
393
|
-
if rel_parts and rel_parts[0] in COPY_AS_IS_DIRS:
|
|
394
|
-
return False
|
|
395
|
-
return True
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
def copy_file(source: Path, target: Path) -> None:
|
|
399
|
-
"""Copy a non-.md file as-is."""
|
|
400
|
-
target.parent.mkdir(parents=True, exist_ok=True)
|
|
401
|
-
shutil.copy2(source, target)
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
def cleanup_stale(source_dir: Path, target_dir: Path) -> int:
|
|
405
|
-
"""Delete files in target that don't exist in any source root. Returns count."""
|
|
406
|
-
del source_dir # multi-root: ignored, kept for signature stability
|
|
407
|
-
deleted = 0
|
|
408
|
-
if not target_dir.exists():
|
|
409
|
-
return 0
|
|
410
|
-
|
|
411
|
-
for target_file in sorted(target_dir.rglob("*")):
|
|
412
|
-
if target_file.is_dir():
|
|
413
|
-
continue
|
|
414
|
-
relative = target_file.relative_to(target_dir)
|
|
415
|
-
if _resolve_source(str(relative)) is None:
|
|
416
|
-
print(f" Deleting stale: {relative}")
|
|
417
|
-
target_file.unlink()
|
|
418
|
-
deleted += 1
|
|
419
|
-
|
|
420
|
-
# Remove empty directories
|
|
421
|
-
for dirpath in sorted(target_dir.rglob("*"), reverse=True):
|
|
422
|
-
if dirpath.is_dir() and not any(dirpath.iterdir()):
|
|
423
|
-
dirpath.rmdir()
|
|
424
|
-
print(f" Removing empty dir: {dirpath.relative_to(target_dir)}")
|
|
425
|
-
|
|
426
|
-
return deleted
|
|
427
|
-
|
|
428
|
-
|
|
429
|
-
def sync_non_md(source_dir: Path, target_dir: Path) -> int:
|
|
430
|
-
"""Copy all non-.md files (and COPY_AS_IS .md files) from every source
|
|
431
|
-
root to target. Returns count."""
|
|
432
|
-
del source_dir # multi-root: ignored, kept for signature stability
|
|
433
|
-
copied = 0
|
|
434
|
-
seen: set[str] = set()
|
|
435
|
-
for source_file, relative in _iter_sources():
|
|
436
|
-
if should_condense(source_file):
|
|
437
|
-
continue # .md files are condensed by the agent, not copied here
|
|
438
|
-
if relative in seen:
|
|
439
|
-
continue
|
|
440
|
-
seen.add(relative)
|
|
441
|
-
target_file = target_dir / relative
|
|
442
|
-
copy_file(source_file, target_file)
|
|
443
|
-
print(f" Copied: {relative}")
|
|
444
|
-
copied += 1
|
|
445
|
-
return copied
|
|
446
|
-
|
|
447
|
-
|
|
448
|
-
def list_md_files(source_dir: Path) -> list:
|
|
449
|
-
"""List all .md files that need condensation by the agent."""
|
|
450
|
-
del source_dir # multi-root: ignored, kept for signature stability
|
|
451
|
-
files: list[str] = []
|
|
452
|
-
seen: set[str] = set()
|
|
453
|
-
for source_file, relative in _iter_sources():
|
|
454
|
-
if not should_condense(source_file):
|
|
455
|
-
continue
|
|
456
|
-
if relative in seen:
|
|
457
|
-
continue
|
|
458
|
-
seen.add(relative)
|
|
459
|
-
files.append(relative)
|
|
460
|
-
return sorted(files)
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
def check_sync(source_dir: Path, target_dir: Path) -> tuple:
|
|
464
|
-
"""Check if target is in sync with source(s). Returns (missing, stale) lists."""
|
|
465
|
-
del source_dir # multi-root: ignored, kept for signature stability
|
|
466
|
-
missing = []
|
|
467
|
-
stale = []
|
|
468
|
-
|
|
469
|
-
# Files in any source root but not in target
|
|
470
|
-
seen: set[str] = set()
|
|
471
|
-
for _source_file, relative in _iter_sources():
|
|
472
|
-
if relative in seen:
|
|
473
|
-
continue
|
|
474
|
-
seen.add(relative)
|
|
475
|
-
if not (target_dir / relative).exists():
|
|
476
|
-
missing.append(relative)
|
|
477
|
-
|
|
478
|
-
# Files in target but not in any source root (stale)
|
|
479
|
-
if target_dir.exists():
|
|
480
|
-
for target_file in sorted(target_dir.rglob("*")):
|
|
481
|
-
if target_file.is_dir():
|
|
482
|
-
continue
|
|
483
|
-
relative = str(target_file.relative_to(target_dir))
|
|
484
|
-
if _resolve_source(relative) is None:
|
|
485
|
-
stale.append(relative)
|
|
486
|
-
|
|
487
|
-
return missing, stale
|
|
488
|
-
|
|
489
|
-
|
|
490
|
-
# āā Multi-agent tool generation āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
|
|
491
|
-
|
|
492
|
-
RULES_SOURCE = PROJECT_ROOT / "dist/agent-src" / "rules"
|
|
493
|
-
|
|
494
|
-
TOOL_DIRS = {
|
|
495
|
-
".claude/rules": "../../dist/agent-src/rules",
|
|
496
|
-
".cursor/rules": "../../dist/agent-src/rules",
|
|
497
|
-
".clinerules": "../dist/agent-src/rules",
|
|
498
|
-
}
|
|
499
|
-
|
|
500
|
-
SKILLS_SOURCE = PROJECT_ROOT / "dist/agent-src" / "skills"
|
|
501
|
-
COMMANDS_SOURCE = PROJECT_ROOT / "dist/agent-src" / "commands"
|
|
502
|
-
PERSONAS_SOURCE = PROJECT_ROOT / "dist/agent-src" / "personas"
|
|
503
|
-
USER_TYPES_SOURCE = PROJECT_ROOT / "dist/agent-src" / "user-types"
|
|
504
|
-
CLAUDE_SKILLS_DIR = PROJECT_ROOT / ".claude" / "skills"
|
|
505
|
-
# Committed plugin-marketplace projection for command-as-skill entries.
|
|
506
|
-
# The marketplace is consumed as a git repo, so every skills[] path must be
|
|
507
|
-
# committed. Real skills resolve to dist/agent-src/skills/<name> (already
|
|
508
|
-
# committed); commands have no <slug>/SKILL.md shape in source, so their
|
|
509
|
-
# committed projection lives here. .claude/skills/ stays a gitignored,
|
|
510
|
-
# generate-tools-rebuilt local auto-discovery channel.
|
|
511
|
-
PLUGIN_SKILLS_DIR = PROJECT_ROOT / ".claude-plugin" / "skills"
|
|
512
|
-
|
|
513
|
-
PERSONA_TOOL_DIRS = {
|
|
514
|
-
".claude/personas": "../../dist/agent-src/personas",
|
|
515
|
-
".cursor/personas": "../../dist/agent-src/personas",
|
|
516
|
-
}
|
|
517
|
-
|
|
518
|
-
USER_TYPE_TOOL_DIRS = {
|
|
519
|
-
".claude/user-types": "../../dist/agent-src/user-types",
|
|
520
|
-
".cursor/user-types": "../../dist/agent-src/user-types",
|
|
521
|
-
}
|
|
522
|
-
|
|
523
|
-
# Map tool-projection directories to the canonical tool ID used by
|
|
524
|
-
# `agents/.agent-tools.yml`. Directories not in this map are always emitted.
|
|
525
|
-
_DIR_TOOL_ID = {
|
|
526
|
-
".claude/rules": "claude-code",
|
|
527
|
-
".cursor/rules": "cursor",
|
|
528
|
-
".clinerules": "cline",
|
|
529
|
-
".claude/personas": "claude-code",
|
|
530
|
-
".cursor/personas": "cursor",
|
|
531
|
-
".claude/user-types": "claude-code",
|
|
532
|
-
".cursor/user-types": "cursor",
|
|
533
|
-
}
|
|
534
|
-
|
|
535
|
-
|
|
536
|
-
def _filter_tool_dirs(mapping: dict[str, str]) -> dict[str, str]:
|
|
537
|
-
"""Drop entries whose tool ID is not active in `agents/.agent-tools.yml`."""
|
|
538
|
-
return {
|
|
539
|
-
d: p for d, p in mapping.items()
|
|
540
|
-
if _tool_active(_DIR_TOOL_ID.get(d, "claude-code"))
|
|
541
|
-
}
|
|
542
|
-
|
|
543
|
-
|
|
544
|
-
def strip_frontmatter(content: str) -> str:
|
|
545
|
-
"""Remove YAML frontmatter (between --- markers) from content."""
|
|
546
|
-
if content.startswith("---"):
|
|
547
|
-
end = content.find("---", 3)
|
|
548
|
-
if end != -1:
|
|
549
|
-
content = content[end + 3:].lstrip("\n")
|
|
550
|
-
return content
|
|
551
|
-
|
|
552
|
-
|
|
553
|
-
# āā Path rewriter (P1 of road-to-path-fixes.md) āāāāāāāāāāāāāāāāāāāāāāāāāāā
|
|
554
|
-
# Source files use logical names that the rewriter resolves at condense
|
|
555
|
-
# time, so the shipped `dist/agent-src/` (and `.augment/` projection) carry
|
|
556
|
-
# deployment-correct relative paths without the agent author having to
|
|
557
|
-
# know how deep their file lives.
|
|
558
|
-
#
|
|
559
|
-
# Frontmatter rewrites:
|
|
560
|
-
# load_context: / load_context_eager:
|
|
561
|
-
# contexts/<area>/<file>.md (logical, preferred)
|
|
562
|
-
# .agent-src.uncondensed/contexts/<area>/<file>.md (legacy)
|
|
563
|
-
# ā ../contexts/<area>/<file>.md (relative from dist/agent-src/rules/)
|
|
564
|
-
# triggers[].path_prefix:
|
|
565
|
-
# LEFT ALONE ā `path_prefix:` is a literal match pattern, not a
|
|
566
|
-
# file reference. Source-of-truth rules that fire on edits under
|
|
567
|
-
# `.agent-src.uncondensed/` keep that prefix verbatim (see
|
|
568
|
-
# road-to-path-fixes.md P2.2 / Modified Option 1).
|
|
569
|
-
#
|
|
570
|
-
# Body-link rewrites:
|
|
571
|
-
# ../../docs/guidelines/<file>.md ā ../docs/guidelines/<file>.md
|
|
572
|
-
# ../../docs/contracts/<file>.md ā ../docs/contracts/<file>.md
|
|
573
|
-
#
|
|
574
|
-
# Idempotent: applying twice is a no-op (rewritten patterns no longer
|
|
575
|
-
# match the source patterns).
|
|
576
|
-
|
|
577
|
-
_LEGACY_SRC_PREFIX = ".agent-src.uncondensed/"
|
|
578
|
-
_PROJECTED_SRC_PREFIX = "dist/agent-src/"
|
|
579
|
-
# Pre-ADR-057 projected prefix ā still stripped defensively so content
|
|
580
|
-
# authored against the old root-level output keeps resolving.
|
|
581
|
-
_LEGACY_PROJECTED_SRC_PREFIX = ".agent-src/"
|
|
582
|
-
|
|
583
|
-
# A YAML list item under load_context*: ` - some/path.md` (optionally quoted)
|
|
584
|
-
_FM_LIST_ITEM_RE = re.compile(r'^(\s*-\s*)(["\']?)([^"\'\n]+?\.md)(["\']?)\s*$')
|
|
585
|
-
|
|
586
|
-
# `path_prefix:` line ā top-level or under `triggers:` (with leading dash)
|
|
587
|
-
_FM_PATH_PREFIX_RE = re.compile(
|
|
588
|
-
r'^(\s*(?:-\s+)?path_prefix:\s*)(["\']?)([^"\'\n]+?)(["\']?)\s*$'
|
|
589
|
-
)
|
|
590
|
-
|
|
591
|
-
# Body-link patterns (relative two-up to docs/) ā capture the docs/... tail
|
|
592
|
-
_BODY_DOCS_RE = re.compile(r'\.\./\.\./(docs/(?:guidelines|contracts)/[^)\s]+\.md)')
|
|
593
|
-
|
|
594
|
-
|
|
595
|
-
def _depth_prefix(source_relative_path: str) -> str:
|
|
596
|
-
"""Return the `../` chain to climb from `<source_relative_path>` back to
|
|
597
|
-
the source root. A file at `rules/X.md` (1 dir deep) needs `../`; a
|
|
598
|
-
file at `commands/council/default.md` (2 dirs deep) needs `../../`.
|
|
599
|
-
"""
|
|
600
|
-
parts = Path(source_relative_path).parts
|
|
601
|
-
depth = max(len(parts) - 1, 1)
|
|
602
|
-
return "../" * depth
|
|
603
|
-
|
|
604
|
-
|
|
605
|
-
def _split_frontmatter(content: str):
|
|
606
|
-
"""Return (frontmatter_lines, body) ā frontmatter_lines is None if no FM."""
|
|
607
|
-
if not content.startswith("---\n"):
|
|
608
|
-
return None, content
|
|
609
|
-
end = content.find("\n---\n", 4)
|
|
610
|
-
if end == -1:
|
|
611
|
-
return None, content
|
|
612
|
-
fm_text = content[4:end]
|
|
613
|
-
body = content[end + len("\n---\n"):]
|
|
614
|
-
return fm_text.split("\n"), body
|
|
615
|
-
|
|
616
|
-
|
|
617
|
-
def _rewrite_load_context_value(value: str, prefix: str) -> str:
|
|
618
|
-
"""Rewrite a single `load_context` list-item value to a deployment path."""
|
|
619
|
-
# Already relative or absolute ā leave alone (idempotence).
|
|
620
|
-
if value.startswith(("../", "./", "/")):
|
|
621
|
-
return value
|
|
622
|
-
# Legacy fully-qualified source prefix.
|
|
623
|
-
if value.startswith(_LEGACY_SRC_PREFIX):
|
|
624
|
-
return prefix + value[len(_LEGACY_SRC_PREFIX):]
|
|
625
|
-
# Projected source prefix (defensive ā also strip).
|
|
626
|
-
if value.startswith(_PROJECTED_SRC_PREFIX):
|
|
627
|
-
return prefix + value[len(_PROJECTED_SRC_PREFIX):]
|
|
628
|
-
if value.startswith(_LEGACY_PROJECTED_SRC_PREFIX):
|
|
629
|
-
return prefix + value[len(_LEGACY_PROJECTED_SRC_PREFIX):]
|
|
630
|
-
# Logical name (e.g. `contexts/execution/foo.md`).
|
|
631
|
-
return prefix + value
|
|
632
|
-
|
|
633
|
-
|
|
634
|
-
def _rewrite_path_prefix_value(value: str) -> str:
|
|
635
|
-
"""No-op for `triggers[].path_prefix:` values.
|
|
636
|
-
|
|
637
|
-
`path_prefix:` is a literal match pattern the host evaluates against
|
|
638
|
-
the file the agent is editing ā not a file reference. Rewriting it
|
|
639
|
-
breaks the workflow it was authored for: source-of-truth rules that
|
|
640
|
-
fire when the agent edits files under `.agent-src.uncondensed/`
|
|
641
|
-
keep that prefix verbatim. The prefix ban therefore applies only to
|
|
642
|
-
`load_context:` entries and body links (see road-to-path-fixes.md
|
|
643
|
-
P2.2 + the AI-Council convergence on 2026-05-06).
|
|
644
|
-
"""
|
|
645
|
-
return value
|
|
646
|
-
|
|
647
|
-
|
|
648
|
-
def _rewrite_frontmatter_lines(lines, prefix):
|
|
649
|
-
"""Apply load_context / path_prefix rewrites to a frontmatter line list."""
|
|
650
|
-
in_load_context = False
|
|
651
|
-
out = []
|
|
652
|
-
for line in lines:
|
|
653
|
-
bare = line.lstrip()
|
|
654
|
-
if bare.startswith(("load_context:", "load_context_eager:")):
|
|
655
|
-
in_load_context = True
|
|
656
|
-
out.append(line)
|
|
657
|
-
continue
|
|
658
|
-
if in_load_context:
|
|
659
|
-
m = _FM_LIST_ITEM_RE.match(line)
|
|
660
|
-
if m:
|
|
661
|
-
indent, q1, value, q2 = m.groups()
|
|
662
|
-
rewritten = _rewrite_load_context_value(value, prefix)
|
|
663
|
-
out.append(f"{indent}{q1}{rewritten}{q2}")
|
|
664
|
-
continue
|
|
665
|
-
in_load_context = False
|
|
666
|
-
# fall through to path_prefix / passthrough
|
|
667
|
-
m = _FM_PATH_PREFIX_RE.match(line)
|
|
668
|
-
if m:
|
|
669
|
-
head, q1, value, q2 = m.groups()
|
|
670
|
-
out.append(f"{head}{q1}{_rewrite_path_prefix_value(value)}{q2}")
|
|
671
|
-
continue
|
|
672
|
-
out.append(line)
|
|
673
|
-
return out
|
|
674
|
-
|
|
675
|
-
|
|
676
|
-
def _rewrite_body_links(body: str, prefix: str) -> str:
|
|
677
|
-
"""Rewrite `../../docs/{guidelines,contracts}/...` to use depth-prefix."""
|
|
678
|
-
return _BODY_DOCS_RE.sub(prefix + r"\1", body)
|
|
679
|
-
|
|
680
|
-
|
|
681
|
-
# āā Human-review banner injection (Phase 5.3 / ADR-018) āāāāāāāāāāāāāāāāāāā
|
|
682
|
-
# Source artefacts may set `trust.human_review_required: true` in their
|
|
683
|
-
# frontmatter. The condenseor injects a short, parser-stable banner block
|
|
684
|
-
# at the top of the projected body so every downstream surface (agent
|
|
685
|
-
# memory, .augment, .claude, etc.) surfaces the gate. Idempotent ā the
|
|
686
|
-
# marker comment prevents double-injection on re-condense.
|
|
687
|
-
|
|
688
|
-
_HRR_BANNER_MARKER = "<!-- agent-config:human-review-banner -->"
|
|
689
|
-
|
|
690
|
-
# Plain YAML list item ā any scalar, used to read `packs:` / `workspaces:`
|
|
691
|
-
# blocks where values are bare identifiers, not file paths.
|
|
692
|
-
_FM_PLAIN_LIST_RE = re.compile(r'^\s*-\s*(["\']?)([^"\'\n]+?)\1\s*$')
|
|
693
|
-
|
|
694
|
-
|
|
695
|
-
def _parse_trust_and_owner(fm_lines):
|
|
696
|
-
"""Extract `trust.level`, `human_review_required`, and an owner hint
|
|
697
|
-
from a frontmatter line list. Owner falls back to the first pack
|
|
698
|
-
prefix (`finance-basic` ā `finance`), then the first workspace,
|
|
699
|
-
then `unknown`.
|
|
700
|
-
"""
|
|
701
|
-
# Default to the schema default `core` so the HUMAN REVIEW banner still
|
|
702
|
-
# renders if a future artefact sets `human_review_required: true` while
|
|
703
|
-
# omitting the (defaulted) `trust.level` line (abstraction-reduction
|
|
704
|
-
# preflight Decision D). The banner only fires when `hrr` is true, so this
|
|
705
|
-
# never affects artefacts that omit the whole trust block.
|
|
706
|
-
level = "core"
|
|
707
|
-
hrr = False
|
|
708
|
-
packs: list[str] = []
|
|
709
|
-
workspaces: list[str] = []
|
|
710
|
-
in_trust = False
|
|
711
|
-
in_packs = False
|
|
712
|
-
in_workspaces = False
|
|
713
|
-
for line in fm_lines:
|
|
714
|
-
stripped = line.lstrip()
|
|
715
|
-
indent = len(line) - len(stripped)
|
|
716
|
-
if indent == 0 and stripped.endswith(":"):
|
|
717
|
-
key = stripped[:-1]
|
|
718
|
-
in_trust = key == "trust"
|
|
719
|
-
in_packs = key == "packs"
|
|
720
|
-
in_workspaces = key == "workspaces"
|
|
721
|
-
continue
|
|
722
|
-
if in_trust and stripped.startswith("level:"):
|
|
723
|
-
level = stripped.split(":", 1)[1].strip().strip('"').strip("'")
|
|
724
|
-
elif in_trust and stripped.startswith("human_review_required:"):
|
|
725
|
-
val = stripped.split(":", 1)[1].strip()
|
|
726
|
-
hrr = val.lower() == "true"
|
|
727
|
-
elif in_packs or in_workspaces:
|
|
728
|
-
m = _FM_PLAIN_LIST_RE.match(line)
|
|
729
|
-
if m:
|
|
730
|
-
value = m.group(2).strip()
|
|
731
|
-
(packs if in_packs else workspaces).append(value)
|
|
732
|
-
owner = "unknown"
|
|
733
|
-
if packs:
|
|
734
|
-
owner = packs[0].split("-")[0]
|
|
735
|
-
elif workspaces:
|
|
736
|
-
owner = workspaces[0]
|
|
737
|
-
return level, hrr, owner
|
|
738
|
-
|
|
739
|
-
|
|
740
|
-
def _inject_hrr_banner(body: str, level: str, owner: str) -> str:
|
|
741
|
-
"""Prepend the HUMAN_REVIEW banner block to `body`. Idempotent ā a
|
|
742
|
-
body that already carries `_HRR_BANNER_MARKER` is returned unchanged.
|
|
743
|
-
"""
|
|
744
|
-
if _HRR_BANNER_MARKER in body:
|
|
745
|
-
return body
|
|
746
|
-
banner = (
|
|
747
|
-
f"{_HRR_BANNER_MARKER}\n"
|
|
748
|
-
f"> HUMAN REVIEW REQUIRED Ā· trust: {level} Ā· owner: {owner}\n\n"
|
|
749
|
-
)
|
|
750
|
-
return banner + body.lstrip("\n")
|
|
751
|
-
|
|
752
|
-
|
|
753
|
-
def _rewrite_paths(content: str, source_relative_path: str) -> str:
|
|
754
|
-
"""Rewrite logical / legacy paths in `content` for a file shipped at
|
|
755
|
-
`dist/agent-src/{source_relative_path}`. Idempotent.
|
|
756
|
-
|
|
757
|
-
See module-level comment above for the full pattern catalog.
|
|
758
|
-
Also injects the HUMAN_REVIEW banner when the source frontmatter
|
|
759
|
-
sets `trust.human_review_required: true` (Phase 5.3 / ADR-018).
|
|
760
|
-
"""
|
|
761
|
-
prefix = _depth_prefix(source_relative_path)
|
|
762
|
-
fm_lines, body = _split_frontmatter(content)
|
|
763
|
-
body = _rewrite_body_links(body, prefix)
|
|
764
|
-
if fm_lines is None:
|
|
765
|
-
return body
|
|
766
|
-
new_fm = _rewrite_frontmatter_lines(fm_lines, prefix)
|
|
767
|
-
level, hrr, owner = _parse_trust_and_owner(fm_lines)
|
|
768
|
-
if hrr and level:
|
|
769
|
-
body = _inject_hrr_banner(body, level, owner)
|
|
770
|
-
return "---\n" + "\n".join(new_fm) + "\n---\n" + body
|
|
771
|
-
|
|
772
|
-
|
|
773
|
-
def generate_rule_symlinks() -> int:
|
|
774
|
-
"""Create symlink directories for rules (.claude/rules/, .cursor/rules/, .clinerules/).
|
|
775
|
-
|
|
776
|
-
Symlinks ALL .md files from dist/agent-src/rules/ into tool-specific directories.
|
|
777
|
-
"""
|
|
778
|
-
# All .md files in dist/agent-src/rules/ ā not just universal ones
|
|
779
|
-
rules = sorted([f.name for f in RULES_SOURCE.glob("*.md")])
|
|
780
|
-
tool_dirs = _filter_tool_dirs(TOOL_DIRS)
|
|
781
|
-
|
|
782
|
-
# Thin-projection opt-in (lean-initial-context Phase 3.1). Default
|
|
783
|
-
# `eager-all` keeps the symlink behaviour below untouched; `thin` writes
|
|
784
|
-
# kernel rules full + non-kernel rules as router-resolved pointers.
|
|
785
|
-
thin_files: dict[str, str] | None = None
|
|
786
|
-
if _lean_projection_mode() == "thin":
|
|
787
|
-
try:
|
|
788
|
-
from scripts.project_thin_rules import build_thin
|
|
789
|
-
except ImportError: # pragma: no cover ā script-style invocation
|
|
790
|
-
from project_thin_rules import build_thin # type: ignore[import-not-found]
|
|
791
|
-
thin_files = build_thin(RULES_SOURCE)
|
|
792
|
-
|
|
793
|
-
total = 0
|
|
794
|
-
for tool_dir, rel_prefix in tool_dirs.items():
|
|
795
|
-
target_dir = PROJECT_ROOT / tool_dir
|
|
796
|
-
target_dir.mkdir(parents=True, exist_ok=True)
|
|
797
|
-
|
|
798
|
-
# Clean stale symlinks
|
|
799
|
-
for item in target_dir.iterdir():
|
|
800
|
-
if item.is_symlink() and item.name not in rules and item.name != "README.md":
|
|
801
|
-
item.unlink()
|
|
802
|
-
|
|
803
|
-
for rule in rules:
|
|
804
|
-
link = target_dir / rule
|
|
805
|
-
if link.exists() or link.is_symlink():
|
|
806
|
-
link.unlink()
|
|
807
|
-
if thin_files is not None:
|
|
808
|
-
# Thin mode: write a real file (kernel full / non-kernel pointer),
|
|
809
|
-
# not a symlink to the full source body.
|
|
810
|
-
link.write_text(thin_files[rule], encoding="utf-8")
|
|
811
|
-
else:
|
|
812
|
-
link.symlink_to(Path(rel_prefix) / rule)
|
|
813
|
-
total += 1
|
|
814
|
-
|
|
815
|
-
# Verify counts match across all tool directories
|
|
816
|
-
source_count = len(rules)
|
|
817
|
-
for tool_dir in tool_dirs:
|
|
818
|
-
target_dir = PROJECT_ROOT / tool_dir
|
|
819
|
-
tool_count = len([f for f in target_dir.iterdir() if f.suffix == ".md"])
|
|
820
|
-
if tool_count != source_count:
|
|
821
|
-
print(f" ā ļø {tool_dir}: {tool_count} rules (expected {source_count})")
|
|
822
|
-
|
|
823
|
-
info(f" ā
Created {total} rule symlinks across {len(tool_dirs)} tool directories ({source_count} rules each)")
|
|
824
|
-
return total
|
|
825
|
-
|
|
826
|
-
|
|
827
|
-
def generate_windsurfrules() -> int:
|
|
828
|
-
"""Generate .windsurfrules by concatenating all rules (no frontmatter).
|
|
829
|
-
"""
|
|
830
|
-
rules = sorted([f.name for f in RULES_SOURCE.glob("*.md")])
|
|
831
|
-
parts = ["# Auto-generated from dist/agent-src/rules/ ā do not edit directly\n"]
|
|
832
|
-
|
|
833
|
-
for rule in rules:
|
|
834
|
-
path = RULES_SOURCE / rule
|
|
835
|
-
content = strip_frontmatter(path.read_text())
|
|
836
|
-
parts.append(f"---\n\n{content.strip()}\n")
|
|
837
|
-
|
|
838
|
-
output = PROJECT_ROOT / ".windsurfrules"
|
|
839
|
-
output.write_text("\n".join(parts) + "\n")
|
|
840
|
-
info(f" ā
Generated .windsurfrules ({len(rules)} rules)")
|
|
841
|
-
return len(rules)
|
|
842
|
-
|
|
843
|
-
|
|
844
|
-
# āā Modern editor formats (road-to-simplicity-and-everywhere Phase 5) ā
|
|
845
|
-
# Cursor `.cursor/rules/*.mdc` (frontmatter: description, globs,
|
|
846
|
-
# alwaysApply) and Windsurf `.windsurf/rules/*.md` (frontmatter:
|
|
847
|
-
# trigger, description, globs) are the formats current editors prefer.
|
|
848
|
-
# Legacy `.windsurfrules` aggregate stays for users who prefer it.
|
|
849
|
-
|
|
850
|
-
CURSOR_RULES_MDC_DIR = PROJECT_ROOT / ".cursor" / "rules"
|
|
851
|
-
WINDSURF_RULES_DIR = PROJECT_ROOT / ".windsurf" / "rules"
|
|
852
|
-
WINDSURF_WORKFLOWS_DIR = PROJECT_ROOT / ".windsurf" / "workflows"
|
|
853
|
-
CURSOR_COMMANDS_DIR = PROJECT_ROOT / ".cursor" / "commands"
|
|
854
|
-
|
|
855
|
-
|
|
856
|
-
def _parse_frontmatter(content: str) -> tuple[dict, str]:
|
|
857
|
-
"""Split a `---`-delimited YAML frontmatter from the body."""
|
|
858
|
-
if not content.startswith("---"):
|
|
859
|
-
return {}, content
|
|
860
|
-
end = content.find("\n---", 3)
|
|
861
|
-
if end == -1:
|
|
862
|
-
return {}, content
|
|
863
|
-
raw = content[3:end].strip()
|
|
864
|
-
body = content[end + 4:].lstrip("\n")
|
|
865
|
-
try:
|
|
866
|
-
meta = yaml.safe_load(raw) or {}
|
|
867
|
-
except yaml.YAMLError:
|
|
868
|
-
meta = {}
|
|
869
|
-
return meta if isinstance(meta, dict) else {}, body
|
|
870
|
-
|
|
871
|
-
|
|
872
|
-
def _yaml_scalar(value: str) -> str:
|
|
873
|
-
"""Return a YAML-safe single-line scalar for a frontmatter value.
|
|
874
|
-
|
|
875
|
-
Descriptions can contain ``:``, ``#``, or quotes ā characters that break
|
|
876
|
-
an unquoted YAML scalar (e.g. ``description: Hard Floor: ...`` is invalid
|
|
877
|
-
YAML). A JSON string is itself a valid YAML double-quoted scalar, so
|
|
878
|
-
``json.dumps`` gives correct escaping for free.
|
|
879
|
-
"""
|
|
880
|
-
return json.dumps(value, ensure_ascii=False)
|
|
881
|
-
|
|
882
|
-
|
|
883
|
-
def _emit_cursor_mdc(source: Path, target: Path) -> None:
|
|
884
|
-
"""Write a Cursor `.mdc` file with Cursor-shaped frontmatter."""
|
|
885
|
-
meta, body = _parse_frontmatter(source.read_text())
|
|
886
|
-
description = (meta.get("description") or "").replace("\n", " ").strip()
|
|
887
|
-
always_apply = bool(meta.get("alwaysApply") or meta.get("type") == "always")
|
|
888
|
-
lines = [
|
|
889
|
-
"---",
|
|
890
|
-
f"description: {_yaml_scalar(description)}",
|
|
891
|
-
"globs: ",
|
|
892
|
-
f"alwaysApply: {'true' if always_apply else 'false'}",
|
|
893
|
-
"---",
|
|
894
|
-
"",
|
|
895
|
-
body.rstrip() + "\n",
|
|
896
|
-
]
|
|
897
|
-
target.parent.mkdir(parents=True, exist_ok=True)
|
|
898
|
-
target.write_text("\n".join(lines))
|
|
899
|
-
|
|
900
|
-
|
|
901
|
-
def _emit_windsurf_rule(source: Path, target: Path) -> None:
|
|
902
|
-
"""Write a Windsurf rule with Wave-8 frontmatter (trigger/description/globs)."""
|
|
903
|
-
meta, body = _parse_frontmatter(source.read_text())
|
|
904
|
-
description = (meta.get("description") or "").replace("\n", " ").strip()
|
|
905
|
-
always_apply = bool(meta.get("alwaysApply") or meta.get("type") == "always")
|
|
906
|
-
trigger = "always_on" if always_apply else "model_decision"
|
|
907
|
-
lines = [
|
|
908
|
-
"---",
|
|
909
|
-
f"trigger: {trigger}",
|
|
910
|
-
f"description: {_yaml_scalar(description)}",
|
|
911
|
-
"globs: ",
|
|
912
|
-
"---",
|
|
913
|
-
"",
|
|
914
|
-
body.rstrip() + "\n",
|
|
915
|
-
]
|
|
916
|
-
target.parent.mkdir(parents=True, exist_ok=True)
|
|
917
|
-
target.write_text("\n".join(lines))
|
|
918
|
-
|
|
919
|
-
|
|
920
|
-
def _clean_modern_dir(target_dir: Path, valid_names: set[str]) -> None:
|
|
921
|
-
"""Drop files in `target_dir` whose names are not in `valid_names`."""
|
|
922
|
-
if not target_dir.exists():
|
|
923
|
-
return
|
|
924
|
-
for item in target_dir.iterdir():
|
|
925
|
-
if item.name == "README.md":
|
|
926
|
-
continue
|
|
927
|
-
if item.name not in valid_names:
|
|
928
|
-
if item.is_dir() and not item.is_symlink():
|
|
929
|
-
shutil.rmtree(item)
|
|
930
|
-
else:
|
|
931
|
-
item.unlink()
|
|
932
|
-
|
|
933
|
-
|
|
934
|
-
def generate_cursor_mdc_rules() -> int:
|
|
935
|
-
"""Emit `.cursor/rules/*.mdc` per source rule (alongside legacy `.md` symlinks)."""
|
|
936
|
-
rules = sorted(RULES_SOURCE.glob("*.md"))
|
|
937
|
-
valid = {f"{r.stem}.mdc" for r in rules}
|
|
938
|
-
_clean_modern_dir(CURSOR_RULES_MDC_DIR, valid | {r.name for r in rules})
|
|
939
|
-
for rule in rules:
|
|
940
|
-
_emit_cursor_mdc(rule, CURSOR_RULES_MDC_DIR / f"{rule.stem}.mdc")
|
|
941
|
-
info(f" ā
Wrote {len(rules)} `.cursor/rules/*.mdc` files")
|
|
942
|
-
return len(rules)
|
|
943
|
-
|
|
944
|
-
|
|
945
|
-
def generate_windsurf_modern_rules() -> int:
|
|
946
|
-
"""Emit `.windsurf/rules/*.md` per source rule (Wave-8 frontmatter)."""
|
|
947
|
-
rules = sorted(RULES_SOURCE.glob("*.md"))
|
|
948
|
-
valid = {r.name for r in rules}
|
|
949
|
-
_clean_modern_dir(WINDSURF_RULES_DIR, valid)
|
|
950
|
-
for rule in rules:
|
|
951
|
-
_emit_windsurf_rule(rule, WINDSURF_RULES_DIR / rule.name)
|
|
952
|
-
info(f" ā
Wrote {len(rules)} `.windsurf/rules/*.md` files")
|
|
953
|
-
return len(rules)
|
|
954
|
-
|
|
955
|
-
|
|
956
|
-
def generate_cursor_commands(active_command_slugs: set[str] | None = None) -> int:
|
|
957
|
-
"""Symlink `.cursor/commands/<slug>.md` per source command.
|
|
958
|
-
|
|
959
|
-
`active_command_slugs` is `None` for legacy-all (every command projected)
|
|
960
|
-
or a slug allowlist for scoped projection; the `_clean_modern_dir` pass
|
|
961
|
-
then reaps any command no longer in the active set.
|
|
962
|
-
"""
|
|
963
|
-
if not (PROJECT_ROOT / "src" / "domains").is_dir():
|
|
964
|
-
return 0
|
|
965
|
-
cmds = [
|
|
966
|
-
(sf, slug)
|
|
967
|
-
for sf, slug in _iter_commands()
|
|
968
|
-
if active_command_slugs is None or slug in active_command_slugs
|
|
969
|
-
]
|
|
970
|
-
valid = {f"{slug}.md" for _, slug in cmds}
|
|
971
|
-
_clean_modern_dir(CURSOR_COMMANDS_DIR, valid)
|
|
972
|
-
CURSOR_COMMANDS_DIR.mkdir(parents=True, exist_ok=True)
|
|
973
|
-
count = 0
|
|
974
|
-
for source_file, slug in cmds:
|
|
975
|
-
link = CURSOR_COMMANDS_DIR / f"{slug}.md"
|
|
976
|
-
if link.exists() or link.is_symlink():
|
|
977
|
-
link.unlink()
|
|
978
|
-
rel = Path("../..") / source_file.relative_to(PROJECT_ROOT)
|
|
979
|
-
link.symlink_to(rel)
|
|
980
|
-
count += 1
|
|
981
|
-
info(f" ā
Linked {count} `.cursor/commands/*.md` files")
|
|
982
|
-
return count
|
|
983
|
-
|
|
984
|
-
|
|
985
|
-
def generate_windsurf_workflows(active_command_slugs: set[str] | None = None) -> int:
|
|
986
|
-
"""Symlink `.windsurf/workflows/<slug>.md` per source command.
|
|
987
|
-
|
|
988
|
-
`active_command_slugs` is `None` for legacy-all or a slug allowlist for
|
|
989
|
-
scoped projection (see `generate_cursor_commands`).
|
|
990
|
-
"""
|
|
991
|
-
if not (PROJECT_ROOT / "src" / "domains").is_dir():
|
|
992
|
-
return 0
|
|
993
|
-
cmds = [
|
|
994
|
-
(sf, slug)
|
|
995
|
-
for sf, slug in _iter_commands()
|
|
996
|
-
if active_command_slugs is None or slug in active_command_slugs
|
|
997
|
-
]
|
|
998
|
-
valid = {f"{slug}.md" for _, slug in cmds}
|
|
999
|
-
_clean_modern_dir(WINDSURF_WORKFLOWS_DIR, valid)
|
|
1000
|
-
WINDSURF_WORKFLOWS_DIR.mkdir(parents=True, exist_ok=True)
|
|
1001
|
-
count = 0
|
|
1002
|
-
for source_file, slug in cmds:
|
|
1003
|
-
link = WINDSURF_WORKFLOWS_DIR / f"{slug}.md"
|
|
1004
|
-
if link.exists() or link.is_symlink():
|
|
1005
|
-
link.unlink()
|
|
1006
|
-
rel = Path("../..") / source_file.relative_to(PROJECT_ROOT)
|
|
1007
|
-
link.symlink_to(rel)
|
|
1008
|
-
count += 1
|
|
1009
|
-
info(f" ā
Linked {count} `.windsurf/workflows/*.md` files")
|
|
1010
|
-
return count
|
|
1011
|
-
|
|
1012
|
-
|
|
1013
|
-
|
|
1014
|
-
|
|
1015
|
-
def generate_gemini_md() -> None:
|
|
1016
|
-
"""Create GEMINI.md symlink to AGENTS.md."""
|
|
1017
|
-
link = PROJECT_ROOT / "GEMINI.md"
|
|
1018
|
-
if link.exists() or link.is_symlink():
|
|
1019
|
-
link.unlink()
|
|
1020
|
-
link.symlink_to("AGENTS.md")
|
|
1021
|
-
info(" ā
Created GEMINI.md ā AGENTS.md symlink")
|
|
1022
|
-
|
|
1023
|
-
|
|
1024
|
-
def _command_slug(source_file: Path) -> str:
|
|
1025
|
-
"""Return the flat .claude/skills/ slug for a command source file.
|
|
1026
|
-
|
|
1027
|
-
Top-level commands keep their stem (`commit.md` ā `commit`). Nested
|
|
1028
|
-
commands flatten the relative path with `-` (`council/default.md` ā
|
|
1029
|
-
`council-default`). Keeps slug collisions out of `.claude/skills/`
|
|
1030
|
-
while preserving native nested invocation in `dist/agent-src/commands/`.
|
|
1031
|
-
"""
|
|
1032
|
-
rel = source_file.relative_to(COMMANDS_SOURCE)
|
|
1033
|
-
return "-".join(rel.with_suffix("").parts)
|
|
1034
|
-
|
|
1035
|
-
|
|
1036
|
-
def _iter_commands():
|
|
1037
|
-
"""Yield (source_file, slug) for every command, sourced from `src/domains/`.
|
|
1038
|
-
|
|
1039
|
-
Repointed (ADR-044 amendment, 2026-06-04) from the legacy
|
|
1040
|
-
`dist/agent-src/commands/` tree to the canonical
|
|
1041
|
-
`src/domains/<pack>/<verb>/command.md` source via
|
|
1042
|
-
`_lib.agent_src.iter_commands()`. The slug is the pack-stripped hyphenated
|
|
1043
|
-
subpath (`_command_path_to_slug`) ā the path is the single source of truth
|
|
1044
|
-
per the amendment. `source_file` is the physical `src/domains/` path; every
|
|
1045
|
-
consumer builds its symlink target relative to `PROJECT_ROOT`, not
|
|
1046
|
-
`COMMANDS_SOURCE`.
|
|
1047
|
-
"""
|
|
1048
|
-
src_domains = PROJECT_ROOT / "src" / "domains"
|
|
1049
|
-
if not src_domains.is_dir():
|
|
1050
|
-
return
|
|
1051
|
-
for source_file in sorted(src_domains.rglob("command.md")):
|
|
1052
|
-
rel = source_file.relative_to(PROJECT_ROOT).as_posix()
|
|
1053
|
-
slug = _command_path_to_slug(rel)
|
|
1054
|
-
if not slug:
|
|
1055
|
-
continue
|
|
1056
|
-
# ADR-044 amendment A3: a pack may opt into a slug prefix; the path
|
|
1057
|
-
# stays the single source of truth (PROJECT_ROOT-aware for tests that
|
|
1058
|
-
# monkey-patch the root).
|
|
1059
|
-
pack_id = source_file.relative_to(src_domains).parts[0]
|
|
1060
|
-
prefix = _domains_slug_prefix(pack_id)
|
|
1061
|
-
if prefix and slug != prefix and not slug.startswith(prefix + "-"):
|
|
1062
|
-
slug = f"{prefix}-{slug}"
|
|
1063
|
-
yield source_file, slug
|
|
1064
|
-
|
|
1065
|
-
|
|
1066
|
-
# --- Per-skill model auto-switch (ADR-035 / road-to-model-capability-tiers) ---
|
|
1067
|
-
|
|
1068
|
-
# The SINGLE generator-owned tierāClaude-model mapping (ADR-035 § 3). Claude
|
|
1069
|
-
# Code is the only surface that consumes a native `model:`, so this is the only
|
|
1070
|
-
# place a capability tier resolves to a concrete model. `inherit`/absent emit
|
|
1071
|
-
# nothing and stay pure symlinks. Non-Claude agents never get a per-vendor
|
|
1072
|
-
# table ā the rule surfaces the tier name as a suggestion.
|
|
1073
|
-
# Tierāmodel mapping + frontmatter regex live in the shared _lib module so the
|
|
1074
|
-
# repo generator (here) and the consumer install finalizer
|
|
1075
|
-
# (install.py::finalize_claude_model_tiers) can never drift (ADR-034/035).
|
|
1076
|
-
from _lib.model_tier import ( # noqa: E402
|
|
1077
|
-
TIER_TO_CLAUDE_MODEL as _TIER_TO_CLAUDE_MODEL,
|
|
1078
|
-
MODEL_TIER_RE as _MODEL_TIER_RE,
|
|
1079
|
-
)
|
|
1080
|
-
|
|
1081
|
-
|
|
1082
|
-
def _read_model_auto_switch() -> str:
|
|
1083
|
-
"""Read `model.auto_switch` from .agent-settings.yml.
|
|
1084
|
-
|
|
1085
|
-
Returns `auto` | `suggest` | `off`; default `suggest`. Only `auto` makes
|
|
1086
|
-
the generator emit a native Claude `model:` key (ADR-034 § 4) ā `suggest`
|
|
1087
|
-
and `off` keep skills/commands as pure symlinks so the package never
|
|
1088
|
-
silently overrides a user's explicit `/model` choice.
|
|
1089
|
-
"""
|
|
1090
|
-
try:
|
|
1091
|
-
from scripts._lib.agent_settings import load_agent_settings
|
|
1092
|
-
except ImportError: # pragma: no cover ā script-style invocation
|
|
1093
|
-
import sys as _sys
|
|
1094
|
-
from pathlib import Path as _Path
|
|
1095
|
-
_sys.path.insert(0, str(_Path(__file__).resolve().parent))
|
|
1096
|
-
from _lib.agent_settings import load_agent_settings # type: ignore[import-not-found]
|
|
1097
|
-
data = load_agent_settings(project_path=SETTINGS_FILE)
|
|
1098
|
-
model = data.get("model")
|
|
1099
|
-
value = model.get("auto_switch") if isinstance(model, dict) else None
|
|
1100
|
-
if isinstance(value, str) and value.strip().lower() in ("auto", "suggest", "off"):
|
|
1101
|
-
return value.strip().lower()
|
|
1102
|
-
return "suggest"
|
|
1103
|
-
|
|
1104
|
-
|
|
1105
|
-
# --- Pack-scoped projection (ADR-040 / road-to-6.0.0-B Step 8) ---------------
|
|
1106
|
-
|
|
1107
|
-
|
|
1108
|
-
def _read_projection_mode() -> str:
|
|
1109
|
-
"""Read `projection.mode` from .agent-settings.yml.
|
|
1110
|
-
|
|
1111
|
-
Returns `legacy-all` | `scoped`; default `legacy-all`. Only `scoped`
|
|
1112
|
-
narrows the projected artefact set; `legacy-all` projects the full surface
|
|
1113
|
-
byte-identically to 5.x. Scoping is opt-in by this key alone ā never
|
|
1114
|
-
inferred from `profile.id`.
|
|
1115
|
-
"""
|
|
1116
|
-
try:
|
|
1117
|
-
from scripts._lib.agent_settings import load_agent_settings
|
|
1118
|
-
except ImportError: # pragma: no cover ā script-style invocation
|
|
1119
|
-
import sys as _sys
|
|
1120
|
-
from pathlib import Path as _Path
|
|
1121
|
-
_sys.path.insert(0, str(_Path(__file__).resolve().parent))
|
|
1122
|
-
from _lib.agent_settings import load_agent_settings # type: ignore[import-not-found]
|
|
1123
|
-
data = load_agent_settings(project_path=SETTINGS_FILE)
|
|
1124
|
-
proj = data.get("projection")
|
|
1125
|
-
value = proj.get("mode") if isinstance(proj, dict) else None
|
|
1126
|
-
if isinstance(value, str) and value.strip().lower() in ("legacy-all", "scoped"):
|
|
1127
|
-
return value.strip().lower()
|
|
1128
|
-
return "legacy-all"
|
|
1129
|
-
|
|
1130
|
-
|
|
1131
|
-
def _domains_slug_prefix(pack_id: str) -> str:
|
|
1132
|
-
"""Return ``slug_prefix`` from ``PROJECT_ROOT/src/domains/<pack>/pack.yaml``.
|
|
1133
|
-
|
|
1134
|
-
PROJECT_ROOT-aware mirror of ``_lib.agent_src.pack_slug_prefix`` so tests
|
|
1135
|
-
that monkey-patch ``condense.PROJECT_ROOT`` see their fixture's manifest.
|
|
1136
|
-
Empty string = no prefix (the default). ADR-044 amendment A3.
|
|
1137
|
-
"""
|
|
1138
|
-
manifest = PROJECT_ROOT / "src" / "domains" / pack_id / "pack.yaml"
|
|
1139
|
-
if not manifest.is_file():
|
|
1140
|
-
return ""
|
|
1141
|
-
m = _SLUG_PREFIX_RE.search(manifest.read_text(encoding="utf-8"))
|
|
1142
|
-
return m.group(1) if m else ""
|
|
1143
|
-
|
|
1144
|
-
|
|
1145
|
-
def _command_path_to_slug(manifest_path: str) -> str:
|
|
1146
|
-
"""Map a manifest command path to the flat generator slug.
|
|
1147
|
-
|
|
1148
|
-
`.../commands/council/analysis.md` ā `council-analysis` ā identical to
|
|
1149
|
-
`_command_slug()` (which flattens the path relative to COMMANDS_SOURCE),
|
|
1150
|
-
so the predicate matches what the generators emit.
|
|
1151
|
-
|
|
1152
|
-
Layout-agnostic: a 6.0.0-D domains command
|
|
1153
|
-
(`src/domains/<pack>/council/analysis/command.md`) carries no literal
|
|
1154
|
-
``commands`` path segment, so derive the slug from the LOGICAL command
|
|
1155
|
-
path (`commands/<subpath>.md`) via ``strip_source_prefix``. Falls back to
|
|
1156
|
-
the literal-segment scan for already-logical / projection paths.
|
|
1157
|
-
"""
|
|
1158
|
-
logical = strip_source_prefix(manifest_path)
|
|
1159
|
-
if logical and logical.startswith("commands/"):
|
|
1160
|
-
return "-".join(Path(logical[len("commands/"):]).with_suffix("").parts)
|
|
1161
|
-
parts = Path(manifest_path).parts
|
|
1162
|
-
i = parts.index("commands")
|
|
1163
|
-
return "-".join(Path(*parts[i + 1:]).with_suffix("").parts)
|
|
1164
|
-
|
|
1165
|
-
|
|
1166
|
-
def _skill_path_to_name(manifest_path: str) -> str:
|
|
1167
|
-
"""Map a manifest skill path to its directory name.
|
|
1168
|
-
|
|
1169
|
-
`.../skills/accessibility-auditor/SKILL.md` ā `accessibility-auditor` ā
|
|
1170
|
-
the key the skills generators iterate on.
|
|
1171
|
-
"""
|
|
1172
|
-
parts = Path(manifest_path).parts
|
|
1173
|
-
i = parts.index("skills")
|
|
1174
|
-
return parts[i + 1]
|
|
1175
|
-
|
|
1176
|
-
|
|
1177
|
-
def _resolve_active_predicates() -> tuple[set[str] | None, set[str] | None]:
|
|
1178
|
-
"""Return `(active_command_slugs, active_skill_names)` for the projection.
|
|
1179
|
-
|
|
1180
|
-
`(None, None)` ā legacy-all (no filtering ā byte-identical to 5.x). In
|
|
1181
|
-
`scoped` mode the selected packs are the active profile's `packs` UNIONED
|
|
1182
|
-
with the `runtime.active_packs` overlay; the resolver expands that over the
|
|
1183
|
-
`requires` graph and resolves the active command/skill set.
|
|
1184
|
-
"""
|
|
1185
|
-
if _read_projection_mode() != "scoped":
|
|
1186
|
-
return None, None
|
|
1187
|
-
try:
|
|
1188
|
-
from scripts.config import packs as packs_mod
|
|
1189
|
-
from scripts.config import session_profiles as sp_mod
|
|
1190
|
-
from scripts.config.profiles import resolve_profile
|
|
1191
|
-
from scripts._lib.agent_settings import load_agent_settings
|
|
1192
|
-
except ImportError: # pragma: no cover ā script-style invocation
|
|
1193
|
-
import sys as _sys
|
|
1194
|
-
_sys.path.insert(0, str(PROJECT_ROOT))
|
|
1195
|
-
from scripts.config import packs as packs_mod # type: ignore
|
|
1196
|
-
from scripts.config import session_profiles as sp_mod # type: ignore
|
|
1197
|
-
from scripts.config.profiles import resolve_profile # type: ignore
|
|
1198
|
-
from scripts._lib.agent_settings import load_agent_settings # type: ignore
|
|
1199
|
-
settings = load_agent_settings(project_path=SETTINGS_FILE)
|
|
1200
|
-
profile = resolve_profile(project_root=PROJECT_ROOT, user_settings=settings)
|
|
1201
|
-
selected = sorted(set(profile.packs) | set(sp_mod.read_overlay(PROJECT_ROOT)))
|
|
1202
|
-
active = packs_mod.resolve_active_set(PROJECT_ROOT, selected)
|
|
1203
|
-
cmd_slugs = {_command_path_to_slug(p) for p in active.commands}
|
|
1204
|
-
skill_names = {_skill_path_to_name(p) for p in active.skills}
|
|
1205
|
-
return cmd_slugs, skill_names
|
|
1206
|
-
|
|
1207
|
-
|
|
1208
|
-
def _model_tier(skill_md: Path) -> str | None:
|
|
1209
|
-
"""Return the `model_tier` frontmatter value, or None if absent."""
|
|
1210
|
-
if not skill_md.exists():
|
|
1211
|
-
return None
|
|
1212
|
-
text = skill_md.read_text(encoding="utf-8", errors="replace")
|
|
1213
|
-
if not text.startswith("---\n"):
|
|
1214
|
-
return None
|
|
1215
|
-
end = text.find("\n---\n", 4)
|
|
1216
|
-
if end == -1:
|
|
1217
|
-
return None
|
|
1218
|
-
m = _MODEL_TIER_RE.search(text[4:end])
|
|
1219
|
-
return m.group(1) if m else None
|
|
1220
|
-
|
|
1221
|
-
|
|
1222
|
-
def _render_native_model_md(src_md: Path, tier: str) -> str:
|
|
1223
|
-
"""Rewrite the source `model_tier: <tier>` line to a native Claude
|
|
1224
|
-
`model: <mapped>` key via the generator-owned mapping (ADR-035 § 3). The
|
|
1225
|
-
rest of the SKILL.md is byte-identical."""
|
|
1226
|
-
text = src_md.read_text(encoding="utf-8")
|
|
1227
|
-
model = _TIER_TO_CLAUDE_MODEL[tier]
|
|
1228
|
-
return _MODEL_TIER_RE.sub(f"model: {model}", text, count=1)
|
|
1229
|
-
|
|
1230
|
-
|
|
1231
|
-
def generate_claude_skills(active_skill_names: set[str] | None = None) -> int:
|
|
1232
|
-
"""Create .claude/skills/ entries for ALL skills in dist/agent-src/skills/.
|
|
1233
|
-
|
|
1234
|
-
Default: a directory symlink ā dist/agent-src/skills/<name> (verbatim).
|
|
1235
|
-
When `model.auto_switch: auto` AND a skill declares
|
|
1236
|
-
`model_tier: lite|medium|high`, the entry becomes a real directory whose
|
|
1237
|
-
sub-files are symlinked but whose SKILL.md is a rendered copy carrying a
|
|
1238
|
-
native Claude `model:` key (ADR-034 Option (b) ā only model-bearing skills
|
|
1239
|
-
break the symlink). Idempotent: each entry is rebuilt from scratch.
|
|
1240
|
-
"""
|
|
1241
|
-
if not SKILLS_SOURCE.exists():
|
|
1242
|
-
print(" ā ļø dist/agent-src/skills/ not found ā skipping skills", file=sys.stderr)
|
|
1243
|
-
return 0
|
|
1244
|
-
|
|
1245
|
-
# All skill directories in dist/agent-src/skills/. Under scoped projection
|
|
1246
|
-
# (active_skill_names not None) keep only the active set; the stale-cleanup
|
|
1247
|
-
# loop below then reaps any now-inactive skill entry.
|
|
1248
|
-
skills = sorted([d.name for d in SKILLS_SOURCE.iterdir() if d.is_dir()])
|
|
1249
|
-
if active_skill_names is not None:
|
|
1250
|
-
skills = [s for s in skills if s in active_skill_names]
|
|
1251
|
-
skill_set = set(skills)
|
|
1252
|
-
# All command slugs (to protect from stale cleanup)
|
|
1253
|
-
command_slugs = {slug for _, slug in _iter_commands()}
|
|
1254
|
-
|
|
1255
|
-
CLAUDE_SKILLS_DIR.mkdir(parents=True, exist_ok=True)
|
|
1256
|
-
auto = _read_model_auto_switch() == "auto"
|
|
1257
|
-
|
|
1258
|
-
# Clean stale entries (symlinks AND rendered skill dirs) ā but never touch
|
|
1259
|
-
# command dirs (real dir whose SKILL.md is a symlink) or the README.
|
|
1260
|
-
for item in CLAUDE_SKILLS_DIR.iterdir():
|
|
1261
|
-
if item.name in skill_set or item.name in command_slugs or item.name == "README.md":
|
|
1262
|
-
continue
|
|
1263
|
-
if item.is_symlink():
|
|
1264
|
-
item.unlink()
|
|
1265
|
-
elif item.is_dir():
|
|
1266
|
-
skill_md = item / "SKILL.md"
|
|
1267
|
-
# A rendered skill dir has a *real* SKILL.md copy; a command dir has
|
|
1268
|
-
# a SKILL.md *symlink* (left for generate_claude_commands to manage).
|
|
1269
|
-
if skill_md.is_file() and not skill_md.is_symlink():
|
|
1270
|
-
shutil.rmtree(item)
|
|
1271
|
-
|
|
1272
|
-
count = 0
|
|
1273
|
-
rendered = 0
|
|
1274
|
-
for skill in skills:
|
|
1275
|
-
link = CLAUDE_SKILLS_DIR / skill
|
|
1276
|
-
src_dir = SKILLS_SOURCE / skill
|
|
1277
|
-
value = _model_tier(src_dir / "SKILL.md") if auto else None
|
|
1278
|
-
# Rebuild from scratch for idempotency (symlink ā rendered-dir flips).
|
|
1279
|
-
if link.is_symlink():
|
|
1280
|
-
link.unlink()
|
|
1281
|
-
elif link.is_dir():
|
|
1282
|
-
shutil.rmtree(link)
|
|
1283
|
-
elif link.exists():
|
|
1284
|
-
link.unlink()
|
|
1285
|
-
if value in _TIER_TO_CLAUDE_MODEL:
|
|
1286
|
-
link.mkdir(parents=True)
|
|
1287
|
-
for entry in sorted(src_dir.iterdir()):
|
|
1288
|
-
if entry.name == "SKILL.md":
|
|
1289
|
-
(link / "SKILL.md").write_text(
|
|
1290
|
-
_render_native_model_md(entry, value), encoding="utf-8"
|
|
1291
|
-
)
|
|
1292
|
-
else:
|
|
1293
|
-
(link / entry.name).symlink_to(
|
|
1294
|
-
Path("../../../dist/agent-src/skills") / skill / entry.name
|
|
1295
|
-
)
|
|
1296
|
-
rendered += 1
|
|
1297
|
-
else:
|
|
1298
|
-
link.symlink_to(Path("../../dist/agent-src/skills") / skill)
|
|
1299
|
-
count += 1
|
|
1300
|
-
|
|
1301
|
-
suffix = f" ({rendered} rendered with native model:)" if rendered else ""
|
|
1302
|
-
info(f" ā
Created {count} skill entries in .claude/skills/{suffix}")
|
|
1303
|
-
return count
|
|
1304
|
-
|
|
1305
|
-
|
|
1306
|
-
def extract_description_from_md(content: str) -> str:
|
|
1307
|
-
"""Extract description from first # heading or first non-empty line."""
|
|
1308
|
-
for line in content.strip().split("\n"):
|
|
1309
|
-
line = line.strip()
|
|
1310
|
-
if line.startswith("# "):
|
|
1311
|
-
return line[2:].strip()
|
|
1312
|
-
if line and not line.startswith("#"):
|
|
1313
|
-
return line[:120]
|
|
1314
|
-
return ""
|
|
1315
|
-
|
|
1316
|
-
|
|
1317
|
-
def generate_claude_commands(active_command_slugs: set[str] | None = None) -> int:
|
|
1318
|
-
"""Create .claude/skills/{slug}/SKILL.md symlinks for ALL Augment commands.
|
|
1319
|
-
|
|
1320
|
-
Commands in dist/agent-src/commands/ are the single source of truth.
|
|
1321
|
-
They must include name: and disable-model-invocation: true in frontmatter
|
|
1322
|
-
(added once, then maintained as part of the command file).
|
|
1323
|
-
|
|
1324
|
-
Top-level commands use their filename stem as the slug. Nested
|
|
1325
|
-
cluster commands (e.g. `commands/council/default.md`) are flattened
|
|
1326
|
-
to `council-default` so directories never collide in `.claude/skills/`.
|
|
1327
|
-
"""
|
|
1328
|
-
if not (PROJECT_ROOT / "src" / "domains").is_dir():
|
|
1329
|
-
print(" ā ļø src/domains/ not found ā skipping commands", file=sys.stderr)
|
|
1330
|
-
return 0
|
|
1331
|
-
|
|
1332
|
-
CLAUDE_SKILLS_DIR.mkdir(parents=True, exist_ok=True)
|
|
1333
|
-
|
|
1334
|
-
# Collect skill names to avoid overwriting real skills with same-name commands
|
|
1335
|
-
skill_names = set()
|
|
1336
|
-
if SKILLS_SOURCE.exists():
|
|
1337
|
-
skill_names = {d.name for d in SKILLS_SOURCE.iterdir() if d.is_dir()}
|
|
1338
|
-
|
|
1339
|
-
# Track current command slugs for stale-directory cleanup
|
|
1340
|
-
current_slugs: set[str] = set()
|
|
1341
|
-
count = 0
|
|
1342
|
-
skipped = 0
|
|
1343
|
-
rendered = 0
|
|
1344
|
-
auto = _read_model_auto_switch() == "auto"
|
|
1345
|
-
for source_file, slug in _iter_commands():
|
|
1346
|
-
# Scoped projection: skip commands outside the active set. They are not
|
|
1347
|
-
# added to current_slugs, so the stale-cleanup pass below reaps any
|
|
1348
|
-
# existing dir for them (switch-down behaviour).
|
|
1349
|
-
if active_command_slugs is not None and slug not in active_command_slugs:
|
|
1350
|
-
continue
|
|
1351
|
-
# Skip if a real skill with the same name exists ā skill takes priority
|
|
1352
|
-
if slug in skill_names:
|
|
1353
|
-
skipped += 1
|
|
1354
|
-
continue
|
|
1355
|
-
|
|
1356
|
-
current_slugs.add(slug)
|
|
1357
|
-
|
|
1358
|
-
# Create skill directory (real dir, SKILL.md symlinked or rendered)
|
|
1359
|
-
skill_dir = CLAUDE_SKILLS_DIR / slug
|
|
1360
|
-
skill_dir.mkdir(parents=True, exist_ok=True)
|
|
1361
|
-
|
|
1362
|
-
skill_file = skill_dir / "SKILL.md"
|
|
1363
|
-
if skill_file.exists() or skill_file.is_symlink():
|
|
1364
|
-
skill_file.unlink()
|
|
1365
|
-
|
|
1366
|
-
rel_path = source_file.relative_to(PROJECT_ROOT)
|
|
1367
|
-
value = _model_tier(source_file) if auto else None
|
|
1368
|
-
if value in _TIER_TO_CLAUDE_MODEL:
|
|
1369
|
-
# Render a copy carrying the native Claude model: key (ADR-034).
|
|
1370
|
-
skill_file.write_text(
|
|
1371
|
-
_render_native_model_md(source_file, value), encoding="utf-8"
|
|
1372
|
-
)
|
|
1373
|
-
rendered += 1
|
|
1374
|
-
else:
|
|
1375
|
-
# Symlink: .claude/skills/{slug}/SKILL.md ā ../../../src/domains/<pack>/<verb>/command.md
|
|
1376
|
-
rel_target = Path("../../..") / rel_path
|
|
1377
|
-
skill_file.symlink_to(rel_target)
|
|
1378
|
-
count += 1
|
|
1379
|
-
|
|
1380
|
-
# Clean stale command skill directories ā real dirs from removed commands.
|
|
1381
|
-
# Only delete if the directory contains exactly the SKILL.md symlink we created.
|
|
1382
|
-
removed_dirs = 0
|
|
1383
|
-
for item in CLAUDE_SKILLS_DIR.iterdir():
|
|
1384
|
-
if not item.is_dir() or item.is_symlink():
|
|
1385
|
-
continue
|
|
1386
|
-
if item.name in skill_names or item.name in current_slugs:
|
|
1387
|
-
continue
|
|
1388
|
-
skill_md = item / "SKILL.md"
|
|
1389
|
-
# Stale command dir = exactly one SKILL.md, either symlinked (default)
|
|
1390
|
-
# or a rendered copy (native-model command, ADR-034). Skill-render dirs
|
|
1391
|
-
# are protected above (item.name in skill_names) and via the skills
|
|
1392
|
-
# generator, so this only reaps removed commands.
|
|
1393
|
-
if skill_md.is_symlink() or skill_md.is_file():
|
|
1394
|
-
entries = list(item.iterdir())
|
|
1395
|
-
if len(entries) == 1 and entries[0].name == "SKILL.md":
|
|
1396
|
-
skill_md.unlink()
|
|
1397
|
-
item.rmdir()
|
|
1398
|
-
removed_dirs += 1
|
|
1399
|
-
|
|
1400
|
-
msg = f" ā
Created {count} command entries in .claude/skills/"
|
|
1401
|
-
if rendered:
|
|
1402
|
-
msg += f" ({rendered} rendered with native model:)"
|
|
1403
|
-
if skipped:
|
|
1404
|
-
msg += f" ({skipped} skipped ā same-name skill exists)"
|
|
1405
|
-
if removed_dirs:
|
|
1406
|
-
msg += f" ({removed_dirs} stale dirs removed)"
|
|
1407
|
-
info(msg)
|
|
1408
|
-
return count
|
|
1409
|
-
|
|
1410
|
-
|
|
1411
|
-
def generate_plugin_command_skills() -> int:
|
|
1412
|
-
"""Mirror command-as-skill entries into the committed .claude-plugin/skills/.
|
|
1413
|
-
|
|
1414
|
-
The plugin marketplace references each command entry as a <slug>/SKILL.md
|
|
1415
|
-
path that must be committed (git-consumed marketplace). Commands have no
|
|
1416
|
-
such shape in source, so this projects them as symlinks:
|
|
1417
|
-
.claude-plugin/skills/<slug>/SKILL.md ā ../../../dist/agent-src/commands/<rel>.
|
|
1418
|
-
|
|
1419
|
-
Symlink-only by design: the committed .claude/skills/ shape was always
|
|
1420
|
-
symlinks (ADR-034 model-rendered copies are local-only, never committed),
|
|
1421
|
-
so the distributed marketplace behaviour is preserved exactly. The local
|
|
1422
|
-
auto-discovery channel (.claude/skills/, gitignored) keeps model rendering
|
|
1423
|
-
for the dev loop.
|
|
1424
|
-
"""
|
|
1425
|
-
if not (PROJECT_ROOT / "src" / "domains").is_dir():
|
|
1426
|
-
return 0
|
|
1427
|
-
|
|
1428
|
-
PLUGIN_SKILLS_DIR.mkdir(parents=True, exist_ok=True)
|
|
1429
|
-
|
|
1430
|
-
skill_names: set[str] = set()
|
|
1431
|
-
if SKILLS_SOURCE.exists():
|
|
1432
|
-
skill_names = {d.name for d in SKILLS_SOURCE.iterdir() if d.is_dir()}
|
|
1433
|
-
|
|
1434
|
-
current_slugs: set[str] = set()
|
|
1435
|
-
count = 0
|
|
1436
|
-
for source_file, slug in _iter_commands():
|
|
1437
|
-
# A real skill of the same name takes priority ā skip the command.
|
|
1438
|
-
if slug in skill_names:
|
|
1439
|
-
continue
|
|
1440
|
-
current_slugs.add(slug)
|
|
1441
|
-
|
|
1442
|
-
skill_dir = PLUGIN_SKILLS_DIR / slug
|
|
1443
|
-
skill_dir.mkdir(parents=True, exist_ok=True)
|
|
1444
|
-
skill_file = skill_dir / "SKILL.md"
|
|
1445
|
-
if skill_file.exists() or skill_file.is_symlink():
|
|
1446
|
-
skill_file.unlink()
|
|
1447
|
-
|
|
1448
|
-
rel_path = source_file.relative_to(PROJECT_ROOT)
|
|
1449
|
-
rel_target = Path("../../..") / rel_path
|
|
1450
|
-
skill_file.symlink_to(rel_target)
|
|
1451
|
-
count += 1
|
|
1452
|
-
|
|
1453
|
-
# Reap stale command dirs from removed commands (exactly one SKILL.md).
|
|
1454
|
-
removed_dirs = 0
|
|
1455
|
-
for item in PLUGIN_SKILLS_DIR.iterdir():
|
|
1456
|
-
if not item.is_dir() or item.is_symlink():
|
|
1457
|
-
continue
|
|
1458
|
-
if item.name in current_slugs:
|
|
1459
|
-
continue
|
|
1460
|
-
skill_md = item / "SKILL.md"
|
|
1461
|
-
if skill_md.is_symlink() or skill_md.is_file():
|
|
1462
|
-
entries = list(item.iterdir())
|
|
1463
|
-
if len(entries) == 1 and entries[0].name == "SKILL.md":
|
|
1464
|
-
skill_md.unlink()
|
|
1465
|
-
item.rmdir()
|
|
1466
|
-
removed_dirs += 1
|
|
1467
|
-
|
|
1468
|
-
msg = f" ā
Created {count} command entries in .claude-plugin/skills/"
|
|
1469
|
-
if removed_dirs:
|
|
1470
|
-
msg += f" ({removed_dirs} stale dirs removed)"
|
|
1471
|
-
info(msg)
|
|
1472
|
-
return count
|
|
1473
|
-
|
|
1474
|
-
|
|
1475
|
-
def generate_persona_symlinks() -> int:
|
|
1476
|
-
"""Create symlink directories for personas (.claude/personas/, .cursor/personas/).
|
|
1477
|
-
|
|
1478
|
-
Symlinks each persona .md file from dist/agent-src/personas/ into tool-specific
|
|
1479
|
-
directories. Excludes README.md ā that's authoring documentation, not a persona.
|
|
1480
|
-
"""
|
|
1481
|
-
if not PERSONAS_SOURCE.exists():
|
|
1482
|
-
print(" ā ļø dist/agent-src/personas/ not found ā skipping personas")
|
|
1483
|
-
return 0
|
|
1484
|
-
|
|
1485
|
-
personas = sorted([
|
|
1486
|
-
f.name for f in PERSONAS_SOURCE.glob("*.md") if f.stem != "README"
|
|
1487
|
-
])
|
|
1488
|
-
tool_dirs = _filter_tool_dirs(PERSONA_TOOL_DIRS)
|
|
1489
|
-
total = 0
|
|
1490
|
-
for tool_dir, rel_prefix in tool_dirs.items():
|
|
1491
|
-
target_dir = PROJECT_ROOT / tool_dir
|
|
1492
|
-
target_dir.mkdir(parents=True, exist_ok=True)
|
|
1493
|
-
|
|
1494
|
-
# Clean stale symlinks
|
|
1495
|
-
for item in target_dir.iterdir():
|
|
1496
|
-
if item.is_symlink() and item.name not in personas and item.name != "README.md":
|
|
1497
|
-
item.unlink()
|
|
1498
|
-
|
|
1499
|
-
for persona in personas:
|
|
1500
|
-
link = target_dir / persona
|
|
1501
|
-
target = Path(rel_prefix) / persona
|
|
1502
|
-
if link.exists() or link.is_symlink():
|
|
1503
|
-
link.unlink()
|
|
1504
|
-
link.symlink_to(target)
|
|
1505
|
-
total += 1
|
|
1506
|
-
|
|
1507
|
-
info(f" ā
Created {total} persona symlinks across {len(tool_dirs)} tool directories ({len(personas)} personas each)")
|
|
1508
|
-
return total
|
|
1509
|
-
|
|
1510
|
-
|
|
1511
|
-
def generate_user_type_symlinks() -> int:
|
|
1512
|
-
"""Create symlink directories for user-types (.claude/user-types/, .cursor/user-types/).
|
|
1513
|
-
|
|
1514
|
-
Symlinks each user-type .md file from dist/agent-src/user-types/ into tool-specific
|
|
1515
|
-
directories. Excludes README.md and _template/ ā those are authoring scaffolding,
|
|
1516
|
-
not user-type lenses.
|
|
1517
|
-
"""
|
|
1518
|
-
if not USER_TYPES_SOURCE.exists():
|
|
1519
|
-
print(" ā ļø dist/agent-src/user-types/ not found ā skipping user-types")
|
|
1520
|
-
return 0
|
|
1521
|
-
|
|
1522
|
-
user_types = sorted([
|
|
1523
|
-
f.name for f in USER_TYPES_SOURCE.glob("*.md") if f.stem != "README"
|
|
1524
|
-
])
|
|
1525
|
-
tool_dirs = _filter_tool_dirs(USER_TYPE_TOOL_DIRS)
|
|
1526
|
-
total = 0
|
|
1527
|
-
for tool_dir, rel_prefix in tool_dirs.items():
|
|
1528
|
-
target_dir = PROJECT_ROOT / tool_dir
|
|
1529
|
-
target_dir.mkdir(parents=True, exist_ok=True)
|
|
1530
|
-
|
|
1531
|
-
# Clean stale symlinks
|
|
1532
|
-
for item in target_dir.iterdir():
|
|
1533
|
-
if item.is_symlink() and item.name not in user_types and item.name != "README.md":
|
|
1534
|
-
item.unlink()
|
|
1535
|
-
|
|
1536
|
-
for user_type in user_types:
|
|
1537
|
-
link = target_dir / user_type
|
|
1538
|
-
target = Path(rel_prefix) / user_type
|
|
1539
|
-
if link.exists() or link.is_symlink():
|
|
1540
|
-
link.unlink()
|
|
1541
|
-
link.symlink_to(target)
|
|
1542
|
-
total += 1
|
|
1543
|
-
|
|
1544
|
-
info(f" ā
Created {total} user-type symlinks across {len(tool_dirs)} tool directories ({len(user_types)} user-types each)")
|
|
1545
|
-
return total
|
|
1546
|
-
|
|
1547
|
-
|
|
1548
|
-
def generate_plugin_hooks() -> int:
|
|
1549
|
-
"""Generate ``hooks/hooks.json`` at the plugin root from the hook manifest.
|
|
1550
|
-
|
|
1551
|
-
Claude Code plugins auto-discover hooks at ``<plugin-root>/hooks/hooks.json``
|
|
1552
|
-
(Plugins reference). The agent-config plugin's source is the repo root
|
|
1553
|
-
(``.claude-plugin/marketplace.json``), so the file lands at
|
|
1554
|
-
``PROJECT_ROOT/hooks/hooks.json``.
|
|
1555
|
-
|
|
1556
|
-
Delivering the Claude lifecycle hooks via **plugin scope** ā instead of
|
|
1557
|
-
writing them into the shared ``.claude/settings.json`` ``hooks`` array ā
|
|
1558
|
-
means no shared ``hooks`` array in any settings file, so there is no
|
|
1559
|
-
collision with a neighbour tool's hooks or with a developer's
|
|
1560
|
-
``settings.local.json``. Claude Code merges plugin-scope and
|
|
1561
|
-
settings-scope hooks and dedups by command string.
|
|
1562
|
-
|
|
1563
|
-
The command resolves the binary project-local-first
|
|
1564
|
-
(``$CLAUDE_PROJECT_DIR/agent-config`` when executable ā maintainer dev-loop)
|
|
1565
|
-
and falls back to ``agent-config`` on PATH (global-only consumer per
|
|
1566
|
-
ADR-020, where the repo carries no wrapper). It always passes
|
|
1567
|
-
``--project-dir "$CLAUDE_PROJECT_DIR"`` so a globally resolved binary still
|
|
1568
|
-
scans the project the event fired in. The universal dispatcher then reads
|
|
1569
|
-
``scripts/hook_manifest.yaml`` at runtime to fan out to the active
|
|
1570
|
-
concerns (each a no-op when its feature is disabled).
|
|
1571
|
-
"""
|
|
1572
|
-
manifest_path = PROJECT_ROOT / "src" / "scripts" / "hook_manifest.yaml"
|
|
1573
|
-
if not manifest_path.exists():
|
|
1574
|
-
print(" ā ļø scripts/hook_manifest.yaml not found ā skipping plugin hooks",
|
|
1575
|
-
file=sys.stderr)
|
|
1576
|
-
return 0
|
|
1577
|
-
|
|
1578
|
-
manifest = yaml.safe_load(manifest_path.read_text(encoding="utf-8")) or {}
|
|
1579
|
-
hook_spec = manifest.get("schema_version", 1)
|
|
1580
|
-
claude_events = manifest.get("platforms", {}).get("claude", {}) or {}
|
|
1581
|
-
aliases = manifest.get("native_event_aliases", {}).get("claude", {}) or {}
|
|
1582
|
-
# Reverse the nativeāagent-config map so we can emit native event names.
|
|
1583
|
-
ac_to_native = {ac: native for native, ac in aliases.items()}
|
|
1584
|
-
|
|
1585
|
-
hooks: dict[str, list] = {}
|
|
1586
|
-
for ac_event, concerns in claude_events.items():
|
|
1587
|
-
if not concerns:
|
|
1588
|
-
continue
|
|
1589
|
-
native = ac_to_native.get(ac_event)
|
|
1590
|
-
if native is None:
|
|
1591
|
-
continue
|
|
1592
|
-
# Resolve the binary project-local-first, PATH-fallback: a maintainer
|
|
1593
|
-
# dev-loop (or any repo carrying a project-local wrapper) uses its own
|
|
1594
|
-
# ./agent-config; a global-only consumer (ADR-020 ā nothing in the
|
|
1595
|
-
# repo) falls back to the globally-installed `agent-config` on PATH.
|
|
1596
|
-
# Either way pass --project-dir "$CLAUDE_PROJECT_DIR" so a globally
|
|
1597
|
-
# resolved binary still scans the project the event fired in.
|
|
1598
|
-
command = (
|
|
1599
|
-
'BIN="$CLAUDE_PROJECT_DIR/agent-config"; [ -x "$BIN" ] || BIN=agent-config; '
|
|
1600
|
-
f'"$BIN" dispatch:hook --platform claude --event {ac_event} '
|
|
1601
|
-
f'--native-event {native} --project-dir "$CLAUDE_PROJECT_DIR" '
|
|
1602
|
-
f'--min-version {hook_spec}'
|
|
1603
|
-
)
|
|
1604
|
-
hooks[native] = [{"hooks": [{"type": "command", "command": command}]}]
|
|
1605
|
-
|
|
1606
|
-
hooks_dir = PROJECT_ROOT / "hooks"
|
|
1607
|
-
hooks_dir.mkdir(parents=True, exist_ok=True)
|
|
1608
|
-
out = hooks_dir / "hooks.json"
|
|
1609
|
-
out.write_text(json.dumps({"hooks": hooks}, indent=2) + "\n", encoding="utf-8")
|
|
1610
|
-
info(f" ā
Generated hooks/hooks.json ({len(hooks)} Claude plugin hooks)")
|
|
1611
|
-
return len(hooks)
|
|
1612
|
-
|
|
1613
|
-
|
|
1614
|
-
def _generate_tools_inner(
|
|
1615
|
-
cmd_slugs: set[str] | None, skill_names: set[str] | None
|
|
1616
|
-
) -> None:
|
|
1617
|
-
"""Run every tool generator once. `cmd_slugs` / `skill_names` are `None`
|
|
1618
|
-
for legacy-all (full surface) or allowlists for scoped projection. Only the
|
|
1619
|
-
four consumer-local native generators honour the allowlists; the committed
|
|
1620
|
-
plugin marketplace (`.claude-plugin/`) and the Augment tree always project
|
|
1621
|
-
the full set in 6.0.0 (ADR-040 / road-to-6.0.0-B Step 8, D3)."""
|
|
1622
|
-
info("š§ Generating multi-agent tool directories...\n")
|
|
1623
|
-
rules = generate_rule_symlinks()
|
|
1624
|
-
windsurfrules = generate_windsurfrules() if _tool_active("windsurf") else 0
|
|
1625
|
-
if _tool_active("gemini"):
|
|
1626
|
-
generate_gemini_md()
|
|
1627
|
-
skills = generate_claude_skills(skill_names) if _tool_active("claude-code") else 0
|
|
1628
|
-
commands = generate_claude_commands(cmd_slugs) if _tool_active("claude-code") else 0
|
|
1629
|
-
plugin_cmd_skills = generate_plugin_command_skills() if _tool_active("claude-code") else 0
|
|
1630
|
-
plugin_hooks = generate_plugin_hooks() if _tool_active("claude-code") else 0
|
|
1631
|
-
personas = generate_persona_symlinks()
|
|
1632
|
-
user_types = generate_user_type_symlinks()
|
|
1633
|
-
cursor_mdc = generate_cursor_mdc_rules() if _tool_active("cursor") else 0
|
|
1634
|
-
windsurf_modern = generate_windsurf_modern_rules() if _tool_active("windsurf") else 0
|
|
1635
|
-
cursor_cmds = generate_cursor_commands(cmd_slugs) if _tool_active("cursor") else 0
|
|
1636
|
-
windsurf_wf = generate_windsurf_workflows(cmd_slugs) if _tool_active("windsurf") else 0
|
|
1637
|
-
summary = (
|
|
1638
|
-
f"ā
generate-tools ā rules={rules} skills={skills} "
|
|
1639
|
-
f"commands={commands} plugin_cmd_skills={plugin_cmd_skills} "
|
|
1640
|
-
f"plugin_hooks={plugin_hooks} "
|
|
1641
|
-
f"personas={personas} user_types={user_types} "
|
|
1642
|
-
f"cursor_mdc={cursor_mdc} windsurf_rules={windsurf_modern} "
|
|
1643
|
-
f"cursor_commands={cursor_cmds} windsurf_workflows={windsurf_wf} "
|
|
1644
|
-
f"windsurfrules={windsurfrules}"
|
|
1645
|
-
)
|
|
1646
|
-
if resolve_level() == "verbose":
|
|
1647
|
-
print(f"\n{summary}")
|
|
1648
|
-
else:
|
|
1649
|
-
success(summary)
|
|
1650
|
-
flush_summary()
|
|
1651
|
-
|
|
1652
|
-
|
|
1653
|
-
def generate_tools() -> None:
|
|
1654
|
-
"""Generate all tool-specific directories and files.
|
|
1655
|
-
|
|
1656
|
-
`agents/.agent-tools.yml` gates per-tool emission. When the file
|
|
1657
|
-
is missing, every tool is emitted (preserves test fixtures and
|
|
1658
|
-
pre-gating behaviour). See `_active_tools()` and `_tool_active()`.
|
|
1659
|
-
|
|
1660
|
-
Honours `projection.mode` (ADR-040): `legacy-all` (default) projects the
|
|
1661
|
-
full surface; `scoped` projects only the active profile + packs. A failed
|
|
1662
|
-
scoped projection restores the full (legacy-all) tree so the host tool is
|
|
1663
|
-
never left with a partial set, then re-raises.
|
|
1664
|
-
"""
|
|
1665
|
-
cmd_slugs, skill_names = _resolve_active_predicates()
|
|
1666
|
-
scoped = cmd_slugs is not None
|
|
1667
|
-
try:
|
|
1668
|
-
_generate_tools_inner(cmd_slugs, skill_names)
|
|
1669
|
-
except Exception:
|
|
1670
|
-
if scoped:
|
|
1671
|
-
info(" ā ļø scoped projection failed ā restoring full (legacy-all) projection")
|
|
1672
|
-
_generate_tools_inner(None, None)
|
|
1673
|
-
raise
|
|
1674
|
-
if not scoped:
|
|
1675
|
-
info(
|
|
1676
|
-
" ā¹ļø Profile mode available ā focused surface. "
|
|
1677
|
-
"Run `agent-config use --profile=developer`."
|
|
1678
|
-
)
|
|
1679
|
-
|
|
1680
|
-
|
|
1681
|
-
# āā .augment/ projection āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
|
|
1682
|
-
# The package uses dist/agent-src/ as the tool-agnostic condensed source of truth.
|
|
1683
|
-
# .augment/ is a generated projection so that Augment Code (which reads from
|
|
1684
|
-
# .augment/) works on the package repo itself. Rules default to copies
|
|
1685
|
-
# because Augment Code historically does not load symlinked rule files;
|
|
1686
|
-
# flip augment.rules_use_symlinks: true in .agent-settings.yml to switch
|
|
1687
|
-
# them to symlinks (everything else is always symlinked).
|
|
1688
|
-
|
|
1689
|
-
# Subdirectories of dist/agent-src/ that map into .augment/ as symlinks.
|
|
1690
|
-
AUGMENT_SYMLINK_DIRS = ("skills", "commands", "guidelines", "personas", "user-types", "templates", "contexts", "scripts")
|
|
1691
|
-
# Top-level files to symlink into .augment/ (README, etc.)
|
|
1692
|
-
AUGMENT_SYMLINK_FILES = ("README.md",)
|
|
1693
|
-
|
|
1694
|
-
|
|
1695
|
-
def project_to_augment() -> None:
|
|
1696
|
-
"""Mirror dist/agent-src/ into .augment/. Symlink everything except rules,
|
|
1697
|
-
which default to copies; opt into rule symlinks via
|
|
1698
|
-
augment.rules_use_symlinks in .agent-settings.yml."""
|
|
1699
|
-
if not TARGET_DIR.exists():
|
|
1700
|
-
print(f" ā ļø {TARGET_DIR.name}/ not found ā nothing to project")
|
|
1701
|
-
return
|
|
1702
|
-
|
|
1703
|
-
AUGMENT_DIR.mkdir(parents=True, exist_ok=True)
|
|
1704
|
-
|
|
1705
|
-
use_symlinks = _read_augment_rules_use_symlinks()
|
|
1706
|
-
|
|
1707
|
-
# Rules: copy by default (Augment Code historically does not load
|
|
1708
|
-
# symlinked rules), or symlink when augment.rules_use_symlinks is true.
|
|
1709
|
-
src_rules = TARGET_DIR / "rules"
|
|
1710
|
-
dst_rules = AUGMENT_DIR / "rules"
|
|
1711
|
-
dst_rules.mkdir(parents=True, exist_ok=True)
|
|
1712
|
-
existing = {f.name for f in dst_rules.iterdir() if f.is_file() or f.is_symlink()}
|
|
1713
|
-
current = set()
|
|
1714
|
-
written = 0
|
|
1715
|
-
if src_rules.exists():
|
|
1716
|
-
for rule in sorted(src_rules.glob("*.md")):
|
|
1717
|
-
target = dst_rules / rule.name
|
|
1718
|
-
# Always remove first to avoid copyāsymlink mode mismatch.
|
|
1719
|
-
if target.is_symlink() or target.exists():
|
|
1720
|
-
target.unlink()
|
|
1721
|
-
if use_symlinks:
|
|
1722
|
-
target.symlink_to(Path("..") / ".." / "dist/agent-src" / "rules" / rule.name)
|
|
1723
|
-
else:
|
|
1724
|
-
shutil.copy2(rule, target)
|
|
1725
|
-
current.add(rule.name)
|
|
1726
|
-
written += 1
|
|
1727
|
-
# Remove stale rule files
|
|
1728
|
-
removed_rules = 0
|
|
1729
|
-
for name in existing - current:
|
|
1730
|
-
(dst_rules / name).unlink()
|
|
1731
|
-
removed_rules += 1
|
|
1732
|
-
mode_label = "Symlinked" if use_symlinks else "Copied"
|
|
1733
|
-
print(f" ā
{mode_label} {written} rules to .augment/rules/" + (f" ({removed_rules} stale removed)" if removed_rules else ""))
|
|
1734
|
-
|
|
1735
|
-
# Subdirectories: replace each with a symlink ā ../dist/agent-src/<subdir>
|
|
1736
|
-
for sub in AUGMENT_SYMLINK_DIRS:
|
|
1737
|
-
dst = AUGMENT_DIR / sub
|
|
1738
|
-
if dst.is_symlink() or dst.exists():
|
|
1739
|
-
if dst.is_dir() and not dst.is_symlink():
|
|
1740
|
-
shutil.rmtree(dst)
|
|
1741
|
-
else:
|
|
1742
|
-
dst.unlink()
|
|
1743
|
-
src = TARGET_DIR / sub
|
|
1744
|
-
if src.exists():
|
|
1745
|
-
dst.symlink_to(Path("..") / "dist/agent-src" / sub, target_is_directory=True)
|
|
1746
|
-
print(f" ā
Symlinked .augment/{sub} ā ../dist/agent-src/{sub}")
|
|
1747
|
-
|
|
1748
|
-
# Top-level files: symlink
|
|
1749
|
-
for name in AUGMENT_SYMLINK_FILES:
|
|
1750
|
-
dst = AUGMENT_DIR / name
|
|
1751
|
-
src = TARGET_DIR / name
|
|
1752
|
-
if dst.is_symlink() or dst.exists():
|
|
1753
|
-
dst.unlink()
|
|
1754
|
-
if src.exists():
|
|
1755
|
-
dst.symlink_to(Path("..") / "dist/agent-src" / name)
|
|
1756
|
-
print(f" ā
Symlinked .augment/{name} ā ../dist/agent-src/{name}")
|
|
1757
|
-
|
|
1758
|
-
# Cleanup: remove any stray top-level entries in .augment/ that are no longer projected.
|
|
1759
|
-
# `state` holds runtime state files written by hooks (onboarding-gate,
|
|
1760
|
-
# context-hygiene, ā¦) and must survive sync ā it is regenerated by
|
|
1761
|
-
# the next hook fire, not by condense.
|
|
1762
|
-
known = set(AUGMENT_SYMLINK_DIRS) | set(AUGMENT_SYMLINK_FILES) | {"rules", "state"}
|
|
1763
|
-
for item in AUGMENT_DIR.iterdir():
|
|
1764
|
-
if item.name in known:
|
|
1765
|
-
continue
|
|
1766
|
-
if item.is_symlink() or item.is_file():
|
|
1767
|
-
item.unlink()
|
|
1768
|
-
print(f" šļø Removed stale .augment/{item.name}")
|
|
1769
|
-
elif item.is_dir():
|
|
1770
|
-
shutil.rmtree(item)
|
|
1771
|
-
print(f" šļø Removed stale .augment/{item.name}/")
|
|
1772
|
-
|
|
1773
|
-
|
|
1774
|
-
def clean_tools() -> None:
|
|
1775
|
-
"""Remove all generated tool directories and files."""
|
|
1776
|
-
import shutil as _shutil
|
|
1777
|
-
targets = [
|
|
1778
|
-
PROJECT_ROOT / ".claude",
|
|
1779
|
-
PROJECT_ROOT / ".cursor",
|
|
1780
|
-
PROJECT_ROOT / ".clinerules",
|
|
1781
|
-
PROJECT_ROOT / ".windsurf",
|
|
1782
|
-
PROJECT_ROOT / ".windsurfrules",
|
|
1783
|
-
PROJECT_ROOT / "GEMINI.md",
|
|
1784
|
-
]
|
|
1785
|
-
for t in targets:
|
|
1786
|
-
if t.is_dir():
|
|
1787
|
-
_shutil.rmtree(t)
|
|
1788
|
-
print(f" šļø Removed {t.relative_to(PROJECT_ROOT)}")
|
|
1789
|
-
elif t.exists() or t.is_symlink():
|
|
1790
|
-
t.unlink()
|
|
1791
|
-
print(f" šļø Removed {t.relative_to(PROJECT_ROOT)}")
|
|
1792
|
-
print("ā
All generated tool files cleaned")
|
|
1793
|
-
|
|
1794
|
-
|
|
1795
|
-
|
|
1796
|
-
def main() -> None:
|
|
1797
|
-
# Multi-root awareness (ADR-017): tolerate a missing legacy
|
|
1798
|
-
# `.agent-src.uncondensed/` as long as at least one package-scoped
|
|
1799
|
-
# source root carries artefacts.
|
|
1800
|
-
if not SOURCE_DIR.exists() and not _any_source_root_exists():
|
|
1801
|
-
print(f"ā No source directory found (looked at {SOURCE_DIR} and packages/*/.agent-src.uncondensed)")
|
|
1802
|
-
sys.exit(1)
|
|
1803
|
-
|
|
1804
|
-
arg = sys.argv[1] if len(sys.argv) > 1 else "--sync"
|
|
1805
|
-
|
|
1806
|
-
if arg == "--list":
|
|
1807
|
-
files = list_md_files(SOURCE_DIR)
|
|
1808
|
-
print(f"š {len(files)} .md files total:\n")
|
|
1809
|
-
for f in files:
|
|
1810
|
-
print(f" {f}")
|
|
1811
|
-
|
|
1812
|
-
elif arg == "--changed":
|
|
1813
|
-
changed = list_changed_md(SOURCE_DIR)
|
|
1814
|
-
if not changed:
|
|
1815
|
-
print("ā
No .md files changed since last condensation")
|
|
1816
|
-
sys.exit(0)
|
|
1817
|
-
print(f"š {len(changed)} .md files changed since last condensation:\n")
|
|
1818
|
-
for f in changed:
|
|
1819
|
-
print(f" {f}")
|
|
1820
|
-
|
|
1821
|
-
elif arg == "--mark-done":
|
|
1822
|
-
if len(sys.argv) < 3:
|
|
1823
|
-
print("Usage: python scripts/condense.py --mark-done <relative-path>")
|
|
1824
|
-
sys.exit(1)
|
|
1825
|
-
mark_done(sys.argv[2])
|
|
1826
|
-
|
|
1827
|
-
elif arg == "--mark-all-done":
|
|
1828
|
-
mark_all_done()
|
|
1829
|
-
|
|
1830
|
-
elif arg == "--check":
|
|
1831
|
-
missing, stale = check_sync(SOURCE_DIR, TARGET_DIR)
|
|
1832
|
-
if not missing and not stale:
|
|
1833
|
-
print("ā
dist/agent-src/ is in sync with .agent-src.uncondensed/")
|
|
1834
|
-
sys.exit(0)
|
|
1835
|
-
if missing:
|
|
1836
|
-
print(f"ā Missing in dist/agent-src/ ({len(missing)}):")
|
|
1837
|
-
for f in missing:
|
|
1838
|
-
print(f" {f}")
|
|
1839
|
-
if stale:
|
|
1840
|
-
print(f"ā Stale in dist/agent-src/ ({len(stale)}):")
|
|
1841
|
-
for f in stale:
|
|
1842
|
-
print(f" {f}")
|
|
1843
|
-
print(f"\nRun 'task sync' to fix non-.md files, then ask the agent to condense .md files.")
|
|
1844
|
-
sys.exit(1)
|
|
1845
|
-
|
|
1846
|
-
elif arg == "--sync":
|
|
1847
|
-
print(f"Source: {SOURCE_DIR}")
|
|
1848
|
-
print(f"Target: {TARGET_DIR}\n")
|
|
1849
|
-
print("--- Syncing non-.md files ---")
|
|
1850
|
-
copied = sync_non_md(SOURCE_DIR, TARGET_DIR)
|
|
1851
|
-
print(f"\n--- Cleanup stale files ---")
|
|
1852
|
-
deleted = cleanup_stale(SOURCE_DIR, TARGET_DIR)
|
|
1853
|
-
# Also cleanup stale hashes (multi-root aware ā resolve against
|
|
1854
|
-
# every artefact root, not just the legacy SOURCE_DIR).
|
|
1855
|
-
hashes = load_hashes()
|
|
1856
|
-
stale_keys = [k for k in hashes if _resolve_source(k) is None]
|
|
1857
|
-
for k in stale_keys:
|
|
1858
|
-
del hashes[k]
|
|
1859
|
-
if stale_keys:
|
|
1860
|
-
save_hashes(hashes)
|
|
1861
|
-
print(f" Cleaned {len(stale_keys)} stale hash entries")
|
|
1862
|
-
changed = list_changed_md(SOURCE_DIR)
|
|
1863
|
-
print(f"\nā
Done: {copied} copied, {deleted} stale deleted")
|
|
1864
|
-
if changed:
|
|
1865
|
-
print(f"š {len(changed)} .md files need condensation (run --changed to see them)")
|
|
1866
|
-
else:
|
|
1867
|
-
print(f"ā
All .md files are up to date")
|
|
1868
|
-
print(f"\n--- Projecting dist/agent-src/ ā .augment/ ---")
|
|
1869
|
-
project_to_augment()
|
|
1870
|
-
|
|
1871
|
-
elif arg == "--check-hashes":
|
|
1872
|
-
has_issues = False
|
|
1873
|
-
changed = list_changed_md(SOURCE_DIR)
|
|
1874
|
-
stale = find_stale_hashes(SOURCE_DIR)
|
|
1875
|
-
|
|
1876
|
-
if stale:
|
|
1877
|
-
has_issues = True
|
|
1878
|
-
print(f"ā ļø {len(stale)} stale hash(es) for deleted source files:\n")
|
|
1879
|
-
for f in stale:
|
|
1880
|
-
print(f" {f}")
|
|
1881
|
-
print(f"\nRun 'task sync-clean-hashes' to remove them.\n")
|
|
1882
|
-
|
|
1883
|
-
if changed:
|
|
1884
|
-
has_issues = True
|
|
1885
|
-
print(f"ā {len(changed)} .md file(s) need recondensation:\n")
|
|
1886
|
-
for f in changed:
|
|
1887
|
-
stored = load_hashes().get(f)
|
|
1888
|
-
reason = "no hash stored" if stored is None else "hash mismatch"
|
|
1889
|
-
print(f" {f} ({reason})")
|
|
1890
|
-
print(f"\nRun '/condense' command to recondense these files.")
|
|
1891
|
-
|
|
1892
|
-
if not has_issues:
|
|
1893
|
-
print("ā
All condensation hashes are clean (no stale, no mismatches)")
|
|
1894
|
-
sys.exit(0)
|
|
1895
|
-
sys.exit(1)
|
|
1896
|
-
|
|
1897
|
-
elif arg == "--clean-hashes":
|
|
1898
|
-
count = clean_stale_hashes(SOURCE_DIR)
|
|
1899
|
-
if count:
|
|
1900
|
-
print(f"ā
Removed {count} stale hash(es)")
|
|
1901
|
-
else:
|
|
1902
|
-
print("ā
No stale hashes found")
|
|
1903
|
-
|
|
1904
|
-
elif arg == "--generate-tools":
|
|
1905
|
-
generate_tools()
|
|
1906
|
-
|
|
1907
|
-
elif arg == "--clean-tools":
|
|
1908
|
-
clean_tools()
|
|
1909
|
-
|
|
1910
|
-
elif arg == "--project-augment":
|
|
1911
|
-
project_to_augment()
|
|
1912
|
-
|
|
1913
|
-
else:
|
|
1914
|
-
print("Usage: python scripts/condense.py [--sync|--list|--changed|--check|--check-hashes|--clean-hashes|--mark-done <path>|--mark-all-done|--generate-tools|--clean-tools|--project-augment]")
|
|
1915
|
-
sys.exit(1)
|
|
1916
|
-
|
|
1917
|
-
|
|
1918
|
-
if __name__ == "__main__":
|
|
1919
|
-
main()
|