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