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,582 +0,0 @@
1
- /**
2
- * GitHub Integration (US-016, US-017)
3
- * Create issues and PRs from content analysis suggestions
4
- */
5
-
6
- import type { Suggestion } from './recommendations.js';
7
- import type { FullAnalysis, PageContent } from './types.js';
8
-
9
- // ============================================
10
- // Types
11
- // ============================================
12
-
13
- export interface GitHubIssue {
14
- title: string;
15
- body: string;
16
- labels: string[];
17
- assignees?: string[];
18
- }
19
-
20
- export interface GitHubPR {
21
- title: string;
22
- body: string;
23
- branch: string;
24
- baseBranch: string;
25
- files: Array<{ path: string; content: string }>;
26
- }
27
-
28
- export interface GitHubConfig {
29
- owner: string;
30
- repo: string;
31
- baseBranch: string;
32
- issueLabels: string[];
33
- prLabels: string[];
34
- assignees?: string[];
35
- }
36
-
37
- export const DEFAULT_GITHUB_CONFIG: GitHubConfig = {
38
- owner: '',
39
- repo: '',
40
- baseBranch: 'main',
41
- issueLabels: ['content', 'automated'],
42
- prLabels: ['content-fix', 'automated'],
43
- assignees: [],
44
- };
45
-
46
- // ============================================
47
- // Issue Generation (US-016)
48
- // ============================================
49
-
50
- /**
51
- * Generate GitHub issue from a single suggestion
52
- */
53
- export function createIssueFromSuggestion(
54
- suggestion: Suggestion,
55
- config?: Partial<GitHubConfig>
56
- ): GitHubIssue {
57
- const cfg = { ...DEFAULT_GITHUB_CONFIG, ...config };
58
-
59
- const title = generateIssueTitle(suggestion);
60
- const body = formatIssueBody(suggestion);
61
- const labels = generateIssueLabels(suggestion, cfg.issueLabels);
62
-
63
- return {
64
- title,
65
- body,
66
- labels,
67
- assignees: cfg.assignees,
68
- };
69
- }
70
-
71
- /**
72
- * Generate batch of issues from full analysis
73
- */
74
- export function createIssuesFromAnalysis(
75
- analysis: FullAnalysis,
76
- suggestions: Suggestion[],
77
- maxIssues: number = 10,
78
- config?: Partial<GitHubConfig>
79
- ): GitHubIssue[] {
80
- // Sort by priority and take top N
81
- const sorted = [...suggestions].sort((a, b) => b.priority - a.priority);
82
- const topSuggestions = sorted.slice(0, maxIssues);
83
-
84
- return topSuggestions.map(s => createIssueFromSuggestion(s, config));
85
- }
86
-
87
- /**
88
- * Generate issue title from suggestion
89
- */
90
- function generateIssueTitle(suggestion: Suggestion): string {
91
- const typeLabel = formatTypeLabel(suggestion.type);
92
- const impactLabel = suggestion.impact === 'high' ? '[HIGH] ' : '';
93
-
94
- // Truncate if too long (GitHub has 256 char limit)
95
- const location = suggestion.sectionId
96
- ? `${suggestion.pageSlug}#${suggestion.sectionId}`
97
- : suggestion.pageSlug;
98
-
99
- let title = `${impactLabel}${typeLabel}: ${location}`;
100
-
101
- // Add brief description
102
- const desc = suggestion.rationale.split('.')[0];
103
- if (title.length + desc.length < 200) {
104
- title += ` - ${desc}`;
105
- }
106
-
107
- return title.slice(0, 250);
108
- }
109
-
110
- /**
111
- * Format issue body with context
112
- */
113
- export function formatIssueBody(suggestion: Suggestion): string {
114
- const lines: string[] = [];
115
-
116
- // Overview
117
- lines.push('## Overview');
118
- lines.push('');
119
- lines.push(`**Type:** ${formatTypeLabel(suggestion.type)}`);
120
- lines.push(`**Page:** \`${suggestion.pageSlug}\``);
121
- if (suggestion.sectionId) {
122
- lines.push(`**Section:** \`${suggestion.sectionId}\``);
123
- }
124
- lines.push(`**Source:** ${formatSourceLabel(suggestion.source)}`);
125
- lines.push(`**Impact:** ${suggestion.impact}`);
126
- lines.push(`**Effort:** ${formatEffortLabel(suggestion.effort)}`);
127
- lines.push('');
128
-
129
- // Current State
130
- lines.push('## Current Content');
131
- lines.push('');
132
- lines.push('```');
133
- lines.push(suggestion.original);
134
- lines.push('```');
135
- lines.push('');
136
-
137
- // Suggested Change
138
- lines.push('## Suggested Change');
139
- lines.push('');
140
- lines.push('```');
141
- lines.push(suggestion.suggested);
142
- lines.push('```');
143
- lines.push('');
144
-
145
- // Rationale
146
- lines.push('## Rationale');
147
- lines.push('');
148
- lines.push(suggestion.rationale);
149
- lines.push('');
150
-
151
- // Metadata
152
- lines.push('---');
153
- lines.push('');
154
- lines.push('*This issue was automatically generated by the content analysis worker.*');
155
- lines.push(`*Suggestion ID: \`${suggestion.id}\`*`);
156
-
157
- return lines.join('\n');
158
- }
159
-
160
- /**
161
- * Generate issue labels based on suggestion properties
162
- */
163
- function generateIssueLabels(
164
- suggestion: Suggestion,
165
- baseLabels: string[]
166
- ): string[] {
167
- const labels = [...baseLabels];
168
-
169
- // Add type-specific label
170
- labels.push(`type:${suggestion.type}`);
171
-
172
- // Add source label
173
- labels.push(`source:${suggestion.source}`);
174
-
175
- // Add priority label
176
- if (suggestion.impact === 'high') {
177
- labels.push('priority:high');
178
- } else if (suggestion.impact === 'medium') {
179
- labels.push('priority:medium');
180
- } else {
181
- labels.push('priority:low');
182
- }
183
-
184
- // Add effort label
185
- if (suggestion.effort === 'quick') {
186
- labels.push('good-first-issue');
187
- }
188
-
189
- return labels;
190
- }
191
-
192
- // ============================================
193
- // PR Generation (US-017)
194
- // ============================================
195
-
196
- /**
197
- * Generate PR from accepted suggestions
198
- */
199
- export function createPRFromSuggestions(
200
- suggestions: Suggestion[],
201
- pageContents: Map<string, { path: string; content: string }>,
202
- config?: Partial<GitHubConfig>
203
- ): GitHubPR {
204
- const cfg = { ...DEFAULT_GITHUB_CONFIG, ...config };
205
-
206
- // Group suggestions by page
207
- const suggestionsByPage = new Map<string, Suggestion[]>();
208
- for (const suggestion of suggestions) {
209
- const existing = suggestionsByPage.get(suggestion.pageSlug) ?? [];
210
- existing.push(suggestion);
211
- suggestionsByPage.set(suggestion.pageSlug, existing);
212
- }
213
-
214
- // Apply suggestions to each page
215
- const files: Array<{ path: string; content: string }> = [];
216
- for (const [pageSlug, pageSuggestions] of suggestionsByPage) {
217
- const pageInfo = pageContents.get(pageSlug);
218
- if (!pageInfo) continue;
219
-
220
- let updatedContent = pageInfo.content;
221
- for (const suggestion of pageSuggestions) {
222
- updatedContent = applySuggestionToContent(suggestion, updatedContent);
223
- }
224
-
225
- files.push({
226
- path: pageInfo.path,
227
- content: updatedContent,
228
- });
229
- }
230
-
231
- // Generate PR metadata
232
- const title = generatePRTitle(suggestions);
233
- const body = formatPRBody(suggestions, suggestionsByPage);
234
- const branch = generateBranchName(suggestions);
235
-
236
- return {
237
- title,
238
- body,
239
- branch,
240
- baseBranch: cfg.baseBranch,
241
- files,
242
- };
243
- }
244
-
245
- /**
246
- * Apply a single suggestion to content
247
- * Uses simple string replacement
248
- */
249
- export function applySuggestionToContent(
250
- suggestion: Suggestion,
251
- content: string
252
- ): string {
253
- // For simple text replacements
254
- if (content.includes(suggestion.original)) {
255
- return content.replace(suggestion.original, suggestion.suggested);
256
- }
257
-
258
- // Try case-insensitive match
259
- const regex = new RegExp(escapeRegExp(suggestion.original), 'gi');
260
- if (regex.test(content)) {
261
- return content.replace(regex, suggestion.suggested);
262
- }
263
-
264
- // No match found, return unchanged
265
- return content;
266
- }
267
-
268
- /**
269
- * Escape special regex characters
270
- */
271
- function escapeRegExp(str: string): string {
272
- return str.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
273
- }
274
-
275
- /**
276
- * Generate commit message for suggestions
277
- */
278
- export function generateCommitMessage(suggestions: Suggestion[]): string {
279
- if (suggestions.length === 0) {
280
- return 'chore: no content changes';
281
- }
282
-
283
- if (suggestions.length === 1) {
284
- const s = suggestions[0];
285
- return `content(${s.pageSlug}): ${formatTypeLabel(s.type).toLowerCase()} update
286
-
287
- ${s.rationale}`;
288
- }
289
-
290
- // Multiple suggestions
291
- const pages = [...new Set(suggestions.map(s => s.pageSlug))];
292
- const types = [...new Set(suggestions.map(s => s.type))];
293
-
294
- let message = `content: update ${suggestions.length} items across ${pages.length} page(s)\n\n`;
295
-
296
- // Group by page
297
- for (const page of pages) {
298
- const pageSuggestions = suggestions.filter(s => s.pageSlug === page);
299
- message += `## ${page}\n`;
300
- for (const s of pageSuggestions) {
301
- message += `- ${formatTypeLabel(s.type)}: ${s.rationale.split('.')[0]}\n`;
302
- }
303
- message += '\n';
304
- }
305
-
306
- return message;
307
- }
308
-
309
- /**
310
- * Generate PR title
311
- */
312
- function generatePRTitle(suggestions: Suggestion[]): string {
313
- if (suggestions.length === 1) {
314
- const s = suggestions[0];
315
- return `[Content] ${formatTypeLabel(s.type)} fix for ${s.pageSlug}`;
316
- }
317
-
318
- const pages = [...new Set(suggestions.map(s => s.pageSlug))];
319
- const highImpact = suggestions.filter(s => s.impact === 'high').length;
320
-
321
- let title = `[Content] ${suggestions.length} improvements`;
322
- if (pages.length === 1) {
323
- title += ` for ${pages[0]}`;
324
- } else {
325
- title += ` across ${pages.length} pages`;
326
- }
327
-
328
- if (highImpact > 0) {
329
- title += ` (${highImpact} high priority)`;
330
- }
331
-
332
- return title;
333
- }
334
-
335
- /**
336
- * Format PR body
337
- */
338
- function formatPRBody(
339
- suggestions: Suggestion[],
340
- suggestionsByPage: Map<string, Suggestion[]>
341
- ): string {
342
- const lines: string[] = [];
343
-
344
- lines.push('## Summary');
345
- lines.push('');
346
- lines.push(`This PR applies **${suggestions.length} content improvements** identified by automated analysis.`);
347
- lines.push('');
348
-
349
- // Stats
350
- const highImpact = suggestions.filter(s => s.impact === 'high').length;
351
- const mediumImpact = suggestions.filter(s => s.impact === 'medium').length;
352
- const lowImpact = suggestions.filter(s => s.impact === 'low').length;
353
-
354
- lines.push('### Impact Breakdown');
355
- lines.push('');
356
- lines.push(`- High: ${highImpact}`);
357
- lines.push(`- Medium: ${mediumImpact}`);
358
- lines.push(`- Low: ${lowImpact}`);
359
- lines.push('');
360
-
361
- // Changes by page
362
- lines.push('## Changes by Page');
363
- lines.push('');
364
-
365
- for (const [pageSlug, pageSuggestions] of suggestionsByPage) {
366
- lines.push(`### ${pageSlug}`);
367
- lines.push('');
368
-
369
- for (const s of pageSuggestions) {
370
- lines.push(`#### ${formatTypeLabel(s.type)}`);
371
- lines.push('');
372
- lines.push('**Before:**');
373
- lines.push('```');
374
- lines.push(truncate(s.original, 200));
375
- lines.push('```');
376
- lines.push('');
377
- lines.push('**After:**');
378
- lines.push('```');
379
- lines.push(truncate(s.suggested, 200));
380
- lines.push('```');
381
- lines.push('');
382
- lines.push(`_${s.rationale}_`);
383
- lines.push('');
384
- }
385
- }
386
-
387
- // Test plan
388
- lines.push('## Test Plan');
389
- lines.push('');
390
- lines.push('- [ ] Review each content change for accuracy');
391
- lines.push('- [ ] Verify formatting is preserved');
392
- lines.push('- [ ] Check page renders correctly');
393
- lines.push('- [ ] Confirm no unintended changes');
394
- lines.push('');
395
-
396
- // Footer
397
- lines.push('---');
398
- lines.push('');
399
- lines.push('*This PR was automatically generated by the content analysis worker.*');
400
-
401
- return lines.join('\n');
402
- }
403
-
404
- /**
405
- * Generate branch name
406
- */
407
- function generateBranchName(suggestions: Suggestion[]): string {
408
- const timestamp = new Date().toISOString().split('T')[0].replace(/-/g, '');
409
-
410
- if (suggestions.length === 1) {
411
- const s = suggestions[0];
412
- const slug = s.pageSlug.replace(/[^a-z0-9]/gi, '-').toLowerCase();
413
- return `content/${timestamp}-${slug}-${s.type}`;
414
- }
415
-
416
- const pages = [...new Set(suggestions.map(s => s.pageSlug))];
417
- if (pages.length === 1) {
418
- const slug = pages[0].replace(/[^a-z0-9]/gi, '-').toLowerCase();
419
- return `content/${timestamp}-${slug}-fixes`;
420
- }
421
-
422
- return `content/${timestamp}-multi-page-fixes`;
423
- }
424
-
425
- // ============================================
426
- // GitHub CLI Commands
427
- // ============================================
428
-
429
- /**
430
- * Generate gh CLI command to create an issue
431
- */
432
- export function generateGitHubIssueCommand(
433
- issue: GitHubIssue,
434
- config?: Partial<GitHubConfig>
435
- ): string {
436
- const cfg = { ...DEFAULT_GITHUB_CONFIG, ...config };
437
-
438
- const parts = ['gh issue create'];
439
-
440
- // Title
441
- parts.push(`--title "${escapeShell(issue.title)}"`);
442
-
443
- // Body (use heredoc for multi-line)
444
- parts.push(`--body "${escapeShell(issue.body)}"`);
445
-
446
- // Labels
447
- if (issue.labels.length > 0) {
448
- parts.push(`--label "${issue.labels.join(',')}"`);
449
- }
450
-
451
- // Assignees
452
- if (issue.assignees && issue.assignees.length > 0) {
453
- parts.push(`--assignee "${issue.assignees.join(',')}"`);
454
- }
455
-
456
- // Repo (if specified)
457
- if (cfg.owner && cfg.repo) {
458
- parts.push(`--repo ${cfg.owner}/${cfg.repo}`);
459
- }
460
-
461
- return parts.join(' \\\n ');
462
- }
463
-
464
- /**
465
- * Generate gh CLI command to create a PR
466
- */
467
- export function generateGitHubPRCommand(
468
- pr: GitHubPR,
469
- config?: Partial<GitHubConfig>
470
- ): string {
471
- const cfg = { ...DEFAULT_GITHUB_CONFIG, ...config };
472
-
473
- const parts = ['gh pr create'];
474
-
475
- // Title
476
- parts.push(`--title "${escapeShell(pr.title)}"`);
477
-
478
- // Body
479
- parts.push(`--body "${escapeShell(pr.body)}"`);
480
-
481
- // Base branch
482
- parts.push(`--base ${pr.baseBranch}`);
483
-
484
- // Head branch
485
- parts.push(`--head ${pr.branch}`);
486
-
487
- // Labels
488
- if (cfg.prLabels.length > 0) {
489
- parts.push(`--label "${cfg.prLabels.join(',')}"`);
490
- }
491
-
492
- // Repo
493
- if (cfg.owner && cfg.repo) {
494
- parts.push(`--repo ${cfg.owner}/${cfg.repo}`);
495
- }
496
-
497
- return parts.join(' \\\n ');
498
- }
499
-
500
- /**
501
- * Escape shell special characters
502
- */
503
- function escapeShell(str: string): string {
504
- return str
505
- .replace(/\\/g, '\\\\')
506
- .replace(/"/g, '\\"')
507
- .replace(/\$/g, '\\$')
508
- .replace(/`/g, '\\`');
509
- }
510
-
511
- // ============================================
512
- // Batch Processing
513
- // ============================================
514
-
515
- /**
516
- * Create issues for all high-impact suggestions
517
- */
518
- export function createHighImpactIssues(
519
- suggestions: Suggestion[],
520
- config?: Partial<GitHubConfig>
521
- ): GitHubIssue[] {
522
- const highImpact = suggestions.filter(s => s.impact === 'high');
523
- return highImpact.map(s => createIssueFromSuggestion(s, config));
524
- }
525
-
526
- /**
527
- * Create issues grouped by page
528
- */
529
- export function createIssuesByPage(
530
- suggestions: Suggestion[],
531
- config?: Partial<GitHubConfig>
532
- ): Map<string, GitHubIssue[]> {
533
- const issuesByPage = new Map<string, GitHubIssue[]>();
534
-
535
- for (const suggestion of suggestions) {
536
- const issue = createIssueFromSuggestion(suggestion, config);
537
- const existing = issuesByPage.get(suggestion.pageSlug) ?? [];
538
- existing.push(issue);
539
- issuesByPage.set(suggestion.pageSlug, existing);
540
- }
541
-
542
- return issuesByPage;
543
- }
544
-
545
- // ============================================
546
- // Helper Functions
547
- // ============================================
548
-
549
- function formatTypeLabel(type: Suggestion['type']): string {
550
- const labels: Record<Suggestion['type'], string> = {
551
- text: 'Text',
552
- structure: 'Structure',
553
- cta: 'CTA',
554
- stat: 'Statistic',
555
- claim: 'Claim',
556
- };
557
- return labels[type] ?? type;
558
- }
559
-
560
- function formatSourceLabel(source: Suggestion['source']): string {
561
- const labels: Record<Suggestion['source'], string> = {
562
- brand: 'Brand Voice',
563
- sales: 'Conversion',
564
- product: 'Product Accuracy',
565
- legal: 'Compliance',
566
- };
567
- return labels[source] ?? source;
568
- }
569
-
570
- function formatEffortLabel(effort: Suggestion['effort']): string {
571
- const labels: Record<Suggestion['effort'], string> = {
572
- quick: 'Quick Fix',
573
- moderate: 'Moderate',
574
- significant: 'Significant',
575
- };
576
- return labels[effort] ?? effort;
577
- }
578
-
579
- function truncate(text: string, maxLen: number): string {
580
- if (text.length <= maxLen) return text;
581
- return text.slice(0, maxLen - 3) + '...';
582
- }