create-hq 5.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.
- package/dist/deps.d.ts +4 -0
- package/dist/deps.d.ts.map +1 -0
- package/dist/deps.js +65 -0
- package/dist/deps.js.map +1 -0
- package/dist/git.d.ts +3 -0
- package/dist/git.d.ts.map +1 -0
- package/dist/git.js +19 -0
- package/dist/git.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +23 -0
- package/dist/index.js.map +1 -0
- package/dist/scaffold.d.ts +8 -0
- package/dist/scaffold.d.ts.map +1 -0
- package/dist/scaffold.js +130 -0
- package/dist/scaffold.js.map +1 -0
- package/dist/ui.d.ts +7 -0
- package/dist/ui.d.ts.map +1 -0
- package/dist/ui.js +36 -0
- package/dist/ui.js.map +1 -0
- package/package.json +41 -0
- package/template/.claude/CLAUDE.md +202 -0
- package/template/.claude/commands/checkpoint.md +127 -0
- package/template/.claude/commands/cleanup.md +307 -0
- package/template/.claude/commands/execute-task.md +440 -0
- package/template/.claude/commands/exit-plan.md +41 -0
- package/template/.claude/commands/handoff.md +97 -0
- package/template/.claude/commands/learn.md +218 -0
- package/template/.claude/commands/metrics.md +118 -0
- package/template/.claude/commands/newworker.md +162 -0
- package/template/.claude/commands/nexttask.md +67 -0
- package/template/.claude/commands/prd.md +238 -0
- package/template/.claude/commands/reanchor.md +51 -0
- package/template/.claude/commands/remember.md +126 -0
- package/template/.claude/commands/run-project.md +348 -0
- package/template/.claude/commands/run.md +110 -0
- package/template/.claude/commands/search-reindex.md +62 -0
- package/template/.claude/commands/search.md +100 -0
- package/template/.claude/commands/setup.md +381 -0
- package/template/.claude/scripts/pure-ralph-loop.ps1 +312 -0
- package/template/.claude/scripts/pure-ralph-loop.sh +859 -0
- package/template/CHANGELOG.md +220 -0
- package/template/LICENSE +21 -0
- package/template/MIGRATION.md +259 -0
- package/template/README.md +368 -0
- package/template/data/journal/.gitkeep +0 -0
- package/template/docs/images/ascii-banner-options.md +122 -0
- package/template/docs/images/hq-banner.svg +105 -0
- package/template/knowledge/Ralph/01-overview.md +71 -0
- package/template/knowledge/Ralph/02-core-concepts.md +114 -0
- package/template/knowledge/Ralph/03-how-ralph-works.md +184 -0
- package/template/knowledge/Ralph/04-back-pressure.md +222 -0
- package/template/knowledge/Ralph/05-specifications.md +210 -0
- package/template/knowledge/Ralph/06-agents-md.md +222 -0
- package/template/knowledge/Ralph/07-implementation.md +316 -0
- package/template/knowledge/Ralph/08-economics.md +182 -0
- package/template/knowledge/Ralph/09-resources.md +145 -0
- package/template/knowledge/Ralph/10-claude-code-workflow.md +212 -0
- package/template/knowledge/Ralph/11-team-training-guide.md +383 -0
- package/template/knowledge/Ralph/README.md +40 -0
- package/template/knowledge/ai-security-framework/CONTRIBUTING.md +139 -0
- package/template/knowledge/ai-security-framework/GLOSSARY.md +176 -0
- package/template/knowledge/ai-security-framework/LICENSE +21 -0
- package/template/knowledge/ai-security-framework/QUICK-START.md +172 -0
- package/template/knowledge/ai-security-framework/README.md +232 -0
- package/template/knowledge/ai-security-framework/checklists/browser-security.md +301 -0
- package/template/knowledge/ai-security-framework/checklists/credential-isolation.md +322 -0
- package/template/knowledge/ai-security-framework/checklists/incident-response.md +288 -0
- package/template/knowledge/ai-security-framework/checklists/pre-flight.md +249 -0
- package/template/knowledge/ai-security-framework/checklists/weekly-audit.md +159 -0
- package/template/knowledge/ai-security-framework/configs/audit-logging.md +372 -0
- package/template/knowledge/ai-security-framework/configs/kill-switches.md +354 -0
- package/template/knowledge/ai-security-framework/docs/01-core-principles.md +256 -0
- package/template/knowledge/ai-security-framework/docs/02-threat-landscape.md +326 -0
- package/template/knowledge/ai-security-framework/docs/03-security-posture.md +250 -0
- package/template/knowledge/ai-security-framework/templates/agents-security.md +233 -0
- package/template/knowledge/design-styles/README.md +42 -0
- package/template/knowledge/design-styles/american-industrial.md +136 -0
- package/template/knowledge/design-styles/ethereal-abstract.md +133 -0
- package/template/knowledge/design-styles/liminal-portal.md +111 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G-3m4YPW0AADdu2.jpeg +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G-JJlt5WwAABK3K.png +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G-JJmj5W0AEbJ-7.png +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G59fgNuXkAAKLJQ (1).jpeg +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G59fgNuXkAAKLJQ.jpeg +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G7fVkn3WEAAM-ST.jpeg +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G8ECO5JWEAIksyn.png +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G9-3GQSWoAA8eqZ.png +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G9xEOqrXkAEZRcs.png +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G_MVeJrXQAA8sx4.jpeg +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G_RSkmGXkAAgAVZ.png +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/README.md +31 -0
- package/template/knowledge/design-styles/swipes/american-industrial/qyqtg7Dq.png +0 -0
- package/template/knowledge/dev-team/README.md +35 -0
- package/template/knowledge/dev-team/patterns/README.md +34 -0
- package/template/knowledge/dev-team/patterns/frontend/react-best-practices.md +178 -0
- package/template/knowledge/dev-team/troubleshooting/README.md +31 -0
- package/template/knowledge/dev-team/workflows/README.md +49 -0
- package/template/knowledge/hq/checkpoint-schema.json +51 -0
- package/template/knowledge/hq/index-md-spec.md +74 -0
- package/template/knowledge/hq/thread-schema.md +153 -0
- package/template/knowledge/hq-core/checkpoint-schema.json +51 -0
- package/template/knowledge/hq-core/index-md-spec.md +74 -0
- package/template/knowledge/hq-core/thread-schema.md +153 -0
- package/template/knowledge/loom/README.md +51 -0
- package/template/knowledge/loom/architecture.md +125 -0
- package/template/knowledge/loom/code-style.md +169 -0
- package/template/knowledge/loom/llm-proxy.md +132 -0
- package/template/knowledge/loom/state-machine.md +131 -0
- package/template/knowledge/loom/thread-system.md +117 -0
- package/template/knowledge/loom/tools.md +94 -0
- package/template/knowledge/loom/weaver.md +96 -0
- package/template/knowledge/loom/web-frontend.md +131 -0
- package/template/knowledge/projects/README.md +72 -0
- package/template/knowledge/projects/templates/README.template.md +28 -0
- package/template/knowledge/workers/README.md +195 -0
- package/template/knowledge/workers/ralph-loop-pattern.md +157 -0
- package/template/knowledge/workers/skill-schema.md +182 -0
- package/template/knowledge/workers/state-machine.md +102 -0
- package/template/knowledge/workers/templates/base-worker.yaml +73 -0
- package/template/knowledge/workers/templates/code-worker.yaml +85 -0
- package/template/knowledge/workers/templates/skill.yaml +49 -0
- package/template/knowledge/workers/templates/social-worker.yaml +70 -0
- package/template/modules/examples/full-manifest.yaml +92 -0
- package/template/modules/examples/minimal.yaml +14 -0
- package/template/modules/modules.yaml +59 -0
- package/template/projects/.gitkeep +0 -0
- package/template/projects/incorporate-workers-into-pure-ralph/prd.json +88 -0
- package/template/projects/pure-ralph-branch-isolation/README.md +114 -0
- package/template/projects/pure-ralph-branch-isolation/prd.json +123 -0
- package/template/projects/purist-ralph-loop/README.md +148 -0
- package/template/projects/purist-ralph-loop/prd.json +135 -0
- package/template/projects/ralph-test/prd.json +50 -0
- package/template/prompts/pure-ralph-base.md +551 -0
- package/template/settings/.gitkeep +0 -0
- package/template/settings/pure-ralph.json +42 -0
- package/template/social-content/drafts/INDEX.md +21 -0
- package/template/social-content/drafts/linkedin/.gitkeep +1 -0
- package/template/social-content/drafts/x/.gitkeep +1 -0
- package/template/social-content/images/.gitkeep +1 -0
- package/template/starter-projects/code-worker/README.md +97 -0
- package/template/starter-projects/code-worker/prd.json +45 -0
- package/template/starter-projects/personal-assistant/README.md +42 -0
- package/template/starter-projects/personal-assistant/prd.json +43 -0
- package/template/starter-projects/social-media/README.md +60 -0
- package/template/starter-projects/social-media/prd.json +43 -0
- package/template/workers/content-brand/README.md +59 -0
- package/template/workers/content-brand/skills/messaging-alignment.md +91 -0
- package/template/workers/content-brand/skills/tone-check.md +76 -0
- package/template/workers/content-brand/skills/voice-analysis.md +68 -0
- package/template/workers/content-brand/worker.yaml +81 -0
- package/template/workers/content-legal/README.md +80 -0
- package/template/workers/content-legal/skills/claim-substantiation.md +150 -0
- package/template/workers/content-legal/skills/compliance-scan.md +123 -0
- package/template/workers/content-legal/skills/disclaimer-check.md +146 -0
- package/template/workers/content-legal/worker.yaml +118 -0
- package/template/workers/content-product/README.md +77 -0
- package/template/workers/content-product/skills/claim-verification.md +96 -0
- package/template/workers/content-product/skills/feature-accuracy.md +117 -0
- package/template/workers/content-product/skills/stats-check.md +128 -0
- package/template/workers/content-product/worker.yaml +97 -0
- package/template/workers/content-sales/README.md +70 -0
- package/template/workers/content-sales/skills/conversion-analysis.md +96 -0
- package/template/workers/content-sales/skills/cta-audit.md +107 -0
- package/template/workers/content-sales/skills/value-prop-check.md +114 -0
- package/template/workers/content-sales/worker.yaml +93 -0
- package/template/workers/content-shared/cli.ts +242 -0
- package/template/workers/content-shared/index.ts +234 -0
- package/template/workers/content-shared/lib/accuracy-analyzer.ts +661 -0
- package/template/workers/content-shared/lib/analyze.ts +370 -0
- package/template/workers/content-shared/lib/brand-analyzer.ts +526 -0
- package/template/workers/content-shared/lib/cms-integration.ts +446 -0
- package/template/workers/content-shared/lib/compliance-analyzer.ts +655 -0
- package/template/workers/content-shared/lib/conversion-analyzer.ts +555 -0
- package/template/workers/content-shared/lib/github-integration.ts +582 -0
- package/template/workers/content-shared/lib/output.ts +373 -0
- package/template/workers/content-shared/lib/parser.ts +771 -0
- package/template/workers/content-shared/lib/priority.ts +439 -0
- package/template/workers/content-shared/lib/recommendations.ts +512 -0
- package/template/workers/content-shared/lib/reporter.ts +749 -0
- package/template/workers/content-shared/lib/restructure.ts +664 -0
- package/template/workers/content-shared/lib/scorer.ts +140 -0
- package/template/workers/content-shared/lib/types.ts +227 -0
- package/template/workers/content-shared/lib/variants.ts +595 -0
- package/template/workers/content-shared/package.json +51 -0
- package/template/workers/content-shared/pnpm-lock.yaml +39 -0
- package/template/workers/content-shared/test/sample-page.json +115 -0
- package/template/workers/content-shared/tsconfig.json +20 -0
- package/template/workers/dev-team/README.md +166 -0
- package/template/workers/dev-team/_template.yaml +70 -0
- package/template/workers/dev-team/architect/package.json +27 -0
- package/template/workers/dev-team/architect/skills/api-design.md +89 -0
- package/template/workers/dev-team/architect/skills/refactor-plan.md +96 -0
- package/template/workers/dev-team/architect/skills/system-design.md +100 -0
- package/template/workers/dev-team/architect/src/index.ts +49 -0
- package/template/workers/dev-team/architect/src/mcp-server.ts +122 -0
- package/template/workers/dev-team/architect/src/skills/api-design.ts +316 -0
- package/template/workers/dev-team/architect/src/skills/refactor-plan.ts +264 -0
- package/template/workers/dev-team/architect/src/skills/system-design.ts +212 -0
- package/template/workers/dev-team/architect/tsconfig.json +19 -0
- package/template/workers/dev-team/architect/worker.yaml +128 -0
- package/template/workers/dev-team/backend-dev/package-lock.json +1252 -0
- package/template/workers/dev-team/backend-dev/package.json +27 -0
- package/template/workers/dev-team/backend-dev/skills/implement-endpoint.md +70 -0
- package/template/workers/dev-team/backend-dev/skills/implement-service.md +62 -0
- package/template/workers/dev-team/backend-dev/src/index.ts +51 -0
- package/template/workers/dev-team/backend-dev/src/mcp-server.ts +109 -0
- package/template/workers/dev-team/backend-dev/src/skills/implement-endpoint.ts +122 -0
- package/template/workers/dev-team/backend-dev/src/skills/implement-service.ts +126 -0
- package/template/workers/dev-team/backend-dev/tsconfig.json +19 -0
- package/template/workers/dev-team/backend-dev/worker.yaml +128 -0
- package/template/workers/dev-team/code-reviewer/package-lock.json +1080 -0
- package/template/workers/dev-team/code-reviewer/package.json +24 -0
- package/template/workers/dev-team/code-reviewer/skills/merge-to-production.md +61 -0
- package/template/workers/dev-team/code-reviewer/skills/merge-to-staging.md +54 -0
- package/template/workers/dev-team/code-reviewer/skills/request-changes.md +63 -0
- package/template/workers/dev-team/code-reviewer/skills/review-pr.md +77 -0
- package/template/workers/dev-team/code-reviewer/src/index.ts +56 -0
- package/template/workers/dev-team/code-reviewer/src/mcp-server.ts +101 -0
- package/template/workers/dev-team/code-reviewer/tsconfig.json +19 -0
- package/template/workers/dev-team/code-reviewer/worker.yaml +90 -0
- package/template/workers/dev-team/database-dev/package.json +22 -0
- package/template/workers/dev-team/database-dev/skills/create-schema.md +48 -0
- package/template/workers/dev-team/database-dev/src/index.ts +50 -0
- package/template/workers/dev-team/database-dev/src/mcp-server.ts +76 -0
- package/template/workers/dev-team/database-dev/tsconfig.json +18 -0
- package/template/workers/dev-team/database-dev/worker.yaml +90 -0
- package/template/workers/dev-team/frontend-dev/package.json +22 -0
- package/template/workers/dev-team/frontend-dev/skills/create-component.md +26 -0
- package/template/workers/dev-team/frontend-dev/src/index.ts +50 -0
- package/template/workers/dev-team/frontend-dev/src/mcp-server.ts +77 -0
- package/template/workers/dev-team/frontend-dev/tsconfig.json +18 -0
- package/template/workers/dev-team/frontend-dev/worker.yaml +132 -0
- package/template/workers/dev-team/infra-dev/package.json +24 -0
- package/template/workers/dev-team/infra-dev/skills/add-monitoring.md +73 -0
- package/template/workers/dev-team/infra-dev/skills/configure-deployment.md +80 -0
- package/template/workers/dev-team/infra-dev/skills/create-dockerfile.md +62 -0
- package/template/workers/dev-team/infra-dev/skills/setup-cicd.md +63 -0
- package/template/workers/dev-team/infra-dev/src/index.ts +55 -0
- package/template/workers/dev-team/infra-dev/src/mcp-server.ts +82 -0
- package/template/workers/dev-team/infra-dev/tsconfig.json +19 -0
- package/template/workers/dev-team/infra-dev/worker.yaml +92 -0
- package/template/workers/dev-team/knowledge-curator/package.json +24 -0
- package/template/workers/dev-team/knowledge-curator/skills/curate-troubleshooting.md +63 -0
- package/template/workers/dev-team/knowledge-curator/skills/process-learnings.md +61 -0
- package/template/workers/dev-team/knowledge-curator/skills/sync-documentation.md +76 -0
- package/template/workers/dev-team/knowledge-curator/skills/update-patterns.md +63 -0
- package/template/workers/dev-team/knowledge-curator/src/index.ts +53 -0
- package/template/workers/dev-team/knowledge-curator/src/mcp-server.ts +92 -0
- package/template/workers/dev-team/knowledge-curator/tsconfig.json +19 -0
- package/template/workers/dev-team/knowledge-curator/worker.yaml +80 -0
- package/template/workers/dev-team/motion-designer/package.json +22 -0
- package/template/workers/dev-team/motion-designer/skills/add-animation.md +25 -0
- package/template/workers/dev-team/motion-designer/skills/generate-image.md +36 -0
- package/template/workers/dev-team/motion-designer/src/index.ts +63 -0
- package/template/workers/dev-team/motion-designer/src/mcp-server.ts +79 -0
- package/template/workers/dev-team/motion-designer/tsconfig.json +18 -0
- package/template/workers/dev-team/motion-designer/worker.yaml +84 -0
- package/template/workers/dev-team/product-planner/queue.json +4 -0
- package/template/workers/dev-team/product-planner/worker.yaml +220 -0
- package/template/workers/dev-team/project-manager/package-lock.json +1252 -0
- package/template/workers/dev-team/project-manager/package.json +27 -0
- package/template/workers/dev-team/project-manager/skills/create-prd.md +66 -0
- package/template/workers/dev-team/project-manager/skills/next-issue.md +51 -0
- package/template/workers/dev-team/project-manager/skills/project-status.md +59 -0
- package/template/workers/dev-team/project-manager/skills/update-learnings.md +65 -0
- package/template/workers/dev-team/project-manager/src/index.ts +54 -0
- package/template/workers/dev-team/project-manager/src/mcp-server.ts +207 -0
- package/template/workers/dev-team/project-manager/src/skills/create-prd.ts +86 -0
- package/template/workers/dev-team/project-manager/src/skills/next-issue.ts +137 -0
- package/template/workers/dev-team/project-manager/src/skills/project-status.ts +131 -0
- package/template/workers/dev-team/project-manager/src/skills/update-learnings.ts +94 -0
- package/template/workers/dev-team/project-manager/tsconfig.json +19 -0
- package/template/workers/dev-team/project-manager/worker.yaml +96 -0
- package/template/workers/dev-team/qa-tester/package.json +24 -0
- package/template/workers/dev-team/qa-tester/skills/create-demo-account.md +36 -0
- package/template/workers/dev-team/qa-tester/skills/run-tests.md +36 -0
- package/template/workers/dev-team/qa-tester/skills/write-test.md +27 -0
- package/template/workers/dev-team/qa-tester/src/index.ts +61 -0
- package/template/workers/dev-team/qa-tester/src/mcp-server.ts +88 -0
- package/template/workers/dev-team/qa-tester/tsconfig.json +18 -0
- package/template/workers/dev-team/qa-tester/worker.yaml +116 -0
- package/template/workers/dev-team/task-executor/package-lock.json +1252 -0
- package/template/workers/dev-team/task-executor/package.json +27 -0
- package/template/workers/dev-team/task-executor/skills/analyze-issue.md +101 -0
- package/template/workers/dev-team/task-executor/skills/execute.md +133 -0
- package/template/workers/dev-team/task-executor/skills/report-learnings.md +106 -0
- package/template/workers/dev-team/task-executor/skills/validate-completion.md +121 -0
- package/template/workers/dev-team/task-executor/src/index.ts +54 -0
- package/template/workers/dev-team/task-executor/src/mcp-server.ts +139 -0
- package/template/workers/dev-team/task-executor/src/skills/analyze-issue.ts +219 -0
- package/template/workers/dev-team/task-executor/src/skills/execute.ts +132 -0
- package/template/workers/dev-team/task-executor/src/skills/report-learnings.ts +119 -0
- package/template/workers/dev-team/task-executor/src/skills/validate-completion.ts +142 -0
- package/template/workers/dev-team/task-executor/tsconfig.json +19 -0
- package/template/workers/dev-team/task-executor/worker.yaml +110 -0
- package/template/workers/registry.yaml +171 -0
- package/template/workers/security-scanner/README.md +73 -0
- package/template/workers/security-scanner/skills/pre-deploy-check.md +205 -0
- package/template/workers/security-scanner/worker.yaml +26 -0
- package/template/workspace/checkpoints/.gitkeep +0 -0
- package/template/workspace/content-ideas/inbox.jsonl +0 -0
- package/template/workspace/drafts/.gitkeep +0 -0
- package/template/workspace/learnings/.gitkeep +3 -0
- package/template/workspace/orchestrator/.gitkeep +0 -0
- package/template/workspace/ralph-test/COMPLETE.md +18 -0
- package/template/workspace/ralph-test/hello.txt +2 -0
- package/template/workspace/reports/.gitkeep +0 -0
- package/template/workspace/scratch/.gitkeep +0 -0
- package/template/workspace/threads/.gitkeep +3 -0
|
@@ -0,0 +1,512 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Recommendations Generator (US-011)
|
|
3
|
+
* Creates improvement recommendations with rationale from analysis findings
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import type {
|
|
7
|
+
FullAnalysis,
|
|
8
|
+
Finding,
|
|
9
|
+
Recommendation,
|
|
10
|
+
ContentSection,
|
|
11
|
+
AnalysisInput,
|
|
12
|
+
} from './types.js';
|
|
13
|
+
|
|
14
|
+
// ============================================
|
|
15
|
+
// Types
|
|
16
|
+
// ============================================
|
|
17
|
+
|
|
18
|
+
export interface Suggestion {
|
|
19
|
+
id: string;
|
|
20
|
+
type: 'text' | 'structure' | 'cta' | 'stat' | 'claim';
|
|
21
|
+
pageSlug: string;
|
|
22
|
+
sectionId?: string;
|
|
23
|
+
original: string;
|
|
24
|
+
suggested: string;
|
|
25
|
+
rationale: string;
|
|
26
|
+
source: 'brand' | 'sales' | 'product' | 'legal';
|
|
27
|
+
impact: 'high' | 'medium' | 'low';
|
|
28
|
+
effort: 'quick' | 'moderate' | 'significant';
|
|
29
|
+
priority: number; // Calculated from impact/effort
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
export interface SuggestionContext {
|
|
33
|
+
pageSlug: string;
|
|
34
|
+
sectionId?: string;
|
|
35
|
+
contentType?: ContentSection['type'];
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
// ============================================
|
|
39
|
+
// Impact/Effort Scoring
|
|
40
|
+
// ============================================
|
|
41
|
+
|
|
42
|
+
const IMPACT_SCORES: Record<Suggestion['impact'], number> = {
|
|
43
|
+
high: 3,
|
|
44
|
+
medium: 2,
|
|
45
|
+
low: 1,
|
|
46
|
+
};
|
|
47
|
+
|
|
48
|
+
const EFFORT_SCORES: Record<Suggestion['effort'], number> = {
|
|
49
|
+
quick: 3, // Lower effort = higher score (easier to do)
|
|
50
|
+
moderate: 2,
|
|
51
|
+
significant: 1,
|
|
52
|
+
};
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* Calculate priority score from impact and effort
|
|
56
|
+
* Higher score = do first (high impact + quick effort = best)
|
|
57
|
+
*/
|
|
58
|
+
function calculateSuggestionPriority(impact: Suggestion['impact'], effort: Suggestion['effort']): number {
|
|
59
|
+
return IMPACT_SCORES[impact] * EFFORT_SCORES[effort];
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
// ============================================
|
|
63
|
+
// Suggestion Generation
|
|
64
|
+
// ============================================
|
|
65
|
+
|
|
66
|
+
let suggestionCounter = 0;
|
|
67
|
+
|
|
68
|
+
function generateSuggestionId(): string {
|
|
69
|
+
suggestionCounter++;
|
|
70
|
+
return `sug-${Date.now()}-${suggestionCounter}`;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
/**
|
|
74
|
+
* Generate suggestions from full analysis results
|
|
75
|
+
*/
|
|
76
|
+
export function generateSuggestions(analysis: FullAnalysis): Suggestion[] {
|
|
77
|
+
const suggestions: Suggestion[] = [];
|
|
78
|
+
const pageSlug = analysis.page;
|
|
79
|
+
|
|
80
|
+
// Process brand findings
|
|
81
|
+
if (analysis.brand) {
|
|
82
|
+
suggestions.push(...convertFindingsToSuggestions(
|
|
83
|
+
analysis.brand.findings,
|
|
84
|
+
analysis.brand.recommendations,
|
|
85
|
+
pageSlug,
|
|
86
|
+
'brand'
|
|
87
|
+
));
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
// Process conversion findings
|
|
91
|
+
if (analysis.conversion) {
|
|
92
|
+
suggestions.push(...convertFindingsToSuggestions(
|
|
93
|
+
analysis.conversion.findings,
|
|
94
|
+
analysis.conversion.recommendations,
|
|
95
|
+
pageSlug,
|
|
96
|
+
'sales'
|
|
97
|
+
));
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
// Process accuracy findings
|
|
101
|
+
if (analysis.accuracy) {
|
|
102
|
+
suggestions.push(...convertFindingsToSuggestions(
|
|
103
|
+
analysis.accuracy.findings,
|
|
104
|
+
analysis.accuracy.recommendations,
|
|
105
|
+
pageSlug,
|
|
106
|
+
'product'
|
|
107
|
+
));
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
// Process compliance findings
|
|
111
|
+
if (analysis.compliance) {
|
|
112
|
+
suggestions.push(...convertFindingsToSuggestions(
|
|
113
|
+
analysis.compliance.findings,
|
|
114
|
+
analysis.compliance.recommendations,
|
|
115
|
+
pageSlug,
|
|
116
|
+
'legal'
|
|
117
|
+
));
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
// Sort by priority (highest first)
|
|
121
|
+
suggestions.sort((a, b) => b.priority - a.priority);
|
|
122
|
+
|
|
123
|
+
return suggestions;
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
/**
|
|
127
|
+
* Convert findings and recommendations to suggestions
|
|
128
|
+
*/
|
|
129
|
+
function convertFindingsToSuggestions(
|
|
130
|
+
findings: Finding[],
|
|
131
|
+
recommendations: Recommendation[],
|
|
132
|
+
pageSlug: string,
|
|
133
|
+
source: Suggestion['source']
|
|
134
|
+
): Suggestion[] {
|
|
135
|
+
const suggestions: Suggestion[] = [];
|
|
136
|
+
|
|
137
|
+
// Create suggestions from recommendations (these have current/suggested pairs)
|
|
138
|
+
for (const rec of recommendations) {
|
|
139
|
+
const impact = mapPriorityToImpact(rec.priority);
|
|
140
|
+
const effort = estimateEffort(rec.current, rec.suggested);
|
|
141
|
+
const type = inferSuggestionType(rec.category, rec.current);
|
|
142
|
+
|
|
143
|
+
suggestions.push({
|
|
144
|
+
id: generateSuggestionId(),
|
|
145
|
+
type,
|
|
146
|
+
pageSlug,
|
|
147
|
+
sectionId: extractSectionId(rec.current),
|
|
148
|
+
original: rec.current,
|
|
149
|
+
suggested: rec.suggested,
|
|
150
|
+
rationale: rec.rationale,
|
|
151
|
+
source,
|
|
152
|
+
impact,
|
|
153
|
+
effort,
|
|
154
|
+
priority: calculateSuggestionPriority(impact, effort),
|
|
155
|
+
});
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
// Create suggestions from critical/warning findings without recommendations
|
|
159
|
+
const recommendedOriginals = new Set(recommendations.map(r => r.current));
|
|
160
|
+
|
|
161
|
+
for (const finding of findings) {
|
|
162
|
+
if (finding.severity === 'pass' || finding.severity === 'info') continue;
|
|
163
|
+
if (finding.evidence && recommendedOriginals.has(finding.evidence)) continue;
|
|
164
|
+
|
|
165
|
+
const impact = finding.severity === 'critical' ? 'high' : 'medium';
|
|
166
|
+
const effort = estimateEffortFromMessage(finding.message);
|
|
167
|
+
const type = inferSuggestionTypeFromCategory(finding.category);
|
|
168
|
+
|
|
169
|
+
suggestions.push({
|
|
170
|
+
id: generateSuggestionId(),
|
|
171
|
+
type,
|
|
172
|
+
pageSlug,
|
|
173
|
+
sectionId: finding.location,
|
|
174
|
+
original: finding.evidence ?? finding.message,
|
|
175
|
+
suggested: generateSuggestedFix(finding),
|
|
176
|
+
rationale: finding.message,
|
|
177
|
+
source,
|
|
178
|
+
impact,
|
|
179
|
+
effort,
|
|
180
|
+
priority: calculateSuggestionPriority(impact, effort),
|
|
181
|
+
});
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
return suggestions;
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
// ============================================
|
|
188
|
+
// Suggestion Enhancement
|
|
189
|
+
// ============================================
|
|
190
|
+
|
|
191
|
+
/**
|
|
192
|
+
* Enhance a suggestion with AI-generated alternatives
|
|
193
|
+
* This provides a richer suggested text based on the type and context
|
|
194
|
+
*/
|
|
195
|
+
export function enhanceSuggestion(suggestion: Suggestion): Suggestion {
|
|
196
|
+
const enhanced = { ...suggestion };
|
|
197
|
+
|
|
198
|
+
// Generate improved suggestion based on type
|
|
199
|
+
switch (suggestion.type) {
|
|
200
|
+
case 'cta':
|
|
201
|
+
enhanced.suggested = enhanceCTA(suggestion.original, suggestion.source);
|
|
202
|
+
break;
|
|
203
|
+
case 'claim':
|
|
204
|
+
enhanced.suggested = enhanceClaim(suggestion.original, suggestion.source);
|
|
205
|
+
break;
|
|
206
|
+
case 'stat':
|
|
207
|
+
enhanced.suggested = enhanceStat(suggestion.original);
|
|
208
|
+
break;
|
|
209
|
+
case 'text':
|
|
210
|
+
enhanced.suggested = enhanceText(suggestion.original, suggestion.source);
|
|
211
|
+
break;
|
|
212
|
+
case 'structure':
|
|
213
|
+
// Structure suggestions typically come pre-formed
|
|
214
|
+
break;
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
// Add additional rationale if enhanced
|
|
218
|
+
if (enhanced.suggested !== suggestion.suggested) {
|
|
219
|
+
enhanced.rationale = `${suggestion.rationale} Enhanced for ${suggestion.source} optimization.`;
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
return enhanced;
|
|
223
|
+
}
|
|
224
|
+
|
|
225
|
+
/**
|
|
226
|
+
* Enhance CTA text
|
|
227
|
+
*/
|
|
228
|
+
function enhanceCTA(original: string, source: Suggestion['source']): string {
|
|
229
|
+
const lowerOriginal = original.toLowerCase();
|
|
230
|
+
|
|
231
|
+
// Generic CTAs to improve
|
|
232
|
+
if (lowerOriginal === 'learn more' || lowerOriginal === 'click here') {
|
|
233
|
+
if (source === 'sales') return 'See How It Works';
|
|
234
|
+
if (source === 'brand') return 'Discover Our Approach';
|
|
235
|
+
return 'Explore the Details';
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
if (lowerOriginal === 'contact us') {
|
|
239
|
+
if (source === 'sales') return 'Talk to an Expert';
|
|
240
|
+
return 'Get in Touch';
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
if (lowerOriginal === 'submit') {
|
|
244
|
+
return 'Send Message';
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
if (lowerOriginal === 'sign up') {
|
|
248
|
+
if (source === 'sales') return 'Start Free Trial';
|
|
249
|
+
return 'Create Your Account';
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
return original;
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
/**
|
|
256
|
+
* Enhance claim text for better substantiation
|
|
257
|
+
*/
|
|
258
|
+
function enhanceClaim(original: string, source: Suggestion['source']): string {
|
|
259
|
+
// Add specificity indicators
|
|
260
|
+
if (original.includes('best') && !original.includes('one of')) {
|
|
261
|
+
return original.replace(/\bbest\b/gi, 'one of the best');
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
if (original.includes('leading') && !original.includes('industry')) {
|
|
265
|
+
return original.replace(/\bleading\b/gi, 'industry-leading');
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
// Add hedging for absolute claims
|
|
269
|
+
if (original.includes('always') || original.includes('never')) {
|
|
270
|
+
return `In most cases, ${original.toLowerCase().replace(/\balways\b/gi, 'typically').replace(/\bnever\b/gi, 'rarely')}`;
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
return original;
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
/**
|
|
277
|
+
* Enhance stat text with context
|
|
278
|
+
*/
|
|
279
|
+
function enhanceStat(original: string): string {
|
|
280
|
+
// Add "up to" for unqualified percentages
|
|
281
|
+
if (/\d+%/.test(original) && !original.includes('up to') && !original.includes('over')) {
|
|
282
|
+
return original.replace(/(\d+%)/, 'up to $1');
|
|
283
|
+
}
|
|
284
|
+
|
|
285
|
+
return original;
|
|
286
|
+
}
|
|
287
|
+
|
|
288
|
+
/**
|
|
289
|
+
* Enhance general text
|
|
290
|
+
*/
|
|
291
|
+
function enhanceText(original: string, source: Suggestion['source']): string {
|
|
292
|
+
// Add brand voice elements
|
|
293
|
+
if (source === 'brand' && original.length > 50) {
|
|
294
|
+
// Break up long sentences
|
|
295
|
+
const sentences = original.split(/[.!?]+/).filter(s => s.trim());
|
|
296
|
+
if (sentences.length === 1 && sentences[0].split(' ').length > 25) {
|
|
297
|
+
// Suggest breaking into two sentences
|
|
298
|
+
const words = sentences[0].split(' ');
|
|
299
|
+
const midpoint = Math.floor(words.length / 2);
|
|
300
|
+
return words.slice(0, midpoint).join(' ') + '. ' + words.slice(midpoint).join(' ');
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
return original;
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
// ============================================
|
|
308
|
+
// Helper Functions
|
|
309
|
+
// ============================================
|
|
310
|
+
|
|
311
|
+
function mapPriorityToImpact(priority: Recommendation['priority']): Suggestion['impact'] {
|
|
312
|
+
return priority; // They use the same values
|
|
313
|
+
}
|
|
314
|
+
|
|
315
|
+
function estimateEffort(original: string, suggested: string): Suggestion['effort'] {
|
|
316
|
+
const originalWords = original.split(/\s+/).length;
|
|
317
|
+
const suggestedWords = suggested.split(/\s+/).length;
|
|
318
|
+
const wordDiff = Math.abs(originalWords - suggestedWords);
|
|
319
|
+
|
|
320
|
+
// Small text changes are quick
|
|
321
|
+
if (wordDiff <= 5 && originalWords <= 20) return 'quick';
|
|
322
|
+
|
|
323
|
+
// Medium changes
|
|
324
|
+
if (wordDiff <= 15 && originalWords <= 50) return 'moderate';
|
|
325
|
+
|
|
326
|
+
// Large changes
|
|
327
|
+
return 'significant';
|
|
328
|
+
}
|
|
329
|
+
|
|
330
|
+
function estimateEffortFromMessage(message: string): Suggestion['effort'] {
|
|
331
|
+
const lowerMessage = message.toLowerCase();
|
|
332
|
+
|
|
333
|
+
// Quick fixes
|
|
334
|
+
if (lowerMessage.includes('missing') || lowerMessage.includes('add')) return 'quick';
|
|
335
|
+
if (lowerMessage.includes('update') || lowerMessage.includes('change')) return 'moderate';
|
|
336
|
+
|
|
337
|
+
// Significant work
|
|
338
|
+
if (lowerMessage.includes('restructure') || lowerMessage.includes('rewrite')) return 'significant';
|
|
339
|
+
|
|
340
|
+
return 'moderate';
|
|
341
|
+
}
|
|
342
|
+
|
|
343
|
+
function inferSuggestionType(category: string, content: string): Suggestion['type'] {
|
|
344
|
+
const lowerCategory = category.toLowerCase();
|
|
345
|
+
const lowerContent = content.toLowerCase();
|
|
346
|
+
|
|
347
|
+
if (lowerCategory.includes('cta') || lowerCategory.includes('button')) return 'cta';
|
|
348
|
+
if (lowerCategory.includes('stat') || /\d+[%x]/.test(content)) return 'stat';
|
|
349
|
+
if (lowerCategory.includes('claim') || lowerCategory.includes('compliance')) return 'claim';
|
|
350
|
+
if (lowerCategory.includes('structure') || lowerCategory.includes('section')) return 'structure';
|
|
351
|
+
|
|
352
|
+
return 'text';
|
|
353
|
+
}
|
|
354
|
+
|
|
355
|
+
function inferSuggestionTypeFromCategory(category: string): Suggestion['type'] {
|
|
356
|
+
const lowerCategory = category.toLowerCase();
|
|
357
|
+
|
|
358
|
+
if (lowerCategory.includes('cta')) return 'cta';
|
|
359
|
+
if (lowerCategory.includes('stat') || lowerCategory.includes('accuracy')) return 'stat';
|
|
360
|
+
if (lowerCategory.includes('claim') || lowerCategory.includes('compliance')) return 'claim';
|
|
361
|
+
if (lowerCategory.includes('structure')) return 'structure';
|
|
362
|
+
|
|
363
|
+
return 'text';
|
|
364
|
+
}
|
|
365
|
+
|
|
366
|
+
function extractSectionId(content: string): string | undefined {
|
|
367
|
+
// Try to extract section reference from content
|
|
368
|
+
const sectionMatch = content.match(/section[:\s]+([a-z0-9-]+)/i);
|
|
369
|
+
if (sectionMatch) return sectionMatch[1];
|
|
370
|
+
|
|
371
|
+
const inMatch = content.match(/in\s+([a-z0-9-]+)\s+section/i);
|
|
372
|
+
if (inMatch) return inMatch[1];
|
|
373
|
+
|
|
374
|
+
return undefined;
|
|
375
|
+
}
|
|
376
|
+
|
|
377
|
+
function generateSuggestedFix(finding: Finding): string {
|
|
378
|
+
const { message, evidence, category } = finding;
|
|
379
|
+
|
|
380
|
+
// Generate context-appropriate suggestions
|
|
381
|
+
if (category.toLowerCase().includes('cta')) {
|
|
382
|
+
return 'Consider using action-oriented language with clear value proposition';
|
|
383
|
+
}
|
|
384
|
+
|
|
385
|
+
if (category.toLowerCase().includes('compliance')) {
|
|
386
|
+
return evidence
|
|
387
|
+
? `Add appropriate disclaimer or substantiation for: "${evidence}"`
|
|
388
|
+
: 'Review and add required disclaimers';
|
|
389
|
+
}
|
|
390
|
+
|
|
391
|
+
if (category.toLowerCase().includes('accuracy')) {
|
|
392
|
+
return evidence
|
|
393
|
+
? `Verify and update: "${evidence}"`
|
|
394
|
+
: 'Verify claims against current product data';
|
|
395
|
+
}
|
|
396
|
+
|
|
397
|
+
if (category.toLowerCase().includes('brand')) {
|
|
398
|
+
return 'Align with brand voice guidelines and approved terminology';
|
|
399
|
+
}
|
|
400
|
+
|
|
401
|
+
return `Address: ${message}`;
|
|
402
|
+
}
|
|
403
|
+
|
|
404
|
+
// ============================================
|
|
405
|
+
// Batch Processing
|
|
406
|
+
// ============================================
|
|
407
|
+
|
|
408
|
+
/**
|
|
409
|
+
* Generate suggestions for multiple pages
|
|
410
|
+
*/
|
|
411
|
+
export function generateSuggestionsForPages(analyses: FullAnalysis[]): Map<string, Suggestion[]> {
|
|
412
|
+
const suggestionsByPage = new Map<string, Suggestion[]>();
|
|
413
|
+
|
|
414
|
+
for (const analysis of analyses) {
|
|
415
|
+
const suggestions = generateSuggestions(analysis);
|
|
416
|
+
suggestionsByPage.set(analysis.page, suggestions);
|
|
417
|
+
}
|
|
418
|
+
|
|
419
|
+
return suggestionsByPage;
|
|
420
|
+
}
|
|
421
|
+
|
|
422
|
+
/**
|
|
423
|
+
* Get all suggestions sorted by priority across all pages
|
|
424
|
+
*/
|
|
425
|
+
export function getAllSuggestionsSorted(suggestionsByPage: Map<string, Suggestion[]>): Suggestion[] {
|
|
426
|
+
const all: Suggestion[] = [];
|
|
427
|
+
|
|
428
|
+
for (const suggestions of suggestionsByPage.values()) {
|
|
429
|
+
all.push(...suggestions);
|
|
430
|
+
}
|
|
431
|
+
|
|
432
|
+
return all.sort((a, b) => b.priority - a.priority);
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
// ============================================
|
|
436
|
+
// Formatting
|
|
437
|
+
// ============================================
|
|
438
|
+
|
|
439
|
+
/**
|
|
440
|
+
* Format a suggestion for display
|
|
441
|
+
*/
|
|
442
|
+
export function formatSuggestion(suggestion: Suggestion): string {
|
|
443
|
+
const impactEmoji = suggestion.impact === 'high' ? '[!]' : suggestion.impact === 'medium' ? '[*]' : '[-]';
|
|
444
|
+
const effortLabel = suggestion.effort === 'quick' ? 'Quick fix' : suggestion.effort === 'moderate' ? 'Moderate' : 'Significant';
|
|
445
|
+
|
|
446
|
+
return `${impactEmoji} [${suggestion.type.toUpperCase()}] ${suggestion.pageSlug}${suggestion.sectionId ? `#${suggestion.sectionId}` : ''}
|
|
447
|
+
Original: "${truncate(suggestion.original, 80)}"
|
|
448
|
+
Suggested: "${truncate(suggestion.suggested, 80)}"
|
|
449
|
+
Rationale: ${suggestion.rationale}
|
|
450
|
+
Impact: ${suggestion.impact} | Effort: ${effortLabel} | Priority: ${suggestion.priority}
|
|
451
|
+
Source: ${suggestion.source}`;
|
|
452
|
+
}
|
|
453
|
+
|
|
454
|
+
function truncate(text: string, maxLength: number): string {
|
|
455
|
+
if (text.length <= maxLength) return text;
|
|
456
|
+
return text.slice(0, maxLength - 3) + '...';
|
|
457
|
+
}
|
|
458
|
+
|
|
459
|
+
/**
|
|
460
|
+
* Format suggestions as markdown
|
|
461
|
+
*/
|
|
462
|
+
export function formatSuggestionsMarkdown(suggestions: Suggestion[]): string {
|
|
463
|
+
const lines: string[] = ['# Content Improvement Suggestions\n'];
|
|
464
|
+
|
|
465
|
+
// Group by impact
|
|
466
|
+
const highImpact = suggestions.filter(s => s.impact === 'high');
|
|
467
|
+
const mediumImpact = suggestions.filter(s => s.impact === 'medium');
|
|
468
|
+
const lowImpact = suggestions.filter(s => s.impact === 'low');
|
|
469
|
+
|
|
470
|
+
if (highImpact.length > 0) {
|
|
471
|
+
lines.push('## High Impact\n');
|
|
472
|
+
for (const s of highImpact) {
|
|
473
|
+
lines.push(formatSuggestionMarkdownItem(s));
|
|
474
|
+
}
|
|
475
|
+
}
|
|
476
|
+
|
|
477
|
+
if (mediumImpact.length > 0) {
|
|
478
|
+
lines.push('\n## Medium Impact\n');
|
|
479
|
+
for (const s of mediumImpact) {
|
|
480
|
+
lines.push(formatSuggestionMarkdownItem(s));
|
|
481
|
+
}
|
|
482
|
+
}
|
|
483
|
+
|
|
484
|
+
if (lowImpact.length > 0) {
|
|
485
|
+
lines.push('\n## Low Impact\n');
|
|
486
|
+
for (const s of lowImpact) {
|
|
487
|
+
lines.push(formatSuggestionMarkdownItem(s));
|
|
488
|
+
}
|
|
489
|
+
}
|
|
490
|
+
|
|
491
|
+
return lines.join('\n');
|
|
492
|
+
}
|
|
493
|
+
|
|
494
|
+
function formatSuggestionMarkdownItem(s: Suggestion): string {
|
|
495
|
+
const effortBadge = s.effort === 'quick' ? 'Quick Win' : s.effort === 'moderate' ? 'Moderate Effort' : 'Major Change';
|
|
496
|
+
|
|
497
|
+
return `### ${s.type.charAt(0).toUpperCase() + s.type.slice(1)}: ${s.pageSlug}
|
|
498
|
+
- **Location**: ${s.sectionId ?? 'Page-level'}
|
|
499
|
+
- **Effort**: ${effortBadge}
|
|
500
|
+
- **Source**: ${s.source}
|
|
501
|
+
|
|
502
|
+
**Current:**
|
|
503
|
+
> ${s.original}
|
|
504
|
+
|
|
505
|
+
**Suggested:**
|
|
506
|
+
> ${s.suggested}
|
|
507
|
+
|
|
508
|
+
**Rationale:** ${s.rationale}
|
|
509
|
+
|
|
510
|
+
---
|
|
511
|
+
`;
|
|
512
|
+
}
|