create-hq 6.0.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 (315) hide show
  1. package/dist/__tests__/scaffold.test.d.ts +2 -0
  2. package/dist/__tests__/scaffold.test.d.ts.map +1 -0
  3. package/dist/__tests__/scaffold.test.js +150 -0
  4. package/dist/__tests__/scaffold.test.js.map +1 -0
  5. package/dist/cloud-sync.d.ts +11 -0
  6. package/dist/cloud-sync.d.ts.map +1 -0
  7. package/dist/cloud-sync.js +25 -0
  8. package/dist/cloud-sync.js.map +1 -0
  9. package/dist/deps.d.ts.map +1 -1
  10. package/dist/deps.js +7 -1
  11. package/dist/deps.js.map +1 -1
  12. package/dist/fetch-template.d.ts +14 -0
  13. package/dist/fetch-template.d.ts.map +1 -0
  14. package/dist/fetch-template.js +136 -0
  15. package/dist/fetch-template.js.map +1 -0
  16. package/dist/index.js +8 -7
  17. package/dist/index.js.map +1 -1
  18. package/dist/scaffold.d.ts +2 -2
  19. package/dist/scaffold.d.ts.map +1 -1
  20. package/dist/scaffold.js +95 -286
  21. package/dist/scaffold.js.map +1 -1
  22. package/dist/ui.d.ts +2 -3
  23. package/dist/ui.d.ts.map +1 -1
  24. package/dist/ui.js +90 -44
  25. package/dist/ui.js.map +1 -1
  26. package/package.json +48 -40
  27. package/template/.claude/CLAUDE.md +0 -202
  28. package/template/.claude/commands/checkpoint.md +0 -127
  29. package/template/.claude/commands/cleanup.md +0 -307
  30. package/template/.claude/commands/execute-task.md +0 -440
  31. package/template/.claude/commands/exit-plan.md +0 -41
  32. package/template/.claude/commands/handoff.md +0 -97
  33. package/template/.claude/commands/learn.md +0 -218
  34. package/template/.claude/commands/metrics.md +0 -118
  35. package/template/.claude/commands/newworker.md +0 -162
  36. package/template/.claude/commands/nexttask.md +0 -67
  37. package/template/.claude/commands/prd.md +0 -238
  38. package/template/.claude/commands/reanchor.md +0 -51
  39. package/template/.claude/commands/remember.md +0 -126
  40. package/template/.claude/commands/run-project.md +0 -348
  41. package/template/.claude/commands/run.md +0 -110
  42. package/template/.claude/commands/search-reindex.md +0 -62
  43. package/template/.claude/commands/search.md +0 -100
  44. package/template/.claude/commands/setup.md +0 -381
  45. package/template/.claude/scripts/pure-ralph-loop.ps1 +0 -312
  46. package/template/.claude/scripts/pure-ralph-loop.sh +0 -859
  47. package/template/CHANGELOG.md +0 -220
  48. package/template/LICENSE +0 -21
  49. package/template/MIGRATION.md +0 -259
  50. package/template/README.md +0 -368
  51. package/template/data/journal/.gitkeep +0 -0
  52. package/template/docs/images/ascii-banner-options.md +0 -122
  53. package/template/docs/images/hq-banner.svg +0 -105
  54. package/template/knowledge/Ralph/01-overview.md +0 -71
  55. package/template/knowledge/Ralph/02-core-concepts.md +0 -114
  56. package/template/knowledge/Ralph/03-how-ralph-works.md +0 -184
  57. package/template/knowledge/Ralph/04-back-pressure.md +0 -222
  58. package/template/knowledge/Ralph/05-specifications.md +0 -210
  59. package/template/knowledge/Ralph/06-agents-md.md +0 -222
  60. package/template/knowledge/Ralph/07-implementation.md +0 -316
  61. package/template/knowledge/Ralph/08-economics.md +0 -182
  62. package/template/knowledge/Ralph/09-resources.md +0 -145
  63. package/template/knowledge/Ralph/10-claude-code-workflow.md +0 -212
  64. package/template/knowledge/Ralph/11-team-training-guide.md +0 -383
  65. package/template/knowledge/Ralph/README.md +0 -40
  66. package/template/knowledge/ai-security-framework/CONTRIBUTING.md +0 -139
  67. package/template/knowledge/ai-security-framework/GLOSSARY.md +0 -176
  68. package/template/knowledge/ai-security-framework/LICENSE +0 -21
  69. package/template/knowledge/ai-security-framework/QUICK-START.md +0 -172
  70. package/template/knowledge/ai-security-framework/README.md +0 -232
  71. package/template/knowledge/ai-security-framework/checklists/browser-security.md +0 -301
  72. package/template/knowledge/ai-security-framework/checklists/credential-isolation.md +0 -322
  73. package/template/knowledge/ai-security-framework/checklists/incident-response.md +0 -288
  74. package/template/knowledge/ai-security-framework/checklists/pre-flight.md +0 -249
  75. package/template/knowledge/ai-security-framework/checklists/weekly-audit.md +0 -159
  76. package/template/knowledge/ai-security-framework/configs/audit-logging.md +0 -372
  77. package/template/knowledge/ai-security-framework/configs/kill-switches.md +0 -354
  78. package/template/knowledge/ai-security-framework/docs/01-core-principles.md +0 -256
  79. package/template/knowledge/ai-security-framework/docs/02-threat-landscape.md +0 -326
  80. package/template/knowledge/ai-security-framework/docs/03-security-posture.md +0 -250
  81. package/template/knowledge/ai-security-framework/templates/agents-security.md +0 -233
  82. package/template/knowledge/design-styles/README.md +0 -42
  83. package/template/knowledge/design-styles/american-industrial.md +0 -136
  84. package/template/knowledge/design-styles/ethereal-abstract.md +0 -133
  85. package/template/knowledge/design-styles/liminal-portal.md +0 -111
  86. package/template/knowledge/design-styles/swipes/american-industrial/G-3m4YPW0AADdu2.jpeg +0 -0
  87. package/template/knowledge/design-styles/swipes/american-industrial/G-JJlt5WwAABK3K.png +0 -0
  88. package/template/knowledge/design-styles/swipes/american-industrial/G-JJmj5W0AEbJ-7.png +0 -0
  89. package/template/knowledge/design-styles/swipes/american-industrial/G59fgNuXkAAKLJQ (1).jpeg +0 -0
  90. package/template/knowledge/design-styles/swipes/american-industrial/G59fgNuXkAAKLJQ.jpeg +0 -0
  91. package/template/knowledge/design-styles/swipes/american-industrial/G7fVkn3WEAAM-ST.jpeg +0 -0
  92. package/template/knowledge/design-styles/swipes/american-industrial/G8ECO5JWEAIksyn.png +0 -0
  93. package/template/knowledge/design-styles/swipes/american-industrial/G9-3GQSWoAA8eqZ.png +0 -0
  94. package/template/knowledge/design-styles/swipes/american-industrial/G9xEOqrXkAEZRcs.png +0 -0
  95. package/template/knowledge/design-styles/swipes/american-industrial/G_MVeJrXQAA8sx4.jpeg +0 -0
  96. package/template/knowledge/design-styles/swipes/american-industrial/G_RSkmGXkAAgAVZ.png +0 -0
  97. package/template/knowledge/design-styles/swipes/american-industrial/README.md +0 -31
  98. package/template/knowledge/design-styles/swipes/american-industrial/qyqtg7Dq.png +0 -0
  99. package/template/knowledge/dev-team/README.md +0 -35
  100. package/template/knowledge/dev-team/patterns/README.md +0 -34
  101. package/template/knowledge/dev-team/patterns/frontend/react-best-practices.md +0 -178
  102. package/template/knowledge/dev-team/troubleshooting/README.md +0 -31
  103. package/template/knowledge/dev-team/workflows/README.md +0 -49
  104. package/template/knowledge/hq/checkpoint-schema.json +0 -51
  105. package/template/knowledge/hq/index-md-spec.md +0 -74
  106. package/template/knowledge/hq/thread-schema.md +0 -153
  107. package/template/knowledge/hq-core/checkpoint-schema.json +0 -51
  108. package/template/knowledge/hq-core/index-md-spec.md +0 -74
  109. package/template/knowledge/hq-core/thread-schema.md +0 -153
  110. package/template/knowledge/loom/README.md +0 -51
  111. package/template/knowledge/loom/architecture.md +0 -125
  112. package/template/knowledge/loom/code-style.md +0 -169
  113. package/template/knowledge/loom/llm-proxy.md +0 -132
  114. package/template/knowledge/loom/state-machine.md +0 -131
  115. package/template/knowledge/loom/thread-system.md +0 -117
  116. package/template/knowledge/loom/tools.md +0 -94
  117. package/template/knowledge/loom/weaver.md +0 -96
  118. package/template/knowledge/loom/web-frontend.md +0 -131
  119. package/template/knowledge/projects/README.md +0 -72
  120. package/template/knowledge/projects/templates/README.template.md +0 -28
  121. package/template/knowledge/workers/README.md +0 -195
  122. package/template/knowledge/workers/ralph-loop-pattern.md +0 -157
  123. package/template/knowledge/workers/skill-schema.md +0 -182
  124. package/template/knowledge/workers/state-machine.md +0 -102
  125. package/template/knowledge/workers/templates/base-worker.yaml +0 -73
  126. package/template/knowledge/workers/templates/code-worker.yaml +0 -85
  127. package/template/knowledge/workers/templates/skill.yaml +0 -49
  128. package/template/knowledge/workers/templates/social-worker.yaml +0 -70
  129. package/template/modules/examples/full-manifest.yaml +0 -92
  130. package/template/modules/examples/minimal.yaml +0 -14
  131. package/template/modules/modules.yaml +0 -59
  132. package/template/projects/.gitkeep +0 -0
  133. package/template/projects/incorporate-workers-into-pure-ralph/prd.json +0 -88
  134. package/template/projects/pure-ralph-branch-isolation/README.md +0 -114
  135. package/template/projects/pure-ralph-branch-isolation/prd.json +0 -123
  136. package/template/projects/purist-ralph-loop/README.md +0 -148
  137. package/template/projects/purist-ralph-loop/prd.json +0 -135
  138. package/template/projects/ralph-test/prd.json +0 -50
  139. package/template/prompts/pure-ralph-base.md +0 -551
  140. package/template/settings/.gitkeep +0 -0
  141. package/template/settings/pure-ralph.json +0 -42
  142. package/template/social-content/drafts/INDEX.md +0 -21
  143. package/template/social-content/drafts/linkedin/.gitkeep +0 -1
  144. package/template/social-content/drafts/x/.gitkeep +0 -1
  145. package/template/social-content/images/.gitkeep +0 -1
  146. package/template/starter-projects/code-worker/README.md +0 -97
  147. package/template/starter-projects/code-worker/prd.json +0 -45
  148. package/template/starter-projects/personal-assistant/README.md +0 -42
  149. package/template/starter-projects/personal-assistant/prd.json +0 -43
  150. package/template/starter-projects/social-media/README.md +0 -60
  151. package/template/starter-projects/social-media/prd.json +0 -43
  152. package/template/workers/content-brand/README.md +0 -59
  153. package/template/workers/content-brand/skills/messaging-alignment.md +0 -91
  154. package/template/workers/content-brand/skills/tone-check.md +0 -76
  155. package/template/workers/content-brand/skills/voice-analysis.md +0 -68
  156. package/template/workers/content-brand/worker.yaml +0 -81
  157. package/template/workers/content-legal/README.md +0 -80
  158. package/template/workers/content-legal/skills/claim-substantiation.md +0 -150
  159. package/template/workers/content-legal/skills/compliance-scan.md +0 -123
  160. package/template/workers/content-legal/skills/disclaimer-check.md +0 -146
  161. package/template/workers/content-legal/worker.yaml +0 -118
  162. package/template/workers/content-product/README.md +0 -77
  163. package/template/workers/content-product/skills/claim-verification.md +0 -96
  164. package/template/workers/content-product/skills/feature-accuracy.md +0 -117
  165. package/template/workers/content-product/skills/stats-check.md +0 -128
  166. package/template/workers/content-product/worker.yaml +0 -97
  167. package/template/workers/content-sales/README.md +0 -70
  168. package/template/workers/content-sales/skills/conversion-analysis.md +0 -96
  169. package/template/workers/content-sales/skills/cta-audit.md +0 -107
  170. package/template/workers/content-sales/skills/value-prop-check.md +0 -114
  171. package/template/workers/content-sales/worker.yaml +0 -93
  172. package/template/workers/content-shared/cli.ts +0 -242
  173. package/template/workers/content-shared/index.ts +0 -234
  174. package/template/workers/content-shared/lib/accuracy-analyzer.ts +0 -661
  175. package/template/workers/content-shared/lib/analyze.ts +0 -370
  176. package/template/workers/content-shared/lib/brand-analyzer.ts +0 -526
  177. package/template/workers/content-shared/lib/cms-integration.ts +0 -446
  178. package/template/workers/content-shared/lib/compliance-analyzer.ts +0 -655
  179. package/template/workers/content-shared/lib/conversion-analyzer.ts +0 -555
  180. package/template/workers/content-shared/lib/github-integration.ts +0 -582
  181. package/template/workers/content-shared/lib/output.ts +0 -373
  182. package/template/workers/content-shared/lib/parser.ts +0 -771
  183. package/template/workers/content-shared/lib/priority.ts +0 -439
  184. package/template/workers/content-shared/lib/recommendations.ts +0 -512
  185. package/template/workers/content-shared/lib/reporter.ts +0 -749
  186. package/template/workers/content-shared/lib/restructure.ts +0 -664
  187. package/template/workers/content-shared/lib/scorer.ts +0 -140
  188. package/template/workers/content-shared/lib/types.ts +0 -227
  189. package/template/workers/content-shared/lib/variants.ts +0 -595
  190. package/template/workers/content-shared/package.json +0 -51
  191. package/template/workers/content-shared/pnpm-lock.yaml +0 -39
  192. package/template/workers/content-shared/test/sample-page.json +0 -115
  193. package/template/workers/content-shared/tsconfig.json +0 -20
  194. package/template/workers/dev-team/README.md +0 -166
  195. package/template/workers/dev-team/_template.yaml +0 -70
  196. package/template/workers/dev-team/architect/package.json +0 -27
  197. package/template/workers/dev-team/architect/skills/api-design.md +0 -89
  198. package/template/workers/dev-team/architect/skills/refactor-plan.md +0 -96
  199. package/template/workers/dev-team/architect/skills/system-design.md +0 -100
  200. package/template/workers/dev-team/architect/src/index.ts +0 -49
  201. package/template/workers/dev-team/architect/src/mcp-server.ts +0 -122
  202. package/template/workers/dev-team/architect/src/skills/api-design.ts +0 -316
  203. package/template/workers/dev-team/architect/src/skills/refactor-plan.ts +0 -264
  204. package/template/workers/dev-team/architect/src/skills/system-design.ts +0 -212
  205. package/template/workers/dev-team/architect/tsconfig.json +0 -19
  206. package/template/workers/dev-team/architect/worker.yaml +0 -128
  207. package/template/workers/dev-team/backend-dev/package-lock.json +0 -1252
  208. package/template/workers/dev-team/backend-dev/package.json +0 -27
  209. package/template/workers/dev-team/backend-dev/skills/implement-endpoint.md +0 -70
  210. package/template/workers/dev-team/backend-dev/skills/implement-service.md +0 -62
  211. package/template/workers/dev-team/backend-dev/src/index.ts +0 -51
  212. package/template/workers/dev-team/backend-dev/src/mcp-server.ts +0 -109
  213. package/template/workers/dev-team/backend-dev/src/skills/implement-endpoint.ts +0 -122
  214. package/template/workers/dev-team/backend-dev/src/skills/implement-service.ts +0 -126
  215. package/template/workers/dev-team/backend-dev/tsconfig.json +0 -19
  216. package/template/workers/dev-team/backend-dev/worker.yaml +0 -128
  217. package/template/workers/dev-team/code-reviewer/package-lock.json +0 -1080
  218. package/template/workers/dev-team/code-reviewer/package.json +0 -24
  219. package/template/workers/dev-team/code-reviewer/skills/merge-to-production.md +0 -61
  220. package/template/workers/dev-team/code-reviewer/skills/merge-to-staging.md +0 -54
  221. package/template/workers/dev-team/code-reviewer/skills/request-changes.md +0 -63
  222. package/template/workers/dev-team/code-reviewer/skills/review-pr.md +0 -77
  223. package/template/workers/dev-team/code-reviewer/src/index.ts +0 -56
  224. package/template/workers/dev-team/code-reviewer/src/mcp-server.ts +0 -101
  225. package/template/workers/dev-team/code-reviewer/tsconfig.json +0 -19
  226. package/template/workers/dev-team/code-reviewer/worker.yaml +0 -90
  227. package/template/workers/dev-team/database-dev/package.json +0 -22
  228. package/template/workers/dev-team/database-dev/skills/create-schema.md +0 -48
  229. package/template/workers/dev-team/database-dev/src/index.ts +0 -50
  230. package/template/workers/dev-team/database-dev/src/mcp-server.ts +0 -76
  231. package/template/workers/dev-team/database-dev/tsconfig.json +0 -18
  232. package/template/workers/dev-team/database-dev/worker.yaml +0 -90
  233. package/template/workers/dev-team/frontend-dev/package.json +0 -22
  234. package/template/workers/dev-team/frontend-dev/skills/create-component.md +0 -26
  235. package/template/workers/dev-team/frontend-dev/src/index.ts +0 -50
  236. package/template/workers/dev-team/frontend-dev/src/mcp-server.ts +0 -77
  237. package/template/workers/dev-team/frontend-dev/tsconfig.json +0 -18
  238. package/template/workers/dev-team/frontend-dev/worker.yaml +0 -132
  239. package/template/workers/dev-team/infra-dev/package.json +0 -24
  240. package/template/workers/dev-team/infra-dev/skills/add-monitoring.md +0 -73
  241. package/template/workers/dev-team/infra-dev/skills/configure-deployment.md +0 -80
  242. package/template/workers/dev-team/infra-dev/skills/create-dockerfile.md +0 -62
  243. package/template/workers/dev-team/infra-dev/skills/setup-cicd.md +0 -63
  244. package/template/workers/dev-team/infra-dev/src/index.ts +0 -55
  245. package/template/workers/dev-team/infra-dev/src/mcp-server.ts +0 -82
  246. package/template/workers/dev-team/infra-dev/tsconfig.json +0 -19
  247. package/template/workers/dev-team/infra-dev/worker.yaml +0 -92
  248. package/template/workers/dev-team/knowledge-curator/package.json +0 -24
  249. package/template/workers/dev-team/knowledge-curator/skills/curate-troubleshooting.md +0 -63
  250. package/template/workers/dev-team/knowledge-curator/skills/process-learnings.md +0 -61
  251. package/template/workers/dev-team/knowledge-curator/skills/sync-documentation.md +0 -76
  252. package/template/workers/dev-team/knowledge-curator/skills/update-patterns.md +0 -63
  253. package/template/workers/dev-team/knowledge-curator/src/index.ts +0 -53
  254. package/template/workers/dev-team/knowledge-curator/src/mcp-server.ts +0 -92
  255. package/template/workers/dev-team/knowledge-curator/tsconfig.json +0 -19
  256. package/template/workers/dev-team/knowledge-curator/worker.yaml +0 -80
  257. package/template/workers/dev-team/motion-designer/package.json +0 -22
  258. package/template/workers/dev-team/motion-designer/skills/add-animation.md +0 -25
  259. package/template/workers/dev-team/motion-designer/skills/generate-image.md +0 -36
  260. package/template/workers/dev-team/motion-designer/src/index.ts +0 -63
  261. package/template/workers/dev-team/motion-designer/src/mcp-server.ts +0 -79
  262. package/template/workers/dev-team/motion-designer/tsconfig.json +0 -18
  263. package/template/workers/dev-team/motion-designer/worker.yaml +0 -84
  264. package/template/workers/dev-team/product-planner/queue.json +0 -4
  265. package/template/workers/dev-team/product-planner/worker.yaml +0 -220
  266. package/template/workers/dev-team/project-manager/package-lock.json +0 -1252
  267. package/template/workers/dev-team/project-manager/package.json +0 -27
  268. package/template/workers/dev-team/project-manager/skills/create-prd.md +0 -66
  269. package/template/workers/dev-team/project-manager/skills/next-issue.md +0 -51
  270. package/template/workers/dev-team/project-manager/skills/project-status.md +0 -59
  271. package/template/workers/dev-team/project-manager/skills/update-learnings.md +0 -65
  272. package/template/workers/dev-team/project-manager/src/index.ts +0 -54
  273. package/template/workers/dev-team/project-manager/src/mcp-server.ts +0 -207
  274. package/template/workers/dev-team/project-manager/src/skills/create-prd.ts +0 -86
  275. package/template/workers/dev-team/project-manager/src/skills/next-issue.ts +0 -137
  276. package/template/workers/dev-team/project-manager/src/skills/project-status.ts +0 -131
  277. package/template/workers/dev-team/project-manager/src/skills/update-learnings.ts +0 -94
  278. package/template/workers/dev-team/project-manager/tsconfig.json +0 -19
  279. package/template/workers/dev-team/project-manager/worker.yaml +0 -96
  280. package/template/workers/dev-team/qa-tester/package.json +0 -24
  281. package/template/workers/dev-team/qa-tester/skills/create-demo-account.md +0 -36
  282. package/template/workers/dev-team/qa-tester/skills/run-tests.md +0 -36
  283. package/template/workers/dev-team/qa-tester/skills/write-test.md +0 -27
  284. package/template/workers/dev-team/qa-tester/src/index.ts +0 -61
  285. package/template/workers/dev-team/qa-tester/src/mcp-server.ts +0 -88
  286. package/template/workers/dev-team/qa-tester/tsconfig.json +0 -18
  287. package/template/workers/dev-team/qa-tester/worker.yaml +0 -116
  288. package/template/workers/dev-team/task-executor/package-lock.json +0 -1252
  289. package/template/workers/dev-team/task-executor/package.json +0 -27
  290. package/template/workers/dev-team/task-executor/skills/analyze-issue.md +0 -101
  291. package/template/workers/dev-team/task-executor/skills/execute.md +0 -133
  292. package/template/workers/dev-team/task-executor/skills/report-learnings.md +0 -106
  293. package/template/workers/dev-team/task-executor/skills/validate-completion.md +0 -121
  294. package/template/workers/dev-team/task-executor/src/index.ts +0 -54
  295. package/template/workers/dev-team/task-executor/src/mcp-server.ts +0 -139
  296. package/template/workers/dev-team/task-executor/src/skills/analyze-issue.ts +0 -219
  297. package/template/workers/dev-team/task-executor/src/skills/execute.ts +0 -132
  298. package/template/workers/dev-team/task-executor/src/skills/report-learnings.ts +0 -119
  299. package/template/workers/dev-team/task-executor/src/skills/validate-completion.ts +0 -142
  300. package/template/workers/dev-team/task-executor/tsconfig.json +0 -19
  301. package/template/workers/dev-team/task-executor/worker.yaml +0 -110
  302. package/template/workers/registry.yaml +0 -171
  303. package/template/workers/security-scanner/README.md +0 -73
  304. package/template/workers/security-scanner/skills/pre-deploy-check.md +0 -205
  305. package/template/workers/security-scanner/worker.yaml +0 -26
  306. package/template/workspace/checkpoints/.gitkeep +0 -0
  307. package/template/workspace/content-ideas/inbox.jsonl +0 -0
  308. package/template/workspace/drafts/.gitkeep +0 -0
  309. package/template/workspace/learnings/.gitkeep +0 -3
  310. package/template/workspace/orchestrator/.gitkeep +0 -0
  311. package/template/workspace/ralph-test/COMPLETE.md +0 -18
  312. package/template/workspace/ralph-test/hello.txt +0 -2
  313. package/template/workspace/reports/.gitkeep +0 -0
  314. package/template/workspace/scratch/.gitkeep +0 -0
  315. package/template/workspace/threads/.gitkeep +0 -3
@@ -1,512 +0,0 @@
1
- /**
2
- * Recommendations Generator (US-011)
3
- * Creates improvement recommendations with rationale from analysis findings
4
- */
5
-
6
- import type {
7
- FullAnalysis,
8
- Finding,
9
- Recommendation,
10
- ContentSection,
11
- AnalysisInput,
12
- } from './types.js';
13
-
14
- // ============================================
15
- // Types
16
- // ============================================
17
-
18
- export interface Suggestion {
19
- id: string;
20
- type: 'text' | 'structure' | 'cta' | 'stat' | 'claim';
21
- pageSlug: string;
22
- sectionId?: string;
23
- original: string;
24
- suggested: string;
25
- rationale: string;
26
- source: 'brand' | 'sales' | 'product' | 'legal';
27
- impact: 'high' | 'medium' | 'low';
28
- effort: 'quick' | 'moderate' | 'significant';
29
- priority: number; // Calculated from impact/effort
30
- }
31
-
32
- export interface SuggestionContext {
33
- pageSlug: string;
34
- sectionId?: string;
35
- contentType?: ContentSection['type'];
36
- }
37
-
38
- // ============================================
39
- // Impact/Effort Scoring
40
- // ============================================
41
-
42
- const IMPACT_SCORES: Record<Suggestion['impact'], number> = {
43
- high: 3,
44
- medium: 2,
45
- low: 1,
46
- };
47
-
48
- const EFFORT_SCORES: Record<Suggestion['effort'], number> = {
49
- quick: 3, // Lower effort = higher score (easier to do)
50
- moderate: 2,
51
- significant: 1,
52
- };
53
-
54
- /**
55
- * Calculate priority score from impact and effort
56
- * Higher score = do first (high impact + quick effort = best)
57
- */
58
- function calculateSuggestionPriority(impact: Suggestion['impact'], effort: Suggestion['effort']): number {
59
- return IMPACT_SCORES[impact] * EFFORT_SCORES[effort];
60
- }
61
-
62
- // ============================================
63
- // Suggestion Generation
64
- // ============================================
65
-
66
- let suggestionCounter = 0;
67
-
68
- function generateSuggestionId(): string {
69
- suggestionCounter++;
70
- return `sug-${Date.now()}-${suggestionCounter}`;
71
- }
72
-
73
- /**
74
- * Generate suggestions from full analysis results
75
- */
76
- export function generateSuggestions(analysis: FullAnalysis): Suggestion[] {
77
- const suggestions: Suggestion[] = [];
78
- const pageSlug = analysis.page;
79
-
80
- // Process brand findings
81
- if (analysis.brand) {
82
- suggestions.push(...convertFindingsToSuggestions(
83
- analysis.brand.findings,
84
- analysis.brand.recommendations,
85
- pageSlug,
86
- 'brand'
87
- ));
88
- }
89
-
90
- // Process conversion findings
91
- if (analysis.conversion) {
92
- suggestions.push(...convertFindingsToSuggestions(
93
- analysis.conversion.findings,
94
- analysis.conversion.recommendations,
95
- pageSlug,
96
- 'sales'
97
- ));
98
- }
99
-
100
- // Process accuracy findings
101
- if (analysis.accuracy) {
102
- suggestions.push(...convertFindingsToSuggestions(
103
- analysis.accuracy.findings,
104
- analysis.accuracy.recommendations,
105
- pageSlug,
106
- 'product'
107
- ));
108
- }
109
-
110
- // Process compliance findings
111
- if (analysis.compliance) {
112
- suggestions.push(...convertFindingsToSuggestions(
113
- analysis.compliance.findings,
114
- analysis.compliance.recommendations,
115
- pageSlug,
116
- 'legal'
117
- ));
118
- }
119
-
120
- // Sort by priority (highest first)
121
- suggestions.sort((a, b) => b.priority - a.priority);
122
-
123
- return suggestions;
124
- }
125
-
126
- /**
127
- * Convert findings and recommendations to suggestions
128
- */
129
- function convertFindingsToSuggestions(
130
- findings: Finding[],
131
- recommendations: Recommendation[],
132
- pageSlug: string,
133
- source: Suggestion['source']
134
- ): Suggestion[] {
135
- const suggestions: Suggestion[] = [];
136
-
137
- // Create suggestions from recommendations (these have current/suggested pairs)
138
- for (const rec of recommendations) {
139
- const impact = mapPriorityToImpact(rec.priority);
140
- const effort = estimateEffort(rec.current, rec.suggested);
141
- const type = inferSuggestionType(rec.category, rec.current);
142
-
143
- suggestions.push({
144
- id: generateSuggestionId(),
145
- type,
146
- pageSlug,
147
- sectionId: extractSectionId(rec.current),
148
- original: rec.current,
149
- suggested: rec.suggested,
150
- rationale: rec.rationale,
151
- source,
152
- impact,
153
- effort,
154
- priority: calculateSuggestionPriority(impact, effort),
155
- });
156
- }
157
-
158
- // Create suggestions from critical/warning findings without recommendations
159
- const recommendedOriginals = new Set(recommendations.map(r => r.current));
160
-
161
- for (const finding of findings) {
162
- if (finding.severity === 'pass' || finding.severity === 'info') continue;
163
- if (finding.evidence && recommendedOriginals.has(finding.evidence)) continue;
164
-
165
- const impact = finding.severity === 'critical' ? 'high' : 'medium';
166
- const effort = estimateEffortFromMessage(finding.message);
167
- const type = inferSuggestionTypeFromCategory(finding.category);
168
-
169
- suggestions.push({
170
- id: generateSuggestionId(),
171
- type,
172
- pageSlug,
173
- sectionId: finding.location,
174
- original: finding.evidence ?? finding.message,
175
- suggested: generateSuggestedFix(finding),
176
- rationale: finding.message,
177
- source,
178
- impact,
179
- effort,
180
- priority: calculateSuggestionPriority(impact, effort),
181
- });
182
- }
183
-
184
- return suggestions;
185
- }
186
-
187
- // ============================================
188
- // Suggestion Enhancement
189
- // ============================================
190
-
191
- /**
192
- * Enhance a suggestion with AI-generated alternatives
193
- * This provides a richer suggested text based on the type and context
194
- */
195
- export function enhanceSuggestion(suggestion: Suggestion): Suggestion {
196
- const enhanced = { ...suggestion };
197
-
198
- // Generate improved suggestion based on type
199
- switch (suggestion.type) {
200
- case 'cta':
201
- enhanced.suggested = enhanceCTA(suggestion.original, suggestion.source);
202
- break;
203
- case 'claim':
204
- enhanced.suggested = enhanceClaim(suggestion.original, suggestion.source);
205
- break;
206
- case 'stat':
207
- enhanced.suggested = enhanceStat(suggestion.original);
208
- break;
209
- case 'text':
210
- enhanced.suggested = enhanceText(suggestion.original, suggestion.source);
211
- break;
212
- case 'structure':
213
- // Structure suggestions typically come pre-formed
214
- break;
215
- }
216
-
217
- // Add additional rationale if enhanced
218
- if (enhanced.suggested !== suggestion.suggested) {
219
- enhanced.rationale = `${suggestion.rationale} Enhanced for ${suggestion.source} optimization.`;
220
- }
221
-
222
- return enhanced;
223
- }
224
-
225
- /**
226
- * Enhance CTA text
227
- */
228
- function enhanceCTA(original: string, source: Suggestion['source']): string {
229
- const lowerOriginal = original.toLowerCase();
230
-
231
- // Generic CTAs to improve
232
- if (lowerOriginal === 'learn more' || lowerOriginal === 'click here') {
233
- if (source === 'sales') return 'See How It Works';
234
- if (source === 'brand') return 'Discover Our Approach';
235
- return 'Explore the Details';
236
- }
237
-
238
- if (lowerOriginal === 'contact us') {
239
- if (source === 'sales') return 'Talk to an Expert';
240
- return 'Get in Touch';
241
- }
242
-
243
- if (lowerOriginal === 'submit') {
244
- return 'Send Message';
245
- }
246
-
247
- if (lowerOriginal === 'sign up') {
248
- if (source === 'sales') return 'Start Free Trial';
249
- return 'Create Your Account';
250
- }
251
-
252
- return original;
253
- }
254
-
255
- /**
256
- * Enhance claim text for better substantiation
257
- */
258
- function enhanceClaim(original: string, source: Suggestion['source']): string {
259
- // Add specificity indicators
260
- if (original.includes('best') && !original.includes('one of')) {
261
- return original.replace(/\bbest\b/gi, 'one of the best');
262
- }
263
-
264
- if (original.includes('leading') && !original.includes('industry')) {
265
- return original.replace(/\bleading\b/gi, 'industry-leading');
266
- }
267
-
268
- // Add hedging for absolute claims
269
- if (original.includes('always') || original.includes('never')) {
270
- return `In most cases, ${original.toLowerCase().replace(/\balways\b/gi, 'typically').replace(/\bnever\b/gi, 'rarely')}`;
271
- }
272
-
273
- return original;
274
- }
275
-
276
- /**
277
- * Enhance stat text with context
278
- */
279
- function enhanceStat(original: string): string {
280
- // Add "up to" for unqualified percentages
281
- if (/\d+%/.test(original) && !original.includes('up to') && !original.includes('over')) {
282
- return original.replace(/(\d+%)/, 'up to $1');
283
- }
284
-
285
- return original;
286
- }
287
-
288
- /**
289
- * Enhance general text
290
- */
291
- function enhanceText(original: string, source: Suggestion['source']): string {
292
- // Add brand voice elements
293
- if (source === 'brand' && original.length > 50) {
294
- // Break up long sentences
295
- const sentences = original.split(/[.!?]+/).filter(s => s.trim());
296
- if (sentences.length === 1 && sentences[0].split(' ').length > 25) {
297
- // Suggest breaking into two sentences
298
- const words = sentences[0].split(' ');
299
- const midpoint = Math.floor(words.length / 2);
300
- return words.slice(0, midpoint).join(' ') + '. ' + words.slice(midpoint).join(' ');
301
- }
302
- }
303
-
304
- return original;
305
- }
306
-
307
- // ============================================
308
- // Helper Functions
309
- // ============================================
310
-
311
- function mapPriorityToImpact(priority: Recommendation['priority']): Suggestion['impact'] {
312
- return priority; // They use the same values
313
- }
314
-
315
- function estimateEffort(original: string, suggested: string): Suggestion['effort'] {
316
- const originalWords = original.split(/\s+/).length;
317
- const suggestedWords = suggested.split(/\s+/).length;
318
- const wordDiff = Math.abs(originalWords - suggestedWords);
319
-
320
- // Small text changes are quick
321
- if (wordDiff <= 5 && originalWords <= 20) return 'quick';
322
-
323
- // Medium changes
324
- if (wordDiff <= 15 && originalWords <= 50) return 'moderate';
325
-
326
- // Large changes
327
- return 'significant';
328
- }
329
-
330
- function estimateEffortFromMessage(message: string): Suggestion['effort'] {
331
- const lowerMessage = message.toLowerCase();
332
-
333
- // Quick fixes
334
- if (lowerMessage.includes('missing') || lowerMessage.includes('add')) return 'quick';
335
- if (lowerMessage.includes('update') || lowerMessage.includes('change')) return 'moderate';
336
-
337
- // Significant work
338
- if (lowerMessage.includes('restructure') || lowerMessage.includes('rewrite')) return 'significant';
339
-
340
- return 'moderate';
341
- }
342
-
343
- function inferSuggestionType(category: string, content: string): Suggestion['type'] {
344
- const lowerCategory = category.toLowerCase();
345
- const lowerContent = content.toLowerCase();
346
-
347
- if (lowerCategory.includes('cta') || lowerCategory.includes('button')) return 'cta';
348
- if (lowerCategory.includes('stat') || /\d+[%x]/.test(content)) return 'stat';
349
- if (lowerCategory.includes('claim') || lowerCategory.includes('compliance')) return 'claim';
350
- if (lowerCategory.includes('structure') || lowerCategory.includes('section')) return 'structure';
351
-
352
- return 'text';
353
- }
354
-
355
- function inferSuggestionTypeFromCategory(category: string): Suggestion['type'] {
356
- const lowerCategory = category.toLowerCase();
357
-
358
- if (lowerCategory.includes('cta')) return 'cta';
359
- if (lowerCategory.includes('stat') || lowerCategory.includes('accuracy')) return 'stat';
360
- if (lowerCategory.includes('claim') || lowerCategory.includes('compliance')) return 'claim';
361
- if (lowerCategory.includes('structure')) return 'structure';
362
-
363
- return 'text';
364
- }
365
-
366
- function extractSectionId(content: string): string | undefined {
367
- // Try to extract section reference from content
368
- const sectionMatch = content.match(/section[:\s]+([a-z0-9-]+)/i);
369
- if (sectionMatch) return sectionMatch[1];
370
-
371
- const inMatch = content.match(/in\s+([a-z0-9-]+)\s+section/i);
372
- if (inMatch) return inMatch[1];
373
-
374
- return undefined;
375
- }
376
-
377
- function generateSuggestedFix(finding: Finding): string {
378
- const { message, evidence, category } = finding;
379
-
380
- // Generate context-appropriate suggestions
381
- if (category.toLowerCase().includes('cta')) {
382
- return 'Consider using action-oriented language with clear value proposition';
383
- }
384
-
385
- if (category.toLowerCase().includes('compliance')) {
386
- return evidence
387
- ? `Add appropriate disclaimer or substantiation for: "${evidence}"`
388
- : 'Review and add required disclaimers';
389
- }
390
-
391
- if (category.toLowerCase().includes('accuracy')) {
392
- return evidence
393
- ? `Verify and update: "${evidence}"`
394
- : 'Verify claims against current product data';
395
- }
396
-
397
- if (category.toLowerCase().includes('brand')) {
398
- return 'Align with brand voice guidelines and approved terminology';
399
- }
400
-
401
- return `Address: ${message}`;
402
- }
403
-
404
- // ============================================
405
- // Batch Processing
406
- // ============================================
407
-
408
- /**
409
- * Generate suggestions for multiple pages
410
- */
411
- export function generateSuggestionsForPages(analyses: FullAnalysis[]): Map<string, Suggestion[]> {
412
- const suggestionsByPage = new Map<string, Suggestion[]>();
413
-
414
- for (const analysis of analyses) {
415
- const suggestions = generateSuggestions(analysis);
416
- suggestionsByPage.set(analysis.page, suggestions);
417
- }
418
-
419
- return suggestionsByPage;
420
- }
421
-
422
- /**
423
- * Get all suggestions sorted by priority across all pages
424
- */
425
- export function getAllSuggestionsSorted(suggestionsByPage: Map<string, Suggestion[]>): Suggestion[] {
426
- const all: Suggestion[] = [];
427
-
428
- for (const suggestions of suggestionsByPage.values()) {
429
- all.push(...suggestions);
430
- }
431
-
432
- return all.sort((a, b) => b.priority - a.priority);
433
- }
434
-
435
- // ============================================
436
- // Formatting
437
- // ============================================
438
-
439
- /**
440
- * Format a suggestion for display
441
- */
442
- export function formatSuggestion(suggestion: Suggestion): string {
443
- const impactEmoji = suggestion.impact === 'high' ? '[!]' : suggestion.impact === 'medium' ? '[*]' : '[-]';
444
- const effortLabel = suggestion.effort === 'quick' ? 'Quick fix' : suggestion.effort === 'moderate' ? 'Moderate' : 'Significant';
445
-
446
- return `${impactEmoji} [${suggestion.type.toUpperCase()}] ${suggestion.pageSlug}${suggestion.sectionId ? `#${suggestion.sectionId}` : ''}
447
- Original: "${truncate(suggestion.original, 80)}"
448
- Suggested: "${truncate(suggestion.suggested, 80)}"
449
- Rationale: ${suggestion.rationale}
450
- Impact: ${suggestion.impact} | Effort: ${effortLabel} | Priority: ${suggestion.priority}
451
- Source: ${suggestion.source}`;
452
- }
453
-
454
- function truncate(text: string, maxLength: number): string {
455
- if (text.length <= maxLength) return text;
456
- return text.slice(0, maxLength - 3) + '...';
457
- }
458
-
459
- /**
460
- * Format suggestions as markdown
461
- */
462
- export function formatSuggestionsMarkdown(suggestions: Suggestion[]): string {
463
- const lines: string[] = ['# Content Improvement Suggestions\n'];
464
-
465
- // Group by impact
466
- const highImpact = suggestions.filter(s => s.impact === 'high');
467
- const mediumImpact = suggestions.filter(s => s.impact === 'medium');
468
- const lowImpact = suggestions.filter(s => s.impact === 'low');
469
-
470
- if (highImpact.length > 0) {
471
- lines.push('## High Impact\n');
472
- for (const s of highImpact) {
473
- lines.push(formatSuggestionMarkdownItem(s));
474
- }
475
- }
476
-
477
- if (mediumImpact.length > 0) {
478
- lines.push('\n## Medium Impact\n');
479
- for (const s of mediumImpact) {
480
- lines.push(formatSuggestionMarkdownItem(s));
481
- }
482
- }
483
-
484
- if (lowImpact.length > 0) {
485
- lines.push('\n## Low Impact\n');
486
- for (const s of lowImpact) {
487
- lines.push(formatSuggestionMarkdownItem(s));
488
- }
489
- }
490
-
491
- return lines.join('\n');
492
- }
493
-
494
- function formatSuggestionMarkdownItem(s: Suggestion): string {
495
- const effortBadge = s.effort === 'quick' ? 'Quick Win' : s.effort === 'moderate' ? 'Moderate Effort' : 'Major Change';
496
-
497
- return `### ${s.type.charAt(0).toUpperCase() + s.type.slice(1)}: ${s.pageSlug}
498
- - **Location**: ${s.sectionId ?? 'Page-level'}
499
- - **Effort**: ${effortBadge}
500
- - **Source**: ${s.source}
501
-
502
- **Current:**
503
- > ${s.original}
504
-
505
- **Suggested:**
506
- > ${s.suggested}
507
-
508
- **Rationale:** ${s.rationale}
509
-
510
- ---
511
- `;
512
- }