@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,1385 +0,0 @@
1
- """External-AI clients for the council.
2
-
3
- Mirrors the contract from `scripts/skill_trigger_eval.py`:
4
- - Tokens come exclusively from ``~/.event4u/agent-config/<provider>.key``
5
- (legacy ``~/.config/agent-config/<provider>.key`` is read as a
6
- fallback so pre-2.4 installs keep working until the user moves the
7
- files into the new namespace).
8
- - File mode must be exactly 0o600. Drift is a hard abort.
9
- - No environment-variable fallback. No keychain fallback.
10
- - Real SDKs (`anthropic`, `openai`) are *soft* dependencies — the
11
- module imports cleanly without them; only `ask()` requires them.
12
-
13
- Tests inject mock clients via the `client=` constructor argument and
14
- never hit the real API.
15
-
16
- Mode contract:
17
- - `billable=True` clients (AnthropicClient, OpenAIClient, GeminiClient,
18
- XAIClient, PerplexityClient) participate in the cost gate — projected
19
- USD spend is checked before each call.
20
- - `billable=False` clients (ManualClient, vendor-official CliClient
21
- subclasses — AnthropicCliClient, OpenAICliClient, GeminiCliClient)
22
- skip the USD cost gate entirely. Spend = $0 to us; provider-side
23
- limits are the user's concern.
24
- - `billable=True` CLI subclasses (XAICliClient, PerplexityCliClient)
25
- wrap community-maintained CLIs that consume the same API key as
26
- their `api` counterparts — they participate in the USD cost gate.
27
- `mode: cli` here is an ergonomic shortcut, not a billing change.
28
-
29
- CLI subclasses additionally consult the optional
30
- `cli_call_budget.max_calls_per_day.<provider>` quota with state
31
- persisted at `~/.event4u/agent-config/cli-calls.json` (daily UTC reset).
32
- """
33
-
34
- from __future__ import annotations
35
-
36
- import json
37
- import shutil
38
- import stat
39
- import subprocess
40
- import sys
41
- import time
42
- from abc import ABC, abstractmethod
43
- from dataclasses import dataclass, field
44
- from datetime import datetime, timezone
45
- from pathlib import Path
46
- from typing import TextIO
47
-
48
- from scripts._lib import user_global_paths
49
-
50
- ANTHROPIC_KEY_FILENAME = "anthropic.key"
51
- OPENAI_KEY_FILENAME = "openai.key"
52
-
53
- #: Canonical write target under the new namespace. Reads route via
54
- #: :func:`_resolve_key_path` so a key still sitting in the legacy
55
- #: ``~/.config/agent-config/`` tree keeps working.
56
- ANTHROPIC_KEY_PATH = user_global_paths.write_target(ANTHROPIC_KEY_FILENAME)
57
- OPENAI_KEY_PATH = user_global_paths.write_target(OPENAI_KEY_FILENAME)
58
-
59
-
60
- def _resolve_key_path(filename: str) -> Path:
61
- """Return the active key path, preferring the new namespace."""
62
- found = user_global_paths.resolve_with_fallback(filename)
63
- if found is not None:
64
- return found
65
- return user_global_paths.write_target(filename)
66
-
67
- DEFAULT_ANTHROPIC_MODEL = "claude-sonnet-4-5"
68
- DEFAULT_OPENAI_MODEL = "gpt-4o"
69
- DEFAULT_GEMINI_MODEL = "gemini-2.5-pro"
70
- DEFAULT_XAI_MODEL = "grok-4"
71
- DEFAULT_PERPLEXITY_MODEL = "sonar-pro"
72
-
73
- #: OpenAI-API-compatible endpoints. xAI and Perplexity both expose the
74
- #: ``/v1/chat/completions`` shape, so their clients reuse the ``openai``
75
- #: SDK with a custom ``base_url``. Gemini has its own SDK (``google-genai``).
76
- XAI_BASE_URL = "https://api.x.ai/v1"
77
- PERPLEXITY_BASE_URL = "https://api.perplexity.ai"
78
-
79
- #: Per-call output budget when no caller-supplied value reaches `ask()`.
80
- #: The CLI resolves the live default from `ai_council.max_output_tokens`
81
- #: in `.agent-settings.yml`; this constant is only the abstract-base /
82
- #: direct-API fallback when nothing else is wired up.
83
- DEFAULT_MAX_TOKENS = 2048
84
-
85
- #: Expansion target when the user sets `max_output_tokens: 0` ("unlimited")
86
- #: in settings. Anthropic requires `max_tokens` to be a positive integer,
87
- #: so 0 is widened to this safe ceiling before the SDK call. Big enough
88
- #: for current frontier models (Sonnet/GPT-4o headroom ≥ 16k); raise
89
- #: explicitly in settings if a larger budget is genuinely needed.
90
- UNLIMITED_TOKENS_FALLBACK = 16384
91
-
92
- # OpenAI reasoning models (o1, o3, o4 families) reject `max_tokens` and the
93
- # `system` role; they require `max_completion_tokens` and accept only `user`
94
- # (and `developer`) messages.
95
- _REASONING_PREFIXES = ("o1", "o3", "o4")
96
-
97
-
98
- def _is_reasoning_model(model: str) -> bool:
99
- name = model.lower()
100
- return any(name == p or name.startswith(p + "-") for p in _REASONING_PREFIXES)
101
-
102
-
103
- class KeyGateError(RuntimeError):
104
- """Raised when a provider key file violates the 0600 contract."""
105
-
106
-
107
- @dataclass
108
- class CouncilResponse:
109
- """Normalised output from a single council member."""
110
-
111
- provider: str
112
- model: str
113
- text: str
114
- input_tokens: int = 0
115
- output_tokens: int = 0
116
- latency_ms: int = 0
117
- error: str | None = None
118
- metadata: dict[str, object] = field(default_factory=dict)
119
-
120
-
121
- def _load_key(path: Path, prefix: str, install_script: str) -> str:
122
- """Shared 0600-gated key loader. Refuses anything outside the contract."""
123
- if not path.exists():
124
- raise KeyGateError(
125
- f"Key not found at {path}.\n"
126
- f" Install it with: bash {install_script}"
127
- )
128
- st = path.stat()
129
- mode = stat.S_IMODE(st.st_mode)
130
- if mode != 0o600:
131
- raise KeyGateError(
132
- f"Unsafe permissions on {path}: got {oct(mode)}, expected 0o600.\n"
133
- f" Fix: chmod 600 {path}"
134
- )
135
- key = path.read_text(encoding="utf-8").strip()
136
- if not key:
137
- raise KeyGateError(f"{path} is empty.")
138
- if not key.startswith(prefix):
139
- raise KeyGateError(
140
- f"{path} does not look like a {prefix!r}-prefixed key."
141
- )
142
- return key
143
-
144
-
145
- def load_anthropic_key(path: Path | None = None) -> str:
146
- resolved = path if path is not None else _resolve_key_path(ANTHROPIC_KEY_FILENAME)
147
- return _load_key(resolved, "sk-ant-", "src/scripts/install_anthropic_key.sh")
148
-
149
-
150
- def load_openai_key(path: Path | None = None) -> str:
151
- resolved = path if path is not None else _resolve_key_path(OPENAI_KEY_FILENAME)
152
- return _load_key(resolved, "sk-", "src/scripts/install_openai_key.sh")
153
-
154
-
155
- class ExternalAIClient(ABC):
156
- """Abstract base for council members."""
157
-
158
- name: str = ""
159
- model: str = ""
160
- billable: bool = True # API-mode subclasses spend money; manual doesn't.
161
- transport: str = "api" # "api" | "cli" | "manual" — surfaced in session manifest.
162
- subscription_label: str = "" # vendor-CLI label (e.g. "claude") for non-billable transports.
163
-
164
- @abstractmethod
165
- def ask(
166
- self,
167
- system_prompt: str,
168
- user_prompt: str,
169
- max_tokens: int = DEFAULT_MAX_TOKENS,
170
- ) -> CouncilResponse:
171
- """Send one independent query. Must never raise on network/API
172
- failure — return a `CouncilResponse` with `error` set instead.
173
- Other members should not be blocked by one failure."""
174
-
175
-
176
- class AnthropicClient(ExternalAIClient):
177
- name = "anthropic"
178
- billable = True
179
-
180
- def __init__(
181
- self,
182
- model: str = DEFAULT_ANTHROPIC_MODEL,
183
- client: object = None,
184
- api_key: str | None = None,
185
- ):
186
- self.model = model
187
- if client is not None:
188
- self._client = client
189
- return
190
- if api_key is None:
191
- raise RuntimeError(
192
- "AnthropicClient requires explicit api_key or injected client. "
193
- "Use load_anthropic_key() — no env-var fallback."
194
- )
195
- try:
196
- import anthropic # type: ignore[import-not-found]
197
- except ImportError as exc: # pragma: no cover - exercised only with real SDK
198
- raise RuntimeError(
199
- "anthropic package not installed. `pip install anthropic`."
200
- ) from exc
201
- self._client = anthropic.Anthropic(api_key=api_key)
202
-
203
- def ask(self, system_prompt: str, user_prompt: str, max_tokens: int = DEFAULT_MAX_TOKENS) -> CouncilResponse:
204
- t0 = time.monotonic()
205
- try:
206
- response = self._client.messages.create(
207
- model=self.model,
208
- max_tokens=max_tokens,
209
- system=system_prompt,
210
- messages=[{"role": "user", "content": user_prompt}],
211
- )
212
- except Exception as exc: # noqa: BLE001 - normalise all SDK errors
213
- return CouncilResponse(
214
- provider=self.name, model=self.model, text="",
215
- latency_ms=int((time.monotonic() - t0) * 1000),
216
- error=f"{type(exc).__name__}: {exc}",
217
- )
218
- latency_ms = int((time.monotonic() - t0) * 1000)
219
- text = ""
220
- content = getattr(response, "content", None)
221
- if content:
222
- text = getattr(content[0], "text", "") or ""
223
- usage = getattr(response, "usage", None)
224
- return CouncilResponse(
225
- provider=self.name, model=self.model, text=text,
226
- input_tokens=getattr(usage, "input_tokens", 0) if usage else 0,
227
- output_tokens=getattr(usage, "output_tokens", 0) if usage else 0,
228
- latency_ms=latency_ms,
229
- )
230
-
231
-
232
- class OpenAIClient(ExternalAIClient):
233
- name = "openai"
234
- billable = True
235
-
236
- def __init__(
237
- self,
238
- model: str = DEFAULT_OPENAI_MODEL,
239
- client: object = None,
240
- api_key: str | None = None,
241
- ):
242
- self.model = model
243
- if client is not None:
244
- self._client = client
245
- return
246
- if api_key is None:
247
- raise RuntimeError(
248
- "OpenAIClient requires explicit api_key or injected client. "
249
- "Use load_openai_key() — no env-var fallback."
250
- )
251
- try:
252
- import openai # type: ignore[import-not-found]
253
- except ImportError as exc: # pragma: no cover - exercised only with real SDK
254
- raise RuntimeError(
255
- "openai package not installed. `pip install openai`."
256
- ) from exc
257
- self._client = openai.OpenAI(api_key=api_key)
258
-
259
- def ask(self, system_prompt: str, user_prompt: str, max_tokens: int = DEFAULT_MAX_TOKENS) -> CouncilResponse:
260
- t0 = time.monotonic()
261
- kwargs: dict[str, object] = {"model": self.model}
262
- if _is_reasoning_model(self.model):
263
- # o1/o3/o4 reasoning models reject `max_tokens` and `system` role.
264
- kwargs["max_completion_tokens"] = max_tokens
265
- kwargs["messages"] = [
266
- {"role": "user", "content": f"{system_prompt}\n\n---\n\n{user_prompt}"},
267
- ]
268
- else:
269
- kwargs["max_tokens"] = max_tokens
270
- kwargs["messages"] = [
271
- {"role": "system", "content": system_prompt},
272
- {"role": "user", "content": user_prompt},
273
- ]
274
- try:
275
- response = self._client.chat.completions.create(**kwargs)
276
- except Exception as exc: # noqa: BLE001 - normalise all SDK errors
277
- return CouncilResponse(
278
- provider=self.name, model=self.model, text="",
279
- latency_ms=int((time.monotonic() - t0) * 1000),
280
- error=f"{type(exc).__name__}: {exc}",
281
- )
282
- latency_ms = int((time.monotonic() - t0) * 1000)
283
- text = ""
284
- choices = getattr(response, "choices", None)
285
- if choices:
286
- msg = getattr(choices[0], "message", None)
287
- text = getattr(msg, "content", "") if msg else ""
288
- usage = getattr(response, "usage", None)
289
- return CouncilResponse(
290
- provider=self.name, model=self.model, text=text or "",
291
- input_tokens=getattr(usage, "prompt_tokens", 0) if usage else 0,
292
- output_tokens=getattr(usage, "completion_tokens", 0) if usage else 0,
293
- latency_ms=latency_ms,
294
- )
295
-
296
-
297
- # ── Gemini / xAI / Perplexity (Phase 0 — Step 6) ─────────────────────
298
-
299
-
300
- class GeminiClient(ExternalAIClient):
301
- """Google Gemini via the ``google-genai`` SDK.
302
-
303
- Lazy-imports ``google.genai`` on first ``ask()`` so disabled
304
- members do not require the SDK to be installed. Tests inject a
305
- mock client shaped like ``genai.Client(api_key=...)`` —
306
- ``self._client.models.generate_content(...)`` returns an object
307
- with ``.text`` and ``.usage_metadata.{prompt_token_count,
308
- candidates_token_count}``.
309
- """
310
-
311
- name = "gemini"
312
- billable = True
313
-
314
- def __init__(
315
- self,
316
- model: str = DEFAULT_GEMINI_MODEL,
317
- client: object = None,
318
- api_key: str | None = None,
319
- ):
320
- self.model = model
321
- if client is not None:
322
- self._client = client
323
- return
324
- if api_key is None:
325
- raise RuntimeError(
326
- "GeminiClient requires explicit api_key or injected client. "
327
- "Use `api_key_ref: env:GEMINI_API_KEY` in ~/.event4u/agent-config/settings/.ai-council.yml."
328
- )
329
- try:
330
- from google import genai # type: ignore[import-not-found]
331
- except ImportError as exc: # pragma: no cover - exercised only with real SDK
332
- raise RuntimeError(
333
- "google-genai package not installed. `pip install google-genai`."
334
- ) from exc
335
- self._client = genai.Client(api_key=api_key)
336
-
337
- def ask(self, system_prompt: str, user_prompt: str, max_tokens: int = DEFAULT_MAX_TOKENS) -> CouncilResponse:
338
- t0 = time.monotonic()
339
- contents = f"{system_prompt}\n\n---\n\n{user_prompt}"
340
- try:
341
- response = self._client.models.generate_content(
342
- model=self.model,
343
- contents=contents,
344
- config={"max_output_tokens": max_tokens},
345
- )
346
- except Exception as exc: # noqa: BLE001 - normalise all SDK errors
347
- return CouncilResponse(
348
- provider=self.name, model=self.model, text="",
349
- latency_ms=int((time.monotonic() - t0) * 1000),
350
- error=f"{type(exc).__name__}: {exc}",
351
- )
352
- latency_ms = int((time.monotonic() - t0) * 1000)
353
- text = getattr(response, "text", "") or ""
354
- usage = getattr(response, "usage_metadata", None)
355
- return CouncilResponse(
356
- provider=self.name, model=self.model, text=text,
357
- input_tokens=getattr(usage, "prompt_token_count", 0) if usage else 0,
358
- output_tokens=getattr(usage, "candidates_token_count", 0) if usage else 0,
359
- latency_ms=latency_ms,
360
- )
361
-
362
-
363
- class _OpenAICompatibleClient(ExternalAIClient):
364
- """Shared shape for OpenAI-API-compatible providers (xAI, Perplexity).
365
-
366
- Both vendors implement ``/v1/chat/completions`` and accept the
367
- ``openai`` Python SDK with a custom ``base_url``. The reasoning-
368
- model branch from :class:`OpenAIClient` is intentionally omitted —
369
- neither xAI nor Perplexity ships a reasoning model that requires
370
- ``max_completion_tokens`` as of 2026-05-14.
371
- """
372
-
373
- billable = True
374
- base_url: str = ""
375
-
376
- def __init__(
377
- self,
378
- model: str,
379
- client: object = None,
380
- api_key: str | None = None,
381
- ):
382
- self.model = model
383
- if client is not None:
384
- self._client = client
385
- return
386
- if api_key is None:
387
- raise RuntimeError(
388
- f"{type(self).__name__} requires explicit api_key or injected client."
389
- )
390
- try:
391
- import openai # type: ignore[import-not-found]
392
- except ImportError as exc: # pragma: no cover - exercised only with real SDK
393
- raise RuntimeError(
394
- "openai package not installed. `pip install openai`."
395
- ) from exc
396
- self._client = openai.OpenAI(api_key=api_key, base_url=self.base_url)
397
-
398
- def ask(self, system_prompt: str, user_prompt: str, max_tokens: int = DEFAULT_MAX_TOKENS) -> CouncilResponse:
399
- t0 = time.monotonic()
400
- try:
401
- response = self._client.chat.completions.create(
402
- model=self.model,
403
- max_tokens=max_tokens,
404
- messages=[
405
- {"role": "system", "content": system_prompt},
406
- {"role": "user", "content": user_prompt},
407
- ],
408
- )
409
- except Exception as exc: # noqa: BLE001 - normalise all SDK errors
410
- return CouncilResponse(
411
- provider=self.name, model=self.model, text="",
412
- latency_ms=int((time.monotonic() - t0) * 1000),
413
- error=f"{type(exc).__name__}: {exc}",
414
- )
415
- latency_ms = int((time.monotonic() - t0) * 1000)
416
- text = ""
417
- choices = getattr(response, "choices", None)
418
- if choices:
419
- msg = getattr(choices[0], "message", None)
420
- text = getattr(msg, "content", "") if msg else ""
421
- usage = getattr(response, "usage", None)
422
- return CouncilResponse(
423
- provider=self.name, model=self.model, text=text or "",
424
- input_tokens=getattr(usage, "prompt_tokens", 0) if usage else 0,
425
- output_tokens=getattr(usage, "completion_tokens", 0) if usage else 0,
426
- latency_ms=latency_ms,
427
- )
428
-
429
-
430
- class XAIClient(_OpenAICompatibleClient):
431
- """xAI Grok via the OpenAI-compatible endpoint at api.x.ai/v1."""
432
-
433
- name = "xai"
434
- base_url = XAI_BASE_URL
435
-
436
- def __init__(
437
- self,
438
- model: str = DEFAULT_XAI_MODEL,
439
- client: object = None,
440
- api_key: str | None = None,
441
- ):
442
- super().__init__(model=model, client=client, api_key=api_key)
443
-
444
-
445
- class PerplexityClient(_OpenAICompatibleClient):
446
- """Perplexity via the OpenAI-compatible endpoint at api.perplexity.ai."""
447
-
448
- name = "perplexity"
449
- base_url = PERPLEXITY_BASE_URL
450
-
451
- def __init__(
452
- self,
453
- model: str = DEFAULT_PERPLEXITY_MODEL,
454
- client: object = None,
455
- api_key: str | None = None,
456
- ):
457
- super().__init__(model=model, client=client, api_key=api_key)
458
-
459
-
460
- # ── CLI transport (step-1 Phase 1+) ──────────────────────────────────
461
-
462
-
463
- CLI_CALLS_FILENAME = "cli-calls.json"
464
-
465
- #: Default subprocess timeout for a single CLI call. Long enough for the
466
- #: largest frontier models to think; short enough to surface a hung
467
- #: subprocess without freezing the council run.
468
- DEFAULT_CLI_TIMEOUT_SECONDS = 120.0
469
-
470
-
471
- class CliClientError(RuntimeError):
472
- """Raised when a CLI member cannot be constructed (binary missing, etc.)."""
473
-
474
-
475
- def _cli_calls_state_path() -> Path:
476
- """Return the canonical write target for the daily-quota counter."""
477
- return user_global_paths.write_target(CLI_CALLS_FILENAME)
478
-
479
-
480
- def _today_utc_iso() -> str:
481
- return datetime.now(timezone.utc).date().isoformat()
482
-
483
-
484
- def load_cli_call_counts(path: Path | None = None) -> dict[str, int]:
485
- """Return today's per-provider call counts. Empty dict on UTC rollover."""
486
- p = path if path is not None else _cli_calls_state_path()
487
- if not p.exists():
488
- return {}
489
- try:
490
- data = json.loads(p.read_text(encoding="utf-8"))
491
- except (json.JSONDecodeError, OSError):
492
- return {}
493
- if not isinstance(data, dict) or data.get("date") != _today_utc_iso():
494
- return {}
495
- counts = data.get("counts", {})
496
- if not isinstance(counts, dict):
497
- return {}
498
- return {str(k): int(v) for k, v in counts.items() if isinstance(v, (int, str))}
499
-
500
-
501
- def record_cli_call(provider: str, path: Path | None = None) -> int:
502
- """Increment today's call count for `provider`. Returns new total."""
503
- p = path if path is not None else _cli_calls_state_path()
504
- counts = load_cli_call_counts(p)
505
- counts[provider] = counts.get(provider, 0) + 1
506
- p.parent.mkdir(parents=True, exist_ok=True)
507
- p.write_text(
508
- json.dumps({"date": _today_utc_iso(), "counts": counts}, indent=2),
509
- encoding="utf-8",
510
- )
511
- return counts[provider]
512
-
513
-
514
- def reset_cli_call_counts(
515
- provider: str | None = None,
516
- path: Path | None = None,
517
- ) -> dict[str, int]:
518
- """Reset the per-provider call counter (step-8 P1, `council quota --reset`).
519
-
520
- ``provider=None`` clears all providers (today's record). Otherwise
521
- only the named provider's count is removed; other providers and
522
- the UTC date marker are preserved. Returns the post-reset counts.
523
- """
524
- p = path if path is not None else _cli_calls_state_path()
525
- counts = load_cli_call_counts(p)
526
- if provider is None:
527
- counts = {}
528
- else:
529
- counts.pop(provider, None)
530
- p.parent.mkdir(parents=True, exist_ok=True)
531
- p.write_text(
532
- json.dumps({"date": _today_utc_iso(), "counts": counts}, indent=2),
533
- encoding="utf-8",
534
- )
535
- return counts
536
-
537
-
538
- def quota_summary_line(
539
- clients: "list[CliClient]",
540
- *,
541
- cli_calls_path: Path | None = None,
542
- ) -> tuple[str, list[str]]:
543
- """Build the pre-run quota summary line (step-8 P1, D1 + D4).
544
-
545
- Returns ``(summary, warn_providers)`` where ``summary`` is the
546
- formatted one-liner (empty string when no CLI member has a
547
- configured cap) and ``warn_providers`` is the subset whose
548
- ``used / max_calls_per_day`` ratio crossed ``warn_at``. Uncapped
549
- providers (``max_calls_per_day is None``) are omitted from the
550
- summary entirely — they cannot exceed a threshold that does not
551
- exist.
552
-
553
- Tested in ``tests/test_cli_quota_warn.py``.
554
- """
555
- capped = [c for c in clients if getattr(c, "max_calls_per_day", None)]
556
- if not capped:
557
- return "", []
558
- # Read state once for the whole summary — call counts only mutate
559
- # inside ``CliClient.ask`` (sequential per-member dispatch), so the
560
- # pre-run snapshot is always consistent with what's about to fire.
561
- counts = load_cli_call_counts(cli_calls_path)
562
- parts: list[str] = []
563
- warn: list[str] = []
564
- for c in capped:
565
- name = getattr(c, "name", "?")
566
- used = int(counts.get(name, 0))
567
- limit = int(c.max_calls_per_day)
568
- parts.append(f"{name} {used}/{limit}")
569
- ratio = used / limit if limit > 0 else 0.0
570
- warn_at = float(getattr(c, "warn_at", 0.8))
571
- if ratio >= warn_at:
572
- warn.append(name)
573
- prefix = "⚠️ " if warn else ""
574
- return f"{prefix}council:quota · " + " · ".join(parts), warn
575
-
576
-
577
- class CliClient(ExternalAIClient):
578
- """Shell-out council member — subscription-authed transport.
579
-
580
- Spawns a locally-installed provider CLI via ``subprocess.run``. Auth
581
- is delegated to the binary itself (Claude CLI, Codex CLI, Gemini
582
- CLI, etc. use the user's logged-in subscription session). Spend is
583
- $0 from this loader's perspective — ``billable=False`` keeps the
584
- USD cost gate from firing.
585
-
586
- Provider subscription quotas (Claude Pro 5h windows, ChatGPT Plus
587
- message caps, Gemini free-tier limits) are guarded by the optional
588
- ``cli_call_budget.max_calls_per_day.<provider>`` config. Counter
589
- state lives at ``~/.event4u/agent-config/cli-calls.json`` and
590
- resets on UTC date rollover.
591
-
592
- Subclass contract:
593
-
594
- - ``name``: provider key (`anthropic`, `openai`, `gemini`, …).
595
- - ``default_binary``: executable name resolved via ``shutil.which``
596
- when the member-level ``binary:`` field is not set.
597
- - ``_build_command(system_prompt, user_prompt, max_tokens)``:
598
- return the argv list to execute.
599
- - ``_parse_output(stdout, stderr)``: return a partial
600
- ``CouncilResponse`` (``provider``, ``model``, ``text``,
601
- ``input_tokens``, ``output_tokens``, ``metadata``). The base
602
- ``ask()`` fills in ``latency_ms``.
603
-
604
- Construction validates the binary up front — a missing CLI fails
605
- fast with ``CliClientError`` so the loader can surface a structured
606
- "skip member with reason" entry rather than crashing the run.
607
-
608
- Stderr heuristics map known failure shapes to short error codes:
609
-
610
- - ``auth_expired`` — authentication / login / unauthorized.
611
- - ``timeout`` — subprocess timeout or deadline exceeded.
612
- - ``cli_quota_exhausted`` — rate-limit / quota messaging from the
613
- provider, OR the local counter has hit ``max_calls_per_day``.
614
- - ``parse_failed`` — non-zero exit absent + stdout was not parseable.
615
- - ``exit_<N>`` — fallback for any non-zero exit code without a known
616
- stderr pattern.
617
- """
618
-
619
- billable = False
620
- transport = "cli"
621
- default_binary: str = ""
622
-
623
- _AUTH_FAILURE_PATTERNS = (
624
- "authentication", "unauthorized", "auth failed", "auth_error",
625
- "login", "not logged in", "session expired", "invalid credentials",
626
- )
627
- _TIMEOUT_PATTERNS = ("timeout", "timed out", "deadline exceeded")
628
- _QUOTA_PATTERNS = (
629
- "rate limit", "rate_limit", "rate-limit", "quota exceeded",
630
- "too many requests", "429", "usage limit",
631
- )
632
-
633
- def __init__(
634
- self,
635
- *,
636
- model: str,
637
- binary: str | None = None,
638
- timeout_seconds: float = DEFAULT_CLI_TIMEOUT_SECONDS,
639
- max_calls_per_day: int | None = None,
640
- warn_at: float = 0.8,
641
- cli_calls_path: Path | None = None,
642
- ):
643
- self.model = model
644
- self.timeout_seconds = timeout_seconds
645
- self.max_calls_per_day = max_calls_per_day
646
- self.warn_at = warn_at
647
- self._cli_calls_path = cli_calls_path
648
- if binary is not None:
649
- self.binary = binary
650
- else:
651
- if not self.default_binary:
652
- raise CliClientError(
653
- f"{type(self).__name__}: no `default_binary` set on subclass; "
654
- f"either fix the class or pass `binary=` explicitly."
655
- )
656
- resolved = shutil.which(self.default_binary)
657
- if resolved is None:
658
- raise CliClientError(
659
- f"{type(self).__name__}: binary {self.default_binary!r} "
660
- f"not found on PATH. Install the provider CLI or set "
661
- f"`members.{self.name}.binary:` in ~/.event4u/agent-config/settings/.ai-council.yml."
662
- )
663
- self.binary = resolved
664
-
665
- # ── subclass hooks ────────────────────────────────────────────
666
-
667
- @abstractmethod
668
- def _build_command(
669
- self, system_prompt: str, user_prompt: str, max_tokens: int
670
- ) -> list[str]:
671
- """Return the argv list the subprocess should execute.
672
-
673
- ``self.binary`` is already resolved to an absolute path. Subclasses
674
- return ``[self.binary, ...flags...]`` and pass the prompt either
675
- via argv (small) or via stdin (large) — see ``_stdin_payload``.
676
- """
677
-
678
- @abstractmethod
679
- def _parse_output(
680
- self, stdout: str, stderr: str
681
- ) -> CouncilResponse:
682
- """Parse provider-specific stdout into a CouncilResponse.
683
-
684
- ``latency_ms`` and ``error`` are set by the base ``ask()`` wrapper;
685
- subclasses populate ``provider``, ``model``, ``text``,
686
- ``input_tokens``, ``output_tokens``, and any ``metadata``.
687
- """
688
-
689
- def _stdin_payload(self, system_prompt: str, user_prompt: str) -> str | None:
690
- """Return text to send on stdin, or ``None`` to inherit caller's stdin.
691
-
692
- Default: ``None`` — subclasses that prefer stdin-piped prompts
693
- override (typical for long prompts that would blow argv limits).
694
- """
695
- return None
696
-
697
- # ── ask() ──────────────────────────────────────────────────────
698
-
699
- def ask(
700
- self,
701
- system_prompt: str,
702
- user_prompt: str,
703
- max_tokens: int = DEFAULT_MAX_TOKENS,
704
- ) -> CouncilResponse:
705
- t0 = time.monotonic()
706
-
707
- # 1. quota gate — local counter check before spawning anything.
708
- if self.max_calls_per_day is not None:
709
- counts = load_cli_call_counts(self._cli_calls_path)
710
- used = counts.get(self.name, 0)
711
- if used >= self.max_calls_per_day:
712
- # step-8 D3 — record the block on the persistent events
713
- # log. Lazy import to keep clients.py independent of the
714
- # CLI layer at module load time.
715
- try:
716
- from scripts.ai_council.events_log import append_event
717
- append_event({
718
- "lens": "",
719
- "invocation": "",
720
- "action": "block_quota",
721
- "verdict": "",
722
- "provider_caps": {
723
- self.name: {
724
- "mode": "cli", "model": self.model,
725
- },
726
- },
727
- "original_ask": user_prompt,
728
- "cli_calls_used": used,
729
- "cli_calls_max": self.max_calls_per_day,
730
- })
731
- except Exception: # pragma: no cover — never crash ask()
732
- pass
733
- return CouncilResponse(
734
- provider=self.name, model=self.model, text="",
735
- latency_ms=int((time.monotonic() - t0) * 1000),
736
- error="cli_quota_exhausted",
737
- metadata={
738
- "cli": True,
739
- "cli_calls_used": used,
740
- "cli_calls_max": self.max_calls_per_day,
741
- },
742
- )
743
-
744
- # 2. build command + spawn.
745
- cmd = self._build_command(system_prompt, user_prompt, max_tokens)
746
- stdin_payload = self._stdin_payload(system_prompt, user_prompt)
747
- try:
748
- proc = subprocess.run(
749
- cmd,
750
- input=stdin_payload,
751
- capture_output=True,
752
- text=True,
753
- timeout=self.timeout_seconds,
754
- check=False,
755
- )
756
- except subprocess.TimeoutExpired:
757
- return CouncilResponse(
758
- provider=self.name, model=self.model, text="",
759
- latency_ms=int((time.monotonic() - t0) * 1000),
760
- error="timeout",
761
- metadata={"cli": True, "timeout_seconds": self.timeout_seconds},
762
- )
763
- except FileNotFoundError:
764
- return CouncilResponse(
765
- provider=self.name, model=self.model, text="",
766
- latency_ms=int((time.monotonic() - t0) * 1000),
767
- error="binary_missing",
768
- metadata={"cli": True, "binary": self.binary},
769
- )
770
- except OSError as exc:
771
- return CouncilResponse(
772
- provider=self.name, model=self.model, text="",
773
- latency_ms=int((time.monotonic() - t0) * 1000),
774
- error=f"os_error: {type(exc).__name__}",
775
- metadata={"cli": True},
776
- )
777
-
778
- # 3. record the call — even failures count against the quota so
779
- # a broken CLI cannot burn the whole budget in a tight loop.
780
- try:
781
- record_cli_call(self.name, self._cli_calls_path)
782
- except OSError: # state-file write failure is non-fatal here.
783
- pass
784
-
785
- latency_ms = int((time.monotonic() - t0) * 1000)
786
-
787
- # 4. non-zero exit → classify and bail.
788
- if proc.returncode != 0:
789
- code = self._classify_stderr(proc.stderr or "", proc.returncode)
790
- return CouncilResponse(
791
- provider=self.name, model=self.model, text="",
792
- latency_ms=latency_ms,
793
- error=code,
794
- metadata={
795
- "cli": True,
796
- "returncode": proc.returncode,
797
- "stderr_tail": (proc.stderr or "")[-500:],
798
- },
799
- )
800
-
801
- # 5. parse stdout via the subclass hook.
802
- try:
803
- response = self._parse_output(proc.stdout or "", proc.stderr or "")
804
- except Exception as exc: # noqa: BLE001 — defensive: parse must never crash the run.
805
- return CouncilResponse(
806
- provider=self.name, model=self.model,
807
- text=proc.stdout or "",
808
- latency_ms=latency_ms,
809
- error=f"parse_failed: {type(exc).__name__}",
810
- metadata={"cli": True, "stderr_tail": (proc.stderr or "")[-500:]},
811
- )
812
- response.latency_ms = latency_ms
813
- meta = dict(response.metadata)
814
- meta.setdefault("cli", True)
815
- response.metadata = meta
816
- return response
817
-
818
- @classmethod
819
- def _classify_stderr(cls, stderr: str, returncode: int) -> str:
820
- haystack = stderr.lower()
821
- if any(p in haystack for p in cls._AUTH_FAILURE_PATTERNS):
822
- return "auth_expired"
823
- if any(p in haystack for p in cls._TIMEOUT_PATTERNS):
824
- return "timeout"
825
- if any(p in haystack for p in cls._QUOTA_PATTERNS):
826
- return "cli_quota_exhausted"
827
- return f"exit_{returncode}"
828
-
829
-
830
- class AnthropicCliClient(CliClient):
831
- """Claude via the official `claude` CLI (subscription-authed).
832
-
833
- Invokes ``claude --print --output-format json`` and consumes the
834
- structured envelope: ``{"result": str, "usage": {"input_tokens":
835
- int, "output_tokens": int}, "session_id": str, ...}``. The prompt
836
- is piped on stdin so it never collides with argv length limits.
837
-
838
- Auth is delegated to the CLI's own session — the user runs
839
- ``claude /login`` once and the orchestrator inherits the
840
- subscription. No API key flows through this process.
841
- """
842
-
843
- name = "anthropic"
844
- default_binary = "claude"
845
- subscription_label = "claude-pro"
846
-
847
- def __init__(
848
- self,
849
- *,
850
- model: str = "claude-sonnet-4-5",
851
- binary: str | None = None,
852
- timeout_seconds: float = DEFAULT_CLI_TIMEOUT_SECONDS,
853
- max_calls_per_day: int | None = None,
854
- warn_at: float = 0.8,
855
- cli_calls_path: Path | None = None,
856
- ):
857
- super().__init__(
858
- model=model,
859
- binary=binary,
860
- timeout_seconds=timeout_seconds,
861
- max_calls_per_day=max_calls_per_day,
862
- warn_at=warn_at,
863
- cli_calls_path=cli_calls_path,
864
- )
865
-
866
- def _build_command(
867
- self, system_prompt: str, user_prompt: str, max_tokens: int # noqa: ARG002
868
- ) -> list[str]:
869
- return [
870
- self.binary,
871
- "--print",
872
- "--output-format", "json",
873
- "--model", self.model,
874
- "--append-system-prompt", system_prompt,
875
- ]
876
-
877
- def _stdin_payload(self, system_prompt: str, user_prompt: str) -> str | None: # noqa: ARG002
878
- return user_prompt
879
-
880
- def _parse_output(self, stdout: str, stderr: str) -> CouncilResponse: # noqa: ARG002
881
- envelope = json.loads(stdout)
882
- if not isinstance(envelope, dict):
883
- raise ValueError("expected JSON object at the top level of claude CLI output")
884
- text = str(envelope.get("result", "")).strip()
885
- usage = envelope.get("usage") or {}
886
- if not isinstance(usage, dict):
887
- usage = {}
888
- meta: dict[str, object] = {}
889
- session_id = envelope.get("session_id")
890
- if session_id:
891
- meta["session_id"] = str(session_id)
892
- total_cost = envelope.get("total_cost_usd")
893
- if total_cost is not None:
894
- meta["reported_cost_usd"] = total_cost
895
- duration_ms = envelope.get("duration_ms")
896
- if duration_ms is not None:
897
- meta["reported_duration_ms"] = duration_ms
898
- return CouncilResponse(
899
- provider=self.name, model=self.model, text=text,
900
- input_tokens=int(usage.get("input_tokens", 0) or 0),
901
- output_tokens=int(usage.get("output_tokens", 0) or 0),
902
- metadata=meta,
903
- )
904
-
905
-
906
- class OpenAICliClient(CliClient):
907
- """OpenAI via the official `codex` CLI (subscription-authed).
908
-
909
- Invokes ``codex exec --json <prompt>`` and consumes the
910
- newline-delimited JSON event stream. The user prompt rides on
911
- argv (Codex does not read prompts from stdin in ``exec`` mode);
912
- the system prompt is passed via ``--system`` when non-empty.
913
-
914
- Auth is delegated to the CLI's own session — the user runs
915
- ``codex login`` once and the orchestrator inherits the
916
- subscription. No API key flows through this process.
917
-
918
- Output shape: one JSON object per line. The terminal event has
919
- ``type == "item.completed"`` with the final assistant message in
920
- ``item.content[0].text``; a separate ``type == "turn.completed"``
921
- event carries token usage in ``usage.input_tokens`` /
922
- ``usage.output_tokens``. Robust against the order of events and
923
- against unknown event types (silently skipped).
924
- """
925
-
926
- name = "openai"
927
- default_binary = "codex"
928
- subscription_label = "chatgpt-plus"
929
-
930
- _AUTH_FAILURE_PATTERNS = CliClient._AUTH_FAILURE_PATTERNS + (
931
- "codex login", "auth_required", "401",
932
- )
933
-
934
- def __init__(
935
- self,
936
- *,
937
- model: str = "gpt-5",
938
- binary: str | None = None,
939
- timeout_seconds: float = DEFAULT_CLI_TIMEOUT_SECONDS,
940
- max_calls_per_day: int | None = None,
941
- warn_at: float = 0.8,
942
- cli_calls_path: Path | None = None,
943
- ):
944
- super().__init__(
945
- model=model,
946
- binary=binary,
947
- timeout_seconds=timeout_seconds,
948
- max_calls_per_day=max_calls_per_day,
949
- warn_at=warn_at,
950
- cli_calls_path=cli_calls_path,
951
- )
952
-
953
- def _build_command(
954
- self, system_prompt: str, user_prompt: str, max_tokens: int # noqa: ARG002
955
- ) -> list[str]:
956
- cmd = [self.binary, "exec", "--json", "--model", self.model]
957
- if system_prompt:
958
- cmd.extend(["--system", system_prompt])
959
- cmd.append(user_prompt)
960
- return cmd
961
-
962
- def _parse_output(self, stdout: str, stderr: str) -> CouncilResponse: # noqa: ARG002
963
- text = ""
964
- input_tokens = 0
965
- output_tokens = 0
966
- meta: dict[str, object] = {}
967
- for line in stdout.splitlines():
968
- line = line.strip()
969
- if not line:
970
- continue
971
- try:
972
- event = json.loads(line)
973
- except json.JSONDecodeError:
974
- continue
975
- if not isinstance(event, dict):
976
- continue
977
- event_type = event.get("type")
978
- if event_type == "item.completed":
979
- item = event.get("item") or {}
980
- if isinstance(item, dict):
981
- content = item.get("content") or []
982
- if isinstance(content, list):
983
- chunks: list[str] = []
984
- for entry in content:
985
- if isinstance(entry, dict) and entry.get("text"):
986
- chunks.append(str(entry["text"]))
987
- if chunks:
988
- text = "\n".join(chunks).strip()
989
- if item.get("id"):
990
- meta["item_id"] = str(item["id"])
991
- elif event_type == "turn.completed":
992
- usage = event.get("usage") or {}
993
- if isinstance(usage, dict):
994
- input_tokens = int(usage.get("input_tokens", 0) or 0)
995
- output_tokens = int(usage.get("output_tokens", 0) or 0)
996
- elif event_type == "session.created":
997
- if event.get("session_id"):
998
- meta["session_id"] = str(event["session_id"])
999
- return CouncilResponse(
1000
- provider=self.name, model=self.model, text=text,
1001
- input_tokens=input_tokens, output_tokens=output_tokens,
1002
- metadata=meta,
1003
- )
1004
-
1005
-
1006
- class GeminiCliClient(CliClient):
1007
- """Google Gemini via the official `gemini` CLI (free-tier subscription).
1008
-
1009
- Invokes ``gemini --prompt <prompt> --output-format json`` and
1010
- consumes the structured envelope: ``{"response": str, "stats":
1011
- {"models": {"<model>": {"tokens": {"prompt": int, "candidates":
1012
- int}}}}, ...}``. Prompt is piped on stdin to dodge argv limits.
1013
-
1014
- Auth is delegated to the CLI's own session — the user runs
1015
- ``gemini`` once interactively to set up OAuth, then the
1016
- orchestrator inherits the consent. Free-tier quotas apply at the
1017
- Google account level; ``cli_call_budget`` enforces a local mirror.
1018
- """
1019
-
1020
- name = "gemini"
1021
- default_binary = "gemini"
1022
- subscription_label = "gemini-pro"
1023
-
1024
- _AUTH_FAILURE_PATTERNS = CliClient._AUTH_FAILURE_PATTERNS + (
1025
- "interactive consent could not be obtained",
1026
- "please run `gemini`",
1027
- "oauth",
1028
- )
1029
-
1030
- def __init__(
1031
- self,
1032
- *,
1033
- model: str = "gemini-2.5-pro",
1034
- binary: str | None = None,
1035
- timeout_seconds: float = DEFAULT_CLI_TIMEOUT_SECONDS,
1036
- max_calls_per_day: int | None = None,
1037
- warn_at: float = 0.8,
1038
- cli_calls_path: Path | None = None,
1039
- ):
1040
- super().__init__(
1041
- model=model,
1042
- binary=binary,
1043
- timeout_seconds=timeout_seconds,
1044
- max_calls_per_day=max_calls_per_day,
1045
- warn_at=warn_at,
1046
- cli_calls_path=cli_calls_path,
1047
- )
1048
-
1049
- def _build_command(
1050
- self, system_prompt: str, user_prompt: str, max_tokens: int # noqa: ARG002
1051
- ) -> list[str]:
1052
- cmd = [
1053
- self.binary,
1054
- "--output-format", "json",
1055
- "--model", self.model,
1056
- ]
1057
- if system_prompt:
1058
- cmd.extend(["--system", system_prompt])
1059
- return cmd
1060
-
1061
- def _stdin_payload(self, system_prompt: str, user_prompt: str) -> str | None: # noqa: ARG002
1062
- return user_prompt
1063
-
1064
- def _parse_output(self, stdout: str, stderr: str) -> CouncilResponse: # noqa: ARG002
1065
- envelope = json.loads(stdout)
1066
- if not isinstance(envelope, dict):
1067
- raise ValueError("expected JSON object at the top level of gemini CLI output")
1068
- text = str(envelope.get("response", "")).strip()
1069
- input_tokens = 0
1070
- output_tokens = 0
1071
- stats = envelope.get("stats") or {}
1072
- if isinstance(stats, dict):
1073
- models = stats.get("models") or {}
1074
- if isinstance(models, dict):
1075
- # gemini emits per-model token counts; pick the configured model
1076
- # if present, else sum across all models in the envelope.
1077
- model_stats = models.get(self.model)
1078
- if not isinstance(model_stats, dict):
1079
- model_stats = next(
1080
- (v for v in models.values() if isinstance(v, dict)),
1081
- {},
1082
- )
1083
- tokens = (model_stats.get("tokens") or {}) if isinstance(model_stats, dict) else {}
1084
- if isinstance(tokens, dict):
1085
- input_tokens = int(tokens.get("prompt", 0) or 0)
1086
- output_tokens = int(tokens.get("candidates", 0) or 0)
1087
- meta: dict[str, object] = {}
1088
- session_id = envelope.get("sessionId") or envelope.get("session_id")
1089
- if session_id:
1090
- meta["session_id"] = str(session_id)
1091
- return CouncilResponse(
1092
- provider=self.name, model=self.model, text=text,
1093
- input_tokens=input_tokens, output_tokens=output_tokens,
1094
- metadata=meta,
1095
- )
1096
-
1097
-
1098
- class XAICliClient(CliClient):
1099
- """xAI Grok via the community `grok` CLI (Superagent project).
1100
-
1101
- Community-maintained wrapper around the xAI API — **not** an
1102
- official subscription transport. The CLI consumes ``XAI_API_KEY``
1103
- from its own environment, so every call is paid per-token exactly
1104
- as ``XAIClient`` (api transport) would be. ``mode: cli`` here is
1105
- an ergonomic shortcut for users who already drive Grok from the
1106
- shell; it does NOT bypass the USD cost gate.
1107
-
1108
- Invokes ``grok -p <prompt>``. Output is plain text — no JSON
1109
- envelope. ``_parse_output`` returns the trimmed stdout and
1110
- estimates token counts heuristically (chars / 4) for the
1111
- audit-trail; estimates feed the post-call spend tracker, not the
1112
- pre-call gate (the orchestrator's ``estimate()`` already projects
1113
- cost from the prompt before this client is invoked).
1114
- """
1115
-
1116
- name = "xai"
1117
- default_binary = "grok"
1118
- billable = True # community CLI consumes an API key — billable applies
1119
-
1120
- _AUTH_FAILURE_PATTERNS = CliClient._AUTH_FAILURE_PATTERNS + (
1121
- "xai_api_key", "401", "unauthorized",
1122
- )
1123
-
1124
- def __init__(
1125
- self,
1126
- *,
1127
- model: str = DEFAULT_XAI_MODEL,
1128
- binary: str | None = None,
1129
- timeout_seconds: float = DEFAULT_CLI_TIMEOUT_SECONDS,
1130
- max_calls_per_day: int | None = None,
1131
- warn_at: float = 0.8,
1132
- cli_calls_path: Path | None = None,
1133
- ):
1134
- super().__init__(
1135
- model=model,
1136
- binary=binary,
1137
- timeout_seconds=timeout_seconds,
1138
- max_calls_per_day=max_calls_per_day,
1139
- warn_at=warn_at,
1140
- cli_calls_path=cli_calls_path,
1141
- )
1142
-
1143
- def _build_command(
1144
- self, system_prompt: str, user_prompt: str, max_tokens: int # noqa: ARG002
1145
- ) -> list[str]:
1146
- cmd = [self.binary, "-p", user_prompt]
1147
- if self.model:
1148
- cmd.extend(["--model", self.model])
1149
- return cmd
1150
-
1151
- def _parse_output(self, stdout: str, stderr: str) -> CouncilResponse: # noqa: ARG002
1152
- text = stdout.strip()
1153
- # Plain-text CLIs surface no token usage — estimate from text
1154
- # length so the audit trail and post-call tracker stay populated.
1155
- # chars / 4 mirrors `pricing.estimate_input_tokens`.
1156
- output_tokens = max(1, len(text) // 4) if text else 0
1157
- return CouncilResponse(
1158
- provider=self.name, model=self.model, text=text,
1159
- input_tokens=0, output_tokens=output_tokens,
1160
- metadata={"cli_output_format": "plain_text", "tokens_estimated": True},
1161
- )
1162
-
1163
-
1164
- class PerplexityCliClient(CliClient):
1165
- """Perplexity via the community `perplexity` CLI (npm package).
1166
-
1167
- Community-maintained wrapper around the Perplexity API — **not**
1168
- an official subscription transport. The CLI consumes
1169
- ``PERPLEXITY_API_KEY`` from its own environment, so every call is
1170
- paid per-token exactly as ``PerplexityClient`` (api transport)
1171
- would be. ``mode: cli`` here is an ergonomic shortcut; it does
1172
- NOT bypass the USD cost gate.
1173
-
1174
- Invokes ``perplexity -p <prompt>``. Output is plain text — no
1175
- JSON envelope. Token counts are estimated heuristically for the
1176
- audit trail; the pre-call cost gate uses the orchestrator's
1177
- prompt-side estimate.
1178
- """
1179
-
1180
- name = "perplexity"
1181
- default_binary = "perplexity"
1182
- billable = True # community CLI consumes an API key — billable applies
1183
-
1184
- _AUTH_FAILURE_PATTERNS = CliClient._AUTH_FAILURE_PATTERNS + (
1185
- "perplexity_api_key", "401", "unauthorized",
1186
- )
1187
-
1188
- def __init__(
1189
- self,
1190
- *,
1191
- model: str = DEFAULT_PERPLEXITY_MODEL,
1192
- binary: str | None = None,
1193
- timeout_seconds: float = DEFAULT_CLI_TIMEOUT_SECONDS,
1194
- max_calls_per_day: int | None = None,
1195
- warn_at: float = 0.8,
1196
- cli_calls_path: Path | None = None,
1197
- ):
1198
- super().__init__(
1199
- model=model,
1200
- binary=binary,
1201
- timeout_seconds=timeout_seconds,
1202
- max_calls_per_day=max_calls_per_day,
1203
- warn_at=warn_at,
1204
- cli_calls_path=cli_calls_path,
1205
- )
1206
-
1207
- def _build_command(
1208
- self, system_prompt: str, user_prompt: str, max_tokens: int # noqa: ARG002
1209
- ) -> list[str]:
1210
- cmd = [self.binary, "-p", user_prompt]
1211
- if self.model:
1212
- cmd.extend(["--model", self.model])
1213
- return cmd
1214
-
1215
- def _parse_output(self, stdout: str, stderr: str) -> CouncilResponse: # noqa: ARG002
1216
- text = stdout.strip()
1217
- output_tokens = max(1, len(text) // 4) if text else 0
1218
- return CouncilResponse(
1219
- provider=self.name, model=self.model, text=text,
1220
- input_tokens=0, output_tokens=output_tokens,
1221
- metadata={"cli_output_format": "plain_text", "tokens_estimated": True},
1222
- )
1223
-
1224
-
1225
- # ── Manual mode (Phase 2b) ───────────────────────────────────────────
1226
-
1227
-
1228
- MANUAL_END_MARKER = "END" # line containing only this terminates a paste block.
1229
-
1230
-
1231
- def _read_until_marker(stream: TextIO, marker: str) -> str:
1232
- """Read lines from `stream` until a line equal to `marker` (after strip).
1233
-
1234
- Returns the joined body without the marker line. EOF before the
1235
- marker is treated as end-of-input — the body collected so far is
1236
- returned; callers decide whether that counts as abort.
1237
- """
1238
- body: list[str] = []
1239
- for raw in stream:
1240
- line = raw.rstrip("\n")
1241
- if line.strip() == marker:
1242
- break
1243
- body.append(line)
1244
- return "\n".join(body).strip()
1245
-
1246
-
1247
- class ManualClient(ExternalAIClient):
1248
- """Copy-paste council member — user is the transport.
1249
-
1250
- `ask()` renders the system prompt + artefact as one Markdown block,
1251
- prints it to `stdout`, and reads pasted replies from `stdin`. After
1252
- each pasted reply, surfaces a 1/2/3 menu (more · next · abort) per
1253
- `user-interaction`. Loops until the user picks 2 or 3.
1254
-
1255
- Spend is $0 — `billable=False` makes the orchestrator skip the cost
1256
- gate for this member regardless of the price table.
1257
-
1258
- Tests inject `stdin` / `stdout` `TextIO` streams. Production usage
1259
- falls back to `sys.stdin` / `sys.stdout`.
1260
- """
1261
-
1262
- billable = False
1263
- transport = "manual"
1264
-
1265
- def __init__(
1266
- self,
1267
- *,
1268
- name: str = "manual",
1269
- model: str = "manual",
1270
- provider_label: str = "your LLM web UI",
1271
- stdin: TextIO | None = None,
1272
- stdout: TextIO | None = None,
1273
- end_marker: str = MANUAL_END_MARKER,
1274
- ):
1275
- self.name = name
1276
- self.model = model
1277
- self.provider_label = provider_label
1278
- self._stdin = stdin if stdin is not None else sys.stdin
1279
- self._stdout = stdout if stdout is not None else sys.stdout
1280
- self._end_marker = end_marker
1281
-
1282
- def ask(
1283
- self,
1284
- system_prompt: str,
1285
- user_prompt: str,
1286
- max_tokens: int = DEFAULT_MAX_TOKENS, # noqa: ARG002 — accepted for ABC parity
1287
- ) -> CouncilResponse:
1288
- t0 = time.monotonic()
1289
- rounds: list[str] = []
1290
- block = self._render_block(system_prompt, user_prompt, follow_up=None)
1291
- self._emit(block)
1292
-
1293
- try:
1294
- while True:
1295
- reply = _read_until_marker(self._stdin, self._end_marker)
1296
- rounds.append(reply)
1297
- choice = self._ask_menu(reply_chars=len(reply))
1298
-
1299
- if choice == "2": # done with this member
1300
- break
1301
- if choice == "3": # abort the council run
1302
- return CouncilResponse(
1303
- provider=self.name, model=self.model, text="",
1304
- latency_ms=int((time.monotonic() - t0) * 1000),
1305
- error="manual_aborted",
1306
- metadata={"rounds": len(rounds), "manual": True},
1307
- )
1308
- # choice == "1": collect follow-up, re-emit context block.
1309
- follow_up = self._read_follow_up()
1310
- if not follow_up:
1311
- break # empty follow-up → treat as "done with this member"
1312
- rounds.append(f"[follow-up sent]\n{follow_up}")
1313
- block = self._render_block(system_prompt, user_prompt, follow_up=follow_up)
1314
- self._emit(block)
1315
- except Exception as exc: # noqa: BLE001 — never break the council on a stdin glitch
1316
- return CouncilResponse(
1317
- provider=self.name, model=self.model, text="\n\n".join(rounds),
1318
- latency_ms=int((time.monotonic() - t0) * 1000),
1319
- error=f"{type(exc).__name__}: {exc}",
1320
- metadata={"rounds": len(rounds), "manual": True},
1321
- )
1322
-
1323
- text = "\n\n---\n\n".join(rounds).strip()
1324
- return CouncilResponse(
1325
- provider=self.name, model=self.model, text=text,
1326
- latency_ms=int((time.monotonic() - t0) * 1000),
1327
- metadata={"rounds": len(rounds), "manual": True},
1328
- )
1329
-
1330
- # ── helpers ──────────────────────────────────────────────────────
1331
-
1332
- def _emit(self, text: str) -> None:
1333
- self._stdout.write(text)
1334
- self._stdout.write("\n")
1335
- self._stdout.flush()
1336
-
1337
- def _render_block(
1338
- self,
1339
- system_prompt: str,
1340
- user_prompt: str,
1341
- *,
1342
- follow_up: str | None,
1343
- ) -> str:
1344
- bar = "═" * 67
1345
- head = (
1346
- f"{bar}\n"
1347
- f"Manual council member: {self.provider_label}\n"
1348
- f"Paste this block into the web UI · then paste the reply below.\n"
1349
- f"{bar}"
1350
- )
1351
- if follow_up is not None:
1352
- body = (
1353
- f"[Follow-up — paste this into the SAME chat thread]\n\n"
1354
- f"{follow_up}"
1355
- )
1356
- else:
1357
- body = f"{system_prompt}\n\n---\n\n{user_prompt}"
1358
- tail = (
1359
- f"{bar}\n"
1360
- f"End your pasted reply with a line containing only: {self._end_marker}\n"
1361
- f"{bar}"
1362
- )
1363
- return f"{head}\n\n{body}\n\n{tail}"
1364
-
1365
- def _ask_menu(self, *, reply_chars: int) -> str:
1366
- prompt = (
1367
- f"\nReply received ({reply_chars} chars). Now what?\n"
1368
- f" 1. More feedback for this member (continue this thread)\n"
1369
- f" 2. Done with this member, move to the next\n"
1370
- f" 3. Abort the council run\n\n"
1371
- f"Choose 1/2/3: "
1372
- )
1373
- self._stdout.write(prompt)
1374
- self._stdout.flush()
1375
- line = self._stdin.readline().strip()
1376
- if line in {"1", "2", "3"}:
1377
- return line
1378
- # unknown input → treat as "next" so we never block forever in tests / piped runs.
1379
- return "2"
1380
-
1381
- def _read_follow_up(self) -> str:
1382
- self._emit(
1383
- f"\nType your follow-up question, end with a line containing only: {self._end_marker}"
1384
- )
1385
- return _read_until_marker(self._stdin, self._end_marker)