codymaster 5.2.0 โ†’ 7.0.1

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 (380) hide show
  1. package/CHANGELOG.md +276 -0
  2. package/README.md +216 -333
  3. package/dist/agent/antigravity.js +152 -0
  4. package/dist/agent/backend.js +2 -0
  5. package/dist/agent/claude.js +196 -0
  6. package/dist/agent/codex.js +204 -0
  7. package/dist/agent/copilot.js +284 -0
  8. package/dist/agent/cursor.js +211 -0
  9. package/dist/agent/factory.js +30 -0
  10. package/dist/agent/gemini.js +142 -0
  11. package/dist/agent/opencode.js +205 -0
  12. package/dist/agent/spawn-helper.js +237 -0
  13. package/dist/agent/version.js +25 -0
  14. package/dist/browse/adapter-factory.js +69 -0
  15. package/dist/browse/adapters/agent-browser-adapter.js +305 -0
  16. package/dist/browse/adapters/playwright-adapter.js +309 -0
  17. package/dist/browse/adapters/types.js +6 -0
  18. package/dist/browse/error-collector.js +132 -0
  19. package/dist/browse/event-log.js +109 -0
  20. package/dist/browse/index.js +17 -0
  21. package/dist/browse-server.js +204 -120
  22. package/dist/cli/command-registry.js +12 -0
  23. package/dist/cli/commands/dashboard.js +76 -2
  24. package/dist/cli/commands/engineering.js +218 -4
  25. package/dist/cli/commands/install.js +160 -0
  26. package/dist/cli/commands/learn.js +181 -0
  27. package/dist/cli/commands/parallel.js +138 -0
  28. package/dist/cli/commands/quality.js +105 -0
  29. package/dist/cli/commands/stack.js +49 -0
  30. package/dist/cli/commands/update.js +159 -0
  31. package/dist/cli/update-check.js +94 -10
  32. package/dist/continuity.js +3 -1
  33. package/dist/dashboard.js +47 -6
  34. package/dist/data.js +35 -0
  35. package/dist/execution/tdd-gate.js +113 -0
  36. package/dist/executor/cancel.js +34 -0
  37. package/dist/executor/gc.js +74 -0
  38. package/dist/executor/index.js +14 -0
  39. package/dist/executor/runner.js +70 -0
  40. package/dist/executor/workdir.js +31 -0
  41. package/dist/handoff/contracts.js +22 -0
  42. package/dist/handoff/index.js +18 -0
  43. package/dist/handoff/io.js +121 -0
  44. package/dist/index.js +7 -3
  45. package/dist/indexer/stack-detect.js +219 -0
  46. package/dist/install/copy.js +98 -0
  47. package/dist/install/engine.js +42 -0
  48. package/dist/install/paths.js +70 -0
  49. package/dist/install/platforms/_simple.js +85 -0
  50. package/dist/install/platforms/antigravity.js +91 -0
  51. package/dist/install/platforms/claude-code.js +107 -0
  52. package/dist/install/platforms/cursor.js +77 -0
  53. package/dist/install/platforms/index.js +27 -0
  54. package/dist/install/platforms/simple.js +163 -0
  55. package/dist/install/profiles.js +75 -0
  56. package/dist/install/types.js +2 -0
  57. package/dist/learnings.js +208 -0
  58. package/dist/mcp-context-server.js +1 -1
  59. package/dist/middleware/metrics.js +30 -0
  60. package/dist/middleware/security-headers.js +14 -0
  61. package/dist/realtime/event-bus.js +29 -0
  62. package/dist/realtime/ws-hub.js +91 -0
  63. package/dist/schemas/task-schema.js +48 -0
  64. package/dist/schemas/validate.js +18 -0
  65. package/dist/skills-lock.js +96 -0
  66. package/dist/sprint-pipeline.js +26 -0
  67. package/dist/storage/index.js +21 -0
  68. package/dist/storage/repos/activity-repo.js +46 -0
  69. package/dist/storage/repos/message-repo.js +39 -0
  70. package/dist/storage/repos/project-repo.js +56 -0
  71. package/dist/storage/repos/task-repo.js +142 -0
  72. package/dist/storage/services/project-service.js +49 -0
  73. package/dist/storage/services/task-service.js +97 -0
  74. package/dist/storage/sqlite.js +113 -0
  75. package/dist/tier-classify.js +131 -0
  76. package/dist/ui/onboarding.js +51 -15
  77. package/dist/utils/cli-utils.js +7 -2
  78. package/dist/utils/design-taste.js +108 -0
  79. package/dist/utils/output-compress.js +143 -0
  80. package/dist/vibecoding-index.js +126 -0
  81. package/package.json +27 -4
  82. package/public/dashboard/app.js +52 -1
  83. package/scripts/build-skills-lock.mjs +88 -0
  84. package/scripts/build-skills.mjs +187 -28
  85. package/scripts/compress-skill.mjs +73 -0
  86. package/scripts/deprecate-skill.mjs +72 -0
  87. package/scripts/install.sh +170 -0
  88. package/scripts/mcp-bridge.js +2 -2
  89. package/scripts/postinstall.js +54 -287
  90. package/scripts/release.sh +126 -0
  91. package/scripts/update-changelog.sh +88 -0
  92. package/scripts/validate-skills.mjs +101 -4
  93. package/skills/_shared/SKILL_TEMPLATE.md +62 -0
  94. package/skills/cm-autopilot/scripts/autopilot.py +19 -2
  95. package/skills/cm-brainstorm-idea/SKILL.md +9 -0
  96. package/skills/cm-clean-code/SKILL.md +20 -0
  97. package/skills/cm-code-review/SKILL.md +21 -0
  98. package/skills/cm-codeintell/SKILL.md +9 -0
  99. package/skills/cm-conductor-worktrees/SKILL.archive.md +28 -0
  100. package/skills/cm-conductor-worktrees/SKILL.md +17 -19
  101. package/skills/cm-continuity/SKILL.md +9 -0
  102. package/skills/cm-dashboard/SKILL.archive.md +15 -0
  103. package/skills/cm-dashboard/SKILL.md +20 -9
  104. package/skills/cm-dashboard/ui/app.js +9 -1
  105. package/skills/cm-debugging/SKILL.md +9 -0
  106. package/skills/cm-design-studio/SKILL.archive.md +34 -0
  107. package/skills/cm-design-studio/SKILL.md +17 -25
  108. package/skills/cm-design-system/SKILL.md +1 -0
  109. package/skills/cm-engineering-meta/SKILL.archive.md +73 -0
  110. package/skills/cm-engineering-meta/SKILL.md +16 -63
  111. package/skills/cm-execution/SKILL.md +98 -0
  112. package/skills/cm-git-worktrees/SKILL.archive.md +157 -0
  113. package/skills/cm-git-worktrees/SKILL.md +15 -146
  114. package/skills/cm-identity-guard/SKILL.md +8 -0
  115. package/skills/cm-planning/SKILL.md +63 -92
  116. package/skills/cm-post-deploy-canary/SKILL.archive.md +22 -0
  117. package/skills/cm-post-deploy-canary/SKILL.md +17 -13
  118. package/skills/cm-qa-visual-cli/SKILL.archive.md +22 -0
  119. package/skills/cm-qa-visual-cli/SKILL.md +16 -12
  120. package/skills/cm-quality-gate/SKILL.md +38 -0
  121. package/skills/cm-safe-deploy/SKILL.md +9 -0
  122. package/skills/cm-second-opinion-cli/SKILL.archive.md +23 -0
  123. package/skills/cm-second-opinion-cli/SKILL.md +17 -14
  124. package/skills/cm-secret-shield/SKILL.archive.md +580 -0
  125. package/skills/cm-secret-shield/SKILL.md +15 -569
  126. package/skills/cm-security-gate/SKILL.archive.md +239 -0
  127. package/skills/cm-security-gate/SKILL.md +15 -228
  128. package/skills/cm-skill-health/SKILL.archive.md +83 -0
  129. package/skills/cm-skill-health/SKILL.md +16 -73
  130. package/skills/cm-skill-index/SKILL.md +8 -0
  131. package/skills/cm-skill-mastery/SKILL.archive.md +156 -0
  132. package/skills/cm-skill-mastery/SKILL.md +16 -146
  133. package/skills/cm-skill-search/SKILL.archive.md +49 -0
  134. package/skills/cm-skill-search/SKILL.md +17 -40
  135. package/skills/cm-skill-share/SKILL.archive.md +58 -0
  136. package/skills/cm-skill-share/SKILL.md +17 -49
  137. package/skills/cm-sprint-bus/SKILL.md +9 -0
  138. package/skills/cm-start/SKILL.md +17 -0
  139. package/skills/cm-tdd/SKILL.md +19 -0
  140. package/skills/cm-terminal/SKILL.md +15 -0
  141. package/skills/cm-test-gate/SKILL.archive.md +245 -0
  142. package/skills/cm-test-gate/SKILL.md +15 -234
  143. package/skills/cm-ui-preview/SKILL.archive.md +153 -0
  144. package/skills/cm-ui-preview/SKILL.md +16 -143
  145. package/skills/cm-ux-master/cli/uxmaster/commands/mcp.py +1 -1
  146. package/skills/cm-ux-master/mcp/mcp-config.json +1 -1
  147. package/skills/cm-ux-master/mcp/server.py +2 -2
  148. package/skills/profiles/design.txt +1 -1
  149. package/skills/profiles/full.txt +0 -10
  150. package/skills/profiles/growth.txt +8 -8
  151. package/skills/profiles/knowledge.txt +1 -1
  152. package/skills/profiles/top35.json +41 -0
  153. package/adapters/antigravity.js +0 -15
  154. package/adapters/claude-code.js +0 -17
  155. package/adapters/cursor.js +0 -16
  156. package/skills/cm-ads-tracker/SKILL.md +0 -401
  157. package/skills/cm-ads-tracker/evals/evals.json +0 -55
  158. package/skills/cm-ads-tracker/references/gtm-architecture.md +0 -321
  159. package/skills/cm-ads-tracker/references/industry-events.md +0 -294
  160. package/skills/cm-ads-tracker/references/platforms-api.md +0 -238
  161. package/skills/cm-ads-tracker/templates/capi-payload.md +0 -79
  162. package/skills/cm-ads-tracker/templates/datalayer-push.js +0 -104
  163. package/skills/cm-ads-tracker/templates/gtm-variables.js +0 -56
  164. package/skills/cm-auto-publisher/SKILL.md +0 -81
  165. package/skills/cm-booking-calendar/SKILL.md +0 -521
  166. package/skills/cm-booking-calendar/references/industry-patterns.md +0 -527
  167. package/skills/cm-booking-calendar/templates/booking-form.css +0 -626
  168. package/skills/cm-booking-calendar/templates/booking-form.html +0 -477
  169. package/skills/cm-booking-calendar/templates/calendar-engine.js +0 -419
  170. package/skills/cm-booking-calendar/templates/calendar-export.js +0 -395
  171. package/skills/cm-booking-calendar/templates/reminder-config.js +0 -629
  172. package/skills/cm-content-factory/.content-factory-state.json +0 -132
  173. package/skills/cm-content-factory/.git 2/logs/refs/heads/main +0 -1
  174. package/skills/cm-content-factory/.git 2/logs/refs/remotes/origin/main +0 -1
  175. package/skills/cm-content-factory/.git 2/objects/02/fb0956734b5f8ba3f918b7defd04a89cfe0076 +0 -0
  176. package/skills/cm-content-factory/.git 2/objects/08/1e129d75dc6feac6c02037272e6bd1a04e3324 +0 -0
  177. package/skills/cm-content-factory/.git 2/objects/0c/5393416f3c5e01c9a655a802bff0dd52f76f0a +0 -0
  178. package/skills/cm-content-factory/.git 2/objects/10/0b9be46978a946a77188f68be725098a122001 +0 -0
  179. package/skills/cm-content-factory/.git 2/objects/10/cf041167fc9843610eb3d90259ef3396315fdc +0 -0
  180. package/skills/cm-content-factory/.git 2/objects/12/5e19538dd6e1338ffe74f6c4c165b00435bf48 +0 -0
  181. package/skills/cm-content-factory/.git 2/objects/16/a9b9d0088d5c1347628b45a2620b479d8ad57c +0 -0
  182. package/skills/cm-content-factory/.git 2/objects/17/8c2a9ef93c33ae4eec9d58e82321f9229843a1 +0 -0
  183. package/skills/cm-content-factory/.git 2/objects/25/397ae41d09104d763bdcac2695209d85cdea89 +0 -0
  184. package/skills/cm-content-factory/.git 2/objects/2f/a836b7947f2d458e1f639788bf4bb0983a3305 +0 -0
  185. package/skills/cm-content-factory/.git 2/objects/3a/baaaf0a1c0909c0828335791557125fba911e0 +0 -0
  186. package/skills/cm-content-factory/.git 2/objects/42/2924221b81f5ce3c4e4daac9a64a24f9b01f9a +0 -0
  187. package/skills/cm-content-factory/.git 2/objects/42/ec0ce707447dc11446a34c9995fb8533801731 +0 -0
  188. package/skills/cm-content-factory/.git 2/objects/46/e43ce92866d56ce74b1d750db307cfe6154a15 +0 -0
  189. package/skills/cm-content-factory/.git 2/objects/48/5e41b633c63f55b8277bcc59f44f67681f671a +0 -0
  190. package/skills/cm-content-factory/.git 2/objects/49/49c596a3a89fa240642acd95dd3258e261eb09 +0 -0
  191. package/skills/cm-content-factory/.git 2/objects/50/9d42d8412ef8eaf7f7e138476bac2e4d10ce60 +0 -0
  192. package/skills/cm-content-factory/.git 2/objects/55/0c8c389d981b463ef849aeb792d8be3ccb6ec8 +0 -0
  193. package/skills/cm-content-factory/.git 2/objects/5d/82d3b18410cdda3ace3677436f0cb599dbe2d2 +0 -0
  194. package/skills/cm-content-factory/.git 2/objects/60/0617c58e871a38b33bf29e282d132bb3c381ad +0 -0
  195. package/skills/cm-content-factory/.git 2/objects/6a/8369a99c687b7245c92ffaf0e0f0dab9014504 +0 -0
  196. package/skills/cm-content-factory/.git 2/objects/79/bea435d40ab531c1aaf6be0432c6a5b7aaed21 +0 -0
  197. package/skills/cm-content-factory/.git 2/objects/7e/5ebd79251c2f14e4aceb86c74b6b6daae6b500 +0 -0
  198. package/skills/cm-content-factory/.git 2/objects/81/98a822a60178d6d5023ddb3e222cddf048742e +0 -0
  199. package/skills/cm-content-factory/.git 2/objects/86/0a0e1943dfe53411d2e499a1f16f46a96ef758 +0 -0
  200. package/skills/cm-content-factory/.git 2/objects/86/971fb55fdc081fdbae52376f0f13e57a4e9b04 +0 -0
  201. package/skills/cm-content-factory/.git 2/objects/88/b89dd609a0a03f8d4fe8bfde20d5b8fc1d326d +0 -0
  202. package/skills/cm-content-factory/.git 2/objects/90/8737edb6b7809e32cc01590b4e08ba42a9d40d +0 -0
  203. package/skills/cm-content-factory/.git 2/objects/93/d5a8a9a7d4fb7f11491cb596a6880528725118 +0 -0
  204. package/skills/cm-content-factory/.git 2/objects/98/46a2ab81d0c3b3eb00ef88fc56989aa7e9f316 +0 -0
  205. package/skills/cm-content-factory/.git 2/objects/9b/d8dd1e49cf274eaf9c555f3ab39dce7af5715e +0 -0
  206. package/skills/cm-content-factory/.git 2/objects/a1/13329fb0cec96ae78b222d33a24c3b5bc7fa1f +0 -0
  207. package/skills/cm-content-factory/.git 2/objects/a9/e6effe626e8a3aea3a8fc3364b492191c6e7d0 +0 -0
  208. package/skills/cm-content-factory/.git 2/objects/ad/6de7e48d9782cca9353d1ff0aa1aab7fe1df85 +0 -0
  209. package/skills/cm-content-factory/.git 2/objects/af/54ae316f771ff692e299ffcd8bf2f06b413b59 +0 -0
  210. package/skills/cm-content-factory/.git 2/objects/b0/4cb8b0b00dad633e731c1472161419e738d674 +0 -0
  211. package/skills/cm-content-factory/.git 2/objects/b3/094abb0b9ed46419b269e4a4e36a459690e3b0 +0 -0
  212. package/skills/cm-content-factory/.git 2/objects/b9/435c5d4baac2cfc5c83009ddd27b46b60db5f1 +0 -0
  213. package/skills/cm-content-factory/.git 2/objects/ba/5da17dbaec5ec2dcfdfd126aead518d1171d5c +0 -0
  214. package/skills/cm-content-factory/.git 2/objects/c0/bf58703aa258ba5dd63083bebaec8f223d844c +0 -0
  215. package/skills/cm-content-factory/.git 2/objects/c4/701a34edf1fc1bad58ccc57bd03f9426acb59a +0 -0
  216. package/skills/cm-content-factory/.git 2/objects/c7/5ccce9a4e5cc74d9b3174550cf6d993ca43638 +0 -0
  217. package/skills/cm-content-factory/.git 2/objects/c7/710d59b5a35b0f1f0a0399386643a0bd94c929 +0 -0
  218. package/skills/cm-content-factory/.git 2/objects/d1/fe58237112e953e5fec52da22cf38e08be3df9 +0 -5
  219. package/skills/cm-content-factory/.git 2/objects/d2/2bbe9fd2f74c95bc5583e803f5e435f1e2cd86 +0 -0
  220. package/skills/cm-content-factory/.git 2/objects/d7/e72852ea2bff74581dbf247d400120086229f4 +0 -0
  221. package/skills/cm-content-factory/.git 2/objects/d8/d4c3b5553e4fd72807e1d4b49ef07d9ef3ac35 +0 -0
  222. package/skills/cm-content-factory/.git 2/objects/dc/75050c2876f6a02ae2a53a3c886f395b622977 +0 -0
  223. package/skills/cm-content-factory/.git 2/objects/ee/e8546f95acec500187c08a28a8b9ee02db0dec +0 -0
  224. package/skills/cm-content-factory/.git 2/objects/ef/263c059208b416c2146434f10cb2b9fabcba16 +0 -0
  225. package/skills/cm-content-factory/.git 2/objects/f3/ae597e84d9a59b88acd21c99bde2eaf686d785 +0 -0
  226. package/skills/cm-content-factory/.git 2/objects/f3/f6f5673c821d3d8e76fa267a9e882e7a5387ea +0 -0
  227. package/skills/cm-content-factory/.git 2/objects/f9/6e6d0ad02624dd11d5848594d056caef7a5e8b +0 -0
  228. package/skills/cm-content-factory/.git 2/objects/ff/278988fc1edf0db3abcf18de795f4cc0b4f3e1 +0 -0
  229. package/skills/cm-content-factory/.git 2/refs/heads/main +0 -1
  230. package/skills/cm-content-factory/.git 2/refs/remotes/origin/main +0 -1
  231. package/skills/cm-content-factory/.pytest_cache 2/v/cache/nodeids +0 -76
  232. package/skills/cm-content-factory/.pytest_cache 2/v/cache/stepwise +0 -1
  233. package/skills/cm-content-factory/AGENTS.md +0 -61
  234. package/skills/cm-content-factory/CLAUDE.md +0 -63
  235. package/skills/cm-content-factory/CURSOR.md +0 -43
  236. package/skills/cm-content-factory/Content Factory.zip +0 -0
  237. package/skills/cm-content-factory/SKILL.md +0 -416
  238. package/skills/cm-content-factory/cf +0 -313
  239. package/skills/cm-content-factory/config.schema.json +0 -397
  240. package/skills/cm-content-factory/dashboard/app.js +0 -556
  241. package/skills/cm-content-factory/dashboard/index.html +0 -397
  242. package/skills/cm-content-factory/dashboard/style.css +0 -1211
  243. package/skills/cm-content-factory/examples/01-real-estate.config.json +0 -146
  244. package/skills/cm-content-factory/examples/02-personal-finance.config.json +0 -146
  245. package/skills/cm-content-factory/examples/03-health-wellness.config.json +0 -147
  246. package/skills/cm-content-factory/examples/04-saas-software.config.json +0 -147
  247. package/skills/cm-content-factory/examples/05-legal-services.config.json +0 -147
  248. package/skills/cm-content-factory/examples/06-insurance.config.json +0 -146
  249. package/skills/cm-content-factory/examples/07-ecommerce-dropship.config.json +0 -146
  250. package/skills/cm-content-factory/examples/08-online-education.config.json +0 -147
  251. package/skills/cm-content-factory/examples/09-crypto-defi.config.json +0 -147
  252. package/skills/cm-content-factory/examples/10-beauty-skincare.config.json +0 -147
  253. package/skills/cm-content-factory/examples/11-home-services.config.json +0 -146
  254. package/skills/cm-content-factory/examples/12-dental-clinic.config.json +0 -147
  255. package/skills/cm-content-factory/examples/13-pet-care.config.json +0 -147
  256. package/skills/cm-content-factory/examples/14-travel-hospitality.config.json +0 -147
  257. package/skills/cm-content-factory/examples/15-ai-automation.config.json +0 -147
  258. package/skills/cm-content-factory/examples/16-wedding-events.config.json +0 -147
  259. package/skills/cm-content-factory/examples/17-fitness-coaching.config.json +0 -148
  260. package/skills/cm-content-factory/examples/18-cybersecurity.config.json +0 -147
  261. package/skills/cm-content-factory/examples/19-food-restaurant.config.json +0 -148
  262. package/skills/cm-content-factory/examples/20-solar-energy.config.json +0 -147
  263. package/skills/cm-content-factory/examples/fitness-blog.config.json +0 -116
  264. package/skills/cm-content-factory/examples/tech-blog.config.json +0 -107
  265. package/skills/cm-content-factory/extensions/EXTENSION_GUIDE.md +0 -72
  266. package/skills/cm-content-factory/extensions/hooks.py +0 -126
  267. package/skills/cm-content-factory/extensions/openclaw_adapter.py +0 -132
  268. package/skills/cm-content-factory/landing/docs/content/changelog.md +0 -36
  269. package/skills/cm-content-factory/landing/docs/content/deployment.md +0 -46
  270. package/skills/cm-content-factory/landing/docs/content/execution-flow.md +0 -67
  271. package/skills/cm-content-factory/landing/docs/content/memory-system.md +0 -38
  272. package/skills/cm-content-factory/landing/docs/content/openspace.md +0 -27
  273. package/skills/cm-content-factory/landing/docs/content/use-cases.md +0 -26
  274. package/skills/cm-content-factory/landing/docs/content/v5-intro.md +0 -28
  275. package/skills/cm-content-factory/landing/docs/index.html +0 -240
  276. package/skills/cm-content-factory/landing/index.html +0 -680
  277. package/skills/cm-content-factory/landing/script.js +0 -143
  278. package/skills/cm-content-factory/landing/style.css +0 -1216
  279. package/skills/cm-content-factory/landing/translations.js +0 -508
  280. package/skills/cm-content-factory/logs/events.jsonl +0 -11
  281. package/skills/cm-content-factory/profiles/_template.profile.json +0 -231
  282. package/skills/cm-content-factory/profiles/finance.profile.json +0 -278
  283. package/skills/cm-content-factory/profiles/legal.profile.json +0 -263
  284. package/skills/cm-content-factory/profiles/medical-research.profile.json +0 -321
  285. package/skills/cm-content-factory/profiles/technology.profile.json +0 -275
  286. package/skills/cm-content-factory/scripts/agent_dispatcher.py +0 -266
  287. package/skills/cm-content-factory/scripts/audit.py +0 -106
  288. package/skills/cm-content-factory/scripts/dashboard_server.py +0 -225
  289. package/skills/cm-content-factory/scripts/deploy.py +0 -146
  290. package/skills/cm-content-factory/scripts/extract.py +0 -132
  291. package/skills/cm-content-factory/scripts/landing_generator.py +0 -459
  292. package/skills/cm-content-factory/scripts/memory.py +0 -521
  293. package/skills/cm-content-factory/scripts/monetize.py +0 -239
  294. package/skills/cm-content-factory/scripts/pipeline.py +0 -357
  295. package/skills/cm-content-factory/scripts/plan.py +0 -163
  296. package/skills/cm-content-factory/scripts/publish.py +0 -145
  297. package/skills/cm-content-factory/scripts/research.py +0 -337
  298. package/skills/cm-content-factory/scripts/scaffold.py +0 -464
  299. package/skills/cm-content-factory/scripts/scoreboard.py +0 -336
  300. package/skills/cm-content-factory/scripts/seo.py +0 -90
  301. package/skills/cm-content-factory/scripts/state_manager.py +0 -320
  302. package/skills/cm-content-factory/scripts/token_manager.py +0 -268
  303. package/skills/cm-content-factory/scripts/validate.py +0 -221
  304. package/skills/cm-content-factory/scripts/wizard.py +0 -329
  305. package/skills/cm-content-factory/scripts/write.py +0 -93
  306. package/skills/cm-content-factory/sites/docs-site/src/assets/houston.webp +0 -0
  307. package/skills/cm-content-factory/sites/docs-site/src/content/docs/architecture.md +0 -90
  308. package/skills/cm-content-factory/sites/docs-site/src/content/docs/data-flow.md +0 -54
  309. package/skills/cm-content-factory/sites/docs-site/src/content/docs/deployment.md +0 -38
  310. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/index.md +0 -65
  311. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/lc-content-lifecycle.md +0 -48
  312. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/seq-write-mode.md +0 -39
  313. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/uj-first-batch.md +0 -42
  314. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/wf-content-pipeline.md +0 -51
  315. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/wf-learning-cycle.md +0 -52
  316. package/skills/cm-content-factory/sites/docs-site/src/content/docs/getting-started/configuration.md +0 -86
  317. package/skills/cm-content-factory/sites/docs-site/src/content/docs/getting-started/installation.md +0 -80
  318. package/skills/cm-content-factory/sites/docs-site/src/content/docs/getting-started/intro.md +0 -58
  319. package/skills/cm-content-factory/sites/docs-site/src/content/docs/index.md +0 -102
  320. package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/index.md +0 -45
  321. package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/optimize-seo.md +0 -29
  322. package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/scale-content-production.md +0 -55
  323. package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/standardize-quality.md +0 -29
  324. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/buyer-cmo-huong.md +0 -41
  325. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/buyer-content-lead-khoa.md +0 -40
  326. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/index.md +0 -56
  327. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/user-content-manager-lan.md +0 -46
  328. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/user-seo-minh.md +0 -45
  329. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/user-writer-tu.md +0 -45
  330. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/content-pipeline.md +0 -108
  331. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/index.md +0 -22
  332. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/memory-system.md +0 -52
  333. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/seo-optimization.md +0 -58
  334. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/troubleshooting-guide.md +0 -92
  335. package/skills/cm-content-factory/sites/docs-site/src/styles/custom.css +0 -575
  336. package/skills/cm-content-factory/tests/conftest.py +0 -66
  337. package/skills/cm-content-factory/tests/test_agent_dispatcher.py +0 -125
  338. package/skills/cm-content-factory/tests/test_memory.py +0 -128
  339. package/skills/cm-content-factory/tests/test_pipeline.py +0 -107
  340. package/skills/cm-content-factory/tests/test_research.py +0 -56
  341. package/skills/cm-content-factory/tests/test_state_manager.py +0 -131
  342. package/skills/cm-content-factory/tests/test_token_manager.py +0 -110
  343. package/skills/cm-content-factory/tests/test_wizard.py +0 -121
  344. package/skills/cm-cro-methodology/SKILL.md +0 -290
  345. package/skills/cm-cro-methodology/references/COPYWRITING.md +0 -178
  346. package/skills/cm-cro-methodology/references/OBJECTIONS.md +0 -135
  347. package/skills/cm-cro-methodology/references/PERSUASION.md +0 -158
  348. package/skills/cm-cro-methodology/references/RESEARCH.md +0 -220
  349. package/skills/cm-cro-methodology/references/funnel-analysis.md +0 -365
  350. package/skills/cm-cro-methodology/references/testing-methodology.md +0 -330
  351. package/skills/cm-google-form/SKILL.md +0 -266
  352. package/skills/cm-google-form/templates/apps-script.js +0 -55
  353. package/skills/cm-google-form/templates/form-markup.html +0 -110
  354. package/skills/cm-google-form/templates/form-submit.js +0 -201
  355. package/skills/cm-google-form/templates/toast.css +0 -152
  356. package/skills/cm-growth-hacking/SKILL.md +0 -282
  357. package/skills/cm-growth-hacking/bottom-sheet-engine.md +0 -261
  358. package/skills/cm-growth-hacking/calendar-integration.md +0 -264
  359. package/skills/cm-growth-hacking/references/engagement-patterns.md +0 -346
  360. package/skills/cm-growth-hacking/templates/bottom-sheet.css +0 -528
  361. package/skills/cm-growth-hacking/templates/bottom-sheet.js +0 -269
  362. package/skills/cm-growth-hacking/templates/calendar-cta.js +0 -213
  363. package/skills/cm-growth-hacking/templates/tracking-events.js +0 -211
  364. package/skills/cm-growth-hacking/templates/trigger-manager.js +0 -254
  365. package/skills/cm-growth-hacking/tracking-events.md +0 -246
  366. package/skills/cm-growth-hacking/trigger-system.md +0 -342
  367. package/skills/cm-jtbd/SKILL.md +0 -98
  368. package/skills/cm-notebooklm/SKILL.md +0 -156
  369. package/skills/cm-notebooklm/references/command_reference.md +0 -94
  370. package/skills/cm-notebooklm/references/workflows.md +0 -60
  371. package/skills/cm-notebooklm/resources/knowledge_sources.md +0 -106
  372. package/skills/cm-notebooklm/scripts/brain-sync.sh +0 -453
  373. package/skills/cm-notebooklm/scripts/graduate_wisdom.py +0 -101
  374. package/skills/cm-readit/SKILL.md +0 -289
  375. package/skills/cm-readit/audio-player.md +0 -206
  376. package/skills/cm-readit/examples/blog-reader.js +0 -352
  377. package/skills/cm-readit/examples/voice-cro.js +0 -390
  378. package/skills/cm-readit/tts-engine.md +0 -262
  379. package/skills/cm-readit/ui-patterns.md +0 -362
  380. package/skills/cm-readit/voice-cro.md +0 -223
@@ -1,239 +0,0 @@
1
- #!/usr/bin/env python3
2
- """
3
- Monetization Scorer โ€” Score content by commercial potential.
4
-
5
- Analyzes articles and topics for monetization potential:
6
- - Commercial intent score
7
- - Conversion potential
8
- - CTA recommendations
9
- - Revenue-priority topic ranking
10
-
11
- Usage:
12
- python3 monetize.py --config content-factory.config.json --score-all
13
- python3 monetize.py --config content-factory.config.json --score article.md
14
- python3 monetize.py --config content-factory.config.json --priority-queue
15
- """
16
-
17
- import json
18
- import sys
19
- import re
20
- import argparse
21
- from pathlib import Path
22
- from datetime import datetime
23
-
24
-
25
- # Commercial intent signals
26
- COMMERCIAL_KEYWORDS_VI = [
27
- "giรก", "bao nhiรชu", "mua", "ฤ‘แบทt lแป‹ch", "dแป‹ch vแปฅ", "chi phรญ",
28
- "khuyแบฟn mรฃi", "ฦฐu ฤ‘รฃi", "trแป‹ liแป‡u", "phรกc ฤ‘แป“", "liแป‡u trรฌnh"
29
- ]
30
-
31
- COMMERCIAL_KEYWORDS_EN = [
32
- "price", "cost", "buy", "book", "service", "deal",
33
- "discount", "treatment", "program", "plan", "subscription"
34
- ]
35
-
36
- TRANSACTIONAL_CATEGORIES = ["phac-do", "chan-dung", "reviews", "programs"]
37
-
38
- CTA_TEMPLATES = {
39
- "vi": {
40
- "soft": "๐Ÿ‘‰ Tรฌm hiแปƒu thรชm vแป dแป‹ch vแปฅ {service} tแบกi {brand}",
41
- "medium": "๐Ÿ“ž ฤแบทt lแป‹ch tฦฐ vแบฅn miแป…n phรญ: {hotline}",
42
- "hard": "๐ŸŽ ฦฏu ฤ‘รฃi ฤ‘แบทc biแป‡t: Giแบฃm 20% cho lแบงn ฤ‘แบงu trแบฃi nghiแป‡m. ฤแบทt lแป‹ch ngay!",
43
- "educational": "๐Ÿ“š Xem thรชm bร i viแบฟt vแป {topic} tแบกi {brand}"
44
- },
45
- "en": {
46
- "soft": "๐Ÿ‘‰ Learn more about {service} at {brand}",
47
- "medium": "๐Ÿ“ž Book a free consultation: {hotline}",
48
- "hard": "๐ŸŽ Special offer: 20% off your first session. Book now!",
49
- "educational": "๐Ÿ“š Read more about {topic} at {brand}"
50
- }
51
- }
52
-
53
-
54
- class MonetizationScorer:
55
- """Score content by commercial and conversion potential."""
56
-
57
- def __init__(self, config_path: str):
58
- with open(config_path, "r", encoding="utf-8") as f:
59
- self.config = json.load(f)
60
-
61
- self.project_root = Path(config_path).resolve().parent
62
- self.language = self.config.get("brand", {}).get("language", "vi")
63
-
64
- def score_article(self, filepath: Path) -> dict:
65
- """Score a single article for monetization potential."""
66
- with open(filepath, "r", encoding="utf-8") as f:
67
- content = f.read()
68
-
69
- # Parse frontmatter
70
- fm = self._parse_frontmatter(content)
71
- body = self._get_body(content)
72
-
73
- score = {
74
- "file": filepath.name,
75
- "slug": filepath.stem,
76
- "category": fm.get("category", "unknown"),
77
- "scores": {
78
- "commercial_intent": 0,
79
- "conversion_potential": 0,
80
- "topic_demand": 0,
81
- "content_quality": 0
82
- },
83
- "total": 0,
84
- "cta_recommendation": "",
85
- "monetization_notes": []
86
- }
87
-
88
- # 1. Commercial intent (0-30)
89
- keywords = COMMERCIAL_KEYWORDS_VI if self.language == "vi" else COMMERCIAL_KEYWORDS_EN
90
- keyword_hits = sum(1 for kw in keywords if kw.lower() in body.lower())
91
- score["scores"]["commercial_intent"] = min(30, keyword_hits * 5)
92
-
93
- # Transactional category bonus
94
- if fm.get("category", "") in TRANSACTIONAL_CATEGORIES:
95
- score["scores"]["commercial_intent"] = min(30, score["scores"]["commercial_intent"] + 10)
96
-
97
- # 2. Conversion potential (0-30)
98
- has_cta = bool(re.search(r'(ฤ‘แบทt lแป‹ch|tฦฐ vแบฅn|liรชn hแป‡|book|contact|call)', body, re.IGNORECASE))
99
- has_pricing = bool(re.search(r'(giรก|chi phรญ|price|cost|\d+\.?\d*\s*(vnฤ‘|ฤ‘|\$|usd))', body, re.IGNORECASE))
100
- has_faq = bool(re.search(r'(cรขu hแปi|faq|q&a)', body, re.IGNORECASE))
101
-
102
- score["scores"]["conversion_potential"] = (
103
- (10 if has_cta else 0) +
104
- (10 if has_pricing else 0) +
105
- (10 if has_faq else 0)
106
- )
107
-
108
- # 3. Topic demand (0-20) โ€” based on title keyword
109
- title = fm.get("title", filepath.stem)
110
- high_demand = ["ฤ‘au", "mแบฅt ngแปง", "stress", "giแบฃm cรขn", "ฤ‘au lฦฐng", "cแป• vai gรกy",
111
- "pain", "sleep", "weight", "back", "neck"]
112
- demand_hits = sum(1 for kw in high_demand if kw.lower() in title.lower())
113
- score["scores"]["topic_demand"] = min(20, demand_hits * 10)
114
-
115
- # 4. Content quality signals (0-20)
116
- words = len(body.split())
117
- h2_count = len(re.findall(r'^## ', body, re.MULTILINE))
118
- score["scores"]["content_quality"] = (
119
- (5 if words >= 500 else 0) +
120
- (5 if words >= 800 else 0) +
121
- (5 if h2_count >= 3 else 0) +
122
- (5 if has_faq else 0)
123
- )
124
-
125
- # Total
126
- score["total"] = sum(score["scores"].values())
127
-
128
- # CTA recommendation
129
- templates = CTA_TEMPLATES.get(self.language, CTA_TEMPLATES["vi"])
130
- if score["total"] >= 60:
131
- score["cta_recommendation"] = "hard"
132
- elif score["total"] >= 35:
133
- score["cta_recommendation"] = "medium"
134
- elif score["scores"]["commercial_intent"] > 0:
135
- score["cta_recommendation"] = "soft"
136
- else:
137
- score["cta_recommendation"] = "educational"
138
-
139
- score["suggested_cta"] = templates.get(score["cta_recommendation"], "")
140
-
141
- # Notes
142
- if not has_cta:
143
- score["monetization_notes"].append("Missing CTA โ€” add a booking/contact link")
144
- if score["scores"]["commercial_intent"] == 0:
145
- score["monetization_notes"].append("Low commercial intent โ€” add service mentions")
146
- if score["total"] >= 50:
147
- score["monetization_notes"].append("๐Ÿ”ฅ High monetization potential!")
148
-
149
- return score
150
-
151
- def score_all(self) -> list:
152
- """Score all articles for monetization."""
153
- content_dir = self.project_root / self.config["output"]["content_dir"]
154
- if not content_dir.exists():
155
- return []
156
-
157
- articles = sorted(content_dir.glob("*.md"))
158
- scores = [self.score_article(a) for a in articles]
159
- return sorted(scores, key=lambda s: s["total"], reverse=True)
160
-
161
- def priority_queue(self) -> list:
162
- """Generate monetization-priority writing queue."""
163
- scores = self.score_all()
164
-
165
- # Identify high-potential articles needing improvement
166
- needs_work = [s for s in scores if s["total"] >= 30 and s["monetization_notes"]]
167
- already_good = [s for s in scores if s["total"] >= 60]
168
- low_potential = [s for s in scores if s["total"] < 30]
169
-
170
- return {
171
- "hot": [s["slug"] for s in already_good[:10]],
172
- "optimize": [{"slug": s["slug"], "notes": s["monetization_notes"]} for s in needs_work[:10]],
173
- "low_priority": len(low_potential),
174
- "avg_score": sum(s["total"] for s in scores) / max(len(scores), 1)
175
- }
176
-
177
- def _parse_frontmatter(self, content: str) -> dict:
178
- match = re.match(r'^---\s*\n(.+?)\n---', content, re.DOTALL)
179
- if not match:
180
- return {}
181
- fm = {}
182
- for line in match.group(1).split('\n'):
183
- if ':' in line:
184
- key, _, val = line.partition(':')
185
- fm[key.strip()] = val.strip().strip('"').strip("'")
186
- return fm
187
-
188
- def _get_body(self, content: str) -> str:
189
- match = re.match(r'^---\s*\n.+?\n---\s*\n(.*)', content, re.DOTALL)
190
- return match.group(1) if match else content
191
-
192
-
193
- def main():
194
- parser = argparse.ArgumentParser(description="Monetization Scorer")
195
- parser.add_argument("--config", required=True)
196
- parser.add_argument("--score", help="Score single article")
197
- parser.add_argument("--score-all", action="store_true")
198
- parser.add_argument("--priority-queue", action="store_true")
199
- args = parser.parse_args()
200
-
201
- scorer = MonetizationScorer(args.config)
202
-
203
- if args.score:
204
- fp = Path(args.score)
205
- if not fp.exists():
206
- fp = scorer.project_root / scorer.config["output"]["content_dir"] / args.score
207
- result = scorer.score_article(fp)
208
- print(f"\n๐Ÿ’ฐ {result['slug']}: {result['total']}/100")
209
- for k, v in result["scores"].items():
210
- print(f" {k}: {v}")
211
- print(f" CTA: {result['cta_recommendation']}")
212
- for note in result["monetization_notes"]:
213
- print(f" ๐Ÿ“ {note}")
214
-
215
- elif args.score_all:
216
- scores = scorer.score_all()
217
- print(f"\n{'โ•' * 60}")
218
- print(f" ๐Ÿ’ฐ MONETIZATION SCORES ({len(scores)} articles)")
219
- print(f"{'โ•' * 60}")
220
- for s in scores[:20]:
221
- bar = "โ–ˆ" * (s["total"] // 5) + "โ–‘" * (20 - s["total"] // 5)
222
- print(f" {s['total']:3d} {bar} {s['slug'][:40]}")
223
- avg = sum(s["total"] for s in scores) / max(len(scores), 1)
224
- print(f"\n Average: {avg:.0f}/100")
225
-
226
- elif args.priority_queue:
227
- queue = scorer.priority_queue()
228
- print(f"\n๐Ÿ”ฅ HOT (ready to monetize): {len(queue['hot'])}")
229
- for slug in queue["hot"]:
230
- print(f" โ†’ {slug}")
231
- print(f"\nโšก OPTIMIZE (needs CTA/improvements): {len(queue['optimize'])}")
232
- for item in queue["optimize"]:
233
- print(f" โ†’ {item['slug']}: {', '.join(item['notes'][:2])}")
234
- print(f"\n๐Ÿ“‰ Low priority: {queue['low_priority']} articles")
235
- print(f" Average score: {queue['avg_score']:.0f}/100")
236
-
237
-
238
- if __name__ == "__main__":
239
- main()
@@ -1,357 +0,0 @@
1
- #!/usr/bin/env python3
2
- """
3
- Content Factory Pipeline โ€” Master orchestrator for the AI Content Factory.
4
-
5
- Reads content-factory.config.json and executes phases:
6
- extract โ†’ plan โ†’ write โ†’ audit โ†’ seo โ†’ publish
7
-
8
- Usage:
9
- python3 .agents/skills/content-factory/scripts/pipeline.py # Run full pipeline
10
- python3 .agents/skills/content-factory/scripts/pipeline.py --phase extract # Run single phase
11
- python3 .agents/skills/content-factory/scripts/pipeline.py --validate-config # Validate config
12
- python3 .agents/skills/content-factory/scripts/pipeline.py --dry-run # Preview actions
13
- python3 .agents/skills/content-factory/scripts/pipeline.py --test-hooks # Test hook system
14
- python3 .agents/skills/content-factory/scripts/pipeline.py --status # Show pipeline status
15
- """
16
-
17
- import json
18
- import sys
19
- import os
20
- import subprocess
21
- import argparse
22
- import threading
23
- from pathlib import Path
24
- from datetime import datetime
25
-
26
- # State integration
27
- try:
28
- from state_manager import StateManager
29
- HAS_STATE = True
30
- except ImportError:
31
- HAS_STATE = False
32
-
33
- # Resolve paths
34
- SCRIPT_DIR = Path(__file__).resolve().parent
35
- SKILL_DIR = SCRIPT_DIR.parent
36
- PROJECT_ROOT = Path(os.getcwd()).resolve()
37
- CONFIG_FILE = PROJECT_ROOT / "content-factory.config.json"
38
- SCHEMA_FILE = SKILL_DIR / "config.schema.json"
39
-
40
- PHASES = ["extract", "plan", "write", "audit", "seo", "publish"]
41
-
42
-
43
- def load_config() -> dict:
44
- """Load and return the project config."""
45
- if not CONFIG_FILE.exists():
46
- print(f"โŒ Config not found: {CONFIG_FILE}")
47
- print(" Create content-factory.config.json in project root.")
48
- print(f" See examples in: {SKILL_DIR / 'examples/'}")
49
- sys.exit(1)
50
-
51
- with open(CONFIG_FILE, "r", encoding="utf-8") as f:
52
- return json.load(f)
53
-
54
-
55
- def validate_config(config: dict) -> bool:
56
- """Validate config structure (basic validation without jsonschema dep)."""
57
- required_keys = ["niche", "brand", "content", "sources", "output"]
58
- missing = [k for k in required_keys if k not in config]
59
- if missing:
60
- print(f"โŒ Missing required config keys: {', '.join(missing)}")
61
- return False
62
-
63
- if "name" not in config.get("brand", {}):
64
- print("โŒ Missing brand.name in config")
65
- return False
66
-
67
- if "article_types" not in config.get("content", {}):
68
- print("โŒ Missing content.article_types in config")
69
- return False
70
-
71
- if "type" not in config.get("sources", {}) or "path" not in config.get("sources", {}):
72
- print("โŒ Missing sources.type or sources.path in config")
73
- return False
74
-
75
- if "content_dir" not in config.get("output", {}):
76
- print("โŒ Missing output.content_dir in config")
77
- return False
78
-
79
- print("โœ… Config validation passed")
80
- return True
81
-
82
-
83
- def fire_hooks(config: dict, hook_name: str):
84
- """Execute registered hooks for a phase."""
85
- hooks = config.get("extensions", {}).get("hooks", {}).get(hook_name, [])
86
- if not hooks:
87
- return
88
-
89
- print(f" ๐Ÿ”— Running {hook_name} hooks ({len(hooks)})...")
90
- for hook_script in hooks:
91
- hook_path = PROJECT_ROOT / hook_script
92
- if hook_path.exists():
93
- try:
94
- result = subprocess.run(
95
- ["python3", str(hook_path), str(CONFIG_FILE)],
96
- cwd=str(PROJECT_ROOT),
97
- capture_output=True, text=True, timeout=300
98
- )
99
- if result.returncode == 0:
100
- print(f" โœ… {hook_script}")
101
- else:
102
- print(f" โš ๏ธ {hook_script}: {result.stderr[:200]}")
103
- except Exception as e:
104
- print(f" โŒ {hook_script}: {e}")
105
- else:
106
- print(f" โš ๏ธ Hook not found: {hook_script}")
107
-
108
-
109
- def run_phase(phase: str, config: dict, dry_run: bool = False, extra_args: list = None,
110
- state_manager: 'StateManager' = None):
111
- """Run a single pipeline phase."""
112
- script_path = SCRIPT_DIR / f"{phase}.py"
113
- if not script_path.exists():
114
- print(f"โŒ Phase script not found: {script_path}")
115
- if state_manager:
116
- state_manager.update_phase(phase, "failed", error=f"Script not found: {phase}.py")
117
- return False
118
-
119
- # Emit state: phase starting
120
- if state_manager:
121
- state_manager.update_phase(phase, "running", progress=0.0)
122
-
123
- # Fire pre-hook
124
- fire_hooks(config, f"pre_{phase}")
125
-
126
- cmd = ["python3", str(script_path), "--config", str(CONFIG_FILE)]
127
- if dry_run:
128
- cmd.append("--dry-run")
129
- if extra_args:
130
- cmd.extend(extra_args)
131
-
132
- print(f"\n{'โ•' * 50}")
133
- print(f" Phase: {phase.upper()}")
134
- print(f"{'โ•' * 50}")
135
-
136
- if dry_run:
137
- print(f" [DRY RUN] Would execute: {' '.join(cmd)}")
138
- if state_manager:
139
- state_manager.update_phase(phase, "done", progress=1.0)
140
- fire_hooks(config, f"post_{phase}")
141
- return True
142
-
143
- try:
144
- result = subprocess.run(
145
- cmd, cwd=str(PROJECT_ROOT),
146
- timeout=3600 # 1h max per phase
147
- )
148
- success = result.returncode == 0
149
- except subprocess.TimeoutExpired:
150
- print(f" โŒ Phase {phase} timed out (1h limit)")
151
- success = False
152
- except Exception as e:
153
- print(f" โŒ Phase {phase} error: {e}")
154
- success = False
155
-
156
- # Emit state: phase result
157
- if state_manager:
158
- if success:
159
- state_manager.update_phase(phase, "done", progress=1.0)
160
- else:
161
- state_manager.update_phase(phase, "failed", error=f"Phase {phase} failed")
162
-
163
- # Fire post-hook
164
- fire_hooks(config, f"post_{phase}")
165
-
166
- return success
167
-
168
-
169
- def show_status(config: dict):
170
- """Show current pipeline status."""
171
- print(f"\n{'โ•' * 60}")
172
- print(f" ๐Ÿญ Content Factory โ€” Status Report")
173
- print(f"{'โ•' * 60}")
174
- print(f" Niche: {config['niche']}")
175
- print(f" Brand: {config['brand']['name']}")
176
- print(f" Language: {config['brand'].get('language', 'vi')}")
177
- print(f" AI: {config.get('pipeline', {}).get('ai_provider', 'gemini-cli')}")
178
- print(f" Parallel: {config.get('pipeline', {}).get('concurrency', 1)}")
179
- print()
180
-
181
- # Knowledge-base status
182
- kb_dir = PROJECT_ROOT / config["output"].get("knowledge_dir", "knowledge-base/")
183
- if kb_dir.exists():
184
- groups = [d for d in kb_dir.iterdir() if d.is_dir()]
185
- index = kb_dir / "index.json"
186
- disease_count = 0
187
- if index.exists():
188
- with open(index) as f:
189
- idx = json.load(f)
190
- disease_count = idx.get("total_diseases", len(idx.get("groups", [])))
191
- print(f" ๐Ÿ“š Knowledge-base: {len(groups)} groups, {disease_count} entries")
192
- else:
193
- print(f" ๐Ÿ“š Knowledge-base: Not yet created")
194
-
195
- # Topics queue status
196
- queue_dir = PROJECT_ROOT / config["output"].get("queue_dir", "topics-queue/")
197
- if queue_dir.exists():
198
- batches = list(queue_dir.glob("*.json"))
199
- total_topics = 0
200
- for b in batches:
201
- with open(b) as f:
202
- data = json.load(f)
203
- total_topics += len(data.get("topics", []))
204
- print(f" ๐Ÿ“‹ Topics queue: {len(batches)} batches, {total_topics} topics")
205
- else:
206
- print(f" ๐Ÿ“‹ Topics queue: Not yet created")
207
-
208
- # Content status
209
- content_dir = PROJECT_ROOT / config["output"]["content_dir"]
210
- if content_dir.exists():
211
- articles = list(content_dir.glob("*.md"))
212
- print(f" ๐Ÿ“„ Content: {len(articles)} articles")
213
- else:
214
- print(f" ๐Ÿ“„ Content: Not yet created")
215
-
216
- # Extensions
217
- ext = config.get("extensions", {})
218
- hooks_count = sum(len(v) for v in ext.get("hooks", {}).values())
219
- openclaw = "โœ… enabled" if ext.get("openclaw", {}).get("enabled") else "โฌœ disabled"
220
- print(f" ๐Ÿ”— Hooks: {hooks_count} registered")
221
- print(f" ๐Ÿ™ OpenClaw: {openclaw}")
222
-
223
- print(f"\n{'โ•' * 60}\n")
224
-
225
-
226
- def test_hooks(config: dict):
227
- """Test that all registered hooks exist and are executable."""
228
- print("๐Ÿงช Testing hooks...")
229
- hooks = config.get("extensions", {}).get("hooks", {})
230
- all_ok = True
231
- for hook_name, scripts in hooks.items():
232
- for script in scripts:
233
- path = PROJECT_ROOT / script
234
- if path.exists():
235
- print(f" โœ… {hook_name}: {script}")
236
- else:
237
- print(f" โŒ {hook_name}: {script} โ€” NOT FOUND")
238
- all_ok = False
239
- if not hooks or all(not v for v in hooks.values()):
240
- print(" โฌœ No hooks registered")
241
- return all_ok
242
-
243
-
244
- def start_dashboard(port: int = 5050):
245
- """Start dashboard server in background thread."""
246
- dashboard_script = SCRIPT_DIR / "dashboard_server.py"
247
- if not dashboard_script.exists():
248
- print(f" โš ๏ธ Dashboard not found: {dashboard_script}")
249
- return None
250
-
251
- import subprocess as sp
252
- proc = sp.Popen(
253
- ["python3", str(dashboard_script), "--port", str(port)],
254
- cwd=str(PROJECT_ROOT),
255
- )
256
- return proc
257
-
258
-
259
- def main():
260
- parser = argparse.ArgumentParser(
261
- description="Content Factory Pipeline โ€” Master orchestrator"
262
- )
263
- parser.add_argument("--phase", choices=PHASES, help="Run single phase")
264
- parser.add_argument("--validate-config", action="store_true", help="Validate config only")
265
- parser.add_argument("--dry-run", action="store_true", help="Preview without executing")
266
- parser.add_argument("--test-hooks", action="store_true", help="Test hook system")
267
- parser.add_argument("--status", action="store_true", help="Show pipeline status")
268
- parser.add_argument("--from-phase", choices=PHASES, help="Start pipeline from phase")
269
- parser.add_argument("--batch", type=int, default=10, help="Batch size for write phase")
270
- parser.add_argument("--group", help="Filter by group code")
271
- parser.add_argument("--dashboard", action="store_true", help="Auto-start dashboard server")
272
- parser.add_argument("--dashboard-port", type=int, default=5050, help="Dashboard port")
273
- parser.add_argument("--budget", type=float, help="Max budget in USD")
274
- args = parser.parse_args()
275
-
276
- config = load_config()
277
-
278
- if args.validate_config:
279
- sys.exit(0 if validate_config(config) else 1)
280
-
281
- if args.status:
282
- show_status(config)
283
- return
284
-
285
- if args.test_hooks:
286
- sys.exit(0 if test_hooks(config) else 1)
287
-
288
- if not validate_config(config):
289
- sys.exit(1)
290
-
291
- # Initialize state manager
292
- sm = None
293
- if HAS_STATE:
294
- sm = StateManager(str(PROJECT_ROOT))
295
- sm.reset()
296
- sm.log_event("info", f"Pipeline started for {config['brand']['name']}")
297
- if args.budget:
298
- sm.set_budget(args.budget)
299
-
300
- # Start dashboard
301
- dashboard_proc = None
302
- if args.dashboard:
303
- dashboard_proc = start_dashboard(args.dashboard_port)
304
- if dashboard_proc:
305
- print(f" ๐ŸŒ Dashboard: http://localhost:{args.dashboard_port}")
306
-
307
- print(f"\n๐Ÿญ Content Factory Pipeline โ€” {config['brand']['name']}")
308
- print(f" Niche: {config['niche']}")
309
- print(f" Started: {datetime.now().strftime('%Y-%m-%d %H:%M')}")
310
-
311
- # Determine phases to run
312
- if args.phase:
313
- phases = [args.phase]
314
- elif args.from_phase:
315
- start_idx = PHASES.index(args.from_phase)
316
- phases = PHASES[start_idx:]
317
- else:
318
- phases = PHASES
319
-
320
- # Build extra args per phase
321
- extra = {}
322
- if args.batch:
323
- extra["write"] = ["--batch", str(args.batch)]
324
- if args.group:
325
- for p in phases:
326
- extra.setdefault(p, []).extend(["--group", args.group])
327
-
328
- # Execute
329
- results = {}
330
- for phase in phases:
331
- phase_extra = extra.get(phase, [])
332
- success = run_phase(phase, config, args.dry_run, phase_extra, state_manager=sm)
333
- results[phase] = success
334
- if not success and not args.dry_run:
335
- print(f"\nโŒ Pipeline stopped at phase: {phase}")
336
- break
337
-
338
- # Summary
339
- print(f"\n{'โ•' * 50}")
340
- print(f" ๐Ÿ“Š PIPELINE SUMMARY")
341
- print(f"{'โ•' * 50}")
342
- for phase, ok in results.items():
343
- icon = "โœ…" if ok else "โŒ"
344
- print(f" {icon} {phase.upper()}")
345
- print(f"{'โ•' * 50}")
346
- print(f" Finished: {datetime.now().strftime('%Y-%m-%d %H:%M')}\n")
347
-
348
- if sm:
349
- sm.log_event("info", "Pipeline finished")
350
-
351
- # Stop dashboard if we started it
352
- if dashboard_proc:
353
- dashboard_proc.terminate()
354
-
355
-
356
- if __name__ == "__main__":
357
- main()