codymaster 4.8.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 (416) hide show
  1. package/CHANGELOG.md +331 -7
  2. package/README.md +226 -296
  3. package/dist/advisory-handoff.js +89 -0
  4. package/dist/advisory-report.js +105 -0
  5. package/dist/agent/antigravity.js +152 -0
  6. package/dist/agent/backend.js +2 -0
  7. package/dist/agent/claude.js +196 -0
  8. package/dist/agent/codex.js +204 -0
  9. package/dist/agent/copilot.js +284 -0
  10. package/dist/agent/cursor.js +211 -0
  11. package/dist/agent/factory.js +30 -0
  12. package/dist/agent/gemini.js +142 -0
  13. package/dist/agent/opencode.js +205 -0
  14. package/dist/agent/spawn-helper.js +237 -0
  15. package/dist/agent/version.js +25 -0
  16. package/dist/browse/adapter-factory.js +69 -0
  17. package/dist/browse/adapters/agent-browser-adapter.js +305 -0
  18. package/dist/browse/adapters/playwright-adapter.js +309 -0
  19. package/dist/browse/adapters/types.js +6 -0
  20. package/dist/browse/error-collector.js +132 -0
  21. package/dist/browse/event-log.js +109 -0
  22. package/dist/browse/index.js +17 -0
  23. package/dist/browse-server.js +204 -120
  24. package/dist/cli/command-registry.js +20 -0
  25. package/dist/cli/commands/bench.js +69 -0
  26. package/dist/cli/commands/brain.js +108 -0
  27. package/dist/cli/commands/dashboard.js +76 -2
  28. package/dist/cli/commands/engineering.js +326 -4
  29. package/dist/cli/commands/evolve.js +123 -0
  30. package/dist/cli/commands/install.js +160 -0
  31. package/dist/cli/commands/learn.js +181 -0
  32. package/dist/cli/commands/mcp-serve.js +104 -0
  33. package/dist/cli/commands/parallel.js +138 -0
  34. package/dist/cli/commands/quality.js +105 -0
  35. package/dist/cli/commands/stack.js +49 -0
  36. package/dist/cli/commands/update.js +159 -0
  37. package/dist/cli/update-check.js +94 -10
  38. package/dist/cm-config.js +0 -18
  39. package/dist/codybench/judges/automated.js +31 -0
  40. package/dist/codybench/runners/claude-code.js +32 -0
  41. package/dist/codybench/suites/memory-retention.js +85 -0
  42. package/dist/codybench/suites/tdd-regression.js +35 -0
  43. package/dist/codybench/suites/token-efficiency.js +55 -0
  44. package/dist/codybench/types.js +2 -0
  45. package/dist/context-db.js +157 -0
  46. package/dist/continuity.js +5 -7
  47. package/dist/dashboard.js +47 -6
  48. package/dist/data.js +35 -0
  49. package/dist/execution/tdd-gate.js +113 -0
  50. package/dist/execution-analyzer.js +138 -0
  51. package/dist/executor/cancel.js +34 -0
  52. package/dist/executor/gc.js +74 -0
  53. package/dist/executor/index.js +14 -0
  54. package/dist/executor/runner.js +70 -0
  55. package/dist/executor/workdir.js +31 -0
  56. package/dist/handoff/contracts.js +22 -0
  57. package/dist/handoff/index.js +18 -0
  58. package/dist/handoff/io.js +121 -0
  59. package/dist/index.js +7 -3
  60. package/dist/indexer/skills-lib.js +533 -0
  61. package/dist/indexer/skills-map.js +1374 -0
  62. package/dist/indexer/skills.js +16 -0
  63. package/dist/indexer/stack-detect.js +219 -0
  64. package/dist/install/copy.js +98 -0
  65. package/dist/install/engine.js +42 -0
  66. package/dist/install/paths.js +70 -0
  67. package/dist/install/platforms/_simple.js +85 -0
  68. package/dist/install/platforms/antigravity.js +91 -0
  69. package/dist/install/platforms/claude-code.js +107 -0
  70. package/dist/install/platforms/cursor.js +77 -0
  71. package/dist/install/platforms/index.js +27 -0
  72. package/dist/install/platforms/simple.js +163 -0
  73. package/dist/install/profiles.js +75 -0
  74. package/dist/install/types.js +2 -0
  75. package/dist/learning-promoter.js +246 -0
  76. package/dist/learnings.js +208 -0
  77. package/dist/mcp-context-server.js +230 -1
  78. package/dist/middleware/metrics.js +30 -0
  79. package/dist/middleware/security-headers.js +14 -0
  80. package/dist/realtime/event-bus.js +29 -0
  81. package/dist/realtime/ws-hub.js +91 -0
  82. package/dist/schemas/task-schema.js +48 -0
  83. package/dist/schemas/validate.js +18 -0
  84. package/dist/skill-chain.js +63 -1
  85. package/dist/skill-evolver.js +456 -0
  86. package/dist/skill-execution-cache.js +254 -0
  87. package/dist/skills-lock.js +96 -0
  88. package/dist/smart-brain-router.js +184 -0
  89. package/dist/sprint-pipeline.js +26 -0
  90. package/dist/storage/index.js +21 -0
  91. package/dist/storage/repos/activity-repo.js +46 -0
  92. package/dist/storage/repos/message-repo.js +39 -0
  93. package/dist/storage/repos/project-repo.js +56 -0
  94. package/dist/storage/repos/task-repo.js +142 -0
  95. package/dist/storage/services/project-service.js +49 -0
  96. package/dist/storage/services/task-service.js +97 -0
  97. package/dist/storage/sqlite.js +113 -0
  98. package/dist/storage-backend.js +10 -8
  99. package/dist/tier-classify.js +131 -0
  100. package/dist/token-budget.js +88 -0
  101. package/dist/ui/onboarding.js +51 -15
  102. package/dist/utils/cli-utils.js +7 -2
  103. package/dist/utils/design-taste.js +108 -0
  104. package/dist/utils/output-compress.js +143 -0
  105. package/dist/vibecoding-index.js +126 -0
  106. package/package.json +20 -6
  107. package/public/dashboard/app.js +52 -1
  108. package/scripts/build-skills-lock.mjs +88 -0
  109. package/scripts/build-skills.mjs +187 -28
  110. package/scripts/compress-skill.mjs +73 -0
  111. package/scripts/deprecate-skill.mjs +72 -0
  112. package/scripts/install.sh +170 -0
  113. package/scripts/mcp-bridge.js +2 -2
  114. package/scripts/postinstall.js +53 -335
  115. package/scripts/update-changelog.sh +88 -0
  116. package/scripts/validate-skills.mjs +101 -4
  117. package/skills/CLAUDE.md +0 -5
  118. package/skills/_shared/SKILL_TEMPLATE.md +62 -0
  119. package/skills/_shared/helpers.md +2 -8
  120. package/skills/cm-autopilot/scripts/autopilot.py +19 -2
  121. package/skills/cm-brainstorm-idea/SKILL.md +9 -0
  122. package/skills/cm-browse/SKILL.md +6 -0
  123. package/skills/cm-clean-code/SKILL.md +20 -0
  124. package/skills/cm-code-review/SKILL.md +21 -0
  125. package/skills/cm-codeintell/SKILL.md +9 -0
  126. package/skills/cm-conductor-worktrees/SKILL.archive.md +28 -0
  127. package/skills/cm-conductor-worktrees/SKILL.md +20 -18
  128. package/skills/cm-continuity/SKILL.md +41 -33
  129. package/skills/cm-dashboard/SKILL.archive.md +15 -0
  130. package/skills/cm-dashboard/SKILL.md +20 -9
  131. package/skills/cm-dashboard/ui/app.js +9 -1
  132. package/skills/cm-debugging/SKILL.md +9 -0
  133. package/skills/cm-design-studio/SKILL.archive.md +34 -0
  134. package/skills/cm-design-studio/SKILL.md +20 -24
  135. package/skills/cm-design-system/SKILL.md +1 -0
  136. package/skills/cm-ecosystem-roadmap/SKILL.md +4 -0
  137. package/skills/cm-engineering-meta/SKILL.archive.md +73 -0
  138. package/skills/cm-engineering-meta/SKILL.md +19 -62
  139. package/skills/cm-execution/SKILL.md +98 -0
  140. package/skills/cm-git-worktrees/SKILL.archive.md +157 -0
  141. package/skills/cm-git-worktrees/SKILL.md +15 -146
  142. package/skills/cm-guardian-runtime/SKILL.md +5 -1
  143. package/skills/cm-identity-guard/SKILL.md +8 -0
  144. package/skills/cm-mcp-engineering/SKILL.md +4 -0
  145. package/skills/cm-planning/SKILL.md +63 -92
  146. package/skills/cm-post-deploy-canary/SKILL.archive.md +22 -0
  147. package/skills/cm-post-deploy-canary/SKILL.md +20 -12
  148. package/skills/cm-project-bootstrap/SKILL.md +11 -0
  149. package/skills/cm-qa-visual-cli/SKILL.archive.md +22 -0
  150. package/skills/cm-qa-visual-cli/SKILL.md +19 -11
  151. package/skills/cm-quality-gate/SKILL.md +38 -0
  152. package/skills/cm-retro-cli/SKILL.md +4 -0
  153. package/skills/cm-safe-deploy/SKILL.md +9 -0
  154. package/skills/cm-second-opinion-cli/SKILL.archive.md +23 -0
  155. package/skills/cm-second-opinion-cli/SKILL.md +20 -13
  156. package/skills/cm-secret-shield/SKILL.archive.md +580 -0
  157. package/skills/cm-secret-shield/SKILL.md +15 -569
  158. package/skills/cm-security-gate/SKILL.archive.md +239 -0
  159. package/skills/cm-security-gate/SKILL.md +16 -228
  160. package/skills/cm-skill-chain/SKILL.md +25 -4
  161. package/skills/cm-skill-evolution/SKILL.md +83 -0
  162. package/skills/cm-skill-health/SKILL.archive.md +83 -0
  163. package/skills/cm-skill-health/SKILL.md +26 -0
  164. package/skills/cm-skill-index/SKILL.md +19 -3
  165. package/skills/cm-skill-mastery/SKILL.archive.md +156 -0
  166. package/skills/cm-skill-mastery/SKILL.md +16 -146
  167. package/skills/cm-skill-search/SKILL.archive.md +49 -0
  168. package/skills/cm-skill-search/SKILL.md +26 -0
  169. package/skills/cm-skill-share/SKILL.archive.md +58 -0
  170. package/skills/cm-skill-share/SKILL.md +26 -0
  171. package/skills/cm-sprint-bus/SKILL.md +13 -0
  172. package/skills/cm-start/SKILL.md +17 -10
  173. package/skills/cm-tdd/SKILL.md +21 -2
  174. package/skills/cm-terminal/SKILL.md +15 -0
  175. package/skills/cm-test-gate/SKILL.archive.md +245 -0
  176. package/skills/cm-test-gate/SKILL.md +15 -234
  177. package/skills/cm-ui-preview/SKILL.archive.md +153 -0
  178. package/skills/cm-ui-preview/SKILL.md +16 -143
  179. package/skills/cm-ux-master/cli/uxmaster/commands/mcp.py +1 -1
  180. package/skills/cm-ux-master/mcp/mcp-config.json +1 -1
  181. package/skills/cm-ux-master/mcp/server.py +2 -2
  182. package/skills/profiles/design.txt +1 -1
  183. package/skills/profiles/full.txt +4 -10
  184. package/skills/profiles/growth.txt +8 -8
  185. package/skills/profiles/knowledge.txt +1 -1
  186. package/skills/profiles/top35.json +41 -0
  187. package/adapters/antigravity.js +0 -15
  188. package/adapters/claude-code.js +0 -17
  189. package/adapters/cursor.js +0 -16
  190. package/install.sh +0 -1125
  191. package/scripts/viking-demo.ts +0 -105
  192. package/skills/cm-ads-tracker/SKILL.md +0 -401
  193. package/skills/cm-ads-tracker/evals/evals.json +0 -55
  194. package/skills/cm-ads-tracker/references/gtm-architecture.md +0 -321
  195. package/skills/cm-ads-tracker/references/industry-events.md +0 -294
  196. package/skills/cm-ads-tracker/references/platforms-api.md +0 -238
  197. package/skills/cm-ads-tracker/templates/capi-payload.md +0 -79
  198. package/skills/cm-ads-tracker/templates/datalayer-push.js +0 -104
  199. package/skills/cm-ads-tracker/templates/gtm-variables.js +0 -56
  200. package/skills/cm-auto-publisher/SKILL.md +0 -81
  201. package/skills/cm-booking-calendar/SKILL.md +0 -521
  202. package/skills/cm-booking-calendar/references/industry-patterns.md +0 -527
  203. package/skills/cm-booking-calendar/templates/booking-form.css +0 -626
  204. package/skills/cm-booking-calendar/templates/booking-form.html +0 -477
  205. package/skills/cm-booking-calendar/templates/calendar-engine.js +0 -419
  206. package/skills/cm-booking-calendar/templates/calendar-export.js +0 -395
  207. package/skills/cm-booking-calendar/templates/reminder-config.js +0 -629
  208. package/skills/cm-content-factory/.content-factory-state.json +0 -132
  209. package/skills/cm-content-factory/.git 2/logs/refs/heads/main +0 -1
  210. package/skills/cm-content-factory/.git 2/logs/refs/remotes/origin/main +0 -1
  211. package/skills/cm-content-factory/.git 2/objects/02/fb0956734b5f8ba3f918b7defd04a89cfe0076 +0 -0
  212. package/skills/cm-content-factory/.git 2/objects/08/1e129d75dc6feac6c02037272e6bd1a04e3324 +0 -0
  213. package/skills/cm-content-factory/.git 2/objects/0c/5393416f3c5e01c9a655a802bff0dd52f76f0a +0 -0
  214. package/skills/cm-content-factory/.git 2/objects/10/0b9be46978a946a77188f68be725098a122001 +0 -0
  215. package/skills/cm-content-factory/.git 2/objects/10/cf041167fc9843610eb3d90259ef3396315fdc +0 -0
  216. package/skills/cm-content-factory/.git 2/objects/12/5e19538dd6e1338ffe74f6c4c165b00435bf48 +0 -0
  217. package/skills/cm-content-factory/.git 2/objects/16/a9b9d0088d5c1347628b45a2620b479d8ad57c +0 -0
  218. package/skills/cm-content-factory/.git 2/objects/17/8c2a9ef93c33ae4eec9d58e82321f9229843a1 +0 -0
  219. package/skills/cm-content-factory/.git 2/objects/25/397ae41d09104d763bdcac2695209d85cdea89 +0 -0
  220. package/skills/cm-content-factory/.git 2/objects/2f/a836b7947f2d458e1f639788bf4bb0983a3305 +0 -0
  221. package/skills/cm-content-factory/.git 2/objects/3a/baaaf0a1c0909c0828335791557125fba911e0 +0 -0
  222. package/skills/cm-content-factory/.git 2/objects/42/2924221b81f5ce3c4e4daac9a64a24f9b01f9a +0 -0
  223. package/skills/cm-content-factory/.git 2/objects/42/ec0ce707447dc11446a34c9995fb8533801731 +0 -0
  224. package/skills/cm-content-factory/.git 2/objects/46/e43ce92866d56ce74b1d750db307cfe6154a15 +0 -0
  225. package/skills/cm-content-factory/.git 2/objects/48/5e41b633c63f55b8277bcc59f44f67681f671a +0 -0
  226. package/skills/cm-content-factory/.git 2/objects/49/49c596a3a89fa240642acd95dd3258e261eb09 +0 -0
  227. package/skills/cm-content-factory/.git 2/objects/50/9d42d8412ef8eaf7f7e138476bac2e4d10ce60 +0 -0
  228. package/skills/cm-content-factory/.git 2/objects/55/0c8c389d981b463ef849aeb792d8be3ccb6ec8 +0 -0
  229. package/skills/cm-content-factory/.git 2/objects/5d/82d3b18410cdda3ace3677436f0cb599dbe2d2 +0 -0
  230. package/skills/cm-content-factory/.git 2/objects/60/0617c58e871a38b33bf29e282d132bb3c381ad +0 -0
  231. package/skills/cm-content-factory/.git 2/objects/6a/8369a99c687b7245c92ffaf0e0f0dab9014504 +0 -0
  232. package/skills/cm-content-factory/.git 2/objects/79/bea435d40ab531c1aaf6be0432c6a5b7aaed21 +0 -0
  233. package/skills/cm-content-factory/.git 2/objects/7e/5ebd79251c2f14e4aceb86c74b6b6daae6b500 +0 -0
  234. package/skills/cm-content-factory/.git 2/objects/81/98a822a60178d6d5023ddb3e222cddf048742e +0 -0
  235. package/skills/cm-content-factory/.git 2/objects/86/0a0e1943dfe53411d2e499a1f16f46a96ef758 +0 -0
  236. package/skills/cm-content-factory/.git 2/objects/86/971fb55fdc081fdbae52376f0f13e57a4e9b04 +0 -0
  237. package/skills/cm-content-factory/.git 2/objects/88/b89dd609a0a03f8d4fe8bfde20d5b8fc1d326d +0 -0
  238. package/skills/cm-content-factory/.git 2/objects/90/8737edb6b7809e32cc01590b4e08ba42a9d40d +0 -0
  239. package/skills/cm-content-factory/.git 2/objects/93/d5a8a9a7d4fb7f11491cb596a6880528725118 +0 -0
  240. package/skills/cm-content-factory/.git 2/objects/98/46a2ab81d0c3b3eb00ef88fc56989aa7e9f316 +0 -0
  241. package/skills/cm-content-factory/.git 2/objects/9b/d8dd1e49cf274eaf9c555f3ab39dce7af5715e +0 -0
  242. package/skills/cm-content-factory/.git 2/objects/a1/13329fb0cec96ae78b222d33a24c3b5bc7fa1f +0 -0
  243. package/skills/cm-content-factory/.git 2/objects/a9/e6effe626e8a3aea3a8fc3364b492191c6e7d0 +0 -0
  244. package/skills/cm-content-factory/.git 2/objects/ad/6de7e48d9782cca9353d1ff0aa1aab7fe1df85 +0 -0
  245. package/skills/cm-content-factory/.git 2/objects/af/54ae316f771ff692e299ffcd8bf2f06b413b59 +0 -0
  246. package/skills/cm-content-factory/.git 2/objects/b0/4cb8b0b00dad633e731c1472161419e738d674 +0 -0
  247. package/skills/cm-content-factory/.git 2/objects/b3/094abb0b9ed46419b269e4a4e36a459690e3b0 +0 -0
  248. package/skills/cm-content-factory/.git 2/objects/b9/435c5d4baac2cfc5c83009ddd27b46b60db5f1 +0 -0
  249. package/skills/cm-content-factory/.git 2/objects/ba/5da17dbaec5ec2dcfdfd126aead518d1171d5c +0 -0
  250. package/skills/cm-content-factory/.git 2/objects/c0/bf58703aa258ba5dd63083bebaec8f223d844c +0 -0
  251. package/skills/cm-content-factory/.git 2/objects/c4/701a34edf1fc1bad58ccc57bd03f9426acb59a +0 -0
  252. package/skills/cm-content-factory/.git 2/objects/c7/5ccce9a4e5cc74d9b3174550cf6d993ca43638 +0 -0
  253. package/skills/cm-content-factory/.git 2/objects/c7/710d59b5a35b0f1f0a0399386643a0bd94c929 +0 -0
  254. package/skills/cm-content-factory/.git 2/objects/d1/fe58237112e953e5fec52da22cf38e08be3df9 +0 -5
  255. package/skills/cm-content-factory/.git 2/objects/d2/2bbe9fd2f74c95bc5583e803f5e435f1e2cd86 +0 -0
  256. package/skills/cm-content-factory/.git 2/objects/d7/e72852ea2bff74581dbf247d400120086229f4 +0 -0
  257. package/skills/cm-content-factory/.git 2/objects/d8/d4c3b5553e4fd72807e1d4b49ef07d9ef3ac35 +0 -0
  258. package/skills/cm-content-factory/.git 2/objects/dc/75050c2876f6a02ae2a53a3c886f395b622977 +0 -0
  259. package/skills/cm-content-factory/.git 2/objects/ee/e8546f95acec500187c08a28a8b9ee02db0dec +0 -0
  260. package/skills/cm-content-factory/.git 2/objects/ef/263c059208b416c2146434f10cb2b9fabcba16 +0 -0
  261. package/skills/cm-content-factory/.git 2/objects/f3/ae597e84d9a59b88acd21c99bde2eaf686d785 +0 -0
  262. package/skills/cm-content-factory/.git 2/objects/f3/f6f5673c821d3d8e76fa267a9e882e7a5387ea +0 -0
  263. package/skills/cm-content-factory/.git 2/objects/f9/6e6d0ad02624dd11d5848594d056caef7a5e8b +0 -0
  264. package/skills/cm-content-factory/.git 2/objects/ff/278988fc1edf0db3abcf18de795f4cc0b4f3e1 +0 -0
  265. package/skills/cm-content-factory/.git 2/refs/heads/main +0 -1
  266. package/skills/cm-content-factory/.git 2/refs/remotes/origin/main +0 -1
  267. package/skills/cm-content-factory/.pytest_cache 2/v/cache/nodeids +0 -76
  268. package/skills/cm-content-factory/.pytest_cache 2/v/cache/stepwise +0 -1
  269. package/skills/cm-content-factory/AGENTS.md +0 -61
  270. package/skills/cm-content-factory/CLAUDE.md +0 -63
  271. package/skills/cm-content-factory/CURSOR.md +0 -43
  272. package/skills/cm-content-factory/Content Factory.zip +0 -0
  273. package/skills/cm-content-factory/SKILL.md +0 -416
  274. package/skills/cm-content-factory/cf +0 -313
  275. package/skills/cm-content-factory/config.schema.json +0 -397
  276. package/skills/cm-content-factory/dashboard/app.js +0 -556
  277. package/skills/cm-content-factory/dashboard/index.html +0 -397
  278. package/skills/cm-content-factory/dashboard/style.css +0 -1211
  279. package/skills/cm-content-factory/examples/01-real-estate.config.json +0 -146
  280. package/skills/cm-content-factory/examples/02-personal-finance.config.json +0 -146
  281. package/skills/cm-content-factory/examples/03-health-wellness.config.json +0 -147
  282. package/skills/cm-content-factory/examples/04-saas-software.config.json +0 -147
  283. package/skills/cm-content-factory/examples/05-legal-services.config.json +0 -147
  284. package/skills/cm-content-factory/examples/06-insurance.config.json +0 -146
  285. package/skills/cm-content-factory/examples/07-ecommerce-dropship.config.json +0 -146
  286. package/skills/cm-content-factory/examples/08-online-education.config.json +0 -147
  287. package/skills/cm-content-factory/examples/09-crypto-defi.config.json +0 -147
  288. package/skills/cm-content-factory/examples/10-beauty-skincare.config.json +0 -147
  289. package/skills/cm-content-factory/examples/11-home-services.config.json +0 -146
  290. package/skills/cm-content-factory/examples/12-dental-clinic.config.json +0 -147
  291. package/skills/cm-content-factory/examples/13-pet-care.config.json +0 -147
  292. package/skills/cm-content-factory/examples/14-travel-hospitality.config.json +0 -147
  293. package/skills/cm-content-factory/examples/15-ai-automation.config.json +0 -147
  294. package/skills/cm-content-factory/examples/16-wedding-events.config.json +0 -147
  295. package/skills/cm-content-factory/examples/17-fitness-coaching.config.json +0 -148
  296. package/skills/cm-content-factory/examples/18-cybersecurity.config.json +0 -147
  297. package/skills/cm-content-factory/examples/19-food-restaurant.config.json +0 -148
  298. package/skills/cm-content-factory/examples/20-solar-energy.config.json +0 -147
  299. package/skills/cm-content-factory/examples/fitness-blog.config.json +0 -116
  300. package/skills/cm-content-factory/examples/tech-blog.config.json +0 -107
  301. package/skills/cm-content-factory/extensions/EXTENSION_GUIDE.md +0 -72
  302. package/skills/cm-content-factory/extensions/hooks.py +0 -126
  303. package/skills/cm-content-factory/extensions/openclaw_adapter.py +0 -132
  304. package/skills/cm-content-factory/landing/docs/content/changelog.md +0 -36
  305. package/skills/cm-content-factory/landing/docs/content/deployment.md +0 -46
  306. package/skills/cm-content-factory/landing/docs/content/execution-flow.md +0 -67
  307. package/skills/cm-content-factory/landing/docs/content/openspace.md +0 -27
  308. package/skills/cm-content-factory/landing/docs/content/openviking.md +0 -33
  309. package/skills/cm-content-factory/landing/docs/content/use-cases.md +0 -26
  310. package/skills/cm-content-factory/landing/docs/content/v5-intro.md +0 -28
  311. package/skills/cm-content-factory/landing/docs/index.html +0 -240
  312. package/skills/cm-content-factory/landing/index.html +0 -680
  313. package/skills/cm-content-factory/landing/script.js +0 -143
  314. package/skills/cm-content-factory/landing/style.css +0 -1216
  315. package/skills/cm-content-factory/landing/translations.js +0 -508
  316. package/skills/cm-content-factory/logs/events.jsonl +0 -11
  317. package/skills/cm-content-factory/profiles/_template.profile.json +0 -231
  318. package/skills/cm-content-factory/profiles/finance.profile.json +0 -278
  319. package/skills/cm-content-factory/profiles/legal.profile.json +0 -263
  320. package/skills/cm-content-factory/profiles/medical-research.profile.json +0 -321
  321. package/skills/cm-content-factory/profiles/technology.profile.json +0 -275
  322. package/skills/cm-content-factory/scripts/agent_dispatcher.py +0 -266
  323. package/skills/cm-content-factory/scripts/audit.py +0 -106
  324. package/skills/cm-content-factory/scripts/dashboard_server.py +0 -225
  325. package/skills/cm-content-factory/scripts/deploy.py +0 -146
  326. package/skills/cm-content-factory/scripts/extract.py +0 -132
  327. package/skills/cm-content-factory/scripts/landing_generator.py +0 -459
  328. package/skills/cm-content-factory/scripts/memory.py +0 -521
  329. package/skills/cm-content-factory/scripts/monetize.py +0 -239
  330. package/skills/cm-content-factory/scripts/pipeline.py +0 -357
  331. package/skills/cm-content-factory/scripts/plan.py +0 -163
  332. package/skills/cm-content-factory/scripts/publish.py +0 -145
  333. package/skills/cm-content-factory/scripts/research.py +0 -337
  334. package/skills/cm-content-factory/scripts/scaffold.py +0 -464
  335. package/skills/cm-content-factory/scripts/scoreboard.py +0 -336
  336. package/skills/cm-content-factory/scripts/seo.py +0 -90
  337. package/skills/cm-content-factory/scripts/state_manager.py +0 -320
  338. package/skills/cm-content-factory/scripts/token_manager.py +0 -268
  339. package/skills/cm-content-factory/scripts/validate.py +0 -221
  340. package/skills/cm-content-factory/scripts/wizard.py +0 -329
  341. package/skills/cm-content-factory/scripts/write.py +0 -93
  342. package/skills/cm-content-factory/sites/docs-site/src/assets/houston.webp +0 -0
  343. package/skills/cm-content-factory/sites/docs-site/src/content/docs/architecture.md +0 -90
  344. package/skills/cm-content-factory/sites/docs-site/src/content/docs/data-flow.md +0 -54
  345. package/skills/cm-content-factory/sites/docs-site/src/content/docs/deployment.md +0 -38
  346. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/index.md +0 -65
  347. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/lc-content-lifecycle.md +0 -48
  348. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/seq-write-mode.md +0 -39
  349. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/uj-first-batch.md +0 -42
  350. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/wf-content-pipeline.md +0 -51
  351. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/wf-learning-cycle.md +0 -52
  352. package/skills/cm-content-factory/sites/docs-site/src/content/docs/getting-started/configuration.md +0 -86
  353. package/skills/cm-content-factory/sites/docs-site/src/content/docs/getting-started/installation.md +0 -80
  354. package/skills/cm-content-factory/sites/docs-site/src/content/docs/getting-started/intro.md +0 -58
  355. package/skills/cm-content-factory/sites/docs-site/src/content/docs/index.md +0 -102
  356. package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/index.md +0 -45
  357. package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/optimize-seo.md +0 -29
  358. package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/scale-content-production.md +0 -55
  359. package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/standardize-quality.md +0 -29
  360. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/buyer-cmo-huong.md +0 -41
  361. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/buyer-content-lead-khoa.md +0 -40
  362. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/index.md +0 -56
  363. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/user-content-manager-lan.md +0 -46
  364. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/user-seo-minh.md +0 -45
  365. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/user-writer-tu.md +0 -45
  366. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/content-pipeline.md +0 -108
  367. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/index.md +0 -22
  368. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/memory-system.md +0 -52
  369. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/seo-optimization.md +0 -58
  370. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/troubleshooting-guide.md +0 -92
  371. package/skills/cm-content-factory/sites/docs-site/src/styles/custom.css +0 -575
  372. package/skills/cm-content-factory/tests/conftest.py +0 -66
  373. package/skills/cm-content-factory/tests/test_agent_dispatcher.py +0 -125
  374. package/skills/cm-content-factory/tests/test_memory.py +0 -128
  375. package/skills/cm-content-factory/tests/test_pipeline.py +0 -107
  376. package/skills/cm-content-factory/tests/test_research.py +0 -56
  377. package/skills/cm-content-factory/tests/test_state_manager.py +0 -131
  378. package/skills/cm-content-factory/tests/test_token_manager.py +0 -110
  379. package/skills/cm-content-factory/tests/test_wizard.py +0 -121
  380. package/skills/cm-cro-methodology/SKILL.md +0 -290
  381. package/skills/cm-cro-methodology/references/COPYWRITING.md +0 -178
  382. package/skills/cm-cro-methodology/references/OBJECTIONS.md +0 -135
  383. package/skills/cm-cro-methodology/references/PERSUASION.md +0 -158
  384. package/skills/cm-cro-methodology/references/RESEARCH.md +0 -220
  385. package/skills/cm-cro-methodology/references/funnel-analysis.md +0 -365
  386. package/skills/cm-cro-methodology/references/testing-methodology.md +0 -330
  387. package/skills/cm-google-form/SKILL.md +0 -266
  388. package/skills/cm-google-form/templates/apps-script.js +0 -55
  389. package/skills/cm-google-form/templates/form-markup.html +0 -110
  390. package/skills/cm-google-form/templates/form-submit.js +0 -201
  391. package/skills/cm-google-form/templates/toast.css +0 -152
  392. package/skills/cm-growth-hacking/SKILL.md +0 -282
  393. package/skills/cm-growth-hacking/bottom-sheet-engine.md +0 -261
  394. package/skills/cm-growth-hacking/calendar-integration.md +0 -264
  395. package/skills/cm-growth-hacking/references/engagement-patterns.md +0 -346
  396. package/skills/cm-growth-hacking/templates/bottom-sheet.css +0 -528
  397. package/skills/cm-growth-hacking/templates/bottom-sheet.js +0 -269
  398. package/skills/cm-growth-hacking/templates/calendar-cta.js +0 -213
  399. package/skills/cm-growth-hacking/templates/tracking-events.js +0 -211
  400. package/skills/cm-growth-hacking/templates/trigger-manager.js +0 -254
  401. package/skills/cm-growth-hacking/tracking-events.md +0 -246
  402. package/skills/cm-growth-hacking/trigger-system.md +0 -342
  403. package/skills/cm-jtbd/SKILL.md +0 -98
  404. package/skills/cm-notebooklm/SKILL.md +0 -156
  405. package/skills/cm-notebooklm/references/command_reference.md +0 -94
  406. package/skills/cm-notebooklm/references/workflows.md +0 -60
  407. package/skills/cm-notebooklm/resources/knowledge_sources.md +0 -106
  408. package/skills/cm-notebooklm/scripts/brain-sync.sh +0 -453
  409. package/skills/cm-notebooklm/scripts/graduate_wisdom.py +0 -101
  410. package/skills/cm-readit/SKILL.md +0 -289
  411. package/skills/cm-readit/audio-player.md +0 -206
  412. package/skills/cm-readit/examples/blog-reader.js +0 -352
  413. package/skills/cm-readit/examples/voice-cro.js +0 -390
  414. package/skills/cm-readit/tts-engine.md +0 -262
  415. package/skills/cm-readit/ui-patterns.md +0 -362
  416. package/skills/cm-readit/voice-cro.md +0 -223
@@ -1,320 +0,0 @@
1
- #!/usr/bin/env python3
2
- """
3
- State Manager — Central state management for Content Factory dashboard.
4
-
5
- All pipeline scripts write state here; dashboard reads it.
6
- State persisted to `.content-factory-state.json` at project root.
7
- Events logged to `logs/events.jsonl` (append-only).
8
-
9
- Usage:
10
- from state_manager import StateManager
11
- sm = StateManager("/path/to/project")
12
- sm.update_phase("write", "running", progress=0.5)
13
- sm.add_task("write-article-1", "writing", {"topic": "SEO Tips"})
14
- sm.log_event("info", "Article 1 started")
15
- """
16
-
17
- import json
18
- import os
19
- import time
20
- import threading
21
- from pathlib import Path
22
- from datetime import datetime
23
-
24
-
25
- class StateManager:
26
- """Thread-safe state manager for Content Factory pipeline."""
27
-
28
- STATE_FILE = ".content-factory-state.json"
29
- EVENTS_FILE = "logs/events.jsonl"
30
-
31
- def __init__(self, project_root: str = None):
32
- self.project_root = Path(project_root or os.getcwd()).resolve()
33
- self.state_path = self.project_root / self.STATE_FILE
34
- self.events_path = self.project_root / self.EVENTS_FILE
35
- self._lock = threading.Lock()
36
- self._ensure_dirs()
37
-
38
- def _ensure_dirs(self):
39
- """Create logs directory if needed."""
40
- self.events_path.parent.mkdir(parents=True, exist_ok=True)
41
-
42
- def _default_state(self) -> dict:
43
- """Return default empty state."""
44
- return {
45
- "version": "2.0",
46
- "started_at": datetime.now().isoformat(),
47
- "updated_at": datetime.now().isoformat(),
48
- "status": "idle",
49
- "pipeline": {
50
- "current_phase": None,
51
- "phases": {
52
- "extract": {"status": "pending", "progress": 0, "started_at": None, "finished_at": None},
53
- "plan": {"status": "pending", "progress": 0, "started_at": None, "finished_at": None},
54
- "write": {"status": "pending", "progress": 0, "started_at": None, "finished_at": None},
55
- "audit": {"status": "pending", "progress": 0, "started_at": None, "finished_at": None},
56
- "seo": {"status": "pending", "progress": 0, "started_at": None, "finished_at": None},
57
- "publish": {"status": "pending", "progress": 0, "started_at": None, "finished_at": None},
58
- },
59
- },
60
- "tasks": [],
61
- "tokens": {
62
- "total_input": 0,
63
- "total_output": 0,
64
- "total_cost_usd": 0.0,
65
- "budget_limit_usd": None,
66
- "providers": {},
67
- },
68
- "errors": [],
69
- "stats": {
70
- "articles_total": 0,
71
- "articles_written": 0,
72
- "articles_passed_audit": 0,
73
- "articles_failed": 0,
74
- },
75
- "agents": [],
76
- }
77
-
78
- def load(self) -> dict:
79
- """Load current state from disk."""
80
- if not self.state_path.exists():
81
- return self._default_state()
82
- try:
83
- with open(self.state_path, "r", encoding="utf-8") as f:
84
- return json.load(f)
85
- except (json.JSONDecodeError, IOError):
86
- return self._default_state()
87
-
88
- def save(self, state: dict):
89
- """Save state to disk (thread-safe)."""
90
- state["updated_at"] = datetime.now().isoformat()
91
- with self._lock:
92
- tmp = self.state_path.with_suffix(".tmp")
93
- with open(tmp, "w", encoding="utf-8") as f:
94
- json.dump(state, f, indent=2, ensure_ascii=False)
95
- tmp.replace(self.state_path)
96
-
97
- def update_phase(self, phase: str, status: str, progress: float = None, error: str = None):
98
- """Update phase status and optional progress (0.0 - 1.0)."""
99
- state = self.load()
100
- phase_data = state["pipeline"]["phases"].get(phase, {})
101
-
102
- phase_data["status"] = status
103
- if progress is not None:
104
- phase_data["progress"] = min(1.0, max(0.0, progress))
105
- if status == "running" and not phase_data.get("started_at"):
106
- phase_data["started_at"] = datetime.now().isoformat()
107
- if status in ("done", "failed"):
108
- phase_data["finished_at"] = datetime.now().isoformat()
109
-
110
- state["pipeline"]["phases"][phase] = phase_data
111
- state["pipeline"]["current_phase"] = phase if status == "running" else state["pipeline"]["current_phase"]
112
-
113
- # Update overall status
114
- if status == "running":
115
- state["status"] = "running"
116
- elif all(p["status"] == "done" for p in state["pipeline"]["phases"].values()):
117
- state["status"] = "completed"
118
- elif any(p["status"] == "failed" for p in state["pipeline"]["phases"].values()):
119
- state["status"] = "error"
120
-
121
- if error:
122
- self._add_error(state, phase, error)
123
-
124
- self.save(state)
125
- self.log_event("info" if status != "failed" else "error",
126
- f"Phase {phase}: {status}" + (f" ({progress*100:.0f}%)" if progress else ""),
127
- {"phase": phase, "status": status})
128
-
129
- def add_task(self, task_id: str, status: str, meta: dict = None):
130
- """Add or update a task in the queue."""
131
- state = self.load()
132
- existing = next((t for t in state["tasks"] if t["id"] == task_id), None)
133
- now = datetime.now().isoformat()
134
-
135
- if existing:
136
- existing["status"] = status
137
- existing["updated_at"] = now
138
- if meta:
139
- existing["meta"].update(meta)
140
- else:
141
- state["tasks"].append({
142
- "id": task_id,
143
- "status": status,
144
- "created_at": now,
145
- "updated_at": now,
146
- "meta": meta or {},
147
- })
148
-
149
- # Keep last 200 tasks max
150
- if len(state["tasks"]) > 200:
151
- done = [t for t in state["tasks"] if t["status"] in ("done", "failed")]
152
- active = [t for t in state["tasks"] if t["status"] not in ("done", "failed")]
153
- state["tasks"] = active + done[-100:]
154
-
155
- self.save(state)
156
-
157
- def update_tokens(self, provider: str, input_tokens: int = 0, output_tokens: int = 0, cost_usd: float = 0.0):
158
- """Track token usage by provider."""
159
- state = self.load()
160
- tokens = state["tokens"]
161
- tokens["total_input"] += input_tokens
162
- tokens["total_output"] += output_tokens
163
- tokens["total_cost_usd"] = round(tokens["total_cost_usd"] + cost_usd, 6)
164
-
165
- if provider not in tokens["providers"]:
166
- tokens["providers"][provider] = {"input": 0, "output": 0, "cost_usd": 0.0, "requests": 0}
167
-
168
- p = tokens["providers"][provider]
169
- p["input"] += input_tokens
170
- p["output"] += output_tokens
171
- p["cost_usd"] = round(p["cost_usd"] + cost_usd, 6)
172
- p["requests"] += 1
173
-
174
- self.save(state)
175
-
176
- def check_budget(self) -> bool:
177
- """Check if spending is within budget. Returns True if OK."""
178
- state = self.load()
179
- limit = state["tokens"].get("budget_limit_usd")
180
- if limit is None:
181
- return True
182
- return state["tokens"]["total_cost_usd"] < limit
183
-
184
- def set_budget(self, budget_usd: float):
185
- """Set budget limit."""
186
- state = self.load()
187
- state["tokens"]["budget_limit_usd"] = budget_usd
188
- self.save(state)
189
-
190
- def update_stats(self, **kwargs):
191
- """Update article statistics."""
192
- state = self.load()
193
- for k, v in kwargs.items():
194
- if k in state["stats"]:
195
- state["stats"][k] = v
196
- self.save(state)
197
-
198
- def increment_stat(self, key: str, amount: int = 1):
199
- """Increment a stat counter."""
200
- state = self.load()
201
- if key in state["stats"]:
202
- state["stats"][key] += amount
203
- self.save(state)
204
-
205
- def register_agent(self, agent_id: str, agent_type: str):
206
- """Register an active agent."""
207
- state = self.load()
208
- existing = next((a for a in state["agents"] if a["id"] == agent_id), None)
209
- now = datetime.now().isoformat()
210
- if existing:
211
- existing["last_seen"] = now
212
- else:
213
- state["agents"].append({
214
- "id": agent_id,
215
- "type": agent_type,
216
- "registered_at": now,
217
- "last_seen": now,
218
- })
219
- self.save(state)
220
-
221
- def _add_error(self, state: dict, source: str, message: str):
222
- """Add error to state (max 50)."""
223
- state["errors"].append({
224
- "source": source,
225
- "message": message,
226
- "timestamp": datetime.now().isoformat(),
227
- })
228
- state["errors"] = state["errors"][-50:]
229
-
230
- def add_error(self, source: str, message: str):
231
- """Public method to add an error."""
232
- state = self.load()
233
- self._add_error(state, source, message)
234
- self.save(state)
235
- self.log_event("error", message, {"source": source})
236
-
237
- def log_event(self, level: str, message: str, data: dict = None):
238
- """Append event to JSONL log file."""
239
- event = {
240
- "ts": datetime.now().isoformat(),
241
- "level": level,
242
- "msg": message,
243
- }
244
- if data:
245
- event["data"] = data
246
-
247
- with self._lock:
248
- with open(self.events_path, "a", encoding="utf-8") as f:
249
- f.write(json.dumps(event, ensure_ascii=False) + "\n")
250
-
251
- def get_recent_events(self, n: int = 100) -> list:
252
- """Read last N events from log."""
253
- if not self.events_path.exists():
254
- return []
255
- try:
256
- with open(self.events_path, "r", encoding="utf-8") as f:
257
- lines = f.readlines()
258
- events = []
259
- for line in lines[-n:]:
260
- line = line.strip()
261
- if line:
262
- try:
263
- events.append(json.loads(line))
264
- except json.JSONDecodeError:
265
- pass
266
- return events
267
- except IOError:
268
- return []
269
-
270
- def reset(self):
271
- """Reset state to default (for new pipeline run)."""
272
- state = self._default_state()
273
- self.save(state)
274
- self.log_event("info", "State reset — new pipeline run")
275
-
276
- def get_snapshot(self) -> dict:
277
- """Get full state snapshot including recent events."""
278
- state = self.load()
279
- state["recent_events"] = self.get_recent_events(50)
280
- return state
281
-
282
-
283
- # CLI interface
284
- if __name__ == "__main__":
285
- import sys
286
- sm = StateManager()
287
-
288
- if len(sys.argv) < 2:
289
- print("Usage: python3 state_manager.py <command>")
290
- print(" status — Show current state")
291
- print(" reset — Reset state")
292
- print(" events — Show recent events")
293
- sys.exit(0)
294
-
295
- cmd = sys.argv[1]
296
- if cmd == "status":
297
- state = sm.load()
298
- print(f"\n🏭 Content Factory State")
299
- print(f" Status: {state['status']}")
300
- print(f" Updated: {state['updated_at']}")
301
- print(f"\n Pipeline:")
302
- for phase, data in state["pipeline"]["phases"].items():
303
- icon = {"pending": "⬜", "running": "🔄", "done": "✅", "failed": "❌"}.get(data["status"], "❓")
304
- progress = f" ({data['progress']*100:.0f}%)" if data["progress"] > 0 else ""
305
- print(f" {icon} {phase}{progress}")
306
- print(f"\n Tokens: {state['tokens']['total_input']}in / {state['tokens']['total_output']}out")
307
- print(f" Cost: ${state['tokens']['total_cost_usd']:.4f}")
308
- print(f" Tasks: {len(state['tasks'])}")
309
- print(f" Errors: {len(state['errors'])}")
310
- print(f" Agents: {len(state['agents'])}")
311
-
312
- elif cmd == "reset":
313
- sm.reset()
314
- print("✅ State reset")
315
-
316
- elif cmd == "events":
317
- events = sm.get_recent_events(20)
318
- for e in events:
319
- icon = {"info": "ℹ️", "warn": "⚠️", "error": "❌"}.get(e["level"], "•")
320
- print(f" {icon} [{e['ts'][-8:]}] {e['msg']}")
@@ -1,268 +0,0 @@
1
- #!/usr/bin/env python3
2
- """
3
- Token Manager — Track token usage, costs, rate limits, and budget for AI providers.
4
-
5
- Monitors API consumption across providers (Gemini, Claude, OpenAI).
6
- Implements circuit breaker pattern and exponential backoff.
7
-
8
- Usage:
9
- from token_manager import TokenManager
10
- tm = TokenManager("/path/to/project")
11
- tm.record_usage("gemini", input_tokens=500, output_tokens=200)
12
- if not tm.check_budget():
13
- print("Budget exceeded!")
14
- tm.wait_if_rate_limited("gemini")
15
- """
16
-
17
- import json
18
- import time
19
- import math
20
- from pathlib import Path
21
- from datetime import datetime
22
-
23
-
24
- # Cost per 1M tokens (USD) — 2025 pricing estimates
25
- PROVIDER_COSTS = {
26
- "gemini": {"input": 0.15, "output": 0.60},
27
- "gemini-pro": {"input": 1.25, "output": 5.00},
28
- "claude-sonnet": {"input": 3.00, "output": 15.00},
29
- "claude-haiku": {"input": 0.25, "output": 1.25},
30
- "openai-gpt4o": {"input": 2.50, "output": 10.00},
31
- "openai-gpt4o-mini": {"input": 0.15, "output": 0.60},
32
- }
33
-
34
- # Default rate limits (requests per minute)
35
- RATE_LIMITS = {
36
- "gemini": 60,
37
- "gemini-pro": 30,
38
- "claude-sonnet": 50,
39
- "claude-haiku": 100,
40
- "openai-gpt4o": 60,
41
- "openai-gpt4o-mini": 100,
42
- }
43
-
44
-
45
- class TokenManager:
46
- """Token usage tracker with rate limiting and circuit breaker."""
47
-
48
- USAGE_FILE = "logs/token_usage.json"
49
- MAX_CONSECUTIVE_FAILURES = 5
50
- BACKOFF_BASE_SECONDS = 2.0
51
- BACKOFF_MAX_SECONDS = 120.0
52
-
53
- def __init__(self, project_root: str = None, budget_usd: float = None):
54
- self.project_root = Path(project_root or ".").resolve()
55
- self.usage_path = self.project_root / self.USAGE_FILE
56
- self.budget_usd = budget_usd
57
- self._request_timestamps: dict[str, list] = {}
58
- self._consecutive_failures: dict[str, int] = {}
59
- self._ensure_dirs()
60
-
61
- def _ensure_dirs(self):
62
- self.usage_path.parent.mkdir(parents=True, exist_ok=True)
63
-
64
- def _load_usage(self) -> dict:
65
- if not self.usage_path.exists():
66
- return self._default_usage()
67
- try:
68
- with open(self.usage_path, "r", encoding="utf-8") as f:
69
- return json.load(f)
70
- except (json.JSONDecodeError, IOError):
71
- return self._default_usage()
72
-
73
- def _default_usage(self) -> dict:
74
- return {
75
- "session_id": datetime.now().strftime("%Y%m%d_%H%M%S"),
76
- "started_at": datetime.now().isoformat(),
77
- "updated_at": datetime.now().isoformat(),
78
- "total_input_tokens": 0,
79
- "total_output_tokens": 0,
80
- "total_cost_usd": 0.0,
81
- "budget_limit_usd": self.budget_usd,
82
- "budget_remaining_usd": self.budget_usd,
83
- "providers": {},
84
- "requests": [],
85
- }
86
-
87
- def _save_usage(self, usage: dict):
88
- usage["updated_at"] = datetime.now().isoformat()
89
- tmp = self.usage_path.with_suffix(".tmp")
90
- with open(tmp, "w", encoding="utf-8") as f:
91
- json.dump(usage, f, indent=2, ensure_ascii=False)
92
- tmp.replace(self.usage_path)
93
-
94
- def estimate_cost(self, provider: str, input_tokens: int, output_tokens: int) -> float:
95
- """Estimate cost for a given request."""
96
- costs = PROVIDER_COSTS.get(provider, {"input": 1.0, "output": 3.0})
97
- input_cost = (input_tokens / 1_000_000) * costs["input"]
98
- output_cost = (output_tokens / 1_000_000) * costs["output"]
99
- return round(input_cost + output_cost, 6)
100
-
101
- def record_usage(self, provider: str, input_tokens: int = 0, output_tokens: int = 0,
102
- success: bool = True, task_id: str = None):
103
- """Record a completed API request."""
104
- cost = self.estimate_cost(provider, input_tokens, output_tokens)
105
- usage = self._load_usage()
106
-
107
- usage["total_input_tokens"] += input_tokens
108
- usage["total_output_tokens"] += output_tokens
109
- usage["total_cost_usd"] = round(usage["total_cost_usd"] + cost, 6)
110
-
111
- if usage["budget_limit_usd"] is not None:
112
- usage["budget_remaining_usd"] = round(
113
- usage["budget_limit_usd"] - usage["total_cost_usd"], 6
114
- )
115
-
116
- # Per-provider stats
117
- if provider not in usage["providers"]:
118
- usage["providers"][provider] = {
119
- "input_tokens": 0, "output_tokens": 0,
120
- "cost_usd": 0.0, "requests": 0,
121
- "failures": 0,
122
- }
123
- p = usage["providers"][provider]
124
- p["input_tokens"] += input_tokens
125
- p["output_tokens"] += output_tokens
126
- p["cost_usd"] = round(p["cost_usd"] + cost, 6)
127
- p["requests"] += 1
128
- if not success:
129
- p["failures"] += 1
130
-
131
- # Append request entry (keep last 500)
132
- usage["requests"].append({
133
- "ts": datetime.now().isoformat(),
134
- "provider": provider,
135
- "input": input_tokens,
136
- "output": output_tokens,
137
- "cost": cost,
138
- "success": success,
139
- "task_id": task_id,
140
- })
141
- usage["requests"] = usage["requests"][-500:]
142
-
143
- self._save_usage(usage)
144
-
145
- # Track for circuit breaker
146
- if success:
147
- self._consecutive_failures[provider] = 0
148
- else:
149
- self._consecutive_failures[provider] = \
150
- self._consecutive_failures.get(provider, 0) + 1
151
-
152
- return cost
153
-
154
- def check_budget(self) -> bool:
155
- """Returns True if within budget (or no budget set)."""
156
- usage = self._load_usage()
157
- limit = usage.get("budget_limit_usd") or self.budget_usd
158
- if limit is None:
159
- return True
160
- return usage["total_cost_usd"] < limit
161
-
162
- def get_budget_status(self) -> dict:
163
- """Get budget status summary."""
164
- usage = self._load_usage()
165
- limit = usage.get("budget_limit_usd") or self.budget_usd
166
- spent = usage["total_cost_usd"]
167
- return {
168
- "limit_usd": limit,
169
- "spent_usd": spent,
170
- "remaining_usd": round((limit or 0) - spent, 6) if limit else None,
171
- "percentage_used": round((spent / limit) * 100, 1) if limit else 0,
172
- "within_budget": spent < limit if limit else True,
173
- }
174
-
175
- def is_circuit_open(self, provider: str) -> bool:
176
- """Check if circuit breaker is tripped (too many failures)."""
177
- return self._consecutive_failures.get(provider, 0) >= self.MAX_CONSECUTIVE_FAILURES
178
-
179
- def get_backoff_seconds(self, provider: str) -> float:
180
- """Calculate exponential backoff delay."""
181
- failures = self._consecutive_failures.get(provider, 0)
182
- if failures == 0:
183
- return 0.0
184
- delay = min(
185
- self.BACKOFF_BASE_SECONDS * math.pow(2, failures - 1),
186
- self.BACKOFF_MAX_SECONDS,
187
- )
188
- return delay
189
-
190
- def wait_if_rate_limited(self, provider: str):
191
- """Wait if needed to respect rate limits."""
192
- now = time.time()
193
- limit = RATE_LIMITS.get(provider, 60)
194
- window = 60.0 # 1 minute window
195
-
196
- if provider not in self._request_timestamps:
197
- self._request_timestamps[provider] = []
198
-
199
- # Clean old timestamps
200
- self._request_timestamps[provider] = [
201
- ts for ts in self._request_timestamps[provider]
202
- if now - ts < window
203
- ]
204
-
205
- if len(self._request_timestamps[provider]) >= limit:
206
- oldest = self._request_timestamps[provider][0]
207
- wait_time = window - (now - oldest) + 0.5
208
- if wait_time > 0:
209
- time.sleep(wait_time)
210
-
211
- self._request_timestamps[provider].append(time.time())
212
-
213
- def reset_circuit(self, provider: str):
214
- """Manually reset circuit breaker for a provider."""
215
- self._consecutive_failures[provider] = 0
216
-
217
- def get_summary(self) -> dict:
218
- """Get usage summary."""
219
- usage = self._load_usage()
220
- return {
221
- "total_input_tokens": usage["total_input_tokens"],
222
- "total_output_tokens": usage["total_output_tokens"],
223
- "total_cost_usd": usage["total_cost_usd"],
224
- "budget": self.get_budget_status(),
225
- "providers": usage["providers"],
226
- "total_requests": len(usage["requests"]),
227
- }
228
-
229
- def reset(self):
230
- """Reset usage for new session."""
231
- usage = self._default_usage()
232
- self._save_usage(usage)
233
- self._request_timestamps.clear()
234
- self._consecutive_failures.clear()
235
-
236
-
237
- # CLI interface
238
- if __name__ == "__main__":
239
- import sys
240
- tm = TokenManager()
241
-
242
- if len(sys.argv) < 2:
243
- print("Usage: python3 token_manager.py <command>")
244
- print(" status — Show token usage summary")
245
- print(" reset — Reset usage for new session")
246
- sys.exit(0)
247
-
248
- cmd = sys.argv[1]
249
- if cmd == "status":
250
- s = tm.get_summary()
251
- print(f"\n💰 Token Usage Summary")
252
- print(f" Input: {s['total_input_tokens']:,} tokens")
253
- print(f" Output: {s['total_output_tokens']:,} tokens")
254
- print(f" Cost: ${s['total_cost_usd']:.4f}")
255
- print(f" Requests: {s['total_requests']}")
256
- b = s["budget"]
257
- if b["limit_usd"]:
258
- bar_len = 20
259
- filled = int(bar_len * b["percentage_used"] / 100)
260
- bar = "█" * filled + "░" * (bar_len - filled)
261
- print(f" Budget: [{bar}] {b['percentage_used']}% (${b['spent_usd']:.4f}/${b['limit_usd']:.2f})")
262
- print(f"\n Per Provider:")
263
- for name, data in s["providers"].items():
264
- print(f" {name}: {data['requests']} req, ${data['cost_usd']:.4f}, {data['failures']} failures")
265
-
266
- elif cmd == "reset":
267
- tm.reset()
268
- print("✅ Token usage reset")