create-hq 5.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 (310) hide show
  1. package/dist/deps.d.ts +4 -0
  2. package/dist/deps.d.ts.map +1 -0
  3. package/dist/deps.js +65 -0
  4. package/dist/deps.js.map +1 -0
  5. package/dist/git.d.ts +3 -0
  6. package/dist/git.d.ts.map +1 -0
  7. package/dist/git.js +19 -0
  8. package/dist/git.js.map +1 -0
  9. package/dist/index.d.ts +3 -0
  10. package/dist/index.d.ts.map +1 -0
  11. package/dist/index.js +23 -0
  12. package/dist/index.js.map +1 -0
  13. package/dist/scaffold.d.ts +8 -0
  14. package/dist/scaffold.d.ts.map +1 -0
  15. package/dist/scaffold.js +130 -0
  16. package/dist/scaffold.js.map +1 -0
  17. package/dist/ui.d.ts +7 -0
  18. package/dist/ui.d.ts.map +1 -0
  19. package/dist/ui.js +36 -0
  20. package/dist/ui.js.map +1 -0
  21. package/package.json +41 -0
  22. package/template/.claude/CLAUDE.md +202 -0
  23. package/template/.claude/commands/checkpoint.md +127 -0
  24. package/template/.claude/commands/cleanup.md +307 -0
  25. package/template/.claude/commands/execute-task.md +440 -0
  26. package/template/.claude/commands/exit-plan.md +41 -0
  27. package/template/.claude/commands/handoff.md +97 -0
  28. package/template/.claude/commands/learn.md +218 -0
  29. package/template/.claude/commands/metrics.md +118 -0
  30. package/template/.claude/commands/newworker.md +162 -0
  31. package/template/.claude/commands/nexttask.md +67 -0
  32. package/template/.claude/commands/prd.md +238 -0
  33. package/template/.claude/commands/reanchor.md +51 -0
  34. package/template/.claude/commands/remember.md +126 -0
  35. package/template/.claude/commands/run-project.md +348 -0
  36. package/template/.claude/commands/run.md +110 -0
  37. package/template/.claude/commands/search-reindex.md +62 -0
  38. package/template/.claude/commands/search.md +100 -0
  39. package/template/.claude/commands/setup.md +381 -0
  40. package/template/.claude/scripts/pure-ralph-loop.ps1 +312 -0
  41. package/template/.claude/scripts/pure-ralph-loop.sh +859 -0
  42. package/template/CHANGELOG.md +220 -0
  43. package/template/LICENSE +21 -0
  44. package/template/MIGRATION.md +259 -0
  45. package/template/README.md +368 -0
  46. package/template/data/journal/.gitkeep +0 -0
  47. package/template/docs/images/ascii-banner-options.md +122 -0
  48. package/template/docs/images/hq-banner.svg +105 -0
  49. package/template/knowledge/Ralph/01-overview.md +71 -0
  50. package/template/knowledge/Ralph/02-core-concepts.md +114 -0
  51. package/template/knowledge/Ralph/03-how-ralph-works.md +184 -0
  52. package/template/knowledge/Ralph/04-back-pressure.md +222 -0
  53. package/template/knowledge/Ralph/05-specifications.md +210 -0
  54. package/template/knowledge/Ralph/06-agents-md.md +222 -0
  55. package/template/knowledge/Ralph/07-implementation.md +316 -0
  56. package/template/knowledge/Ralph/08-economics.md +182 -0
  57. package/template/knowledge/Ralph/09-resources.md +145 -0
  58. package/template/knowledge/Ralph/10-claude-code-workflow.md +212 -0
  59. package/template/knowledge/Ralph/11-team-training-guide.md +383 -0
  60. package/template/knowledge/Ralph/README.md +40 -0
  61. package/template/knowledge/ai-security-framework/CONTRIBUTING.md +139 -0
  62. package/template/knowledge/ai-security-framework/GLOSSARY.md +176 -0
  63. package/template/knowledge/ai-security-framework/LICENSE +21 -0
  64. package/template/knowledge/ai-security-framework/QUICK-START.md +172 -0
  65. package/template/knowledge/ai-security-framework/README.md +232 -0
  66. package/template/knowledge/ai-security-framework/checklists/browser-security.md +301 -0
  67. package/template/knowledge/ai-security-framework/checklists/credential-isolation.md +322 -0
  68. package/template/knowledge/ai-security-framework/checklists/incident-response.md +288 -0
  69. package/template/knowledge/ai-security-framework/checklists/pre-flight.md +249 -0
  70. package/template/knowledge/ai-security-framework/checklists/weekly-audit.md +159 -0
  71. package/template/knowledge/ai-security-framework/configs/audit-logging.md +372 -0
  72. package/template/knowledge/ai-security-framework/configs/kill-switches.md +354 -0
  73. package/template/knowledge/ai-security-framework/docs/01-core-principles.md +256 -0
  74. package/template/knowledge/ai-security-framework/docs/02-threat-landscape.md +326 -0
  75. package/template/knowledge/ai-security-framework/docs/03-security-posture.md +250 -0
  76. package/template/knowledge/ai-security-framework/templates/agents-security.md +233 -0
  77. package/template/knowledge/design-styles/README.md +42 -0
  78. package/template/knowledge/design-styles/american-industrial.md +136 -0
  79. package/template/knowledge/design-styles/ethereal-abstract.md +133 -0
  80. package/template/knowledge/design-styles/liminal-portal.md +111 -0
  81. package/template/knowledge/design-styles/swipes/american-industrial/G-3m4YPW0AADdu2.jpeg +0 -0
  82. package/template/knowledge/design-styles/swipes/american-industrial/G-JJlt5WwAABK3K.png +0 -0
  83. package/template/knowledge/design-styles/swipes/american-industrial/G-JJmj5W0AEbJ-7.png +0 -0
  84. package/template/knowledge/design-styles/swipes/american-industrial/G59fgNuXkAAKLJQ (1).jpeg +0 -0
  85. package/template/knowledge/design-styles/swipes/american-industrial/G59fgNuXkAAKLJQ.jpeg +0 -0
  86. package/template/knowledge/design-styles/swipes/american-industrial/G7fVkn3WEAAM-ST.jpeg +0 -0
  87. package/template/knowledge/design-styles/swipes/american-industrial/G8ECO5JWEAIksyn.png +0 -0
  88. package/template/knowledge/design-styles/swipes/american-industrial/G9-3GQSWoAA8eqZ.png +0 -0
  89. package/template/knowledge/design-styles/swipes/american-industrial/G9xEOqrXkAEZRcs.png +0 -0
  90. package/template/knowledge/design-styles/swipes/american-industrial/G_MVeJrXQAA8sx4.jpeg +0 -0
  91. package/template/knowledge/design-styles/swipes/american-industrial/G_RSkmGXkAAgAVZ.png +0 -0
  92. package/template/knowledge/design-styles/swipes/american-industrial/README.md +31 -0
  93. package/template/knowledge/design-styles/swipes/american-industrial/qyqtg7Dq.png +0 -0
  94. package/template/knowledge/dev-team/README.md +35 -0
  95. package/template/knowledge/dev-team/patterns/README.md +34 -0
  96. package/template/knowledge/dev-team/patterns/frontend/react-best-practices.md +178 -0
  97. package/template/knowledge/dev-team/troubleshooting/README.md +31 -0
  98. package/template/knowledge/dev-team/workflows/README.md +49 -0
  99. package/template/knowledge/hq/checkpoint-schema.json +51 -0
  100. package/template/knowledge/hq/index-md-spec.md +74 -0
  101. package/template/knowledge/hq/thread-schema.md +153 -0
  102. package/template/knowledge/hq-core/checkpoint-schema.json +51 -0
  103. package/template/knowledge/hq-core/index-md-spec.md +74 -0
  104. package/template/knowledge/hq-core/thread-schema.md +153 -0
  105. package/template/knowledge/loom/README.md +51 -0
  106. package/template/knowledge/loom/architecture.md +125 -0
  107. package/template/knowledge/loom/code-style.md +169 -0
  108. package/template/knowledge/loom/llm-proxy.md +132 -0
  109. package/template/knowledge/loom/state-machine.md +131 -0
  110. package/template/knowledge/loom/thread-system.md +117 -0
  111. package/template/knowledge/loom/tools.md +94 -0
  112. package/template/knowledge/loom/weaver.md +96 -0
  113. package/template/knowledge/loom/web-frontend.md +131 -0
  114. package/template/knowledge/projects/README.md +72 -0
  115. package/template/knowledge/projects/templates/README.template.md +28 -0
  116. package/template/knowledge/workers/README.md +195 -0
  117. package/template/knowledge/workers/ralph-loop-pattern.md +157 -0
  118. package/template/knowledge/workers/skill-schema.md +182 -0
  119. package/template/knowledge/workers/state-machine.md +102 -0
  120. package/template/knowledge/workers/templates/base-worker.yaml +73 -0
  121. package/template/knowledge/workers/templates/code-worker.yaml +85 -0
  122. package/template/knowledge/workers/templates/skill.yaml +49 -0
  123. package/template/knowledge/workers/templates/social-worker.yaml +70 -0
  124. package/template/modules/examples/full-manifest.yaml +92 -0
  125. package/template/modules/examples/minimal.yaml +14 -0
  126. package/template/modules/modules.yaml +59 -0
  127. package/template/projects/.gitkeep +0 -0
  128. package/template/projects/incorporate-workers-into-pure-ralph/prd.json +88 -0
  129. package/template/projects/pure-ralph-branch-isolation/README.md +114 -0
  130. package/template/projects/pure-ralph-branch-isolation/prd.json +123 -0
  131. package/template/projects/purist-ralph-loop/README.md +148 -0
  132. package/template/projects/purist-ralph-loop/prd.json +135 -0
  133. package/template/projects/ralph-test/prd.json +50 -0
  134. package/template/prompts/pure-ralph-base.md +551 -0
  135. package/template/settings/.gitkeep +0 -0
  136. package/template/settings/pure-ralph.json +42 -0
  137. package/template/social-content/drafts/INDEX.md +21 -0
  138. package/template/social-content/drafts/linkedin/.gitkeep +1 -0
  139. package/template/social-content/drafts/x/.gitkeep +1 -0
  140. package/template/social-content/images/.gitkeep +1 -0
  141. package/template/starter-projects/code-worker/README.md +97 -0
  142. package/template/starter-projects/code-worker/prd.json +45 -0
  143. package/template/starter-projects/personal-assistant/README.md +42 -0
  144. package/template/starter-projects/personal-assistant/prd.json +43 -0
  145. package/template/starter-projects/social-media/README.md +60 -0
  146. package/template/starter-projects/social-media/prd.json +43 -0
  147. package/template/workers/content-brand/README.md +59 -0
  148. package/template/workers/content-brand/skills/messaging-alignment.md +91 -0
  149. package/template/workers/content-brand/skills/tone-check.md +76 -0
  150. package/template/workers/content-brand/skills/voice-analysis.md +68 -0
  151. package/template/workers/content-brand/worker.yaml +81 -0
  152. package/template/workers/content-legal/README.md +80 -0
  153. package/template/workers/content-legal/skills/claim-substantiation.md +150 -0
  154. package/template/workers/content-legal/skills/compliance-scan.md +123 -0
  155. package/template/workers/content-legal/skills/disclaimer-check.md +146 -0
  156. package/template/workers/content-legal/worker.yaml +118 -0
  157. package/template/workers/content-product/README.md +77 -0
  158. package/template/workers/content-product/skills/claim-verification.md +96 -0
  159. package/template/workers/content-product/skills/feature-accuracy.md +117 -0
  160. package/template/workers/content-product/skills/stats-check.md +128 -0
  161. package/template/workers/content-product/worker.yaml +97 -0
  162. package/template/workers/content-sales/README.md +70 -0
  163. package/template/workers/content-sales/skills/conversion-analysis.md +96 -0
  164. package/template/workers/content-sales/skills/cta-audit.md +107 -0
  165. package/template/workers/content-sales/skills/value-prop-check.md +114 -0
  166. package/template/workers/content-sales/worker.yaml +93 -0
  167. package/template/workers/content-shared/cli.ts +242 -0
  168. package/template/workers/content-shared/index.ts +234 -0
  169. package/template/workers/content-shared/lib/accuracy-analyzer.ts +661 -0
  170. package/template/workers/content-shared/lib/analyze.ts +370 -0
  171. package/template/workers/content-shared/lib/brand-analyzer.ts +526 -0
  172. package/template/workers/content-shared/lib/cms-integration.ts +446 -0
  173. package/template/workers/content-shared/lib/compliance-analyzer.ts +655 -0
  174. package/template/workers/content-shared/lib/conversion-analyzer.ts +555 -0
  175. package/template/workers/content-shared/lib/github-integration.ts +582 -0
  176. package/template/workers/content-shared/lib/output.ts +373 -0
  177. package/template/workers/content-shared/lib/parser.ts +771 -0
  178. package/template/workers/content-shared/lib/priority.ts +439 -0
  179. package/template/workers/content-shared/lib/recommendations.ts +512 -0
  180. package/template/workers/content-shared/lib/reporter.ts +749 -0
  181. package/template/workers/content-shared/lib/restructure.ts +664 -0
  182. package/template/workers/content-shared/lib/scorer.ts +140 -0
  183. package/template/workers/content-shared/lib/types.ts +227 -0
  184. package/template/workers/content-shared/lib/variants.ts +595 -0
  185. package/template/workers/content-shared/package.json +51 -0
  186. package/template/workers/content-shared/pnpm-lock.yaml +39 -0
  187. package/template/workers/content-shared/test/sample-page.json +115 -0
  188. package/template/workers/content-shared/tsconfig.json +20 -0
  189. package/template/workers/dev-team/README.md +166 -0
  190. package/template/workers/dev-team/_template.yaml +70 -0
  191. package/template/workers/dev-team/architect/package.json +27 -0
  192. package/template/workers/dev-team/architect/skills/api-design.md +89 -0
  193. package/template/workers/dev-team/architect/skills/refactor-plan.md +96 -0
  194. package/template/workers/dev-team/architect/skills/system-design.md +100 -0
  195. package/template/workers/dev-team/architect/src/index.ts +49 -0
  196. package/template/workers/dev-team/architect/src/mcp-server.ts +122 -0
  197. package/template/workers/dev-team/architect/src/skills/api-design.ts +316 -0
  198. package/template/workers/dev-team/architect/src/skills/refactor-plan.ts +264 -0
  199. package/template/workers/dev-team/architect/src/skills/system-design.ts +212 -0
  200. package/template/workers/dev-team/architect/tsconfig.json +19 -0
  201. package/template/workers/dev-team/architect/worker.yaml +128 -0
  202. package/template/workers/dev-team/backend-dev/package-lock.json +1252 -0
  203. package/template/workers/dev-team/backend-dev/package.json +27 -0
  204. package/template/workers/dev-team/backend-dev/skills/implement-endpoint.md +70 -0
  205. package/template/workers/dev-team/backend-dev/skills/implement-service.md +62 -0
  206. package/template/workers/dev-team/backend-dev/src/index.ts +51 -0
  207. package/template/workers/dev-team/backend-dev/src/mcp-server.ts +109 -0
  208. package/template/workers/dev-team/backend-dev/src/skills/implement-endpoint.ts +122 -0
  209. package/template/workers/dev-team/backend-dev/src/skills/implement-service.ts +126 -0
  210. package/template/workers/dev-team/backend-dev/tsconfig.json +19 -0
  211. package/template/workers/dev-team/backend-dev/worker.yaml +128 -0
  212. package/template/workers/dev-team/code-reviewer/package-lock.json +1080 -0
  213. package/template/workers/dev-team/code-reviewer/package.json +24 -0
  214. package/template/workers/dev-team/code-reviewer/skills/merge-to-production.md +61 -0
  215. package/template/workers/dev-team/code-reviewer/skills/merge-to-staging.md +54 -0
  216. package/template/workers/dev-team/code-reviewer/skills/request-changes.md +63 -0
  217. package/template/workers/dev-team/code-reviewer/skills/review-pr.md +77 -0
  218. package/template/workers/dev-team/code-reviewer/src/index.ts +56 -0
  219. package/template/workers/dev-team/code-reviewer/src/mcp-server.ts +101 -0
  220. package/template/workers/dev-team/code-reviewer/tsconfig.json +19 -0
  221. package/template/workers/dev-team/code-reviewer/worker.yaml +90 -0
  222. package/template/workers/dev-team/database-dev/package.json +22 -0
  223. package/template/workers/dev-team/database-dev/skills/create-schema.md +48 -0
  224. package/template/workers/dev-team/database-dev/src/index.ts +50 -0
  225. package/template/workers/dev-team/database-dev/src/mcp-server.ts +76 -0
  226. package/template/workers/dev-team/database-dev/tsconfig.json +18 -0
  227. package/template/workers/dev-team/database-dev/worker.yaml +90 -0
  228. package/template/workers/dev-team/frontend-dev/package.json +22 -0
  229. package/template/workers/dev-team/frontend-dev/skills/create-component.md +26 -0
  230. package/template/workers/dev-team/frontend-dev/src/index.ts +50 -0
  231. package/template/workers/dev-team/frontend-dev/src/mcp-server.ts +77 -0
  232. package/template/workers/dev-team/frontend-dev/tsconfig.json +18 -0
  233. package/template/workers/dev-team/frontend-dev/worker.yaml +132 -0
  234. package/template/workers/dev-team/infra-dev/package.json +24 -0
  235. package/template/workers/dev-team/infra-dev/skills/add-monitoring.md +73 -0
  236. package/template/workers/dev-team/infra-dev/skills/configure-deployment.md +80 -0
  237. package/template/workers/dev-team/infra-dev/skills/create-dockerfile.md +62 -0
  238. package/template/workers/dev-team/infra-dev/skills/setup-cicd.md +63 -0
  239. package/template/workers/dev-team/infra-dev/src/index.ts +55 -0
  240. package/template/workers/dev-team/infra-dev/src/mcp-server.ts +82 -0
  241. package/template/workers/dev-team/infra-dev/tsconfig.json +19 -0
  242. package/template/workers/dev-team/infra-dev/worker.yaml +92 -0
  243. package/template/workers/dev-team/knowledge-curator/package.json +24 -0
  244. package/template/workers/dev-team/knowledge-curator/skills/curate-troubleshooting.md +63 -0
  245. package/template/workers/dev-team/knowledge-curator/skills/process-learnings.md +61 -0
  246. package/template/workers/dev-team/knowledge-curator/skills/sync-documentation.md +76 -0
  247. package/template/workers/dev-team/knowledge-curator/skills/update-patterns.md +63 -0
  248. package/template/workers/dev-team/knowledge-curator/src/index.ts +53 -0
  249. package/template/workers/dev-team/knowledge-curator/src/mcp-server.ts +92 -0
  250. package/template/workers/dev-team/knowledge-curator/tsconfig.json +19 -0
  251. package/template/workers/dev-team/knowledge-curator/worker.yaml +80 -0
  252. package/template/workers/dev-team/motion-designer/package.json +22 -0
  253. package/template/workers/dev-team/motion-designer/skills/add-animation.md +25 -0
  254. package/template/workers/dev-team/motion-designer/skills/generate-image.md +36 -0
  255. package/template/workers/dev-team/motion-designer/src/index.ts +63 -0
  256. package/template/workers/dev-team/motion-designer/src/mcp-server.ts +79 -0
  257. package/template/workers/dev-team/motion-designer/tsconfig.json +18 -0
  258. package/template/workers/dev-team/motion-designer/worker.yaml +84 -0
  259. package/template/workers/dev-team/product-planner/queue.json +4 -0
  260. package/template/workers/dev-team/product-planner/worker.yaml +220 -0
  261. package/template/workers/dev-team/project-manager/package-lock.json +1252 -0
  262. package/template/workers/dev-team/project-manager/package.json +27 -0
  263. package/template/workers/dev-team/project-manager/skills/create-prd.md +66 -0
  264. package/template/workers/dev-team/project-manager/skills/next-issue.md +51 -0
  265. package/template/workers/dev-team/project-manager/skills/project-status.md +59 -0
  266. package/template/workers/dev-team/project-manager/skills/update-learnings.md +65 -0
  267. package/template/workers/dev-team/project-manager/src/index.ts +54 -0
  268. package/template/workers/dev-team/project-manager/src/mcp-server.ts +207 -0
  269. package/template/workers/dev-team/project-manager/src/skills/create-prd.ts +86 -0
  270. package/template/workers/dev-team/project-manager/src/skills/next-issue.ts +137 -0
  271. package/template/workers/dev-team/project-manager/src/skills/project-status.ts +131 -0
  272. package/template/workers/dev-team/project-manager/src/skills/update-learnings.ts +94 -0
  273. package/template/workers/dev-team/project-manager/tsconfig.json +19 -0
  274. package/template/workers/dev-team/project-manager/worker.yaml +96 -0
  275. package/template/workers/dev-team/qa-tester/package.json +24 -0
  276. package/template/workers/dev-team/qa-tester/skills/create-demo-account.md +36 -0
  277. package/template/workers/dev-team/qa-tester/skills/run-tests.md +36 -0
  278. package/template/workers/dev-team/qa-tester/skills/write-test.md +27 -0
  279. package/template/workers/dev-team/qa-tester/src/index.ts +61 -0
  280. package/template/workers/dev-team/qa-tester/src/mcp-server.ts +88 -0
  281. package/template/workers/dev-team/qa-tester/tsconfig.json +18 -0
  282. package/template/workers/dev-team/qa-tester/worker.yaml +116 -0
  283. package/template/workers/dev-team/task-executor/package-lock.json +1252 -0
  284. package/template/workers/dev-team/task-executor/package.json +27 -0
  285. package/template/workers/dev-team/task-executor/skills/analyze-issue.md +101 -0
  286. package/template/workers/dev-team/task-executor/skills/execute.md +133 -0
  287. package/template/workers/dev-team/task-executor/skills/report-learnings.md +106 -0
  288. package/template/workers/dev-team/task-executor/skills/validate-completion.md +121 -0
  289. package/template/workers/dev-team/task-executor/src/index.ts +54 -0
  290. package/template/workers/dev-team/task-executor/src/mcp-server.ts +139 -0
  291. package/template/workers/dev-team/task-executor/src/skills/analyze-issue.ts +219 -0
  292. package/template/workers/dev-team/task-executor/src/skills/execute.ts +132 -0
  293. package/template/workers/dev-team/task-executor/src/skills/report-learnings.ts +119 -0
  294. package/template/workers/dev-team/task-executor/src/skills/validate-completion.ts +142 -0
  295. package/template/workers/dev-team/task-executor/tsconfig.json +19 -0
  296. package/template/workers/dev-team/task-executor/worker.yaml +110 -0
  297. package/template/workers/registry.yaml +171 -0
  298. package/template/workers/security-scanner/README.md +73 -0
  299. package/template/workers/security-scanner/skills/pre-deploy-check.md +205 -0
  300. package/template/workers/security-scanner/worker.yaml +26 -0
  301. package/template/workspace/checkpoints/.gitkeep +0 -0
  302. package/template/workspace/content-ideas/inbox.jsonl +0 -0
  303. package/template/workspace/drafts/.gitkeep +0 -0
  304. package/template/workspace/learnings/.gitkeep +3 -0
  305. package/template/workspace/orchestrator/.gitkeep +0 -0
  306. package/template/workspace/ralph-test/COMPLETE.md +18 -0
  307. package/template/workspace/ralph-test/hello.txt +2 -0
  308. package/template/workspace/reports/.gitkeep +0 -0
  309. package/template/workspace/scratch/.gitkeep +0 -0
  310. package/template/workspace/threads/.gitkeep +3 -0
@@ -0,0 +1,446 @@
1
+ /**
2
+ * CMS API Integration (US-018)
3
+ * Submit suggestions to CMS review queue
4
+ */
5
+
6
+ import type { Suggestion } from './recommendations.js';
7
+
8
+ // ============================================
9
+ // Types
10
+ // ============================================
11
+
12
+ export interface CMSSuggestion {
13
+ pageSlug: string;
14
+ sectionId?: string;
15
+ original: string;
16
+ suggested: string;
17
+ rationale: string;
18
+ source: string;
19
+ priority: 'high' | 'medium' | 'low';
20
+ }
21
+
22
+ export interface CMSClientConfig {
23
+ baseUrl: string;
24
+ apiKey?: string;
25
+ timeout?: number;
26
+ }
27
+
28
+ export interface CMSClientInstance {
29
+ config: CMSClientConfig;
30
+ submitSuggestion: (suggestion: CMSSuggestion) => Promise<CMSSubmitResult>;
31
+ submitSuggestions: (suggestions: CMSSuggestion[]) => Promise<CMSBatchResult>;
32
+ getSuggestionStatus: (suggestionId: string) => Promise<CMSSuggestionStatus>;
33
+ listPendingSuggestions: (pageSlug?: string) => Promise<CMSSuggestion[]>;
34
+ }
35
+
36
+ export interface CMSSubmitResult {
37
+ success: boolean;
38
+ suggestionId?: string;
39
+ message?: string;
40
+ error?: string;
41
+ }
42
+
43
+ export interface CMSBatchResult {
44
+ submitted: number;
45
+ failed: number;
46
+ results: CMSSubmitResult[];
47
+ }
48
+
49
+ export interface CMSSuggestionStatus {
50
+ id: string;
51
+ status: 'pending' | 'approved' | 'rejected' | 'applied';
52
+ reviewedBy?: string;
53
+ reviewedAt?: string;
54
+ comments?: string;
55
+ }
56
+
57
+ export const DEFAULT_CMS_CONFIG: CMSClientConfig = {
58
+ baseUrl: '',
59
+ apiKey: undefined,
60
+ timeout: 30000,
61
+ };
62
+
63
+ // ============================================
64
+ // CMS Client Factory
65
+ // ============================================
66
+
67
+ /**
68
+ * Create CMS client instance
69
+ */
70
+ export function createCMSClient(config: CMSClientConfig): CMSClientInstance {
71
+ const cfg = { ...DEFAULT_CMS_CONFIG, ...config };
72
+
73
+ return {
74
+ config: cfg,
75
+ submitSuggestion: (suggestion) => submitSuggestion(cfg, suggestion),
76
+ submitSuggestions: (suggestions) => submitSuggestions(cfg, suggestions),
77
+ getSuggestionStatus: (id) => getSuggestionStatus(cfg, id),
78
+ listPendingSuggestions: (pageSlug) => listPendingSuggestions(cfg, pageSlug),
79
+ };
80
+ }
81
+
82
+ // ============================================
83
+ // CMS API Functions
84
+ // ============================================
85
+
86
+ /**
87
+ * Submit a single suggestion to CMS review queue
88
+ */
89
+ async function submitSuggestion(
90
+ config: CMSClientConfig,
91
+ suggestion: CMSSuggestion
92
+ ): Promise<CMSSubmitResult> {
93
+ if (!config.baseUrl) {
94
+ return {
95
+ success: false,
96
+ error: 'CMS base URL not configured',
97
+ };
98
+ }
99
+
100
+ try {
101
+ const response = await fetch(`${config.baseUrl}/api/suggestions`, {
102
+ method: 'POST',
103
+ headers: buildHeaders(config),
104
+ body: JSON.stringify(suggestion),
105
+ signal: AbortSignal.timeout(config.timeout ?? 30000),
106
+ });
107
+
108
+ if (!response.ok) {
109
+ const errorText = await response.text();
110
+ return {
111
+ success: false,
112
+ error: `HTTP ${response.status}: ${errorText}`,
113
+ };
114
+ }
115
+
116
+ const data = await response.json() as { id?: string; message?: string };
117
+ return {
118
+ success: true,
119
+ suggestionId: data.id,
120
+ message: data.message ?? 'Suggestion submitted successfully',
121
+ };
122
+ } catch (error) {
123
+ return {
124
+ success: false,
125
+ error: error instanceof Error ? error.message : 'Unknown error',
126
+ };
127
+ }
128
+ }
129
+
130
+ /**
131
+ * Submit batch of suggestions to CMS
132
+ */
133
+ async function submitSuggestions(
134
+ config: CMSClientConfig,
135
+ suggestions: CMSSuggestion[]
136
+ ): Promise<CMSBatchResult> {
137
+ const results: CMSSubmitResult[] = [];
138
+ let submitted = 0;
139
+ let failed = 0;
140
+
141
+ // Process in batches of 10 to avoid overwhelming the API
142
+ const batchSize = 10;
143
+ for (let i = 0; i < suggestions.length; i += batchSize) {
144
+ const batch = suggestions.slice(i, i + batchSize);
145
+
146
+ // Submit batch in parallel
147
+ const batchResults = await Promise.all(
148
+ batch.map(s => submitSuggestion(config, s))
149
+ );
150
+
151
+ for (const result of batchResults) {
152
+ results.push(result);
153
+ if (result.success) {
154
+ submitted++;
155
+ } else {
156
+ failed++;
157
+ }
158
+ }
159
+ }
160
+
161
+ return { submitted, failed, results };
162
+ }
163
+
164
+ /**
165
+ * Get suggestion status from CMS
166
+ */
167
+ async function getSuggestionStatus(
168
+ config: CMSClientConfig,
169
+ suggestionId: string
170
+ ): Promise<CMSSuggestionStatus> {
171
+ if (!config.baseUrl) {
172
+ throw new Error('CMS base URL not configured');
173
+ }
174
+
175
+ const response = await fetch(
176
+ `${config.baseUrl}/api/suggestions/${suggestionId}`,
177
+ {
178
+ method: 'GET',
179
+ headers: buildHeaders(config),
180
+ signal: AbortSignal.timeout(config.timeout ?? 30000),
181
+ }
182
+ );
183
+
184
+ if (!response.ok) {
185
+ throw new Error(`Failed to get suggestion status: HTTP ${response.status}`);
186
+ }
187
+
188
+ return response.json() as Promise<CMSSuggestionStatus>;
189
+ }
190
+
191
+ /**
192
+ * List pending suggestions from CMS
193
+ */
194
+ async function listPendingSuggestions(
195
+ config: CMSClientConfig,
196
+ pageSlug?: string
197
+ ): Promise<CMSSuggestion[]> {
198
+ if (!config.baseUrl) {
199
+ throw new Error('CMS base URL not configured');
200
+ }
201
+
202
+ const url = new URL(`${config.baseUrl}/api/suggestions`);
203
+ url.searchParams.set('status', 'pending');
204
+ if (pageSlug) {
205
+ url.searchParams.set('pageSlug', pageSlug);
206
+ }
207
+
208
+ const response = await fetch(url.toString(), {
209
+ method: 'GET',
210
+ headers: buildHeaders(config),
211
+ signal: AbortSignal.timeout(config.timeout ?? 30000),
212
+ });
213
+
214
+ if (!response.ok) {
215
+ throw new Error(`Failed to list suggestions: HTTP ${response.status}`);
216
+ }
217
+
218
+ const data = await response.json() as { suggestions: CMSSuggestion[] };
219
+ return data.suggestions;
220
+ }
221
+
222
+ /**
223
+ * Build request headers
224
+ */
225
+ function buildHeaders(config: CMSClientConfig): Record<string, string> {
226
+ const headers: Record<string, string> = {
227
+ 'Content-Type': 'application/json',
228
+ };
229
+
230
+ if (config.apiKey) {
231
+ headers['Authorization'] = `Bearer ${config.apiKey}`;
232
+ }
233
+
234
+ return headers;
235
+ }
236
+
237
+ // ============================================
238
+ // Suggestion Conversion
239
+ // ============================================
240
+
241
+ /**
242
+ * Convert internal suggestion to CMS format
243
+ */
244
+ export function toCMSSuggestion(suggestion: Suggestion): CMSSuggestion {
245
+ return {
246
+ pageSlug: suggestion.pageSlug,
247
+ sectionId: suggestion.sectionId,
248
+ original: suggestion.original,
249
+ suggested: suggestion.suggested,
250
+ rationale: suggestion.rationale,
251
+ source: formatSource(suggestion.source),
252
+ priority: suggestion.impact,
253
+ };
254
+ }
255
+
256
+ /**
257
+ * Convert batch of suggestions to CMS format
258
+ */
259
+ export function toCMSSuggestions(suggestions: Suggestion[]): CMSSuggestion[] {
260
+ return suggestions.map(toCMSSuggestion);
261
+ }
262
+
263
+ /**
264
+ * Format source for CMS
265
+ */
266
+ function formatSource(source: Suggestion['source']): string {
267
+ const sourceMap: Record<Suggestion['source'], string> = {
268
+ brand: 'Brand Voice Analysis',
269
+ sales: 'Conversion Analysis',
270
+ product: 'Product Accuracy Check',
271
+ legal: 'Compliance Review',
272
+ };
273
+ return sourceMap[source] ?? source;
274
+ }
275
+
276
+ // ============================================
277
+ // Status Sync
278
+ // ============================================
279
+
280
+ /**
281
+ * Sync suggestion status from CMS
282
+ */
283
+ export async function syncSuggestionStatus(
284
+ client: CMSClientInstance,
285
+ suggestionId: string
286
+ ): Promise<string> {
287
+ const status = await client.getSuggestionStatus(suggestionId);
288
+ return status.status;
289
+ }
290
+
291
+ /**
292
+ * Sync multiple suggestion statuses
293
+ */
294
+ export async function syncSuggestionStatuses(
295
+ client: CMSClientInstance,
296
+ suggestionIds: string[]
297
+ ): Promise<Map<string, string>> {
298
+ const statuses = new Map<string, string>();
299
+
300
+ // Fetch in parallel with rate limiting
301
+ const batchSize = 5;
302
+ for (let i = 0; i < suggestionIds.length; i += batchSize) {
303
+ const batch = suggestionIds.slice(i, i + batchSize);
304
+ const results = await Promise.all(
305
+ batch.map(async (id) => {
306
+ try {
307
+ const status = await syncSuggestionStatus(client, id);
308
+ return { id, status };
309
+ } catch {
310
+ return { id, status: 'unknown' };
311
+ }
312
+ })
313
+ );
314
+
315
+ for (const { id, status } of results) {
316
+ statuses.set(id, status);
317
+ }
318
+ }
319
+
320
+ return statuses;
321
+ }
322
+
323
+ // ============================================
324
+ // Dry Run / Preview Mode
325
+ // ============================================
326
+
327
+ /**
328
+ * Create a dry-run CMS client that logs instead of submitting
329
+ */
330
+ export function createDryRunCMSClient(): CMSClientInstance {
331
+ const dryRunResults: CMSSuggestion[] = [];
332
+
333
+ return {
334
+ config: { baseUrl: 'dry-run://localhost' },
335
+
336
+ async submitSuggestion(suggestion: CMSSuggestion): Promise<CMSSubmitResult> {
337
+ dryRunResults.push(suggestion);
338
+ console.log('[DRY RUN] Would submit suggestion:', suggestion.pageSlug);
339
+ return {
340
+ success: true,
341
+ suggestionId: `dry-run-${Date.now()}`,
342
+ message: 'Dry run - not actually submitted',
343
+ };
344
+ },
345
+
346
+ async submitSuggestions(suggestions: CMSSuggestion[]): Promise<CMSBatchResult> {
347
+ const results: CMSSubmitResult[] = [];
348
+ for (const s of suggestions) {
349
+ const result = await this.submitSuggestion(s);
350
+ results.push(result);
351
+ }
352
+ return {
353
+ submitted: suggestions.length,
354
+ failed: 0,
355
+ results,
356
+ };
357
+ },
358
+
359
+ async getSuggestionStatus(id: string): Promise<CMSSuggestionStatus> {
360
+ return {
361
+ id,
362
+ status: 'pending',
363
+ comments: 'Dry run mode - no actual status available',
364
+ };
365
+ },
366
+
367
+ async listPendingSuggestions(): Promise<CMSSuggestion[]> {
368
+ return dryRunResults;
369
+ },
370
+ };
371
+ }
372
+
373
+ // ============================================
374
+ // Formatting for Review
375
+ // ============================================
376
+
377
+ /**
378
+ * Format CMS suggestion for human review
379
+ */
380
+ export function formatCMSSuggestionForReview(suggestion: CMSSuggestion): string {
381
+ const lines: string[] = [];
382
+
383
+ lines.push(`## ${suggestion.pageSlug}${suggestion.sectionId ? ` > ${suggestion.sectionId}` : ''}`);
384
+ lines.push('');
385
+ lines.push(`**Priority:** ${suggestion.priority}`);
386
+ lines.push(`**Source:** ${suggestion.source}`);
387
+ lines.push('');
388
+ lines.push('### Current');
389
+ lines.push('```');
390
+ lines.push(suggestion.original);
391
+ lines.push('```');
392
+ lines.push('');
393
+ lines.push('### Suggested');
394
+ lines.push('```');
395
+ lines.push(suggestion.suggested);
396
+ lines.push('```');
397
+ lines.push('');
398
+ lines.push(`**Rationale:** ${suggestion.rationale}`);
399
+ lines.push('');
400
+
401
+ return lines.join('\n');
402
+ }
403
+
404
+ /**
405
+ * Format batch of suggestions as review document
406
+ */
407
+ export function formatSuggestionsForReview(suggestions: CMSSuggestion[]): string {
408
+ const lines: string[] = [];
409
+
410
+ lines.push('# CMS Suggestions Review');
411
+ lines.push('');
412
+ lines.push(`**Total Suggestions:** ${suggestions.length}`);
413
+ lines.push(`**Generated:** ${new Date().toISOString()}`);
414
+ lines.push('');
415
+
416
+ // Group by priority
417
+ const high = suggestions.filter(s => s.priority === 'high');
418
+ const medium = suggestions.filter(s => s.priority === 'medium');
419
+ const low = suggestions.filter(s => s.priority === 'low');
420
+
421
+ if (high.length > 0) {
422
+ lines.push('# High Priority');
423
+ lines.push('');
424
+ for (const s of high) {
425
+ lines.push(formatCMSSuggestionForReview(s));
426
+ }
427
+ }
428
+
429
+ if (medium.length > 0) {
430
+ lines.push('# Medium Priority');
431
+ lines.push('');
432
+ for (const s of medium) {
433
+ lines.push(formatCMSSuggestionForReview(s));
434
+ }
435
+ }
436
+
437
+ if (low.length > 0) {
438
+ lines.push('# Low Priority');
439
+ lines.push('');
440
+ for (const s of low) {
441
+ lines.push(formatCMSSuggestionForReview(s));
442
+ }
443
+ }
444
+
445
+ return lines.join('\n');
446
+ }