@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
|
@@ -1,1669 +0,0 @@
|
|
|
1
|
-
"""``agent-config doctor`` — install + manifest health report.
|
|
2
|
-
|
|
3
|
-
Phase 4 of road-to-multi-package-coexistence (drift detection) and
|
|
4
|
-
Phase 2 of road-to-surface-discipline (diagnostic hub). Read-only
|
|
5
|
-
sibling to ``prune``/``validate``: walks the project manifest and the
|
|
6
|
-
on-disk deploy roots, runs a battery of health checks, and produces:
|
|
7
|
-
|
|
8
|
-
Drift categories (manifest ↔ filesystem):
|
|
9
|
-
|
|
10
|
-
* ``missing`` — manifest entry has a ``path`` that is **not** on disk.
|
|
11
|
-
* ``modified`` — manifest entry records a ``sha256`` that does not
|
|
12
|
-
match the current bytes on disk.
|
|
13
|
-
* ``foreign`` — file present under one of the ``deploy_roots`` that
|
|
14
|
-
no manifest entry claims (potential neighbour-tool drift).
|
|
15
|
-
* ``tag-drift`` — manifest-claimed ``.md`` file carries a frontmatter
|
|
16
|
-
``package:`` value that disagrees with this package's identifier
|
|
17
|
-
(P5.2). Hand-edited tags or accidental cross-package writes show up
|
|
18
|
-
here; files without frontmatter are skipped (P5.1 contract).
|
|
19
|
-
|
|
20
|
-
Health checks (see :data:`CHECK_IDS`):
|
|
21
|
-
scope · stale-orphans · manifest-integrity · lockfile-freshness · bridge-drift ·
|
|
22
|
-
mcp-mode · mcp-beta-readiness · offline-readiness · python-runtime ·
|
|
23
|
-
tier-usage-readiness · council-cli · unsupported-combos ·
|
|
24
|
-
wizard-state.
|
|
25
|
-
Each emits a structured ``{id, status, message, remedy}`` record with
|
|
26
|
-
``status`` ∈ ``ok`` / ``warn`` / ``fail`` / ``skipped`` (rendered
|
|
27
|
-
``✅`` / ``⚠️`` / ``❌`` / ``⏭️``). ``--check <id>`` runs a single check.
|
|
28
|
-
|
|
29
|
-
Checks are split by scope (:data:`GLOBAL_CHECK_IDS` vs
|
|
30
|
-
:data:`MANIFEST_REQUIRED_CHECK_IDS`) so an ADR-020 global-only consumer
|
|
31
|
-
(bridge marker present, no ``agents/installed-tools.lock``) gets a
|
|
32
|
-
green-capable report instead of a hard bail: the global checks run, the
|
|
33
|
-
manifest-required checks report ``skipped``, and ``bridge-drift`` returns
|
|
34
|
-
a scope-aware "drift not applicable" verdict. See :func:`main` for the
|
|
35
|
-
no-manifest branch and exit-code contract.
|
|
36
|
-
|
|
37
|
-
Repair affordances: ``--repair wizard-state`` resets a malformed or
|
|
38
|
-
orphaned ``state/wizard-state.json`` under the user-global root (the
|
|
39
|
-
file the unified Setup-Wizard uses for resume continuity). Used as the
|
|
40
|
-
recovery path when ``1.9``'s npm downgrade is not viable
|
|
41
|
-
(road-to-global-only-install § 1.10 / A6).
|
|
42
|
-
|
|
43
|
-
Exit codes: ``0`` (clean) · ``1`` (drift or any ``fail`` check) · ``2``
|
|
44
|
-
(error such as "manifest missing"). Both human and ``--json`` output
|
|
45
|
-
emit the drift category lists and the structured checks array. Every
|
|
46
|
-
drift entry carries a one-line ``fix`` hint (P4.3).
|
|
47
|
-
"""
|
|
48
|
-
from __future__ import annotations
|
|
49
|
-
|
|
50
|
-
import argparse
|
|
51
|
-
import hashlib
|
|
52
|
-
import json
|
|
53
|
-
import os
|
|
54
|
-
import re
|
|
55
|
-
import shutil
|
|
56
|
-
import sys
|
|
57
|
-
from pathlib import Path
|
|
58
|
-
from typing import Any
|
|
59
|
-
|
|
60
|
-
from scripts._lib import installed_lock, installed_tools
|
|
61
|
-
from scripts._lib.agent_settings import (
|
|
62
|
-
PROJECT_ROOT_ENV,
|
|
63
|
-
ROOT_OVERRIDE_ENV,
|
|
64
|
-
ProjectRootError,
|
|
65
|
-
find_project_root_with_trace,
|
|
66
|
-
project_settings_path,
|
|
67
|
-
resolve_project_root,
|
|
68
|
-
)
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
class _Sentinel:
|
|
72
|
-
"""Tiny stand-in for a private sentinel value type."""
|
|
73
|
-
|
|
74
|
-
__slots__ = ()
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
#: Returned by :func:`_read_inline_package_tag` when the file is out
|
|
78
|
-
#: of scope for tag-drift detection (no ``.md`` suffix, unreadable, or
|
|
79
|
-
#: no leading frontmatter block).
|
|
80
|
-
NO_FRONTMATTER = _Sentinel()
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
def _resolve_project_root(arg: str | None) -> tuple[Path, str]:
|
|
84
|
-
"""Resolve the doctor's project root via the shared Phase-3 helper.
|
|
85
|
-
|
|
86
|
-
Returns ``(root, origin)`` where ``origin`` is one of the anchor
|
|
87
|
-
names from :mod:`scripts._lib.agent_settings` (``agents-dir``,
|
|
88
|
-
``git``, ``agent-settings``) or one of the origin tags
|
|
89
|
-
``root-flag`` / ``explicit`` / ``env`` / ``cwd-fallback``. The tag
|
|
90
|
-
is surfaced in both human and JSON output so subdir invocations
|
|
91
|
-
are auditable.
|
|
92
|
-
"""
|
|
93
|
-
return resolve_project_root(arg)
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
#: Path of the install-mode marker file (Step 8 A5). One-line file:
|
|
97
|
-
#: ``minimal\n`` or ``full\n``. Written by ``install.py``; consumed by
|
|
98
|
-
#: ``doctor --context`` to surface install state without re-deriving it
|
|
99
|
-
#: from filesystem heuristics.
|
|
100
|
-
_INSTALL_MODE_MARKER_REL = "agents/.agent-state/install-mode.txt"
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
def _detect_install_mode(project_root: Path) -> tuple[str, str]:
|
|
104
|
-
"""Return ``(mode, source)`` for the project install state.
|
|
105
|
-
|
|
106
|
-
Hybrid detection (Step 8 council decision Q5):
|
|
107
|
-
|
|
108
|
-
1. ``agents/.agent-state/install-mode.txt`` if present (authoritative
|
|
109
|
-
for installs since Step 8) → ``source="marker-file"``.
|
|
110
|
-
2. Filesystem heuristic for back-compat: ``AGENTS.md`` + copilot
|
|
111
|
-
bridges → ``full``; otherwise ``minimal`` → ``source="heuristic"``.
|
|
112
|
-
|
|
113
|
-
Heuristic is fragile by design — users who delete ``AGENTS.md`` but
|
|
114
|
-
keep copilot bridges get misreported. The marker file is the
|
|
115
|
-
intended source of truth for installs from Step 8 onward.
|
|
116
|
-
"""
|
|
117
|
-
marker = project_root / _INSTALL_MODE_MARKER_REL
|
|
118
|
-
if marker.is_file():
|
|
119
|
-
try:
|
|
120
|
-
value = marker.read_text(encoding="utf-8").strip()
|
|
121
|
-
except OSError:
|
|
122
|
-
value = ""
|
|
123
|
-
if value in ("minimal", "full"):
|
|
124
|
-
return value, "marker-file"
|
|
125
|
-
has_agents_md = (project_root / "AGENTS.md").exists()
|
|
126
|
-
has_copilot = (project_root / ".github" / "copilot-instructions.md").exists()
|
|
127
|
-
if has_agents_md and has_copilot:
|
|
128
|
-
return "full", "heuristic"
|
|
129
|
-
return "minimal", "heuristic"
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
def _settings_layer_chain(project_root: Path) -> list[str]:
|
|
133
|
-
"""Return the ordered ``.agent-settings.yml`` layer files that exist.
|
|
134
|
-
|
|
135
|
-
Shallowest first (user-global → project root). Used by ``--context``
|
|
136
|
-
to surface which layers will participate in the cascade merge.
|
|
137
|
-
"""
|
|
138
|
-
layers: list[str] = []
|
|
139
|
-
from scripts._lib import user_global_paths
|
|
140
|
-
user_global = user_global_paths.resolve_with_fallback("agent-settings.yml")
|
|
141
|
-
if user_global is not None and user_global.is_file():
|
|
142
|
-
layers.append(str(user_global))
|
|
143
|
-
project_settings = project_settings_path(project_root)
|
|
144
|
-
if project_settings.is_file():
|
|
145
|
-
layers.append(str(project_settings))
|
|
146
|
-
return layers
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
def _detect_wrapper(project_root: Path) -> dict[str, Any]:
|
|
150
|
-
"""Return ``{path, exists, embedded_root}`` for the project wrapper.
|
|
151
|
-
|
|
152
|
-
The ``./agent-config`` wrapper at the project root pins
|
|
153
|
-
``AGENT_CONFIG_PROJECT_ROOT`` to its own ``SELF_DIR``, so the
|
|
154
|
-
"embedded root" is just the wrapper's directory. Surfaced for
|
|
155
|
-
debugging wrapper-coupling issues (Step 8 A3-coupling).
|
|
156
|
-
"""
|
|
157
|
-
wrapper = project_root / "agent-config"
|
|
158
|
-
if not wrapper.exists():
|
|
159
|
-
return {"path": str(wrapper), "exists": False, "embedded_root": None}
|
|
160
|
-
return {
|
|
161
|
-
"path": str(wrapper),
|
|
162
|
-
"exists": True,
|
|
163
|
-
"embedded_root": str(project_root),
|
|
164
|
-
}
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
def _emit_trace_text(
|
|
168
|
-
root: Path | None,
|
|
169
|
-
anchor: str | None,
|
|
170
|
-
trace: list[dict[str, Any]],
|
|
171
|
-
origin: str,
|
|
172
|
-
) -> None:
|
|
173
|
-
"""Render the discovery trace as human-readable text."""
|
|
174
|
-
print(f" 📍 start: {Path.cwd()}")
|
|
175
|
-
print(f" 📍 origin: {origin}")
|
|
176
|
-
if trace:
|
|
177
|
-
print(" trace:")
|
|
178
|
-
for record in trace:
|
|
179
|
-
hit = record["hit"]
|
|
180
|
-
symbol = "✅" if hit else "·"
|
|
181
|
-
tag = f"[{record['pass']}]"
|
|
182
|
-
anchor_str = f" → {hit}" if hit else ""
|
|
183
|
-
print(
|
|
184
|
-
f" {symbol} {tag} {record['ancestor']}"
|
|
185
|
-
f"{anchor_str} ({record['reason']})"
|
|
186
|
-
)
|
|
187
|
-
if root is not None:
|
|
188
|
-
print(f" 📍 resolved root: {root} (anchor: {anchor or 'n/a'})")
|
|
189
|
-
else:
|
|
190
|
-
print(" ⚠️ no anchor found in chain")
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
def _emit_trace_json(
|
|
194
|
-
root: Path | None,
|
|
195
|
-
anchor: str | None,
|
|
196
|
-
trace: list[dict[str, Any]],
|
|
197
|
-
origin: str,
|
|
198
|
-
) -> None:
|
|
199
|
-
"""Render the discovery trace as a JSON payload."""
|
|
200
|
-
payload: dict[str, Any] = {
|
|
201
|
-
"start": str(Path.cwd()),
|
|
202
|
-
"origin": origin,
|
|
203
|
-
"resolved_root": str(root) if root is not None else None,
|
|
204
|
-
"anchor": anchor,
|
|
205
|
-
"trace": trace,
|
|
206
|
-
}
|
|
207
|
-
print(json.dumps(payload, indent=2))
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
def _emit_context_text(ctx: dict[str, Any]) -> None:
|
|
211
|
-
"""Render the install / discovery context as human-readable text."""
|
|
212
|
-
print(f" 📍 project_root: {ctx['project_root']} (origin: {ctx['origin']})")
|
|
213
|
-
print(f" 📦 install_mode: {ctx['install_mode']} (source: {ctx['install_mode_source']})")
|
|
214
|
-
env_pin = ctx.get("env_pin")
|
|
215
|
-
if env_pin:
|
|
216
|
-
marker = " (--root override)" if ctx.get("root_override") else ""
|
|
217
|
-
print(f" 🔒 env_pin: {env_pin}{marker}")
|
|
218
|
-
else:
|
|
219
|
-
print(" 🔒 env_pin: (unset)")
|
|
220
|
-
layers = ctx.get("settings_layers") or []
|
|
221
|
-
if layers:
|
|
222
|
-
print(f" 📑 settings layers ({len(layers)}):")
|
|
223
|
-
for layer in layers:
|
|
224
|
-
print(f" - {layer}")
|
|
225
|
-
else:
|
|
226
|
-
print(" 📑 settings layers: (none)")
|
|
227
|
-
wrapper = ctx.get("wrapper") or {}
|
|
228
|
-
if wrapper.get("exists"):
|
|
229
|
-
print(f" 🧩 wrapper: {wrapper['path']} (embedded root: {wrapper.get('embedded_root')})")
|
|
230
|
-
else:
|
|
231
|
-
print(f" 🧩 wrapper: (not present at {wrapper.get('path')})")
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
def _resolve_path(project_root: Path, raw: str) -> Path:
|
|
235
|
-
p = Path(raw).expanduser()
|
|
236
|
-
if not p.is_absolute():
|
|
237
|
-
p = project_root / p
|
|
238
|
-
return p
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
def _sha256(path: Path) -> str | None:
|
|
242
|
-
try:
|
|
243
|
-
return hashlib.sha256(path.read_bytes()).hexdigest()
|
|
244
|
-
except OSError:
|
|
245
|
-
return None
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
#: Inline-tag identifier this package writes into deployed Markdown
|
|
249
|
-
#: frontmatter (P5.1). Kept in sync with ``install.PACKAGE_TAG_ID``;
|
|
250
|
-
#: duplicated here to keep ``cmd_doctor`` import-light (no pull on the
|
|
251
|
-
#: installer module from the CLI).
|
|
252
|
-
PACKAGE_TAG_ID = "event4u/agent-config"
|
|
253
|
-
|
|
254
|
-
_FRONTMATTER_PACKAGE_RE = re.compile(
|
|
255
|
-
r"^package:\s*(.+?)\s*$", re.MULTILINE,
|
|
256
|
-
)
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
def _read_inline_package_tag(path: Path) -> str | None | _Sentinel:
|
|
260
|
-
"""Extract the inline ``package:`` value from a Markdown frontmatter.
|
|
261
|
-
|
|
262
|
-
Returns ``NO_FRONTMATTER`` when ``path`` is not a Markdown file or
|
|
263
|
-
has no leading ``---`` block (P5.1: those files are out of scope).
|
|
264
|
-
Returns ``None`` when frontmatter is present but lacks a
|
|
265
|
-
``package:`` key. Returns the string value otherwise.
|
|
266
|
-
"""
|
|
267
|
-
if path.suffix != ".md":
|
|
268
|
-
return NO_FRONTMATTER
|
|
269
|
-
try:
|
|
270
|
-
text = path.read_text(encoding="utf-8")
|
|
271
|
-
except OSError:
|
|
272
|
-
return NO_FRONTMATTER
|
|
273
|
-
if not (text.startswith("---\n") or text.startswith("---\r\n")):
|
|
274
|
-
return NO_FRONTMATTER
|
|
275
|
-
lines = text.splitlines()
|
|
276
|
-
close_idx: int | None = None
|
|
277
|
-
for i in range(1, len(lines)):
|
|
278
|
-
if lines[i].rstrip() == "---":
|
|
279
|
-
close_idx = i
|
|
280
|
-
break
|
|
281
|
-
if close_idx is None:
|
|
282
|
-
return NO_FRONTMATTER
|
|
283
|
-
block = "\n".join(lines[1:close_idx])
|
|
284
|
-
m = _FRONTMATTER_PACKAGE_RE.search(block)
|
|
285
|
-
if not m:
|
|
286
|
-
return None
|
|
287
|
-
return m.group(1).strip().strip("'\"")
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
def _fix_hint(category: str, kind: str | None) -> str:
|
|
291
|
-
"""Return a one-line remediation hint for the surfaced item."""
|
|
292
|
-
if category == "missing":
|
|
293
|
-
return "run `./agent-config sync` to re-install"
|
|
294
|
-
if category == "modified":
|
|
295
|
-
return "commit the local change, or re-install with --force"
|
|
296
|
-
if category == "foreign":
|
|
297
|
-
return (
|
|
298
|
-
"identify owning tool, or run `./agent-config prune` "
|
|
299
|
-
"if confirmed orphan"
|
|
300
|
-
)
|
|
301
|
-
if category == "tag-drift":
|
|
302
|
-
return (
|
|
303
|
-
"re-install with --force to restore the inline tag, "
|
|
304
|
-
"or remove the file if it is no longer ours"
|
|
305
|
-
)
|
|
306
|
-
return ""
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
def _collect_manifest_entries(
|
|
310
|
-
project_root: Path, manifest: dict[str, Any],
|
|
311
|
-
) -> tuple[
|
|
312
|
-
list[tuple[str, Path, str, str | None]], # (tool, abs_path, kind, sha)
|
|
313
|
-
set[Path], # resolved-known set
|
|
314
|
-
]:
|
|
315
|
-
"""Flatten v2 ``tools[].files[]`` into per-file records.
|
|
316
|
-
|
|
317
|
-
Returns the records list and a set of resolved absolute paths so
|
|
318
|
-
the foreign-file scan can skip anything the manifest claims.
|
|
319
|
-
"""
|
|
320
|
-
records: list[tuple[str, Path, str, str | None]] = []
|
|
321
|
-
known: set[Path] = set()
|
|
322
|
-
for tool in manifest.get("tools") or []:
|
|
323
|
-
if tool.get("scope") != "project":
|
|
324
|
-
continue
|
|
325
|
-
tool_id = str(tool.get("name", ""))
|
|
326
|
-
for entry in tool.get("files") or []:
|
|
327
|
-
raw = entry.get("path") or ""
|
|
328
|
-
if not raw:
|
|
329
|
-
continue
|
|
330
|
-
kind = entry.get("kind") or ""
|
|
331
|
-
target = _resolve_path(project_root, raw)
|
|
332
|
-
try:
|
|
333
|
-
resolved = target.resolve()
|
|
334
|
-
except OSError:
|
|
335
|
-
resolved = target
|
|
336
|
-
records.append((tool_id, target, kind, entry.get("sha256")))
|
|
337
|
-
known.add(resolved)
|
|
338
|
-
return records, known
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
def _scan_foreign(
|
|
342
|
-
project_root: Path,
|
|
343
|
-
manifest: dict[str, Any],
|
|
344
|
-
known: set[Path],
|
|
345
|
-
) -> list[Path]:
|
|
346
|
-
"""Walk every declared deploy root and surface unclaimed files.
|
|
347
|
-
|
|
348
|
-
Only ``regular files`` under ``deploy_roots`` count; directories and
|
|
349
|
-
symlinks are followed but the bookkeeping is on the resolved final
|
|
350
|
-
path so a manifest claim via either path silences both surfaces.
|
|
351
|
-
Falls back to :data:`installed_tools.DEFAULT_DEPLOY_ROOTS` when the
|
|
352
|
-
manifest lacks an explicit ``deploy_roots`` list.
|
|
353
|
-
"""
|
|
354
|
-
roots = manifest.get("deploy_roots") or list(
|
|
355
|
-
installed_tools.DEFAULT_DEPLOY_ROOTS,
|
|
356
|
-
)
|
|
357
|
-
foreign: list[Path] = []
|
|
358
|
-
seen: set[Path] = set()
|
|
359
|
-
for root_rel in roots:
|
|
360
|
-
root = _resolve_path(project_root, str(root_rel))
|
|
361
|
-
if not root.exists() or not root.is_dir():
|
|
362
|
-
continue
|
|
363
|
-
for child in root.rglob("*"):
|
|
364
|
-
if not child.is_file():
|
|
365
|
-
continue
|
|
366
|
-
try:
|
|
367
|
-
resolved = child.resolve()
|
|
368
|
-
except OSError:
|
|
369
|
-
resolved = child
|
|
370
|
-
if resolved in known or resolved in seen:
|
|
371
|
-
continue
|
|
372
|
-
seen.add(resolved)
|
|
373
|
-
foreign.append(child)
|
|
374
|
-
foreign.sort()
|
|
375
|
-
return foreign
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
def _classify(
|
|
379
|
-
records: list[tuple[str, Path, str, str | None]],
|
|
380
|
-
) -> tuple[
|
|
381
|
-
list[dict[str, Any]],
|
|
382
|
-
list[dict[str, Any]],
|
|
383
|
-
list[dict[str, Any]],
|
|
384
|
-
]:
|
|
385
|
-
"""Split manifest records into missing / modified / tag-drift lists.
|
|
386
|
-
|
|
387
|
-
Tag-drift inspection (P5.2) is restricted to manifest entries that
|
|
388
|
-
point at a present ``.md`` file with a frontmatter block. A file
|
|
389
|
-
that has frontmatter but whose ``package:`` value disagrees with
|
|
390
|
-
:data:`PACKAGE_TAG_ID` — or that has dropped the key entirely —
|
|
391
|
-
surfaces here. Files without frontmatter are silently ignored per
|
|
392
|
-
the P5.1 contract (we never synthesise frontmatter).
|
|
393
|
-
"""
|
|
394
|
-
missing: list[dict[str, Any]] = []
|
|
395
|
-
modified: list[dict[str, Any]] = []
|
|
396
|
-
tag_drift: list[dict[str, Any]] = []
|
|
397
|
-
for tool_id, target, kind, expected in records:
|
|
398
|
-
if not target.exists():
|
|
399
|
-
missing.append({
|
|
400
|
-
"tool": tool_id, "path": str(target), "kind": kind,
|
|
401
|
-
"fix": _fix_hint("missing", kind),
|
|
402
|
-
})
|
|
403
|
-
continue
|
|
404
|
-
tag = _read_inline_package_tag(target)
|
|
405
|
-
if not isinstance(tag, _Sentinel) and tag != PACKAGE_TAG_ID:
|
|
406
|
-
tag_drift.append({
|
|
407
|
-
"tool": tool_id, "path": str(target), "kind": kind,
|
|
408
|
-
"expected": PACKAGE_TAG_ID,
|
|
409
|
-
"found": "" if tag is None else tag,
|
|
410
|
-
"fix": _fix_hint("tag-drift", kind),
|
|
411
|
-
})
|
|
412
|
-
if expected is None:
|
|
413
|
-
continue
|
|
414
|
-
actual = _sha256(target)
|
|
415
|
-
if actual is None or actual == expected:
|
|
416
|
-
continue
|
|
417
|
-
modified.append({
|
|
418
|
-
"tool": tool_id, "path": str(target), "kind": kind,
|
|
419
|
-
"fix": _fix_hint("modified", kind),
|
|
420
|
-
})
|
|
421
|
-
return missing, modified, tag_drift
|
|
422
|
-
|
|
423
|
-
|
|
424
|
-
def _foreign_records(
|
|
425
|
-
project_root: Path, foreign: list[Path],
|
|
426
|
-
) -> list[dict[str, Any]]:
|
|
427
|
-
out: list[dict[str, Any]] = []
|
|
428
|
-
for p in foreign:
|
|
429
|
-
try:
|
|
430
|
-
rel = p.relative_to(project_root)
|
|
431
|
-
path_str = str(rel)
|
|
432
|
-
except ValueError:
|
|
433
|
-
path_str = str(p)
|
|
434
|
-
out.append({
|
|
435
|
-
"tool": "",
|
|
436
|
-
"path": path_str,
|
|
437
|
-
"kind": "deployed",
|
|
438
|
-
"fix": _fix_hint("foreign", "deployed"),
|
|
439
|
-
})
|
|
440
|
-
return out
|
|
441
|
-
|
|
442
|
-
|
|
443
|
-
# ---------------------------------------------------------------------------
|
|
444
|
-
# Health checks (Phase 2 of road-to-surface-discipline).
|
|
445
|
-
# ---------------------------------------------------------------------------
|
|
446
|
-
|
|
447
|
-
#: Ordered registry of structured health-check identifiers. The order
|
|
448
|
-
#: is preserved in text and JSON output. Adding a check requires both
|
|
449
|
-
#: a runner below and an entry in :func:`_run_checks`.
|
|
450
|
-
CHECK_IDS = (
|
|
451
|
-
"scope",
|
|
452
|
-
"global-binary",
|
|
453
|
-
"stale-orphans",
|
|
454
|
-
"manifest-integrity",
|
|
455
|
-
"lockfile-freshness",
|
|
456
|
-
"bridge-drift",
|
|
457
|
-
"mcp-mode",
|
|
458
|
-
"mcp-beta-readiness",
|
|
459
|
-
"offline-readiness",
|
|
460
|
-
"python-runtime",
|
|
461
|
-
"tier-usage-readiness",
|
|
462
|
-
"council-cli",
|
|
463
|
-
"unsupported-combos",
|
|
464
|
-
"wizard-state",
|
|
465
|
-
)
|
|
466
|
-
|
|
467
|
-
#: Checks that need only the project root (or no input at all) and run
|
|
468
|
-
#: regardless of whether a project lockfile exists. Under ADR-020
|
|
469
|
-
#: global-only (bridge marker present, no ``installed-tools.lock``) these
|
|
470
|
-
#: still produce a real verdict. ``scope`` lives here because
|
|
471
|
-
#: :func:`_check_scope` reads only ``project_root`` — the roadmap prose
|
|
472
|
-
#: that grouped it with the manifest checks predates this code reality
|
|
473
|
-
#: (AI council, claude-sonnet-4-5 + gpt-4o, design lens, 2026-06-02).
|
|
474
|
-
GLOBAL_CHECK_IDS: frozenset[str] = frozenset({
|
|
475
|
-
"scope",
|
|
476
|
-
"global-binary",
|
|
477
|
-
"stale-orphans",
|
|
478
|
-
"mcp-mode",
|
|
479
|
-
"mcp-beta-readiness",
|
|
480
|
-
"offline-readiness",
|
|
481
|
-
"python-runtime",
|
|
482
|
-
"tier-usage-readiness",
|
|
483
|
-
"council-cli",
|
|
484
|
-
"wizard-state",
|
|
485
|
-
})
|
|
486
|
-
|
|
487
|
-
#: Checks that genuinely cannot run without the project manifest. Without
|
|
488
|
-
#: a lockfile they report ``skipped`` rather than a misleading verdict.
|
|
489
|
-
#: ``bridge-drift`` is deliberately **not** here: it is scope-aware —
|
|
490
|
-
#: a manifest-derived drift roll-up when the lockfile exists, and a
|
|
491
|
-
#: "drift not applicable (global-only consumer)" verdict when it does
|
|
492
|
-
#: not (computed in :func:`_check_bridge_drift_no_manifest`, keeping
|
|
493
|
-
#: :func:`_check_bridge_drift` a pure roll-up per the council's SRP point).
|
|
494
|
-
MANIFEST_REQUIRED_CHECK_IDS: frozenset[str] = frozenset({
|
|
495
|
-
"manifest-integrity",
|
|
496
|
-
"lockfile-freshness",
|
|
497
|
-
"unsupported-combos",
|
|
498
|
-
})
|
|
499
|
-
|
|
500
|
-
#: Project-root-relative path of the ADR-020 global-only consumer marker.
|
|
501
|
-
BRIDGE_MARKER_RELATIVE = "agents/.event4u-bridge.yml"
|
|
502
|
-
|
|
503
|
-
#: Repair targets that ``--repair <id>`` accepts. Each id maps to a
|
|
504
|
-
#: function in :func:`_run_repair` that resets the named artefact and
|
|
505
|
-
#: returns an exit code. Additive set: introduce by adding a new id
|
|
506
|
-
#: here, a runner inside :func:`_run_repair`, and (optionally) a
|
|
507
|
-
#: matching health check above.
|
|
508
|
-
REPAIR_IDS = ("wizard-state",)
|
|
509
|
-
|
|
510
|
-
#: Six gates that govern the MCP `experimental → beta` promotion. The
|
|
511
|
-
#: artefact path under each gate id mirrors `tests/test_mcp_beta_gates.py`
|
|
512
|
-
#: and the contract in `docs/contracts/mcp-beta-criteria.md`.
|
|
513
|
-
MCP_BETA_GATES: tuple[tuple[str, str], ...] = (
|
|
514
|
-
("gate-1-external-client", "tests/mcp/external-clients"),
|
|
515
|
-
("gate-2-bearer-auth", "tests/mcp/auth"),
|
|
516
|
-
("gate-3-parity-smoke", "tests/mcp/parity"),
|
|
517
|
-
("gate-4-healthz-load", "tests/mcp/load/healthz.k6.js"),
|
|
518
|
-
("gate-5-rate-limit", "docs/contracts/mcp-rate-limit.md"),
|
|
519
|
-
("gate-6-no-drift", ".github/workflows/mcp-no-drift.yml"),
|
|
520
|
-
)
|
|
521
|
-
|
|
522
|
-
#: Visible status → glyph map. ``warn`` keeps a trailing space so the
|
|
523
|
-
#: rendered output stays in a single visual column with the other glyphs.
|
|
524
|
-
STATUS_SYMBOLS = {"ok": "✅", "warn": "⚠️ ", "fail": "❌", "skipped": "⏭️ "}
|
|
525
|
-
|
|
526
|
-
#: Minimum Python interpreter the CLI targets. Bumped in lockstep with
|
|
527
|
-
#: ``from __future__ import annotations`` + PEP-604 syntax usage.
|
|
528
|
-
MIN_PYTHON = (3, 10)
|
|
529
|
-
|
|
530
|
-
|
|
531
|
-
def _check_scope(project_root: Path) -> dict[str, Any]:
|
|
532
|
-
"""Distinguish a standalone project root from a monorepo workspace.
|
|
533
|
-
|
|
534
|
-
Surfaces ``warn`` when ``project_root`` looks like a monorepo root
|
|
535
|
-
so the operator knows to run ``doctor`` per-package; otherwise
|
|
536
|
-
reports the project as a standalone target.
|
|
537
|
-
"""
|
|
538
|
-
for marker in ("pnpm-workspace.yaml", "lerna.json"):
|
|
539
|
-
if (project_root / marker).exists():
|
|
540
|
-
return {
|
|
541
|
-
"id": "scope", "status": "warn",
|
|
542
|
-
"message": f"monorepo root detected ({marker})",
|
|
543
|
-
"remedy": "run `agent-config doctor` from each workspace package",
|
|
544
|
-
}
|
|
545
|
-
pkg = project_root / "package.json"
|
|
546
|
-
if pkg.exists():
|
|
547
|
-
try:
|
|
548
|
-
data = json.loads(pkg.read_text(encoding="utf-8"))
|
|
549
|
-
if data.get("workspaces"):
|
|
550
|
-
return {
|
|
551
|
-
"id": "scope", "status": "warn",
|
|
552
|
-
"message": "package.json declares workspaces (monorepo root)",
|
|
553
|
-
"remedy": "run `agent-config doctor` from each workspace package",
|
|
554
|
-
}
|
|
555
|
-
except (OSError, ValueError):
|
|
556
|
-
pass
|
|
557
|
-
return {
|
|
558
|
-
"id": "scope", "status": "ok",
|
|
559
|
-
"message": "standalone project root",
|
|
560
|
-
"remedy": "",
|
|
561
|
-
}
|
|
562
|
-
|
|
563
|
-
|
|
564
|
-
def _check_manifest_integrity(manifest: dict[str, Any]) -> dict[str, Any]:
|
|
565
|
-
"""Verify the manifest carries a writer version and a known schema."""
|
|
566
|
-
schema = manifest.get("schema_version")
|
|
567
|
-
version = manifest.get("agent_config_version")
|
|
568
|
-
if not version:
|
|
569
|
-
return {
|
|
570
|
-
"id": "manifest-integrity", "status": "warn",
|
|
571
|
-
"message": "manifest lacks `agent_config_version`",
|
|
572
|
-
"remedy": "re-run `./agent-config init` to record the writer version",
|
|
573
|
-
}
|
|
574
|
-
if schema not in installed_tools.SCHEMA_VERSIONS_SUPPORTED:
|
|
575
|
-
return {
|
|
576
|
-
"id": "manifest-integrity", "status": "warn",
|
|
577
|
-
"message": f"unknown schema_version: {schema!r}",
|
|
578
|
-
"remedy": "upgrade @event4u/agent-config to a writer that "
|
|
579
|
-
"recognises this schema",
|
|
580
|
-
}
|
|
581
|
-
return {
|
|
582
|
-
"id": "manifest-integrity", "status": "ok",
|
|
583
|
-
"message": f"schema v{schema}, written by agent-config {version}",
|
|
584
|
-
"remedy": "",
|
|
585
|
-
}
|
|
586
|
-
|
|
587
|
-
|
|
588
|
-
def _package_root() -> Path:
|
|
589
|
-
"""Resolve the @event4u/agent-config package root (this repo)."""
|
|
590
|
-
return Path(__file__).resolve().parents[3]
|
|
591
|
-
|
|
592
|
-
|
|
593
|
-
def _current_package_version() -> str | None:
|
|
594
|
-
"""Read ``version`` from this package's ``package.json``; ``None`` on error."""
|
|
595
|
-
try:
|
|
596
|
-
data = json.loads(
|
|
597
|
-
(_package_root() / "package.json").read_text(encoding="utf-8"),
|
|
598
|
-
)
|
|
599
|
-
v = data.get("version")
|
|
600
|
-
if isinstance(v, str) and v.strip():
|
|
601
|
-
return v.strip()
|
|
602
|
-
except (OSError, ValueError):
|
|
603
|
-
pass
|
|
604
|
-
return None
|
|
605
|
-
|
|
606
|
-
|
|
607
|
-
def _check_lockfile_freshness(manifest: dict[str, Any]) -> dict[str, Any]:
|
|
608
|
-
"""Compare the manifest writer version against the current package."""
|
|
609
|
-
recorded = manifest.get("agent_config_version")
|
|
610
|
-
current = _current_package_version()
|
|
611
|
-
if not recorded:
|
|
612
|
-
return {
|
|
613
|
-
"id": "lockfile-freshness", "status": "warn",
|
|
614
|
-
"message": "manifest has no writer version recorded",
|
|
615
|
-
"remedy": "re-run `./agent-config init` to refresh the manifest",
|
|
616
|
-
}
|
|
617
|
-
if current is None:
|
|
618
|
-
return {
|
|
619
|
-
"id": "lockfile-freshness", "status": "warn",
|
|
620
|
-
"message": f"manifest written by {recorded}; current package version unknown",
|
|
621
|
-
"remedy": "verify the package install (package.json missing or unreadable)",
|
|
622
|
-
}
|
|
623
|
-
if recorded != current:
|
|
624
|
-
return {
|
|
625
|
-
"id": "lockfile-freshness", "status": "warn",
|
|
626
|
-
"message": f"manifest writer {recorded} != current package {current}",
|
|
627
|
-
"remedy": "re-run `./agent-config sync` to refresh the manifest "
|
|
628
|
-
"against the current package",
|
|
629
|
-
}
|
|
630
|
-
return {
|
|
631
|
-
"id": "lockfile-freshness", "status": "ok",
|
|
632
|
-
"message": f"manifest and package both at {current}",
|
|
633
|
-
"remedy": "",
|
|
634
|
-
}
|
|
635
|
-
|
|
636
|
-
|
|
637
|
-
def _check_global_binary(project_root: Path) -> dict[str, Any]:
|
|
638
|
-
"""`agent-config` PATH resolution + global-install version parity.
|
|
639
|
-
|
|
640
|
-
The Claude plugin hook resolves `agent-config` from PATH (ADR-020
|
|
641
|
-
global-only); if it is missing, every PostToolUse hook fails silently —
|
|
642
|
-
this is the root cause of the "dashboard never updates" report. Version
|
|
643
|
-
drift between the binary on PATH and the global install root means the
|
|
644
|
-
plugin may emit a hook spec the installed binary predates.
|
|
645
|
-
"""
|
|
646
|
-
binary = shutil.which("agent-config")
|
|
647
|
-
if not binary:
|
|
648
|
-
return {
|
|
649
|
-
"id": "global-binary", "status": "fail",
|
|
650
|
-
"message": "`agent-config` is not on PATH — Claude plugin hooks "
|
|
651
|
-
"cannot resolve it (hooks silently no-op)",
|
|
652
|
-
"remedy": "npm install -g @event4u/agent-config, then re-run doctor",
|
|
653
|
-
}
|
|
654
|
-
lock = installed_lock.read_lockfile()
|
|
655
|
-
global_v = ""
|
|
656
|
-
if isinstance(lock, dict):
|
|
657
|
-
raw = lock.get("agent_config_version")
|
|
658
|
-
global_v = raw.strip().lstrip("v") if isinstance(raw, str) else ""
|
|
659
|
-
binary_v = (_current_package_version() or "").lstrip("v")
|
|
660
|
-
if global_v and binary_v and global_v != binary_v:
|
|
661
|
-
return {
|
|
662
|
-
"id": "global-binary", "status": "warn",
|
|
663
|
-
"message": f"version drift — binary {binary_v}, global install "
|
|
664
|
-
f"{global_v}",
|
|
665
|
-
"remedy": "agent-config upgrade (refresh the global install + plugin)",
|
|
666
|
-
}
|
|
667
|
-
bridge = project_root / "agents" / ".event4u-bridge.yml"
|
|
668
|
-
bridge_note = "" if bridge.is_file() else " · no project bridge marker"
|
|
669
|
-
return {
|
|
670
|
-
"id": "global-binary", "status": "ok",
|
|
671
|
-
"message": f"on PATH ({binary}); version {binary_v or 'unknown'}"
|
|
672
|
-
f"{bridge_note}",
|
|
673
|
-
"remedy": "" if not bridge_note else
|
|
674
|
-
"agent-config refresh --project (scaffold the bridge marker)",
|
|
675
|
-
}
|
|
676
|
-
|
|
677
|
-
|
|
678
|
-
def _check_bridge_drift(
|
|
679
|
-
missing: list[dict[str, Any]],
|
|
680
|
-
modified: list[dict[str, Any]],
|
|
681
|
-
foreign: list[dict[str, Any]],
|
|
682
|
-
tag_drift: list[dict[str, Any]],
|
|
683
|
-
) -> dict[str, Any]:
|
|
684
|
-
"""Roll the four drift category counts into a single health verdict."""
|
|
685
|
-
total = len(missing) + len(modified) + len(foreign) + len(tag_drift)
|
|
686
|
-
if total == 0:
|
|
687
|
-
return {
|
|
688
|
-
"id": "bridge-drift", "status": "ok",
|
|
689
|
-
"message": "manifest matches filesystem (no drift)",
|
|
690
|
-
"remedy": "",
|
|
691
|
-
}
|
|
692
|
-
parts = []
|
|
693
|
-
if missing:
|
|
694
|
-
parts.append(f"{len(missing)} missing")
|
|
695
|
-
if modified:
|
|
696
|
-
parts.append(f"{len(modified)} modified")
|
|
697
|
-
if foreign:
|
|
698
|
-
parts.append(f"{len(foreign)} foreign")
|
|
699
|
-
if tag_drift:
|
|
700
|
-
parts.append(f"{len(tag_drift)} tag-drift")
|
|
701
|
-
return {
|
|
702
|
-
"id": "bridge-drift", "status": "fail",
|
|
703
|
-
"message": f"{total} drift item(s): {', '.join(parts)}",
|
|
704
|
-
"remedy": "see the drift section below or run `./agent-config sync`",
|
|
705
|
-
}
|
|
706
|
-
|
|
707
|
-
|
|
708
|
-
def _check_mcp_mode(project_root: Path) -> dict[str, Any]:
|
|
709
|
-
"""Detect which MCP config file the project advertises, if any."""
|
|
710
|
-
candidates = (
|
|
711
|
-
(".cursor/mcp.json", "cursor"),
|
|
712
|
-
(".ai/mcp/mcp.json", "ai-mcp"),
|
|
713
|
-
("mcp.json", "root"),
|
|
714
|
-
)
|
|
715
|
-
found: list[str] = []
|
|
716
|
-
for rel, label in candidates:
|
|
717
|
-
path = project_root / rel
|
|
718
|
-
if not path.exists():
|
|
719
|
-
continue
|
|
720
|
-
try:
|
|
721
|
-
json.loads(path.read_text(encoding="utf-8"))
|
|
722
|
-
found.append(f"{label} ({rel})")
|
|
723
|
-
except (OSError, ValueError):
|
|
724
|
-
return {
|
|
725
|
-
"id": "mcp-mode", "status": "warn",
|
|
726
|
-
"message": f"MCP config at {rel} is not valid JSON",
|
|
727
|
-
"remedy": f"fix or remove `{rel}` (see docs/architecture.md § MCP)",
|
|
728
|
-
}
|
|
729
|
-
if not found:
|
|
730
|
-
return {
|
|
731
|
-
"id": "mcp-mode", "status": "ok",
|
|
732
|
-
"message": "no MCP config present (MCP Beta off)",
|
|
733
|
-
"remedy": "",
|
|
734
|
-
}
|
|
735
|
-
return {
|
|
736
|
-
"id": "mcp-mode", "status": "ok",
|
|
737
|
-
"message": f"MCP config detected: {', '.join(found)}",
|
|
738
|
-
"remedy": "",
|
|
739
|
-
}
|
|
740
|
-
|
|
741
|
-
|
|
742
|
-
def _check_offline_readiness() -> dict[str, Any]:
|
|
743
|
-
"""Verify the verified-offline install entrypoint ships with the package."""
|
|
744
|
-
script = _package_root() / "src" / "scripts" / "hermetic-install.sh"
|
|
745
|
-
if not script.exists():
|
|
746
|
-
return {
|
|
747
|
-
"id": "offline-readiness", "status": "warn",
|
|
748
|
-
"message": "src/scripts/hermetic-install.sh not found in package",
|
|
749
|
-
"remedy": "reinstall @event4u/agent-config or pull missing files",
|
|
750
|
-
}
|
|
751
|
-
return {
|
|
752
|
-
"id": "offline-readiness", "status": "ok",
|
|
753
|
-
"message": "verified-offline install entrypoint present",
|
|
754
|
-
"remedy": "",
|
|
755
|
-
}
|
|
756
|
-
|
|
757
|
-
|
|
758
|
-
def _check_stale_orphans() -> dict[str, Any]:
|
|
759
|
-
"""Surface package-tagged files on disk that the global-deploy inventory
|
|
760
|
-
no longer tracks — leftovers from a pre-inventory installer or a
|
|
761
|
-
since-removed / renamed artefact (e.g. ``create-pr`` → ``pr/create``).
|
|
762
|
-
|
|
763
|
-
Read-only: counts candidates per recorded anchor, never deletes. The
|
|
764
|
-
remedy is a global redeploy, whose always-run tag sweep
|
|
765
|
-
(``reap_tagged_orphans``) reconciles them. Scans only the package-owned
|
|
766
|
-
subtrees the inventory recorded (not the whole anchor), and counts a
|
|
767
|
-
file only when it carries this package's ``package:`` tag — user-authored
|
|
768
|
-
files in shared anchors never register.
|
|
769
|
-
"""
|
|
770
|
-
try: # package-style import (installed package / pytest)
|
|
771
|
-
from scripts._lib import global_deploy_inventory as gdi
|
|
772
|
-
except ImportError: # pragma: no cover — script-style sys.path fallback
|
|
773
|
-
from _lib import global_deploy_inventory as gdi # type: ignore[no-redef]
|
|
774
|
-
|
|
775
|
-
tools = gdi.load_inventory().get("tools", {})
|
|
776
|
-
if not isinstance(tools, dict) or not tools:
|
|
777
|
-
return {
|
|
778
|
-
"id": "stale-orphans", "status": "ok",
|
|
779
|
-
"message": "no global-deploy inventory yet — nothing to reconcile",
|
|
780
|
-
"remedy": "",
|
|
781
|
-
}
|
|
782
|
-
orphan_count = 0
|
|
783
|
-
sample: list[str] = []
|
|
784
|
-
for tool_id, entry in sorted(tools.items()):
|
|
785
|
-
if not isinstance(entry, dict):
|
|
786
|
-
continue
|
|
787
|
-
anchor_raw = entry.get("anchor")
|
|
788
|
-
recorded = entry.get("files")
|
|
789
|
-
if not isinstance(anchor_raw, str) or not isinstance(recorded, list):
|
|
790
|
-
continue
|
|
791
|
-
anchor = Path(anchor_raw).expanduser()
|
|
792
|
-
if not anchor.is_dir():
|
|
793
|
-
continue
|
|
794
|
-
recorded_set = {r for r in recorded if isinstance(r, str)}
|
|
795
|
-
# Bound the scan to the top-level subtrees the package actually owns.
|
|
796
|
-
owned_roots = {r.split("/", 1)[0] for r in recorded_set if "/" in r}
|
|
797
|
-
for root_name in sorted(owned_roots):
|
|
798
|
-
root = anchor / root_name
|
|
799
|
-
if not root.is_dir():
|
|
800
|
-
continue
|
|
801
|
-
for md in root.rglob("*.md"):
|
|
802
|
-
if md.is_dir():
|
|
803
|
-
continue
|
|
804
|
-
try:
|
|
805
|
-
rel = md.relative_to(anchor).as_posix()
|
|
806
|
-
except ValueError:
|
|
807
|
-
continue
|
|
808
|
-
if rel in recorded_set:
|
|
809
|
-
continue
|
|
810
|
-
tag = _read_inline_package_tag(md)
|
|
811
|
-
if isinstance(tag, _Sentinel) or tag != PACKAGE_TAG_ID:
|
|
812
|
-
continue
|
|
813
|
-
orphan_count += 1
|
|
814
|
-
if len(sample) < 5:
|
|
815
|
-
sample.append(f"{tool_id}:{rel}")
|
|
816
|
-
if orphan_count == 0:
|
|
817
|
-
return {
|
|
818
|
-
"id": "stale-orphans", "status": "ok",
|
|
819
|
-
"message": "no stale package-tagged files under recorded anchors",
|
|
820
|
-
"remedy": "",
|
|
821
|
-
}
|
|
822
|
-
return {
|
|
823
|
-
"id": "stale-orphans", "status": "warn",
|
|
824
|
-
"message": (
|
|
825
|
-
f"{orphan_count} stale package-tagged file(s) not tracked by the "
|
|
826
|
-
f"deploy inventory (e.g. {', '.join(sample)})"
|
|
827
|
-
),
|
|
828
|
-
"remedy": "run `agent-config global` to reap them "
|
|
829
|
-
"(the tag sweep reconciles on every deploy)",
|
|
830
|
-
}
|
|
831
|
-
|
|
832
|
-
|
|
833
|
-
def _check_python_runtime() -> dict[str, Any]:
|
|
834
|
-
"""Confirm the interpreter is at least :data:`MIN_PYTHON`."""
|
|
835
|
-
cur = sys.version_info[:2]
|
|
836
|
-
need = MIN_PYTHON
|
|
837
|
-
if cur < need:
|
|
838
|
-
return {
|
|
839
|
-
"id": "python-runtime", "status": "fail",
|
|
840
|
-
"message": f"python {cur[0]}.{cur[1]} below required {need[0]}.{need[1]}",
|
|
841
|
-
"remedy": f"install python >= {need[0]}.{need[1]} and re-run",
|
|
842
|
-
}
|
|
843
|
-
return {
|
|
844
|
-
"id": "python-runtime", "status": "ok",
|
|
845
|
-
"message": f"python {cur[0]}.{cur[1]} meets {need[0]}.{need[1]}+ requirement",
|
|
846
|
-
"remedy": "",
|
|
847
|
-
}
|
|
848
|
-
|
|
849
|
-
|
|
850
|
-
def _check_mcp_beta_readiness(project_root: Path) -> dict[str, Any]:
|
|
851
|
-
"""Report the MCP `experimental → beta` promotion gate sheet.
|
|
852
|
-
|
|
853
|
-
Walks the six gates in :data:`MCP_BETA_GATES` and asks whether the
|
|
854
|
-
artefact named by each gate (test directory, k6 script, contract
|
|
855
|
-
doc, or workflow file) is present. Green when all six pass, warn
|
|
856
|
-
when any are missing — the gate stays red until evidence lands.
|
|
857
|
-
Mirrors `tests/test_mcp_beta_gates.py` so the doctor verdict and
|
|
858
|
-
the pytest sheet cannot drift.
|
|
859
|
-
"""
|
|
860
|
-
pending: list[str] = []
|
|
861
|
-
for gate_id, rel in MCP_BETA_GATES:
|
|
862
|
-
if not (project_root / rel).exists():
|
|
863
|
-
pending.append(f"{gate_id} ({rel})")
|
|
864
|
-
if not pending:
|
|
865
|
-
return {
|
|
866
|
-
"id": "mcp-beta-readiness", "status": "ok",
|
|
867
|
-
"message": "all 6 MCP beta gates green — promotion authorized",
|
|
868
|
-
"remedy": "",
|
|
869
|
-
}
|
|
870
|
-
return {
|
|
871
|
-
"id": "mcp-beta-readiness", "status": "warn",
|
|
872
|
-
"message": (
|
|
873
|
-
f"{len(pending)}/6 MCP beta gate(s) pending: "
|
|
874
|
-
f"{', '.join(pending)}"
|
|
875
|
-
),
|
|
876
|
-
"remedy": (
|
|
877
|
-
"produce the artefacts listed in "
|
|
878
|
-
"docs/contracts/mcp-beta-criteria.md (one per gate); "
|
|
879
|
-
"do not flip `experimental` wording until all 6 are green"
|
|
880
|
-
),
|
|
881
|
-
}
|
|
882
|
-
|
|
883
|
-
|
|
884
|
-
def _check_tier_usage_readiness(project_root: Path) -> dict[str, Any]:
|
|
885
|
-
"""Report whether tier-usage telemetry can drive empirical retiering.
|
|
886
|
-
|
|
887
|
-
Phase 5 of road-to-surface-discipline. Three terminal states:
|
|
888
|
-
|
|
889
|
-
* **ok** — telemetry on, log present, at least one record survived
|
|
890
|
-
the privacy floor.
|
|
891
|
-
* **warn (disabled)** — telemetry off; no signal collected.
|
|
892
|
-
Default-off doctrine; user opt-in is the unlock.
|
|
893
|
-
* **warn (no data)** — telemetry on but the log is absent / empty;
|
|
894
|
-
retiering decisions still rely on operator judgement until
|
|
895
|
-
enough records accumulate.
|
|
896
|
-
* **fail (poisoned)** — every record was rejected by the privacy
|
|
897
|
-
floor; the report would refuse to render. The log needs a manual
|
|
898
|
-
inspection.
|
|
899
|
-
|
|
900
|
-
Contract: ``docs/contracts/command-clusters.md`` § tier-usage signal.
|
|
901
|
-
"""
|
|
902
|
-
settings_file = project_settings_path(project_root)
|
|
903
|
-
log_path = project_root / ".agent-tier-usage.jsonl"
|
|
904
|
-
enabled = False
|
|
905
|
-
if settings_file.is_file():
|
|
906
|
-
try:
|
|
907
|
-
import yaml # type: ignore[import-not-found]
|
|
908
|
-
except ImportError:
|
|
909
|
-
yaml = None # type: ignore[assignment]
|
|
910
|
-
if yaml is not None:
|
|
911
|
-
try:
|
|
912
|
-
raw = yaml.safe_load(settings_file.read_text(encoding="utf-8")) or {}
|
|
913
|
-
except Exception:
|
|
914
|
-
raw = {}
|
|
915
|
-
tele = (raw.get("telemetry") if isinstance(raw, dict) else None) or {}
|
|
916
|
-
tu = tele.get("tier_usage") if isinstance(tele, dict) else None
|
|
917
|
-
if isinstance(tu, dict):
|
|
918
|
-
output = tu.get("output") if isinstance(tu.get("output"), dict) else {}
|
|
919
|
-
if isinstance(output.get("path"), str) and output["path"].strip():
|
|
920
|
-
log_path = project_root / output["path"].strip()
|
|
921
|
-
val = tu.get("enabled")
|
|
922
|
-
enabled = bool(val) if isinstance(val, bool) else (
|
|
923
|
-
isinstance(val, str) and val.strip().lower() in ("true", "yes", "on", "1")
|
|
924
|
-
)
|
|
925
|
-
|
|
926
|
-
if not enabled:
|
|
927
|
-
return {
|
|
928
|
-
"id": "tier-usage-readiness", "status": "warn",
|
|
929
|
-
"message": (
|
|
930
|
-
"tier-usage telemetry disabled — empirical retiering "
|
|
931
|
-
"decisions fall back to operator judgement"
|
|
932
|
-
),
|
|
933
|
-
"remedy": (
|
|
934
|
-
"set `telemetry.tier_usage.enabled: true` in "
|
|
935
|
-
".agent-settings.yml (default-off; opt-in)"
|
|
936
|
-
),
|
|
937
|
-
}
|
|
938
|
-
if not log_path.exists():
|
|
939
|
-
return {
|
|
940
|
-
"id": "tier-usage-readiness", "status": "warn",
|
|
941
|
-
"message": (
|
|
942
|
-
f"tier-usage telemetry on but {log_path.name} not yet "
|
|
943
|
-
"written — no signal accumulated"
|
|
944
|
-
),
|
|
945
|
-
"remedy": (
|
|
946
|
-
"run any tracked command to seed the log; the dispatcher "
|
|
947
|
-
"writes one record per invocation"
|
|
948
|
-
),
|
|
949
|
-
}
|
|
950
|
-
total = 0
|
|
951
|
-
valid = 0
|
|
952
|
-
allowed_fields = {"ts_bucket", "command", "tier", "outcome", "user_hash"}
|
|
953
|
-
allowed_outcomes = {"success", "error", "blocked"}
|
|
954
|
-
try:
|
|
955
|
-
with log_path.open("r", encoding="utf-8") as fh:
|
|
956
|
-
for line in fh:
|
|
957
|
-
line = line.strip()
|
|
958
|
-
if not line:
|
|
959
|
-
continue
|
|
960
|
-
total += 1
|
|
961
|
-
try:
|
|
962
|
-
obj = json.loads(line)
|
|
963
|
-
except json.JSONDecodeError:
|
|
964
|
-
continue
|
|
965
|
-
if not isinstance(obj, dict):
|
|
966
|
-
continue
|
|
967
|
-
if not set(obj.keys()).issubset(allowed_fields):
|
|
968
|
-
continue
|
|
969
|
-
if (
|
|
970
|
-
isinstance(obj.get("command"), str) and obj["command"]
|
|
971
|
-
and isinstance(obj.get("tier"), int) and obj["tier"] in (0, 1, 2, 3)
|
|
972
|
-
and obj.get("outcome") in allowed_outcomes
|
|
973
|
-
and isinstance(obj.get("user_hash"), str) and len(obj["user_hash"]) == 16
|
|
974
|
-
and isinstance(obj.get("ts_bucket"), str)
|
|
975
|
-
):
|
|
976
|
-
valid += 1
|
|
977
|
-
except OSError as exc:
|
|
978
|
-
return {
|
|
979
|
-
"id": "tier-usage-readiness", "status": "fail",
|
|
980
|
-
"message": f"cannot read {log_path.name}: {exc}",
|
|
981
|
-
"remedy": "fix permissions on the tier-usage log",
|
|
982
|
-
}
|
|
983
|
-
if total > 0 and valid == 0:
|
|
984
|
-
return {
|
|
985
|
-
"id": "tier-usage-readiness", "status": "fail",
|
|
986
|
-
"message": (
|
|
987
|
-
f"{total} record(s) in {log_path.name} but 0 passed the "
|
|
988
|
-
"privacy floor — report would refuse to render"
|
|
989
|
-
),
|
|
990
|
-
"remedy": (
|
|
991
|
-
"inspect the log; the dispatcher is writing records the "
|
|
992
|
-
"contract forbids (paths, argv, message bodies)"
|
|
993
|
-
),
|
|
994
|
-
}
|
|
995
|
-
if valid == 0:
|
|
996
|
-
return {
|
|
997
|
-
"id": "tier-usage-readiness", "status": "warn",
|
|
998
|
-
"message": f"{log_path.name} present but empty — no signal yet",
|
|
999
|
-
"remedy": "run any tracked command to seed the log",
|
|
1000
|
-
}
|
|
1001
|
-
return {
|
|
1002
|
-
"id": "tier-usage-readiness", "status": "ok",
|
|
1003
|
-
"message": (
|
|
1004
|
-
f"{valid} record(s) past the privacy floor in {log_path.name} "
|
|
1005
|
-
"— retiering signal available"
|
|
1006
|
-
),
|
|
1007
|
-
"remedy": "",
|
|
1008
|
-
}
|
|
1009
|
-
|
|
1010
|
-
|
|
1011
|
-
#: Provider → (default binary, billable flag). Mirrors the
|
|
1012
|
-
#: ``CliClient`` subclass attributes in ``scripts/ai_council/clients.py``
|
|
1013
|
-
#: without importing them at module load time (keeps doctor lightweight
|
|
1014
|
-
#: and robust if council deps fail to load).
|
|
1015
|
-
_CLI_PROVIDER_META: dict[str, tuple[str, bool]] = {
|
|
1016
|
-
"anthropic": ("claude", False),
|
|
1017
|
-
"openai": ("codex", False),
|
|
1018
|
-
"gemini": ("gemini", False),
|
|
1019
|
-
"xai": ("grok", True),
|
|
1020
|
-
"perplexity": ("perplexity", True),
|
|
1021
|
-
}
|
|
1022
|
-
|
|
1023
|
-
|
|
1024
|
-
def _check_council_cli(project_root: Path) -> dict[str, Any]:
|
|
1025
|
-
"""Health-probe each enabled ``mode: cli`` council member.
|
|
1026
|
-
|
|
1027
|
-
For every CLI member in ``agents/settings/.ai-council.yml`` reports binary
|
|
1028
|
-
presence (via ``shutil.which``), billable flag, daily-quota state
|
|
1029
|
-
(used/cap or used/—), and rolls up to a single status icon.
|
|
1030
|
-
|
|
1031
|
-
No subprocess is spawned: ``--version`` probes would defeat the
|
|
1032
|
-
cheap-by-default contract and surface flaky network paths (Codex
|
|
1033
|
-
talks home on first run). Binary presence + the cached
|
|
1034
|
-
``cli-calls.json`` counter cover the actionable failure modes.
|
|
1035
|
-
|
|
1036
|
-
Status rules:
|
|
1037
|
-
|
|
1038
|
-
- ``ok`` — every enabled CLI member has its binary on PATH AND
|
|
1039
|
-
(when capped) usage is below ``warn_at``.
|
|
1040
|
-
- ``warn`` — at least one binary is missing OR usage crosses
|
|
1041
|
-
``warn_at`` for at least one capped member.
|
|
1042
|
-
- returns ``ok`` with "no council config" if no config is found in any
|
|
1043
|
-
scope — user-global ``~/.event4u/agent-config/settings/.ai-council.yml``, an
|
|
1044
|
-
explicit ``$AI_COUNCIL_CONFIG``, or a project-local
|
|
1045
|
-
``agents/settings/.ai-council.yml`` — e.g. the council is not set up
|
|
1046
|
-
yet.
|
|
1047
|
-
"""
|
|
1048
|
-
try:
|
|
1049
|
-
from scripts.ai_council.clients import load_cli_call_counts
|
|
1050
|
-
from scripts.ai_council.config import (
|
|
1051
|
-
load_council_config, resolve_config_path,
|
|
1052
|
-
)
|
|
1053
|
-
except Exception as exc: # noqa: BLE001 — defensive: doctor must not crash
|
|
1054
|
-
return {
|
|
1055
|
-
"id": "council-cli", "status": "warn",
|
|
1056
|
-
"message": f"council deps unavailable ({type(exc).__name__})",
|
|
1057
|
-
"remedy": "install PyYAML and ensure scripts/ai_council is importable",
|
|
1058
|
-
}
|
|
1059
|
-
council_path = resolve_config_path(project_root)
|
|
1060
|
-
if not council_path.exists():
|
|
1061
|
-
return {
|
|
1062
|
-
"id": "council-cli", "status": "ok",
|
|
1063
|
-
"message": f"no council config ({council_path} not present)",
|
|
1064
|
-
"remedy": (
|
|
1065
|
-
"create the user-global council config at "
|
|
1066
|
-
f"{council_path} (see docs/contracts/ai-council-config.md)"
|
|
1067
|
-
),
|
|
1068
|
-
}
|
|
1069
|
-
try:
|
|
1070
|
-
cfg = load_council_config(council_path)
|
|
1071
|
-
except Exception as exc: # noqa: BLE001
|
|
1072
|
-
return {
|
|
1073
|
-
"id": "council-cli", "status": "warn",
|
|
1074
|
-
"message": f"council config invalid: {exc}",
|
|
1075
|
-
"remedy": f"fix {council_path} and re-run doctor",
|
|
1076
|
-
}
|
|
1077
|
-
cli_members: list[tuple[str, Any]] = [
|
|
1078
|
-
(name, m) for name, m in cfg.members.items()
|
|
1079
|
-
if m.enabled and m.mode == "cli" and name in _CLI_PROVIDER_META
|
|
1080
|
-
]
|
|
1081
|
-
if not cli_members:
|
|
1082
|
-
return {
|
|
1083
|
-
"id": "council-cli", "status": "ok",
|
|
1084
|
-
"message": "no enabled CLI-mode members",
|
|
1085
|
-
"remedy": "",
|
|
1086
|
-
}
|
|
1087
|
-
counts = load_cli_call_counts()
|
|
1088
|
-
caps = cfg.cli_call_budget.max_calls_per_day
|
|
1089
|
-
warn_at = float(cfg.cli_call_budget.warn_at)
|
|
1090
|
-
missing: list[str] = []
|
|
1091
|
-
over_warn: list[str] = []
|
|
1092
|
-
lines: list[str] = []
|
|
1093
|
-
for name, member in cli_members:
|
|
1094
|
-
default_bin, billable = _CLI_PROVIDER_META[name]
|
|
1095
|
-
binary_name = member.binary or default_bin
|
|
1096
|
-
resolved = shutil.which(binary_name)
|
|
1097
|
-
binary_glyph = "✅" if resolved else "❌"
|
|
1098
|
-
if resolved is None:
|
|
1099
|
-
missing.append(name)
|
|
1100
|
-
used = int(counts.get(name, 0))
|
|
1101
|
-
cap = caps.get(name)
|
|
1102
|
-
if cap is not None:
|
|
1103
|
-
ratio = used / cap if cap > 0 else 0.0
|
|
1104
|
-
quota_glyph = "⚠️" if ratio >= warn_at else "✅"
|
|
1105
|
-
if ratio >= warn_at:
|
|
1106
|
-
over_warn.append(name)
|
|
1107
|
-
quota_str = f"{used}/{cap}"
|
|
1108
|
-
else:
|
|
1109
|
-
quota_glyph = "—"
|
|
1110
|
-
quota_str = f"{used}/—"
|
|
1111
|
-
billable_str = "billable" if billable else "subscription"
|
|
1112
|
-
lines.append(
|
|
1113
|
-
f"{name}: binary {binary_glyph} ({binary_name}) · "
|
|
1114
|
-
f"quota {quota_glyph} {quota_str} · {billable_str}"
|
|
1115
|
-
)
|
|
1116
|
-
detail = " | ".join(lines)
|
|
1117
|
-
if missing:
|
|
1118
|
-
return {
|
|
1119
|
-
"id": "council-cli", "status": "warn",
|
|
1120
|
-
"message": (
|
|
1121
|
-
f"{len(missing)}/{len(cli_members)} CLI member(s) missing binary "
|
|
1122
|
-
f"({', '.join(missing)}) · {detail}"
|
|
1123
|
-
),
|
|
1124
|
-
"remedy": (
|
|
1125
|
-
"install the missing CLI(s) — see `council:estimate` pre-flight "
|
|
1126
|
-
"for per-provider install hints, or flip "
|
|
1127
|
-
"ai_council.members.<name>.mode to 'api'"
|
|
1128
|
-
),
|
|
1129
|
-
}
|
|
1130
|
-
if over_warn:
|
|
1131
|
-
return {
|
|
1132
|
-
"id": "council-cli", "status": "warn",
|
|
1133
|
-
"message": (
|
|
1134
|
-
f"{len(over_warn)}/{len(cli_members)} CLI member(s) at/over "
|
|
1135
|
-
f"quota warn_at={warn_at} ({', '.join(over_warn)}) · {detail}"
|
|
1136
|
-
),
|
|
1137
|
-
"remedy": (
|
|
1138
|
-
"wait for UTC rollover or run "
|
|
1139
|
-
"`python3 scripts/council_cli.py quota --reset` to clear the counter"
|
|
1140
|
-
),
|
|
1141
|
-
}
|
|
1142
|
-
return {
|
|
1143
|
-
"id": "council-cli", "status": "ok",
|
|
1144
|
-
"message": f"{len(cli_members)} CLI member(s) healthy · {detail}",
|
|
1145
|
-
"remedy": "",
|
|
1146
|
-
}
|
|
1147
|
-
|
|
1148
|
-
|
|
1149
|
-
def _check_unsupported_combos(manifest: dict[str, Any]) -> dict[str, Any]:
|
|
1150
|
-
"""Flag tools whose ``scope`` violates the global-only or project-only rules."""
|
|
1151
|
-
global_only = {"droid", "qoder"}
|
|
1152
|
-
bad: list[str] = []
|
|
1153
|
-
for tool in manifest.get("tools") or []:
|
|
1154
|
-
name = str(tool.get("name", ""))
|
|
1155
|
-
scope = tool.get("scope")
|
|
1156
|
-
if name in global_only and scope != "global":
|
|
1157
|
-
bad.append(f"{name} (scope={scope}, requires global)")
|
|
1158
|
-
if bad:
|
|
1159
|
-
return {
|
|
1160
|
-
"id": "unsupported-combos", "status": "fail",
|
|
1161
|
-
"message": f"{len(bad)} tool(s) with unsupported scope: {', '.join(bad)}",
|
|
1162
|
-
"remedy": "re-install the listed tools with `--global --force`",
|
|
1163
|
-
}
|
|
1164
|
-
return {
|
|
1165
|
-
"id": "unsupported-combos", "status": "ok",
|
|
1166
|
-
"message": "all installed tools use supported scopes",
|
|
1167
|
-
"remedy": "",
|
|
1168
|
-
}
|
|
1169
|
-
|
|
1170
|
-
|
|
1171
|
-
def _wizard_state_path() -> Path:
|
|
1172
|
-
"""Return the on-disk location of the unified Setup-Wizard state file.
|
|
1173
|
-
|
|
1174
|
-
Mirrors ``STATE_REL`` in :mod:`src/server/routes/wizard.ts` — the
|
|
1175
|
-
server writes ``<writeRoot>/state/wizard-state.json``. Under the
|
|
1176
|
-
global-only install scope (ADR-020) ``writeRoot`` resolves to
|
|
1177
|
-
:func:`user_global_paths.event4u_root` (typically
|
|
1178
|
-
``~/.event4u/agent-config/``); the ``EVENT4U_CONFIG_HOME`` override
|
|
1179
|
-
is honoured implicitly so tests stay hermetic.
|
|
1180
|
-
"""
|
|
1181
|
-
from scripts._lib.user_global_paths import event4u_root
|
|
1182
|
-
return event4u_root() / "state" / "wizard-state.json"
|
|
1183
|
-
|
|
1184
|
-
|
|
1185
|
-
def _check_wizard_state() -> dict[str, Any]:
|
|
1186
|
-
"""Health-check the resumable Setup-Wizard state file.
|
|
1187
|
-
|
|
1188
|
-
Returns ``ok`` when the file is absent (no active session) or when
|
|
1189
|
-
its JSON shape matches the contract from
|
|
1190
|
-
:mod:`src/server/routes/wizard.ts` (``step: int``, ``partial: dict``,
|
|
1191
|
-
optional ``totalSteps: int``, ``startedAt: str | null``).
|
|
1192
|
-
Returns ``fail`` when the file exists but is unreadable, not valid
|
|
1193
|
-
JSON, or violates the shape — ``--repair wizard-state`` resets it.
|
|
1194
|
-
"""
|
|
1195
|
-
state_pth = _wizard_state_path()
|
|
1196
|
-
if not state_pth.exists():
|
|
1197
|
-
return {
|
|
1198
|
-
"id": "wizard-state", "status": "ok",
|
|
1199
|
-
"message": "no active wizard session",
|
|
1200
|
-
"remedy": "",
|
|
1201
|
-
}
|
|
1202
|
-
try:
|
|
1203
|
-
raw = state_pth.read_text(encoding="utf-8")
|
|
1204
|
-
except OSError as exc:
|
|
1205
|
-
return {
|
|
1206
|
-
"id": "wizard-state", "status": "fail",
|
|
1207
|
-
"message": f"unreadable wizard-state at {state_pth}: {exc}",
|
|
1208
|
-
"remedy": "agent-config doctor --repair wizard-state",
|
|
1209
|
-
}
|
|
1210
|
-
try:
|
|
1211
|
-
data = json.loads(raw)
|
|
1212
|
-
except json.JSONDecodeError as exc:
|
|
1213
|
-
return {
|
|
1214
|
-
"id": "wizard-state", "status": "fail",
|
|
1215
|
-
"message": f"malformed JSON in wizard-state ({exc.msg} at line {exc.lineno})",
|
|
1216
|
-
"remedy": "agent-config doctor --repair wizard-state",
|
|
1217
|
-
}
|
|
1218
|
-
if not isinstance(data, dict):
|
|
1219
|
-
return {
|
|
1220
|
-
"id": "wizard-state", "status": "fail",
|
|
1221
|
-
"message": f"wizard-state root is {type(data).__name__}, expected object",
|
|
1222
|
-
"remedy": "agent-config doctor --repair wizard-state",
|
|
1223
|
-
}
|
|
1224
|
-
step = data.get("step")
|
|
1225
|
-
partial = data.get("partial", {})
|
|
1226
|
-
if not isinstance(step, int) or step < 0:
|
|
1227
|
-
return {
|
|
1228
|
-
"id": "wizard-state", "status": "fail",
|
|
1229
|
-
"message": f"wizard-state.step is {step!r}, expected non-negative integer",
|
|
1230
|
-
"remedy": "agent-config doctor --repair wizard-state",
|
|
1231
|
-
}
|
|
1232
|
-
if not isinstance(partial, dict):
|
|
1233
|
-
return {
|
|
1234
|
-
"id": "wizard-state", "status": "fail",
|
|
1235
|
-
"message": f"wizard-state.partial is {type(partial).__name__}, expected object",
|
|
1236
|
-
"remedy": "agent-config doctor --repair wizard-state",
|
|
1237
|
-
}
|
|
1238
|
-
total = data.get("totalSteps")
|
|
1239
|
-
if total is not None and (not isinstance(total, int) or total < 1):
|
|
1240
|
-
return {
|
|
1241
|
-
"id": "wizard-state", "status": "fail",
|
|
1242
|
-
"message": f"wizard-state.totalSteps is {total!r}, expected positive integer or omitted",
|
|
1243
|
-
"remedy": "agent-config doctor --repair wizard-state",
|
|
1244
|
-
}
|
|
1245
|
-
suffix = f" of {total}" if isinstance(total, int) else ""
|
|
1246
|
-
return {
|
|
1247
|
-
"id": "wizard-state", "status": "ok",
|
|
1248
|
-
"message": f"resumable wizard session at step {step + 1}{suffix}",
|
|
1249
|
-
"remedy": "",
|
|
1250
|
-
}
|
|
1251
|
-
|
|
1252
|
-
|
|
1253
|
-
def _run_checks(
|
|
1254
|
-
project_root: Path,
|
|
1255
|
-
manifest: dict[str, Any],
|
|
1256
|
-
drift: dict[str, list[dict[str, Any]]],
|
|
1257
|
-
only: str | None = None,
|
|
1258
|
-
) -> list[dict[str, Any]]:
|
|
1259
|
-
"""Run the health-check registry and return ordered structured results.
|
|
1260
|
-
|
|
1261
|
-
``only`` filters to a single check id; ``None`` runs the full set.
|
|
1262
|
-
"""
|
|
1263
|
-
runners: dict[str, Any] = {
|
|
1264
|
-
"scope": lambda: _check_scope(project_root),
|
|
1265
|
-
"global-binary": lambda: _check_global_binary(project_root),
|
|
1266
|
-
"stale-orphans": _check_stale_orphans,
|
|
1267
|
-
"manifest-integrity": lambda: _check_manifest_integrity(manifest),
|
|
1268
|
-
"lockfile-freshness": lambda: _check_lockfile_freshness(manifest),
|
|
1269
|
-
"bridge-drift": lambda: _check_bridge_drift(
|
|
1270
|
-
drift["missing"], drift["modified"],
|
|
1271
|
-
drift["foreign"], drift["tag_drift"],
|
|
1272
|
-
),
|
|
1273
|
-
"mcp-mode": lambda: _check_mcp_mode(project_root),
|
|
1274
|
-
"mcp-beta-readiness": lambda: _check_mcp_beta_readiness(project_root),
|
|
1275
|
-
"offline-readiness": lambda: _check_offline_readiness(),
|
|
1276
|
-
"python-runtime": lambda: _check_python_runtime(),
|
|
1277
|
-
"tier-usage-readiness": lambda: _check_tier_usage_readiness(project_root),
|
|
1278
|
-
"council-cli": lambda: _check_council_cli(project_root),
|
|
1279
|
-
"unsupported-combos": lambda: _check_unsupported_combos(manifest),
|
|
1280
|
-
"wizard-state": _check_wizard_state,
|
|
1281
|
-
}
|
|
1282
|
-
out: list[dict[str, Any]] = []
|
|
1283
|
-
for cid in CHECK_IDS:
|
|
1284
|
-
if only is not None and cid != only:
|
|
1285
|
-
continue
|
|
1286
|
-
out.append(runners[cid]())
|
|
1287
|
-
return out
|
|
1288
|
-
|
|
1289
|
-
|
|
1290
|
-
def _skipped_manifest_check(check_id: str) -> dict[str, Any]:
|
|
1291
|
-
"""A ``skipped`` verdict for a manifest-required check with no lockfile.
|
|
1292
|
-
|
|
1293
|
-
Explicit machine-readable record (not omitted, not ``null``) so the
|
|
1294
|
-
``--json`` ``checks`` array keeps a stable shape for a global-only
|
|
1295
|
-
consumer — a council convergence point (2026-06-02).
|
|
1296
|
-
"""
|
|
1297
|
-
return {
|
|
1298
|
-
"id": check_id, "status": "skipped",
|
|
1299
|
-
"message": "requires a project lockfile (agents/installed-tools.lock)",
|
|
1300
|
-
"remedy": "run `agent-config init` to create a project lockfile, "
|
|
1301
|
-
"then re-run this check",
|
|
1302
|
-
}
|
|
1303
|
-
|
|
1304
|
-
|
|
1305
|
-
def _check_bridge_drift_no_manifest(bridge_present: bool) -> dict[str, Any]:
|
|
1306
|
-
"""Scope-aware ``bridge-drift`` verdict when no project manifest exists.
|
|
1307
|
-
|
|
1308
|
-
Kept out of :func:`_check_bridge_drift` (which stays a pure
|
|
1309
|
-
manifest-derived roll-up) per the council's single-responsibility
|
|
1310
|
-
point. A global-only consumer has no distributed tools to drift, so
|
|
1311
|
-
a bridge-present repo reports ``ok`` ("not applicable"); a repo with
|
|
1312
|
-
neither lockfile nor bridge marker reports ``skipped``.
|
|
1313
|
-
"""
|
|
1314
|
-
if bridge_present:
|
|
1315
|
-
return {
|
|
1316
|
-
"id": "bridge-drift", "status": "ok",
|
|
1317
|
-
"message": "no project lockfile → distributed-tool drift not "
|
|
1318
|
-
"applicable (global-only consumer)",
|
|
1319
|
-
"remedy": "",
|
|
1320
|
-
}
|
|
1321
|
-
return {
|
|
1322
|
-
"id": "bridge-drift", "status": "skipped",
|
|
1323
|
-
"message": "no project lockfile and no bridge marker → drift check "
|
|
1324
|
-
"not applicable",
|
|
1325
|
-
"remedy": "run `agent-config init` (project install) or "
|
|
1326
|
-
"`agent-config refresh --project` (global-only consumer)",
|
|
1327
|
-
}
|
|
1328
|
-
|
|
1329
|
-
|
|
1330
|
-
def _run_checks_no_manifest(
|
|
1331
|
-
project_root: Path,
|
|
1332
|
-
bridge_present: bool,
|
|
1333
|
-
only: str | None = None,
|
|
1334
|
-
) -> list[dict[str, Any]]:
|
|
1335
|
-
"""Run the registry with no project manifest available.
|
|
1336
|
-
|
|
1337
|
-
Mirrors :func:`_run_checks` and preserves :data:`CHECK_IDS` order.
|
|
1338
|
-
Global checks (:data:`GLOBAL_CHECK_IDS`) run unchanged; manifest-required
|
|
1339
|
-
checks (:data:`MANIFEST_REQUIRED_CHECK_IDS`) report ``skipped``;
|
|
1340
|
-
``bridge-drift`` gets the scope-aware no-manifest verdict.
|
|
1341
|
-
"""
|
|
1342
|
-
runners: dict[str, Any] = {
|
|
1343
|
-
"scope": lambda: _check_scope(project_root),
|
|
1344
|
-
"global-binary": lambda: _check_global_binary(project_root),
|
|
1345
|
-
"stale-orphans": _check_stale_orphans,
|
|
1346
|
-
"manifest-integrity": lambda: _skipped_manifest_check("manifest-integrity"),
|
|
1347
|
-
"lockfile-freshness": lambda: _skipped_manifest_check("lockfile-freshness"),
|
|
1348
|
-
"bridge-drift": lambda: _check_bridge_drift_no_manifest(bridge_present),
|
|
1349
|
-
"mcp-mode": lambda: _check_mcp_mode(project_root),
|
|
1350
|
-
"mcp-beta-readiness": lambda: _check_mcp_beta_readiness(project_root),
|
|
1351
|
-
"offline-readiness": lambda: _check_offline_readiness(),
|
|
1352
|
-
"python-runtime": lambda: _check_python_runtime(),
|
|
1353
|
-
"tier-usage-readiness": lambda: _check_tier_usage_readiness(project_root),
|
|
1354
|
-
"council-cli": lambda: _check_council_cli(project_root),
|
|
1355
|
-
"unsupported-combos": lambda: _skipped_manifest_check("unsupported-combos"),
|
|
1356
|
-
"wizard-state": _check_wizard_state,
|
|
1357
|
-
}
|
|
1358
|
-
out: list[dict[str, Any]] = []
|
|
1359
|
-
for cid in CHECK_IDS:
|
|
1360
|
-
if only is not None and cid != only:
|
|
1361
|
-
continue
|
|
1362
|
-
out.append(runners[cid]())
|
|
1363
|
-
return out
|
|
1364
|
-
|
|
1365
|
-
|
|
1366
|
-
def _run_no_manifest(
|
|
1367
|
-
opts: argparse.Namespace,
|
|
1368
|
-
project_root: Path,
|
|
1369
|
-
origin: str,
|
|
1370
|
-
bridge_present: bool,
|
|
1371
|
-
) -> int:
|
|
1372
|
-
"""Handle the no-project-lockfile path without the old hard bail.
|
|
1373
|
-
|
|
1374
|
-
Exit-code contract (council-defined, 2026-06-02):
|
|
1375
|
-
|
|
1376
|
-
* ``0`` — bare report for a recognised global-only consumer
|
|
1377
|
-
(bridge marker present), or a single global ``--check`` that passed.
|
|
1378
|
-
* ``1`` — a runnable health check requested via ``--check`` failed.
|
|
1379
|
-
* ``2`` — a requested ``--check`` cannot run (manifest-required check
|
|
1380
|
-
with no lockfile, or ``bridge-drift`` with neither lockfile nor
|
|
1381
|
-
bridge marker), or a bare report in an **uninitialised** repo
|
|
1382
|
-
(neither lockfile nor bridge marker) — preserves the spirit of the
|
|
1383
|
-
pre-existing "run init" signal while still printing a real report.
|
|
1384
|
-
"""
|
|
1385
|
-
checks = _run_checks_no_manifest(project_root, bridge_present, only=opts.check)
|
|
1386
|
-
fail_check = any(c["status"] == "fail" for c in checks)
|
|
1387
|
-
skipped_requested = opts.check is not None and any(
|
|
1388
|
-
c["id"] == opts.check and c["status"] == "skipped" for c in checks
|
|
1389
|
-
)
|
|
1390
|
-
|
|
1391
|
-
if opts.json:
|
|
1392
|
-
_emit_json(project_root, [], [], [], [], checks=checks, origin=origin)
|
|
1393
|
-
elif opts.check is None:
|
|
1394
|
-
print(f" 📍 project_root: {project_root} (origin: {origin})")
|
|
1395
|
-
if bridge_present:
|
|
1396
|
-
print(" ℹ️ global-only consumer: bridge marker present, no "
|
|
1397
|
-
"project lockfile (expected under ADR-020)")
|
|
1398
|
-
print(" project-manifest checks are skipped — they apply only "
|
|
1399
|
-
"to project-local distributed tools")
|
|
1400
|
-
else:
|
|
1401
|
-
print(" ⚠️ no project lockfile and no bridge marker at "
|
|
1402
|
-
f"{project_root}", file=sys.stderr)
|
|
1403
|
-
print(" run `agent-config init` (project install) or "
|
|
1404
|
-
"`agent-config refresh --project` (global-only consumer)",
|
|
1405
|
-
file=sys.stderr)
|
|
1406
|
-
_emit_checks_text(checks)
|
|
1407
|
-
else:
|
|
1408
|
-
_emit_checks_text(checks)
|
|
1409
|
-
|
|
1410
|
-
if opts.check is not None:
|
|
1411
|
-
if skipped_requested:
|
|
1412
|
-
return 2
|
|
1413
|
-
return 1 if fail_check else 0
|
|
1414
|
-
return 0 if bridge_present else 2
|
|
1415
|
-
|
|
1416
|
-
|
|
1417
|
-
def _emit_json(
|
|
1418
|
-
project_root: Path,
|
|
1419
|
-
missing: list[dict[str, Any]],
|
|
1420
|
-
modified: list[dict[str, Any]],
|
|
1421
|
-
foreign: list[dict[str, Any]],
|
|
1422
|
-
tag_drift: list[dict[str, Any]],
|
|
1423
|
-
checks: list[dict[str, Any]] | None = None,
|
|
1424
|
-
origin: str | None = None,
|
|
1425
|
-
) -> None:
|
|
1426
|
-
payload: dict[str, Any] = {
|
|
1427
|
-
"project_root": str(project_root),
|
|
1428
|
-
"missing": missing,
|
|
1429
|
-
"modified": modified,
|
|
1430
|
-
"foreign": foreign,
|
|
1431
|
-
"tag_drift": tag_drift,
|
|
1432
|
-
}
|
|
1433
|
-
if origin is not None:
|
|
1434
|
-
payload["project_root_origin"] = origin
|
|
1435
|
-
if checks is not None:
|
|
1436
|
-
payload["checks"] = checks
|
|
1437
|
-
print(json.dumps(payload, indent=2))
|
|
1438
|
-
|
|
1439
|
-
|
|
1440
|
-
def _emit_checks_text(checks: list[dict[str, Any]]) -> None:
|
|
1441
|
-
"""Render the health-check block above the drift section."""
|
|
1442
|
-
if not checks:
|
|
1443
|
-
return
|
|
1444
|
-
print("checks:")
|
|
1445
|
-
for c in checks:
|
|
1446
|
-
sym = STATUS_SYMBOLS.get(c["status"], "?")
|
|
1447
|
-
print(f" {sym} {c['id']}: {c['message']}")
|
|
1448
|
-
if c["status"] != "ok" and c.get("remedy"):
|
|
1449
|
-
print(f" fix: {c['remedy']}")
|
|
1450
|
-
print("")
|
|
1451
|
-
|
|
1452
|
-
|
|
1453
|
-
def _emit_text(
|
|
1454
|
-
project_root: Path,
|
|
1455
|
-
missing: list[dict[str, Any]],
|
|
1456
|
-
modified: list[dict[str, Any]],
|
|
1457
|
-
foreign: list[dict[str, Any]],
|
|
1458
|
-
tag_drift: list[dict[str, Any]],
|
|
1459
|
-
) -> None:
|
|
1460
|
-
total = len(missing) + len(modified) + len(foreign) + len(tag_drift)
|
|
1461
|
-
if total == 0:
|
|
1462
|
-
print(f"✅ doctor: manifest matches filesystem under {project_root}")
|
|
1463
|
-
return
|
|
1464
|
-
print(f"⚠️ doctor: {total} drift item(s) under {project_root}")
|
|
1465
|
-
for label, items in (
|
|
1466
|
-
("missing", missing),
|
|
1467
|
-
("modified", modified),
|
|
1468
|
-
("foreign", foreign),
|
|
1469
|
-
("tag-drift", tag_drift),
|
|
1470
|
-
):
|
|
1471
|
-
if not items:
|
|
1472
|
-
continue
|
|
1473
|
-
print(f"\n {label} ({len(items)}):")
|
|
1474
|
-
for it in items:
|
|
1475
|
-
tool = it["tool"] or "?"
|
|
1476
|
-
print(f" · [{tool}] {it['path']}")
|
|
1477
|
-
if label == "tag-drift":
|
|
1478
|
-
found = it.get("found") or "(missing)"
|
|
1479
|
-
expected = it.get("expected", PACKAGE_TAG_ID)
|
|
1480
|
-
print(f" expected: {expected}")
|
|
1481
|
-
print(f" found: {found}")
|
|
1482
|
-
print(f" fix: {it['fix']}")
|
|
1483
|
-
|
|
1484
|
-
|
|
1485
|
-
def _parse(argv: list[str]) -> argparse.Namespace:
|
|
1486
|
-
parser = argparse.ArgumentParser(
|
|
1487
|
-
prog="agent-config doctor",
|
|
1488
|
-
description=(
|
|
1489
|
-
"Manifest ↔ filesystem drift report plus install health "
|
|
1490
|
-
"checks. See --check for the individual check ids."
|
|
1491
|
-
),
|
|
1492
|
-
)
|
|
1493
|
-
parser.add_argument("--project", default=None,
|
|
1494
|
-
help="project root (default: cwd)")
|
|
1495
|
-
parser.add_argument("--json", action="store_true",
|
|
1496
|
-
help="emit a JSON report instead of human text")
|
|
1497
|
-
parser.add_argument(
|
|
1498
|
-
"--check", default=None, metavar="ID",
|
|
1499
|
-
choices=list(CHECK_IDS),
|
|
1500
|
-
help=("run a single health check by id "
|
|
1501
|
-
f"({' · '.join(CHECK_IDS)})"),
|
|
1502
|
-
)
|
|
1503
|
-
parser.add_argument(
|
|
1504
|
-
"--trace-root", action="store_true",
|
|
1505
|
-
help=("print every ancestor checked during project-root discovery "
|
|
1506
|
-
"plus the winning anchor; short-circuits the drift report"),
|
|
1507
|
-
)
|
|
1508
|
-
parser.add_argument(
|
|
1509
|
-
"--context", action="store_true",
|
|
1510
|
-
help=("print effective project root, anchor, env-pin, settings "
|
|
1511
|
-
"layer chain, wrapper, and install-mode; short-circuits "
|
|
1512
|
-
"the drift report"),
|
|
1513
|
-
)
|
|
1514
|
-
parser.add_argument(
|
|
1515
|
-
"--repair", default=None, metavar="ID",
|
|
1516
|
-
choices=list(REPAIR_IDS),
|
|
1517
|
-
help=("reset a recoverable artefact and exit "
|
|
1518
|
-
f"({' · '.join(REPAIR_IDS)}); short-circuits the drift "
|
|
1519
|
-
"report. Idempotent — absent files report 'nothing to "
|
|
1520
|
-
"repair' and exit 0."),
|
|
1521
|
-
)
|
|
1522
|
-
return parser.parse_args(argv)
|
|
1523
|
-
|
|
1524
|
-
|
|
1525
|
-
def _run_trace_root(opts: argparse.Namespace) -> int:
|
|
1526
|
-
"""Handle ``--trace-root``: discovery walk + records, no drift report."""
|
|
1527
|
-
start = Path.cwd()
|
|
1528
|
-
root, anchor, trace = find_project_root_with_trace(start)
|
|
1529
|
-
# Determine origin label even when no anchor is found.
|
|
1530
|
-
if os.environ.get(ROOT_OVERRIDE_ENV) == "1" and os.environ.get(PROJECT_ROOT_ENV):
|
|
1531
|
-
origin = "root-flag"
|
|
1532
|
-
elif opts.project:
|
|
1533
|
-
origin = "explicit"
|
|
1534
|
-
elif os.environ.get(PROJECT_ROOT_ENV):
|
|
1535
|
-
origin = "env"
|
|
1536
|
-
elif root is not None:
|
|
1537
|
-
origin = anchor or "unknown"
|
|
1538
|
-
else:
|
|
1539
|
-
origin = "cwd-fallback"
|
|
1540
|
-
if opts.json:
|
|
1541
|
-
_emit_trace_json(root, anchor, trace, origin)
|
|
1542
|
-
else:
|
|
1543
|
-
_emit_trace_text(root, anchor, trace, origin)
|
|
1544
|
-
return 0
|
|
1545
|
-
|
|
1546
|
-
|
|
1547
|
-
def _run_context(opts: argparse.Namespace) -> int:
|
|
1548
|
-
"""Handle ``--context``: install + discovery snapshot, no drift report."""
|
|
1549
|
-
try:
|
|
1550
|
-
project_root, origin = _resolve_project_root(opts.project)
|
|
1551
|
-
except ProjectRootError as exc:
|
|
1552
|
-
print(f"❌ doctor: {exc}", file=sys.stderr)
|
|
1553
|
-
return 2
|
|
1554
|
-
mode, mode_source = _detect_install_mode(project_root)
|
|
1555
|
-
ctx: dict[str, Any] = {
|
|
1556
|
-
"project_root": str(project_root),
|
|
1557
|
-
"origin": origin,
|
|
1558
|
-
"install_mode": mode,
|
|
1559
|
-
"install_mode_source": mode_source,
|
|
1560
|
-
"env_pin": os.environ.get(PROJECT_ROOT_ENV) or None,
|
|
1561
|
-
"root_override": os.environ.get(ROOT_OVERRIDE_ENV) == "1",
|
|
1562
|
-
"settings_layers": _settings_layer_chain(project_root),
|
|
1563
|
-
"wrapper": _detect_wrapper(project_root),
|
|
1564
|
-
}
|
|
1565
|
-
if opts.json:
|
|
1566
|
-
print(json.dumps(ctx, indent=2))
|
|
1567
|
-
else:
|
|
1568
|
-
_emit_context_text(ctx)
|
|
1569
|
-
return 0
|
|
1570
|
-
|
|
1571
|
-
|
|
1572
|
-
def _run_repair(opts: argparse.Namespace) -> int:
|
|
1573
|
-
"""Handle ``--repair <id>``: reset the named artefact, no drift report.
|
|
1574
|
-
|
|
1575
|
-
Currently the only target is ``wizard-state`` — unlinks the
|
|
1576
|
-
resumable session file so the next ``agent-config setup`` boots
|
|
1577
|
-
from step 1. Absent files are a no-op (idempotent re-run).
|
|
1578
|
-
"""
|
|
1579
|
-
target = opts.repair
|
|
1580
|
-
if target == "wizard-state":
|
|
1581
|
-
state_pth = _wizard_state_path()
|
|
1582
|
-
payload = {
|
|
1583
|
-
"id": "wizard-state",
|
|
1584
|
-
"path": str(state_pth),
|
|
1585
|
-
"action": "remove" if state_pth.exists() else "noop",
|
|
1586
|
-
}
|
|
1587
|
-
if state_pth.exists():
|
|
1588
|
-
try:
|
|
1589
|
-
state_pth.unlink()
|
|
1590
|
-
except OSError as exc:
|
|
1591
|
-
payload["action"] = "error"
|
|
1592
|
-
payload["error"] = str(exc)
|
|
1593
|
-
if opts.json:
|
|
1594
|
-
print(json.dumps(payload, indent=2))
|
|
1595
|
-
else:
|
|
1596
|
-
print(f"❌ doctor: could not remove {state_pth}: {exc}",
|
|
1597
|
-
file=sys.stderr)
|
|
1598
|
-
return 2
|
|
1599
|
-
if opts.json:
|
|
1600
|
-
print(json.dumps(payload, indent=2))
|
|
1601
|
-
else:
|
|
1602
|
-
if payload["action"] == "remove":
|
|
1603
|
-
print(f"✅ doctor: reset wizard-state ({state_pth})")
|
|
1604
|
-
else:
|
|
1605
|
-
print(f"✅ doctor: nothing to repair (no wizard-state at {state_pth})")
|
|
1606
|
-
return 0
|
|
1607
|
-
# argparse `choices=` already constrains target; defensive default.
|
|
1608
|
-
print(f"❌ doctor: unknown repair target {target!r}", file=sys.stderr)
|
|
1609
|
-
return 2
|
|
1610
|
-
|
|
1611
|
-
|
|
1612
|
-
def main(argv: list[str] | None = None) -> int:
|
|
1613
|
-
opts = _parse(list(argv) if argv is not None else sys.argv[1:])
|
|
1614
|
-
if opts.repair is not None:
|
|
1615
|
-
return _run_repair(opts)
|
|
1616
|
-
if opts.trace_root:
|
|
1617
|
-
return _run_trace_root(opts)
|
|
1618
|
-
if opts.context:
|
|
1619
|
-
return _run_context(opts)
|
|
1620
|
-
try:
|
|
1621
|
-
project_root, origin = _resolve_project_root(opts.project)
|
|
1622
|
-
except ProjectRootError as exc:
|
|
1623
|
-
print(f"❌ doctor: {exc}", file=sys.stderr)
|
|
1624
|
-
return 2
|
|
1625
|
-
manifest_pth = installed_tools.manifest_path(project_root)
|
|
1626
|
-
manifest = installed_tools.read_manifest(manifest_pth)
|
|
1627
|
-
if manifest is None:
|
|
1628
|
-
# No project lockfile. Under ADR-020 global-only this is a
|
|
1629
|
-
# legitimate state, not an error — run the lockfile-independent
|
|
1630
|
-
# checks and report consumer state instead of a hard bail. The
|
|
1631
|
-
# per-scope split + exit-code contract live in _run_no_manifest.
|
|
1632
|
-
bridge_present = (project_root / BRIDGE_MARKER_RELATIVE).is_file()
|
|
1633
|
-
return _run_no_manifest(opts, project_root, origin, bridge_present)
|
|
1634
|
-
|
|
1635
|
-
records, known = _collect_manifest_entries(project_root, manifest)
|
|
1636
|
-
missing, modified, tag_drift = _classify(records)
|
|
1637
|
-
foreign = _foreign_records(
|
|
1638
|
-
project_root, _scan_foreign(project_root, manifest, known),
|
|
1639
|
-
)
|
|
1640
|
-
drift_groups = {
|
|
1641
|
-
"missing": missing, "modified": modified,
|
|
1642
|
-
"foreign": foreign, "tag_drift": tag_drift,
|
|
1643
|
-
}
|
|
1644
|
-
checks = _run_checks(project_root, manifest, drift_groups, only=opts.check)
|
|
1645
|
-
fail_check = any(c["status"] == "fail" for c in checks)
|
|
1646
|
-
|
|
1647
|
-
if opts.json:
|
|
1648
|
-
_emit_json(
|
|
1649
|
-
project_root, missing, modified, foreign, tag_drift,
|
|
1650
|
-
checks=checks, origin=origin,
|
|
1651
|
-
)
|
|
1652
|
-
else:
|
|
1653
|
-
if opts.check is None:
|
|
1654
|
-
print(f" 📍 project_root: {project_root} (origin: {origin})")
|
|
1655
|
-
_emit_checks_text(checks)
|
|
1656
|
-
if opts.check is None:
|
|
1657
|
-
_emit_text(project_root, missing, modified, foreign, tag_drift)
|
|
1658
|
-
|
|
1659
|
-
# Full-suite mode: exit code reflects drift only; failing health
|
|
1660
|
-
# checks are surfaced visually but do not change the rc, preserving
|
|
1661
|
-
# the original drift-detection contract. ``--check <id>`` mode is
|
|
1662
|
-
# explicit and propagates the single check's verdict.
|
|
1663
|
-
if opts.check is not None:
|
|
1664
|
-
return 1 if fail_check else 0
|
|
1665
|
-
return 1 if (missing or modified or foreign or tag_drift) else 0
|
|
1666
|
-
|
|
1667
|
-
|
|
1668
|
-
if __name__ == "__main__": # pragma: no cover
|
|
1669
|
-
raise SystemExit(main())
|