create-hq 5.0.0 → 5.2.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 (301) hide show
  1. package/dist/deps.d.ts +2 -2
  2. package/dist/deps.d.ts.map +1 -1
  3. package/dist/deps.js +138 -29
  4. package/dist/deps.js.map +1 -1
  5. package/dist/index.js +3 -3
  6. package/dist/index.js.map +1 -1
  7. package/dist/scaffold.d.ts +1 -1
  8. package/dist/scaffold.d.ts.map +1 -1
  9. package/dist/scaffold.js +125 -26
  10. package/dist/scaffold.js.map +1 -1
  11. package/dist/ui.js +1 -1
  12. package/package.json +4 -8
  13. package/template/.claude/CLAUDE.md +0 -202
  14. package/template/.claude/commands/checkpoint.md +0 -127
  15. package/template/.claude/commands/cleanup.md +0 -307
  16. package/template/.claude/commands/execute-task.md +0 -440
  17. package/template/.claude/commands/exit-plan.md +0 -41
  18. package/template/.claude/commands/handoff.md +0 -97
  19. package/template/.claude/commands/learn.md +0 -218
  20. package/template/.claude/commands/metrics.md +0 -118
  21. package/template/.claude/commands/newworker.md +0 -162
  22. package/template/.claude/commands/nexttask.md +0 -67
  23. package/template/.claude/commands/prd.md +0 -238
  24. package/template/.claude/commands/reanchor.md +0 -51
  25. package/template/.claude/commands/remember.md +0 -126
  26. package/template/.claude/commands/run-project.md +0 -348
  27. package/template/.claude/commands/run.md +0 -110
  28. package/template/.claude/commands/search-reindex.md +0 -62
  29. package/template/.claude/commands/search.md +0 -100
  30. package/template/.claude/commands/setup.md +0 -381
  31. package/template/.claude/scripts/pure-ralph-loop.ps1 +0 -312
  32. package/template/.claude/scripts/pure-ralph-loop.sh +0 -859
  33. package/template/CHANGELOG.md +0 -220
  34. package/template/LICENSE +0 -21
  35. package/template/MIGRATION.md +0 -259
  36. package/template/README.md +0 -368
  37. package/template/data/journal/.gitkeep +0 -0
  38. package/template/docs/images/ascii-banner-options.md +0 -122
  39. package/template/docs/images/hq-banner.svg +0 -105
  40. package/template/knowledge/Ralph/01-overview.md +0 -71
  41. package/template/knowledge/Ralph/02-core-concepts.md +0 -114
  42. package/template/knowledge/Ralph/03-how-ralph-works.md +0 -184
  43. package/template/knowledge/Ralph/04-back-pressure.md +0 -222
  44. package/template/knowledge/Ralph/05-specifications.md +0 -210
  45. package/template/knowledge/Ralph/06-agents-md.md +0 -222
  46. package/template/knowledge/Ralph/07-implementation.md +0 -316
  47. package/template/knowledge/Ralph/08-economics.md +0 -182
  48. package/template/knowledge/Ralph/09-resources.md +0 -145
  49. package/template/knowledge/Ralph/10-claude-code-workflow.md +0 -212
  50. package/template/knowledge/Ralph/11-team-training-guide.md +0 -383
  51. package/template/knowledge/Ralph/README.md +0 -40
  52. package/template/knowledge/ai-security-framework/CONTRIBUTING.md +0 -139
  53. package/template/knowledge/ai-security-framework/GLOSSARY.md +0 -176
  54. package/template/knowledge/ai-security-framework/LICENSE +0 -21
  55. package/template/knowledge/ai-security-framework/QUICK-START.md +0 -172
  56. package/template/knowledge/ai-security-framework/README.md +0 -232
  57. package/template/knowledge/ai-security-framework/checklists/browser-security.md +0 -301
  58. package/template/knowledge/ai-security-framework/checklists/credential-isolation.md +0 -322
  59. package/template/knowledge/ai-security-framework/checklists/incident-response.md +0 -288
  60. package/template/knowledge/ai-security-framework/checklists/pre-flight.md +0 -249
  61. package/template/knowledge/ai-security-framework/checklists/weekly-audit.md +0 -159
  62. package/template/knowledge/ai-security-framework/configs/audit-logging.md +0 -372
  63. package/template/knowledge/ai-security-framework/configs/kill-switches.md +0 -354
  64. package/template/knowledge/ai-security-framework/docs/01-core-principles.md +0 -256
  65. package/template/knowledge/ai-security-framework/docs/02-threat-landscape.md +0 -326
  66. package/template/knowledge/ai-security-framework/docs/03-security-posture.md +0 -250
  67. package/template/knowledge/ai-security-framework/templates/agents-security.md +0 -233
  68. package/template/knowledge/design-styles/README.md +0 -42
  69. package/template/knowledge/design-styles/american-industrial.md +0 -136
  70. package/template/knowledge/design-styles/ethereal-abstract.md +0 -133
  71. package/template/knowledge/design-styles/liminal-portal.md +0 -111
  72. package/template/knowledge/design-styles/swipes/american-industrial/G-3m4YPW0AADdu2.jpeg +0 -0
  73. package/template/knowledge/design-styles/swipes/american-industrial/G-JJlt5WwAABK3K.png +0 -0
  74. package/template/knowledge/design-styles/swipes/american-industrial/G-JJmj5W0AEbJ-7.png +0 -0
  75. package/template/knowledge/design-styles/swipes/american-industrial/G59fgNuXkAAKLJQ (1).jpeg +0 -0
  76. package/template/knowledge/design-styles/swipes/american-industrial/G59fgNuXkAAKLJQ.jpeg +0 -0
  77. package/template/knowledge/design-styles/swipes/american-industrial/G7fVkn3WEAAM-ST.jpeg +0 -0
  78. package/template/knowledge/design-styles/swipes/american-industrial/G8ECO5JWEAIksyn.png +0 -0
  79. package/template/knowledge/design-styles/swipes/american-industrial/G9-3GQSWoAA8eqZ.png +0 -0
  80. package/template/knowledge/design-styles/swipes/american-industrial/G9xEOqrXkAEZRcs.png +0 -0
  81. package/template/knowledge/design-styles/swipes/american-industrial/G_MVeJrXQAA8sx4.jpeg +0 -0
  82. package/template/knowledge/design-styles/swipes/american-industrial/G_RSkmGXkAAgAVZ.png +0 -0
  83. package/template/knowledge/design-styles/swipes/american-industrial/README.md +0 -31
  84. package/template/knowledge/design-styles/swipes/american-industrial/qyqtg7Dq.png +0 -0
  85. package/template/knowledge/dev-team/README.md +0 -35
  86. package/template/knowledge/dev-team/patterns/README.md +0 -34
  87. package/template/knowledge/dev-team/patterns/frontend/react-best-practices.md +0 -178
  88. package/template/knowledge/dev-team/troubleshooting/README.md +0 -31
  89. package/template/knowledge/dev-team/workflows/README.md +0 -49
  90. package/template/knowledge/hq/checkpoint-schema.json +0 -51
  91. package/template/knowledge/hq/index-md-spec.md +0 -74
  92. package/template/knowledge/hq/thread-schema.md +0 -153
  93. package/template/knowledge/hq-core/checkpoint-schema.json +0 -51
  94. package/template/knowledge/hq-core/index-md-spec.md +0 -74
  95. package/template/knowledge/hq-core/thread-schema.md +0 -153
  96. package/template/knowledge/loom/README.md +0 -51
  97. package/template/knowledge/loom/architecture.md +0 -125
  98. package/template/knowledge/loom/code-style.md +0 -169
  99. package/template/knowledge/loom/llm-proxy.md +0 -132
  100. package/template/knowledge/loom/state-machine.md +0 -131
  101. package/template/knowledge/loom/thread-system.md +0 -117
  102. package/template/knowledge/loom/tools.md +0 -94
  103. package/template/knowledge/loom/weaver.md +0 -96
  104. package/template/knowledge/loom/web-frontend.md +0 -131
  105. package/template/knowledge/projects/README.md +0 -72
  106. package/template/knowledge/projects/templates/README.template.md +0 -28
  107. package/template/knowledge/workers/README.md +0 -195
  108. package/template/knowledge/workers/ralph-loop-pattern.md +0 -157
  109. package/template/knowledge/workers/skill-schema.md +0 -182
  110. package/template/knowledge/workers/state-machine.md +0 -102
  111. package/template/knowledge/workers/templates/base-worker.yaml +0 -73
  112. package/template/knowledge/workers/templates/code-worker.yaml +0 -85
  113. package/template/knowledge/workers/templates/skill.yaml +0 -49
  114. package/template/knowledge/workers/templates/social-worker.yaml +0 -70
  115. package/template/modules/examples/full-manifest.yaml +0 -92
  116. package/template/modules/examples/minimal.yaml +0 -14
  117. package/template/modules/modules.yaml +0 -59
  118. package/template/projects/.gitkeep +0 -0
  119. package/template/projects/incorporate-workers-into-pure-ralph/prd.json +0 -88
  120. package/template/projects/pure-ralph-branch-isolation/README.md +0 -114
  121. package/template/projects/pure-ralph-branch-isolation/prd.json +0 -123
  122. package/template/projects/purist-ralph-loop/README.md +0 -148
  123. package/template/projects/purist-ralph-loop/prd.json +0 -135
  124. package/template/projects/ralph-test/prd.json +0 -50
  125. package/template/prompts/pure-ralph-base.md +0 -551
  126. package/template/settings/.gitkeep +0 -0
  127. package/template/settings/pure-ralph.json +0 -42
  128. package/template/social-content/drafts/INDEX.md +0 -21
  129. package/template/social-content/drafts/linkedin/.gitkeep +0 -1
  130. package/template/social-content/drafts/x/.gitkeep +0 -1
  131. package/template/social-content/images/.gitkeep +0 -1
  132. package/template/starter-projects/code-worker/README.md +0 -97
  133. package/template/starter-projects/code-worker/prd.json +0 -45
  134. package/template/starter-projects/personal-assistant/README.md +0 -42
  135. package/template/starter-projects/personal-assistant/prd.json +0 -43
  136. package/template/starter-projects/social-media/README.md +0 -60
  137. package/template/starter-projects/social-media/prd.json +0 -43
  138. package/template/workers/content-brand/README.md +0 -59
  139. package/template/workers/content-brand/skills/messaging-alignment.md +0 -91
  140. package/template/workers/content-brand/skills/tone-check.md +0 -76
  141. package/template/workers/content-brand/skills/voice-analysis.md +0 -68
  142. package/template/workers/content-brand/worker.yaml +0 -81
  143. package/template/workers/content-legal/README.md +0 -80
  144. package/template/workers/content-legal/skills/claim-substantiation.md +0 -150
  145. package/template/workers/content-legal/skills/compliance-scan.md +0 -123
  146. package/template/workers/content-legal/skills/disclaimer-check.md +0 -146
  147. package/template/workers/content-legal/worker.yaml +0 -118
  148. package/template/workers/content-product/README.md +0 -77
  149. package/template/workers/content-product/skills/claim-verification.md +0 -96
  150. package/template/workers/content-product/skills/feature-accuracy.md +0 -117
  151. package/template/workers/content-product/skills/stats-check.md +0 -128
  152. package/template/workers/content-product/worker.yaml +0 -97
  153. package/template/workers/content-sales/README.md +0 -70
  154. package/template/workers/content-sales/skills/conversion-analysis.md +0 -96
  155. package/template/workers/content-sales/skills/cta-audit.md +0 -107
  156. package/template/workers/content-sales/skills/value-prop-check.md +0 -114
  157. package/template/workers/content-sales/worker.yaml +0 -93
  158. package/template/workers/content-shared/cli.ts +0 -242
  159. package/template/workers/content-shared/index.ts +0 -234
  160. package/template/workers/content-shared/lib/accuracy-analyzer.ts +0 -661
  161. package/template/workers/content-shared/lib/analyze.ts +0 -370
  162. package/template/workers/content-shared/lib/brand-analyzer.ts +0 -526
  163. package/template/workers/content-shared/lib/cms-integration.ts +0 -446
  164. package/template/workers/content-shared/lib/compliance-analyzer.ts +0 -655
  165. package/template/workers/content-shared/lib/conversion-analyzer.ts +0 -555
  166. package/template/workers/content-shared/lib/github-integration.ts +0 -582
  167. package/template/workers/content-shared/lib/output.ts +0 -373
  168. package/template/workers/content-shared/lib/parser.ts +0 -771
  169. package/template/workers/content-shared/lib/priority.ts +0 -439
  170. package/template/workers/content-shared/lib/recommendations.ts +0 -512
  171. package/template/workers/content-shared/lib/reporter.ts +0 -749
  172. package/template/workers/content-shared/lib/restructure.ts +0 -664
  173. package/template/workers/content-shared/lib/scorer.ts +0 -140
  174. package/template/workers/content-shared/lib/types.ts +0 -227
  175. package/template/workers/content-shared/lib/variants.ts +0 -595
  176. package/template/workers/content-shared/package.json +0 -51
  177. package/template/workers/content-shared/pnpm-lock.yaml +0 -39
  178. package/template/workers/content-shared/test/sample-page.json +0 -115
  179. package/template/workers/content-shared/tsconfig.json +0 -20
  180. package/template/workers/dev-team/README.md +0 -166
  181. package/template/workers/dev-team/_template.yaml +0 -70
  182. package/template/workers/dev-team/architect/package.json +0 -27
  183. package/template/workers/dev-team/architect/skills/api-design.md +0 -89
  184. package/template/workers/dev-team/architect/skills/refactor-plan.md +0 -96
  185. package/template/workers/dev-team/architect/skills/system-design.md +0 -100
  186. package/template/workers/dev-team/architect/src/index.ts +0 -49
  187. package/template/workers/dev-team/architect/src/mcp-server.ts +0 -122
  188. package/template/workers/dev-team/architect/src/skills/api-design.ts +0 -316
  189. package/template/workers/dev-team/architect/src/skills/refactor-plan.ts +0 -264
  190. package/template/workers/dev-team/architect/src/skills/system-design.ts +0 -212
  191. package/template/workers/dev-team/architect/tsconfig.json +0 -19
  192. package/template/workers/dev-team/architect/worker.yaml +0 -128
  193. package/template/workers/dev-team/backend-dev/package-lock.json +0 -1252
  194. package/template/workers/dev-team/backend-dev/package.json +0 -27
  195. package/template/workers/dev-team/backend-dev/skills/implement-endpoint.md +0 -70
  196. package/template/workers/dev-team/backend-dev/skills/implement-service.md +0 -62
  197. package/template/workers/dev-team/backend-dev/src/index.ts +0 -51
  198. package/template/workers/dev-team/backend-dev/src/mcp-server.ts +0 -109
  199. package/template/workers/dev-team/backend-dev/src/skills/implement-endpoint.ts +0 -122
  200. package/template/workers/dev-team/backend-dev/src/skills/implement-service.ts +0 -126
  201. package/template/workers/dev-team/backend-dev/tsconfig.json +0 -19
  202. package/template/workers/dev-team/backend-dev/worker.yaml +0 -128
  203. package/template/workers/dev-team/code-reviewer/package-lock.json +0 -1080
  204. package/template/workers/dev-team/code-reviewer/package.json +0 -24
  205. package/template/workers/dev-team/code-reviewer/skills/merge-to-production.md +0 -61
  206. package/template/workers/dev-team/code-reviewer/skills/merge-to-staging.md +0 -54
  207. package/template/workers/dev-team/code-reviewer/skills/request-changes.md +0 -63
  208. package/template/workers/dev-team/code-reviewer/skills/review-pr.md +0 -77
  209. package/template/workers/dev-team/code-reviewer/src/index.ts +0 -56
  210. package/template/workers/dev-team/code-reviewer/src/mcp-server.ts +0 -101
  211. package/template/workers/dev-team/code-reviewer/tsconfig.json +0 -19
  212. package/template/workers/dev-team/code-reviewer/worker.yaml +0 -90
  213. package/template/workers/dev-team/database-dev/package.json +0 -22
  214. package/template/workers/dev-team/database-dev/skills/create-schema.md +0 -48
  215. package/template/workers/dev-team/database-dev/src/index.ts +0 -50
  216. package/template/workers/dev-team/database-dev/src/mcp-server.ts +0 -76
  217. package/template/workers/dev-team/database-dev/tsconfig.json +0 -18
  218. package/template/workers/dev-team/database-dev/worker.yaml +0 -90
  219. package/template/workers/dev-team/frontend-dev/package.json +0 -22
  220. package/template/workers/dev-team/frontend-dev/skills/create-component.md +0 -26
  221. package/template/workers/dev-team/frontend-dev/src/index.ts +0 -50
  222. package/template/workers/dev-team/frontend-dev/src/mcp-server.ts +0 -77
  223. package/template/workers/dev-team/frontend-dev/tsconfig.json +0 -18
  224. package/template/workers/dev-team/frontend-dev/worker.yaml +0 -132
  225. package/template/workers/dev-team/infra-dev/package.json +0 -24
  226. package/template/workers/dev-team/infra-dev/skills/add-monitoring.md +0 -73
  227. package/template/workers/dev-team/infra-dev/skills/configure-deployment.md +0 -80
  228. package/template/workers/dev-team/infra-dev/skills/create-dockerfile.md +0 -62
  229. package/template/workers/dev-team/infra-dev/skills/setup-cicd.md +0 -63
  230. package/template/workers/dev-team/infra-dev/src/index.ts +0 -55
  231. package/template/workers/dev-team/infra-dev/src/mcp-server.ts +0 -82
  232. package/template/workers/dev-team/infra-dev/tsconfig.json +0 -19
  233. package/template/workers/dev-team/infra-dev/worker.yaml +0 -92
  234. package/template/workers/dev-team/knowledge-curator/package.json +0 -24
  235. package/template/workers/dev-team/knowledge-curator/skills/curate-troubleshooting.md +0 -63
  236. package/template/workers/dev-team/knowledge-curator/skills/process-learnings.md +0 -61
  237. package/template/workers/dev-team/knowledge-curator/skills/sync-documentation.md +0 -76
  238. package/template/workers/dev-team/knowledge-curator/skills/update-patterns.md +0 -63
  239. package/template/workers/dev-team/knowledge-curator/src/index.ts +0 -53
  240. package/template/workers/dev-team/knowledge-curator/src/mcp-server.ts +0 -92
  241. package/template/workers/dev-team/knowledge-curator/tsconfig.json +0 -19
  242. package/template/workers/dev-team/knowledge-curator/worker.yaml +0 -80
  243. package/template/workers/dev-team/motion-designer/package.json +0 -22
  244. package/template/workers/dev-team/motion-designer/skills/add-animation.md +0 -25
  245. package/template/workers/dev-team/motion-designer/skills/generate-image.md +0 -36
  246. package/template/workers/dev-team/motion-designer/src/index.ts +0 -63
  247. package/template/workers/dev-team/motion-designer/src/mcp-server.ts +0 -79
  248. package/template/workers/dev-team/motion-designer/tsconfig.json +0 -18
  249. package/template/workers/dev-team/motion-designer/worker.yaml +0 -84
  250. package/template/workers/dev-team/product-planner/queue.json +0 -4
  251. package/template/workers/dev-team/product-planner/worker.yaml +0 -220
  252. package/template/workers/dev-team/project-manager/package-lock.json +0 -1252
  253. package/template/workers/dev-team/project-manager/package.json +0 -27
  254. package/template/workers/dev-team/project-manager/skills/create-prd.md +0 -66
  255. package/template/workers/dev-team/project-manager/skills/next-issue.md +0 -51
  256. package/template/workers/dev-team/project-manager/skills/project-status.md +0 -59
  257. package/template/workers/dev-team/project-manager/skills/update-learnings.md +0 -65
  258. package/template/workers/dev-team/project-manager/src/index.ts +0 -54
  259. package/template/workers/dev-team/project-manager/src/mcp-server.ts +0 -207
  260. package/template/workers/dev-team/project-manager/src/skills/create-prd.ts +0 -86
  261. package/template/workers/dev-team/project-manager/src/skills/next-issue.ts +0 -137
  262. package/template/workers/dev-team/project-manager/src/skills/project-status.ts +0 -131
  263. package/template/workers/dev-team/project-manager/src/skills/update-learnings.ts +0 -94
  264. package/template/workers/dev-team/project-manager/tsconfig.json +0 -19
  265. package/template/workers/dev-team/project-manager/worker.yaml +0 -96
  266. package/template/workers/dev-team/qa-tester/package.json +0 -24
  267. package/template/workers/dev-team/qa-tester/skills/create-demo-account.md +0 -36
  268. package/template/workers/dev-team/qa-tester/skills/run-tests.md +0 -36
  269. package/template/workers/dev-team/qa-tester/skills/write-test.md +0 -27
  270. package/template/workers/dev-team/qa-tester/src/index.ts +0 -61
  271. package/template/workers/dev-team/qa-tester/src/mcp-server.ts +0 -88
  272. package/template/workers/dev-team/qa-tester/tsconfig.json +0 -18
  273. package/template/workers/dev-team/qa-tester/worker.yaml +0 -116
  274. package/template/workers/dev-team/task-executor/package-lock.json +0 -1252
  275. package/template/workers/dev-team/task-executor/package.json +0 -27
  276. package/template/workers/dev-team/task-executor/skills/analyze-issue.md +0 -101
  277. package/template/workers/dev-team/task-executor/skills/execute.md +0 -133
  278. package/template/workers/dev-team/task-executor/skills/report-learnings.md +0 -106
  279. package/template/workers/dev-team/task-executor/skills/validate-completion.md +0 -121
  280. package/template/workers/dev-team/task-executor/src/index.ts +0 -54
  281. package/template/workers/dev-team/task-executor/src/mcp-server.ts +0 -139
  282. package/template/workers/dev-team/task-executor/src/skills/analyze-issue.ts +0 -219
  283. package/template/workers/dev-team/task-executor/src/skills/execute.ts +0 -132
  284. package/template/workers/dev-team/task-executor/src/skills/report-learnings.ts +0 -119
  285. package/template/workers/dev-team/task-executor/src/skills/validate-completion.ts +0 -142
  286. package/template/workers/dev-team/task-executor/tsconfig.json +0 -19
  287. package/template/workers/dev-team/task-executor/worker.yaml +0 -110
  288. package/template/workers/registry.yaml +0 -171
  289. package/template/workers/security-scanner/README.md +0 -73
  290. package/template/workers/security-scanner/skills/pre-deploy-check.md +0 -205
  291. package/template/workers/security-scanner/worker.yaml +0 -26
  292. package/template/workspace/checkpoints/.gitkeep +0 -0
  293. package/template/workspace/content-ideas/inbox.jsonl +0 -0
  294. package/template/workspace/drafts/.gitkeep +0 -0
  295. package/template/workspace/learnings/.gitkeep +0 -3
  296. package/template/workspace/orchestrator/.gitkeep +0 -0
  297. package/template/workspace/ralph-test/COMPLETE.md +0 -18
  298. package/template/workspace/ralph-test/hello.txt +0 -2
  299. package/template/workspace/reports/.gitkeep +0 -0
  300. package/template/workspace/scratch/.gitkeep +0 -0
  301. 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
- }