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,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
- }