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