@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.
Files changed (1537) hide show
  1. package/.claude-plugin/marketplace.json +35 -3
  2. package/AGENTS.md +8 -7
  3. package/CHANGELOG.md +408 -0
  4. package/CONTRIBUTING.md +1 -1
  5. package/README.md +17 -15
  6. package/dist/agent-src/commands/agent-status.md +2 -2
  7. package/dist/agent-src/commands/agents/audit.md +3 -3
  8. package/dist/agent-src/commands/agents/init.md +1 -1
  9. package/dist/agent-src/commands/agents/optimize.md +4 -4
  10. package/dist/agent-src/commands/analyze/decision.md +108 -0
  11. package/dist/agent-src/commands/analyze/incident.md +120 -0
  12. package/dist/agent-src/commands/analyze/near-miss.md +113 -0
  13. package/dist/agent-src/commands/analyze/postmortem.md +130 -0
  14. package/dist/agent-src/commands/analyze/premortem.md +104 -0
  15. package/dist/agent-src/commands/analyze.md +124 -0
  16. package/dist/agent-src/commands/brand/identity.md +27 -0
  17. package/dist/agent-src/commands/brand/review.md +27 -0
  18. package/dist/agent-src/commands/brand/strategy.md +27 -0
  19. package/dist/agent-src/commands/brand/tokens.md +28 -0
  20. package/dist/agent-src/commands/brand/voice.md +27 -0
  21. package/dist/agent-src/commands/brand.md +58 -0
  22. package/dist/agent-src/commands/check-current-md.md +3 -3
  23. package/dist/agent-src/commands/condense.md +2 -2
  24. package/dist/agent-src/commands/council/debate.md +2 -2
  25. package/dist/agent-src/commands/council/default.md +45 -18
  26. package/dist/agent-src/commands/fix/portability.md +3 -3
  27. package/dist/agent-src/commands/fix/refs.md +3 -3
  28. package/dist/agent-src/commands/implement-ticket.md +36 -6
  29. package/dist/agent-src/commands/knowledge/cross-repo.md +1 -1
  30. package/dist/agent-src/commands/memory/add.md +1 -1
  31. package/dist/agent-src/commands/mission/upgrade.md +182 -0
  32. package/dist/agent-src/commands/optimize/skills.md +2 -2
  33. package/dist/agent-src/commands/orchestrate.md +1 -1
  34. package/dist/agent-src/commands/pr/create.md +6 -4
  35. package/dist/agent-src/commands/review-changes.md +8 -0
  36. package/dist/agent-src/commands/roadmap/materialize.md +73 -0
  37. package/dist/agent-src/commands/skill/preview.md +1 -1
  38. package/dist/agent-src/commands/skills/discover.md +1 -1
  39. package/dist/agent-src/commands/threat-model.md +4 -4
  40. package/dist/agent-src/commands/upstream-contribute.md +3 -3
  41. package/dist/agent-src/commands/video/from-script.md +2 -2
  42. package/dist/agent-src/commands/video/from-song.md +3 -3
  43. package/dist/agent-src/commands/video/scene.md +1 -1
  44. package/dist/agent-src/commands/video/storyboard.md +1 -1
  45. package/dist/agent-src/commands/video.md +3 -3
  46. package/dist/agent-src/contexts/communication/rules-auto/source-of-truth-mechanics.md +3 -3
  47. package/dist/agent-src/contexts/communication/rules-auto/user-interaction-mechanics.md +1 -1
  48. package/dist/agent-src/contexts/execution/evidence-discipline.md +153 -0
  49. package/dist/agent-src/contexts/execution/project-intelligence.md +264 -0
  50. package/dist/agent-src/contexts/execution/roadmap-process-loop.md +2 -1
  51. package/dist/agent-src/personas/ai-video-technical-director.md +1 -1
  52. package/dist/agent-src/personas/brand-strategist.md +74 -0
  53. package/dist/agent-src/personas/design-director.md +74 -0
  54. package/dist/agent-src/rules/brand-consistency.md +77 -0
  55. package/dist/agent-src/rules/brand-source-of-truth.md +57 -0
  56. package/dist/agent-src/rules/direct-answers.md +2 -0
  57. package/dist/agent-src/rules/domain-safety-disclaimer.md +2 -0
  58. package/dist/agent-src/rules/git-history-discipline.md +1 -0
  59. package/dist/agent-src/rules/icon-consistency.md +53 -0
  60. package/dist/agent-src/rules/image-likeness-and-rights.md +67 -0
  61. package/dist/agent-src/rules/lethal-trifecta-guard.md +1 -1
  62. package/dist/agent-src/rules/persona-governance.md +2 -2
  63. package/dist/agent-src/rules/provider-lifecycle-discipline.md +3 -1
  64. package/dist/agent-src/rules/roadmap-progress-sync.md +10 -0
  65. package/dist/agent-src/rules/security-sensitive-stop.md +9 -3
  66. package/dist/agent-src/rules/size-enforcement.md +1 -1
  67. package/dist/agent-src/rules/source-confidentiality.md +3 -3
  68. package/dist/agent-src/rules/source-discovery-gate.md +98 -0
  69. package/dist/agent-src/rules/think-before-action.md +1 -0
  70. package/dist/agent-src/rules/ui-audit-gate.md +2 -0
  71. package/dist/agent-src/rules/untrusted-input-defense.md +1 -1
  72. package/dist/agent-src/rules/user-interaction.md +1 -1
  73. package/dist/agent-src/scripts/archive_completed_roadmaps.ts +392 -0
  74. package/dist/agent-src/scripts/update_roadmap_progress.ts +824 -0
  75. package/dist/agent-src/skills/adr-create/SKILL.md +5 -5
  76. package/dist/agent-src/skills/agent-security-review/evals/triggers.json +1 -0
  77. package/dist/agent-src/skills/agents-md-thin-root/SKILL.md +1 -1
  78. package/dist/agent-src/skills/ai-council/SKILL.md +1 -1
  79. package/dist/agent-src/skills/analysis-autonomous-mode/SKILL.md +9 -13
  80. package/dist/agent-src/skills/blade-ui/SKILL.md +12 -5
  81. package/dist/agent-src/skills/blameless-post-mortem/SKILL.md +199 -0
  82. package/dist/agent-src/skills/brand/ATTRIBUTION.md +38 -0
  83. package/dist/agent-src/skills/brand/SKILL.md +115 -0
  84. package/dist/agent-src/skills/brand/data/archetypes.csv +13 -0
  85. package/dist/agent-src/skills/brand/data/color-psychology.csv +14 -0
  86. package/dist/agent-src/skills/brand/data/logo-style-fit.csv +13 -0
  87. package/dist/agent-src/skills/brand/data/manifest.json +226 -0
  88. package/dist/agent-src/skills/brand/data/messaging-frameworks.csv +13 -0
  89. package/dist/agent-src/skills/brand/data/naming-patterns.csv +13 -0
  90. package/dist/agent-src/skills/brand/data/typography-principles.csv +13 -0
  91. package/dist/agent-src/skills/brand/data/voice-tone.csv +13 -0
  92. package/dist/agent-src/skills/brand/evals/triggers.json +17 -0
  93. package/dist/agent-src/skills/brand-asset-generation/SKILL.md +89 -0
  94. package/dist/agent-src/skills/brand-asset-generation/evals/triggers.json +17 -0
  95. package/dist/agent-src/skills/brand-audit/SKILL.md +67 -0
  96. package/dist/agent-src/skills/brand-audit/evals/triggers.json +17 -0
  97. package/dist/agent-src/skills/brand-identity/SKILL.md +101 -0
  98. package/dist/agent-src/skills/brand-identity/evals/triggers.json +17 -0
  99. package/dist/agent-src/skills/brand-strategy/SKILL.md +83 -0
  100. package/dist/agent-src/skills/brand-strategy/evals/triggers.json +17 -0
  101. package/dist/agent-src/skills/brand-to-tokens/SKILL.md +102 -0
  102. package/dist/agent-src/skills/brand-to-tokens/evals/triggers.json +17 -0
  103. package/dist/agent-src/skills/brand-to-tokens/templates/marp-brand-deck.md.example +46 -0
  104. package/dist/agent-src/skills/brand-to-tokens/templates/reveal-brand-deck.yaml +32 -0
  105. package/dist/agent-src/skills/canvas-design/evals/triggers.json +1 -0
  106. package/dist/agent-src/skills/check-refs/SKILL.md +5 -5
  107. package/dist/agent-src/skills/code-review/SKILL.md +6 -15
  108. package/dist/agent-src/skills/command-writing/SKILL.md +2 -2
  109. package/dist/agent-src/skills/complexity-first-planning/evals/triggers.json +1 -0
  110. package/dist/agent-src/skills/context-authoring/SKILL.md +2 -2
  111. package/dist/agent-src/skills/context-document/SKILL.md +35 -2
  112. package/dist/agent-src/skills/corpus-grounding/evals/triggers.json +1 -0
  113. package/dist/agent-src/skills/corpus-grounding/scripts/bm25_search.ts +482 -0
  114. package/dist/agent-src/skills/corpus-grounding/scripts/decision_engine.ts +803 -0
  115. package/dist/agent-src/skills/corpus-grounding/scripts/ground.ts +541 -0
  116. package/dist/agent-src/skills/corpus-grounding/scripts/schema_validator.ts +309 -0
  117. package/dist/agent-src/skills/database/SKILL.md +26 -4
  118. package/dist/agent-src/skills/decision-record/SKILL.md +1 -1
  119. package/dist/agent-src/skills/decision-record/evals/triggers.json +17 -0
  120. package/dist/agent-src/skills/decision-review/SKILL.md +179 -0
  121. package/dist/agent-src/skills/description-assist/SKILL.md +1 -1
  122. package/dist/agent-src/skills/design-intelligence/SKILL.md +1 -1
  123. package/dist/agent-src/skills/design-intelligence/data/manifest.json +23 -6
  124. package/dist/agent-src/skills/design-intelligence/evals/triggers.json +1 -0
  125. package/dist/agent-src/skills/design-tokens/evals/triggers.json +1 -0
  126. package/dist/agent-src/skills/design-tokens/scripts/tokens.ts +888 -0
  127. package/dist/agent-src/skills/doc-coauthoring/evals/triggers.json +1 -0
  128. package/dist/agent-src/skills/eloquent/evals/triggers.json +1 -0
  129. package/dist/agent-src/skills/emit-tickets/SKILL.md +198 -0
  130. package/dist/agent-src/skills/estimate-ticket/evals/triggers.json +1 -0
  131. package/dist/agent-src/skills/git-workflow/SKILL.md +33 -0
  132. package/dist/agent-src/skills/guideline-writing/SKILL.md +2 -2
  133. package/dist/agent-src/skills/iconography/SKILL.md +88 -0
  134. package/dist/agent-src/skills/iconography/evals/triggers.json +17 -0
  135. package/dist/agent-src/skills/image-analyser/evals/triggers.json +1 -0
  136. package/dist/agent-src/skills/image-creator/evals/triggers.json +1 -0
  137. package/dist/agent-src/skills/image-editing/SKILL.md +100 -0
  138. package/dist/agent-src/skills/image-editing/evals/triggers.json +17 -0
  139. package/dist/agent-src/skills/image-generation/SKILL.md +95 -0
  140. package/dist/agent-src/skills/image-generation/evals/triggers.json +17 -0
  141. package/dist/agent-src/skills/image-provider-routing/SKILL.md +96 -0
  142. package/dist/agent-src/skills/image-provider-routing/evals/triggers.json +17 -0
  143. package/dist/agent-src/skills/launch-readiness/SKILL.md +21 -0
  144. package/dist/agent-src/skills/learning-to-rule-or-skill/SKILL.md +12 -8
  145. package/dist/agent-src/skills/lint-skills/SKILL.md +5 -5
  146. package/dist/agent-src/skills/logo-generation/SKILL.md +98 -0
  147. package/dist/agent-src/skills/logo-generation/evals/triggers.json +17 -0
  148. package/dist/agent-src/skills/markitdown/SKILL.md +1 -1
  149. package/dist/agent-src/skills/md-language-check/SKILL.md +1 -1
  150. package/dist/agent-src/skills/motion-choreographer/SKILL.md +1 -1
  151. package/dist/agent-src/skills/php-coder/evals/triggers.json +1 -0
  152. package/dist/agent-src/skills/prediction-pool-optimizer/evals/triggers.json +1 -0
  153. package/dist/agent-src/skills/premortem/SKILL.md +137 -0
  154. package/dist/agent-src/skills/prompt-engineering-image/SKILL.md +115 -0
  155. package/dist/agent-src/skills/prompt-engineering-image/evals/triggers.json +17 -0
  156. package/dist/agent-src/skills/prompt-validator/evals/triggers.json +1 -0
  157. package/dist/agent-src/skills/react-shadcn-ui/SKILL.md +12 -5
  158. package/dist/agent-src/skills/react-shadcn-ui/scripts/shadcn_add.ts +388 -0
  159. package/dist/agent-src/skills/reasoning-orchestrator/SKILL.md +1 -1
  160. package/dist/agent-src/skills/reasoning-orchestrator/evals/triggers.json +1 -0
  161. package/dist/agent-src/skills/refine-ticket/evals/triggers.json +1 -0
  162. package/dist/agent-src/skills/roadmap-management/SKILL.md +16 -3
  163. package/dist/agent-src/skills/roadmap-writing/SKILL.md +76 -0
  164. package/dist/agent-src/skills/root-cause-frameworks/SKILL.md +146 -0
  165. package/dist/agent-src/skills/rule-refactor/SKILL.md +9 -9
  166. package/dist/agent-src/skills/rule-writing/SKILL.md +7 -7
  167. package/dist/agent-src/skills/script-writing/SKILL.md +2 -2
  168. package/dist/agent-src/skills/security-audit/SKILL.md +5 -0
  169. package/dist/agent-src/skills/skill-improvement-pipeline/SKILL.md +19 -3
  170. package/dist/agent-src/skills/skill-management/SKILL.md +3 -3
  171. package/dist/agent-src/skills/skill-reviewer/SKILL.md +1 -1
  172. package/dist/agent-src/skills/skill-writing/SKILL.md +5 -5
  173. package/dist/agent-src/skills/skill-writing/evals/triggers.json +1 -0
  174. package/dist/agent-src/skills/source-discovery/SKILL.md +182 -0
  175. package/dist/agent-src/skills/standards-from-config/SKILL.md +93 -0
  176. package/dist/agent-src/skills/systematic-debugging/SKILL.md +7 -0
  177. package/dist/agent-src/skills/tailwind-engineer/scripts/tailwind_config_gen.ts +561 -0
  178. package/dist/agent-src/skills/threat-modeling/SKILL.md +1 -0
  179. package/dist/agent-src/skills/typography-system/SKILL.md +138 -0
  180. package/dist/agent-src/skills/typography-system/evals/triggers.json +17 -0
  181. package/dist/agent-src/skills/upstream-contribute/SKILL.md +3 -3
  182. package/dist/agent-src/skills/verify-repair-loop/SKILL.md +209 -0
  183. package/dist/agent-src/skills/verify-repair-loop/evals/output-schema.yml +20 -0
  184. package/dist/agent-src/skills/verify-repair-loop/evals/triggers.json +17 -0
  185. package/dist/agent-src/templates/agent-settings.md +7 -0
  186. package/dist/agent-src/templates/contexts/knowledge-card.md +69 -0
  187. package/dist/agent-src/templates/contexts/lesson-card.md +73 -0
  188. package/dist/agent-src/templates/roadmaps.md +29 -1
  189. package/dist/agent-src/templates/scripts/README.md +6 -6
  190. package/dist/agent-src/templates/scripts/check_memory.ts +640 -0
  191. package/dist/agent-src/templates/scripts/check_memory_proposal.ts +351 -0
  192. package/dist/agent-src/templates/scripts/implement_ticket/__main__.ts +27 -0
  193. package/dist/agent-src/templates/scripts/memory_hash.ts +333 -0
  194. package/dist/agent-src/templates/scripts/memory_lookup.ts +1067 -0
  195. package/dist/agent-src/templates/scripts/memory_report.ts +846 -0
  196. package/dist/agent-src/templates/scripts/memory_signal.ts +422 -0
  197. package/dist/agent-src/templates/scripts/memory_status.ts +191 -0
  198. package/dist/agent-src/templates/scripts/pr_review_routing.ts +523 -0
  199. package/dist/agent-src/templates/scripts/pr_risk_review.ts +0 -0
  200. package/dist/agent-src/templates/scripts/telemetry/aggregator.ts +0 -0
  201. package/dist/agent-src/templates/scripts/telemetry/boundary.ts +164 -0
  202. package/dist/agent-src/templates/scripts/telemetry/engagement.ts +479 -0
  203. package/dist/agent-src/templates/scripts/telemetry/report_renderer.ts +394 -0
  204. package/dist/agent-src/templates/scripts/telemetry/settings.ts +210 -0
  205. package/dist/agent-src/templates/scripts/telemetry_record.ts +255 -0
  206. package/dist/agent-src/templates/scripts/telemetry_report.ts +189 -0
  207. package/dist/agent-src/templates/scripts/telemetry_status.ts +312 -0
  208. package/dist/agent-src/templates/scripts/tier_usage_report.ts +597 -0
  209. package/dist/agent-src/templates/scripts/work_engine/__main__.ts +14 -0
  210. package/dist/agent-src/templates/scripts/work_engine/_lib/agent_settings.ts +1118 -0
  211. package/dist/agent-src/templates/scripts/work_engine/_lib/user_global_paths.ts +329 -0
  212. package/dist/agent-src/templates/scripts/work_engine/cli.ts +206 -0
  213. package/dist/agent-src/templates/scripts/work_engine/cli_args.ts +249 -0
  214. package/dist/agent-src/templates/scripts/work_engine/delivery_state.ts +225 -0
  215. package/dist/agent-src/templates/scripts/work_engine/directives/backend/analyze.ts +125 -0
  216. package/dist/agent-src/templates/scripts/work_engine/directives/backend/implement.ts +189 -0
  217. package/dist/agent-src/templates/scripts/work_engine/directives/backend/index.ts +94 -0
  218. package/dist/agent-src/templates/scripts/work_engine/directives/backend/memory.ts +193 -0
  219. package/dist/agent-src/templates/scripts/work_engine/directives/backend/plan.ts +267 -0
  220. package/dist/agent-src/templates/scripts/work_engine/directives/backend/refine.ts +518 -0
  221. package/dist/agent-src/templates/scripts/work_engine/directives/backend/report.ts +379 -0
  222. package/dist/agent-src/templates/scripts/work_engine/directives/backend/test.ts +268 -0
  223. package/dist/agent-src/templates/scripts/work_engine/directives/backend/verify.ts +258 -0
  224. package/dist/agent-src/templates/scripts/work_engine/directives/index.ts +32 -0
  225. package/dist/agent-src/templates/scripts/work_engine/directives/mixed/contract.ts +243 -0
  226. package/dist/agent-src/templates/scripts/work_engine/directives/mixed/index.ts +108 -0
  227. package/dist/agent-src/templates/scripts/work_engine/directives/mixed/stitch.ts +259 -0
  228. package/dist/agent-src/templates/scripts/work_engine/directives/mixed/ui.ts +216 -0
  229. package/dist/agent-src/templates/scripts/work_engine/directives/ui/_passthrough.ts +40 -0
  230. package/dist/agent-src/templates/scripts/work_engine/directives/ui/app_spec.ts +241 -0
  231. package/dist/agent-src/templates/scripts/work_engine/directives/ui/apply.ts +216 -0
  232. package/dist/agent-src/templates/scripts/work_engine/directives/ui/audit.ts +506 -0
  233. package/dist/agent-src/templates/scripts/work_engine/directives/ui/design.ts +325 -0
  234. package/dist/agent-src/templates/scripts/work_engine/directives/ui/index.ts +102 -0
  235. package/dist/agent-src/templates/scripts/work_engine/directives/ui/polish.ts +462 -0
  236. package/dist/agent-src/templates/scripts/work_engine/directives/ui/review.ts +474 -0
  237. package/dist/agent-src/templates/scripts/work_engine/directives/ui/scaffold.ts +352 -0
  238. package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/_skipped.ts +33 -0
  239. package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/apply.ts +213 -0
  240. package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/index.ts +111 -0
  241. package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/refine.ts +126 -0
  242. package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/report.ts +112 -0
  243. package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/test.ts +164 -0
  244. package/dist/agent-src/templates/scripts/work_engine/dispatcher.ts +515 -0
  245. package/dist/agent-src/templates/scripts/work_engine/emitters.ts +119 -0
  246. package/dist/agent-src/templates/scripts/work_engine/errors.ts +24 -0
  247. package/dist/agent-src/templates/scripts/work_engine/hook_bootstrap.ts +104 -0
  248. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/_chat_history_base.ts +176 -0
  249. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/chat_history_append.ts +41 -0
  250. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/chat_history_halt_append.ts +89 -0
  251. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/decision_gate.ts +193 -0
  252. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/decision_trace.ts +304 -0
  253. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/directive_set_guard.ts +110 -0
  254. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/halt_surface_audit.ts +118 -0
  255. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/index.ts +17 -0
  256. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/memory_visibility.ts +161 -0
  257. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/state_shape_validation.ts +45 -0
  258. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/trace.ts +134 -0
  259. package/dist/agent-src/templates/scripts/work_engine/hooks/context.ts +94 -0
  260. package/dist/agent-src/templates/scripts/work_engine/hooks/events.ts +58 -0
  261. package/dist/agent-src/templates/scripts/work_engine/hooks/exceptions.ts +85 -0
  262. package/dist/agent-src/templates/scripts/work_engine/hooks/index.ts +27 -0
  263. package/dist/agent-src/templates/scripts/work_engine/hooks/registry.ts +66 -0
  264. package/dist/agent-src/templates/scripts/work_engine/hooks/runner.ts +90 -0
  265. package/dist/agent-src/templates/scripts/work_engine/hooks/settings.ts +260 -0
  266. package/dist/agent-src/templates/scripts/work_engine/input_builders.ts +260 -0
  267. package/dist/agent-src/templates/scripts/work_engine/intent/classify.ts +466 -0
  268. package/dist/agent-src/templates/scripts/work_engine/migration/v0_to_v1.ts +531 -0
  269. package/dist/agent-src/templates/scripts/work_engine/orchestration.ts +366 -0
  270. package/dist/agent-src/templates/scripts/work_engine/persona_policy.ts +97 -0
  271. package/dist/agent-src/templates/scripts/work_engine/resolvers/diff.ts +135 -0
  272. package/dist/agent-src/templates/scripts/work_engine/resolvers/file.ts +175 -0
  273. package/dist/agent-src/templates/scripts/work_engine/resolvers/prompt.ts +115 -0
  274. package/dist/agent-src/templates/scripts/work_engine/scoring/confidence.ts +415 -0
  275. package/dist/agent-src/templates/scripts/work_engine/scoring/decision_engine.ts +466 -0
  276. package/dist/agent-src/templates/scripts/work_engine/scoring/decision_trace.ts +298 -0
  277. package/dist/agent-src/templates/scripts/work_engine/scoring/memory_visibility.ts +444 -0
  278. package/dist/agent-src/templates/scripts/work_engine/stack/detect.ts +252 -0
  279. package/dist/agent-src/templates/scripts/work_engine/stack/runner.ts +745 -0
  280. package/dist/agent-src/templates/scripts/work_engine/state.ts +1151 -0
  281. package/dist/agent-src/templates/scripts/work_engine/state_io.ts +413 -0
  282. package/dist/agent-src/templates/tickets.md +120 -0
  283. package/dist/cli/commands/commands.js +2 -2
  284. package/dist/cli/commands/commands.js.map +1 -1
  285. package/dist/cli/commands/doctorShell.js +4 -22
  286. package/dist/cli/commands/doctorShell.js.map +1 -1
  287. package/dist/cli/commands/packs.js +1 -1
  288. package/dist/cli/commands/packs.js.map +1 -1
  289. package/dist/cli/commands/recordTriggerEval.js +179 -0
  290. package/dist/cli/commands/recordTriggerEval.js.map +1 -0
  291. package/dist/cli/commands/recordTriggerEval.test.js +113 -0
  292. package/dist/cli/commands/recordTriggerEval.test.js.map +1 -0
  293. package/dist/cli/commands/workspaces.js +1 -1
  294. package/dist/cli/commands/workspaces.js.map +1 -1
  295. package/dist/cli/main.js +22 -1
  296. package/dist/cli/main.js.map +1 -1
  297. package/dist/cli/python/knowledge_ingest.js +1048 -0
  298. package/dist/cli/python/knowledge_ingest.js.map +1 -0
  299. package/dist/cli/python/workspace_analytics.js +1085 -0
  300. package/dist/cli/python/workspace_analytics.js.map +1 -0
  301. package/dist/cli/python/workspace_crypto.js +544 -0
  302. package/dist/cli/python/workspace_crypto.js.map +1 -0
  303. package/dist/cli/python/workspace_documents.js +1216 -0
  304. package/dist/cli/python/workspace_documents.js.map +1 -0
  305. package/dist/cli/python/workspace_drive.js +574 -0
  306. package/dist/cli/python/workspace_drive.js.map +1 -0
  307. package/dist/cli/python/workspace_drive_health.js +628 -0
  308. package/dist/cli/python/workspace_drive_health.js.map +1 -0
  309. package/dist/cli/python/workspace_explain.js +765 -0
  310. package/dist/cli/python/workspace_explain.js.map +1 -0
  311. package/dist/cli/python/workspace_hosts.js +349 -0
  312. package/dist/cli/python/workspace_hosts.js.map +1 -0
  313. package/dist/cli/python/workspace_inbox.js +692 -0
  314. package/dist/cli/python/workspace_inbox.js.map +1 -0
  315. package/dist/cli/python/workspace_render.js +816 -0
  316. package/dist/cli/python/workspace_render.js.map +1 -0
  317. package/dist/cli/python/workspace_roles.js +487 -0
  318. package/dist/cli/python/workspace_roles.js.map +1 -0
  319. package/dist/cli/python/workspace_secrets.js +180 -0
  320. package/dist/cli/python/workspace_secrets.js.map +1 -0
  321. package/dist/cli/python/workspace_sessions.js +1079 -0
  322. package/dist/cli/python/workspace_sessions.js.map +1 -0
  323. package/dist/cli/python/workspace_skills.js +417 -0
  324. package/dist/cli/python/workspace_skills.js.map +1 -0
  325. package/dist/cli/registry.js +2 -0
  326. package/dist/cli/registry.js.map +1 -1
  327. package/dist/discovery/deprecation-report.md +1 -1
  328. package/dist/discovery/discovery-manifest.json +1174 -123
  329. package/dist/discovery/discovery-manifest.json.sha256 +1 -1
  330. package/dist/discovery/discovery-manifest.summary.md +9 -6
  331. package/dist/discovery/orphan-report.md +1 -1
  332. package/dist/discovery/packs.json +163 -15
  333. package/dist/discovery/trust-report.md +4 -4
  334. package/dist/discovery/workspaces.json +73 -12
  335. package/dist/install/install.mjs +13934 -0
  336. package/dist/mcp/registry-manifest.json +4 -4
  337. package/dist/router.json +1 -1
  338. package/dist/server/routes/wizard.js +50 -21
  339. package/dist/server/routes/wizard.js.map +1 -1
  340. package/dist/server/routes/workspace.js +44 -25
  341. package/dist/server/routes/workspace.js.map +1 -1
  342. package/dist/server/schemas/settings.js +15 -0
  343. package/dist/server/schemas/settings.js.map +1 -1
  344. package/docs/SKILL_CENSUS.md +344 -0
  345. package/docs/architecture/augment-projection.md +1 -1
  346. package/docs/architecture/multi-tool-projection.md +3 -3
  347. package/docs/architecture.md +37 -6
  348. package/docs/benchmark.md +24 -27
  349. package/docs/capability-matrix.md +32 -0
  350. package/docs/catalog.md +50 -9
  351. package/docs/command-naming-audit.md +60 -0
  352. package/docs/contracts/STABILITY.md +32 -0
  353. package/docs/contracts/agents-md-tech-stack.md +1 -1
  354. package/docs/contracts/ai-council-config.md +22 -22
  355. package/docs/contracts/analysis-memory-loop.md +149 -0
  356. package/docs/contracts/benchmark-ab-contract.md +3 -3
  357. package/docs/contracts/branch-protection-policy.md +27 -0
  358. package/docs/contracts/brand-token-consumption.md +69 -0
  359. package/docs/contracts/command-clusters.md +2 -1
  360. package/docs/contracts/command-surface-tiers.md +13 -0
  361. package/docs/contracts/discovery-manifest.schema.json +24 -5
  362. package/docs/contracts/implement-ticket-flow.md +9 -9
  363. package/docs/contracts/install-layout.md +249 -0
  364. package/docs/contracts/kernel-membership.md +1 -1
  365. package/docs/contracts/linear-ai-rules-inclusion.md +2 -2
  366. package/docs/contracts/linter-structural-model.md +1 -1
  367. package/docs/contracts/mcp-discovery-phase-notice.md +1 -1
  368. package/docs/contracts/multi-tool-projection-fidelity.md +1 -1
  369. package/docs/contracts/namespace.md +2 -2
  370. package/docs/contracts/no-runtime-boundary.md +56 -0
  371. package/docs/contracts/package-self-orientation.md +24 -0
  372. package/docs/contracts/provider-lifecycle.md +3 -3
  373. package/docs/contracts/reasoning-discipline-protocol.md +83 -0
  374. package/docs/contracts/rule-classification.md +3 -3
  375. package/docs/contracts/skill-domains.md +1 -1
  376. package/docs/contracts/smoke-contracts.md +1 -1
  377. package/docs/contracts/surface-tiers.md +81 -0
  378. package/docs/contracts/ticket-bundle-format.md +228 -0
  379. package/docs/cookbook.md +152 -0
  380. package/docs/customization.md +12 -1
  381. package/docs/decisions/ADR-013-discovery-frontmatter-contract.md +16 -0
  382. package/docs/decisions/ADR-056-unvalidated-video-adapters-disposition.md +1 -1
  383. package/docs/decisions/ADR-059-render-resume-filesystem-as-state.md +1 -1
  384. package/docs/decisions/ADR-060-comfyui-sandbox-model.md +1 -1
  385. package/docs/decisions/ADR-061-corpus-grounding-layer.md +48 -1
  386. package/docs/decisions/ADR-096-analysis-workbench.md +110 -0
  387. package/docs/decisions/ADR-097-mission-recipe-privilege-boundary.md +121 -0
  388. package/docs/decisions/ADR-098-evidence-first-structure-discovery.md +154 -0
  389. package/docs/decisions/ADR-099-file-first-pattern-library.md +87 -0
  390. package/docs/decisions/ADR-100-global-knowledge-card-sharing.md +133 -0
  391. package/docs/decisions/ADR-101-ticket-bundle-emission.md +109 -0
  392. package/docs/decisions/ADR-102-ticket-handoff-paste-and-mcp.md +72 -0
  393. package/docs/decisions/ADR-103-global-knowledge-default-off-until-measured.md +92 -0
  394. package/docs/decisions/ADR-200-python-to-typescript-migration.md +193 -0
  395. package/docs/decisions/INDEX.md +9 -0
  396. package/docs/distribution/mcp-submission-checklist.md +3 -3
  397. package/docs/featured-commands.md +1 -1
  398. package/docs/featured-skills.md +1 -1
  399. package/docs/getting-started-by-role.md +2 -0
  400. package/docs/getting-started.md +2 -2
  401. package/docs/guidelines/agent-infra/failure-signatures.md +35 -0
  402. package/docs/guidelines/agent-infra/frontier-reasoning-operating-profile.md +5 -0
  403. package/docs/guidelines/agent-infra/size-and-scope.md +17 -0
  404. package/docs/guidelines/agent-infra/skill-quality-checklist.md +2 -2
  405. package/docs/guides/frontend-design-corpus-refresh.md +83 -0
  406. package/docs/guides/skill-preview.md +1 -1
  407. package/docs/hook-payload-capture.md +4 -4
  408. package/docs/mcp.md +1 -1
  409. package/docs/migration/consumer-template-consumption-model.md +145 -0
  410. package/docs/migration/divergences/README.md +55 -0
  411. package/docs/migration/divergences/_template.md +50 -0
  412. package/docs/migration/divergences/bench-stats-float-precision.md +72 -0
  413. package/docs/migration/divergences/mcp-telemetry-node-sqlite.md +61 -0
  414. package/docs/migration/divergences/pack-mcp-content-gzip-body.md +53 -0
  415. package/docs/migration/divergences/src-scripts-build_cloud_bundle.md +63 -0
  416. package/docs/migration/divergences/src-scripts-check_memory.md +91 -0
  417. package/docs/migration/divergences/src-scripts-inventory_abstraction_budget.md +65 -0
  418. package/docs/migration/divergences/src-scripts-lint_marketplace.md +57 -0
  419. package/docs/migration/divergences/src-scripts-lint_mcp_registry_manifest.md +70 -0
  420. package/docs/migration/divergences/src-scripts-spotcheck_thin_root.md +60 -0
  421. package/docs/migration/divergences/src-scripts-validate_agent_settings.md +58 -0
  422. package/docs/migration/node-floor.md +86 -0
  423. package/docs/migration/yaml-roundtrip-spike.md +163 -0
  424. package/docs/personas.md +6 -1
  425. package/docs/role-experiences.md +19 -0
  426. package/docs/setup/per-ide/windsurf.md +1 -1
  427. package/docs/skills-catalog.md +24 -3
  428. package/docs/threat-model.md +28 -0
  429. package/llms.txt +23 -2
  430. package/package.json +10 -15
  431. package/src/config/agent-settings.template.yml +64 -1
  432. package/src/config/discovery/packs.yml +31 -0
  433. package/src/config/discovery/unassigned-artefacts.yml +6 -0
  434. package/src/config/discovery/workspaces.yml +2 -2
  435. package/src/config/gitignore-block.txt +7 -0
  436. package/src/scripts/_cli/cmd_doctor.ts +2306 -0
  437. package/src/scripts/_cli/cmd_explain.ts +748 -0
  438. package/src/scripts/_cli/cmd_export.ts +375 -0
  439. package/src/scripts/_cli/cmd_migrate.ts +951 -0
  440. package/src/scripts/_cli/cmd_prune.ts +610 -0
  441. package/src/scripts/_cli/cmd_refresh.ts +530 -0
  442. package/src/scripts/_cli/cmd_settings_check.ts +407 -0
  443. package/src/scripts/_cli/cmd_settings_migrate.ts +344 -0
  444. package/src/scripts/_cli/cmd_sync.ts +381 -0
  445. package/src/scripts/_cli/cmd_uninstall.ts +833 -0
  446. package/src/scripts/_cli/cmd_update.ts +585 -0
  447. package/src/scripts/_cli/cmd_upgrade.ts +390 -0
  448. package/src/scripts/_cli/cmd_validate.ts +394 -0
  449. package/src/scripts/_cli/cmd_versions.ts +492 -0
  450. package/src/scripts/_cli/explain_last/assumptions.ts +114 -0
  451. package/src/scripts/_cli/explain_last/council.ts +197 -0
  452. package/src/scripts/_cli/explain_last/halt.ts +73 -0
  453. package/src/scripts/_cli/explain_last/index.ts +155 -0
  454. package/src/scripts/_cli/explain_last/inputs.ts +211 -0
  455. package/src/scripts/_cli/explain_last/memory.ts +231 -0
  456. package/src/scripts/_cli/explain_last/provider.ts +82 -0
  457. package/src/scripts/_cli/explain_last/render.ts +54 -0
  458. package/src/scripts/_cli/explain_last/route.ts +70 -0
  459. package/src/scripts/_cli/explain_last/scrubber.ts +138 -0
  460. package/src/scripts/_cli/explain_last/sections/assumptions.ts +51 -0
  461. package/src/scripts/_cli/explain_last/sections/council.ts +56 -0
  462. package/src/scripts/_cli/explain_last/sections/halt.ts +60 -0
  463. package/src/scripts/_cli/explain_last/sections/header.ts +50 -0
  464. package/src/scripts/_cli/explain_last/sections/index.ts +21 -0
  465. package/src/scripts/_cli/explain_last/sections/inputs.ts +63 -0
  466. package/src/scripts/_cli/explain_last/sections/memory.ts +124 -0
  467. package/src/scripts/_cli/explain_last/sections/pack.ts +42 -0
  468. package/src/scripts/_cli/explain_last/sections/provider.ts +51 -0
  469. package/src/scripts/_cli/explain_last/sections/route.ts +48 -0
  470. package/src/scripts/_cli/explain_last/state_loader.ts +119 -0
  471. package/src/scripts/_dispatch.bash +179 -163
  472. package/src/scripts/_lib/agent_settings.ts +1123 -0
  473. package/src/scripts/_lib/agent_src.ts +654 -0
  474. package/src/scripts/_lib/agents_overlay.ts +183 -0
  475. package/src/scripts/_lib/bench_ab_cache.ts +399 -0
  476. package/src/scripts/_lib/bench_ab_scoring.ts +352 -0
  477. package/src/scripts/_lib/bench_ab_scoring_v2.ts +751 -0
  478. package/src/scripts/_lib/bench_cost.ts +396 -0
  479. package/src/scripts/_lib/bench_quality.ts +237 -0
  480. package/src/scripts/_lib/bench_report.ts +255 -0
  481. package/src/scripts/_lib/bench_telegraph.ts +516 -0
  482. package/src/scripts/_lib/bench_telegraph_report.ts +272 -0
  483. package/src/scripts/_lib/changelog_eras.ts +398 -0
  484. package/src/scripts/_lib/claude_desktop_bundler.ts +324 -0
  485. package/src/scripts/_lib/cli_wrapper.ts +89 -0
  486. package/src/scripts/_lib/fs_atomic.ts +172 -0
  487. package/src/scripts/_lib/global_deploy_inventory.ts +639 -0
  488. package/src/scripts/_lib/install_layout.ts +87 -0
  489. package/src/scripts/_lib/install_regenerator.ts +157 -0
  490. package/src/scripts/_lib/installed_lock.ts +451 -0
  491. package/src/scripts/_lib/installed_tools.ts +518 -0
  492. package/src/scripts/_lib/json_pointers.ts +388 -0
  493. package/src/scripts/_lib/knowledge_global.ts +770 -0
  494. package/src/scripts/_lib/knowledge_global_promote.ts +453 -0
  495. package/src/scripts/_lib/knowledge_global_redaction.ts +448 -0
  496. package/src/scripts/_lib/link_crypto.ts +325 -0
  497. package/src/scripts/_lib/linked_projects.ts +613 -0
  498. package/src/scripts/_lib/model_tier.ts +65 -0
  499. package/src/scripts/_lib/module_detection.ts +275 -0
  500. package/src/scripts/_lib/node_sqlite.d.ts +32 -0
  501. package/src/scripts/_lib/pin_resolver.ts +264 -0
  502. package/src/scripts/_lib/py_random.ts +212 -0
  503. package/src/scripts/_lib/script_output.ts +147 -0
  504. package/src/scripts/_lib/security_lint.ts +623 -0
  505. package/src/scripts/_lib/surface_tiers.ts +127 -0
  506. package/src/scripts/_lib/token_count.ts +126 -0
  507. package/src/scripts/_lib/update_check.ts +297 -0
  508. package/src/scripts/_lib/user_global_paths.ts +329 -0
  509. package/src/scripts/_lib/value_ladder.ts +882 -0
  510. package/src/scripts/_lib/value_report.ts +617 -0
  511. package/src/scripts/_lib/zip_min.ts +175 -0
  512. package/src/scripts/adoption_report.ts +357 -0
  513. package/src/scripts/adoption_snapshot.ts +392 -0
  514. package/src/scripts/adoption_status.ts +424 -0
  515. package/src/scripts/adr/regenerate_index.ts +257 -0
  516. package/src/scripts/ai-image/adapters/flux.sh +45 -0
  517. package/src/scripts/ai-image/adapters/gemini-image.sh +45 -0
  518. package/src/scripts/ai-image/adapters/ideogram.sh +45 -0
  519. package/src/scripts/ai-image/adapters/recraft.sh +47 -0
  520. package/src/scripts/ai-video/adapters/comfyui.sh +3 -3
  521. package/src/scripts/ai-video/adapters/fal.sh +3 -3
  522. package/src/scripts/ai-video/adapters/gemini-veo.sh +3 -3
  523. package/src/scripts/ai-video/adapters/higgsfield.sh +3 -3
  524. package/src/scripts/ai-video/adapters/kling.sh +3 -3
  525. package/src/scripts/ai-video/adapters/musetalk.sh +2 -2
  526. package/src/scripts/ai-video/adapters/openai-images.sh +3 -3
  527. package/src/scripts/ai-video/adapters/replicate.sh +3 -3
  528. package/src/scripts/ai-video/adapters/sora.sh +3 -3
  529. package/src/scripts/ai-video/adapters/syncso.sh +3 -3
  530. package/src/scripts/ai-video/audio-adapters/allin1.sh +2 -2
  531. package/src/scripts/ai-video/audio-adapters/whisperx.sh +2 -2
  532. package/src/scripts/ai-video/lib/audio-adapter-contract.md +1 -1
  533. package/src/scripts/ai-video/lib/embed-provenance.sh +2 -2
  534. package/src/scripts/ai-video/lib/ingest-song.sh +2 -2
  535. package/src/scripts/ai-video/lib/parse-blueprint.sh +1 -1
  536. package/src/scripts/ai-video/lib/resume-scan.sh +2 -2
  537. package/src/scripts/ai-video/smoke-trace.sh +16 -7
  538. package/src/scripts/ai-video/stitch.sh +2 -2
  539. package/src/scripts/ai_council/_default_prices.ts +73 -0
  540. package/src/scripts/ai_council/advisors.ts +244 -0
  541. package/src/scripts/ai_council/airgap.ts +249 -0
  542. package/src/scripts/ai_council/budget_guard.ts +492 -0
  543. package/src/scripts/ai_council/bundler.ts +376 -0
  544. package/src/scripts/ai_council/cli_hints.ts +120 -0
  545. package/src/scripts/ai_council/clients.ts +2214 -0
  546. package/src/scripts/ai_council/compile_corpus.ts +681 -0
  547. package/src/scripts/ai_council/confidence_gate.ts +230 -0
  548. package/src/scripts/ai_council/config.ts +1729 -0
  549. package/src/scripts/ai_council/consensus.ts +551 -0
  550. package/src/scripts/ai_council/events_log.ts +327 -0
  551. package/src/scripts/ai_council/learn_low_impact_preview.ts +317 -0
  552. package/src/scripts/ai_council/low_impact.ts +1069 -0
  553. package/src/scripts/ai_council/low_impact_corpus.ts +662 -0
  554. package/src/scripts/ai_council/low_impact_intake.ts +222 -0
  555. package/src/scripts/ai_council/modes.ts +169 -0
  556. package/src/scripts/ai_council/necessity.ts +933 -0
  557. package/src/scripts/ai_council/orchestrator.ts +1689 -0
  558. package/src/scripts/ai_council/pricing.ts +267 -0
  559. package/src/scripts/ai_council/probation_gate.ts +282 -0
  560. package/src/scripts/ai_council/project_context.ts +308 -0
  561. package/src/scripts/ai_council/prompts.ts +600 -0
  562. package/src/scripts/ai_council/redact_low_impact_entry.ts +291 -0
  563. package/src/scripts/ai_council/replay.ts +314 -0
  564. package/src/scripts/ai_council/session.ts +558 -0
  565. package/src/scripts/ai_council/shadow_dispatch.ts +509 -0
  566. package/src/scripts/ai_council/solo_dispatch.ts +281 -0
  567. package/src/scripts/analysis_freshness.ts +343 -0
  568. package/src/scripts/annotate_discovery.ts +288 -0
  569. package/src/scripts/apply_modules_config.ts +537 -0
  570. package/src/scripts/audit_adr_coverage.ts +357 -0
  571. package/src/scripts/audit_auto_rules.ts +415 -0
  572. package/src/scripts/audit_cloud_compatibility.ts +608 -0
  573. package/src/scripts/audit_command_surface.ts +1227 -0
  574. package/src/scripts/audit_initial_context.ts +694 -0
  575. package/src/scripts/audit_likelihood.ts +434 -0
  576. package/src/scripts/audit_mcp_tools.ts +252 -0
  577. package/src/scripts/audit_overlap.ts +421 -0
  578. package/src/scripts/audit_skill_descriptions.ts +402 -0
  579. package/src/scripts/audit_skill_overlap.ts +576 -0
  580. package/src/scripts/audit_user_type_axis.ts +264 -0
  581. package/src/scripts/backfill_model_tier.ts +349 -0
  582. package/src/scripts/bench_ab_cache_dispatch.ts +126 -0
  583. package/src/scripts/bench_ab_clone.ts +610 -0
  584. package/src/scripts/bench_ab_diff.ts +609 -0
  585. package/src/scripts/bench_ab_integrity.ts +261 -0
  586. package/src/scripts/bench_ab_run.ts +417 -0
  587. package/src/scripts/bench_ab_task_runner.ts +1382 -0
  588. package/src/scripts/bench_ab_tracka_run.ts +436 -0
  589. package/src/scripts/bench_ab_v2_run.ts +585 -0
  590. package/src/scripts/bench_ab_v2_stats.ts +1018 -0
  591. package/src/scripts/bench_baseline_ready.ts +326 -0
  592. package/src/scripts/bench_condense_memory.ts +479 -0
  593. package/src/scripts/bench_drift_check.ts +503 -0
  594. package/src/scripts/bench_per_tool.ts +591 -0
  595. package/src/scripts/bench_rtk_savings.ts +710 -0
  596. package/src/scripts/bench_run.ts +509 -0
  597. package/src/scripts/bench_runner.ts +519 -0
  598. package/src/scripts/build_cloud_bundle.ts +692 -0
  599. package/src/scripts/build_discovery_manifest.ts +1371 -0
  600. package/src/scripts/build_linear_digest.ts +368 -0
  601. package/src/scripts/build_mcp_registry_manifest.ts +351 -0
  602. package/src/scripts/build_rule_trigger_matrix.ts +469 -0
  603. package/src/scripts/capture_showcase_session.ts +735 -0
  604. package/src/scripts/chat_history.ts +2301 -0
  605. package/src/scripts/check_always_budget.ts +694 -0
  606. package/src/scripts/check_artefact_checksums.ts +281 -0
  607. package/src/scripts/check_augment_description_cap.ts +133 -0
  608. package/src/scripts/check_augmentignore.ts +108 -0
  609. package/src/scripts/check_beta_review_markers.ts +234 -0
  610. package/src/scripts/check_bite_sized_granularity.ts +116 -0
  611. package/src/scripts/check_cluster_patterns.ts +285 -0
  612. package/src/scripts/check_command_count_messaging.ts +224 -0
  613. package/src/scripts/check_condensation.ts +900 -0
  614. package/src/scripts/check_condensed_paths.ts +414 -0
  615. package/src/scripts/check_context_paths.ts +388 -0
  616. package/src/scripts/check_council_config_location.ts +260 -0
  617. package/src/scripts/check_council_layout.ts +180 -0
  618. package/src/scripts/check_council_references.ts +345 -0
  619. package/src/scripts/check_discovery_determinism.ts +124 -0
  620. package/src/scripts/check_gate_paths.ts +230 -0
  621. package/src/scripts/check_iron_law_prominence.ts +298 -0
  622. package/src/scripts/check_kernel_rule_bundle.ts +242 -0
  623. package/src/scripts/check_knowledge_cards.ts +759 -0
  624. package/src/scripts/check_md_language.ts +291 -0
  625. package/src/scripts/check_memory.ts +845 -0
  626. package/src/scripts/check_memory_proposal.ts +351 -0
  627. package/src/scripts/check_module_management_neutral.ts +238 -0
  628. package/src/scripts/check_no_conflict_markers.ts +298 -0
  629. package/src/scripts/check_no_conflict_markers_allowlist.json +4 -0
  630. package/src/scripts/check_no_external_sources.ts +351 -0
  631. package/src/scripts/check_no_local_settings_committed.ts +69 -0
  632. package/src/scripts/check_no_new_legacy_path.ts +188 -0
  633. package/src/scripts/check_no_roadmap_refs.ts +304 -0
  634. package/src/scripts/check_one_off_location.ts +165 -0
  635. package/src/scripts/check_overlay_cascade_subdirs.ts +188 -0
  636. package/src/scripts/check_portability.ts +860 -0
  637. package/src/scripts/check_proposal.ts +0 -0
  638. package/src/scripts/check_public_catalog_links.ts +204 -0
  639. package/src/scripts/check_public_links.ts +357 -0
  640. package/src/scripts/check_references.ts +963 -0
  641. package/src/scripts/check_release_includes_discovery.ts +94 -0
  642. package/src/scripts/check_release_pr_shape.ts +222 -0
  643. package/src/scripts/check_release_published.ts +235 -0
  644. package/src/scripts/check_release_trunk_sync.ts +203 -0
  645. package/src/scripts/check_reply_consistency.ts +359 -0
  646. package/src/scripts/check_roadmap_trackable.ts +268 -0
  647. package/src/scripts/check_role_doc_links.ts +187 -0
  648. package/src/scripts/check_safety_floor_untouched.ts +160 -0
  649. package/src/scripts/check_skill_requires.ts +205 -0
  650. package/src/scripts/check_structural_breaking.ts +170 -0
  651. package/src/scripts/check_surface_tiers.ts +567 -0
  652. package/src/scripts/check_template_pin_drift.ts +222 -0
  653. package/src/scripts/check_test_coverage_diff.ts +235 -0
  654. package/src/scripts/check_token_optimizer_freshness.ts +183 -0
  655. package/src/scripts/check_trigger_evals.ts +375 -0
  656. package/src/scripts/check_update_banner.ts +143 -0
  657. package/src/scripts/ci_status.ts +0 -0
  658. package/src/scripts/ci_summary.ts +235 -0
  659. package/src/scripts/ci_time_ratio.ts +526 -0
  660. package/src/scripts/command_suggester/cooldown.ts +176 -0
  661. package/src/scripts/command_suggester/index.ts +41 -0
  662. package/src/scripts/command_suggester/loader.ts +205 -0
  663. package/src/scripts/command_suggester/match.ts +294 -0
  664. package/src/scripts/command_suggester/rank.ts +201 -0
  665. package/src/scripts/command_suggester/render.ts +122 -0
  666. package/src/scripts/command_suggester/sanitize.ts +114 -0
  667. package/src/scripts/command_suggester/settings.ts +186 -0
  668. package/src/scripts/command_suggester/types.ts +0 -0
  669. package/src/scripts/compile_router.ts +297 -0
  670. package/src/scripts/condense.sh +7 -1
  671. package/src/scripts/condense.ts +2035 -0
  672. package/src/scripts/condense_memory.ts +334 -0
  673. package/src/scripts/config/index.ts +15 -0
  674. package/src/scripts/config/packs.ts +310 -0
  675. package/src/scripts/config/presets.ts +369 -0
  676. package/src/scripts/config/profile_explain.ts +114 -0
  677. package/src/scripts/config/profiles.ts +277 -0
  678. package/src/scripts/config/session_profiles.ts +1064 -0
  679. package/src/scripts/context_hygiene_hook.ts +272 -0
  680. package/src/scripts/cost_by_conversation.ts +444 -0
  681. package/src/scripts/cost_summary.ts +407 -0
  682. package/src/scripts/council_cli.ts +2827 -0
  683. package/src/scripts/council_prune.ts +153 -0
  684. package/src/scripts/cross_repo_retrieve.ts +694 -0
  685. package/src/scripts/discovery_stats.ts +218 -0
  686. package/src/scripts/evidence_report.ts +580 -0
  687. package/src/scripts/external_sources_denylist.json +1 -0
  688. package/src/scripts/extract_audit_patterns.ts +394 -0
  689. package/src/scripts/first_run_gate_hook.ts +246 -0
  690. package/src/scripts/gen_discovery_baseline.ts +297 -0
  691. package/src/scripts/generate_capabilities_index.ts +496 -0
  692. package/src/scripts/generate_capability_matrix.ts +430 -0
  693. package/src/scripts/generate_catalog.ts +178 -0
  694. package/src/scripts/generate_command_flows.ts +316 -0
  695. package/src/scripts/generate_cookbook.ts +302 -0
  696. package/src/scripts/generate_index.ts +500 -0
  697. package/src/scripts/generate_ownership_matrix.ts +646 -0
  698. package/src/scripts/generate_pack_manifests.ts +1025 -0
  699. package/src/scripts/generate_role_experiences_catalog.ts +265 -0
  700. package/src/scripts/hermetic-install.sh +22 -11
  701. package/src/scripts/hook_manifest.yaml +24 -10
  702. package/src/scripts/hooks/augment-chat-history.sh +3 -10
  703. package/src/scripts/hooks/augment-context-hygiene.sh +3 -10
  704. package/src/scripts/hooks/augment-dispatcher.sh +3 -10
  705. package/src/scripts/hooks/augment-onboarding-gate.sh +3 -10
  706. package/src/scripts/hooks/augment-roadmap-progress.sh +3 -10
  707. package/src/scripts/hooks/block_no_verify.ts +413 -0
  708. package/src/scripts/hooks/cline-dispatcher.sh +3 -10
  709. package/src/scripts/hooks/cowork-dispatcher.sh +3 -14
  710. package/src/scripts/hooks/cursor-dispatcher.sh +3 -10
  711. package/src/scripts/hooks/dispatch_hook.ts +851 -0
  712. package/src/scripts/hooks/dispatch_issues.ts +226 -0
  713. package/src/scripts/hooks/envelope.ts +140 -0
  714. package/src/scripts/hooks/gemini-dispatcher.sh +3 -8
  715. package/src/scripts/hooks/replay_hook.ts +364 -0
  716. package/src/scripts/hooks/state_io.ts +293 -0
  717. package/src/scripts/hooks/windsurf-dispatcher.sh +3 -9
  718. package/src/scripts/hooks_doctor.ts +418 -0
  719. package/src/scripts/hooks_status.ts +292 -0
  720. package/src/scripts/injection_scan_hook.ts +285 -0
  721. package/src/scripts/install +36 -22
  722. package/src/scripts/install-hooks.sh +20 -14
  723. package/src/scripts/install.sh +38 -14
  724. package/src/scripts/install.ts +4515 -0
  725. package/src/scripts/inventory_abstraction_budget.ts +1104 -0
  726. package/src/scripts/inventory_frontmatter.ts +320 -0
  727. package/src/scripts/inventory_meta_layers.ts +516 -0
  728. package/src/scripts/iron_law_sha.ts +233 -0
  729. package/src/scripts/knowledge_global_cli.ts +1105 -0
  730. package/src/scripts/linked_projects_list.ts +310 -0
  731. package/src/scripts/lint_agent_security.ts +224 -0
  732. package/src/scripts/lint_agent_skill_names.ts +241 -0
  733. package/src/scripts/lint_agents_layout.ts +205 -0
  734. package/src/scripts/lint_agents_md.ts +294 -0
  735. package/src/scripts/lint_archived_skills.ts +309 -0
  736. package/src/scripts/lint_artefact_frontmatter.ts +359 -0
  737. package/src/scripts/lint_bench_ab.ts +319 -0
  738. package/src/scripts/lint_bench_corpus.ts +421 -0
  739. package/src/scripts/lint_command_flow_coverage.ts +231 -0
  740. package/src/scripts/lint_command_routing.ts +377 -0
  741. package/src/scripts/lint_command_tiers.ts +345 -0
  742. package/src/scripts/lint_command_verbs.ts +379 -0
  743. package/src/scripts/lint_commit_subjects.ts +243 -0
  744. package/src/scripts/lint_context_spine_usage.ts +198 -0
  745. package/src/scripts/lint_discovery_manifest.ts +540 -0
  746. package/src/scripts/lint_discovery_vocabulary.ts +393 -0
  747. package/src/scripts/lint_empty_roadmaps.ts +147 -0
  748. package/src/scripts/lint_eval_freshness.ts +335 -0
  749. package/src/scripts/lint_examples.ts +183 -0
  750. package/src/scripts/lint_explain_trace.ts +381 -0
  751. package/src/scripts/lint_featured_skills.ts +0 -0
  752. package/src/scripts/lint_flows.ts +701 -0
  753. package/src/scripts/lint_framework_leakage.ts +497 -0
  754. package/src/scripts/lint_framework_leakage_allowlist.json +8 -1
  755. package/src/scripts/lint_frontmatter_boilerplate.ts +356 -0
  756. package/src/scripts/lint_ghostwriter_source.ts +389 -0
  757. package/src/scripts/lint_global_paths.ts +420 -0
  758. package/src/scripts/lint_handoffs.ts +362 -0
  759. package/src/scripts/lint_hidden_unicode.ts +350 -0
  760. package/src/scripts/lint_hook_concern_budget.ts +319 -0
  761. package/src/scripts/lint_hook_manifest.ts +354 -0
  762. package/src/scripts/lint_instruction_smuggling.ts +173 -0
  763. package/src/scripts/lint_load_context.ts +371 -0
  764. package/src/scripts/lint_marketplace.ts +286 -0
  765. package/src/scripts/lint_marketplace_install_completeness.ts +309 -0
  766. package/src/scripts/lint_mcp_config_security.ts +225 -0
  767. package/src/scripts/lint_mcp_registry_manifest.ts +350 -0
  768. package/src/scripts/lint_media_policy_linkage.ts +224 -0
  769. package/src/scripts/lint_missions.ts +774 -0
  770. package/src/scripts/lint_model_tier_coverage.ts +151 -0
  771. package/src/scripts/lint_namespace.ts +295 -0
  772. package/src/scripts/lint_namespace_collisions.ts +203 -0
  773. package/src/scripts/lint_new_skill_gate.ts +462 -0
  774. package/src/scripts/lint_no_new_atomic_commands.ts +342 -0
  775. package/src/scripts/lint_one_off_age.ts +348 -0
  776. package/src/scripts/lint_orchestration_dsl.ts +377 -0
  777. package/src/scripts/lint_orchestrator_auto_detect.ts +177 -0
  778. package/src/scripts/lint_pack_boundaries.ts +366 -0
  779. package/src/scripts/lint_pack_dependencies.ts +541 -0
  780. package/src/scripts/lint_pack_first_win.ts +202 -0
  781. package/src/scripts/lint_persona_governance.ts +292 -0
  782. package/src/scripts/lint_positioning.ts +257 -0
  783. package/src/scripts/lint_profile_overlay_set_only.ts +324 -0
  784. package/src/scripts/lint_readme_jargon.ts +189 -0
  785. package/src/scripts/lint_readme_size.ts +73 -0
  786. package/src/scripts/lint_regression.ts +497 -0
  787. package/src/scripts/lint_roadmap_ci_steps.ts +252 -0
  788. package/src/scripts/lint_roadmap_complexity.ts +295 -0
  789. package/src/scripts/lint_roadmap_later_disposition.ts +357 -0
  790. package/src/scripts/lint_role_experiences.ts +410 -0
  791. package/src/scripts/lint_rule_interactions.ts +281 -0
  792. package/src/scripts/lint_rule_tiers.ts +169 -0
  793. package/src/scripts/lint_showcase_sessions.ts +254 -0
  794. package/src/scripts/lint_skill_frontmatter_safety.ts +279 -0
  795. package/src/scripts/lint_skill_originality.ts +586 -0
  796. package/src/scripts/lint_skill_originality_allowlist.json +20 -0
  797. package/src/scripts/lint_skill_tools.ts +320 -0
  798. package/src/scripts/lint_ticket_buildable.ts +1027 -0
  799. package/src/scripts/lint_topics_yaml.ts +203 -0
  800. package/src/scripts/lint_trust_coherence.ts +377 -0
  801. package/src/scripts/lint_value_dashboard.ts +314 -0
  802. package/src/scripts/lint_workflow_security.ts +637 -0
  803. package/src/scripts/lint_workflow_security_allowlist.json +20 -0
  804. package/src/scripts/lint_workspace_boundary.ts +248 -0
  805. package/src/scripts/mcp_parity_smoke.ts +638 -0
  806. package/src/scripts/mcp_render.ts +346 -0
  807. package/src/scripts/mcp_server/__main__.ts +28 -0
  808. package/src/scripts/mcp_server/catalog.ts +154 -0
  809. package/src/scripts/mcp_server/index.ts +24 -0
  810. package/src/scripts/mcp_server/metadata.ts +83 -0
  811. package/src/scripts/mcp_server/prompts.ts +711 -0
  812. package/src/scripts/mcp_server/resources.ts +343 -0
  813. package/src/scripts/mcp_server/server.ts +439 -0
  814. package/src/scripts/mcp_server/telemetry.ts +154 -0
  815. package/src/scripts/mcp_server/tools.ts +1031 -0
  816. package/src/scripts/mcp_setup.sh +25 -52
  817. package/src/scripts/mcp_telemetry_health.ts +362 -0
  818. package/src/scripts/mcp_telemetry_query.ts +371 -0
  819. package/src/scripts/mcp_telemetry_store.ts +422 -0
  820. package/src/scripts/measure_augment_budget.ts +453 -0
  821. package/src/scripts/measure_density.ts +618 -0
  822. package/src/scripts/measure_frugality_savings.ts +353 -0
  823. package/src/scripts/measure_markitdown_lift.ts +299 -0
  824. package/src/scripts/measure_patterns.ts +682 -0
  825. package/src/scripts/measure_projection_bytes.ts +425 -0
  826. package/src/scripts/measure_rule_budget.ts +627 -0
  827. package/src/scripts/measure_skill_reduction.ts +442 -0
  828. package/src/scripts/media/lib/adapter-common.sh +247 -0
  829. package/src/scripts/media/lib/adapter-contract.md +329 -0
  830. package/src/scripts/media/lib/fixtures/comfyui/result.json +1 -0
  831. package/src/scripts/media/lib/fixtures/fal/result.json +1 -0
  832. package/src/scripts/media/lib/fixtures/flux/asset-0001.png +0 -0
  833. package/src/scripts/media/lib/fixtures/flux/result.json +1 -0
  834. package/src/scripts/media/lib/fixtures/gemini-image/asset-0001.png +0 -0
  835. package/src/scripts/media/lib/fixtures/gemini-image/result.json +1 -0
  836. package/src/scripts/media/lib/fixtures/gemini-veo/result.json +1 -0
  837. package/src/scripts/media/lib/fixtures/higgsfield/result.json +1 -0
  838. package/src/scripts/media/lib/fixtures/ideogram/asset-0001.png +0 -0
  839. package/src/scripts/media/lib/fixtures/ideogram/result.json +1 -0
  840. package/src/scripts/media/lib/fixtures/kling/result.json +1 -0
  841. package/src/scripts/media/lib/fixtures/musetalk/result.json +1 -0
  842. package/src/scripts/media/lib/fixtures/openai-images/result.json +1 -0
  843. package/src/scripts/media/lib/fixtures/recraft/asset-0001.svg +1 -0
  844. package/src/scripts/media/lib/fixtures/recraft/result.json +1 -0
  845. package/src/scripts/media/lib/fixtures/replicate/result.json +1 -0
  846. package/src/scripts/media/lib/fixtures/sora/result.json +1 -0
  847. package/src/scripts/media/lib/fixtures/syncso/result.json +1 -0
  848. package/src/scripts/media/lib/load-config.sh +180 -0
  849. package/src/scripts/media/lib/redact.sh +85 -0
  850. package/src/scripts/memory_hash.ts +331 -0
  851. package/src/scripts/memory_lookup.ts +1278 -0
  852. package/src/scripts/memory_report.ts +845 -0
  853. package/src/scripts/memory_signal.ts +417 -0
  854. package/src/scripts/memory_status.ts +189 -0
  855. package/src/scripts/migrate_command_suggestions.ts +341 -0
  856. package/src/scripts/migrate_frontmatter_defaults.ts +539 -0
  857. package/src/scripts/migration_status.ts +301 -0
  858. package/src/scripts/mine_session.ts +645 -0
  859. package/src/scripts/minimal_safe_diff_hook.ts +355 -0
  860. package/src/scripts/move_artefact.ts +869 -0
  861. package/src/scripts/new_skill.ts +404 -0
  862. package/src/scripts/onboarding_gate_hook.ts +224 -0
  863. package/src/scripts/pack_dependency_allowlist.json +1 -1
  864. package/src/scripts/pack_mcp_content.ts +552 -0
  865. package/src/scripts/parity/README.md +140 -0
  866. package/src/scripts/parity/compare.ts +189 -0
  867. package/src/scripts/parity/coverage_diff.ts +199 -0
  868. package/src/scripts/parity/phase-manifest.json +93 -0
  869. package/src/scripts/parity/phase_gate.ts +270 -0
  870. package/src/scripts/parity/replay.ts +320 -0
  871. package/src/scripts/pattern_share.ts +363 -0
  872. package/src/scripts/plan_physical_move.ts +605 -0
  873. package/src/scripts/prediction-pool/poisson_sim.ts +537 -0
  874. package/src/scripts/prediction-pool/pool_winsim.ts +677 -0
  875. package/src/scripts/prediction-pool/score_ev.ts +546 -0
  876. package/src/scripts/print_required_checks.ts +249 -0
  877. package/src/scripts/probe_projection_fidelity.ts +468 -0
  878. package/src/scripts/probe_skill_registration.ts +787 -0
  879. package/src/scripts/profile_staleness_hook.ts +169 -0
  880. package/src/scripts/profile_use.ts +227 -0
  881. package/src/scripts/project_thin_rules.ts +387 -0
  882. package/src/scripts/propose_modules_config.ts +311 -0
  883. package/src/scripts/prototype_lint_contradictions.ts +414 -0
  884. package/src/scripts/prove_pack_extractable.ts +388 -0
  885. package/src/scripts/readme_linter.ts +913 -0
  886. package/src/scripts/redact_hook_capture.ts +325 -0
  887. package/src/scripts/refine_ticket_detect.ts +703 -0
  888. package/src/scripts/release.ts +1697 -0
  889. package/src/scripts/render_benchmark_md.ts +664 -0
  890. package/src/scripts/render_value_md.ts +506 -0
  891. package/src/scripts/repro/repro_marketplace_install_gap.sh +1 -1
  892. package/src/scripts/roadmap_progress_hook.ts +410 -0
  893. package/src/scripts/router_telemetry.ts +972 -0
  894. package/src/scripts/run.ts +98 -0
  895. package/src/scripts/run_skill_evals.ts +477 -0
  896. package/src/scripts/runtime_dispatcher.ts +586 -0
  897. package/src/scripts/runtime_handler.ts +231 -0
  898. package/src/scripts/runtime_registry.ts +394 -0
  899. package/src/scripts/schemas/command.schema.json +3 -2
  900. package/src/scripts/schemas/mission-catalog.schema.json +112 -0
  901. package/src/scripts/schemas/mission.schema.json +87 -0
  902. package/src/scripts/schemas/pack.schema.json +6 -0
  903. package/src/scripts/schemas/rule.schema.json +1 -0
  904. package/src/scripts/schemas/skill.schema.json +1 -0
  905. package/src/scripts/schemas/ticket-manifest.schema.json +35 -0
  906. package/src/scripts/schemas/ticket.schema.json +60 -0
  907. package/src/scripts/score_skill_selection.ts +570 -0
  908. package/src/scripts/security_audit_config.ts +423 -0
  909. package/src/scripts/skill_collision_clusters.ts +448 -0
  910. package/src/scripts/skill_discovery.ts +690 -0
  911. package/src/scripts/skill_linter.ts +4276 -0
  912. package/src/scripts/skill_overlap.ts +414 -0
  913. package/src/scripts/skill_preview.ts +548 -0
  914. package/src/scripts/skill_tools/audit_persona_coverage.ts +427 -0
  915. package/src/scripts/skill_tools/audit_user_type_coverage.ts +507 -0
  916. package/src/scripts/skill_tools/index.ts +28 -0
  917. package/src/scripts/skill_tools/run_block_d_eval.ts +373 -0
  918. package/src/scripts/skill_tools/score_skill_relevance.ts +475 -0
  919. package/src/scripts/skill_tools/suggest_skill_for_task.ts +288 -0
  920. package/src/scripts/skill_trigger_eval.ts +1046 -0
  921. package/src/scripts/skill_usage_collect.ts +465 -0
  922. package/src/scripts/skill_usage_report.ts +364 -0
  923. package/src/scripts/smoke/kernel.sh +4 -5
  924. package/src/scripts/smoke/router.sh +76 -76
  925. package/src/scripts/smoke/schema.sh +2 -2
  926. package/src/scripts/smoke/skills.sh +73 -52
  927. package/src/scripts/smoke_path_resolution.ts +194 -0
  928. package/src/scripts/smoke_quickstart.ts +224 -0
  929. package/src/scripts/snapshot_agent_outputs.ts +375 -0
  930. package/src/scripts/spotcheck_thin_root.ts +247 -0
  931. package/src/scripts/surface-tiers.yml +68 -0
  932. package/src/scripts/sync_agent_settings.ts +763 -0
  933. package/src/scripts/sync_github_metadata.ts +550 -0
  934. package/src/scripts/sync_gitignore.ts +630 -0
  935. package/src/scripts/sync_yaml_rt.ts +910 -0
  936. package/src/scripts/telegraph_stats.ts +447 -0
  937. package/src/scripts/tool_registry.ts +330 -0
  938. package/src/scripts/tools/adapter_errors.ts +93 -0
  939. package/src/scripts/tools/base_adapter.ts +147 -0
  940. package/src/scripts/tools/github_adapter.ts +229 -0
  941. package/src/scripts/tools/jira_adapter.ts +196 -0
  942. package/src/scripts/trigger_coverage.ts +251 -0
  943. package/src/scripts/update_counts.ts +284 -0
  944. package/src/scripts/update_prices.ts +219 -0
  945. package/src/scripts/validate_agent_settings.ts +265 -0
  946. package/src/scripts/validate_decision_engine.ts +366 -0
  947. package/src/scripts/validate_discovery_manifest.ts +160 -0
  948. package/src/scripts/validate_frontmatter.ts +1030 -0
  949. package/src/scripts/validate_pack_yaml.ts +0 -0
  950. package/src/scripts/validate_safe_paths.ts +164 -0
  951. package/src/scripts/validate_telegraph_carveouts.ts +485 -0
  952. package/src/scripts/verify_before_complete_hook.ts +306 -0
  953. package/src/scripts/verify_physical_move.ts +411 -0
  954. package/src/scripts/wrapper_freshness_hook.ts +179 -0
  955. package/dist/agent-src/scripts/archive_completed_roadmaps.py +0 -171
  956. package/dist/agent-src/scripts/update_roadmap_progress.py +0 -537
  957. package/dist/agent-src/skills/corpus-grounding/scripts/bm25_search.py +0 -212
  958. package/dist/agent-src/skills/corpus-grounding/scripts/decision_engine.py +0 -438
  959. package/dist/agent-src/skills/corpus-grounding/scripts/ground.py +0 -166
  960. package/dist/agent-src/skills/corpus-grounding/scripts/schema_validator.py +0 -160
  961. package/dist/agent-src/skills/design-tokens/scripts/tokens.py +0 -296
  962. package/dist/agent-src/skills/react-shadcn-ui/scripts/shadcn_add.py +0 -299
  963. package/dist/agent-src/skills/tailwind-engineer/scripts/tailwind_config_gen.py +0 -463
  964. package/dist/agent-src/templates/scripts/check_memory.py +0 -282
  965. package/dist/agent-src/templates/scripts/check_memory_proposal.py +0 -180
  966. package/dist/agent-src/templates/scripts/implement_ticket/__init__.py +0 -94
  967. package/dist/agent-src/templates/scripts/implement_ticket/__main__.py +0 -15
  968. package/dist/agent-src/templates/scripts/memory_hash.py +0 -75
  969. package/dist/agent-src/templates/scripts/memory_lookup.py +0 -436
  970. package/dist/agent-src/templates/scripts/memory_report.py +0 -314
  971. package/dist/agent-src/templates/scripts/memory_signal.py +0 -165
  972. package/dist/agent-src/templates/scripts/memory_status.py +0 -76
  973. package/dist/agent-src/templates/scripts/pr_review_routing.py +0 -340
  974. package/dist/agent-src/templates/scripts/pr_risk_review.py +0 -211
  975. package/dist/agent-src/templates/scripts/telemetry/__init__.py +0 -42
  976. package/dist/agent-src/templates/scripts/telemetry/aggregator.py +0 -169
  977. package/dist/agent-src/templates/scripts/telemetry/boundary.py +0 -171
  978. package/dist/agent-src/templates/scripts/telemetry/engagement.py +0 -297
  979. package/dist/agent-src/templates/scripts/telemetry/report_renderer.py +0 -197
  980. package/dist/agent-src/templates/scripts/telemetry/settings.py +0 -177
  981. package/dist/agent-src/templates/scripts/telemetry_record.py +0 -179
  982. package/dist/agent-src/templates/scripts/telemetry_report.py +0 -161
  983. package/dist/agent-src/templates/scripts/telemetry_status.py +0 -142
  984. package/dist/agent-src/templates/scripts/tier_usage_report.py +0 -183
  985. package/dist/agent-src/templates/scripts/work_engine/__init__.py +0 -58
  986. package/dist/agent-src/templates/scripts/work_engine/__main__.py +0 -9
  987. package/dist/agent-src/templates/scripts/work_engine/_lib/__init__.py +0 -7
  988. package/dist/agent-src/templates/scripts/work_engine/_lib/agent_settings.py +0 -840
  989. package/dist/agent-src/templates/scripts/work_engine/_lib/user_global_paths.py +0 -249
  990. package/dist/agent-src/templates/scripts/work_engine/cli.py +0 -195
  991. package/dist/agent-src/templates/scripts/work_engine/cli_args.py +0 -116
  992. package/dist/agent-src/templates/scripts/work_engine/delivery_state.py +0 -137
  993. package/dist/agent-src/templates/scripts/work_engine/directives/__init__.py +0 -33
  994. package/dist/agent-src/templates/scripts/work_engine/directives/backend/__init__.py +0 -98
  995. package/dist/agent-src/templates/scripts/work_engine/directives/backend/analyze.py +0 -98
  996. package/dist/agent-src/templates/scripts/work_engine/directives/backend/implement.py +0 -145
  997. package/dist/agent-src/templates/scripts/work_engine/directives/backend/memory.py +0 -136
  998. package/dist/agent-src/templates/scripts/work_engine/directives/backend/plan.py +0 -175
  999. package/dist/agent-src/templates/scripts/work_engine/directives/backend/refine.py +0 -396
  1000. package/dist/agent-src/templates/scripts/work_engine/directives/backend/report.py +0 -227
  1001. package/dist/agent-src/templates/scripts/work_engine/directives/backend/test.py +0 -180
  1002. package/dist/agent-src/templates/scripts/work_engine/directives/backend/verify.py +0 -170
  1003. package/dist/agent-src/templates/scripts/work_engine/directives/mixed/__init__.py +0 -116
  1004. package/dist/agent-src/templates/scripts/work_engine/directives/mixed/contract.py +0 -254
  1005. package/dist/agent-src/templates/scripts/work_engine/directives/mixed/stitch.py +0 -229
  1006. package/dist/agent-src/templates/scripts/work_engine/directives/mixed/ui.py +0 -231
  1007. package/dist/agent-src/templates/scripts/work_engine/directives/ui/__init__.py +0 -113
  1008. package/dist/agent-src/templates/scripts/work_engine/directives/ui/_passthrough.py +0 -44
  1009. package/dist/agent-src/templates/scripts/work_engine/directives/ui/apply.py +0 -241
  1010. package/dist/agent-src/templates/scripts/work_engine/directives/ui/audit.py +0 -414
  1011. package/dist/agent-src/templates/scripts/work_engine/directives/ui/design.py +0 -335
  1012. package/dist/agent-src/templates/scripts/work_engine/directives/ui/polish.py +0 -513
  1013. package/dist/agent-src/templates/scripts/work_engine/directives/ui/review.py +0 -471
  1014. package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/__init__.py +0 -119
  1015. package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/_skipped.py +0 -37
  1016. package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/apply.py +0 -165
  1017. package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/refine.py +0 -66
  1018. package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/report.py +0 -62
  1019. package/dist/agent-src/templates/scripts/work_engine/directives/ui_trivial/test.py +0 -115
  1020. package/dist/agent-src/templates/scripts/work_engine/dispatcher.py +0 -331
  1021. package/dist/agent-src/templates/scripts/work_engine/emitters.py +0 -68
  1022. package/dist/agent-src/templates/scripts/work_engine/errors.py +0 -19
  1023. package/dist/agent-src/templates/scripts/work_engine/hook_bootstrap.py +0 -91
  1024. package/dist/agent-src/templates/scripts/work_engine/hooks/__init__.py +0 -54
  1025. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/__init__.py +0 -35
  1026. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/_chat_history_base.py +0 -59
  1027. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/chat_history_append.py +0 -43
  1028. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/chat_history_halt_append.py +0 -41
  1029. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/decision_gate.py +0 -162
  1030. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/decision_trace.py +0 -163
  1031. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/directive_set_guard.py +0 -53
  1032. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/halt_surface_audit.py +0 -50
  1033. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/memory_visibility.py +0 -141
  1034. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/state_shape_validation.py +0 -52
  1035. package/dist/agent-src/templates/scripts/work_engine/hooks/builtin/trace.py +0 -84
  1036. package/dist/agent-src/templates/scripts/work_engine/hooks/context.py +0 -66
  1037. package/dist/agent-src/templates/scripts/work_engine/hooks/events.py +0 -44
  1038. package/dist/agent-src/templates/scripts/work_engine/hooks/exceptions.py +0 -79
  1039. package/dist/agent-src/templates/scripts/work_engine/hooks/registry.py +0 -60
  1040. package/dist/agent-src/templates/scripts/work_engine/hooks/runner.py +0 -73
  1041. package/dist/agent-src/templates/scripts/work_engine/hooks/settings.py +0 -196
  1042. package/dist/agent-src/templates/scripts/work_engine/input_builders.py +0 -163
  1043. package/dist/agent-src/templates/scripts/work_engine/intent/__init__.py +0 -47
  1044. package/dist/agent-src/templates/scripts/work_engine/intent/classify.py +0 -280
  1045. package/dist/agent-src/templates/scripts/work_engine/migration/__init__.py +0 -8
  1046. package/dist/agent-src/templates/scripts/work_engine/migration/v0_to_v1.py +0 -231
  1047. package/dist/agent-src/templates/scripts/work_engine/orchestration.py +0 -193
  1048. package/dist/agent-src/templates/scripts/work_engine/persona_policy.py +0 -85
  1049. package/dist/agent-src/templates/scripts/work_engine/resolvers/__init__.py +0 -22
  1050. package/dist/agent-src/templates/scripts/work_engine/resolvers/diff.py +0 -106
  1051. package/dist/agent-src/templates/scripts/work_engine/resolvers/file.py +0 -113
  1052. package/dist/agent-src/templates/scripts/work_engine/resolvers/prompt.py +0 -90
  1053. package/dist/agent-src/templates/scripts/work_engine/scoring/__init__.py +0 -14
  1054. package/dist/agent-src/templates/scripts/work_engine/scoring/confidence.py +0 -300
  1055. package/dist/agent-src/templates/scripts/work_engine/scoring/decision_engine.py +0 -351
  1056. package/dist/agent-src/templates/scripts/work_engine/scoring/decision_trace.py +0 -141
  1057. package/dist/agent-src/templates/scripts/work_engine/scoring/memory_visibility.py +0 -283
  1058. package/dist/agent-src/templates/scripts/work_engine/stack/__init__.py +0 -31
  1059. package/dist/agent-src/templates/scripts/work_engine/stack/detect.py +0 -187
  1060. package/dist/agent-src/templates/scripts/work_engine/stack/runner.py +0 -481
  1061. package/dist/agent-src/templates/scripts/work_engine/state.py +0 -694
  1062. package/dist/agent-src/templates/scripts/work_engine/state_io.py +0 -202
  1063. package/dist/cli/python/resolvePython.js +0 -38
  1064. package/dist/cli/python/resolvePython.js.map +0 -1
  1065. package/src/scripts/__pycache__/validate_frontmatter.cpython-312.pyc +0 -0
  1066. package/src/scripts/_archive/_backfill_skill_domains.py +0 -140
  1067. package/src/scripts/_archive/_bootstrap_tier_frontmatter.py +0 -151
  1068. package/src/scripts/_archive/_p43_bodies.py +0 -235
  1069. package/src/scripts/_archive/_p43_condense.py +0 -118
  1070. package/src/scripts/_archive/_p4_migrate.py +0 -199
  1071. package/src/scripts/_archive/_phase2_shim_helper.py +0 -109
  1072. package/src/scripts/_archive/_pilot_council_question.py +0 -57
  1073. package/src/scripts/_cli/__init__.py +0 -0
  1074. package/src/scripts/_cli/cmd_doctor.py +0 -1669
  1075. package/src/scripts/_cli/cmd_explain.py +0 -355
  1076. package/src/scripts/_cli/cmd_export.py +0 -157
  1077. package/src/scripts/_cli/cmd_migrate.py +0 -524
  1078. package/src/scripts/_cli/cmd_prune.py +0 -322
  1079. package/src/scripts/_cli/cmd_refresh.py +0 -179
  1080. package/src/scripts/_cli/cmd_settings_check.py +0 -171
  1081. package/src/scripts/_cli/cmd_settings_migrate.py +0 -147
  1082. package/src/scripts/_cli/cmd_sync.py +0 -166
  1083. package/src/scripts/_cli/cmd_uninstall.py +0 -476
  1084. package/src/scripts/_cli/cmd_update.py +0 -279
  1085. package/src/scripts/_cli/cmd_upgrade.py +0 -172
  1086. package/src/scripts/_cli/cmd_validate.py +0 -177
  1087. package/src/scripts/_cli/cmd_versions.py +0 -160
  1088. package/src/scripts/_cli/explain_last/__init__.py +0 -122
  1089. package/src/scripts/_cli/explain_last/assumptions.py +0 -59
  1090. package/src/scripts/_cli/explain_last/council.py +0 -105
  1091. package/src/scripts/_cli/explain_last/halt.py +0 -44
  1092. package/src/scripts/_cli/explain_last/inputs.py +0 -128
  1093. package/src/scripts/_cli/explain_last/memory.py +0 -94
  1094. package/src/scripts/_cli/explain_last/provider.py +0 -52
  1095. package/src/scripts/_cli/explain_last/render.py +0 -52
  1096. package/src/scripts/_cli/explain_last/route.py +0 -59
  1097. package/src/scripts/_cli/explain_last/scrubber.py +0 -105
  1098. package/src/scripts/_cli/explain_last/sections/__init__.py +0 -35
  1099. package/src/scripts/_cli/explain_last/sections/assumptions.py +0 -21
  1100. package/src/scripts/_cli/explain_last/sections/council.py +0 -27
  1101. package/src/scripts/_cli/explain_last/sections/halt.py +0 -31
  1102. package/src/scripts/_cli/explain_last/sections/header.py +0 -24
  1103. package/src/scripts/_cli/explain_last/sections/inputs.py +0 -27
  1104. package/src/scripts/_cli/explain_last/sections/memory.py +0 -21
  1105. package/src/scripts/_cli/explain_last/sections/pack.py +0 -16
  1106. package/src/scripts/_cli/explain_last/sections/provider.py +0 -26
  1107. package/src/scripts/_cli/explain_last/sections/route.py +0 -22
  1108. package/src/scripts/_cli/explain_last/state_loader.py +0 -76
  1109. package/src/scripts/_emit_domain_table.py +0 -35
  1110. package/src/scripts/_lib/__init__.py +0 -5
  1111. package/src/scripts/_lib/__pycache__/__init__.cpython-312.pyc +0 -0
  1112. package/src/scripts/_lib/__pycache__/agent_src.cpython-312.pyc +0 -0
  1113. package/src/scripts/_lib/agent_settings.py +0 -840
  1114. package/src/scripts/_lib/agent_src.py +0 -491
  1115. package/src/scripts/_lib/agents_overlay.py +0 -120
  1116. package/src/scripts/_lib/bench_ab_cache.py +0 -162
  1117. package/src/scripts/_lib/bench_ab_scoring.py +0 -209
  1118. package/src/scripts/_lib/bench_ab_scoring_v2.py +0 -227
  1119. package/src/scripts/_lib/bench_cost.py +0 -138
  1120. package/src/scripts/_lib/bench_quality.py +0 -118
  1121. package/src/scripts/_lib/bench_report.py +0 -149
  1122. package/src/scripts/_lib/bench_telegraph.py +0 -273
  1123. package/src/scripts/_lib/bench_telegraph_report.py +0 -151
  1124. package/src/scripts/_lib/changelog_eras.py +0 -330
  1125. package/src/scripts/_lib/claude_desktop_bundler.py +0 -238
  1126. package/src/scripts/_lib/cli_wrapper.py +0 -64
  1127. package/src/scripts/_lib/fs_atomic.py +0 -116
  1128. package/src/scripts/_lib/global_deploy_inventory.py +0 -312
  1129. package/src/scripts/_lib/install_regenerator.py +0 -134
  1130. package/src/scripts/_lib/installed_lock.py +0 -256
  1131. package/src/scripts/_lib/installed_tools.py +0 -381
  1132. package/src/scripts/_lib/json_pointers.py +0 -260
  1133. package/src/scripts/_lib/link_crypto.py +0 -206
  1134. package/src/scripts/_lib/linked_projects.py +0 -238
  1135. package/src/scripts/_lib/model_tier.py +0 -52
  1136. package/src/scripts/_lib/module_detection.py +0 -223
  1137. package/src/scripts/_lib/pin_resolver.py +0 -152
  1138. package/src/scripts/_lib/script_output.py +0 -144
  1139. package/src/scripts/_lib/security_lint.py +0 -228
  1140. package/src/scripts/_lib/token_count.py +0 -95
  1141. package/src/scripts/_lib/update_check.py +0 -207
  1142. package/src/scripts/_lib/user_global_paths.py +0 -249
  1143. package/src/scripts/_lib/value_ladder.py +0 -696
  1144. package/src/scripts/_lib/value_report.py +0 -455
  1145. package/src/scripts/_phase4_bucket.py +0 -210
  1146. package/src/scripts/_pilot_measure.py +0 -53
  1147. package/src/scripts/_tmp_scan_framework_leakage.py +0 -119
  1148. package/src/scripts/adoption_report.py +0 -195
  1149. package/src/scripts/adoption_snapshot.py +0 -219
  1150. package/src/scripts/adoption_status.py +0 -166
  1151. package/src/scripts/adr/regenerate_index.py +0 -79
  1152. package/src/scripts/ai-video/lib/adapter-common.sh +0 -231
  1153. package/src/scripts/ai-video/lib/adapter-contract.md +0 -329
  1154. package/src/scripts/ai-video/lib/fixtures/comfyui/result.json +0 -1
  1155. package/src/scripts/ai-video/lib/fixtures/fal/result.json +0 -1
  1156. package/src/scripts/ai-video/lib/fixtures/gemini-veo/result.json +0 -1
  1157. package/src/scripts/ai-video/lib/fixtures/higgsfield/result.json +0 -1
  1158. package/src/scripts/ai-video/lib/fixtures/kling/result.json +0 -1
  1159. package/src/scripts/ai-video/lib/fixtures/musetalk/result.json +0 -1
  1160. package/src/scripts/ai-video/lib/fixtures/openai-images/result.json +0 -1
  1161. package/src/scripts/ai-video/lib/fixtures/replicate/result.json +0 -1
  1162. package/src/scripts/ai-video/lib/fixtures/sora/result.json +0 -1
  1163. package/src/scripts/ai-video/lib/fixtures/syncso/result.json +0 -1
  1164. package/src/scripts/ai-video/lib/load-config.sh +0 -180
  1165. package/src/scripts/ai-video/lib/redact.sh +0 -85
  1166. package/src/scripts/ai_council/__init__.py +0 -40
  1167. package/src/scripts/ai_council/_default_prices.py +0 -50
  1168. package/src/scripts/ai_council/advisors.py +0 -148
  1169. package/src/scripts/ai_council/airgap.py +0 -165
  1170. package/src/scripts/ai_council/budget_guard.py +0 -202
  1171. package/src/scripts/ai_council/bundler.py +0 -263
  1172. package/src/scripts/ai_council/cli_hints.py +0 -123
  1173. package/src/scripts/ai_council/clients.py +0 -1385
  1174. package/src/scripts/ai_council/compile_corpus.py +0 -179
  1175. package/src/scripts/ai_council/confidence_gate.py +0 -156
  1176. package/src/scripts/ai_council/config.py +0 -1419
  1177. package/src/scripts/ai_council/consensus.py +0 -329
  1178. package/src/scripts/ai_council/events_log.py +0 -141
  1179. package/src/scripts/ai_council/learn_low_impact_preview.py +0 -252
  1180. package/src/scripts/ai_council/low_impact.py +0 -714
  1181. package/src/scripts/ai_council/low_impact_corpus.py +0 -466
  1182. package/src/scripts/ai_council/low_impact_intake.py +0 -163
  1183. package/src/scripts/ai_council/modes.py +0 -131
  1184. package/src/scripts/ai_council/necessity.py +0 -782
  1185. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_2a4_acceptance.py +0 -208
  1186. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_add_quiet.py +0 -149
  1187. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_budget_v2_audit.py +0 -206
  1188. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_context_layer_v1_estimate.py +0 -67
  1189. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_context_layer_v1_review.py +0 -292
  1190. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_followups_review.py +0 -259
  1191. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_inject_quiet_flag.py +0 -33
  1192. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_measure_v2.sh +0 -36
  1193. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_measure_verbosity.sh +0 -26
  1194. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_nondestructive_inline_audit.py +0 -209
  1195. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_per_task.sh +0 -41
  1196. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_phase4_dispatch_latency.py +0 -108
  1197. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_phase6_trigger_jaccard.py +0 -92
  1198. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_phase_2a_budget_rebalance.py +0 -257
  1199. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_phase_2a_post_revert.py +0 -197
  1200. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_rebalancing_audit.py +0 -149
  1201. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_roundtrip.py +0 -111
  1202. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_rule_hardening_v1.py +0 -251
  1203. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_silent_taskfiles.py +0 -98
  1204. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_structural_open_questions.py +0 -232
  1205. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_structural_optimization.py +0 -144
  1206. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_structural_v3_gaps.py +0 -252
  1207. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_structural_v3_review.py +0 -240
  1208. package/src/scripts/ai_council/one_off_archive/2026-05/_one_off_tier_retrofit.py +0 -180
  1209. package/src/scripts/ai_council/orchestrator.py +0 -1206
  1210. package/src/scripts/ai_council/pricing.py +0 -215
  1211. package/src/scripts/ai_council/probation_gate.py +0 -152
  1212. package/src/scripts/ai_council/project_context.py +0 -159
  1213. package/src/scripts/ai_council/prompts.py +0 -567
  1214. package/src/scripts/ai_council/redact_low_impact_entry.py +0 -155
  1215. package/src/scripts/ai_council/replay.py +0 -155
  1216. package/src/scripts/ai_council/session.py +0 -366
  1217. package/src/scripts/ai_council/shadow_dispatch.py +0 -235
  1218. package/src/scripts/ai_council/solo_dispatch.py +0 -226
  1219. package/src/scripts/annotate_discovery.py +0 -149
  1220. package/src/scripts/apply_modules_config.py +0 -290
  1221. package/src/scripts/audit_adr_coverage.py +0 -173
  1222. package/src/scripts/audit_auto_rules.py +0 -175
  1223. package/src/scripts/audit_cloud_compatibility.py +0 -362
  1224. package/src/scripts/audit_command_surface.py +0 -682
  1225. package/src/scripts/audit_initial_context.py +0 -237
  1226. package/src/scripts/audit_likelihood.py +0 -148
  1227. package/src/scripts/audit_mcp_tools.py +0 -146
  1228. package/src/scripts/audit_overlap.py +0 -145
  1229. package/src/scripts/audit_skill_descriptions.py +0 -180
  1230. package/src/scripts/audit_skill_overlap.py +0 -207
  1231. package/src/scripts/audit_user_type_axis.py +0 -140
  1232. package/src/scripts/backfill_model_tier.py +0 -184
  1233. package/src/scripts/bench_ab_cache_dispatch.py +0 -68
  1234. package/src/scripts/bench_ab_clone.py +0 -220
  1235. package/src/scripts/bench_ab_diff.py +0 -220
  1236. package/src/scripts/bench_ab_integrity.py +0 -143
  1237. package/src/scripts/bench_ab_run.py +0 -235
  1238. package/src/scripts/bench_ab_task_runner.py +0 -814
  1239. package/src/scripts/bench_ab_tracka_run.py +0 -202
  1240. package/src/scripts/bench_ab_v2_run.py +0 -247
  1241. package/src/scripts/bench_ab_v2_stats.py +0 -347
  1242. package/src/scripts/bench_baseline_ready.py +0 -108
  1243. package/src/scripts/bench_condense_memory.py +0 -168
  1244. package/src/scripts/bench_drift_check.py +0 -151
  1245. package/src/scripts/bench_per_tool.py +0 -216
  1246. package/src/scripts/bench_rtk_savings.py +0 -320
  1247. package/src/scripts/bench_run.py +0 -272
  1248. package/src/scripts/bench_runner.py +0 -158
  1249. package/src/scripts/build_cloud_bundle.py +0 -458
  1250. package/src/scripts/build_discovery_manifest.py +0 -757
  1251. package/src/scripts/build_linear_digest.py +0 -260
  1252. package/src/scripts/build_mcp_registry_manifest.py +0 -181
  1253. package/src/scripts/build_rule_trigger_matrix.py +0 -350
  1254. package/src/scripts/capture_showcase_session.py +0 -361
  1255. package/src/scripts/chat_history.py +0 -1799
  1256. package/src/scripts/check_always_budget.py +0 -532
  1257. package/src/scripts/check_artefact_checksums.py +0 -111
  1258. package/src/scripts/check_augment_description_cap.py +0 -79
  1259. package/src/scripts/check_augmentignore.py +0 -72
  1260. package/src/scripts/check_beta_review_markers.py +0 -127
  1261. package/src/scripts/check_bite_sized_granularity.py +0 -98
  1262. package/src/scripts/check_cluster_patterns.py +0 -206
  1263. package/src/scripts/check_command_count_messaging.py +0 -152
  1264. package/src/scripts/check_condensation.py +0 -375
  1265. package/src/scripts/check_condensed_paths.py +0 -231
  1266. package/src/scripts/check_context_paths.py +0 -202
  1267. package/src/scripts/check_council_layout.py +0 -125
  1268. package/src/scripts/check_council_references.py +0 -228
  1269. package/src/scripts/check_discovery_determinism.py +0 -70
  1270. package/src/scripts/check_gate_paths.py +0 -128
  1271. package/src/scripts/check_iron_law_prominence.py +0 -145
  1272. package/src/scripts/check_kernel_rule_bundle.py +0 -151
  1273. package/src/scripts/check_md_language.py +0 -161
  1274. package/src/scripts/check_memory.py +0 -429
  1275. package/src/scripts/check_memory_proposal.py +0 -182
  1276. package/src/scripts/check_module_management_neutral.py +0 -147
  1277. package/src/scripts/check_no_external_sources.py +0 -101
  1278. package/src/scripts/check_no_local_settings_committed.py +0 -51
  1279. package/src/scripts/check_no_new_legacy_path.py +0 -100
  1280. package/src/scripts/check_no_roadmap_refs.py +0 -155
  1281. package/src/scripts/check_one_off_location.py +0 -81
  1282. package/src/scripts/check_overlay_cascade_subdirs.py +0 -129
  1283. package/src/scripts/check_portability.py +0 -574
  1284. package/src/scripts/check_proposal.py +0 -269
  1285. package/src/scripts/check_public_catalog_links.py +0 -125
  1286. package/src/scripts/check_public_links.py +0 -185
  1287. package/src/scripts/check_references.py +0 -559
  1288. package/src/scripts/check_release_includes_discovery.py +0 -61
  1289. package/src/scripts/check_release_pr_shape.py +0 -123
  1290. package/src/scripts/check_release_published.py +0 -145
  1291. package/src/scripts/check_release_trunk_sync.py +0 -152
  1292. package/src/scripts/check_reply_consistency.py +0 -169
  1293. package/src/scripts/check_roadmap_trackable.py +0 -114
  1294. package/src/scripts/check_role_doc_links.py +0 -110
  1295. package/src/scripts/check_safety_floor_untouched.py +0 -125
  1296. package/src/scripts/check_skill_requires.py +0 -147
  1297. package/src/scripts/check_template_pin_drift.py +0 -129
  1298. package/src/scripts/check_test_coverage_diff.py +0 -180
  1299. package/src/scripts/check_token_optimizer_freshness.py +0 -146
  1300. package/src/scripts/check_update_banner.py +0 -86
  1301. package/src/scripts/ci_status.py +0 -301
  1302. package/src/scripts/ci_summary.py +0 -131
  1303. package/src/scripts/ci_time_ratio.py +0 -168
  1304. package/src/scripts/command_suggester/__init__.py +0 -51
  1305. package/src/scripts/command_suggester/cooldown.py +0 -132
  1306. package/src/scripts/command_suggester/loader.py +0 -73
  1307. package/src/scripts/command_suggester/match.py +0 -180
  1308. package/src/scripts/command_suggester/rank.py +0 -120
  1309. package/src/scripts/command_suggester/render.py +0 -86
  1310. package/src/scripts/command_suggester/sanitize.py +0 -113
  1311. package/src/scripts/command_suggester/settings.py +0 -127
  1312. package/src/scripts/command_suggester/types.py +0 -78
  1313. package/src/scripts/compile_router.py +0 -232
  1314. package/src/scripts/condense.py +0 -1919
  1315. package/src/scripts/condense_memory.py +0 -178
  1316. package/src/scripts/config/__init__.py +0 -9
  1317. package/src/scripts/config/packs.py +0 -157
  1318. package/src/scripts/config/presets.py +0 -224
  1319. package/src/scripts/config/profile_explain.py +0 -89
  1320. package/src/scripts/config/profiles.py +0 -191
  1321. package/src/scripts/config/session_profiles.py +0 -542
  1322. package/src/scripts/context_hygiene_hook.py +0 -181
  1323. package/src/scripts/cost_by_conversation.py +0 -78
  1324. package/src/scripts/cost_summary.py +0 -97
  1325. package/src/scripts/council_cli.py +0 -2571
  1326. package/src/scripts/council_prune.py +0 -81
  1327. package/src/scripts/cross_repo_retrieve.py +0 -172
  1328. package/src/scripts/discovery_stats.py +0 -70
  1329. package/src/scripts/extract_audit_patterns.py +0 -202
  1330. package/src/scripts/first_run_gate_hook.py +0 -179
  1331. package/src/scripts/gen_discovery_baseline.py +0 -127
  1332. package/src/scripts/generate_catalog.py +0 -116
  1333. package/src/scripts/generate_command_flows.py +0 -191
  1334. package/src/scripts/generate_index.py +0 -303
  1335. package/src/scripts/generate_ownership_matrix.py +0 -378
  1336. package/src/scripts/generate_pack_manifests.py +0 -340
  1337. package/src/scripts/hooks/__init__.py +0 -1
  1338. package/src/scripts/hooks/dispatch_hook.py +0 -461
  1339. package/src/scripts/hooks/dispatch_issues.py +0 -136
  1340. package/src/scripts/hooks/envelope.py +0 -98
  1341. package/src/scripts/hooks/replay_hook.py +0 -144
  1342. package/src/scripts/hooks/state_io.py +0 -145
  1343. package/src/scripts/hooks_doctor.py +0 -223
  1344. package/src/scripts/hooks_status.py +0 -157
  1345. package/src/scripts/injection_scan_hook.py +0 -145
  1346. package/src/scripts/install.py +0 -5258
  1347. package/src/scripts/inventory_abstraction_budget.py +0 -622
  1348. package/src/scripts/inventory_frontmatter.py +0 -164
  1349. package/src/scripts/inventory_meta_layers.py +0 -288
  1350. package/src/scripts/iron_law_sha.py +0 -107
  1351. package/src/scripts/linked_projects_list.py +0 -91
  1352. package/src/scripts/lint_agent_security.py +0 -112
  1353. package/src/scripts/lint_agent_skill_names.py +0 -150
  1354. package/src/scripts/lint_agents_layout.py +0 -197
  1355. package/src/scripts/lint_agents_md.py +0 -210
  1356. package/src/scripts/lint_archived_skills.py +0 -159
  1357. package/src/scripts/lint_artefact_frontmatter.py +0 -188
  1358. package/src/scripts/lint_bench_ab.py +0 -173
  1359. package/src/scripts/lint_bench_corpus.py +0 -255
  1360. package/src/scripts/lint_command_flow_coverage.py +0 -132
  1361. package/src/scripts/lint_command_routing.py +0 -160
  1362. package/src/scripts/lint_command_tiers.py +0 -216
  1363. package/src/scripts/lint_command_verbs.py +0 -206
  1364. package/src/scripts/lint_commit_subjects.py +0 -139
  1365. package/src/scripts/lint_context_spine_usage.py +0 -137
  1366. package/src/scripts/lint_discovery_manifest.py +0 -176
  1367. package/src/scripts/lint_discovery_vocabulary.py +0 -222
  1368. package/src/scripts/lint_empty_roadmaps.py +0 -80
  1369. package/src/scripts/lint_examples.py +0 -102
  1370. package/src/scripts/lint_explain_trace.py +0 -80
  1371. package/src/scripts/lint_featured_skills.py +0 -144
  1372. package/src/scripts/lint_flows.py +0 -215
  1373. package/src/scripts/lint_framework_leakage.py +0 -375
  1374. package/src/scripts/lint_frontmatter_boilerplate.py +0 -77
  1375. package/src/scripts/lint_ghostwriter_source.py +0 -242
  1376. package/src/scripts/lint_global_paths.py +0 -148
  1377. package/src/scripts/lint_handoffs.py +0 -217
  1378. package/src/scripts/lint_hidden_unicode.py +0 -132
  1379. package/src/scripts/lint_hook_concern_budget.py +0 -207
  1380. package/src/scripts/lint_hook_manifest.py +0 -217
  1381. package/src/scripts/lint_instruction_smuggling.py +0 -107
  1382. package/src/scripts/lint_load_context.py +0 -196
  1383. package/src/scripts/lint_marketplace.py +0 -180
  1384. package/src/scripts/lint_marketplace_install_completeness.py +0 -198
  1385. package/src/scripts/lint_mcp_config_security.py +0 -124
  1386. package/src/scripts/lint_mcp_registry_manifest.py +0 -69
  1387. package/src/scripts/lint_media_policy_linkage.py +0 -140
  1388. package/src/scripts/lint_model_tier_coverage.py +0 -73
  1389. package/src/scripts/lint_namespace.py +0 -135
  1390. package/src/scripts/lint_namespace_collisions.py +0 -103
  1391. package/src/scripts/lint_new_skill_gate.py +0 -144
  1392. package/src/scripts/lint_no_new_atomic_commands.py +0 -180
  1393. package/src/scripts/lint_one_off_age.py +0 -184
  1394. package/src/scripts/lint_orchestration_dsl.py +0 -217
  1395. package/src/scripts/lint_orchestrator_auto_detect.py +0 -111
  1396. package/src/scripts/lint_pack_boundaries.py +0 -147
  1397. package/src/scripts/lint_pack_dependencies.py +0 -137
  1398. package/src/scripts/lint_pack_first_win.py +0 -121
  1399. package/src/scripts/lint_persona_governance.py +0 -164
  1400. package/src/scripts/lint_positioning.py +0 -143
  1401. package/src/scripts/lint_profile_overlay_set_only.py +0 -179
  1402. package/src/scripts/lint_readme_jargon.py +0 -131
  1403. package/src/scripts/lint_readme_size.py +0 -33
  1404. package/src/scripts/lint_regression.py +0 -251
  1405. package/src/scripts/lint_roadmap_ci_steps.py +0 -186
  1406. package/src/scripts/lint_roadmap_complexity.py +0 -220
  1407. package/src/scripts/lint_role_experiences.py +0 -255
  1408. package/src/scripts/lint_rule_interactions.py +0 -170
  1409. package/src/scripts/lint_rule_tiers.py +0 -90
  1410. package/src/scripts/lint_showcase_sessions.py +0 -148
  1411. package/src/scripts/lint_skill_frontmatter_safety.py +0 -144
  1412. package/src/scripts/lint_skill_tools.py +0 -168
  1413. package/src/scripts/lint_topics_yaml.py +0 -89
  1414. package/src/scripts/lint_trust_coherence.py +0 -212
  1415. package/src/scripts/lint_value_dashboard.py +0 -218
  1416. package/src/scripts/lint_workspace_boundary.py +0 -122
  1417. package/src/scripts/mcp_parity_smoke.py +0 -316
  1418. package/src/scripts/mcp_render.py +0 -173
  1419. package/src/scripts/mcp_server/__init__.py +0 -19
  1420. package/src/scripts/mcp_server/__main__.py +0 -12
  1421. package/src/scripts/mcp_server/catalog.py +0 -125
  1422. package/src/scripts/mcp_server/metadata.py +0 -75
  1423. package/src/scripts/mcp_server/prompts.py +0 -442
  1424. package/src/scripts/mcp_server/requirements.txt +0 -4
  1425. package/src/scripts/mcp_server/resources.py +0 -201
  1426. package/src/scripts/mcp_server/server.py +0 -270
  1427. package/src/scripts/mcp_server/telemetry.py +0 -128
  1428. package/src/scripts/mcp_server/tools.py +0 -926
  1429. package/src/scripts/mcp_telemetry_health.py +0 -214
  1430. package/src/scripts/mcp_telemetry_query.py +0 -203
  1431. package/src/scripts/mcp_telemetry_store.py +0 -211
  1432. package/src/scripts/measure_augment_budget.py +0 -214
  1433. package/src/scripts/measure_density.py +0 -232
  1434. package/src/scripts/measure_frugality_savings.py +0 -164
  1435. package/src/scripts/measure_markitdown_lift.py +0 -127
  1436. package/src/scripts/measure_patterns.py +0 -376
  1437. package/src/scripts/measure_projection_bytes.py +0 -159
  1438. package/src/scripts/measure_rule_budget.py +0 -347
  1439. package/src/scripts/measure_skill_reduction.py +0 -102
  1440. package/src/scripts/memory_hash.py +0 -75
  1441. package/src/scripts/memory_lookup.py +0 -436
  1442. package/src/scripts/memory_report.py +0 -314
  1443. package/src/scripts/memory_signal.py +0 -165
  1444. package/src/scripts/memory_status.py +0 -76
  1445. package/src/scripts/migrate_command_suggestions.py +0 -151
  1446. package/src/scripts/migrate_frontmatter_defaults.py +0 -245
  1447. package/src/scripts/mine_session.py +0 -356
  1448. package/src/scripts/minimal_safe_diff_hook.py +0 -245
  1449. package/src/scripts/move_artefact.py +0 -143
  1450. package/src/scripts/new_skill.py +0 -148
  1451. package/src/scripts/onboarding_gate_hook.py +0 -142
  1452. package/src/scripts/pack_mcp_content.py +0 -293
  1453. package/src/scripts/plan_physical_move.py +0 -353
  1454. package/src/scripts/prediction-pool/poisson_sim.py +0 -167
  1455. package/src/scripts/prediction-pool/pool_winsim.py +0 -236
  1456. package/src/scripts/prediction-pool/score_ev.py +0 -188
  1457. package/src/scripts/print_required_checks.py +0 -196
  1458. package/src/scripts/probe_projection_fidelity.py +0 -202
  1459. package/src/scripts/probe_skill_registration.py +0 -413
  1460. package/src/scripts/profile_staleness_hook.py +0 -69
  1461. package/src/scripts/profile_use.py +0 -164
  1462. package/src/scripts/project_thin_rules.py +0 -168
  1463. package/src/scripts/propose_modules_config.py +0 -145
  1464. package/src/scripts/prototype_lint_contradictions.py +0 -150
  1465. package/src/scripts/prove_pack_extractable.py +0 -187
  1466. package/src/scripts/readme_linter.py +0 -589
  1467. package/src/scripts/redact_hook_capture.py +0 -148
  1468. package/src/scripts/refine_ticket_detect.py +0 -646
  1469. package/src/scripts/release.py +0 -1091
  1470. package/src/scripts/render_benchmark_md.py +0 -401
  1471. package/src/scripts/render_value_md.py +0 -347
  1472. package/src/scripts/requirements-evals.txt +0 -8
  1473. package/src/scripts/roadmap_progress_hook.py +0 -274
  1474. package/src/scripts/router_telemetry.py +0 -470
  1475. package/src/scripts/run_skill_evals.py +0 -185
  1476. package/src/scripts/runtime_dispatcher.py +0 -276
  1477. package/src/scripts/runtime_handler.py +0 -148
  1478. package/src/scripts/runtime_registry.py +0 -166
  1479. package/src/scripts/score_skill_selection.py +0 -198
  1480. package/src/scripts/security_audit_config.py +0 -153
  1481. package/src/scripts/setup_eval_venv.sh +0 -58
  1482. package/src/scripts/skill_collision_clusters.py +0 -162
  1483. package/src/scripts/skill_discovery.py +0 -254
  1484. package/src/scripts/skill_linter.py +0 -3694
  1485. package/src/scripts/skill_overlap.py +0 -204
  1486. package/src/scripts/skill_preview.py +0 -179
  1487. package/src/scripts/skill_tools/__init__.py +0 -22
  1488. package/src/scripts/skill_tools/audit_persona_coverage.py +0 -147
  1489. package/src/scripts/skill_tools/audit_user_type_coverage.py +0 -148
  1490. package/src/scripts/skill_tools/run_block_d_eval.py +0 -129
  1491. package/src/scripts/skill_tools/score_skill_relevance.py +0 -169
  1492. package/src/scripts/skill_tools/suggest_skill_for_task.py +0 -113
  1493. package/src/scripts/skill_trigger_eval.py +0 -682
  1494. package/src/scripts/skill_usage_collect.py +0 -191
  1495. package/src/scripts/skill_usage_report.py +0 -162
  1496. package/src/scripts/smoke_path_resolution.py +0 -93
  1497. package/src/scripts/smoke_quickstart.py +0 -144
  1498. package/src/scripts/snapshot_agent_outputs.py +0 -144
  1499. package/src/scripts/spotcheck_thin_root.py +0 -134
  1500. package/src/scripts/sync_agent_settings.py +0 -180
  1501. package/src/scripts/sync_github_metadata.py +0 -147
  1502. package/src/scripts/sync_gitignore.py +0 -291
  1503. package/src/scripts/sync_yaml_rt.py +0 -734
  1504. package/src/scripts/telegraph_stats.py +0 -119
  1505. package/src/scripts/tool_registry.py +0 -146
  1506. package/src/scripts/tools/__init__.py +0 -1
  1507. package/src/scripts/tools/adapter_errors.py +0 -63
  1508. package/src/scripts/tools/base_adapter.py +0 -91
  1509. package/src/scripts/tools/github_adapter.py +0 -128
  1510. package/src/scripts/tools/jira_adapter.py +0 -115
  1511. package/src/scripts/trigger_coverage.py +0 -129
  1512. package/src/scripts/update_counts.py +0 -199
  1513. package/src/scripts/update_prices.py +0 -125
  1514. package/src/scripts/validate_agent_settings.py +0 -124
  1515. package/src/scripts/validate_decision_engine.py +0 -136
  1516. package/src/scripts/validate_discovery_manifest.py +0 -94
  1517. package/src/scripts/validate_frontmatter.py +0 -647
  1518. package/src/scripts/validate_pack_yaml.py +0 -179
  1519. package/src/scripts/validate_safe_paths.py +0 -118
  1520. package/src/scripts/validate_telegraph_carveouts.py +0 -129
  1521. package/src/scripts/verify_before_complete_hook.py +0 -216
  1522. package/src/scripts/verify_physical_move.py +0 -185
  1523. package/src/scripts/wrapper_freshness_hook.py +0 -86
  1524. /package/dist/agent-src/skills/design-intelligence/data/{typography.csv → font-pairings-reference.csv} +0 -0
  1525. /package/src/scripts/{ai-video → media}/lib/fixtures/allin1/analysis.json +0 -0
  1526. /package/src/scripts/{ai-video → media}/lib/fixtures/comfyui/scene-0001.mp4 +0 -0
  1527. /package/src/scripts/{ai-video → media}/lib/fixtures/fal/scene-0001.mp4 +0 -0
  1528. /package/src/scripts/{ai-video → media}/lib/fixtures/gemini-veo/scene-0001.mp4 +0 -0
  1529. /package/src/scripts/{ai-video → media}/lib/fixtures/higgsfield/scene-0001.mp4 +0 -0
  1530. /package/src/scripts/{ai-video → media}/lib/fixtures/kling/scene-0001.mp4 +0 -0
  1531. /package/src/scripts/{ai-video → media}/lib/fixtures/musetalk/lipsync-0001.mp4 +0 -0
  1532. /package/src/scripts/{ai-video → media}/lib/fixtures/openai-images/scene-0001.png +0 -0
  1533. /package/src/scripts/{ai-video → media}/lib/fixtures/replicate/scene-0001.mp4 +0 -0
  1534. /package/src/scripts/{ai-video → media}/lib/fixtures/sora/scene-0001.mp4 +0 -0
  1535. /package/src/scripts/{ai-video → media}/lib/fixtures/syncso/lipsync-0001.mp4 +0 -0
  1536. /package/src/scripts/{ai-video → media}/lib/fixtures/whisperx/transcript.json +0 -0
  1537. /package/src/scripts/{ai-video → media}/lib/telemetry.sh +0 -0
@@ -1,2571 +0,0 @@
1
- #!/usr/bin/env python3
2
- """Council CLI — `./agent-config council:{estimate,run,render}`.
3
-
4
- Wraps `scripts.ai_council.orchestrator` for non-interactive callers.
5
- Subcommands:
6
-
7
- estimate Bundle + estimate per-member cost (no API call, no spend).
8
- run Same + estimate, then call the council. Requires --confirm.
9
- render Re-render a saved responses JSON to the markdown report.
10
-
11
- `./agent-config` is non-interactive by contract — the cost gate is an
12
- explicit `--confirm` flag, never an interactive y/n.
13
- """
14
- from __future__ import annotations
15
-
16
- import argparse
17
- import json
18
- import sys
19
- from dataclasses import asdict
20
- from pathlib import Path
21
- try: # invocation-agnostic import (repo-root-on-path vs scripts-on-path)
22
- from scripts._lib.agent_settings import (
23
- project_settings_path, resolve_project_root,
24
- )
25
- except ModuleNotFoundError: # pragma: no cover
26
- from _lib.agent_settings import (
27
- project_settings_path, resolve_project_root,
28
- )
29
- from typing import Any
30
-
31
- import yaml
32
-
33
- # `PACKAGE_ROOT` is where `scripts.ai_council.*` lives — fixed relative to
34
- # this file, used only for the import path below. `REPO_ROOT` is the project
35
- # the council operates on: when invoked via the global `agent-config`
36
- # wrapper from a consumer project it is that project (anchor-walked from
37
- # CWD, or pinned via AGENT_CONFIG_PROJECT_ROOT / --root), NOT the package
38
- # install dir.
39
- #
40
- # `AI_COUNCIL_FILE` is the council config the CLI reads. Resolution is
41
- # user-global-first (see `resolve_config_path`): an explicit
42
- # `$AI_COUNCIL_CONFIG` wins, else a project-local
43
- # `agents/settings/.ai-council.yml` if checked in, else the canonical
44
- # per-user `~/.event4u/agent-config/settings/.ai-council.yml`. A single developer
45
- # configures the council once globally and it applies in every project —
46
- # the project no longer needs (or should keep) its own copy.
47
- PACKAGE_ROOT = Path(__file__).resolve().parents[1]
48
- REPO_ROOT, _ = resolve_project_root(None)
49
- SETTINGS_FILE = project_settings_path(REPO_ROOT)
50
- # `AI_COUNCIL_FILE` is resolved below, after `sys.path` is set up and
51
- # `resolve_config_path` is importable (it lives in `scripts.ai_council.config`,
52
- # which itself imports `scripts._lib`).
53
-
54
- # Canonical output dirs per ai-council § "Output path convention".
55
- # Enforced at write-time by `_validate_council_output_path` so shell-side
56
- # `>` redirects and forgetful agents can't strand artefacts at agents/ root.
57
- COUNCIL_CANONICAL_DIRS: dict[str, str] = {
58
- "responses": "agents/runtime/council/responses",
59
- "sessions": "agents/runtime/council/sessions",
60
- "questions": "agents/runtime/council/questions",
61
- }
62
-
63
-
64
- def _validate_council_output_path(
65
- path_str: str, *, kind: str, subcommand: str,
66
- ) -> Path:
67
- """Reject non-canonical --output paths at write-time.
68
-
69
- `kind` selects the expected canonical dir (`responses`, `sessions`,
70
- `questions`). Raises `argparse.ArgumentTypeError` on violation so
71
- `main()` surfaces a clean ❌ message and returns 2.
72
- """
73
- expected_rel = COUNCIL_CANONICAL_DIRS[kind]
74
- expected_abs = (REPO_ROOT / expected_rel).resolve()
75
- p = Path(path_str)
76
- target = p if p.is_absolute() else (REPO_ROOT / p)
77
- target_resolved = target.resolve()
78
- try:
79
- target_resolved.relative_to(expected_abs)
80
- except ValueError as exc:
81
- raise argparse.ArgumentTypeError(
82
- f"council:{subcommand} --output must live under "
83
- f"{expected_rel}/ (per ai-council § Output path convention); "
84
- f"got {path_str!r}."
85
- ) from exc
86
- return p
87
-
88
- sys.path.insert(0, str(PACKAGE_ROOT))
89
-
90
- from scripts.ai_council.bundler import ( # noqa: E402
91
- BundleTooLarge, bundle_prompt, bundle_roadmap,
92
- )
93
- from scripts.ai_council.clients import ( # noqa: E402
94
- DEFAULT_MAX_TOKENS, UNLIMITED_TOKENS_FALLBACK,
95
- AnthropicClient, AnthropicCliClient, CliClient, CliClientError,
96
- CouncilResponse, ExternalAIClient, GeminiClient, GeminiCliClient,
97
- ManualClient, OpenAIClient, OpenAICliClient, PerplexityClient,
98
- PerplexityCliClient, XAIClient, XAICliClient,
99
- load_anthropic_key, load_cli_call_counts, load_openai_key,
100
- quota_summary_line, reset_cli_call_counts,
101
- )
102
- from scripts.ai_council.advisors import ( # noqa: E402
103
- AdvisorPlan, build_persona_labels, plan_advisor_swap,
104
- )
105
- from scripts.ai_council.cli_hints import format_install_hints # noqa: E402
106
- from scripts.ai_council.config import ( # noqa: E402
107
- AdvisorConfig, CouncilConfig, CouncilConfigError,
108
- load_council_config, resolve_api_key, resolve_config_path,
109
- )
110
-
111
- # User-global-first resolution: explicit `$AI_COUNCIL_CONFIG`, else a
112
- # project-local `agents/settings/.ai-council.yml`, else the canonical
113
- # `~/.event4u/agent-config/settings/.ai-council.yml`. Computed here (not at the
114
- # REPO_ROOT block above) so `resolve_config_path` is importable.
115
- AI_COUNCIL_FILE = resolve_config_path(REPO_ROOT) # noqa: E402
116
- from scripts.ai_council.solo_dispatch import ( # noqa: E402
117
- AuthCache, select_solo_member,
118
- )
119
- from scripts.ai_council.modes import ( # noqa: E402
120
- InvalidModeError, resolve_mode,
121
- )
122
- from scripts.ai_council.events_log import append_event # noqa: E402
123
- from scripts.ai_council.necessity import ( # noqa: E402
124
- ClassificationResult, SizeFitVerdict, classify_necessity,
125
- classify_size_fit, downgrade_message, educate_message,
126
- )
127
- from scripts.ai_council.orchestrator import ( # noqa: E402
128
- ConsensusResult,
129
- CostBudget, CouncilQuestion, DebateCapExceeded, DebateCheckpoint,
130
- DebateCostEstimate,
131
- PeerReviewResult, consult, estimate, estimate_debate_cost, render,
132
- run_consensus_scoring, run_debate, run_peer_review,
133
- )
134
- from scripts.ai_council.pricing import ( # noqa: E402
135
- PriceTable, estimate_cost, load_prices,
136
- )
137
- from scripts.ai_council.project_context import detect_project_context # noqa: E402
138
- from scripts.ai_council.replay import ( # noqa: E402
139
- DecisionReplayInputs, render_decision_replay,
140
- )
141
-
142
- SCHEMA_VERSION = 1
143
-
144
- #: Provider names accepted under `mode=api`. Mirrors the routing table
145
- #: in ``_construct_api_member``; both must stay in sync.
146
- _API_PROVIDERS = frozenset({"anthropic", "openai", "gemini", "xai", "perplexity"})
147
-
148
- #: Provider names with a wired ``mode=cli`` subclass. Mirrors the
149
- #: routing table in ``_construct_cli_member``; both must stay in sync.
150
- #: Phase 2 ships ``anthropic``; Phase 3 adds ``openai`` + ``gemini``;
151
- #: Phase 4 adds ``xai`` + ``perplexity`` (community CLIs, no
152
- #: subscription savings — they still consume the API key and remain
153
- #: ``billable=True``).
154
- _CLI_PROVIDERS = frozenset({"anthropic", "openai", "gemini", "xai", "perplexity"})
155
-
156
-
157
- class CouncilDisabledError(RuntimeError):
158
- """Raised when ai_council.enabled is false or no member is enabled."""
159
-
160
-
161
- def load_settings(
162
- path: Path = SETTINGS_FILE,
163
- *,
164
- ai_council_path: Path = AI_COUNCIL_FILE,
165
- ) -> dict[str, Any]:
166
- """Load merged settings via the centralized loader.
167
-
168
- road-to-portable-dev-preferences P3 migration: tolerance contract
169
- (missing file / malformed YAML / no PyYAML) is handled uniformly by
170
- ``load_agent_settings``. ``ai_council.*`` keys are not whitelisted,
171
- so the project file remains authoritative for council config.
172
-
173
- Step-2 council-redesign overlay: when ``agents/settings/.ai-council.yml``
174
- exists it is the single source of truth — the validated config is
175
- synthesized back into ``settings['ai_council']`` and wins over any
176
- legacy block in ``.agent-settings.yml``. The pre-2 path stays alive
177
- so the migration breadcrumb in ``.agent-settings.yml`` can ship
178
- independently.
179
- """
180
- from scripts._lib.agent_settings import load_agent_settings
181
- settings = load_agent_settings(project_path=path)
182
- if ai_council_path.exists():
183
- cfg = load_council_config(ai_council_path)
184
- settings["ai_council"] = _synthesize_ai_council_block(cfg)
185
- return settings
186
-
187
-
188
- def _synthesize_ai_council_block(cfg: CouncilConfig) -> dict[str, Any]:
189
- """Project a validated ``CouncilConfig`` onto the legacy dict shape.
190
-
191
- ``build_members`` and the ``_resolve_*`` helpers read the legacy
192
- ``ai_council.*`` keys — keeping the projection identical means no
193
- downstream caller changes. ``api_key_ref`` is carried through; raw
194
- keys are never resolved here (resolution is lazy, per enabled
195
- member, inside ``_construct_api_member``).
196
- """
197
- members: dict[str, dict[str, Any]] = {}
198
- for name, m in cfg.members.items():
199
- entry: dict[str, Any] = {"enabled": m.enabled, "model": m.model}
200
- if m.api_key_ref is not None:
201
- entry["api_key_ref"] = m.api_key_ref
202
- if m.mode is not None:
203
- entry["mode"] = m.mode
204
- if m.binary is not None:
205
- entry["binary"] = m.binary
206
- if m.model_ladder:
207
- entry["model_ladder"] = list(m.model_ladder)
208
- members[name] = entry
209
- advisors: dict[str, dict[str, Any]] = {}
210
- for name, a in cfg.advisors.items():
211
- entry = {
212
- "enabled": a.enabled,
213
- "member": a.member,
214
- "persona": a.persona,
215
- }
216
- if a.model is not None:
217
- entry["model"] = a.model
218
- advisors[name] = entry
219
- return {
220
- "enabled": cfg.enabled,
221
- "mode": cfg.defaults.mode,
222
- "min_rounds": cfg.defaults.min_rounds,
223
- "deep_min_rounds": cfg.defaults.deep_min_rounds,
224
- "max_output_tokens": cfg.defaults.max_output_tokens,
225
- "session_retention_days": cfg.defaults.session_retention_days,
226
- "debate_max_rounds": cfg.defaults.debate_max_rounds,
227
- "cost_budget": {
228
- "max_input_tokens": cfg.cost_budget.max_input_tokens,
229
- "max_output_tokens": cfg.cost_budget.max_output_tokens,
230
- "max_calls": cfg.cost_budget.max_calls,
231
- "max_total_usd": cfg.cost_budget.max_total_usd,
232
- },
233
- "consensus_scoring": {
234
- "enabled": cfg.consensus_scoring.enabled,
235
- "strong_threshold": cfg.consensus_scoring.strong_threshold,
236
- "minority_threshold": cfg.consensus_scoring.minority_threshold,
237
- "lenses": list(cfg.consensus_scoring.lenses),
238
- },
239
- "cli_call_budget": {
240
- "max_calls_per_day": dict(cfg.cli_call_budget.max_calls_per_day),
241
- "warn_at": cfg.cli_call_budget.warn_at,
242
- },
243
- "necessity_classifier": {
244
- "enabled": cfg.necessity_classifier.enabled,
245
- "mode": cfg.necessity_classifier.mode,
246
- "user_explicit_mode": cfg.necessity_classifier.user_explicit_mode,
247
- },
248
- "model_downgrade": {
249
- "enabled": cfg.model_downgrade.enabled,
250
- "auto_apply": cfg.model_downgrade.auto_apply,
251
- },
252
- "debate": {
253
- "max_cost_usd": cfg.debate.max_cost_usd,
254
- "cost_disclosure": {
255
- "mode": cfg.debate.cost_disclosure.mode,
256
- "threshold_usd": cfg.debate.cost_disclosure.threshold_usd,
257
- "show_per_member": cfg.debate.cost_disclosure.show_per_member,
258
- },
259
- },
260
- "lens_overrides": {
261
- "necessity_classifier_mode": dict(
262
- cfg.lens_overrides.necessity_classifier_mode,
263
- ),
264
- "necessity_classifier_user_explicit_mode": dict(
265
- cfg.lens_overrides.necessity_classifier_user_explicit_mode,
266
- ),
267
- "model_downgrade": {
268
- lens: {"enabled": md.enabled, "auto_apply": md.auto_apply}
269
- for lens, md in cfg.lens_overrides.model_downgrade.items()
270
- },
271
- "cost_disclosure": {
272
- lens: {
273
- "mode": cd.mode,
274
- "threshold_usd": cd.threshold_usd,
275
- "show_per_member": cd.show_per_member,
276
- }
277
- for lens, cd in cfg.lens_overrides.cost_disclosure.items()
278
- },
279
- },
280
- "members": members,
281
- "advisors": advisors,
282
- }
283
-
284
-
285
- def build_members(
286
- settings: dict[str, Any],
287
- *,
288
- invocation_mode: str | None = None,
289
- model_overrides: dict[str, str] | None = None,
290
- siblings_overrides: dict[str, list[str]] | None = None,
291
- skipped: list[dict[str, Any]] | None = None,
292
- ) -> list[ExternalAIClient]:
293
- """Construct enabled council members from settings.
294
-
295
- Honours `ai_council.enabled` (master switch) and per-member
296
- `enabled` flags. Raises `CouncilDisabledError` when the council is
297
- off or no member is wired up.
298
-
299
- `model_overrides` is a per-invocation `{member_name: model_id}`
300
- map that wins over the per-member `model` in settings. Members not
301
- listed fall back to the settings value, then the per-client default.
302
-
303
- `siblings_overrides` is a per-invocation `{member_name: [model, ...]}`
304
- map that fans the named provider out to multiple sibling models in
305
- one run (e.g. claude-sonnet-4-5 + claude-opus-4-1). Each model
306
- becomes its own billable member with independent cost tracking.
307
- Mutually exclusive with `model_overrides` for the same provider;
308
- requires `mode=api`; provider must be enabled in settings.
309
-
310
- `skipped` is an optional caller-owned list. When provided, each
311
- cli-mode member that fails to construct (binary missing) is appended
312
- as `{"member": <name>, "reason": "binary_missing", "detail": <msg>}`
313
- instead of crashing the loop. The skip is also surfaced on stderr
314
- as `[council] SKIP <name>: <detail>` so the run log carries it
315
- even when the caller passes ``None``. Phase 5 Step 2 contract:
316
- a missing CLI binary degrades that member only — never silently
317
- drops, never crashes the whole council unless every configured
318
- member ends up skipped.
319
- """
320
- ai = (settings.get("ai_council") or {}) if isinstance(settings, dict) else {}
321
- if not ai.get("enabled"):
322
- raise CouncilDisabledError(
323
- "ai_council.enabled is false in .agent-settings.yml — "
324
- "flip it on before invoking council:* commands."
325
- )
326
- members_cfg = ai.get("members") or {}
327
- global_mode = ai.get("mode")
328
- cli_budget_cfg = (ai.get("cli_call_budget") or {}) if isinstance(ai, dict) else {}
329
- cli_caps = (cli_budget_cfg.get("max_calls_per_day") or {}) if isinstance(cli_budget_cfg, dict) else {}
330
- cli_warn_at = float(cli_budget_cfg.get("warn_at", 0.8)) if isinstance(cli_budget_cfg, dict) else 0.8
331
- overrides = model_overrides or {}
332
- siblings = siblings_overrides or {}
333
- unknown = set(overrides) - set(members_cfg)
334
- if unknown:
335
- raise CouncilDisabledError(
336
- f"--model targets unknown member(s) {sorted(unknown)!r}; "
337
- f"known members: {sorted(members_cfg)!r}."
338
- )
339
- unknown_sib = set(siblings) - set(members_cfg)
340
- if unknown_sib:
341
- raise CouncilDisabledError(
342
- f"--siblings targets unknown member(s) {sorted(unknown_sib)!r}; "
343
- f"known members: {sorted(members_cfg)!r}."
344
- )
345
- conflict = set(overrides) & set(siblings)
346
- if conflict:
347
- raise CouncilDisabledError(
348
- f"--model and --siblings target the same member(s) {sorted(conflict)!r}; "
349
- f"pick one per provider per invocation."
350
- )
351
- members: list[ExternalAIClient] = []
352
- for name, cfg in members_cfg.items():
353
- cfg = cfg or {}
354
- if not cfg.get("enabled"):
355
- if name in siblings:
356
- raise CouncilDisabledError(
357
- f"--siblings targets member {name!r} but it is not "
358
- f"enabled in .agent-settings.yml (ai_council.members.{name}.enabled)."
359
- )
360
- continue
361
- mode = resolve_mode(
362
- name,
363
- invocation_mode=invocation_mode,
364
- member_settings=cfg,
365
- global_mode=global_mode,
366
- )
367
- if name in siblings:
368
- if mode != "api":
369
- raise CouncilDisabledError(
370
- f"--siblings requires mode=api for member {name!r} (got {mode!r})."
371
- )
372
- api_key_ref = cfg.get("api_key_ref")
373
- for sib_model in siblings[name]:
374
- members.append(
375
- _construct_api_member(name, sib_model, api_key_ref=api_key_ref),
376
- )
377
- continue
378
- model = overrides.get(name) or cfg.get("model")
379
- if mode == "api" and name in _API_PROVIDERS:
380
- members.append(
381
- _construct_api_member(name, model, api_key_ref=cfg.get("api_key_ref")),
382
- )
383
- elif mode == "cli" and name in _CLI_PROVIDERS:
384
- try:
385
- members.append(
386
- _construct_cli_member(
387
- name,
388
- model,
389
- binary=cfg.get("binary"),
390
- max_calls_per_day=cli_caps.get(name),
391
- warn_at=cli_warn_at,
392
- ),
393
- )
394
- except CliClientError as exc:
395
- _, _, display = _CLI_FACTORY[name]
396
- detail = (
397
- f"{exc} Install the {display} CLI or flip "
398
- f"ai_council.members.{name}.mode back to 'api'."
399
- )
400
- entry = {
401
- "member": name,
402
- "reason": "binary_missing",
403
- "detail": detail,
404
- }
405
- if skipped is not None:
406
- skipped.append(entry)
407
- print(f"[council] SKIP {name}: {detail}", file=sys.stderr)
408
- continue
409
- elif mode == "cli":
410
- raise CouncilDisabledError(
411
- f"member {name!r} resolves to mode=cli but no CLI client is "
412
- f"wired (known: {sorted(_CLI_PROVIDERS)!r})."
413
- )
414
- elif mode == "manual":
415
- members.append(ManualClient(name=name, model=model or "manual"))
416
- elif mode == "playwright":
417
- raise CouncilDisabledError(
418
- f"member {name!r} resolves to mode=playwright (Phase 2c, not wired)."
419
- )
420
- else:
421
- raise CouncilDisabledError(
422
- f"member {name!r} has no transport — mode={mode}, "
423
- f"name not in {sorted(_API_PROVIDERS)!r}."
424
- )
425
- if not members:
426
- if skipped:
427
- names = ", ".join(s["member"] for s in skipped)
428
- raise CouncilDisabledError(
429
- f"no council member could be constructed — every enabled "
430
- f"member was skipped ({names}). See [council] SKIP entries "
431
- f"on stderr for the per-member reason."
432
- )
433
- raise CouncilDisabledError(
434
- "no council member has `enabled: true` — enable at least one in "
435
- ".agent-settings.yml under ai_council.members.*."
436
- )
437
- return members
438
-
439
-
440
- def _build_advisor_plans(
441
- ai_cfg: dict[str, Any],
442
- repo_root: Path,
443
- ) -> dict[str, AdvisorPlan]:
444
- """Reconstruct AdvisorConfig from the projected dict, then plan swaps.
445
-
446
- The legacy ``ai_council.advisors`` dict shape is the projection
447
- written by ``_synthesize_ai_council_block``. Disabled advisors are
448
- silently skipped by ``plan_advisor_swap``; one-per-provider is
449
- enforced there. Returns empty when no advisor block is present.
450
- """
451
- raw = ai_cfg.get("advisors") if isinstance(ai_cfg, dict) else None
452
- if not raw:
453
- return {}
454
- advisors: dict[str, AdvisorConfig] = {}
455
- for name, entry in raw.items():
456
- if not isinstance(entry, dict):
457
- continue
458
- advisors[name] = AdvisorConfig(
459
- name=name,
460
- enabled=bool(entry.get("enabled", False)),
461
- member=str(entry.get("member", "")),
462
- persona=str(entry.get("persona", "")),
463
- model=entry.get("model"),
464
- )
465
- return plan_advisor_swap(advisors, repo_root)
466
-
467
-
468
- def _advisor_model_overrides(
469
- plans: dict[str, AdvisorPlan],
470
- explicit: dict[str, str] | None,
471
- ) -> dict[str, str]:
472
- """Merge advisor model_overrides under explicit ``--model`` flags.
473
-
474
- Explicit CLI ``--model`` overrides win over advisor-bound model
475
- overrides — the user's flag is always authoritative.
476
- """
477
- merged: dict[str, str] = {}
478
- for member, plan in plans.items():
479
- if plan.model_override:
480
- merged[member] = plan.model_override
481
- if explicit:
482
- merged.update(explicit)
483
- return merged
484
-
485
-
486
- def _format_advisor_summary(
487
- plans: dict[str, AdvisorPlan],
488
- members: list[ExternalAIClient],
489
- ) -> str:
490
- """Render the ``advisor: <persona> on <member> via <model>`` lines."""
491
- if not plans:
492
- return ""
493
- member_models = {m.name: m.model for m in members}
494
- rows: list[str] = []
495
- for member, plan in plans.items():
496
- model = member_models.get(member, plan.model_override or "?")
497
- rows.append(
498
- f" advisor: {plan.display_name} on {member} via {model}"
499
- )
500
- return "\n".join(rows)
501
-
502
-
503
- def _construct_api_member(
504
- name: str,
505
- model: str | None,
506
- *,
507
- api_key_ref: str | None = None,
508
- ) -> ExternalAIClient:
509
- """Build an api-mode client for a known provider name.
510
-
511
- ``api_key_ref`` carries the validated ``file:<path>`` / ``env:<VAR>``
512
- reference from ``agents/settings/.ai-council.yml`` and is resolved lazily here
513
- so the council does not require keys for disabled providers. When
514
- ``api_key_ref`` is ``None`` (no new config yet, or legacy code path),
515
- fall back to the per-provider loaders so the pre-step-2
516
- ``.agent-settings.yml`` flow keeps working during migration. Tests
517
- monkeypatch the legacy loaders — that path stays intact.
518
- """
519
- if name == "anthropic":
520
- api_key = (
521
- resolve_api_key(api_key_ref, scope="ai_council.members.anthropic")
522
- if api_key_ref else load_anthropic_key()
523
- )
524
- return AnthropicClient(model=model or "claude-sonnet-4-5", api_key=api_key)
525
- if name == "openai":
526
- api_key = (
527
- resolve_api_key(api_key_ref, scope="ai_council.members.openai")
528
- if api_key_ref else load_openai_key()
529
- )
530
- return OpenAIClient(model=model or "gpt-4o", api_key=api_key)
531
- if name == "gemini":
532
- if not api_key_ref:
533
- raise CouncilDisabledError(
534
- "member 'gemini' requires api_key_ref in ~/.event4u/agent-config/settings/.ai-council.yml "
535
- "(e.g. `env:GEMINI_API_KEY`) — no legacy fallback."
536
- )
537
- api_key = resolve_api_key(api_key_ref, scope="ai_council.members.gemini")
538
- return GeminiClient(model=model or "gemini-2.5-pro", api_key=api_key)
539
- if name == "xai":
540
- if not api_key_ref:
541
- raise CouncilDisabledError(
542
- "member 'xai' requires api_key_ref in ~/.event4u/agent-config/settings/.ai-council.yml "
543
- "(e.g. `env:XAI_API_KEY`) — no legacy fallback."
544
- )
545
- api_key = resolve_api_key(api_key_ref, scope="ai_council.members.xai")
546
- return XAIClient(model=model or "grok-4", api_key=api_key)
547
- if name == "perplexity":
548
- if not api_key_ref:
549
- raise CouncilDisabledError(
550
- "member 'perplexity' requires api_key_ref in ~/.event4u/agent-config/settings/.ai-council.yml "
551
- "(e.g. `env:PERPLEXITY_API_KEY`) — no legacy fallback."
552
- )
553
- api_key = resolve_api_key(api_key_ref, scope="ai_council.members.perplexity")
554
- return PerplexityClient(model=model or "sonar-pro", api_key=api_key)
555
- raise CouncilDisabledError(
556
- f"member {name!r} has no api transport "
557
- f"(known: {sorted(_API_PROVIDERS)!r})."
558
- )
559
-
560
-
561
- #: Provider → (class-attribute-name, default_model, human_display) for
562
- #: cli-mode routing. The class ref is looked up via ``getattr`` on this
563
- #: module at call time so ``monkeypatch.setattr(council_cli, "AnthropicCliClient", X)``
564
- #: keeps working from tests. The display string is used by
565
- #: ``build_members`` to render the "Install the <X> CLI" hint in
566
- #: skip-with-reason logs without re-importing every subclass at the
567
- #: call site.
568
- _CLI_FACTORY: dict[str, tuple[str, str, str]] = {
569
- "anthropic": ("AnthropicCliClient", "claude-sonnet-4-5", "Claude"),
570
- "openai": ("OpenAICliClient", "gpt-5", "Codex"),
571
- "gemini": ("GeminiCliClient", "gemini-2.5-pro", "Gemini"),
572
- "xai": ("XAICliClient", "grok-4", "Grok (community)"),
573
- "perplexity": ("PerplexityCliClient", "sonar-pro", "Perplexity (community)"),
574
- }
575
-
576
-
577
- def _construct_cli_member(
578
- name: str,
579
- model: str | None,
580
- *,
581
- binary: str | None = None,
582
- max_calls_per_day: int | None = None,
583
- warn_at: float = 0.8,
584
- ) -> ExternalAIClient:
585
- """Build a cli-mode client for a known provider name.
586
-
587
- ``binary`` overrides the provider default (e.g. ``/opt/claude``);
588
- ``None`` falls through to ``shutil.which(default_binary)``. The
589
- daily quota is plumbed through to the subclass; ``None`` disables
590
- the local counter (only stderr-based quota detection remains).
591
- ``warn_at`` (step-8 P1) is the fractional threshold flipping the
592
- pre-run quota summary to its ``⚠️`` shape; default 0.8 mirrors
593
- ``CliCallBudgetConfig``.
594
- Lets the subclass' ``CliClientError`` propagate so ``build_members``
595
- can convert it into a structured per-member skip entry without
596
- crashing the whole council (the original "fail loudly for the
597
- entire council" contract is preserved when no other member
598
- survives — the empty-members guard at the end of ``build_members``
599
- fires with the skip log attached).
600
- """
601
- if name in _CLI_FACTORY:
602
- attr, default_model, _display = _CLI_FACTORY[name]
603
- cls = globals()[attr]
604
- return cls(
605
- model=model or default_model,
606
- binary=binary,
607
- max_calls_per_day=max_calls_per_day,
608
- warn_at=warn_at,
609
- )
610
- raise CouncilDisabledError(
611
- f"member {name!r} has no cli transport "
612
- f"(known: {sorted(_CLI_PROVIDERS)!r})."
613
- )
614
-
615
-
616
- def build_question(
617
- *,
618
- input_path: Path,
619
- input_mode: str,
620
- max_tokens: int,
621
- prompt_mode_override: str | None = None,
622
- ) -> tuple[CouncilQuestion, str]:
623
- """Bundle the input file. Returns (question, artefact_label).
624
-
625
- `prompt_mode_override` swaps the per-mode neutrality addendum looked
626
- up by `system_prompt_for(question.mode, ...)`. The bundle shape is
627
- unchanged — the bundler still uses `input_mode` to format the
628
- artefact. Routed by the `/council pr|design|optimize|analysis`
629
- wrappers via the `--prompt-mode` CLI flag.
630
- """
631
- if input_mode == "prompt":
632
- text = input_path.read_text(encoding="utf-8")
633
- ctx = bundle_prompt(text)
634
- artefact = str(input_path)
635
- elif input_mode == "roadmap":
636
- ctx = bundle_roadmap(input_path)
637
- artefact = str(input_path)
638
- else:
639
- raise ValueError(
640
- f"unsupported input mode: {input_mode!r} (use prompt | roadmap)"
641
- )
642
- mode = prompt_mode_override or ctx.mode
643
- return CouncilQuestion(mode=mode, user_prompt=ctx.text,
644
- max_tokens=max_tokens), artefact
645
-
646
-
647
- def format_estimate_table(
648
- members: list[ExternalAIClient],
649
- estimates: list[Any],
650
- *,
651
- consensus_delta_usd: float = 0.0,
652
- consensus_extra_calls: int = 0,
653
- peer_review_delta_usd: float = 0.0,
654
- peer_review_extra_calls: int = 0,
655
- ) -> str:
656
- rows = [
657
- f" {m.name}/{m.model}: "
658
- f"~{e.input_tokens} in + {e.output_tokens} out = ${e.total_usd:.4f}"
659
- for m, e in zip(members, estimates)
660
- ]
661
- total = sum(e.total_usd for e in estimates)
662
- if consensus_extra_calls > 0:
663
- rows.append(
664
- f" +consensus scoring: +{consensus_extra_calls} calls "
665
- f"(~+${consensus_delta_usd:.4f})"
666
- )
667
- total += consensus_delta_usd
668
- if peer_review_extra_calls > 0:
669
- rows.append(
670
- f" +peer-review: +{peer_review_extra_calls} calls "
671
- f"(~+${peer_review_delta_usd:.4f})"
672
- )
673
- total += peer_review_delta_usd
674
- rows.append(f" TOTAL: ${total:.4f}")
675
- return "\n".join(rows)
676
-
677
-
678
- def _consensus_cost_delta(
679
- ai_cfg: dict[str, Any],
680
- prompt_mode: str,
681
- estimates: list[Any],
682
- n_billable: int,
683
- ) -> tuple[int, float]:
684
- """Return ``(extra_calls, extra_usd)`` for the consensus round.
685
-
686
- Active when ``ai_council.consensus_scoring.enabled`` is true AND the
687
- invocation's lens is in ``consensus_scoring.lenses``. Each member
688
- contributes two extra calls (extraction + scoring); the worst-case
689
- cost uses the base per-member estimate as a ceiling.
690
- """
691
- cs = ai_cfg.get("consensus_scoring") or {}
692
- if not cs.get("enabled"):
693
- return 0, 0.0
694
- lenses = cs.get("lenses") or ["analysis"]
695
- if prompt_mode not in lenses:
696
- return 0, 0.0
697
- extra_calls = 2 * n_billable
698
- extra_usd = 2.0 * sum(e.total_usd for e in estimates)
699
- return extra_calls, extra_usd
700
-
701
-
702
- def _maybe_run_consensus(
703
- ai_cfg: dict[str, Any],
704
- question: CouncilQuestion,
705
- members: list[ExternalAIClient],
706
- responses: list[CouncilResponse],
707
- budget: CostBudget,
708
- table: PriceTable,
709
- project: Any,
710
- args: argparse.Namespace,
711
- ) -> ConsensusResult | None:
712
- """Run the consensus scoring round when enabled for this lens."""
713
- cs = ai_cfg.get("consensus_scoring") or {}
714
- if not cs.get("enabled"):
715
- return None
716
- lenses = cs.get("lenses") or ["analysis"]
717
- if question.mode not in lenses:
718
- return None
719
- return run_consensus_scoring(
720
- members, responses,
721
- budget=budget, table=table, project=project,
722
- original_ask=args.original_ask,
723
- max_tokens=question.max_tokens,
724
- strong_threshold=float(cs.get("strong_threshold", 0.7)),
725
- minority_threshold=float(cs.get("minority_threshold", 0.4)),
726
- )
727
-
728
-
729
- def _serialise_consensus(consensus: ConsensusResult) -> dict[str, Any]:
730
- """Project ConsensusResult onto a JSON-safe dict for session payloads."""
731
- return {
732
- "findings": [
733
- {"id": f.id, "source": f.source, "text": f.text}
734
- for f in consensus.findings
735
- ],
736
- "scores": [
737
- {
738
- "finding_id": s.finding_id, "scorer": s.scorer,
739
- "score": s.score, "agree": s.agree, "reason": s.reason,
740
- }
741
- for s in consensus.scores
742
- ],
743
- "metadata": {
744
- fid: {
745
- "mean_score": m.mean_score,
746
- "agreement_rate": m.agreement_rate,
747
- "consensus_strength": m.consensus_strength,
748
- "dissent_count": m.dissent_count,
749
- "scorers": list(m.scorers),
750
- "concur_count": m.concur_count,
751
- "dissent_reasons": [list(pair) for pair in m.dissent_reasons],
752
- "evidence_quality": m.evidence_quality,
753
- }
754
- for fid, m in consensus.metadata.items()
755
- },
756
- "extraction_responses": _serialise_responses(consensus.extraction_responses),
757
- "scoring_responses": _serialise_responses(consensus.scoring_responses),
758
- }
759
-
760
-
761
- def _decision_replay_settings(
762
- ai_cfg: dict[str, Any], lens: str,
763
- ) -> tuple[bool, bool]:
764
- """Resolve (enabled, include_member_arguments) for ``lens``.
765
-
766
- Per-lens override under ``lenses.<lens>.decision_replay`` beats the
767
- global ``decision_replay`` block. Defaults: enabled=True,
768
- include_member_arguments=True (Phase 9 ships ON by default — the
769
- artefact is the audit trail GPT review of PR #148 called out as
770
- missing).
771
- """
772
- global_block = ai_cfg.get("decision_replay") or {}
773
- enabled = global_block.get("enabled", True)
774
- include_args = global_block.get("include_member_arguments", True)
775
- lenses = ai_cfg.get("lenses") or {}
776
- lens_block = (lenses.get(lens) or {}).get("decision_replay")
777
- if isinstance(lens_block, dict):
778
- if "enabled" in lens_block:
779
- enabled = lens_block["enabled"]
780
- if "include_member_arguments" in lens_block:
781
- include_args = lens_block["include_member_arguments"]
782
- return bool(enabled), bool(include_args)
783
-
784
-
785
- def _maybe_write_decision_replay(
786
- *,
787
- ai_cfg: dict[str, Any],
788
- lens: str,
789
- out_path: Path,
790
- consensus: ConsensusResult | None,
791
- deliberation: list[CouncilResponse],
792
- original_ask: str,
793
- ) -> Path | None:
794
- """Write ``decision-replay.md`` alongside ``out_path`` when enabled.
795
-
796
- No-op when ``decision_replay.enabled`` resolves to ``False`` for the
797
- lens or when ``consensus`` is ``None`` (nothing to replay). Returns
798
- the artefact path on success, ``None`` otherwise.
799
- """
800
- enabled, include_args = _decision_replay_settings(ai_cfg, lens)
801
- if not enabled or consensus is None:
802
- return None
803
- replay = render_decision_replay(
804
- DecisionReplayInputs(
805
- findings=list(consensus.findings),
806
- scores=list(consensus.scores),
807
- metadata=dict(consensus.metadata),
808
- deliberation=deliberation,
809
- original_ask=original_ask,
810
- include_member_arguments=include_args,
811
- ),
812
- )
813
- target = out_path.parent / "decision-replay.md"
814
- target.parent.mkdir(parents=True, exist_ok=True)
815
- target.write_text(replay, encoding="utf-8")
816
- return target
817
-
818
-
819
- # ── peer-review (Phase 5 / F1, Karpathy anonymous review) ──────────
820
-
821
-
822
- def _peer_review_active(ai_cfg: dict[str, Any], args: argparse.Namespace) -> bool:
823
- """Return True when peer-review should fire for this invocation.
824
-
825
- Resolution chain (highest priority first):
826
- 1. ``--peer-review`` CLI flag — explicit opt-in.
827
- 2. ``ai_council.peer_review.enabled: true`` in
828
- ``agents/settings/.ai-council.yml`` — opt-in via config.
829
- Both default to false; peer-review is opt-in by R2 verdict.
830
- """
831
- if getattr(args, "peer_review", False):
832
- return True
833
- pr_cfg = ai_cfg.get("peer_review") or {}
834
- return bool(pr_cfg.get("enabled"))
835
-
836
-
837
- def _peer_review_cost_delta(
838
- ai_cfg: dict[str, Any],
839
- args: argparse.Namespace,
840
- estimates: list[Any],
841
- n_billable: int,
842
- ) -> tuple[int, float]:
843
- """Return ``(extra_calls, extra_usd)`` for the peer-review round.
844
-
845
- One extra call per billable member (each reviews the others). The
846
- worst-case cost uses the base per-member estimate as a ceiling —
847
- same heuristic as ``_consensus_cost_delta``.
848
- """
849
- if not _peer_review_active(ai_cfg, args):
850
- return 0, 0.0
851
- if n_billable < 2:
852
- # Need ≥ 2 distinct deliberation outputs for peer-review to
853
- # have anything to review. The orchestrator no-ops below 2.
854
- return 0, 0.0
855
- extra_calls = n_billable
856
- extra_usd = sum(e.total_usd for e in estimates)
857
- return extra_calls, extra_usd
858
-
859
-
860
- def _maybe_run_peer_review(
861
- ai_cfg: dict[str, Any],
862
- args: argparse.Namespace,
863
- question: CouncilQuestion,
864
- members: list[ExternalAIClient],
865
- responses: list[CouncilResponse],
866
- budget: CostBudget,
867
- table: PriceTable,
868
- project: Any,
869
- *,
870
- persona_labels: dict[str, str] | None = None,
871
- ) -> PeerReviewResult | None:
872
- """Run the peer-review pass when opted in.
873
-
874
- No-ops if fewer than 2 successful deliberation responses exist —
875
- the orchestrator surfaces the empty result in that case.
876
-
877
- ``persona_labels`` (Phase 6) flows through to ``anonymize_responses``
878
- so advisor-mode runs render as ``Response A (Contrarian)`` instead
879
- of bare ``Response A``. Plain-member runs pass ``None``.
880
- """
881
- if not _peer_review_active(ai_cfg, args):
882
- return None
883
- result = run_peer_review(
884
- members, responses,
885
- budget=budget, table=table, project=project,
886
- original_ask=args.original_ask,
887
- max_tokens=question.max_tokens,
888
- persona_labels=persona_labels,
889
- )
890
- if not result.responses:
891
- return None
892
- return result
893
-
894
-
895
- def _serialise_peer_review(peer_review: PeerReviewResult) -> dict[str, Any]:
896
- """Project PeerReviewResult onto a JSON-safe dict for session payloads."""
897
- return {
898
- "responses": _serialise_responses(peer_review.responses),
899
- "label_to_source": dict(peer_review.label_to_source),
900
- "persona_labels": dict(peer_review.persona_labels),
901
- }
902
-
903
-
904
- def _deserialise_peer_review(
905
- data: dict[str, Any] | None,
906
- ) -> PeerReviewResult | None:
907
- """Reconstruct a PeerReviewResult from a session payload section.
908
-
909
- Returns ``None`` for payloads predating Phase 5 or runs where the
910
- flag was not passed.
911
- """
912
- if not data:
913
- return None
914
- return PeerReviewResult(
915
- responses=_deserialise_responses(data.get("responses") or []),
916
- label_to_source=dict(data.get("label_to_source") or {}),
917
- persona_labels=dict(data.get("persona_labels") or {}),
918
- )
919
-
920
-
921
- # ── subcommands ─────────────────────────────────────────────────────
922
-
923
-
924
- def _resolve_rounds(args: argparse.Namespace, ai_cfg: dict[str, Any]) -> int:
925
- """Resolve effective debate round count from CLI args + settings.
926
-
927
- Resolution chain (highest priority first):
928
- 1. ``--rounds N`` — explicit user override, any value.
929
- 2. ``--depth deep`` — uses ``ai_council.deep_min_rounds``,
930
- floored at ``min_rounds`` so the deep tier is monotonic.
931
- 3. ``ai_council.min_rounds`` — default 2.
932
-
933
- Sub-commands (rule/skill/command) declare ``council_depth: deep``
934
- in their frontmatter; the host agent reads that and translates it
935
- to ``--depth deep`` on the CLI invocation. The CLI itself stays
936
- unaware of frontmatter — the contract is the flag.
937
- """
938
- if getattr(args, "rounds", None) is not None:
939
- return int(args.rounds)
940
- min_rounds = int(ai_cfg.get("min_rounds", 2))
941
- if getattr(args, "depth", "standard") == "deep":
942
- deep = int(ai_cfg.get("deep_min_rounds", min_rounds))
943
- return max(deep, min_rounds)
944
- return min_rounds
945
-
946
-
947
- def _resolve_max_tokens(args: argparse.Namespace, ai_cfg: dict[str, Any]) -> int:
948
- """Resolve the per-call output budget passed to each member.
949
-
950
- Resolution chain (highest priority first):
951
- 1. ``--max-tokens N`` — explicit invocation override.
952
- 2. ``ai_council.max_output_tokens`` — settings value (project file
953
- is authoritative; this key is not user-global-mergeable).
954
- 3. ``DEFAULT_MAX_TOKENS`` — package fallback (2048).
955
-
956
- A value of ``0`` at any layer means "unlimited"; it is widened to
957
- ``UNLIMITED_TOKENS_FALLBACK`` before reaching the SDK because
958
- Anthropic rejects ``max_tokens=0``. Estimation uses the same expanded
959
- value so the cost preview reflects the worst-case ceiling.
960
- """
961
- cli = getattr(args, "max_tokens", None)
962
- if cli is not None:
963
- value = int(cli)
964
- elif "max_output_tokens" in ai_cfg:
965
- value = int(ai_cfg.get("max_output_tokens") or 0)
966
- else:
967
- value = DEFAULT_MAX_TOKENS
968
- if value <= 0:
969
- return UNLIMITED_TOKENS_FALLBACK
970
- return value
971
-
972
-
973
- def cmd_estimate(
974
- args: argparse.Namespace,
975
- *,
976
- settings: dict[str, Any] | None = None,
977
- members: list[ExternalAIClient] | None = None,
978
- table: PriceTable | None = None,
979
- ) -> int:
980
- """Print per-member cost preview. No API calls."""
981
- if settings is None:
982
- settings = load_settings()
983
- ai_cfg = (settings.get("ai_council") or {}) if isinstance(settings, dict) else {}
984
- advisor_plans = _build_advisor_plans(ai_cfg, REPO_ROOT)
985
- explicit_overrides = _parse_model_overrides(getattr(args, "model", None))
986
- skipped: list[dict[str, Any]] = []
987
- if members is None:
988
- members = build_members(
989
- settings,
990
- invocation_mode=args.mode_override,
991
- model_overrides=_advisor_model_overrides(
992
- advisor_plans, explicit_overrides,
993
- ),
994
- siblings_overrides=_parse_siblings_overrides(getattr(args, "siblings", None)),
995
- skipped=skipped,
996
- )
997
- if table is None:
998
- table = load_prices()
999
- question, _ = build_question(
1000
- input_path=Path(args.question), input_mode=args.input_mode,
1001
- max_tokens=_resolve_max_tokens(args, ai_cfg),
1002
- prompt_mode_override=getattr(args, "prompt_mode", None),
1003
- )
1004
- project = detect_project_context(REPO_ROOT)
1005
- billable = [m for m in members if getattr(m, "billable", True)]
1006
- estimates = estimate(question, billable, table,
1007
- project=project, original_ask=args.original_ask,
1008
- advisor_plans=advisor_plans)
1009
- if getattr(args, "debate", False):
1010
- return _emit_debate_estimate(
1011
- args, ai_cfg, members, billable, estimates, advisor_plans,
1012
- skipped=skipped,
1013
- )
1014
- extra_calls, extra_usd = _consensus_cost_delta(
1015
- ai_cfg, question.mode, estimates, len(billable),
1016
- )
1017
- pr_extra_calls, pr_extra_usd = _peer_review_cost_delta(
1018
- ai_cfg, args, estimates, len(billable),
1019
- )
1020
- sys.stdout.write(
1021
- f"council:estimate · mode={question.mode} · members={len(members)} "
1022
- f"(billable={len(billable)})\n"
1023
- )
1024
- advisor_summary = _format_advisor_summary(advisor_plans, billable)
1025
- if advisor_summary:
1026
- sys.stdout.write(advisor_summary + "\n")
1027
- if skipped:
1028
- sys.stdout.write(format_install_hints(skipped) + "\n")
1029
- sys.stdout.write(
1030
- format_estimate_table(
1031
- billable, estimates,
1032
- consensus_delta_usd=extra_usd,
1033
- consensus_extra_calls=extra_calls,
1034
- peer_review_delta_usd=pr_extra_usd,
1035
- peer_review_extra_calls=pr_extra_calls,
1036
- ) + "\n"
1037
- )
1038
- return 0
1039
-
1040
-
1041
- def _emit_debate_estimate(
1042
- args: argparse.Namespace,
1043
- ai_cfg: dict[str, Any],
1044
- members: list[ExternalAIClient],
1045
- billable: list[ExternalAIClient],
1046
- estimates: list[Any],
1047
- advisor_plans: Any,
1048
- *,
1049
- skipped: list[dict[str, Any]] | None = None,
1050
- ) -> int:
1051
- """Render the round-by-round debate cost projection.
1052
-
1053
- Upper bound only — progressive disclosure may stop the debate early.
1054
- Cost shape mirrors ``cmd_debate``: one call per billable member per
1055
- round, default ``ai_council.min_rounds`` (typically 2), capped at
1056
- ``ai_council.debate_max_rounds`` (typically 4).
1057
- """
1058
- min_rounds = int(ai_cfg.get("min_rounds", 2))
1059
- max_rounds_cap = int(ai_cfg.get("debate_max_rounds", 4))
1060
- requested = (
1061
- int(args.rounds) if getattr(args, "rounds", None) is not None
1062
- else min_rounds
1063
- )
1064
- if requested < 1:
1065
- raise argparse.ArgumentTypeError(
1066
- f"--rounds must be >= 1 (got {requested})"
1067
- )
1068
- if requested > max_rounds_cap:
1069
- raise argparse.ArgumentTypeError(
1070
- f"--rounds={requested} exceeds debate_max_rounds={max_rounds_cap}; "
1071
- f"raise the cap in ~/.event4u/agent-config/settings/.ai-council.yml or lower --rounds."
1072
- )
1073
- rounds = requested
1074
- per_round_usd = sum(e.total_usd for e in estimates)
1075
- projected_total = per_round_usd * rounds
1076
- sys.stdout.write(
1077
- f"council:estimate · mode=debate · members={len(members)} "
1078
- f"(billable={len(billable)}) · rounds={rounds} "
1079
- f"(cap={max_rounds_cap})\n"
1080
- )
1081
- advisor_summary = _format_advisor_summary(advisor_plans, billable)
1082
- if advisor_summary:
1083
- sys.stdout.write(advisor_summary + "\n")
1084
- if skipped:
1085
- sys.stdout.write(format_install_hints(skipped) + "\n")
1086
- for round_idx in range(1, rounds + 1):
1087
- sys.stdout.write(f"\nRound {round_idx} of {rounds}:\n")
1088
- sys.stdout.write(format_estimate_table(billable, estimates) + "\n")
1089
- if round_idx < rounds:
1090
- sys.stdout.write(" " + "─" * 40 + "\n")
1091
- sys.stdout.write(
1092
- f"\n PROJECTED TOTAL ({rounds} rounds): ${projected_total:.4f}\n"
1093
- )
1094
- sys.stdout.write(
1095
- " Note: progressive disclosure may stop the debate early; "
1096
- "this is an upper bound.\n"
1097
- )
1098
- return 0
1099
-
1100
-
1101
- def _serialise_responses(responses: list[CouncilResponse]) -> list[dict[str, Any]]:
1102
- out: list[dict[str, Any]] = []
1103
- for r in responses:
1104
- d = asdict(r)
1105
- # `metadata` may contain non-JSON types; coerce.
1106
- d["metadata"] = {k: str(v) for k, v in (d.get("metadata") or {}).items()}
1107
- out.append(d)
1108
- return out
1109
-
1110
-
1111
- def _deserialise_responses(items: list[dict[str, Any]]) -> list[CouncilResponse]:
1112
- out: list[CouncilResponse] = []
1113
- for d in items:
1114
- out.append(CouncilResponse(
1115
- provider=d.get("provider", ""),
1116
- model=d.get("model", ""),
1117
- text=d.get("text", ""),
1118
- input_tokens=int(d.get("input_tokens", 0) or 0),
1119
- output_tokens=int(d.get("output_tokens", 0) or 0),
1120
- latency_ms=int(d.get("latency_ms", 0) or 0),
1121
- error=d.get("error"),
1122
- metadata=dict(d.get("metadata") or {}),
1123
- ))
1124
- return out
1125
-
1126
-
1127
- def _deserialise_consensus(data: dict[str, Any] | None) -> ConsensusResult | None:
1128
- """Reconstruct a ConsensusResult from a serialised payload section.
1129
-
1130
- Used by ``cmd_render`` to re-render saved sessions that captured a
1131
- consensus round. Returns ``None`` when the payload predates Phase 4
1132
- or the round was skipped for the lens.
1133
- """
1134
- if not data:
1135
- return None
1136
- from scripts.ai_council.consensus import (
1137
- ConsensusMetadata, Finding, FindingScore,
1138
- aggregate_scores, bucket_by_threshold,
1139
- )
1140
- findings = [
1141
- Finding(id=f["id"], source=f["source"], text=f["text"])
1142
- for f in (data.get("findings") or [])
1143
- ]
1144
- scores = [
1145
- FindingScore(
1146
- finding_id=s["finding_id"], scorer=s["scorer"],
1147
- score=int(s["score"]), agree=bool(s["agree"]),
1148
- reason=s.get("reason", ""),
1149
- )
1150
- for s in (data.get("scores") or [])
1151
- ]
1152
- metadata = aggregate_scores(findings, scores)
1153
- bucket = bucket_by_threshold(findings, metadata)
1154
- return ConsensusResult(
1155
- bucket=bucket, findings=findings, scores=scores, metadata=metadata,
1156
- extraction_responses=_deserialise_responses(
1157
- data.get("extraction_responses") or [],
1158
- ),
1159
- scoring_responses=_deserialise_responses(
1160
- data.get("scoring_responses") or [],
1161
- ),
1162
- )
1163
-
1164
-
1165
- def _resolve_necessity_mode(
1166
- ai_cfg: dict[str, Any],
1167
- lens: str,
1168
- invocation: str = "agent",
1169
- ) -> tuple[bool, str]:
1170
- """Return ``(enabled, effective_mode)`` for the necessity classifier.
1171
-
1172
- Two-tier resolution (step-8 D2):
1173
-
1174
- - ``invocation="agent"`` → reads ``necessity_classifier.mode`` with
1175
- per-lens override at ``lenses.<lens>.necessity_classifier.mode``
1176
- (default ``educate``).
1177
- - ``invocation="user_explicit"`` → reads
1178
- ``necessity_classifier.user_explicit_mode`` with per-lens override
1179
- at ``lenses.<lens>.necessity_classifier.user_explicit_mode``
1180
- (default ``warn-only``).
1181
-
1182
- Reads the synthesized dict shape produced by
1183
- :func:`_synthesize_ai_council_block`, so both typed-config and
1184
- legacy-settings paths are honoured.
1185
- """
1186
- nc_block = ai_cfg.get("necessity_classifier") or {}
1187
- enabled = bool(nc_block.get("enabled", True))
1188
- lens_overrides = ai_cfg.get("lens_overrides") or {}
1189
- if invocation == "user_explicit":
1190
- global_mode = str(nc_block.get("user_explicit_mode", "warn-only"))
1191
- overrides = (
1192
- lens_overrides.get("necessity_classifier_user_explicit_mode") or {}
1193
- )
1194
- else:
1195
- global_mode = str(nc_block.get("mode", "educate"))
1196
- overrides = lens_overrides.get("necessity_classifier_mode") or {}
1197
- return enabled, str(overrides.get(lens, global_mode))
1198
-
1199
-
1200
- def _provider_caps_snapshot(ai_cfg: dict[str, Any]) -> dict[str, dict[str, str]]:
1201
- """Return ``{provider: {mode, model}}`` for enabled members.
1202
-
1203
- Step-8 D3 events-log snapshot. Captures only public capability
1204
- metadata (no API keys, no prompt content) so the log line stays
1205
- within the privacy floor. Disabled members are excluded.
1206
- """
1207
- members = ai_cfg.get("members") or {}
1208
- snapshot: dict[str, dict[str, str]] = {}
1209
- if not isinstance(members, dict):
1210
- return snapshot
1211
- for name, cfg in members.items():
1212
- if not isinstance(cfg, dict) or not cfg.get("enabled", True):
1213
- continue
1214
- snapshot[str(name)] = {
1215
- "mode": str(cfg.get("mode", "")),
1216
- "model": str(cfg.get("model", "")),
1217
- }
1218
- return snapshot
1219
-
1220
-
1221
- def _necessity_gate(
1222
- *, prompt: str, lens: str, invocation: str, proceed_anyway: bool,
1223
- ai_cfg: dict[str, Any], stdout=None, original_ask: str = "",
1224
- ) -> tuple[bool, int, ClassificationResult | None]:
1225
- """Apply the Phase-6 necessity classifier before any member fires.
1226
-
1227
- Returns ``(proceed, exit_code, result)``. ``proceed=True`` means the
1228
- dispatcher continues; ``proceed=False`` means the caller should
1229
- return ``exit_code`` immediately. ``result`` carries the verdict for
1230
- session.md provenance on the proceed path (None when classifier is
1231
- disabled / off).
1232
-
1233
- Step-8 D3: every non-disabled branch emits one
1234
- :func:`append_event` line. ``original_ask`` is forwarded to the
1235
- events log so the sha256[:12] hash anchors the line to the
1236
- user-side question without leaking content. When the caller does
1237
- not have an ``original_ask`` value, the prompt itself is hashed
1238
- (legacy CLIs route through this path).
1239
- """
1240
- out = stdout if stdout is not None else sys.stdout
1241
- enabled, mode = _resolve_necessity_mode(ai_cfg, lens, invocation=invocation)
1242
- if not enabled or mode == "off":
1243
- return True, 0, None
1244
- result = classify_necessity(prompt, lens=lens, invocation=invocation)
1245
- caps = _provider_caps_snapshot(ai_cfg)
1246
- hashed = original_ask or prompt
1247
-
1248
- def _emit(action: str) -> None:
1249
- append_event({
1250
- "lens": lens, "invocation": invocation,
1251
- "action": action, "verdict": result.verdict,
1252
- "category": result.category,
1253
- "mode": mode, "provider_caps": caps,
1254
- "original_ask": hashed,
1255
- })
1256
-
1257
- if result.verdict != "unnecessary":
1258
- if result.verdict == "borderline":
1259
- out.write(
1260
- f"council:necessity · borderline ({result.category}) · "
1261
- f"{result.rationale}\n"
1262
- )
1263
- _emit("proceed")
1264
- return True, 0, result
1265
- # verdict == "unnecessary"
1266
- if mode == "warn-only":
1267
- # Annotated but never skips (step-8 D2). Applies to both
1268
- # invocation tiers when the mode resolves to warn-only.
1269
- out.write(
1270
- f"council:necessity · warn-only ({result.category}) · "
1271
- f"{result.rationale}\n"
1272
- )
1273
- _emit("proceed")
1274
- return True, 0, result
1275
- if mode == "block":
1276
- out.write(
1277
- f"council:necessity · skipped ({result.category}) · "
1278
- f"{result.rationale}\n"
1279
- f"council:necessity · mode=block — `--proceed-anyway` has "
1280
- f"no effect on the block path.\n"
1281
- )
1282
- _emit("skip_necessity")
1283
- return False, 0, result
1284
- # mode == "educate"
1285
- if invocation == "agent":
1286
- out.write(
1287
- f"council:necessity · skipped (agent, {result.category}) · "
1288
- f"{result.rationale}\n"
1289
- )
1290
- _emit("skip_necessity")
1291
- return False, 0, result
1292
- # invocation == "user_explicit"
1293
- if proceed_anyway:
1294
- out.write(
1295
- f"council:necessity · override (user_explicit + "
1296
- f"--proceed-anyway, {result.category}) · "
1297
- f"{result.rationale}\n"
1298
- )
1299
- _emit("proceed")
1300
- return True, 0, result
1301
- out.write(educate_message(result, lens) + "\n")
1302
- _emit("skip_necessity")
1303
- return False, 2, result
1304
-
1305
-
1306
- def _resolve_model_downgrade(
1307
- ai_cfg: dict[str, Any], lens: str,
1308
- ) -> tuple[bool, bool]:
1309
- """Return ``(enabled, auto_apply)`` for the size-fit downgrade gate.
1310
-
1311
- Per-lens override at ``lenses.<lens>.model_downgrade`` wins over the
1312
- global ``model_downgrade`` block. Reads the synthesized dict shape
1313
- from :func:`_synthesize_ai_council_block` so both typed-config and
1314
- legacy paths are honoured.
1315
- """
1316
- md_block = ai_cfg.get("model_downgrade") or {}
1317
- enabled = bool(md_block.get("enabled", True))
1318
- auto_apply = bool(md_block.get("auto_apply", False))
1319
- overrides = (
1320
- (ai_cfg.get("lens_overrides") or {}).get("model_downgrade") or {}
1321
- )
1322
- lens_override = overrides.get(lens) if isinstance(overrides, dict) else None
1323
- if isinstance(lens_override, dict):
1324
- enabled = bool(lens_override.get("enabled", enabled))
1325
- auto_apply = bool(lens_override.get("auto_apply", auto_apply))
1326
- return enabled, auto_apply
1327
-
1328
-
1329
- def _size_fit_gate(
1330
- *, prompt: str, lens: str, members: list[ExternalAIClient],
1331
- ai_cfg: dict[str, Any], stdout=None,
1332
- ) -> list[tuple[str, SizeFitVerdict, bool]]:
1333
- """Apply the Phase-7 size-fit classifier across enabled members.
1334
-
1335
- Iterates every member with a configured ``model_ladder`` and runs
1336
- :func:`classify_size_fit`. When ``auto_apply`` is true and a
1337
- downgrade is suggested, the member's ``model`` attribute is rewritten
1338
- in place; otherwise the suggestion is surfaced as a stdout notice
1339
- and the original model stands. Members without a ladder are skipped
1340
- silently.
1341
-
1342
- Returns a list of ``(member_name, verdict, applied)`` tuples for
1343
- session.md provenance. Never blocks the dispatch — Phase 7 is a
1344
- suggestion gate, not a refusal gate.
1345
- """
1346
- out = stdout if stdout is not None else sys.stdout
1347
- enabled, auto_apply = _resolve_model_downgrade(ai_cfg, lens)
1348
- decisions: list[tuple[str, SizeFitVerdict, bool]] = []
1349
- if not enabled:
1350
- return decisions
1351
- members_cfg = ai_cfg.get("members") or {}
1352
- for member in members:
1353
- member_cfg = members_cfg.get(member.name) or {}
1354
- ladder = member_cfg.get("model_ladder") or ()
1355
- if not ladder:
1356
- continue
1357
- verdict = classify_size_fit(
1358
- prompt, current_model=member.model, ladder=ladder, lens=lens,
1359
- )
1360
- applied = False
1361
- if not verdict.fit and verdict.suggested_model:
1362
- if auto_apply:
1363
- out.write(
1364
- f"council:size-fit · {member.name} · auto-downgrade "
1365
- f"`{member.model}` → `{verdict.suggested_model}` · "
1366
- f"{verdict.reason}\n"
1367
- )
1368
- member.model = verdict.suggested_model
1369
- applied = True
1370
- else:
1371
- out.write(
1372
- f"council:size-fit · {member.name} · "
1373
- f"{downgrade_message(verdict, member.model)}\n"
1374
- )
1375
- decisions.append((member.name, verdict, applied))
1376
- return decisions
1377
-
1378
-
1379
- def _resolve_cost_disclosure(
1380
- ai_cfg: dict[str, Any], lens: str,
1381
- ) -> tuple[str, float, bool]:
1382
- """Return ``(mode, threshold_usd, show_per_member)`` for the lens.
1383
-
1384
- Per-lens override at ``lenses.<lens>.cost_disclosure`` wins over the
1385
- global ``debate.cost_disclosure`` block. The ``debate`` lens gets
1386
- the debate-scoped defaults; other lenses default to ``off`` unless
1387
- explicitly overridden (Phase 8 step 5 \u2014 cheap lenses are opt-in).
1388
- """
1389
- debate_block = ai_cfg.get("debate") or {}
1390
- debate_disc = debate_block.get("cost_disclosure") or {}
1391
- if lens == "debate":
1392
- mode = str(debate_disc.get("mode", "always"))
1393
- threshold = float(debate_disc.get("threshold_usd", 1.00))
1394
- show_per_member = bool(debate_disc.get("show_per_member", True))
1395
- else:
1396
- mode = "off"
1397
- threshold = 1.00
1398
- show_per_member = True
1399
- overrides = (
1400
- (ai_cfg.get("lens_overrides") or {}).get("cost_disclosure") or {}
1401
- )
1402
- lens_override = overrides.get(lens) if isinstance(overrides, dict) else None
1403
- if isinstance(lens_override, dict):
1404
- mode = str(lens_override.get("mode", mode))
1405
- threshold = float(lens_override.get("threshold_usd", threshold))
1406
- show_per_member = bool(lens_override.get("show_per_member", show_per_member))
1407
- return mode, threshold, show_per_member
1408
-
1409
-
1410
- def _format_cost_disclosure(
1411
- est: DebateCostEstimate, *, lens: str, show_per_member: bool,
1412
- ) -> str:
1413
- """Render the pre-flight disclosure block for stdout.
1414
-
1415
- Mirrors the roadmap spec: total range across N members \u00d7 R rounds,
1416
- optional per-member breakdown, and a subscription-member call-out
1417
- for CLI / manual transports that don't sum into USD totals.
1418
- """
1419
- lines = [
1420
- f"council:{lens} \u00b7 cost-disclosure \u00b7 estimated "
1421
- f"${est.low_usd:.4f} \u2013 ${est.high_usd:.4f} "
1422
- f"(expected ${est.expected_usd:.4f}) across "
1423
- f"{len(est.per_member)} billable members \u00d7 {est.rounds} rounds",
1424
- ]
1425
- if show_per_member and est.per_member:
1426
- lines.append(" per member:")
1427
- for pm in est.per_member:
1428
- lines.append(
1429
- f" \u00b7 {pm['name']:<14} {pm['model']:<22} "
1430
- f"${pm['low_usd']:.4f} \u2013 ${pm['high_usd']:.4f}",
1431
- )
1432
- if est.subscription_members:
1433
- lines.append(" subscription (no USD spend):")
1434
- for sm in est.subscription_members:
1435
- label = sm.get("subscription_label") or sm.get("transport", "")
1436
- lines.append(
1437
- f" \u00b7 {sm['name']:<14} {sm['model']:<22} ({label})",
1438
- )
1439
- return "\n".join(lines) + "\n"
1440
-
1441
-
1442
- def _debate_refusal_cap(
1443
- ai_cfg: dict[str, Any],
1444
- ) -> float:
1445
- """Resolve the hard refusal cap (``debate.max_cost_usd``).
1446
-
1447
- Returns 0.0 when disabled. The cap is unconditional \u2014 no
1448
- ``--proceed-anyway`` override (the user must lower rounds, drop
1449
- members, or raise the cap explicitly).
1450
- """
1451
- debate_block = ai_cfg.get("debate") or {}
1452
- return float(debate_block.get("max_cost_usd", 5.00) or 0.0)
1453
-
1454
-
1455
- def _emit_shadow_slo_banner() -> None:
1456
- """Pre-flight SLO banner for solo-dispatch invocations (step-9 P10).
1457
-
1458
- Reads ``agents/runtime/council/shadow-log.jsonl`` and prints the 7-day rolling
1459
- disagreement rate. ``OK``, ``WARN``, ``BREACH`` are all surfaced so the
1460
- user can see when single-member quality is drifting. Never auto-flips
1461
- back to full council \u2014 visibility-first, action-second (D10).
1462
- """
1463
- try:
1464
- from scripts.ai_council import shadow_dispatch as _sd
1465
- rate, n = _sd.compute_disagreement_rate(_sd.SHADOW_LOG_PATH)
1466
- if n == 0:
1467
- return
1468
- sys.stdout.write(_sd.slo_banner(rate, n) + "\n")
1469
- except Exception: # noqa: BLE001 \u2014 banner must never break dispatch.
1470
- return
1471
-
1472
-
1473
- def _apply_solo_dispatch(
1474
- members: list[ExternalAIClient],
1475
- ) -> tuple[list[ExternalAIClient], str | None]:
1476
- """Filter ``members`` to a single solo-dispatch pick (step-9 P9).
1477
-
1478
- Loads the routing chain from ``agents/settings/.ai-council.yml`` and asks
1479
- :func:`select_solo_member` for the first chain entry whose member
1480
- is runtime-present. The probe is conservative: a member counts as
1481
- auth-valid iff ``build_members`` returned a runtime client for it
1482
- \u2014 build_members has already filtered out missing binaries / bad
1483
- keys via the ``skipped`` list. Deep CLI auth probes (e.g.
1484
- ``claude auth status``) are reserved for the shadow-mode path.
1485
-
1486
- Returns ``(filtered_members, marker)``. ``marker`` is a one-line
1487
- info banner the caller prints to stdout (``None`` when no banner
1488
- is needed, e.g. config missing). Returns the unfiltered list when
1489
- no solo member can be picked \u2014 caller never fails the decision.
1490
- """
1491
- try:
1492
- cfg = load_council_config(AI_COUNCIL_FILE)
1493
- except (CouncilConfigError, FileNotFoundError):
1494
- return members, None
1495
- if not cfg.routing.solo_member_fallback_chain:
1496
- return (
1497
- members,
1498
- "council:solo \u00b7 WARN \u00b7 --single requested but "
1499
- "routing.solo_member_fallback_chain is empty \u2014 "
1500
- "escalating to full council.",
1501
- )
1502
- runtime_names = {getattr(m, "name", "") for m in members}
1503
- pick = select_solo_member(
1504
- cfg.routing,
1505
- cfg.members,
1506
- auth_cache=AuthCache(),
1507
- probe=lambda name, _t: name in runtime_names,
1508
- )
1509
- if pick is None:
1510
- return (
1511
- members,
1512
- "council:solo \u00b7 WARN \u00b7 solo dispatch unavailable "
1513
- "(no chain member runtime-present) \u2014 escalating to "
1514
- "full council.",
1515
- )
1516
- filtered = [m for m in members if getattr(m, "name", "") == pick]
1517
- if not filtered:
1518
- # Defensive: ``pick`` came from runtime_names so this should
1519
- # be unreachable. If we ever get here, escalate rather than
1520
- # ship an empty council.
1521
- return (
1522
- members,
1523
- "council:solo \u00b7 WARN \u00b7 selected member vanished "
1524
- "between probe and filter \u2014 escalating to full council.",
1525
- )
1526
- return (
1527
- filtered,
1528
- f"council:solo \u00b7 dispatching to {pick} only "
1529
- f"(routing.solo_member_fallback_chain).",
1530
- )
1531
-
1532
-
1533
- def cmd_run(
1534
- args: argparse.Namespace,
1535
- *,
1536
- settings: dict[str, Any] | None = None,
1537
- members: list[ExternalAIClient] | None = None,
1538
- table: PriceTable | None = None,
1539
- ) -> int:
1540
- """Estimate, then run the council. Requires --confirm to spend."""
1541
- if settings is None:
1542
- settings = load_settings()
1543
- ai_cfg = (settings.get("ai_council") or {}) if isinstance(settings, dict) else {}
1544
- advisor_plans = _build_advisor_plans(ai_cfg, REPO_ROOT)
1545
- explicit_overrides = _parse_model_overrides(getattr(args, "model", None))
1546
- skipped: list[dict[str, Any]] = []
1547
- if members is None:
1548
- members = build_members(
1549
- settings,
1550
- invocation_mode=args.mode_override,
1551
- model_overrides=_advisor_model_overrides(
1552
- advisor_plans, explicit_overrides,
1553
- ),
1554
- siblings_overrides=_parse_siblings_overrides(getattr(args, "siblings", None)),
1555
- skipped=skipped,
1556
- )
1557
- if getattr(args, "single", False):
1558
- members, solo_banner = _apply_solo_dispatch(members)
1559
- if solo_banner:
1560
- sys.stdout.write(solo_banner + "\n")
1561
- _emit_shadow_slo_banner()
1562
- if table is None:
1563
- table = load_prices()
1564
- question, artefact = build_question(
1565
- input_path=Path(args.question), input_mode=args.input_mode,
1566
- max_tokens=_resolve_max_tokens(args, ai_cfg),
1567
- prompt_mode_override=getattr(args, "prompt_mode", None),
1568
- )
1569
- proceed, gate_exit, _necessity_result = _necessity_gate(
1570
- prompt=question.user_prompt,
1571
- lens=question.mode,
1572
- invocation=getattr(args, "invocation", "agent"),
1573
- proceed_anyway=getattr(args, "proceed_anyway", False),
1574
- ai_cfg=ai_cfg,
1575
- original_ask=getattr(args, "original_ask", "") or "",
1576
- )
1577
- if not proceed:
1578
- return gate_exit
1579
- _size_fit_gate(
1580
- prompt=question.user_prompt,
1581
- lens=question.mode,
1582
- members=members,
1583
- ai_cfg=ai_cfg,
1584
- )
1585
- project = detect_project_context(REPO_ROOT)
1586
- billable = [m for m in members if getattr(m, "billable", True)]
1587
- estimates = estimate(question, billable, table,
1588
- project=project, original_ask=args.original_ask,
1589
- advisor_plans=advisor_plans)
1590
- extra_calls, extra_usd = _consensus_cost_delta(
1591
- ai_cfg, question.mode, estimates, len(billable),
1592
- )
1593
- pr_extra_calls, pr_extra_usd = _peer_review_cost_delta(
1594
- ai_cfg, args, estimates, len(billable),
1595
- )
1596
- sys.stdout.write(
1597
- f"council:run · mode={question.mode} · members={len(members)} "
1598
- f"(billable={len(billable)})\n"
1599
- )
1600
- advisor_summary = _format_advisor_summary(advisor_plans, billable)
1601
- if advisor_summary:
1602
- sys.stdout.write(advisor_summary + "\n")
1603
- if skipped:
1604
- sys.stdout.write(format_install_hints(skipped) + "\n")
1605
- sys.stdout.write(
1606
- format_estimate_table(
1607
- billable, estimates,
1608
- consensus_delta_usd=extra_usd,
1609
- consensus_extra_calls=extra_calls,
1610
- peer_review_delta_usd=pr_extra_usd,
1611
- peer_review_extra_calls=pr_extra_calls,
1612
- ) + "\n"
1613
- )
1614
-
1615
- # Step-8 P1 — pre-run quota summary. After estimate / before
1616
- # dispatch so the user sees the budget shape before --confirm.
1617
- # Uncapped providers are omitted by ``quota_summary_line``; when
1618
- # no CLI member has a configured cap the summary is empty and we
1619
- # write nothing.
1620
- cli_members = [m for m in members if isinstance(m, CliClient)]
1621
- summary, warn_providers = quota_summary_line(cli_members)
1622
- if summary:
1623
- sys.stdout.write(summary + "\n")
1624
- for prov in warn_providers:
1625
- sys.stdout.write(f"council:quota · WARN · {prov} near limit\n")
1626
-
1627
- # Phase 8 step 5 — opt-in cost disclosure for non-debate lenses.
1628
- # Default mode is "off" for analysis / default (cheap enough that
1629
- # the disclosure is friction); users opt in by setting
1630
- # `lenses.<name>.cost_disclosure.mode` in ~/.event4u/agent-config/settings/.ai-council.yml.
1631
- disc_mode, disc_threshold, disc_show = _resolve_cost_disclosure(
1632
- ai_cfg, question.mode,
1633
- )
1634
- if disc_mode != "off":
1635
- run_estimate = estimate_debate_cost(
1636
- question, members, table,
1637
- rounds=1, project=project,
1638
- original_ask=args.original_ask,
1639
- advisor_plans=advisor_plans,
1640
- )
1641
- if disc_mode == "always" or (
1642
- disc_mode == "above_threshold"
1643
- and run_estimate.expected_usd > disc_threshold
1644
- ):
1645
- sys.stdout.write(
1646
- _format_cost_disclosure(
1647
- run_estimate, lens=question.mode,
1648
- show_per_member=disc_show,
1649
- )
1650
- )
1651
-
1652
- if not args.confirm:
1653
- sys.stdout.write(
1654
- "\nNo --confirm flag — estimate only. Re-run with --confirm to "
1655
- "invoke the council and write the response.\n"
1656
- )
1657
- return 0
1658
-
1659
- cost_cfg = ai_cfg.get("cost_budget") or {}
1660
- budget = CostBudget(
1661
- max_input_tokens=int(cost_cfg.get("max_input_tokens", 50_000)),
1662
- max_output_tokens=int(cost_cfg.get("max_output_tokens", 20_000)),
1663
- max_calls=int(cost_cfg.get("max_calls", 10)),
1664
- max_total_usd=float(cost_cfg.get("max_total_usd", 0.0) or 0.0),
1665
- )
1666
- rounds = _resolve_rounds(args, ai_cfg)
1667
- responses = consult(
1668
- members, question, budget,
1669
- table=table, project=project,
1670
- original_ask=args.original_ask, rounds=rounds,
1671
- advisor_plans=advisor_plans,
1672
- )
1673
- # Pipeline order (R4 verdict): deliberation → peer-review → consensus
1674
- # → synthesis. Peer-review anonymises only deliberation outputs;
1675
- # consensus-scoring runs on the de-anonymised findings.
1676
- persona_labels = build_persona_labels(advisor_plans, billable)
1677
- peer_review = _maybe_run_peer_review(
1678
- ai_cfg, args, question, members, responses, budget, table, project,
1679
- persona_labels=persona_labels,
1680
- )
1681
- consensus = _maybe_run_consensus(
1682
- ai_cfg, question, members, responses, budget, table, project, args,
1683
- )
1684
- estimated_total = sum(e.total_usd for e in estimates)
1685
- actual_total = 0.0
1686
- all_responses: list[CouncilResponse] = list(responses)
1687
- if peer_review is not None:
1688
- all_responses.extend(peer_review.responses)
1689
- if consensus is not None:
1690
- all_responses.extend(consensus.extraction_responses)
1691
- all_responses.extend(consensus.scoring_responses)
1692
- for r in all_responses:
1693
- if r.error:
1694
- continue
1695
- ce = estimate_cost(r.provider, r.model, r.input_tokens, r.output_tokens, table)
1696
- actual_total += ce.total_usd
1697
- payload = {
1698
- "schema_version": SCHEMA_VERSION,
1699
- "mode": question.mode,
1700
- "prompt_mode": getattr(args, "prompt_mode", None),
1701
- "prose_synthesis": getattr(args, "prose_synthesis", None),
1702
- "peer_review_enabled": _peer_review_active(ai_cfg, args),
1703
- "artefact": artefact,
1704
- "original_ask": args.original_ask,
1705
- "members": [f"{m.name}/{m.model}" for m in members],
1706
- "rounds": rounds,
1707
- "cost_usd_estimated": round(estimated_total, 6),
1708
- "cost_usd_actual": round(actual_total, 6),
1709
- "responses": _serialise_responses(responses),
1710
- }
1711
- if peer_review is not None:
1712
- payload["peer_review"] = _serialise_peer_review(peer_review)
1713
- if consensus is not None:
1714
- payload["consensus"] = _serialise_consensus(consensus)
1715
- out_path = _validate_council_output_path(
1716
- args.output, kind="responses", subcommand="run",
1717
- )
1718
- out_path.parent.mkdir(parents=True, exist_ok=True)
1719
- out_path.write_text(json.dumps(payload, indent=2) + "\n", encoding="utf-8")
1720
- sys.stdout.write(
1721
- f"\ncouncil:run · wrote {out_path} "
1722
- f"(estimated ${estimated_total:.4f} / actual ${actual_total:.4f})\n"
1723
- )
1724
- replay_path = _maybe_write_decision_replay(
1725
- ai_cfg=ai_cfg, lens=question.mode, out_path=out_path,
1726
- consensus=consensus, deliberation=responses,
1727
- original_ask=args.original_ask,
1728
- )
1729
- if replay_path is not None:
1730
- sys.stdout.write(f"council:run · wrote {replay_path}\n")
1731
- errors = [r for r in responses if r.error]
1732
- return 1 if errors and len(errors) == len(responses) else 0
1733
-
1734
-
1735
- def _debate_round_filename(round_number: int) -> str:
1736
- return f"debate-round-{round_number}.json"
1737
-
1738
-
1739
- def _write_debate_round(
1740
- out_dir: Path,
1741
- round_number: int,
1742
- responses: list[CouncilResponse],
1743
- *,
1744
- question: CouncilQuestion,
1745
- members: list[ExternalAIClient],
1746
- artefact: str,
1747
- original_ask: str,
1748
- total_planned_rounds: int,
1749
- table: PriceTable,
1750
- prompt_mode: str | None,
1751
- prose_synthesis: bool | None,
1752
- ) -> Path:
1753
- """Persist a single debate round as a self-contained JSON.
1754
-
1755
- Each round file mirrors the ``cmd_run`` payload shape — re-rendering
1756
- via ``council render <debate-round-N.json>`` works without special
1757
- handling. Round-specific keys (``debate_round``, ``debate_total_rounds``)
1758
- are additive so the renderer can ignore them safely.
1759
- """
1760
- out_dir.mkdir(parents=True, exist_ok=True)
1761
- actual_total = 0.0
1762
- for r in responses:
1763
- if r.error:
1764
- continue
1765
- ce = estimate_cost(r.provider, r.model, r.input_tokens, r.output_tokens, table)
1766
- actual_total += ce.total_usd
1767
- payload = {
1768
- "schema_version": SCHEMA_VERSION,
1769
- "mode": question.mode,
1770
- "prompt_mode": prompt_mode,
1771
- "prose_synthesis": prose_synthesis,
1772
- "artefact": artefact,
1773
- "original_ask": original_ask,
1774
- "members": [f"{m.name}/{m.model}" for m in members],
1775
- "debate_round": round_number,
1776
- "debate_total_rounds": total_planned_rounds,
1777
- "rounds": 1,
1778
- "cost_usd_actual": round(actual_total, 6),
1779
- "responses": _serialise_responses(responses),
1780
- }
1781
- out_path = out_dir / _debate_round_filename(round_number)
1782
- out_path.write_text(json.dumps(payload, indent=2) + "\n", encoding="utf-8")
1783
- return out_path
1784
-
1785
-
1786
- def _load_debate_seed(
1787
- path: Path,
1788
- expected_members: list[ExternalAIClient],
1789
- ) -> list[CouncilResponse]:
1790
- """Load `--continue-as-debate` seed: round-1 responses from a prior session.
1791
-
1792
- The seed file must be the JSON written by ``cmd_run`` (or a prior
1793
- debate round). Members + models must match the current invocation —
1794
- a mismatch is a hard error per the Phase 7 contract, not a silent
1795
- fallback. The host agent surfaces the mismatch and asks the user
1796
- to either re-run with matching members or drop ``--continue-as-debate``.
1797
- """
1798
- if not path.exists():
1799
- raise FileNotFoundError(
1800
- f"--continue-as-debate path not found: {path}"
1801
- )
1802
- payload = json.loads(path.read_text(encoding="utf-8"))
1803
- source_members = list(payload.get("members") or [])
1804
- expected_labels = [f"{m.name}/{m.model}" for m in expected_members]
1805
- if source_members != expected_labels:
1806
- raise CouncilDisabledError(
1807
- f"--continue-as-debate member mismatch: source session has "
1808
- f"{source_members!r}, current invocation has {expected_labels!r}. "
1809
- f"Re-run with matching members or drop --continue-as-debate."
1810
- )
1811
- return _deserialise_responses(payload.get("responses") or [])
1812
-
1813
-
1814
- def _make_debate_continue_prompt(
1815
- *, auto_continue: bool,
1816
- stream: Any = None,
1817
- ) -> Any:
1818
- """Build the on_continue callback for `run_debate()`.
1819
-
1820
- ``--auto-continue`` returns ``None`` so the orchestrator skips the
1821
- gate entirely (still subject to the hard-cap check). Interactive
1822
- mode prints the checkpoint line and reads y/N from stdin.
1823
- """
1824
- if auto_continue:
1825
- return None
1826
- out = stream or sys.stdout
1827
-
1828
- def _prompt(checkpoint: DebateCheckpoint) -> bool:
1829
- out.write(
1830
- f"\ndebate:checkpoint round={checkpoint.completed_round}/"
1831
- f"{checkpoint.total_planned_rounds} "
1832
- f"cost_so_far=${checkpoint.cost_so_far_usd:.4f} "
1833
- f"next_round_estimate=${checkpoint.next_round_estimate_usd:.4f} "
1834
- f"— continue? [y/N]: "
1835
- )
1836
- out.flush()
1837
- try:
1838
- answer = sys.stdin.readline().strip().lower()
1839
- except (EOFError, KeyboardInterrupt):
1840
- return False
1841
- return answer in {"y", "yes"}
1842
-
1843
- return _prompt
1844
-
1845
-
1846
- def cmd_debate(
1847
- args: argparse.Namespace,
1848
- *,
1849
- settings: dict[str, Any] | None = None,
1850
- members: list[ExternalAIClient] | None = None,
1851
- table: PriceTable | None = None,
1852
- ) -> int:
1853
- """Run a multi-round debate with progressive cost disclosure.
1854
-
1855
- Phase 7 contract: each member produces an initial position in
1856
- Round 1, then rebuts the strongest opposing position in subsequent
1857
- rounds. The orchestrator pauses after each round and asks the user
1858
- to continue (``--auto-continue`` bypasses the prompt). Round files
1859
- are persisted incrementally so an interrupted debate leaves a
1860
- recoverable trail.
1861
- """
1862
- if settings is None:
1863
- settings = load_settings()
1864
- ai_cfg = (settings.get("ai_council") or {}) if isinstance(settings, dict) else {}
1865
- advisor_plans = _build_advisor_plans(ai_cfg, REPO_ROOT)
1866
- explicit_overrides = _parse_model_overrides(getattr(args, "model", None))
1867
- skipped: list[dict[str, Any]] = []
1868
- if members is None:
1869
- members = build_members(
1870
- settings,
1871
- invocation_mode=args.mode_override,
1872
- model_overrides=_advisor_model_overrides(
1873
- advisor_plans, explicit_overrides,
1874
- ),
1875
- siblings_overrides=_parse_siblings_overrides(
1876
- getattr(args, "siblings", None),
1877
- ),
1878
- skipped=skipped,
1879
- )
1880
- if table is None:
1881
- table = load_prices()
1882
- question, artefact = build_question(
1883
- input_path=Path(args.question), input_mode=args.input_mode,
1884
- max_tokens=_resolve_max_tokens(args, ai_cfg),
1885
- prompt_mode_override="debate",
1886
- )
1887
- proceed, gate_exit, _necessity_result = _necessity_gate(
1888
- prompt=question.user_prompt,
1889
- lens="debate",
1890
- invocation=getattr(args, "invocation", "agent"),
1891
- proceed_anyway=getattr(args, "proceed_anyway", False),
1892
- ai_cfg=ai_cfg,
1893
- original_ask=getattr(args, "original_ask", "") or "",
1894
- )
1895
- if not proceed:
1896
- return gate_exit
1897
- _size_fit_gate(
1898
- prompt=question.user_prompt,
1899
- lens="debate",
1900
- members=members,
1901
- ai_cfg=ai_cfg,
1902
- )
1903
- project = detect_project_context(REPO_ROOT)
1904
- billable = [m for m in members if getattr(m, "billable", True)]
1905
-
1906
- # Resolve round count: explicit --rounds wins; otherwise default 2.
1907
- # Hard ceiling: ai_council.debate_max_rounds (Phase 0 reserved key).
1908
- max_rounds_cap = int(ai_cfg.get("debate_max_rounds", 4))
1909
- requested = (
1910
- int(args.rounds) if getattr(args, "rounds", None) is not None else 2
1911
- )
1912
- if requested < 1:
1913
- raise argparse.ArgumentTypeError(
1914
- f"--rounds must be >= 1 (got {requested})"
1915
- )
1916
- if requested > max_rounds_cap:
1917
- raise argparse.ArgumentTypeError(
1918
- f"--rounds={requested} exceeds debate_max_rounds={max_rounds_cap}; "
1919
- f"raise the cap in ~/.event4u/agent-config/settings/.ai-council.yml or lower --rounds."
1920
- )
1921
- rounds = requested
1922
-
1923
- estimates = estimate(
1924
- question, billable, table,
1925
- project=project, original_ask=args.original_ask,
1926
- advisor_plans=advisor_plans,
1927
- )
1928
- per_round_usd = sum(e.total_usd for e in estimates)
1929
- projected_total = per_round_usd * rounds
1930
- sys.stdout.write(
1931
- f"council:debate · members={len(members)} (billable={len(billable)}) "
1932
- f"· rounds={rounds} (cap={max_rounds_cap})\n"
1933
- )
1934
- advisor_summary = _format_advisor_summary(advisor_plans, billable)
1935
- if advisor_summary:
1936
- sys.stdout.write(advisor_summary + "\n")
1937
- if skipped:
1938
- sys.stdout.write(format_install_hints(skipped) + "\n")
1939
- sys.stdout.write(
1940
- format_estimate_table(billable, estimates) + "\n"
1941
- )
1942
- sys.stdout.write(
1943
- f" × {rounds} rounds (worst case, before progressive disclosure)\n"
1944
- f" PROJECTED TOTAL: ${projected_total:.4f}\n"
1945
- )
1946
-
1947
- # Phase 8 — pre-flight cost disclosure + hard refusal cap.
1948
- debate_estimate = estimate_debate_cost(
1949
- question, members, table,
1950
- rounds=rounds, project=project,
1951
- original_ask=args.original_ask,
1952
- advisor_plans=advisor_plans,
1953
- )
1954
- disc_mode, disc_threshold, disc_show = _resolve_cost_disclosure(
1955
- ai_cfg, "debate",
1956
- )
1957
- should_disclose = (
1958
- disc_mode == "always"
1959
- or (
1960
- disc_mode == "above_threshold"
1961
- and debate_estimate.expected_usd > disc_threshold
1962
- )
1963
- )
1964
- if should_disclose:
1965
- sys.stdout.write(
1966
- _format_cost_disclosure(
1967
- debate_estimate, lens="debate", show_per_member=disc_show,
1968
- )
1969
- )
1970
- cap = _debate_refusal_cap(ai_cfg)
1971
- if cap > 0 and debate_estimate.high_usd > cap:
1972
- sys.stderr.write(
1973
- f"❌ council:debate refused · high-end estimate "
1974
- f"${debate_estimate.high_usd:.4f} exceeds "
1975
- f"debate.max_cost_usd=${cap:.2f}. Lower --rounds, drop "
1976
- f"members, or raise the cap in ~/.event4u/agent-config/settings/.ai-council.yml.\n"
1977
- )
1978
- return 4
1979
-
1980
- if not args.confirm:
1981
- sys.stdout.write(
1982
- "\nNo --confirm flag — estimate only. Re-run with --confirm to "
1983
- "start the debate.\n"
1984
- )
1985
- return 0
1986
-
1987
- cost_cfg = ai_cfg.get("cost_budget") or {}
1988
- budget = CostBudget(
1989
- max_input_tokens=int(cost_cfg.get("max_input_tokens", 50_000)),
1990
- max_output_tokens=int(cost_cfg.get("max_output_tokens", 20_000)),
1991
- max_calls=int(cost_cfg.get("max_calls", 10)),
1992
- max_total_usd=float(cost_cfg.get("max_total_usd", 0.0) or 0.0),
1993
- )
1994
-
1995
- out_dir = _validate_council_output_path(
1996
- args.output, kind="responses", subcommand="debate",
1997
- )
1998
- seed: list[CouncilResponse] | None = None
1999
- if getattr(args, "continue_as_debate", None):
2000
- seed = _load_debate_seed(Path(args.continue_as_debate), billable)
2001
- sys.stdout.write(
2002
- f"council:debate · seeding round 1 from "
2003
- f"{args.continue_as_debate} ({len(seed)} responses)\n"
2004
- )
2005
-
2006
- written: list[Path] = []
2007
-
2008
- def _on_round_complete(round_number: int, results: list[CouncilResponse]) -> None:
2009
- path = _write_debate_round(
2010
- out_dir, round_number, results,
2011
- question=question, members=members,
2012
- artefact=artefact, original_ask=args.original_ask,
2013
- total_planned_rounds=rounds, table=table,
2014
- prompt_mode="debate",
2015
- prose_synthesis=getattr(args, "prose_synthesis", None),
2016
- )
2017
- written.append(path)
2018
- errors = [r for r in results if r.error]
2019
- sys.stdout.write(
2020
- f"council:debate · wrote {path} "
2021
- f"({len(results) - len(errors)}/{len(results)} ok)\n"
2022
- )
2023
-
2024
- on_continue = _make_debate_continue_prompt(
2025
- auto_continue=bool(getattr(args, "auto_continue", False)),
2026
- )
2027
-
2028
- try:
2029
- all_rounds = run_debate(
2030
- members, question,
2031
- budget=budget, table=table, project=project,
2032
- original_ask=args.original_ask,
2033
- max_rounds=rounds,
2034
- on_round_complete=_on_round_complete,
2035
- on_continue=on_continue,
2036
- advisor_plans=advisor_plans,
2037
- seed_round_1=seed,
2038
- )
2039
- except DebateCapExceeded as exc:
2040
- sys.stderr.write(
2041
- f"❌ council:debate cap reached after round {exc.completed_round}: "
2042
- f"{exc}\n"
2043
- f"Partial debate persisted under {out_dir} "
2044
- f"({len(written)} rounds).\n"
2045
- )
2046
- return 3
2047
-
2048
- actual_total = 0.0
2049
- for rnd in all_rounds:
2050
- for r in rnd:
2051
- if r.error:
2052
- continue
2053
- ce = estimate_cost(
2054
- r.provider, r.model, r.input_tokens, r.output_tokens, table,
2055
- )
2056
- actual_total += ce.total_usd
2057
- sys.stdout.write(
2058
- f"\ncouncil:debate · {len(all_rounds)} round(s) complete · "
2059
- f"actual ${actual_total:.4f} (cap projection ${projected_total:.4f})\n"
2060
- )
2061
- errors_last = [r for r in all_rounds[-1] if r.error] if all_rounds else []
2062
- return 1 if errors_last and len(errors_last) == len(all_rounds[-1]) else 0
2063
-
2064
-
2065
- def cmd_render(args: argparse.Namespace) -> int:
2066
- """Re-render a saved responses JSON to the markdown report.
2067
-
2068
- Lens resolution order: explicit ``--prompt-mode`` > ``prompt_mode``
2069
- in the payload > ``mode`` in the payload > ``None`` (default decision
2070
- template). R4 Q4 escape hatch ``--prose-synthesis`` overrides the
2071
- table. ``--output`` writes to ``agents/runtime/council/sessions/`` (enforced);
2072
- omit it for stdout.
2073
- """
2074
- payload = json.loads(Path(args.responses).read_text(encoding="utf-8"))
2075
- items = payload.get("responses") or []
2076
- explicit = getattr(args, "prompt_mode", None)
2077
- mode = explicit or payload.get("prompt_mode") or payload.get("mode")
2078
- prose = getattr(args, "prose_synthesis", None)
2079
- if prose is None:
2080
- prose = payload.get("prose_synthesis")
2081
- consensus = _deserialise_consensus(payload.get("consensus"))
2082
- peer_review = _deserialise_peer_review(payload.get("peer_review"))
2083
- body = render(
2084
- _deserialise_responses(items),
2085
- mode=mode,
2086
- prose_synthesis=prose,
2087
- consensus=consensus,
2088
- peer_review=peer_review,
2089
- )
2090
- if getattr(args, "output", None):
2091
- out_path = _validate_council_output_path(
2092
- args.output, kind="sessions", subcommand="render",
2093
- )
2094
- out_path.parent.mkdir(parents=True, exist_ok=True)
2095
- out_path.write_text(body + "\n", encoding="utf-8")
2096
- sys.stdout.write(f"council:render · wrote {out_path}\n")
2097
- return 0
2098
- sys.stdout.write(body + "\n")
2099
- return 0
2100
-
2101
-
2102
- def _cmd_replay_low_impact_stats(args: argparse.Namespace) -> int:
2103
- """Summarise the session's ``low-impact-resolutions.md`` (Phase 11).
2104
-
2105
- The log file lives next to the ``responses`` JSON. Missing or empty
2106
- log → prints an explicit "no entries" line and returns 0 (a session
2107
- with no low-impact resolutions is not an error).
2108
- """
2109
- from scripts.ai_council.low_impact import ( # noqa: WPS433 — local import
2110
- parse_low_impact_log,
2111
- render_low_impact_stats,
2112
- )
2113
-
2114
- responses_path = Path(args.responses)
2115
- log_path = responses_path.parent / "low-impact-resolutions.md"
2116
- if not log_path.exists():
2117
- sys.stdout.write(
2118
- "council:replay · no low-impact-resolutions.md alongside "
2119
- f"{responses_path} — session had no fast-path entries.\n",
2120
- )
2121
- return 0
2122
- body = log_path.read_text(encoding="utf-8")
2123
- stats = parse_low_impact_log(body)
2124
- out = render_low_impact_stats(stats)
2125
- if getattr(args, "output", None):
2126
- target = _validate_council_output_path(
2127
- args.output, kind="sessions", subcommand="replay",
2128
- )
2129
- target.parent.mkdir(parents=True, exist_ok=True)
2130
- target.write_text(out, encoding="utf-8")
2131
- sys.stdout.write(f"council:replay · wrote {target}\n")
2132
- return 0
2133
- sys.stdout.write(out)
2134
- return 0
2135
-
2136
-
2137
- def cmd_replay(args: argparse.Namespace) -> int:
2138
- """Re-render the ``decision-replay.md`` audit trail (Phase 9).
2139
-
2140
- Reads a saved ``council:run`` JSON payload, rebuilds the consensus
2141
- bundle, and emits the replay markdown to stdout (default) or to
2142
- ``--output``. Pure re-projection — no model calls. Returns 2 when
2143
- the payload lacks consensus data (Phase 9 prerequisite).
2144
-
2145
- When ``--low-impact-stats`` is set, the consensus replay is skipped
2146
- and the session's ``low-impact-resolutions.md`` (Phase 11) is
2147
- summarised instead — count, status breakdown, members used, cost.
2148
- """
2149
- if getattr(args, "low_impact_stats", False):
2150
- return _cmd_replay_low_impact_stats(args)
2151
- payload = json.loads(Path(args.responses).read_text(encoding="utf-8"))
2152
- consensus = _deserialise_consensus(payload.get("consensus"))
2153
- if consensus is None:
2154
- sys.stderr.write(
2155
- "❌ council:replay: payload has no `consensus` block — "
2156
- "rerun with consensus_scoring enabled for this lens.\n"
2157
- )
2158
- return 2
2159
- deliberation = _deserialise_responses(payload.get("responses") or [])
2160
- include_args = (
2161
- bool(args.include_member_arguments)
2162
- if args.include_member_arguments is not None
2163
- else True
2164
- )
2165
- body = render_decision_replay(
2166
- DecisionReplayInputs(
2167
- findings=list(consensus.findings),
2168
- scores=list(consensus.scores),
2169
- metadata=dict(consensus.metadata),
2170
- deliberation=deliberation,
2171
- original_ask=str(payload.get("original_ask", "")),
2172
- include_member_arguments=include_args,
2173
- ),
2174
- )
2175
- if getattr(args, "output", None):
2176
- out_path = _validate_council_output_path(
2177
- args.output, kind="sessions", subcommand="replay",
2178
- )
2179
- out_path.parent.mkdir(parents=True, exist_ok=True)
2180
- out_path.write_text(body, encoding="utf-8")
2181
- sys.stdout.write(f"council:replay · wrote {out_path}\n")
2182
- else:
2183
- sys.stdout.write(body)
2184
- return 0
2185
-
2186
-
2187
- # ── argparse + main ─────────────────────────────────────────────────
2188
-
2189
-
2190
- def _parse_model_overrides(items: list[str] | None) -> dict[str, str]:
2191
- """Parse repeated `--model name=model-id` flags into a dict.
2192
-
2193
- Empty/None list → empty dict (no override). Bad shape raises
2194
- `argparse.ArgumentTypeError` so the CLI surfaces the error.
2195
- """
2196
- out: dict[str, str] = {}
2197
- for raw in items or []:
2198
- if "=" not in raw:
2199
- raise argparse.ArgumentTypeError(
2200
- f"--model expects '<member>=<model-id>', got {raw!r}."
2201
- )
2202
- name, model = raw.split("=", 1)
2203
- name, model = name.strip(), model.strip()
2204
- if not name or not model:
2205
- raise argparse.ArgumentTypeError(
2206
- f"--model member and model-id must both be non-empty: {raw!r}."
2207
- )
2208
- out[name] = model
2209
- return out
2210
-
2211
-
2212
- def _parse_siblings_overrides(items: list[str] | None) -> dict[str, list[str]]:
2213
- """Parse repeated `--siblings name=model1,model2[,...]` flags.
2214
-
2215
- Requires ≥ 2 distinct, non-empty models per provider — sibling
2216
- mode without diversity has no purpose. Repeating the same provider
2217
- flag is rejected as ambiguous.
2218
- """
2219
- out: dict[str, list[str]] = {}
2220
- for raw in items or []:
2221
- if "=" not in raw:
2222
- raise argparse.ArgumentTypeError(
2223
- f"--siblings expects '<member>=<model1>,<model2>[,...]', got {raw!r}."
2224
- )
2225
- name, models_csv = raw.split("=", 1)
2226
- name = name.strip()
2227
- models = [m.strip() for m in models_csv.split(",") if m.strip()]
2228
- if not name or not models:
2229
- raise argparse.ArgumentTypeError(
2230
- f"--siblings member and model list must both be non-empty: {raw!r}."
2231
- )
2232
- if len(set(models)) < 2:
2233
- raise argparse.ArgumentTypeError(
2234
- f"--siblings requires ≥ 2 distinct models for {name!r}, got {models!r}."
2235
- )
2236
- if name in out:
2237
- raise argparse.ArgumentTypeError(
2238
- f"--siblings repeated for member {name!r}; combine into one flag."
2239
- )
2240
- out[name] = models
2241
- return out
2242
-
2243
-
2244
- def _add_common_input_args(p: argparse.ArgumentParser) -> None:
2245
- p.add_argument("question", type=str,
2246
- help="Path to the question file (text or roadmap).")
2247
- p.add_argument("--input-mode", choices=["prompt", "roadmap"],
2248
- default="prompt",
2249
- help="How to bundle the file (default: prompt).")
2250
- p.add_argument("--prompt-mode",
2251
- choices=["pr", "design", "optimize", "analysis"],
2252
- default=None, dest="prompt_mode",
2253
- help="Lens-override for the system-prompt addendum. "
2254
- "The bundle shape stays as --input-mode; only "
2255
- "the per-mode neutrality addendum is swapped "
2256
- "(see scripts/ai_council/prompts.py _MODE_TABLE). "
2257
- "Routed by the /council pr|design|optimize|"
2258
- "analysis wrappers.")
2259
- p.add_argument("--max-tokens", type=int, default=None,
2260
- help="Per-member output budget. Default reads "
2261
- "ai_council.max_output_tokens from .agent-settings.yml "
2262
- "(2048 if unset). 0 = unlimited (widened to the safe "
2263
- "provider ceiling before the SDK call).")
2264
- p.add_argument("--mode-override", choices=["api", "manual"], default=None,
2265
- help="Override every member's transport mode.")
2266
- p.add_argument("--model", action="append", default=None, dest="model",
2267
- metavar="MEMBER=MODEL_ID",
2268
- help="Per-invocation model override, e.g. "
2269
- "--model anthropic=claude-sonnet-4-5. Repeatable. "
2270
- "Wins over `ai_council.members.<name>.model` in "
2271
- ".agent-settings.yml; the settings file is not "
2272
- "modified.")
2273
- p.add_argument("--siblings", action="append", default=None, dest="siblings",
2274
- metavar="MEMBER=MODEL1,MODEL2[,...]",
2275
- help="Fan one provider out to ≥ 2 sibling models in a "
2276
- "single run, e.g. --siblings anthropic=claude-sonnet-4-5,"
2277
- "claude-opus-4-1. Each model becomes its own billable "
2278
- "member with independent cost tracking. Mutually "
2279
- "exclusive with --model for the same provider; "
2280
- "requires the provider to be enabled with mode=api. "
2281
- "Single-provider degraded-run strategy per ai-council "
2282
- "skill.")
2283
- p.add_argument("--original-ask", default="",
2284
- help="The user's framing sentence (flows into handoff).")
2285
- p.add_argument("--peer-review", dest="peer_review", action="store_true",
2286
- default=False,
2287
- help="Run an anonymous peer-review pass after the main "
2288
- "deliberation. Each member critiques the others' "
2289
- "(anonymised) responses for blind spots before "
2290
- "synthesis. Adds N extra API calls. Opt-in per the "
2291
- "R2 verdict; also accepts ai_council.peer_review."
2292
- "enabled: true in ~/.event4u/agent-config/settings/.ai-council.yml.")
2293
-
2294
-
2295
- def cmd_shadow_report(args: argparse.Namespace) -> int:
2296
- """Print the 7-day rolling disagreement rate + SLO status (step-9 P10)."""
2297
- from pathlib import Path as _Path
2298
-
2299
- from scripts.ai_council import shadow_dispatch as _sd
2300
-
2301
- log_path = _Path(args.log) if args.log else _sd.SHADOW_LOG_PATH
2302
- rate, n = _sd.compute_disagreement_rate(
2303
- log_path, window_days=int(args.window_days)
2304
- )
2305
- print(_sd.slo_banner(rate, n))
2306
- return 0
2307
-
2308
-
2309
- def cmd_quota(
2310
- args: argparse.Namespace,
2311
- *,
2312
- settings: dict[str, Any] | None = None,
2313
- ) -> int:
2314
- """Dump today's CLI-quota state (step-8 P1, D1).
2315
-
2316
- Reads ``~/.event4u/agent-config/cli-calls.json`` plus the configured
2317
- caps from ``.agent-settings.yml`` and prints one line per provider
2318
- that has a configured ``max_calls_per_day``. ``--reset <provider>``
2319
- (gated behind ``--confirm``) clears the counter for that provider.
2320
- """
2321
- s = settings if settings is not None else load_settings()
2322
- ai_cfg = (s.get("ai_council") or {}) if isinstance(s, dict) else {}
2323
- cli_budget_cfg = (
2324
- (ai_cfg.get("cli_call_budget") or {}) if isinstance(ai_cfg, dict) else {}
2325
- )
2326
- caps = (
2327
- (cli_budget_cfg.get("max_calls_per_day") or {})
2328
- if isinstance(cli_budget_cfg, dict)
2329
- else {}
2330
- )
2331
- warn_at = (
2332
- float(cli_budget_cfg.get("warn_at", 0.8))
2333
- if isinstance(cli_budget_cfg, dict)
2334
- else 0.8
2335
- )
2336
-
2337
- if getattr(args, "reset", None):
2338
- provider = args.reset
2339
- if not getattr(args, "confirm", False):
2340
- sys.stderr.write(
2341
- f"❌ council:quota: --reset {provider} requires --confirm.\n"
2342
- )
2343
- return 2
2344
- reset_cli_call_counts(provider=provider)
2345
- sys.stdout.write(f"council:quota · reset · {provider}\n")
2346
- return 0
2347
-
2348
- counts = load_cli_call_counts()
2349
- if not caps:
2350
- sys.stdout.write(
2351
- "council:quota · no providers have a configured "
2352
- "cli_call_budget.max_calls_per_day cap.\n"
2353
- )
2354
- return 0
2355
- for provider in sorted(caps):
2356
- limit = int(caps[provider])
2357
- used = int(counts.get(provider, 0))
2358
- ratio = used / limit if limit > 0 else 0.0
2359
- status = "ok"
2360
- if used >= limit:
2361
- status = "exhausted"
2362
- elif ratio >= warn_at:
2363
- status = "warn"
2364
- sys.stdout.write(
2365
- f"council:quota · {provider} · {used}/{limit} · {status}\n"
2366
- )
2367
- return 0
2368
-
2369
-
2370
- def build_parser() -> argparse.ArgumentParser:
2371
- parser = argparse.ArgumentParser(
2372
- prog="agent-config council",
2373
- description="Non-interactive council orchestration.",
2374
- )
2375
- sub = parser.add_subparsers(dest="cmd", required=True)
2376
-
2377
- p_est = sub.add_parser("estimate", help="Pre-call cost preview (no spend).")
2378
- _add_common_input_args(p_est)
2379
- p_est.add_argument("--debate", action="store_true", default=False,
2380
- help="Render the round-by-round projection for a "
2381
- "debate run (one call per member per round). "
2382
- "Progressive disclosure may stop the debate "
2383
- "early — this is an upper bound.")
2384
- p_est.add_argument("--rounds", type=int, default=None,
2385
- help="Debate round count for --debate. Defaults to "
2386
- "ai_council.min_rounds (typically 2); capped "
2387
- "at ai_council.debate_max_rounds (typically 4).")
2388
-
2389
- p_run = sub.add_parser("run", help="Run the council; --confirm required to spend.")
2390
- _add_common_input_args(p_run)
2391
- p_run.add_argument("--output", required=True,
2392
- help="Path to write the responses JSON.")
2393
- p_run.add_argument("--confirm", action="store_true",
2394
- help="Required to actually invoke the council.")
2395
- p_run.add_argument("--rounds", type=int, default=None,
2396
- help="Number of debate rounds (1-3). Explicit override; "
2397
- "wins over --depth. Defaults to ai_council.min_rounds "
2398
- "in .agent-settings.yml (or 2 if unset).")
2399
- p_run.add_argument("--depth", choices=["standard", "deep"], default="standard",
2400
- help="Reasoning-depth tier. 'deep' floors rounds at "
2401
- "ai_council.deep_min_rounds (max'd with min_rounds) "
2402
- "for architecture, refactoring, or bug-diagnosis "
2403
- "artefacts. Set by the host agent when the consuming "
2404
- "rule/skill/command declares council_depth: deep. "
2405
- "Overridden by explicit --rounds.")
2406
- p_run.add_argument("--invocation", choices=["agent", "user_explicit"],
2407
- default="agent",
2408
- help="Source signal for the necessity classifier "
2409
- "(Phase 6). 'agent' = autonomous (default; silent "
2410
- "skip when unnecessary). 'user_explicit' = manual "
2411
- "user invocation (educate path when unnecessary, "
2412
- "requires --proceed-anyway to override).")
2413
- p_run.add_argument("--proceed-anyway", action="store_true",
2414
- dest="proceed_anyway", default=False,
2415
- help="Override the necessity-classifier skip / educate "
2416
- "verdict for this invocation (Phase 6). Has no "
2417
- "effect when the classifier verdict is "
2418
- "`necessary` or `borderline`.")
2419
- p_run.add_argument("--single", action="store_true", default=False,
2420
- help="Dispatch to a single member from "
2421
- "routing.solo_member_fallback_chain (step-9 P9). "
2422
- "Falls back to the full council when the chain is "
2423
- "empty or no chain member is runtime-present. "
2424
- "Overridden by env "
2425
- "AGENT_CONFIG_FORCE_FULL_COUNCIL=1.")
2426
- _add_prose_synthesis_arg(p_run)
2427
-
2428
- p_deb = sub.add_parser(
2429
- "debate",
2430
- help="Multi-round debate with progressive cost disclosure (Phase 7).",
2431
- )
2432
- _add_common_input_args(p_deb)
2433
- p_deb.add_argument("--output", required=True,
2434
- help="Directory to write debate-round-N.json files.")
2435
- p_deb.add_argument("--confirm", action="store_true",
2436
- help="Required to actually start the debate.")
2437
- p_deb.add_argument("--rounds", type=int, default=None,
2438
- help="Number of debate rounds (default 2). Capped by "
2439
- "ai_council.debate_max_rounds in ~/.event4u/agent-config/settings/.ai-council.yml.")
2440
- p_deb.add_argument("--auto-continue", action="store_true",
2441
- default=False, dest="auto_continue",
2442
- help="Skip the between-round y/N prompt. The hard cap "
2443
- "against cost_budget.max_total_usd still applies.")
2444
- p_deb.add_argument("--continue-as-debate", default=None,
2445
- dest="continue_as_debate", metavar="PATH",
2446
- help="Seed round 1 from an existing council session "
2447
- "JSON. Members + models must match the current "
2448
- "invocation.")
2449
- p_deb.add_argument("--invocation", choices=["agent", "user_explicit"],
2450
- default="agent",
2451
- help="Source signal for the necessity classifier "
2452
- "(Phase 6). 'agent' = autonomous (default; silent "
2453
- "skip when unnecessary). 'user_explicit' = manual "
2454
- "user invocation (educate path when unnecessary, "
2455
- "requires --proceed-anyway to override).")
2456
- p_deb.add_argument("--proceed-anyway", action="store_true",
2457
- dest="proceed_anyway", default=False,
2458
- help="Override the necessity-classifier skip / educate "
2459
- "verdict for this invocation (Phase 6). Has no "
2460
- "effect when the classifier verdict is "
2461
- "`necessary` or `borderline`.")
2462
- _add_prose_synthesis_arg(p_deb)
2463
-
2464
- p_ren = sub.add_parser("render", help="Re-render a saved responses JSON.")
2465
- p_ren.add_argument("responses",
2466
- help="Path to the JSON written by `council run`.")
2467
- p_ren.add_argument("--prompt-mode",
2468
- choices=["default", "pr", "design", "optimize", "analysis",
2469
- "prompt", "roadmap", "diff", "files"],
2470
- default=None, dest="prompt_mode",
2471
- help="Override the synthesis-template lens. Defaults "
2472
- "to the `mode` recorded in the responses JSON.")
2473
- p_ren.add_argument("--output", default=None,
2474
- help="Write the rendered markdown to a file under "
2475
- "agents/runtime/council/sessions/ (enforced). Omit for "
2476
- "stdout. Prefer this over shell redirects so "
2477
- "the canonical-path check fires at write-time.")
2478
- _add_prose_synthesis_arg(p_ren)
2479
-
2480
- p_rep = sub.add_parser(
2481
- "replay",
2482
- help="Re-render decision-replay.md from a saved responses JSON (Phase 9).",
2483
- )
2484
- p_rep.add_argument("responses",
2485
- help="Path to the JSON written by `council run`.")
2486
- p_rep.add_argument("--output", default=None,
2487
- help="Optional file to write the replay markdown. "
2488
- "Defaults to stdout.")
2489
- rep_group = p_rep.add_mutually_exclusive_group()
2490
- rep_group.add_argument("--redact-member-arguments",
2491
- dest="include_member_arguments",
2492
- action="store_const", const=False, default=None,
2493
- help="Emit the redacted view (consensus + dissent "
2494
- "counts only, no per-member arguments).")
2495
- rep_group.add_argument("--include-member-arguments",
2496
- dest="include_member_arguments",
2497
- action="store_const", const=True,
2498
- help="Include per-member arguments (default).")
2499
- p_rep.add_argument("--low-impact-stats", action="store_true", default=False,
2500
- help="Skip the decision replay and print a summary of "
2501
- "low-impact fast-path resolutions for the session "
2502
- "(parses `low-impact-resolutions.md` alongside the "
2503
- "responses JSON).")
2504
-
2505
- p_quo = sub.add_parser(
2506
- "quota",
2507
- help="Dump today's CLI-quota state and configured caps (step-8 P1).",
2508
- )
2509
- p_quo.add_argument("--reset", default=None, metavar="PROVIDER",
2510
- help="Reset today's counter for one provider. "
2511
- "Requires --confirm.")
2512
- p_quo.add_argument("--confirm", action="store_true", default=False,
2513
- help="Confirm a mutating --reset operation.")
2514
-
2515
- p_sha = sub.add_parser(
2516
- "shadow-report",
2517
- help="Read agents/runtime/council/shadow-log.jsonl and print the 7-day "
2518
- "rolling disagreement rate + SLO status (step-9 P10).",
2519
- )
2520
- p_sha.add_argument("--log", default=None,
2521
- help="Path to the shadow log (default: "
2522
- "agents/runtime/council/shadow-log.jsonl).")
2523
- p_sha.add_argument("--window-days", type=int, default=7,
2524
- help="Rolling window in days (default: 7).")
2525
-
2526
- return parser
2527
-
2528
-
2529
- def _add_prose_synthesis_arg(p: argparse.ArgumentParser) -> None:
2530
- """R4 Q4 escape hatch — toggle structured vs prose synthesis."""
2531
- group = p.add_mutually_exclusive_group()
2532
- group.add_argument("--prose-synthesis", dest="prose_synthesis",
2533
- action="store_const", const=True, default=None,
2534
- help="Force open-ended prose synthesis (bare slot) "
2535
- "regardless of lens. R4 Q4 escape hatch.")
2536
- group.add_argument("--no-prose-synthesis", dest="prose_synthesis",
2537
- action="store_const", const=False,
2538
- help="Force the structured default decision-lens "
2539
- "template even on a creative lens "
2540
- "(design / optimize). Symmetric escape hatch.")
2541
-
2542
-
2543
- def main(argv: list[str] | None = None) -> int:
2544
- args = build_parser().parse_args(argv)
2545
- try:
2546
- if args.cmd == "estimate":
2547
- return cmd_estimate(args)
2548
- if args.cmd == "run":
2549
- return cmd_run(args)
2550
- if args.cmd == "debate":
2551
- return cmd_debate(args)
2552
- if args.cmd == "render":
2553
- return cmd_render(args)
2554
- if args.cmd == "replay":
2555
- return cmd_replay(args)
2556
- if args.cmd == "quota":
2557
- return cmd_quota(args)
2558
- if args.cmd == "shadow-report":
2559
- return cmd_shadow_report(args)
2560
- except CouncilDisabledError as exc:
2561
- sys.stderr.write(f"❌ council:{args.cmd}: {exc}\n")
2562
- return 2
2563
- except (BundleTooLarge, InvalidModeError, FileNotFoundError,
2564
- argparse.ArgumentTypeError) as exc:
2565
- sys.stderr.write(f"❌ council:{args.cmd}: {exc}\n")
2566
- return 2
2567
- return 1
2568
-
2569
-
2570
- if __name__ == "__main__":
2571
- raise SystemExit(main())