@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
|
@@ -0,0 +1,1729 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Council configuration loader — single source of truth.
|
|
3
|
+
*
|
|
4
|
+
* TypeScript twin of `src/scripts/ai_council/config.py` (ADR-200 —
|
|
5
|
+
* Python→TS migration, Phase 12). Mirrors the Python module's public
|
|
6
|
+
* surface and validation behaviour byte-for-byte: same exported
|
|
7
|
+
* snake_case names, same defaults, same error messages, same precedence.
|
|
8
|
+
*
|
|
9
|
+
* Reads `agents/settings/.ai-council.yml` per the contract in
|
|
10
|
+
* `docs/contracts/ai-council-config.md`. Replaces the fragmented
|
|
11
|
+
* `.agent-settings.yml` `ai_council` block (Phase 0 migration).
|
|
12
|
+
*
|
|
13
|
+
* Validation contract (8 rules, all enforced at load time):
|
|
14
|
+
*
|
|
15
|
+
* 1. `enabled` is a bool.
|
|
16
|
+
* 2. `defaults.mode` ∈ {`api`, `manual`, `cli`}; per-member mode same
|
|
17
|
+
* set. Semantics: `api` = SDK call against a stored key (billable);
|
|
18
|
+
* `manual` = copy & paste — human transports prompt + reply between
|
|
19
|
+
* the agent and an external chat surface (free); `cli` = shell out to
|
|
20
|
+
* a locally-installed CLI under subscription auth (free for
|
|
21
|
+
* first-party CLIs, billable for community wrappers).
|
|
22
|
+
* 3. `members.<name>` keys are restricted to the known provider set.
|
|
23
|
+
* 4. `cost_budget.*` numeric fields are >= 0.
|
|
24
|
+
* 5. Enabled members carry a non-empty `model` and `api_key_ref` when
|
|
25
|
+
* their effective mode is `api`. CLI-mode members do NOT require
|
|
26
|
+
* `api_key_ref` (subscription auth is provided by the CLI binary
|
|
27
|
+
* itself).
|
|
28
|
+
* 6. `api_key_ref` starts with `file:` or `env:` — raw keys are refused
|
|
29
|
+
* even if syntactically plausible.
|
|
30
|
+
* 7. Resolved `file:` key paths must have mode 0o600 (delegated to
|
|
31
|
+
* `resolve_api_key`; runs at use-time, not parse-time).
|
|
32
|
+
* 8. `binary:` is only valid when the member's effective mode is `cli`;
|
|
33
|
+
* `cli_call_budget.max_calls_per_day.<provider>` keys must be valid
|
|
34
|
+
* providers.
|
|
35
|
+
*
|
|
36
|
+
* Parity notes (intentional, documented):
|
|
37
|
+
* - YAML is parsed with `yaml` (npm) at `version: '1.1'`, matching
|
|
38
|
+
* PyYAML's `safe_load` scalar grammar — the same approach the sibling
|
|
39
|
+
* twins (`move_artefact.ts`, `new_skill.ts`, `agent_settings.ts`) use.
|
|
40
|
+
* - JS has no int/float distinction, so a `.0`-suffixed YAML float in
|
|
41
|
+
* an int-only field (`max_members`, `auth_check_timeout_seconds`,
|
|
42
|
+
* `max_tokens`) parses to a JS integer and passes the int check where
|
|
43
|
+
* Python would reject it as a `float`. This is the same parity
|
|
44
|
+
* boundary the YAML twins accept; integers written as `2` (the only
|
|
45
|
+
* realistic shape) behave identically.
|
|
46
|
+
*/
|
|
47
|
+
import * as fs from 'node:fs';
|
|
48
|
+
import * as os from 'node:os';
|
|
49
|
+
import * as path from 'node:path';
|
|
50
|
+
|
|
51
|
+
import { parse as parseYaml } from 'yaml';
|
|
52
|
+
|
|
53
|
+
import * as user_global_paths from '../_lib/user_global_paths.js';
|
|
54
|
+
|
|
55
|
+
const _VALID_PROVIDERS: ReadonlySet<string> = new Set([
|
|
56
|
+
'anthropic',
|
|
57
|
+
'openai',
|
|
58
|
+
'gemini',
|
|
59
|
+
'xai',
|
|
60
|
+
'perplexity',
|
|
61
|
+
]);
|
|
62
|
+
const _VALID_MODES: ReadonlySet<string> = new Set(['api', 'manual', 'cli']);
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* Prefixes that signal "this is a raw API key" so we refuse it loudly
|
|
66
|
+
* even when the user accidentally inlined it in `api_key_ref`.
|
|
67
|
+
*/
|
|
68
|
+
export const _RAW_KEY_PREFIXES: readonly string[] = [
|
|
69
|
+
'sk-',
|
|
70
|
+
'sk-ant-',
|
|
71
|
+
'ya29.',
|
|
72
|
+
'AIza',
|
|
73
|
+
'xai-',
|
|
74
|
+
'pplx-',
|
|
75
|
+
'gsk_',
|
|
76
|
+
];
|
|
77
|
+
|
|
78
|
+
// ── value type shapes ──────────────────────────────────────────────
|
|
79
|
+
|
|
80
|
+
/** Any JSON-like value parsed out of the YAML (mirrors Python `Any`). */
|
|
81
|
+
export type Json =
|
|
82
|
+
| string
|
|
83
|
+
| number
|
|
84
|
+
| boolean
|
|
85
|
+
| null
|
|
86
|
+
| Json[]
|
|
87
|
+
| { [key: string]: Json };
|
|
88
|
+
|
|
89
|
+
type Dict = { [key: string]: Json };
|
|
90
|
+
|
|
91
|
+
/** Mirror of `pathlib.Path` for the public surface — a filesystem string. */
|
|
92
|
+
export type PathLike = string;
|
|
93
|
+
|
|
94
|
+
// ── error type ─────────────────────────────────────────────────────
|
|
95
|
+
|
|
96
|
+
/** Raised when `agents/settings/.ai-council.yml` violates the schema. */
|
|
97
|
+
export class CouncilConfigError extends Error {
|
|
98
|
+
constructor(message: string) {
|
|
99
|
+
super(message);
|
|
100
|
+
this.name = 'CouncilConfigError';
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
// ── Python-format helpers (byte-faithful error messages) ───────────
|
|
105
|
+
|
|
106
|
+
/** Python `repr()` for a string scalar (single-quoted, escaped). */
|
|
107
|
+
function _pyReprStr(s: string): string {
|
|
108
|
+
const hasSingle = s.includes("'");
|
|
109
|
+
const hasDouble = s.includes('"');
|
|
110
|
+
const quote = hasSingle && !hasDouble ? '"' : "'";
|
|
111
|
+
let body = s
|
|
112
|
+
.replace(/\\/g, '\\\\')
|
|
113
|
+
.replace(/\n/g, '\\n')
|
|
114
|
+
.replace(/\r/g, '\\r')
|
|
115
|
+
.replace(/\t/g, '\\t');
|
|
116
|
+
if (quote === "'") {
|
|
117
|
+
body = body.replace(/'/g, "\\'");
|
|
118
|
+
} else {
|
|
119
|
+
body = body.replace(/"/g, '\\"');
|
|
120
|
+
}
|
|
121
|
+
return `${quote}${body}${quote}`;
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
/** Python `repr()` for a float value (shortest round-trip, `N.0` for ints). */
|
|
125
|
+
function _pyReprFloat(value: number): string {
|
|
126
|
+
if (Number.isInteger(value) && Number.isFinite(value)) {
|
|
127
|
+
return `${value}.0`;
|
|
128
|
+
}
|
|
129
|
+
if (value === Infinity) {
|
|
130
|
+
return 'inf';
|
|
131
|
+
}
|
|
132
|
+
if (value === -Infinity) {
|
|
133
|
+
return '-inf';
|
|
134
|
+
}
|
|
135
|
+
if (Number.isNaN(value)) {
|
|
136
|
+
return 'nan';
|
|
137
|
+
}
|
|
138
|
+
return String(value);
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
/**
|
|
142
|
+
* Python `repr()` for an arbitrary parsed value. Floats are tracked via
|
|
143
|
+
* `_FLOAT` so int-valued floats render `N.0`; bare numbers render as
|
|
144
|
+
* Python ints (no decimal). Mirrors `{value!r}` formatting.
|
|
145
|
+
*/
|
|
146
|
+
function _pyRepr(value: unknown): string {
|
|
147
|
+
if (value instanceof _Float) {
|
|
148
|
+
return _pyReprFloat(value.value);
|
|
149
|
+
}
|
|
150
|
+
if (value === null || value === undefined) {
|
|
151
|
+
return 'None';
|
|
152
|
+
}
|
|
153
|
+
if (typeof value === 'boolean') {
|
|
154
|
+
return value ? 'True' : 'False';
|
|
155
|
+
}
|
|
156
|
+
if (typeof value === 'string') {
|
|
157
|
+
return _pyReprStr(value);
|
|
158
|
+
}
|
|
159
|
+
if (typeof value === 'number') {
|
|
160
|
+
return String(value);
|
|
161
|
+
}
|
|
162
|
+
if (Array.isArray(value)) {
|
|
163
|
+
return `[${value.map((v) => _pyRepr(v)).join(', ')}]`;
|
|
164
|
+
}
|
|
165
|
+
if (typeof value === 'object') {
|
|
166
|
+
const parts: string[] = [];
|
|
167
|
+
for (const [k, v] of Object.entries(value as Record<string, unknown>)) {
|
|
168
|
+
parts.push(`${_pyReprStr(k)}: ${_pyRepr(v)}`);
|
|
169
|
+
}
|
|
170
|
+
return `{${parts.join(', ')}}`;
|
|
171
|
+
}
|
|
172
|
+
return String(value);
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
/** Wrapper marking a number that should `repr()` as a Python float. */
|
|
176
|
+
class _Float {
|
|
177
|
+
constructor(readonly value: number) {}
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
/** Mark `n` so `_pyRepr` renders it with a Python float repr (`N.0`). */
|
|
181
|
+
function _f(n: number): _Float {
|
|
182
|
+
return new _Float(n);
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
/** Python `type(value).__name__`. */
|
|
186
|
+
function _pyTypeName(value: unknown): string {
|
|
187
|
+
if (value === null || value === undefined) {
|
|
188
|
+
return 'NoneType';
|
|
189
|
+
}
|
|
190
|
+
if (typeof value === 'boolean') {
|
|
191
|
+
return 'bool';
|
|
192
|
+
}
|
|
193
|
+
if (typeof value === 'number') {
|
|
194
|
+
return Number.isInteger(value) ? 'int' : 'float';
|
|
195
|
+
}
|
|
196
|
+
if (typeof value === 'string') {
|
|
197
|
+
return 'str';
|
|
198
|
+
}
|
|
199
|
+
if (Array.isArray(value)) {
|
|
200
|
+
return 'list';
|
|
201
|
+
}
|
|
202
|
+
if (typeof value === 'object') {
|
|
203
|
+
return 'dict';
|
|
204
|
+
}
|
|
205
|
+
return typeof value;
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
/** Python `sorted(set_of_strings)` rendered as a list repr `['a', 'b']`. */
|
|
209
|
+
function _sortedListRepr(items: Iterable<string>): string {
|
|
210
|
+
const sorted = [...items].sort();
|
|
211
|
+
return `[${sorted.map((s) => _pyReprStr(s)).join(', ')}]`;
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
/** Python `oct(mode)` → `0o600`-shaped string. */
|
|
215
|
+
function _pyOct(mode: number): string {
|
|
216
|
+
return `0o${mode.toString(8)}`;
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
// ── Python int()/float() coercion + isinstance helpers ─────────────
|
|
220
|
+
|
|
221
|
+
/**
|
|
222
|
+
* Mirror Python `int(x)`: ints pass through, floats truncate toward
|
|
223
|
+
* zero, bools → 0/1, numeric strings parse. Anything else raises —
|
|
224
|
+
* but the config only ever feeds values that `int()` accepts (the
|
|
225
|
+
* call sites are `int(d.get(...))` with numeric defaults).
|
|
226
|
+
*/
|
|
227
|
+
function _pyInt(value: Json | undefined): number {
|
|
228
|
+
if (typeof value === 'boolean') {
|
|
229
|
+
return value ? 1 : 0;
|
|
230
|
+
}
|
|
231
|
+
if (typeof value === 'number') {
|
|
232
|
+
return Math.trunc(value);
|
|
233
|
+
}
|
|
234
|
+
if (typeof value === 'string') {
|
|
235
|
+
const trimmed = value.trim();
|
|
236
|
+
// Python int("3") works; int("3.0") raises ValueError. The config
|
|
237
|
+
// never relies on that edge — keep it simple: parse base-10 ints.
|
|
238
|
+
if (/^[+-]?\d+$/.test(trimmed)) {
|
|
239
|
+
return parseInt(trimmed, 10);
|
|
240
|
+
}
|
|
241
|
+
throw new CouncilConfigError(
|
|
242
|
+
`invalid literal for int() with base 10: ${_pyReprStr(value)}`,
|
|
243
|
+
);
|
|
244
|
+
}
|
|
245
|
+
throw new CouncilConfigError(
|
|
246
|
+
`int() argument must be a string or a number, not ` +
|
|
247
|
+
`'${_pyTypeName(value)}'`,
|
|
248
|
+
);
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
/**
|
|
252
|
+
* Mirror Python `float(x)`: numbers + bools coerce, numeric strings
|
|
253
|
+
* parse. The config call sites pass numbers/bools/strings.
|
|
254
|
+
*/
|
|
255
|
+
function _pyFloat(value: Json | undefined): number {
|
|
256
|
+
if (typeof value === 'boolean') {
|
|
257
|
+
return value ? 1.0 : 0.0;
|
|
258
|
+
}
|
|
259
|
+
if (typeof value === 'number') {
|
|
260
|
+
return value;
|
|
261
|
+
}
|
|
262
|
+
if (typeof value === 'string') {
|
|
263
|
+
const trimmed = value.trim();
|
|
264
|
+
const parsed = Number(trimmed);
|
|
265
|
+
if (trimmed !== '' && Number.isFinite(parsed)) {
|
|
266
|
+
return parsed;
|
|
267
|
+
}
|
|
268
|
+
if (/^[+-]?inf(inity)?$/i.test(trimmed)) {
|
|
269
|
+
return trimmed.startsWith('-') ? -Infinity : Infinity;
|
|
270
|
+
}
|
|
271
|
+
if (/^[+-]?nan$/i.test(trimmed)) {
|
|
272
|
+
return NaN;
|
|
273
|
+
}
|
|
274
|
+
throw new CouncilConfigError(
|
|
275
|
+
`could not convert string to float: ${_pyReprStr(value)}`,
|
|
276
|
+
);
|
|
277
|
+
}
|
|
278
|
+
throw new CouncilConfigError(
|
|
279
|
+
`float() argument must be a string or a number, not ` +
|
|
280
|
+
`'${_pyTypeName(value)}'`,
|
|
281
|
+
);
|
|
282
|
+
}
|
|
283
|
+
|
|
284
|
+
/** Mirror Python `isinstance(x, bool)`. */
|
|
285
|
+
function _isBool(value: unknown): value is boolean {
|
|
286
|
+
return typeof value === 'boolean';
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
/** Mirror Python `isinstance(x, int) and not isinstance(x, bool)`. */
|
|
290
|
+
function _isInt(value: unknown): value is number {
|
|
291
|
+
return typeof value === 'number' && !Number.isNaN(value) && Number.isInteger(value);
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
/** Mirror Python `isinstance(x, (int, float)) and not isinstance(x, bool)`. */
|
|
295
|
+
function _isNumber(value: unknown): value is number {
|
|
296
|
+
return typeof value === 'number';
|
|
297
|
+
}
|
|
298
|
+
|
|
299
|
+
/** Mirror Python `isinstance(x, dict)`. */
|
|
300
|
+
function _isDict(value: unknown): value is Dict {
|
|
301
|
+
return (
|
|
302
|
+
typeof value === 'object' &&
|
|
303
|
+
value !== null &&
|
|
304
|
+
!Array.isArray(value)
|
|
305
|
+
);
|
|
306
|
+
}
|
|
307
|
+
|
|
308
|
+
/** Mirror Python `isinstance(x, str)`. */
|
|
309
|
+
function _isStr(value: unknown): value is string {
|
|
310
|
+
return typeof value === 'string';
|
|
311
|
+
}
|
|
312
|
+
|
|
313
|
+
/** Mirror Python `isinstance(x, (list, tuple))`. */
|
|
314
|
+
function _isList(value: unknown): value is Json[] {
|
|
315
|
+
return Array.isArray(value);
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
/**
|
|
319
|
+
* Mirror Python truthiness for `d.get(k) or {}` / `or ()` patterns:
|
|
320
|
+
* `None`, `False`, `0`, `0.0`, `''`, `[]`, `{}` are falsy. Used to
|
|
321
|
+
* replicate the `raw.get("x") or {}` idiom exactly.
|
|
322
|
+
*/
|
|
323
|
+
function _getOr<T extends Json>(d: Dict, key: string, fallback: T): Json | T {
|
|
324
|
+
const v = d[key];
|
|
325
|
+
if (v === undefined) {
|
|
326
|
+
return fallback;
|
|
327
|
+
}
|
|
328
|
+
if (
|
|
329
|
+
v === null ||
|
|
330
|
+
v === false ||
|
|
331
|
+
v === 0 ||
|
|
332
|
+
v === '' ||
|
|
333
|
+
(Array.isArray(v) && v.length === 0) ||
|
|
334
|
+
(_isDict(v) && Object.keys(v).length === 0)
|
|
335
|
+
) {
|
|
336
|
+
return fallback;
|
|
337
|
+
}
|
|
338
|
+
return v;
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
/** Mirror `d.get(key, default)` — `undefined` (absent) → default. */
|
|
342
|
+
function _get(d: Dict, key: string, fallback: Json): Json {
|
|
343
|
+
const v = d[key];
|
|
344
|
+
return v === undefined ? fallback : v;
|
|
345
|
+
}
|
|
346
|
+
|
|
347
|
+
// ── config dataclasses (frozen → readonly interfaces) ──────────────
|
|
348
|
+
|
|
349
|
+
export interface DefaultsConfig {
|
|
350
|
+
readonly mode: string;
|
|
351
|
+
readonly member_mode: string;
|
|
352
|
+
readonly min_rounds: number;
|
|
353
|
+
readonly deep_min_rounds: number;
|
|
354
|
+
readonly max_output_tokens: number;
|
|
355
|
+
readonly session_retention_days: number;
|
|
356
|
+
readonly debate_max_rounds: number;
|
|
357
|
+
}
|
|
358
|
+
|
|
359
|
+
export interface CostBudgetConfig {
|
|
360
|
+
readonly max_input_tokens: number;
|
|
361
|
+
readonly max_output_tokens: number;
|
|
362
|
+
readonly max_calls: number;
|
|
363
|
+
readonly max_total_usd: number;
|
|
364
|
+
}
|
|
365
|
+
|
|
366
|
+
export interface MemberConfig {
|
|
367
|
+
readonly name: string;
|
|
368
|
+
readonly enabled: boolean;
|
|
369
|
+
readonly model: string;
|
|
370
|
+
readonly api_key_ref: string | null;
|
|
371
|
+
readonly mode: string | null;
|
|
372
|
+
readonly binary: string | null;
|
|
373
|
+
readonly model_ladder: readonly string[];
|
|
374
|
+
readonly participate_low_impact: boolean;
|
|
375
|
+
}
|
|
376
|
+
|
|
377
|
+
/**
|
|
378
|
+
* Replace-mode advisor binding (Phase 6).
|
|
379
|
+
*
|
|
380
|
+
* `member` names the provider whose plain call is replaced by this
|
|
381
|
+
* advisor-persona call. `persona` is the path to the advisor persona
|
|
382
|
+
* file (resolved relative to the package root). `model` is an optional
|
|
383
|
+
* override of the bound member's plain model.
|
|
384
|
+
*/
|
|
385
|
+
export interface AdvisorConfig {
|
|
386
|
+
readonly name: string;
|
|
387
|
+
readonly enabled: boolean;
|
|
388
|
+
readonly member: string;
|
|
389
|
+
readonly persona: string;
|
|
390
|
+
readonly model: string | null;
|
|
391
|
+
}
|
|
392
|
+
|
|
393
|
+
/**
|
|
394
|
+
* Consensus-scoring round settings (Phase 4 / F3). Only the `analysis`
|
|
395
|
+
* lens activates the scoring round today; other lenses see this as
|
|
396
|
+
* inert config. Thresholds are inclusive on the `strong` side and
|
|
397
|
+
* exclusive on the `minority` side. Defaults mirror the roadmap.
|
|
398
|
+
*/
|
|
399
|
+
export interface ConsensusScoringConfig {
|
|
400
|
+
readonly enabled: boolean;
|
|
401
|
+
readonly strong_threshold: number;
|
|
402
|
+
readonly minority_threshold: number;
|
|
403
|
+
readonly lenses: readonly string[];
|
|
404
|
+
}
|
|
405
|
+
|
|
406
|
+
const _VALID_NECESSITY_MODES: ReadonlySet<string> = new Set([
|
|
407
|
+
'off',
|
|
408
|
+
'educate',
|
|
409
|
+
'block',
|
|
410
|
+
'warn-only',
|
|
411
|
+
]);
|
|
412
|
+
const _VALID_DISCLOSURE_MODES: ReadonlySet<string> = new Set([
|
|
413
|
+
'always',
|
|
414
|
+
'above_threshold',
|
|
415
|
+
'off',
|
|
416
|
+
]);
|
|
417
|
+
|
|
418
|
+
/**
|
|
419
|
+
* Council-necessity classifier toggle (Phase 6). `mode` controls the
|
|
420
|
+
* agent invocation path; `user_explicit_mode` the user-explicit path.
|
|
421
|
+
*/
|
|
422
|
+
export interface NecessityClassifierConfig {
|
|
423
|
+
readonly enabled: boolean;
|
|
424
|
+
readonly mode: string;
|
|
425
|
+
readonly user_explicit_mode: string;
|
|
426
|
+
}
|
|
427
|
+
|
|
428
|
+
/** Model-size downgrade-suggestion toggle (Phase 7). */
|
|
429
|
+
export interface ModelDowngradeConfig {
|
|
430
|
+
readonly enabled: boolean;
|
|
431
|
+
readonly auto_apply: boolean;
|
|
432
|
+
}
|
|
433
|
+
|
|
434
|
+
/** Pre-flight cost-disclosure toggle (Phase 8). */
|
|
435
|
+
export interface CostDisclosureConfig {
|
|
436
|
+
readonly mode: string;
|
|
437
|
+
readonly threshold_usd: number;
|
|
438
|
+
readonly show_per_member: boolean;
|
|
439
|
+
}
|
|
440
|
+
|
|
441
|
+
/** Debate cost-visibility + hard refusal cap (Phase 8). */
|
|
442
|
+
export interface DebateConfig {
|
|
443
|
+
readonly max_cost_usd: number;
|
|
444
|
+
readonly cost_disclosure: CostDisclosureConfig;
|
|
445
|
+
}
|
|
446
|
+
|
|
447
|
+
/** Decision-replay artefact toggle (Phase 9). */
|
|
448
|
+
export interface DecisionReplayConfig {
|
|
449
|
+
readonly enabled: boolean;
|
|
450
|
+
readonly include_member_arguments: boolean;
|
|
451
|
+
}
|
|
452
|
+
|
|
453
|
+
/** Routing entry for one impact class (Phase 10). */
|
|
454
|
+
export interface DecisionResolutionEntry {
|
|
455
|
+
readonly mode: string;
|
|
456
|
+
readonly confidence_threshold: number;
|
|
457
|
+
}
|
|
458
|
+
|
|
459
|
+
/** Opt-in fuzzy matching for the corpus-aware classifier (step-9 P5). */
|
|
460
|
+
export interface FuzzyMatchConfig {
|
|
461
|
+
readonly enabled: boolean;
|
|
462
|
+
readonly threshold: number;
|
|
463
|
+
}
|
|
464
|
+
|
|
465
|
+
/** Hard caps for the lightweight-QA fast-path (Phase 11). */
|
|
466
|
+
export interface LowImpactFastPathConfig {
|
|
467
|
+
readonly max_members: number;
|
|
468
|
+
readonly max_rounds: number;
|
|
469
|
+
readonly max_tokens: number;
|
|
470
|
+
readonly max_cost_usd: number;
|
|
471
|
+
readonly fuzzy_match: FuzzyMatchConfig;
|
|
472
|
+
}
|
|
473
|
+
|
|
474
|
+
/** Impact-class → routing map (Phase 10). */
|
|
475
|
+
export interface DecisionResolutionConfig {
|
|
476
|
+
readonly enabled: boolean;
|
|
477
|
+
readonly classes: ReadonlyMap<string, DecisionResolutionEntry>;
|
|
478
|
+
readonly fast_path: LowImpactFastPathConfig;
|
|
479
|
+
}
|
|
480
|
+
|
|
481
|
+
/** Per-lens overrides keyed by lens name (Phase 6+). */
|
|
482
|
+
export interface LensOverridesConfig {
|
|
483
|
+
readonly necessity_classifier_mode: ReadonlyMap<string, string>;
|
|
484
|
+
readonly necessity_classifier_user_explicit_mode: ReadonlyMap<string, string>;
|
|
485
|
+
readonly model_downgrade: ReadonlyMap<string, ModelDowngradeConfig>;
|
|
486
|
+
readonly cost_disclosure: ReadonlyMap<string, CostDisclosureConfig>;
|
|
487
|
+
readonly decision_replay: ReadonlyMap<string, DecisionReplayConfig>;
|
|
488
|
+
}
|
|
489
|
+
|
|
490
|
+
/** Solo-member dispatch fallback chain (step-9 P8/P9 · U2). */
|
|
491
|
+
export interface RoutingConfig {
|
|
492
|
+
readonly solo_member_fallback_chain: readonly string[];
|
|
493
|
+
readonly auth_check_timeout_seconds: number;
|
|
494
|
+
}
|
|
495
|
+
|
|
496
|
+
/** Low-impact dispatch + shadow-mode config (step-9 P8/P10 · U3). */
|
|
497
|
+
export interface LowImpactConfig {
|
|
498
|
+
readonly dispatch: string;
|
|
499
|
+
readonly shadow_sample_rate: number;
|
|
500
|
+
readonly solo_confidence_floor: number;
|
|
501
|
+
}
|
|
502
|
+
|
|
503
|
+
/** Per-day call-count guard for `mode: cli` members (Phase 0). */
|
|
504
|
+
export interface CliCallBudgetConfig {
|
|
505
|
+
readonly max_calls_per_day: ReadonlyMap<string, number>;
|
|
506
|
+
readonly warn_at: number;
|
|
507
|
+
}
|
|
508
|
+
|
|
509
|
+
export interface CouncilConfig {
|
|
510
|
+
readonly enabled: boolean;
|
|
511
|
+
readonly defaults: DefaultsConfig;
|
|
512
|
+
readonly cost_budget: CostBudgetConfig;
|
|
513
|
+
readonly members: ReadonlyMap<string, MemberConfig>;
|
|
514
|
+
readonly advisors: ReadonlyMap<string, AdvisorConfig>;
|
|
515
|
+
readonly consensus_scoring: ConsensusScoringConfig;
|
|
516
|
+
readonly cli_call_budget: CliCallBudgetConfig;
|
|
517
|
+
readonly necessity_classifier: NecessityClassifierConfig;
|
|
518
|
+
readonly model_downgrade: ModelDowngradeConfig;
|
|
519
|
+
readonly debate: DebateConfig;
|
|
520
|
+
readonly decision_replay: DecisionReplayConfig;
|
|
521
|
+
readonly decision_resolution: DecisionResolutionConfig;
|
|
522
|
+
readonly routing: RoutingConfig;
|
|
523
|
+
readonly low_impact: LowImpactConfig;
|
|
524
|
+
readonly lens_overrides: LensOverridesConfig;
|
|
525
|
+
readonly source_path: PathLike | null;
|
|
526
|
+
}
|
|
527
|
+
|
|
528
|
+
// ── default factories (dataclass defaults) ─────────────────────────
|
|
529
|
+
|
|
530
|
+
function _defaultCostDisclosure(): CostDisclosureConfig {
|
|
531
|
+
return { mode: 'always', threshold_usd: 1.0, show_per_member: true };
|
|
532
|
+
}
|
|
533
|
+
|
|
534
|
+
function _defaultFuzzyMatch(): FuzzyMatchConfig {
|
|
535
|
+
return { enabled: false, threshold: 0.92 };
|
|
536
|
+
}
|
|
537
|
+
|
|
538
|
+
function _defaultFastPath(): LowImpactFastPathConfig {
|
|
539
|
+
return {
|
|
540
|
+
max_members: 2,
|
|
541
|
+
max_rounds: 1,
|
|
542
|
+
max_tokens: 2500,
|
|
543
|
+
max_cost_usd: 0.05,
|
|
544
|
+
fuzzy_match: _defaultFuzzyMatch(),
|
|
545
|
+
};
|
|
546
|
+
}
|
|
547
|
+
|
|
548
|
+
// ── path constants ─────────────────────────────────────────────────
|
|
549
|
+
|
|
550
|
+
/** Dotfile name for the council config in any scope. */
|
|
551
|
+
export const COUNCIL_CONFIG_RELNAME = '.ai-council.yml';
|
|
552
|
+
|
|
553
|
+
/**
|
|
554
|
+
* User-global location, relative to `event4u_root()` — under `settings/`
|
|
555
|
+
* alongside the other per-user config (`.agent-settings.yml`,
|
|
556
|
+
* `.agent-user.yml`). This is exactly the path the browser setup wizard
|
|
557
|
+
* reads/writes (`<writeRoot>/settings/.ai-council.yml` in
|
|
558
|
+
* `src/server/routes/wizard.ts`), so a council configured in the wizard
|
|
559
|
+
* is the same file the CLI reads.
|
|
560
|
+
*/
|
|
561
|
+
export const COUNCIL_CONFIG_USER_GLOBAL_REL = 'settings/.ai-council.yml';
|
|
562
|
+
|
|
563
|
+
/**
|
|
564
|
+
* Env var pinning the council config to an explicit absolute path, ahead
|
|
565
|
+
* of the project → user-global search. Mirrors `EVENT4U_CONFIG_HOME` but
|
|
566
|
+
* targets the config file itself (tests / power users).
|
|
567
|
+
*/
|
|
568
|
+
export const COUNCIL_CONFIG_ENV = 'AI_COUNCIL_CONFIG';
|
|
569
|
+
|
|
570
|
+
// ── expanduser (Python Path.expanduser parity) ────────────────────
|
|
571
|
+
|
|
572
|
+
function _expanduser(p: string): string {
|
|
573
|
+
if (p === '~') {
|
|
574
|
+
return os.homedir();
|
|
575
|
+
}
|
|
576
|
+
if (
|
|
577
|
+
p.startsWith('~/') ||
|
|
578
|
+
(process.platform === 'win32' && p.startsWith('~\\'))
|
|
579
|
+
) {
|
|
580
|
+
return path.join(os.homedir(), p.slice(2));
|
|
581
|
+
}
|
|
582
|
+
return p;
|
|
583
|
+
}
|
|
584
|
+
|
|
585
|
+
/** Mirror of `pathlib.Path.is_absolute()` (POSIX + Windows). */
|
|
586
|
+
function _isAbsoluteLikePython(p: string): boolean {
|
|
587
|
+
if (process.platform === 'win32') {
|
|
588
|
+
return (
|
|
589
|
+
/^[a-zA-Z]:[\\/]/.test(p) ||
|
|
590
|
+
/^([\\/]{2})[^\\/]+[\\/][^\\/]+/.test(p)
|
|
591
|
+
);
|
|
592
|
+
}
|
|
593
|
+
return p.startsWith('/');
|
|
594
|
+
}
|
|
595
|
+
|
|
596
|
+
// ── path resolution ────────────────────────────────────────────────
|
|
597
|
+
|
|
598
|
+
/**
|
|
599
|
+
* Resolve which `.ai-council.yml` the council reads.
|
|
600
|
+
*
|
|
601
|
+
* Precedence (first match wins):
|
|
602
|
+
*
|
|
603
|
+
* 1. `$AI_COUNCIL_CONFIG` — explicit absolute override (tests / power
|
|
604
|
+
* users). Honoured even when the target is absent.
|
|
605
|
+
* 2. Project-local `<project_root>/agents/settings/.ai-council.yml`.
|
|
606
|
+
* 3. User-global `~/.event4u/agent-config/settings/.ai-council.yml`
|
|
607
|
+
* (with the legacy `~/.config/agent-config/` read-fallback).
|
|
608
|
+
*
|
|
609
|
+
* Always returns a path (never `null`): when nothing exists yet it
|
|
610
|
+
* returns the user-global write target.
|
|
611
|
+
*/
|
|
612
|
+
export function resolve_config_path(
|
|
613
|
+
project_root: PathLike,
|
|
614
|
+
options: { env?: user_global_paths.EnvMap | null } = {},
|
|
615
|
+
): PathLike {
|
|
616
|
+
const env_map = options.env != null ? options.env : process.env;
|
|
617
|
+
const override = env_map[COUNCIL_CONFIG_ENV];
|
|
618
|
+
if (override) {
|
|
619
|
+
return _expanduser(override);
|
|
620
|
+
}
|
|
621
|
+
const project_path = path.join(
|
|
622
|
+
project_root,
|
|
623
|
+
'agents',
|
|
624
|
+
'settings',
|
|
625
|
+
COUNCIL_CONFIG_RELNAME,
|
|
626
|
+
);
|
|
627
|
+
if (fs.existsSync(project_path)) {
|
|
628
|
+
return project_path;
|
|
629
|
+
}
|
|
630
|
+
const found = user_global_paths.resolve_with_fallback(
|
|
631
|
+
COUNCIL_CONFIG_USER_GLOBAL_REL,
|
|
632
|
+
{ env: options.env ?? null },
|
|
633
|
+
);
|
|
634
|
+
if (found !== null) {
|
|
635
|
+
return found;
|
|
636
|
+
}
|
|
637
|
+
return user_global_paths.write_target(COUNCIL_CONFIG_USER_GLOBAL_REL, {
|
|
638
|
+
env: options.env ?? null,
|
|
639
|
+
});
|
|
640
|
+
}
|
|
641
|
+
|
|
642
|
+
// ── YAML loading ───────────────────────────────────────────────────
|
|
643
|
+
|
|
644
|
+
/** Load and validate the council YAML at `path`. */
|
|
645
|
+
export function load_council_config(p: PathLike): CouncilConfig {
|
|
646
|
+
if (!fs.existsSync(p)) {
|
|
647
|
+
throw new CouncilConfigError(
|
|
648
|
+
`Council config not found at ${p}. ` +
|
|
649
|
+
`Create it per docs/contracts/ai-council-config.md.`,
|
|
650
|
+
);
|
|
651
|
+
}
|
|
652
|
+
let raw: Json;
|
|
653
|
+
try {
|
|
654
|
+
const text = fs.readFileSync(p, 'utf-8');
|
|
655
|
+
// version '1.1' matches PyYAML's safe_load scalar grammar.
|
|
656
|
+
const parsed = parseYaml(text, { version: '1.1' }) as Json;
|
|
657
|
+
raw = parsed === null || parsed === undefined ? {} : parsed;
|
|
658
|
+
} catch (exc) {
|
|
659
|
+
if (exc instanceof CouncilConfigError) {
|
|
660
|
+
throw exc;
|
|
661
|
+
}
|
|
662
|
+
const msg = exc instanceof Error ? exc.message : String(exc);
|
|
663
|
+
throw new CouncilConfigError(`${p}: invalid YAML — ${msg}`);
|
|
664
|
+
}
|
|
665
|
+
if (!_isDict(raw)) {
|
|
666
|
+
throw new CouncilConfigError(`${p}: top-level must be a mapping.`);
|
|
667
|
+
}
|
|
668
|
+
return _build_config(raw, p);
|
|
669
|
+
}
|
|
670
|
+
|
|
671
|
+
// ── builders ───────────────────────────────────────────────────────
|
|
672
|
+
|
|
673
|
+
export function _build_config(raw: Dict, source_path: PathLike): CouncilConfig {
|
|
674
|
+
const enabled = _get(raw, 'enabled', false);
|
|
675
|
+
if (!_isBool(enabled)) {
|
|
676
|
+
throw new CouncilConfigError('`enabled` must be a bool.');
|
|
677
|
+
}
|
|
678
|
+
|
|
679
|
+
const defaults = _build_defaults(_asDict(_getOr(raw, 'defaults', {})));
|
|
680
|
+
const cost_budget = _build_cost_budget(
|
|
681
|
+
_asDict(_getOr(raw, 'cost_budget', {})),
|
|
682
|
+
);
|
|
683
|
+
|
|
684
|
+
const members_raw = _getOr(raw, 'members', {});
|
|
685
|
+
if (!_isDict(members_raw)) {
|
|
686
|
+
throw new CouncilConfigError('`members` must be a mapping.');
|
|
687
|
+
}
|
|
688
|
+
const members = new Map<string, MemberConfig>();
|
|
689
|
+
for (const [name, cfg] of Object.entries(members_raw)) {
|
|
690
|
+
members.set(
|
|
691
|
+
name,
|
|
692
|
+
_build_member(name, _asDict(_orEmpty(cfg)), defaults.mode),
|
|
693
|
+
);
|
|
694
|
+
}
|
|
695
|
+
|
|
696
|
+
const advisors_raw = _getOr(raw, 'advisors', {});
|
|
697
|
+
if (!_isDict(advisors_raw)) {
|
|
698
|
+
throw new CouncilConfigError('`advisors` must be a mapping.');
|
|
699
|
+
}
|
|
700
|
+
const advisors = new Map<string, AdvisorConfig>();
|
|
701
|
+
for (const [adv_name, adv_cfg] of Object.entries(advisors_raw)) {
|
|
702
|
+
advisors.set(adv_name, _build_advisor(adv_name, _asDict(_orEmpty(adv_cfg))));
|
|
703
|
+
}
|
|
704
|
+
|
|
705
|
+
// Cross-validate enabled advisors against the members block. An
|
|
706
|
+
// advisor referencing a missing or disabled member is a hard error
|
|
707
|
+
// — never a silent skip — so a typo never costs the user money on
|
|
708
|
+
// an unintended call plan.
|
|
709
|
+
for (const adv of advisors.values()) {
|
|
710
|
+
if (!adv.enabled) {
|
|
711
|
+
continue;
|
|
712
|
+
}
|
|
713
|
+
const bound = members.get(adv.member);
|
|
714
|
+
if (bound === undefined) {
|
|
715
|
+
throw new CouncilConfigError(
|
|
716
|
+
`advisors.${adv.name}.member=${_pyReprStr(adv.member)}: no such ` +
|
|
717
|
+
`member in the \`members\` block.`,
|
|
718
|
+
);
|
|
719
|
+
}
|
|
720
|
+
if (!bound.enabled) {
|
|
721
|
+
throw new CouncilConfigError(
|
|
722
|
+
`advisors.${adv.name}.member=${_pyReprStr(adv.member)}: member ` +
|
|
723
|
+
`exists but is disabled. Enable the member or disable ` +
|
|
724
|
+
`the advisor.`,
|
|
725
|
+
);
|
|
726
|
+
}
|
|
727
|
+
}
|
|
728
|
+
|
|
729
|
+
const consensus = _build_consensus_scoring(
|
|
730
|
+
_asDict(_getOr(raw, 'consensus_scoring', {})),
|
|
731
|
+
);
|
|
732
|
+
const cli_call_budget = _build_cli_call_budget(
|
|
733
|
+
_asDict(_getOr(raw, 'cli_call_budget', {})),
|
|
734
|
+
);
|
|
735
|
+
const necessity_classifier = _build_necessity_classifier(
|
|
736
|
+
_asDict(_getOr(raw, 'necessity_classifier', {})),
|
|
737
|
+
);
|
|
738
|
+
const model_downgrade = _build_model_downgrade(
|
|
739
|
+
_asDict(_getOr(raw, 'model_downgrade', {})),
|
|
740
|
+
);
|
|
741
|
+
const debate = _build_debate(_asDict(_getOr(raw, 'debate', {})));
|
|
742
|
+
const decision_replay = _build_decision_replay(
|
|
743
|
+
_asDict(_getOr(raw, 'decision_replay', {})),
|
|
744
|
+
'decision_replay',
|
|
745
|
+
);
|
|
746
|
+
const decision_resolution = _build_decision_resolution(
|
|
747
|
+
_asDict(_getOr(raw, 'decision_resolution', {})),
|
|
748
|
+
);
|
|
749
|
+
const routing = _build_routing(_asDict(_getOr(raw, 'routing', {})), members);
|
|
750
|
+
const low_impact = _build_low_impact(
|
|
751
|
+
_asDict(_getOr(raw, 'low_impact', {})),
|
|
752
|
+
members,
|
|
753
|
+
routing,
|
|
754
|
+
);
|
|
755
|
+
_reject_top_level_locked_dispatch(raw);
|
|
756
|
+
const lens_overrides = _build_lens_overrides(
|
|
757
|
+
_asDict(_getOr(raw, 'lenses', {})),
|
|
758
|
+
);
|
|
759
|
+
|
|
760
|
+
return {
|
|
761
|
+
enabled,
|
|
762
|
+
defaults,
|
|
763
|
+
cost_budget,
|
|
764
|
+
members,
|
|
765
|
+
advisors,
|
|
766
|
+
consensus_scoring: consensus,
|
|
767
|
+
cli_call_budget,
|
|
768
|
+
necessity_classifier,
|
|
769
|
+
model_downgrade,
|
|
770
|
+
debate,
|
|
771
|
+
decision_replay,
|
|
772
|
+
decision_resolution,
|
|
773
|
+
routing,
|
|
774
|
+
low_impact,
|
|
775
|
+
lens_overrides,
|
|
776
|
+
source_path,
|
|
777
|
+
};
|
|
778
|
+
}
|
|
779
|
+
|
|
780
|
+
/**
|
|
781
|
+
* Mirror Python's `cfg or {}` for a per-entry value: falsy → `{}`.
|
|
782
|
+
* Used for `cfg or {}` / `adv_cfg or {}` in the members/advisors loops.
|
|
783
|
+
*/
|
|
784
|
+
function _orEmpty(value: Json): Json {
|
|
785
|
+
if (
|
|
786
|
+
value === null ||
|
|
787
|
+
value === undefined ||
|
|
788
|
+
value === false ||
|
|
789
|
+
value === 0 ||
|
|
790
|
+
value === '' ||
|
|
791
|
+
(Array.isArray(value) && value.length === 0) ||
|
|
792
|
+
(_isDict(value) && Object.keys(value).length === 0)
|
|
793
|
+
) {
|
|
794
|
+
return {};
|
|
795
|
+
}
|
|
796
|
+
return value;
|
|
797
|
+
}
|
|
798
|
+
|
|
799
|
+
/**
|
|
800
|
+
* Narrow a `Json` known to be a dict (after a `_getOr(..., {})` /
|
|
801
|
+
* `_orEmpty` upstream) — but a non-dict here means the caller passed a
|
|
802
|
+
* raw value that the builder will reject with its own message. Pass the
|
|
803
|
+
* raw value through so per-builder `isinstance(d, dict)` checks fire.
|
|
804
|
+
*/
|
|
805
|
+
function _asDict(value: Json): Dict {
|
|
806
|
+
return value as Dict;
|
|
807
|
+
}
|
|
808
|
+
|
|
809
|
+
function _build_necessity_classifier(d: Dict): NecessityClassifierConfig {
|
|
810
|
+
if (!_isDict(d)) {
|
|
811
|
+
throw new CouncilConfigError('`necessity_classifier` must be a mapping.');
|
|
812
|
+
}
|
|
813
|
+
const enabled = _get(d, 'enabled', true);
|
|
814
|
+
if (!_isBool(enabled)) {
|
|
815
|
+
throw new CouncilConfigError(
|
|
816
|
+
'`necessity_classifier.enabled` must be a bool.',
|
|
817
|
+
);
|
|
818
|
+
}
|
|
819
|
+
const mode = _get(d, 'mode', 'educate');
|
|
820
|
+
if (!(_isStr(mode) && _VALID_NECESSITY_MODES.has(mode))) {
|
|
821
|
+
throw new CouncilConfigError(
|
|
822
|
+
`necessity_classifier.mode=${_pyRepr(mode)} not in ` +
|
|
823
|
+
`${_sortedListRepr(_VALID_NECESSITY_MODES)}.`,
|
|
824
|
+
);
|
|
825
|
+
}
|
|
826
|
+
const user_explicit_mode = _get(d, 'user_explicit_mode', 'warn-only');
|
|
827
|
+
if (
|
|
828
|
+
!(_isStr(user_explicit_mode) && _VALID_NECESSITY_MODES.has(user_explicit_mode))
|
|
829
|
+
) {
|
|
830
|
+
throw new CouncilConfigError(
|
|
831
|
+
`necessity_classifier.user_explicit_mode=` +
|
|
832
|
+
`${_pyRepr(user_explicit_mode)} not in ` +
|
|
833
|
+
`${_sortedListRepr(_VALID_NECESSITY_MODES)}.`,
|
|
834
|
+
);
|
|
835
|
+
}
|
|
836
|
+
return {
|
|
837
|
+
enabled: Boolean(enabled),
|
|
838
|
+
mode,
|
|
839
|
+
user_explicit_mode,
|
|
840
|
+
};
|
|
841
|
+
}
|
|
842
|
+
|
|
843
|
+
function _build_model_downgrade(d: Dict): ModelDowngradeConfig {
|
|
844
|
+
if (!_isDict(d)) {
|
|
845
|
+
throw new CouncilConfigError('`model_downgrade` must be a mapping.');
|
|
846
|
+
}
|
|
847
|
+
const enabled = _get(d, 'enabled', true);
|
|
848
|
+
if (!_isBool(enabled)) {
|
|
849
|
+
throw new CouncilConfigError('`model_downgrade.enabled` must be a bool.');
|
|
850
|
+
}
|
|
851
|
+
const auto_apply = _get(d, 'auto_apply', false);
|
|
852
|
+
if (!_isBool(auto_apply)) {
|
|
853
|
+
throw new CouncilConfigError('`model_downgrade.auto_apply` must be a bool.');
|
|
854
|
+
}
|
|
855
|
+
return { enabled: Boolean(enabled), auto_apply: Boolean(auto_apply) };
|
|
856
|
+
}
|
|
857
|
+
|
|
858
|
+
function _build_cost_disclosure(d: Dict, scope: string): CostDisclosureConfig {
|
|
859
|
+
if (!_isDict(d)) {
|
|
860
|
+
throw new CouncilConfigError(`\`${scope}\` must be a mapping.`);
|
|
861
|
+
}
|
|
862
|
+
const mode = _get(d, 'mode', 'always');
|
|
863
|
+
if (!(_isStr(mode) && _VALID_DISCLOSURE_MODES.has(mode))) {
|
|
864
|
+
throw new CouncilConfigError(
|
|
865
|
+
`${scope}.mode=${_pyRepr(mode)} not in ` +
|
|
866
|
+
`${_sortedListRepr(_VALID_DISCLOSURE_MODES)}.`,
|
|
867
|
+
);
|
|
868
|
+
}
|
|
869
|
+
const threshold = _pyFloat(_get(d, 'threshold_usd', 1.0));
|
|
870
|
+
if (threshold < 0) {
|
|
871
|
+
throw new CouncilConfigError(
|
|
872
|
+
`${scope}.threshold_usd must be >= 0 (got ${_pyRepr(_f(threshold))}).`,
|
|
873
|
+
);
|
|
874
|
+
}
|
|
875
|
+
const show_per_member = _get(d, 'show_per_member', true);
|
|
876
|
+
if (!_isBool(show_per_member)) {
|
|
877
|
+
throw new CouncilConfigError(`\`${scope}.show_per_member\` must be a bool.`);
|
|
878
|
+
}
|
|
879
|
+
return {
|
|
880
|
+
mode,
|
|
881
|
+
threshold_usd: threshold,
|
|
882
|
+
show_per_member: Boolean(show_per_member),
|
|
883
|
+
};
|
|
884
|
+
}
|
|
885
|
+
|
|
886
|
+
function _build_debate(d: Dict): DebateConfig {
|
|
887
|
+
if (!_isDict(d)) {
|
|
888
|
+
throw new CouncilConfigError('`debate` must be a mapping.');
|
|
889
|
+
}
|
|
890
|
+
const cap = _pyFloat(_get(d, 'max_cost_usd', 5.0));
|
|
891
|
+
if (cap < 0) {
|
|
892
|
+
throw new CouncilConfigError(
|
|
893
|
+
`debate.max_cost_usd must be >= 0 (got ${_pyRepr(_f(cap))}; ` +
|
|
894
|
+
`use 0 to disable the cap).`,
|
|
895
|
+
);
|
|
896
|
+
}
|
|
897
|
+
const disclosure_raw = _getOr(d, 'cost_disclosure', {});
|
|
898
|
+
const disclosure = _build_cost_disclosure(
|
|
899
|
+
_asDict(disclosure_raw),
|
|
900
|
+
'debate.cost_disclosure',
|
|
901
|
+
);
|
|
902
|
+
return { max_cost_usd: cap, cost_disclosure: disclosure };
|
|
903
|
+
}
|
|
904
|
+
|
|
905
|
+
function _build_decision_replay(d: Dict, scope: string): DecisionReplayConfig {
|
|
906
|
+
if (!_isDict(d)) {
|
|
907
|
+
throw new CouncilConfigError(`\`${scope}\` must be a mapping.`);
|
|
908
|
+
}
|
|
909
|
+
const enabled = _get(d, 'enabled', true);
|
|
910
|
+
if (!_isBool(enabled)) {
|
|
911
|
+
throw new CouncilConfigError(`\`${scope}.enabled\` must be a bool.`);
|
|
912
|
+
}
|
|
913
|
+
const include_args = _get(d, 'include_member_arguments', true);
|
|
914
|
+
if (!_isBool(include_args)) {
|
|
915
|
+
throw new CouncilConfigError(
|
|
916
|
+
`\`${scope}.include_member_arguments\` must be a bool.`,
|
|
917
|
+
);
|
|
918
|
+
}
|
|
919
|
+
return {
|
|
920
|
+
enabled: Boolean(enabled),
|
|
921
|
+
include_member_arguments: Boolean(include_args),
|
|
922
|
+
};
|
|
923
|
+
}
|
|
924
|
+
|
|
925
|
+
const _VALID_RESOLUTION_MODES: ReadonlySet<string> = new Set([
|
|
926
|
+
'agent',
|
|
927
|
+
'council',
|
|
928
|
+
'user',
|
|
929
|
+
]);
|
|
930
|
+
const _IMPACT_CLASSES: readonly string[] = [
|
|
931
|
+
'trivial',
|
|
932
|
+
'low_impact',
|
|
933
|
+
'medium_impact',
|
|
934
|
+
'high_impact',
|
|
935
|
+
'user_required',
|
|
936
|
+
];
|
|
937
|
+
const _LOCKED_IMPACT_CLASSES: ReadonlySet<string> = new Set([
|
|
938
|
+
'high_impact',
|
|
939
|
+
'user_required',
|
|
940
|
+
]);
|
|
941
|
+
|
|
942
|
+
const _DEFAULT_RESOLUTION_MODES: Readonly<Record<string, string>> = {
|
|
943
|
+
trivial: 'agent',
|
|
944
|
+
low_impact: 'agent',
|
|
945
|
+
medium_impact: 'council',
|
|
946
|
+
high_impact: 'user',
|
|
947
|
+
user_required: 'user',
|
|
948
|
+
};
|
|
949
|
+
|
|
950
|
+
function _build_decision_resolution(d: Dict): DecisionResolutionConfig {
|
|
951
|
+
if (!_isDict(d)) {
|
|
952
|
+
throw new CouncilConfigError('`decision_resolution` must be a mapping.');
|
|
953
|
+
}
|
|
954
|
+
const enabled = _get(d, 'enabled', true);
|
|
955
|
+
if (!_isBool(enabled)) {
|
|
956
|
+
throw new CouncilConfigError(
|
|
957
|
+
'`decision_resolution.enabled` must be a bool.',
|
|
958
|
+
);
|
|
959
|
+
}
|
|
960
|
+
const classes_raw = _getOr(d, 'classes', {});
|
|
961
|
+
if (!_isDict(classes_raw)) {
|
|
962
|
+
throw new CouncilConfigError(
|
|
963
|
+
'`decision_resolution.classes` must be a mapping.',
|
|
964
|
+
);
|
|
965
|
+
}
|
|
966
|
+
const classes = new Map<string, DecisionResolutionEntry>();
|
|
967
|
+
for (const cls of _IMPACT_CLASSES) {
|
|
968
|
+
const entry_raw = _orEmpty(classes_raw[cls] ?? null);
|
|
969
|
+
if (!_isDict(entry_raw)) {
|
|
970
|
+
throw new CouncilConfigError(
|
|
971
|
+
`\`decision_resolution.classes.${cls}\` must be a mapping.`,
|
|
972
|
+
);
|
|
973
|
+
}
|
|
974
|
+
const mode = _get(entry_raw, 'mode', _DEFAULT_RESOLUTION_MODES[cls] as string);
|
|
975
|
+
if (!(_isStr(mode) && _VALID_RESOLUTION_MODES.has(mode))) {
|
|
976
|
+
throw new CouncilConfigError(
|
|
977
|
+
`decision_resolution.classes.${cls}.mode=${_pyRepr(mode)} not in ` +
|
|
978
|
+
`${_sortedListRepr(_VALID_RESOLUTION_MODES)}.`,
|
|
979
|
+
);
|
|
980
|
+
}
|
|
981
|
+
// Iron Law: high_impact + user_required are locked to user.
|
|
982
|
+
if (_LOCKED_IMPACT_CLASSES.has(cls) && mode !== 'user') {
|
|
983
|
+
throw new CouncilConfigError(
|
|
984
|
+
`decision_resolution.classes.${cls}.mode=${_pyRepr(mode)}: ` +
|
|
985
|
+
`class \`${cls}\` is LOCKED to \`user\` (Iron Law) — ` +
|
|
986
|
+
`high-impact and user-required decisions never bypass ` +
|
|
987
|
+
`the user.`,
|
|
988
|
+
);
|
|
989
|
+
}
|
|
990
|
+
// Iron Law: `dispatch` is not configurable for locked classes
|
|
991
|
+
// (step-9 P8/P11 · U3). Any nested `dispatch` key — including
|
|
992
|
+
// smuggled-in YAML anchor merges — is a hard schema error.
|
|
993
|
+
if (_LOCKED_IMPACT_CLASSES.has(cls) && 'dispatch' in entry_raw) {
|
|
994
|
+
throw new CouncilConfigError(
|
|
995
|
+
`decision_resolution.classes.${cls}.dispatch=` +
|
|
996
|
+
`${_pyRepr(entry_raw.dispatch)}: dispatch is not ` +
|
|
997
|
+
`configurable for high-impact / user-required ` +
|
|
998
|
+
`decisions — always full council.`,
|
|
999
|
+
);
|
|
1000
|
+
}
|
|
1001
|
+
const threshold = _pyFloat(_get(entry_raw, 'confidence_threshold', 0.6));
|
|
1002
|
+
if (!(0.0 <= threshold && threshold <= 1.0)) {
|
|
1003
|
+
throw new CouncilConfigError(
|
|
1004
|
+
`decision_resolution.classes.${cls}.confidence_threshold ` +
|
|
1005
|
+
`must be in [0.0, 1.0] (got ${_pyRepr(_f(threshold))}).`,
|
|
1006
|
+
);
|
|
1007
|
+
}
|
|
1008
|
+
classes.set(cls, { mode, confidence_threshold: threshold });
|
|
1009
|
+
}
|
|
1010
|
+
const fast_path_raw = _getOr(d, 'fast_path', {});
|
|
1011
|
+
if (!_isDict(fast_path_raw)) {
|
|
1012
|
+
throw new CouncilConfigError(
|
|
1013
|
+
'`decision_resolution.fast_path` must be a mapping.',
|
|
1014
|
+
);
|
|
1015
|
+
}
|
|
1016
|
+
const fast_path = _build_fast_path(_asDict(fast_path_raw));
|
|
1017
|
+
return { enabled: Boolean(enabled), classes, fast_path };
|
|
1018
|
+
}
|
|
1019
|
+
|
|
1020
|
+
function _build_fast_path(d: Dict): LowImpactFastPathConfig {
|
|
1021
|
+
const max_members = _get(d, 'max_members', 2);
|
|
1022
|
+
if (!_isInt(max_members) || _isBool(max_members)) {
|
|
1023
|
+
throw new CouncilConfigError(
|
|
1024
|
+
'decision_resolution.fast_path.max_members must be an int ' +
|
|
1025
|
+
`(got ${_pyTypeName(max_members)}).`,
|
|
1026
|
+
);
|
|
1027
|
+
}
|
|
1028
|
+
if (max_members < 1 || max_members > 2) {
|
|
1029
|
+
throw new CouncilConfigError(
|
|
1030
|
+
'decision_resolution.fast_path.max_members must be 1 or 2 ' +
|
|
1031
|
+
`(got ${max_members}). Fast-path is by design a 1-2 member ` +
|
|
1032
|
+
'lookup — wider fan-out belongs in the standard council path.',
|
|
1033
|
+
);
|
|
1034
|
+
}
|
|
1035
|
+
const max_rounds = _get(d, 'max_rounds', 1);
|
|
1036
|
+
if (max_rounds !== 1) {
|
|
1037
|
+
throw new CouncilConfigError(
|
|
1038
|
+
'decision_resolution.fast_path.max_rounds is LOCKED to 1 ' +
|
|
1039
|
+
`(got ${_pyRepr(max_rounds)}). Multi-round fast-paths defeat the ` +
|
|
1040
|
+
'purpose — escalate to standard council instead.',
|
|
1041
|
+
);
|
|
1042
|
+
}
|
|
1043
|
+
const max_tokens = _get(d, 'max_tokens', 2500);
|
|
1044
|
+
if (!_isInt(max_tokens) || _isBool(max_tokens) || max_tokens <= 0) {
|
|
1045
|
+
throw new CouncilConfigError(
|
|
1046
|
+
'decision_resolution.fast_path.max_tokens must be a positive ' +
|
|
1047
|
+
`int (got ${_pyRepr(max_tokens)}).`,
|
|
1048
|
+
);
|
|
1049
|
+
}
|
|
1050
|
+
const max_cost_raw = _get(d, 'max_cost_usd', 0.05);
|
|
1051
|
+
if (_isBool(max_cost_raw) || !_isNumber(max_cost_raw)) {
|
|
1052
|
+
throw new CouncilConfigError(
|
|
1053
|
+
'decision_resolution.fast_path.max_cost_usd must be a ' +
|
|
1054
|
+
`number (got ${_pyTypeName(max_cost_raw)}).`,
|
|
1055
|
+
);
|
|
1056
|
+
}
|
|
1057
|
+
const max_cost = _pyFloat(max_cost_raw);
|
|
1058
|
+
if (max_cost <= 0.0) {
|
|
1059
|
+
throw new CouncilConfigError(
|
|
1060
|
+
'decision_resolution.fast_path.max_cost_usd must be > 0 ' +
|
|
1061
|
+
`(got ${_pyRepr(_f(max_cost))}).`,
|
|
1062
|
+
);
|
|
1063
|
+
}
|
|
1064
|
+
const fuzzy_match = _build_fuzzy_match(_asDict(_getOr(d, 'fuzzy_match', {})));
|
|
1065
|
+
return {
|
|
1066
|
+
max_members,
|
|
1067
|
+
max_rounds: 1,
|
|
1068
|
+
max_tokens,
|
|
1069
|
+
max_cost_usd: max_cost,
|
|
1070
|
+
fuzzy_match,
|
|
1071
|
+
};
|
|
1072
|
+
}
|
|
1073
|
+
|
|
1074
|
+
function _build_fuzzy_match(d: Dict): FuzzyMatchConfig {
|
|
1075
|
+
if (!_isDict(d)) {
|
|
1076
|
+
throw new CouncilConfigError(
|
|
1077
|
+
'decision_resolution.fast_path.fuzzy_match must be a mapping.',
|
|
1078
|
+
);
|
|
1079
|
+
}
|
|
1080
|
+
const enabled = _get(d, 'enabled', false);
|
|
1081
|
+
if (!_isBool(enabled)) {
|
|
1082
|
+
throw new CouncilConfigError(
|
|
1083
|
+
'decision_resolution.fast_path.fuzzy_match.enabled must be a bool ' +
|
|
1084
|
+
`(got ${_pyTypeName(enabled)}).`,
|
|
1085
|
+
);
|
|
1086
|
+
}
|
|
1087
|
+
const threshold_raw = _get(d, 'threshold', 0.92);
|
|
1088
|
+
if (_isBool(threshold_raw) || !_isNumber(threshold_raw)) {
|
|
1089
|
+
throw new CouncilConfigError(
|
|
1090
|
+
'decision_resolution.fast_path.fuzzy_match.threshold must be a ' +
|
|
1091
|
+
`number (got ${_pyTypeName(threshold_raw)}).`,
|
|
1092
|
+
);
|
|
1093
|
+
}
|
|
1094
|
+
const threshold = _pyFloat(threshold_raw);
|
|
1095
|
+
if (!(0.0 < threshold && threshold <= 1.0)) {
|
|
1096
|
+
throw new CouncilConfigError(
|
|
1097
|
+
'decision_resolution.fast_path.fuzzy_match.threshold must be in ' +
|
|
1098
|
+
`(0.0, 1.0] (got ${_pyRepr(_f(threshold))}).`,
|
|
1099
|
+
);
|
|
1100
|
+
}
|
|
1101
|
+
return { enabled, threshold };
|
|
1102
|
+
}
|
|
1103
|
+
|
|
1104
|
+
function _build_lens_overrides(d: Dict): LensOverridesConfig {
|
|
1105
|
+
if (!_isDict(d)) {
|
|
1106
|
+
throw new CouncilConfigError('`lenses` must be a mapping.');
|
|
1107
|
+
}
|
|
1108
|
+
const nc_overrides = new Map<string, string>();
|
|
1109
|
+
const nc_user_overrides = new Map<string, string>();
|
|
1110
|
+
const md_overrides = new Map<string, ModelDowngradeConfig>();
|
|
1111
|
+
const cd_overrides = new Map<string, CostDisclosureConfig>();
|
|
1112
|
+
const dr_overrides = new Map<string, DecisionReplayConfig>();
|
|
1113
|
+
for (const [lens_name, lens_cfg] of Object.entries(d)) {
|
|
1114
|
+
if (!_isDict(lens_cfg)) {
|
|
1115
|
+
throw new CouncilConfigError(
|
|
1116
|
+
`\`lenses.${lens_name}\` must be a mapping.`,
|
|
1117
|
+
);
|
|
1118
|
+
}
|
|
1119
|
+
const nc_block = lens_cfg.necessity_classifier;
|
|
1120
|
+
if (nc_block !== undefined && nc_block !== null) {
|
|
1121
|
+
if (!_isDict(nc_block)) {
|
|
1122
|
+
throw new CouncilConfigError(
|
|
1123
|
+
`\`lenses.${lens_name}.necessity_classifier\` must be a mapping.`,
|
|
1124
|
+
);
|
|
1125
|
+
}
|
|
1126
|
+
const mode = nc_block.mode;
|
|
1127
|
+
if (mode !== undefined && mode !== null) {
|
|
1128
|
+
if (!(_isStr(mode) && _VALID_NECESSITY_MODES.has(mode))) {
|
|
1129
|
+
throw new CouncilConfigError(
|
|
1130
|
+
`lenses.${lens_name}.necessity_classifier.mode=${_pyRepr(mode)} ` +
|
|
1131
|
+
`not in ${_sortedListRepr(_VALID_NECESSITY_MODES)}.`,
|
|
1132
|
+
);
|
|
1133
|
+
}
|
|
1134
|
+
nc_overrides.set(lens_name, mode);
|
|
1135
|
+
}
|
|
1136
|
+
const user_mode = nc_block.user_explicit_mode;
|
|
1137
|
+
if (user_mode !== undefined && user_mode !== null) {
|
|
1138
|
+
if (!(_isStr(user_mode) && _VALID_NECESSITY_MODES.has(user_mode))) {
|
|
1139
|
+
throw new CouncilConfigError(
|
|
1140
|
+
`lenses.${lens_name}.necessity_classifier.` +
|
|
1141
|
+
`user_explicit_mode=${_pyRepr(user_mode)} ` +
|
|
1142
|
+
`not in ${_sortedListRepr(_VALID_NECESSITY_MODES)}.`,
|
|
1143
|
+
);
|
|
1144
|
+
}
|
|
1145
|
+
nc_user_overrides.set(lens_name, user_mode);
|
|
1146
|
+
}
|
|
1147
|
+
}
|
|
1148
|
+
const md_block = lens_cfg.model_downgrade;
|
|
1149
|
+
if (md_block !== undefined && md_block !== null) {
|
|
1150
|
+
if (!_isDict(md_block)) {
|
|
1151
|
+
throw new CouncilConfigError(
|
|
1152
|
+
`\`lenses.${lens_name}.model_downgrade\` must be a mapping.`,
|
|
1153
|
+
);
|
|
1154
|
+
}
|
|
1155
|
+
const md_enabled = _get(md_block, 'enabled', true);
|
|
1156
|
+
if (!_isBool(md_enabled)) {
|
|
1157
|
+
throw new CouncilConfigError(
|
|
1158
|
+
`\`lenses.${lens_name}.model_downgrade.enabled\` must be a bool.`,
|
|
1159
|
+
);
|
|
1160
|
+
}
|
|
1161
|
+
const md_auto = _get(md_block, 'auto_apply', false);
|
|
1162
|
+
if (!_isBool(md_auto)) {
|
|
1163
|
+
throw new CouncilConfigError(
|
|
1164
|
+
`\`lenses.${lens_name}.model_downgrade.auto_apply\` must be a bool.`,
|
|
1165
|
+
);
|
|
1166
|
+
}
|
|
1167
|
+
md_overrides.set(lens_name, {
|
|
1168
|
+
enabled: Boolean(md_enabled),
|
|
1169
|
+
auto_apply: Boolean(md_auto),
|
|
1170
|
+
});
|
|
1171
|
+
}
|
|
1172
|
+
const cd_block = lens_cfg.cost_disclosure;
|
|
1173
|
+
if (cd_block !== undefined && cd_block !== null) {
|
|
1174
|
+
cd_overrides.set(
|
|
1175
|
+
lens_name,
|
|
1176
|
+
_build_cost_disclosure(
|
|
1177
|
+
_asDict(cd_block),
|
|
1178
|
+
`lenses.${lens_name}.cost_disclosure`,
|
|
1179
|
+
),
|
|
1180
|
+
);
|
|
1181
|
+
}
|
|
1182
|
+
const dr_block = lens_cfg.decision_replay;
|
|
1183
|
+
if (dr_block !== undefined && dr_block !== null) {
|
|
1184
|
+
dr_overrides.set(
|
|
1185
|
+
lens_name,
|
|
1186
|
+
_build_decision_replay(
|
|
1187
|
+
_asDict(dr_block),
|
|
1188
|
+
`lenses.${lens_name}.decision_replay`,
|
|
1189
|
+
),
|
|
1190
|
+
);
|
|
1191
|
+
}
|
|
1192
|
+
}
|
|
1193
|
+
return {
|
|
1194
|
+
necessity_classifier_mode: nc_overrides,
|
|
1195
|
+
necessity_classifier_user_explicit_mode: nc_user_overrides,
|
|
1196
|
+
model_downgrade: md_overrides,
|
|
1197
|
+
cost_disclosure: cd_overrides,
|
|
1198
|
+
decision_replay: dr_overrides,
|
|
1199
|
+
};
|
|
1200
|
+
}
|
|
1201
|
+
|
|
1202
|
+
function _build_consensus_scoring(d: Dict): ConsensusScoringConfig {
|
|
1203
|
+
if (!_isDict(d)) {
|
|
1204
|
+
throw new CouncilConfigError('`consensus_scoring` must be a mapping.');
|
|
1205
|
+
}
|
|
1206
|
+
const strong = _pyFloat(_get(d, 'strong_threshold', 0.7));
|
|
1207
|
+
const minority = _pyFloat(_get(d, 'minority_threshold', 0.4));
|
|
1208
|
+
if (!(0.0 <= minority && minority <= strong && strong <= 1.0)) {
|
|
1209
|
+
throw new CouncilConfigError(
|
|
1210
|
+
`consensus_scoring thresholds broken: require ` +
|
|
1211
|
+
`0 <= minority (${_pyNum(minority)}) <= strong (${_pyNum(strong)}) <= 1.`,
|
|
1212
|
+
);
|
|
1213
|
+
}
|
|
1214
|
+
const lenses_raw = _get(d, 'lenses', ['analysis']);
|
|
1215
|
+
if (!_isList(lenses_raw) || !lenses_raw.every((x) => _isStr(x))) {
|
|
1216
|
+
throw new CouncilConfigError(
|
|
1217
|
+
'`consensus_scoring.lenses` must be a list of strings.',
|
|
1218
|
+
);
|
|
1219
|
+
}
|
|
1220
|
+
return {
|
|
1221
|
+
enabled: Boolean(_pyTruthy(_get(d, 'enabled', false))),
|
|
1222
|
+
strong_threshold: strong,
|
|
1223
|
+
minority_threshold: minority,
|
|
1224
|
+
lenses: lenses_raw as string[],
|
|
1225
|
+
};
|
|
1226
|
+
}
|
|
1227
|
+
|
|
1228
|
+
const _VALID_MEMBER_MODES: ReadonlySet<string> = new Set(['cli', 'api']);
|
|
1229
|
+
|
|
1230
|
+
function _build_defaults(d: Dict): DefaultsConfig {
|
|
1231
|
+
if (!_isDict(d)) {
|
|
1232
|
+
throw new CouncilConfigError('`defaults` must be a mapping.');
|
|
1233
|
+
}
|
|
1234
|
+
const mode = _get(d, 'mode', 'api');
|
|
1235
|
+
if (!(_isStr(mode) && _VALID_MODES.has(mode))) {
|
|
1236
|
+
throw new CouncilConfigError(
|
|
1237
|
+
`defaults.mode=${_pyRepr(mode)} not in ${_sortedListRepr(_VALID_MODES)}.`,
|
|
1238
|
+
);
|
|
1239
|
+
}
|
|
1240
|
+
// `member_mode` (step-9 P8 · U1) — global preference for solo /
|
|
1241
|
+
// CLI-mode invocations. Narrower set than `defaults.mode` because
|
|
1242
|
+
// `manual` makes no sense as a per-member dispatch default.
|
|
1243
|
+
const member_mode = _get(d, 'member_mode', 'cli');
|
|
1244
|
+
if (!(_isStr(member_mode) && _VALID_MEMBER_MODES.has(member_mode))) {
|
|
1245
|
+
throw new CouncilConfigError(
|
|
1246
|
+
`defaults.member_mode=${_pyRepr(member_mode)} not in ` +
|
|
1247
|
+
`${_sortedListRepr(_VALID_MEMBER_MODES)}.`,
|
|
1248
|
+
);
|
|
1249
|
+
}
|
|
1250
|
+
return {
|
|
1251
|
+
mode,
|
|
1252
|
+
member_mode,
|
|
1253
|
+
min_rounds: _pyInt(_get(d, 'min_rounds', 2)),
|
|
1254
|
+
deep_min_rounds: _pyInt(_get(d, 'deep_min_rounds', 3)),
|
|
1255
|
+
max_output_tokens: _pyInt(_get(d, 'max_output_tokens', 0)),
|
|
1256
|
+
session_retention_days: _pyInt(_get(d, 'session_retention_days', 7)),
|
|
1257
|
+
debate_max_rounds: _pyInt(_get(d, 'debate_max_rounds', 4)),
|
|
1258
|
+
};
|
|
1259
|
+
}
|
|
1260
|
+
|
|
1261
|
+
const _VALID_DISPATCH_MODES: ReadonlySet<string> = new Set(['full', 'single']);
|
|
1262
|
+
|
|
1263
|
+
function _build_routing(d: Dict, members: Map<string, MemberConfig>): RoutingConfig {
|
|
1264
|
+
if (!_isDict(d)) {
|
|
1265
|
+
throw new CouncilConfigError('`routing` must be a mapping.');
|
|
1266
|
+
}
|
|
1267
|
+
const chain_raw = _get(d, 'solo_member_fallback_chain', []);
|
|
1268
|
+
if (!_isList(chain_raw)) {
|
|
1269
|
+
throw new CouncilConfigError(
|
|
1270
|
+
'`routing.solo_member_fallback_chain` must be a list ' +
|
|
1271
|
+
`(got ${_pyTypeName(chain_raw)}).`,
|
|
1272
|
+
);
|
|
1273
|
+
}
|
|
1274
|
+
const chain: string[] = [];
|
|
1275
|
+
const seen = new Set<string>();
|
|
1276
|
+
for (let idx = 0; idx < chain_raw.length; idx++) {
|
|
1277
|
+
const entry = chain_raw[idx];
|
|
1278
|
+
if (!_isStr(entry) || entry.trim() === '') {
|
|
1279
|
+
throw new CouncilConfigError(
|
|
1280
|
+
`routing.solo_member_fallback_chain[${idx}]: each ` +
|
|
1281
|
+
`entry must be a non-empty string (got ${_pyRepr(entry)}).`,
|
|
1282
|
+
);
|
|
1283
|
+
}
|
|
1284
|
+
if (seen.has(entry)) {
|
|
1285
|
+
throw new CouncilConfigError(
|
|
1286
|
+
`routing.solo_member_fallback_chain[${idx}]: ` +
|
|
1287
|
+
`duplicate entry ${_pyReprStr(entry)} — chain order must be ` +
|
|
1288
|
+
`unique.`,
|
|
1289
|
+
);
|
|
1290
|
+
}
|
|
1291
|
+
if (!members.has(entry)) {
|
|
1292
|
+
throw new CouncilConfigError(
|
|
1293
|
+
`routing.solo_member_fallback_chain[${idx}]=${_pyReprStr(entry)}: ` +
|
|
1294
|
+
`no such member in the \`members\` block.`,
|
|
1295
|
+
);
|
|
1296
|
+
}
|
|
1297
|
+
seen.add(entry);
|
|
1298
|
+
chain.push(entry);
|
|
1299
|
+
}
|
|
1300
|
+
const timeout_raw = _get(d, 'auth_check_timeout_seconds', 3);
|
|
1301
|
+
if (
|
|
1302
|
+
!_isInt(timeout_raw) ||
|
|
1303
|
+
_isBool(timeout_raw) ||
|
|
1304
|
+
!(1 <= timeout_raw && timeout_raw <= 30)
|
|
1305
|
+
) {
|
|
1306
|
+
throw new CouncilConfigError(
|
|
1307
|
+
'routing.auth_check_timeout_seconds must be an int in ' +
|
|
1308
|
+
`[1, 30] (got ${_pyRepr(timeout_raw)}).`,
|
|
1309
|
+
);
|
|
1310
|
+
}
|
|
1311
|
+
return {
|
|
1312
|
+
solo_member_fallback_chain: chain,
|
|
1313
|
+
auth_check_timeout_seconds: timeout_raw,
|
|
1314
|
+
};
|
|
1315
|
+
}
|
|
1316
|
+
|
|
1317
|
+
function _build_low_impact(
|
|
1318
|
+
d: Dict,
|
|
1319
|
+
members: Map<string, MemberConfig>,
|
|
1320
|
+
routing: RoutingConfig,
|
|
1321
|
+
): LowImpactConfig {
|
|
1322
|
+
if (!_isDict(d)) {
|
|
1323
|
+
throw new CouncilConfigError('`low_impact` must be a mapping.');
|
|
1324
|
+
}
|
|
1325
|
+
const dispatch = _get(d, 'dispatch', 'full');
|
|
1326
|
+
if (!(_isStr(dispatch) && _VALID_DISPATCH_MODES.has(dispatch))) {
|
|
1327
|
+
throw new CouncilConfigError(
|
|
1328
|
+
`low_impact.dispatch=${_pyRepr(dispatch)} not in ` +
|
|
1329
|
+
`${_sortedListRepr(_VALID_DISPATCH_MODES)}.`,
|
|
1330
|
+
);
|
|
1331
|
+
}
|
|
1332
|
+
if (dispatch === 'single') {
|
|
1333
|
+
const enabled_in_chain = routing.solo_member_fallback_chain.filter(
|
|
1334
|
+
(name) => members.get(name) !== undefined && members.get(name)!.enabled,
|
|
1335
|
+
);
|
|
1336
|
+
if (enabled_in_chain.length === 0) {
|
|
1337
|
+
throw new CouncilConfigError(
|
|
1338
|
+
"low_impact.dispatch='single' requires at least one " +
|
|
1339
|
+
'enabled member in routing.solo_member_fallback_chain ' +
|
|
1340
|
+
`(chain=${_pyRepr([...routing.solo_member_fallback_chain])}). ` +
|
|
1341
|
+
"Enable a chain member or set dispatch back to 'full'.",
|
|
1342
|
+
);
|
|
1343
|
+
}
|
|
1344
|
+
}
|
|
1345
|
+
const shadow_raw = _get(d, 'shadow_sample_rate', 0.1);
|
|
1346
|
+
if (_isBool(shadow_raw) || !_isNumber(shadow_raw)) {
|
|
1347
|
+
throw new CouncilConfigError(
|
|
1348
|
+
'low_impact.shadow_sample_rate must be a number ' +
|
|
1349
|
+
`(got ${_pyTypeName(shadow_raw)}).`,
|
|
1350
|
+
);
|
|
1351
|
+
}
|
|
1352
|
+
const shadow = _pyFloat(shadow_raw);
|
|
1353
|
+
if (!(0.0 <= shadow && shadow <= 1.0)) {
|
|
1354
|
+
throw new CouncilConfigError(
|
|
1355
|
+
'low_impact.shadow_sample_rate must be in [0.0, 1.0] ' +
|
|
1356
|
+
`(got ${_pyRepr(_f(shadow))}).`,
|
|
1357
|
+
);
|
|
1358
|
+
}
|
|
1359
|
+
const floor_raw = _get(d, 'solo_confidence_floor', 0.7);
|
|
1360
|
+
if (_isBool(floor_raw) || !_isNumber(floor_raw)) {
|
|
1361
|
+
throw new CouncilConfigError(
|
|
1362
|
+
'low_impact.solo_confidence_floor must be a number ' +
|
|
1363
|
+
`(got ${_pyTypeName(floor_raw)}).`,
|
|
1364
|
+
);
|
|
1365
|
+
}
|
|
1366
|
+
const floor = _pyFloat(floor_raw);
|
|
1367
|
+
if (!(0.0 <= floor && floor <= 1.0)) {
|
|
1368
|
+
throw new CouncilConfigError(
|
|
1369
|
+
'low_impact.solo_confidence_floor must be in [0.0, 1.0] ' +
|
|
1370
|
+
`(got ${_pyRepr(_f(floor))}).`,
|
|
1371
|
+
);
|
|
1372
|
+
}
|
|
1373
|
+
return {
|
|
1374
|
+
dispatch,
|
|
1375
|
+
shadow_sample_rate: shadow,
|
|
1376
|
+
solo_confidence_floor: floor,
|
|
1377
|
+
};
|
|
1378
|
+
}
|
|
1379
|
+
|
|
1380
|
+
function _reject_top_level_locked_dispatch(raw: Dict): void {
|
|
1381
|
+
for (const cls of _LOCKED_IMPACT_CLASSES) {
|
|
1382
|
+
const block = raw[cls];
|
|
1383
|
+
if (!_isDict(block)) {
|
|
1384
|
+
continue;
|
|
1385
|
+
}
|
|
1386
|
+
if ('dispatch' in block) {
|
|
1387
|
+
throw new CouncilConfigError(
|
|
1388
|
+
`${cls}.dispatch=${_pyRepr(block.dispatch)}: dispatch is ` +
|
|
1389
|
+
`not configurable for high-impact / user-required ` +
|
|
1390
|
+
`decisions — always full council.`,
|
|
1391
|
+
);
|
|
1392
|
+
}
|
|
1393
|
+
if ('solo_confidence_floor' in block) {
|
|
1394
|
+
throw new CouncilConfigError(
|
|
1395
|
+
`${cls}.solo_confidence_floor=` +
|
|
1396
|
+
`${_pyRepr(block.solo_confidence_floor)}: irrelevant on ` +
|
|
1397
|
+
`high-impact / user-required classes — they never ` +
|
|
1398
|
+
`dispatch solo. Set on \`low_impact\` instead.`,
|
|
1399
|
+
);
|
|
1400
|
+
}
|
|
1401
|
+
}
|
|
1402
|
+
}
|
|
1403
|
+
|
|
1404
|
+
function _build_cost_budget(d: Dict): CostBudgetConfig {
|
|
1405
|
+
if (!_isDict(d)) {
|
|
1406
|
+
throw new CouncilConfigError('`cost_budget` must be a mapping.');
|
|
1407
|
+
}
|
|
1408
|
+
const cb: CostBudgetConfig = {
|
|
1409
|
+
max_input_tokens: _pyInt(_get(d, 'max_input_tokens', 500_000)),
|
|
1410
|
+
max_output_tokens: _pyInt(_get(d, 'max_output_tokens', 200_000)),
|
|
1411
|
+
max_calls: _pyInt(_get(d, 'max_calls', 50)),
|
|
1412
|
+
max_total_usd: _pyFloat(_get(d, 'max_total_usd', 20.0)),
|
|
1413
|
+
};
|
|
1414
|
+
const fields: Array<[keyof CostBudgetConfig, boolean]> = [
|
|
1415
|
+
['max_input_tokens', false],
|
|
1416
|
+
['max_output_tokens', false],
|
|
1417
|
+
['max_calls', false],
|
|
1418
|
+
['max_total_usd', true],
|
|
1419
|
+
];
|
|
1420
|
+
for (const [fname, isFloat] of fields) {
|
|
1421
|
+
const val = cb[fname];
|
|
1422
|
+
if (val < 0) {
|
|
1423
|
+
throw new CouncilConfigError(
|
|
1424
|
+
`cost_budget.${fname} must be >= 0 (got ` +
|
|
1425
|
+
`${isFloat ? _pyRepr(_f(val)) : _pyRepr(val)}).`,
|
|
1426
|
+
);
|
|
1427
|
+
}
|
|
1428
|
+
}
|
|
1429
|
+
return cb;
|
|
1430
|
+
}
|
|
1431
|
+
|
|
1432
|
+
function _build_member(
|
|
1433
|
+
name: string,
|
|
1434
|
+
cfg: Dict,
|
|
1435
|
+
default_mode = 'api',
|
|
1436
|
+
): MemberConfig {
|
|
1437
|
+
if (!_VALID_PROVIDERS.has(name)) {
|
|
1438
|
+
throw new CouncilConfigError(
|
|
1439
|
+
`members.${name}: unknown provider; valid: ` +
|
|
1440
|
+
`${_sortedListRepr(_VALID_PROVIDERS)}.`,
|
|
1441
|
+
);
|
|
1442
|
+
}
|
|
1443
|
+
const member_enabled = Boolean(_pyTruthy(_get(cfg, 'enabled', false)));
|
|
1444
|
+
const modelVal = _get(cfg, 'model', null);
|
|
1445
|
+
const model = _pyTruthy(modelVal) ? (modelVal as string) : '';
|
|
1446
|
+
const api_key_ref_raw = _get(cfg, 'api_key_ref', null);
|
|
1447
|
+
const api_key_ref = api_key_ref_raw === undefined ? null : api_key_ref_raw;
|
|
1448
|
+
const member_mode_raw = _get(cfg, 'mode', null);
|
|
1449
|
+
const member_mode = member_mode_raw === undefined ? null : member_mode_raw;
|
|
1450
|
+
if (member_mode !== null && !(_isStr(member_mode) && _VALID_MODES.has(member_mode))) {
|
|
1451
|
+
throw new CouncilConfigError(
|
|
1452
|
+
`members.${name}.mode=${_pyRepr(member_mode)} not in ` +
|
|
1453
|
+
`${_sortedListRepr(_VALID_MODES)}.`,
|
|
1454
|
+
);
|
|
1455
|
+
}
|
|
1456
|
+
const effective_mode = member_mode !== null ? (member_mode as string) : default_mode;
|
|
1457
|
+
const binary_raw = _get(cfg, 'binary', null);
|
|
1458
|
+
const binary = binary_raw === undefined ? null : binary_raw;
|
|
1459
|
+
if (binary !== null) {
|
|
1460
|
+
if (!_isStr(binary) || binary.trim() === '') {
|
|
1461
|
+
throw new CouncilConfigError(
|
|
1462
|
+
`members.${name}.binary must be a non-empty string when set.`,
|
|
1463
|
+
);
|
|
1464
|
+
}
|
|
1465
|
+
if (effective_mode !== 'cli') {
|
|
1466
|
+
throw new CouncilConfigError(
|
|
1467
|
+
`members.${name}.binary is only valid when the member's ` +
|
|
1468
|
+
`effective mode is 'cli' (got ${_pyRepr(effective_mode)}). Set ` +
|
|
1469
|
+
`\`mode: cli\` on the member or \`defaults.mode: cli\` to use ` +
|
|
1470
|
+
`this field.`,
|
|
1471
|
+
);
|
|
1472
|
+
}
|
|
1473
|
+
}
|
|
1474
|
+
if (member_enabled) {
|
|
1475
|
+
if (!model) {
|
|
1476
|
+
throw new CouncilConfigError(
|
|
1477
|
+
`members.${name}: enabled members require a non-empty \`model\`.`,
|
|
1478
|
+
);
|
|
1479
|
+
}
|
|
1480
|
+
// CLI-mode members authenticate via the subscription bound to
|
|
1481
|
+
// the local CLI binary; api_key_ref is not required for them.
|
|
1482
|
+
// Manual mode is human-transported and also key-free. Only
|
|
1483
|
+
// api-mode members must supply an api_key_ref.
|
|
1484
|
+
if (effective_mode === 'api' && !_pyTruthy(api_key_ref)) {
|
|
1485
|
+
throw new CouncilConfigError(
|
|
1486
|
+
`members.${name}: enabled api-mode members require an \`api_key_ref\`.`,
|
|
1487
|
+
);
|
|
1488
|
+
}
|
|
1489
|
+
}
|
|
1490
|
+
if (api_key_ref !== null) {
|
|
1491
|
+
_validate_api_key_ref(`members.${name}`, api_key_ref);
|
|
1492
|
+
}
|
|
1493
|
+
const ladder_raw = _getOr(cfg, 'model_ladder', []);
|
|
1494
|
+
if (!_isList(ladder_raw)) {
|
|
1495
|
+
throw new CouncilConfigError(
|
|
1496
|
+
`members.${name}.model_ladder must be a list (got ` +
|
|
1497
|
+
`${_pyTypeName(ladder_raw)}).`,
|
|
1498
|
+
);
|
|
1499
|
+
}
|
|
1500
|
+
let ladder: string[] = [];
|
|
1501
|
+
if (ladder_raw.length > 0) {
|
|
1502
|
+
const entries: string[] = [];
|
|
1503
|
+
for (const entry of ladder_raw) {
|
|
1504
|
+
if (!_isStr(entry) || entry.trim() === '') {
|
|
1505
|
+
throw new CouncilConfigError(
|
|
1506
|
+
`members.${name}.model_ladder entries must be non-empty ` +
|
|
1507
|
+
`strings (got ${_pyRepr(entry)}).`,
|
|
1508
|
+
);
|
|
1509
|
+
}
|
|
1510
|
+
entries.push(entry);
|
|
1511
|
+
}
|
|
1512
|
+
if (member_enabled && model && !entries.includes(model)) {
|
|
1513
|
+
throw new CouncilConfigError(
|
|
1514
|
+
`members.${name}.model_ladder must include the active ` +
|
|
1515
|
+
`\`model\` (${_pyReprStr(model)}); got ${_pyRepr(entries)}.`,
|
|
1516
|
+
);
|
|
1517
|
+
}
|
|
1518
|
+
ladder = entries;
|
|
1519
|
+
}
|
|
1520
|
+
const participate_raw = _get(cfg, 'participate_low_impact', false);
|
|
1521
|
+
if (!_isBool(participate_raw)) {
|
|
1522
|
+
throw new CouncilConfigError(
|
|
1523
|
+
`members.${name}.participate_low_impact must be a bool ` +
|
|
1524
|
+
`(got ${_pyTypeName(participate_raw)}).`,
|
|
1525
|
+
);
|
|
1526
|
+
}
|
|
1527
|
+
return {
|
|
1528
|
+
name,
|
|
1529
|
+
enabled: member_enabled,
|
|
1530
|
+
model,
|
|
1531
|
+
api_key_ref: (api_key_ref as string | null) ?? null,
|
|
1532
|
+
mode: (member_mode as string | null) ?? null,
|
|
1533
|
+
binary: (binary as string | null) ?? null,
|
|
1534
|
+
model_ladder: ladder,
|
|
1535
|
+
participate_low_impact: participate_raw,
|
|
1536
|
+
};
|
|
1537
|
+
}
|
|
1538
|
+
|
|
1539
|
+
function _build_cli_call_budget(d: Dict): CliCallBudgetConfig {
|
|
1540
|
+
if (!_isDict(d)) {
|
|
1541
|
+
throw new CouncilConfigError('`cli_call_budget` must be a mapping.');
|
|
1542
|
+
}
|
|
1543
|
+
const raw_caps = _getOr(d, 'max_calls_per_day', {});
|
|
1544
|
+
if (!_isDict(raw_caps)) {
|
|
1545
|
+
throw new CouncilConfigError(
|
|
1546
|
+
'`cli_call_budget.max_calls_per_day` must be a mapping.',
|
|
1547
|
+
);
|
|
1548
|
+
}
|
|
1549
|
+
const caps = new Map<string, number>();
|
|
1550
|
+
for (const [provider, value] of Object.entries(raw_caps)) {
|
|
1551
|
+
if (!_VALID_PROVIDERS.has(provider)) {
|
|
1552
|
+
throw new CouncilConfigError(
|
|
1553
|
+
`cli_call_budget.max_calls_per_day.${provider}: unknown ` +
|
|
1554
|
+
`provider; valid: ${_sortedListRepr(_VALID_PROVIDERS)}.`,
|
|
1555
|
+
);
|
|
1556
|
+
}
|
|
1557
|
+
if (!_isInt(value) || _isBool(value) || value < 0) {
|
|
1558
|
+
throw new CouncilConfigError(
|
|
1559
|
+
`cli_call_budget.max_calls_per_day.${provider} must be a ` +
|
|
1560
|
+
`non-negative integer (got ${_pyRepr(value)}).`,
|
|
1561
|
+
);
|
|
1562
|
+
}
|
|
1563
|
+
caps.set(provider, value);
|
|
1564
|
+
}
|
|
1565
|
+
const warn_at_raw = _get(d, 'warn_at', 0.8);
|
|
1566
|
+
if (_isBool(warn_at_raw) || !_isNumber(warn_at_raw)) {
|
|
1567
|
+
throw new CouncilConfigError(
|
|
1568
|
+
`cli_call_budget.warn_at must be a number in [0.0, 1.0] ` +
|
|
1569
|
+
`(got ${_pyRepr(warn_at_raw)}).`,
|
|
1570
|
+
);
|
|
1571
|
+
}
|
|
1572
|
+
const warn_at = _pyFloat(warn_at_raw);
|
|
1573
|
+
if (!(0.0 <= warn_at && warn_at <= 1.0)) {
|
|
1574
|
+
throw new CouncilConfigError(
|
|
1575
|
+
`cli_call_budget.warn_at must be in [0.0, 1.0] (got ${_pyNum(warn_at)}).`,
|
|
1576
|
+
);
|
|
1577
|
+
}
|
|
1578
|
+
return { max_calls_per_day: caps, warn_at };
|
|
1579
|
+
}
|
|
1580
|
+
|
|
1581
|
+
function _build_advisor(name: string, cfg: Dict): AdvisorConfig {
|
|
1582
|
+
if (!_isDict(cfg)) {
|
|
1583
|
+
throw new CouncilConfigError(`advisors.${name}: must be a mapping.`);
|
|
1584
|
+
}
|
|
1585
|
+
const member = _get(cfg, 'member', null);
|
|
1586
|
+
if (!(_isStr(member) && _VALID_PROVIDERS.has(member))) {
|
|
1587
|
+
throw new CouncilConfigError(
|
|
1588
|
+
`advisors.${name}.member=${_pyRepr(member)} not a valid provider; ` +
|
|
1589
|
+
`valid: ${_sortedListRepr(_VALID_PROVIDERS)}.`,
|
|
1590
|
+
);
|
|
1591
|
+
}
|
|
1592
|
+
// `persona` may be set explicitly; otherwise default to the
|
|
1593
|
+
// convention path so the YAML stays terse.
|
|
1594
|
+
const personaVal = _get(cfg, 'persona', null);
|
|
1595
|
+
const persona = _pyTruthy(personaVal)
|
|
1596
|
+
? (personaVal as string)
|
|
1597
|
+
: `personas/advisors/${name}.md`;
|
|
1598
|
+
const modelVal = _get(cfg, 'model', null);
|
|
1599
|
+
const model = modelVal === undefined ? null : modelVal;
|
|
1600
|
+
if (model !== null && !_isStr(model)) {
|
|
1601
|
+
throw new CouncilConfigError(
|
|
1602
|
+
`advisors.${name}.model must be a string when set.`,
|
|
1603
|
+
);
|
|
1604
|
+
}
|
|
1605
|
+
return {
|
|
1606
|
+
name,
|
|
1607
|
+
enabled: Boolean(_pyTruthy(_get(cfg, 'enabled', false))),
|
|
1608
|
+
member,
|
|
1609
|
+
persona,
|
|
1610
|
+
model: (model as string | null) ?? null,
|
|
1611
|
+
};
|
|
1612
|
+
}
|
|
1613
|
+
|
|
1614
|
+
function _validate_api_key_ref(scope: string, ref: Json): void {
|
|
1615
|
+
if (!_isStr(ref) || ref === '') {
|
|
1616
|
+
throw new CouncilConfigError(`${scope}.api_key_ref must be a non-empty string.`);
|
|
1617
|
+
}
|
|
1618
|
+
if (_RAW_KEY_PREFIXES.some((prefix) => ref.startsWith(prefix))) {
|
|
1619
|
+
throw new CouncilConfigError(
|
|
1620
|
+
`${scope}.api_key_ref looks like a raw API key. ` +
|
|
1621
|
+
`Use \`file:<path>\` (0600) or \`env:<VAR>\` — never inline secrets.`,
|
|
1622
|
+
);
|
|
1623
|
+
}
|
|
1624
|
+
if (ref.startsWith('file:')) {
|
|
1625
|
+
if (ref.slice('file:'.length).trim() === '') {
|
|
1626
|
+
throw new CouncilConfigError(`${scope}.api_key_ref \`file:\` ref missing path.`);
|
|
1627
|
+
}
|
|
1628
|
+
return;
|
|
1629
|
+
}
|
|
1630
|
+
if (ref.startsWith('env:')) {
|
|
1631
|
+
if (ref.slice('env:'.length).trim() === '') {
|
|
1632
|
+
throw new CouncilConfigError(
|
|
1633
|
+
`${scope}.api_key_ref \`env:\` ref missing variable name.`,
|
|
1634
|
+
);
|
|
1635
|
+
}
|
|
1636
|
+
return;
|
|
1637
|
+
}
|
|
1638
|
+
throw new CouncilConfigError(
|
|
1639
|
+
`${scope}.api_key_ref must start with \`file:\` or \`env:\` (got ${_pyRepr(ref)}).`,
|
|
1640
|
+
);
|
|
1641
|
+
}
|
|
1642
|
+
|
|
1643
|
+
/**
|
|
1644
|
+
* Resolve `file:<path>` or `env:<VAR>` to the raw key string.
|
|
1645
|
+
*
|
|
1646
|
+
* `file:` — relative paths resolve under the user-global namespace
|
|
1647
|
+
* (`~/.event4u/agent-config/` today, with the pre-2.4
|
|
1648
|
+
* `~/.config/agent-config/` tree read as a fallback). Mode must be
|
|
1649
|
+
* 0o600. `env:` — reads from the environment; empty/missing is a hard
|
|
1650
|
+
* error. Never echoes the value.
|
|
1651
|
+
*/
|
|
1652
|
+
export function resolve_api_key(ref: string, scope = 'api_key_ref'): string {
|
|
1653
|
+
_validate_api_key_ref(scope, ref);
|
|
1654
|
+
if (ref.startsWith('env:')) {
|
|
1655
|
+
const variable = ref.slice('env:'.length).trim();
|
|
1656
|
+
if (variable === '') {
|
|
1657
|
+
throw new CouncilConfigError(`${scope}: \`env:\` ref missing variable name.`);
|
|
1658
|
+
}
|
|
1659
|
+
const value = (process.env[variable] ?? '').trim();
|
|
1660
|
+
if (value === '') {
|
|
1661
|
+
throw new CouncilConfigError(
|
|
1662
|
+
`${scope}: env var ${_pyReprStr(variable)} is unset or empty.`,
|
|
1663
|
+
);
|
|
1664
|
+
}
|
|
1665
|
+
return value;
|
|
1666
|
+
}
|
|
1667
|
+
const spec = ref.slice('file:'.length).trim();
|
|
1668
|
+
if (spec === '') {
|
|
1669
|
+
throw new CouncilConfigError(`${scope}: \`file:\` ref missing path.`);
|
|
1670
|
+
}
|
|
1671
|
+
let resolved = _expanduser(spec);
|
|
1672
|
+
if (!_isAbsoluteLikePython(resolved)) {
|
|
1673
|
+
const found = user_global_paths.resolve_with_fallback(spec);
|
|
1674
|
+
if (found === null) {
|
|
1675
|
+
const target = user_global_paths.write_target(spec);
|
|
1676
|
+
throw new CouncilConfigError(
|
|
1677
|
+
`${scope}: key file not found at ${target} (or legacy fallback).`,
|
|
1678
|
+
);
|
|
1679
|
+
}
|
|
1680
|
+
resolved = found;
|
|
1681
|
+
}
|
|
1682
|
+
if (!fs.existsSync(resolved)) {
|
|
1683
|
+
throw new CouncilConfigError(`${scope}: key file does not exist at ${resolved}.`);
|
|
1684
|
+
}
|
|
1685
|
+
const mode = fs.statSync(resolved).mode & 0o7777;
|
|
1686
|
+
if (mode !== 0o600) {
|
|
1687
|
+
throw new CouncilConfigError(
|
|
1688
|
+
`${scope}: unsafe permissions on ${resolved}: got ${_pyOct(mode)}, ` +
|
|
1689
|
+
`expected 0o600. Fix: chmod 600 ${resolved}`,
|
|
1690
|
+
);
|
|
1691
|
+
}
|
|
1692
|
+
const value = fs.readFileSync(resolved, 'utf-8').trim();
|
|
1693
|
+
if (value === '') {
|
|
1694
|
+
throw new CouncilConfigError(`${scope}: key file at ${resolved} is empty.`);
|
|
1695
|
+
}
|
|
1696
|
+
return value;
|
|
1697
|
+
}
|
|
1698
|
+
|
|
1699
|
+
// ── small numeric helpers for f-string `{x}` (non-repr) interpolation ─
|
|
1700
|
+
|
|
1701
|
+
/**
|
|
1702
|
+
* Mirror Python `str(number)` inside an f-string `{x}` (no `!r`). For
|
|
1703
|
+
* the consensus / warn_at messages Python interpolates the FLOAT value
|
|
1704
|
+
* via `{minority}` / `{warn_at}` — these came through `float(...)` so a
|
|
1705
|
+
* whole value renders `1.0`, not `1`.
|
|
1706
|
+
*/
|
|
1707
|
+
function _pyNum(value: number): string {
|
|
1708
|
+
return _pyReprFloat(value);
|
|
1709
|
+
}
|
|
1710
|
+
|
|
1711
|
+
/** Mirror Python truthiness for `bool(x)` / `x or ...` scalar tests. */
|
|
1712
|
+
function _pyTruthy(value: Json): boolean {
|
|
1713
|
+
if (value === null || value === undefined) {
|
|
1714
|
+
return false;
|
|
1715
|
+
}
|
|
1716
|
+
if (typeof value === 'boolean') {
|
|
1717
|
+
return value;
|
|
1718
|
+
}
|
|
1719
|
+
if (typeof value === 'number') {
|
|
1720
|
+
return value !== 0;
|
|
1721
|
+
}
|
|
1722
|
+
if (typeof value === 'string') {
|
|
1723
|
+
return value !== '';
|
|
1724
|
+
}
|
|
1725
|
+
if (Array.isArray(value)) {
|
|
1726
|
+
return value.length > 0;
|
|
1727
|
+
}
|
|
1728
|
+
return Object.keys(value).length > 0;
|
|
1729
|
+
}
|