@rankcli/agent-runtime 0.0.1
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/README.md +242 -0
- package/dist/analyzer-2CSWIQGD.mjs +6 -0
- package/dist/chunk-YNZYHEYM.mjs +774 -0
- package/dist/index.d.mts +4012 -0
- package/dist/index.d.ts +4012 -0
- package/dist/index.js +29672 -0
- package/dist/index.mjs +28602 -0
- package/package.json +53 -0
- package/scripts/build-deno.ts +134 -0
- package/src/audit/ai/analyzer.ts +347 -0
- package/src/audit/ai/index.ts +29 -0
- package/src/audit/ai/prompts/content-analysis.ts +271 -0
- package/src/audit/ai/types.ts +179 -0
- package/src/audit/checks/additional-checks.ts +439 -0
- package/src/audit/checks/ai-citation-worthiness.ts +399 -0
- package/src/audit/checks/ai-content-structure.ts +325 -0
- package/src/audit/checks/ai-readiness.ts +339 -0
- package/src/audit/checks/anchor-text.ts +179 -0
- package/src/audit/checks/answer-conciseness.ts +322 -0
- package/src/audit/checks/asset-minification.ts +270 -0
- package/src/audit/checks/bing-optimization.ts +206 -0
- package/src/audit/checks/brand-mention-optimization.ts +349 -0
- package/src/audit/checks/caching-headers.ts +305 -0
- package/src/audit/checks/canonical-advanced.ts +150 -0
- package/src/audit/checks/canonical-domain.ts +196 -0
- package/src/audit/checks/citation-quality.ts +358 -0
- package/src/audit/checks/client-rendering.ts +542 -0
- package/src/audit/checks/color-contrast.ts +342 -0
- package/src/audit/checks/content-freshness.ts +170 -0
- package/src/audit/checks/content-science.ts +589 -0
- package/src/audit/checks/conversion-elements.ts +526 -0
- package/src/audit/checks/crawlability.ts +220 -0
- package/src/audit/checks/directory-listing.ts +172 -0
- package/src/audit/checks/dom-analysis.ts +191 -0
- package/src/audit/checks/dom-size.ts +246 -0
- package/src/audit/checks/duplicate-content.ts +194 -0
- package/src/audit/checks/eeat-signals.ts +990 -0
- package/src/audit/checks/entity-seo.ts +396 -0
- package/src/audit/checks/featured-snippet.ts +473 -0
- package/src/audit/checks/freshness-signals.ts +443 -0
- package/src/audit/checks/funnel-intent.ts +463 -0
- package/src/audit/checks/hreflang.ts +174 -0
- package/src/audit/checks/html-compliance.ts +302 -0
- package/src/audit/checks/image-dimensions.ts +167 -0
- package/src/audit/checks/images.ts +160 -0
- package/src/audit/checks/indexnow.ts +275 -0
- package/src/audit/checks/interactive-tools.ts +475 -0
- package/src/audit/checks/internal-link-graph.ts +436 -0
- package/src/audit/checks/keyword-analysis.ts +239 -0
- package/src/audit/checks/keyword-cannibalization.ts +385 -0
- package/src/audit/checks/keyword-placement.ts +471 -0
- package/src/audit/checks/links.ts +203 -0
- package/src/audit/checks/llms-txt.ts +224 -0
- package/src/audit/checks/local-seo.ts +296 -0
- package/src/audit/checks/mobile.ts +167 -0
- package/src/audit/checks/modern-images.ts +226 -0
- package/src/audit/checks/navboost-signals.ts +395 -0
- package/src/audit/checks/on-page.ts +209 -0
- package/src/audit/checks/page-resources.ts +285 -0
- package/src/audit/checks/pagination.ts +180 -0
- package/src/audit/checks/performance.ts +153 -0
- package/src/audit/checks/platform-presence.ts +580 -0
- package/src/audit/checks/redirect-analysis.ts +153 -0
- package/src/audit/checks/redirect-chain.ts +389 -0
- package/src/audit/checks/resource-hints.ts +420 -0
- package/src/audit/checks/responsive-css.ts +247 -0
- package/src/audit/checks/responsive-images.ts +396 -0
- package/src/audit/checks/review-ecosystem.ts +415 -0
- package/src/audit/checks/robots-validation.ts +373 -0
- package/src/audit/checks/security-headers.ts +172 -0
- package/src/audit/checks/security.ts +144 -0
- package/src/audit/checks/serp-preview.ts +251 -0
- package/src/audit/checks/site-maturity.ts +444 -0
- package/src/audit/checks/social-meta.test.ts +275 -0
- package/src/audit/checks/social-meta.ts +134 -0
- package/src/audit/checks/soft-404.ts +151 -0
- package/src/audit/checks/structured-data.ts +238 -0
- package/src/audit/checks/tech-detection.ts +496 -0
- package/src/audit/checks/topical-clusters.ts +435 -0
- package/src/audit/checks/tracker-bloat.ts +462 -0
- package/src/audit/checks/tracking-verification.test.ts +371 -0
- package/src/audit/checks/tracking-verification.ts +636 -0
- package/src/audit/checks/url-safety.ts +682 -0
- package/src/audit/deno-entry.ts +66 -0
- package/src/audit/discovery/index.ts +15 -0
- package/src/audit/discovery/link-crawler.ts +232 -0
- package/src/audit/discovery/repo-routes.ts +347 -0
- package/src/audit/engine.ts +620 -0
- package/src/audit/fixes/index.ts +209 -0
- package/src/audit/fixes/social-meta-fixes.test.ts +329 -0
- package/src/audit/fixes/social-meta-fixes.ts +463 -0
- package/src/audit/index.ts +74 -0
- package/src/audit/runner.test.ts +299 -0
- package/src/audit/runner.ts +130 -0
- package/src/audit/types.ts +1953 -0
- package/src/content/featured-snippet.ts +367 -0
- package/src/content/generator.test.ts +534 -0
- package/src/content/generator.ts +501 -0
- package/src/content/headline.ts +317 -0
- package/src/content/index.ts +62 -0
- package/src/content/intent.ts +258 -0
- package/src/content/keyword-density.ts +349 -0
- package/src/content/readability.ts +262 -0
- package/src/executor.ts +336 -0
- package/src/fixer.ts +416 -0
- package/src/frameworks/detector.test.ts +248 -0
- package/src/frameworks/detector.ts +371 -0
- package/src/frameworks/index.ts +68 -0
- package/src/frameworks/recipes/angular.yaml +171 -0
- package/src/frameworks/recipes/astro.yaml +206 -0
- package/src/frameworks/recipes/django.yaml +180 -0
- package/src/frameworks/recipes/laravel.yaml +137 -0
- package/src/frameworks/recipes/nextjs.yaml +268 -0
- package/src/frameworks/recipes/nuxt.yaml +175 -0
- package/src/frameworks/recipes/rails.yaml +188 -0
- package/src/frameworks/recipes/react.yaml +202 -0
- package/src/frameworks/recipes/sveltekit.yaml +154 -0
- package/src/frameworks/recipes/vue.yaml +137 -0
- package/src/frameworks/recipes/wordpress.yaml +209 -0
- package/src/frameworks/suggestion-engine.ts +320 -0
- package/src/geo/geo-content.test.ts +305 -0
- package/src/geo/geo-content.ts +266 -0
- package/src/geo/geo-history.test.ts +473 -0
- package/src/geo/geo-history.ts +433 -0
- package/src/geo/geo-tracker.test.ts +359 -0
- package/src/geo/geo-tracker.ts +411 -0
- package/src/geo/index.ts +10 -0
- package/src/git/commit-helper.test.ts +261 -0
- package/src/git/commit-helper.ts +329 -0
- package/src/git/index.ts +12 -0
- package/src/git/pr-helper.test.ts +284 -0
- package/src/git/pr-helper.ts +307 -0
- package/src/index.ts +66 -0
- package/src/keywords/ai-keyword-engine.ts +1062 -0
- package/src/keywords/ai-summarizer.ts +387 -0
- package/src/keywords/ci-mode.ts +555 -0
- package/src/keywords/engine.ts +359 -0
- package/src/keywords/index.ts +151 -0
- package/src/keywords/llm-judge.ts +357 -0
- package/src/keywords/nlp-analysis.ts +706 -0
- package/src/keywords/prioritizer.ts +295 -0
- package/src/keywords/site-crawler.ts +342 -0
- package/src/keywords/sources/autocomplete.ts +139 -0
- package/src/keywords/sources/competitive-search.ts +450 -0
- package/src/keywords/sources/competitor-analysis.ts +374 -0
- package/src/keywords/sources/dataforseo.ts +206 -0
- package/src/keywords/sources/free-sources.ts +294 -0
- package/src/keywords/sources/gsc.ts +123 -0
- package/src/keywords/topic-grouping.ts +327 -0
- package/src/keywords/types.ts +144 -0
- package/src/keywords/wizard.ts +457 -0
- package/src/loader.ts +40 -0
- package/src/reports/index.ts +7 -0
- package/src/reports/report-generator.test.ts +293 -0
- package/src/reports/report-generator.ts +713 -0
- package/src/scheduler/alerts.test.ts +458 -0
- package/src/scheduler/alerts.ts +328 -0
- package/src/scheduler/index.ts +8 -0
- package/src/scheduler/scheduled-audit.test.ts +377 -0
- package/src/scheduler/scheduled-audit.ts +149 -0
- package/src/test/integration-test.ts +325 -0
- package/src/tools/analyzer.ts +373 -0
- package/src/tools/crawl.ts +293 -0
- package/src/tools/files.ts +301 -0
- package/src/tools/h1-fixer.ts +249 -0
- package/src/tools/index.ts +67 -0
- package/src/tracking/github-action.ts +326 -0
- package/src/tracking/google-analytics.ts +265 -0
- package/src/tracking/index.ts +45 -0
- package/src/tracking/report-generator.ts +386 -0
- package/src/tracking/search-console.ts +335 -0
- package/src/types.ts +134 -0
- package/src/utils/http.ts +302 -0
- package/src/wasm-adapter.ts +297 -0
- package/src/wasm-entry.ts +14 -0
- package/tsconfig.json +17 -0
- package/tsup.wasm.config.ts +26 -0
- package/vitest.config.ts +15 -0
|
@@ -0,0 +1,327 @@
|
|
|
1
|
+
// Keyword Topic Grouping/Clustering
|
|
2
|
+
// Groups keywords by parent topic for content planning
|
|
3
|
+
|
|
4
|
+
import type { KeywordData } from './types.js';
|
|
5
|
+
|
|
6
|
+
export interface KeywordTopic {
|
|
7
|
+
topic: string;
|
|
8
|
+
keywords: KeywordData[];
|
|
9
|
+
totalVolume: number;
|
|
10
|
+
avgDifficulty: number;
|
|
11
|
+
contentType: 'hub' | 'pillar' | 'cluster';
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export interface TopicClusterResult {
|
|
15
|
+
topics: KeywordTopic[];
|
|
16
|
+
uncategorized: KeywordData[];
|
|
17
|
+
contentStrategy: ContentRecommendation[];
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export interface ContentRecommendation {
|
|
21
|
+
topic: string;
|
|
22
|
+
pageType: 'pillar-page' | 'blog-post' | 'faq-page' | 'landing-page';
|
|
23
|
+
targetKeywords: string[];
|
|
24
|
+
potentialTraffic: number;
|
|
25
|
+
difficulty: 'easy' | 'medium' | 'hard';
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Group keywords into topic clusters using multiple strategies:
|
|
30
|
+
* 1. Shared root words
|
|
31
|
+
* 2. Semantic similarity (common terms)
|
|
32
|
+
* 3. Intent matching (questions, how-to, etc.)
|
|
33
|
+
* 4. N-gram overlap
|
|
34
|
+
*/
|
|
35
|
+
export function groupKeywordsByTopic(keywords: KeywordData[]): TopicClusterResult {
|
|
36
|
+
const topicMap = new Map<string, KeywordData[]>();
|
|
37
|
+
const uncategorized: KeywordData[] = [];
|
|
38
|
+
|
|
39
|
+
// Step 1: Extract potential topics from keywords
|
|
40
|
+
const topicCandidates = extractTopicCandidates(keywords);
|
|
41
|
+
|
|
42
|
+
// Step 2: Assign keywords to topics
|
|
43
|
+
for (const kw of keywords) {
|
|
44
|
+
const matchedTopic = findBestTopic(kw.keyword, topicCandidates);
|
|
45
|
+
|
|
46
|
+
if (matchedTopic) {
|
|
47
|
+
if (!topicMap.has(matchedTopic)) {
|
|
48
|
+
topicMap.set(matchedTopic, []);
|
|
49
|
+
}
|
|
50
|
+
topicMap.get(matchedTopic)!.push(kw);
|
|
51
|
+
} else {
|
|
52
|
+
uncategorized.push(kw);
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
// Step 3: Create topic objects with metrics
|
|
57
|
+
const topics: KeywordTopic[] = [];
|
|
58
|
+
|
|
59
|
+
for (const [topic, kwList] of topicMap) {
|
|
60
|
+
if (kwList.length < 2) {
|
|
61
|
+
// Move single-keyword topics to uncategorized
|
|
62
|
+
uncategorized.push(...kwList);
|
|
63
|
+
continue;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
const totalVolume = kwList.reduce((sum, k) => sum + k.searchVolume, 0);
|
|
67
|
+
const avgDifficulty = Math.round(
|
|
68
|
+
kwList.reduce((sum, k) => sum + k.keywordDifficulty, 0) / kwList.length
|
|
69
|
+
);
|
|
70
|
+
|
|
71
|
+
// Determine content type based on cluster size and volume
|
|
72
|
+
let contentType: 'hub' | 'pillar' | 'cluster';
|
|
73
|
+
if (kwList.length >= 10 || totalVolume >= 5000) {
|
|
74
|
+
contentType = 'hub';
|
|
75
|
+
} else if (kwList.length >= 5 || totalVolume >= 1000) {
|
|
76
|
+
contentType = 'pillar';
|
|
77
|
+
} else {
|
|
78
|
+
contentType = 'cluster';
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
topics.push({
|
|
82
|
+
topic,
|
|
83
|
+
keywords: kwList.sort((a, b) => b.searchVolume - a.searchVolume),
|
|
84
|
+
totalVolume,
|
|
85
|
+
avgDifficulty,
|
|
86
|
+
contentType,
|
|
87
|
+
});
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
// Sort topics by total volume
|
|
91
|
+
topics.sort((a, b) => b.totalVolume - a.totalVolume);
|
|
92
|
+
|
|
93
|
+
// Step 4: Generate content strategy recommendations
|
|
94
|
+
const contentStrategy = generateContentStrategy(topics);
|
|
95
|
+
|
|
96
|
+
return { topics, uncategorized, contentStrategy };
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
function extractTopicCandidates(keywords: KeywordData[]): Map<string, number> {
|
|
100
|
+
const wordFrequency = new Map<string, number>();
|
|
101
|
+
const bigramFrequency = new Map<string, number>();
|
|
102
|
+
|
|
103
|
+
// Count word and bigram frequencies
|
|
104
|
+
for (const kw of keywords) {
|
|
105
|
+
const words = kw.keyword.toLowerCase()
|
|
106
|
+
.replace(/[^\w\s]/g, '')
|
|
107
|
+
.split(/\s+/)
|
|
108
|
+
.filter(w => w.length > 2 && !isStopWord(w));
|
|
109
|
+
|
|
110
|
+
// Single words
|
|
111
|
+
for (const word of words) {
|
|
112
|
+
wordFrequency.set(word, (wordFrequency.get(word) || 0) + 1);
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
// Bigrams (2-word phrases)
|
|
116
|
+
for (let i = 0; i < words.length - 1; i++) {
|
|
117
|
+
const bigram = `${words[i]} ${words[i + 1]}`;
|
|
118
|
+
bigramFrequency.set(bigram, (bigramFrequency.get(bigram) || 0) + 1);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
// Combine and filter - prefer bigrams that appear frequently
|
|
123
|
+
const candidates = new Map<string, number>();
|
|
124
|
+
|
|
125
|
+
// Add frequent bigrams first (they make better topics)
|
|
126
|
+
for (const [bigram, count] of bigramFrequency) {
|
|
127
|
+
if (count >= 2) {
|
|
128
|
+
candidates.set(bigram, count * 2); // Weight bigrams higher
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
// Add frequent single words
|
|
133
|
+
for (const [word, count] of wordFrequency) {
|
|
134
|
+
if (count >= 3 && !candidates.has(word)) {
|
|
135
|
+
candidates.set(word, count);
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
return candidates;
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
function findBestTopic(keyword: string, topicCandidates: Map<string, number>): string | null {
|
|
143
|
+
const kwLower = keyword.toLowerCase();
|
|
144
|
+
let bestTopic: string | null = null;
|
|
145
|
+
let bestScore = 0;
|
|
146
|
+
|
|
147
|
+
for (const [topic, frequency] of topicCandidates) {
|
|
148
|
+
if (kwLower.includes(topic)) {
|
|
149
|
+
// Score based on how much of the keyword this topic covers
|
|
150
|
+
const coverage = topic.length / kwLower.length;
|
|
151
|
+
const score = frequency * coverage;
|
|
152
|
+
|
|
153
|
+
if (score > bestScore) {
|
|
154
|
+
bestScore = score;
|
|
155
|
+
bestTopic = topic;
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
// Also check for intent-based grouping
|
|
161
|
+
if (!bestTopic) {
|
|
162
|
+
const intent = classifyIntent(keyword);
|
|
163
|
+
if (intent !== 'other') {
|
|
164
|
+
// Group questions together, how-tos together, etc.
|
|
165
|
+
const intentTopics: Record<string, string> = {
|
|
166
|
+
question: 'FAQ / Questions',
|
|
167
|
+
'how-to': 'How-To Guides',
|
|
168
|
+
comparison: 'Comparisons',
|
|
169
|
+
best: 'Best / Top Lists',
|
|
170
|
+
review: 'Reviews',
|
|
171
|
+
};
|
|
172
|
+
if (intentTopics[intent]) {
|
|
173
|
+
return intentTopics[intent];
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
return bestTopic;
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
function classifyIntent(keyword: string): string {
|
|
182
|
+
const kw = keyword.toLowerCase();
|
|
183
|
+
|
|
184
|
+
if (/^(what|why|when|where|who|which|how|can|is|are|does|do)\b/.test(kw)) {
|
|
185
|
+
return 'question';
|
|
186
|
+
}
|
|
187
|
+
if (kw.startsWith('how to ')) {
|
|
188
|
+
return 'how-to';
|
|
189
|
+
}
|
|
190
|
+
if (/\b(vs|versus|compared|comparison|alternative)\b/.test(kw)) {
|
|
191
|
+
return 'comparison';
|
|
192
|
+
}
|
|
193
|
+
if (/^(best|top|greatest)\b/.test(kw)) {
|
|
194
|
+
return 'best';
|
|
195
|
+
}
|
|
196
|
+
if (/\b(review|reviews)\b/.test(kw)) {
|
|
197
|
+
return 'review';
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
return 'other';
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
function isStopWord(word: string): boolean {
|
|
204
|
+
const stopWords = new Set([
|
|
205
|
+
'the', 'a', 'an', 'and', 'or', 'but', 'in', 'on', 'at', 'to', 'for',
|
|
206
|
+
'of', 'with', 'by', 'from', 'as', 'is', 'was', 'are', 'were', 'been',
|
|
207
|
+
'be', 'have', 'has', 'had', 'do', 'does', 'did', 'will', 'would',
|
|
208
|
+
'could', 'should', 'may', 'might', 'must', 'shall', 'can', 'need',
|
|
209
|
+
'it', 'its', 'this', 'that', 'these', 'those', 'i', 'you', 'he', 'she',
|
|
210
|
+
'we', 'they', 'what', 'which', 'who', 'when', 'where', 'why', 'how',
|
|
211
|
+
'all', 'each', 'every', 'both', 'few', 'more', 'most', 'other', 'some',
|
|
212
|
+
'such', 'no', 'not', 'only', 'own', 'same', 'so', 'than', 'too', 'very',
|
|
213
|
+
'just', 'also', 'now', 'new', 'first', 'last', 'long', 'great', 'little',
|
|
214
|
+
'own', 'other', 'old', 'right', 'big', 'high', 'different', 'small',
|
|
215
|
+
'large', 'next', 'early', 'young', 'important', 'public', 'bad', 'same',
|
|
216
|
+
]);
|
|
217
|
+
return stopWords.has(word);
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
function generateContentStrategy(topics: KeywordTopic[]): ContentRecommendation[] {
|
|
221
|
+
const recommendations: ContentRecommendation[] = [];
|
|
222
|
+
|
|
223
|
+
for (const topic of topics.slice(0, 10)) {
|
|
224
|
+
// Determine page type based on topic characteristics
|
|
225
|
+
let pageType: ContentRecommendation['pageType'];
|
|
226
|
+
let targetKeywords: string[] = [];
|
|
227
|
+
|
|
228
|
+
if (topic.topic === 'FAQ / Questions') {
|
|
229
|
+
pageType = 'faq-page';
|
|
230
|
+
targetKeywords = topic.keywords.slice(0, 10).map(k => k.keyword);
|
|
231
|
+
} else if (topic.topic === 'How-To Guides') {
|
|
232
|
+
pageType = 'blog-post';
|
|
233
|
+
targetKeywords = topic.keywords.slice(0, 5).map(k => k.keyword);
|
|
234
|
+
} else if (topic.contentType === 'hub') {
|
|
235
|
+
pageType = 'pillar-page';
|
|
236
|
+
targetKeywords = topic.keywords.slice(0, 8).map(k => k.keyword);
|
|
237
|
+
} else if (topic.avgDifficulty < 20) {
|
|
238
|
+
pageType = 'blog-post';
|
|
239
|
+
targetKeywords = topic.keywords.slice(0, 5).map(k => k.keyword);
|
|
240
|
+
} else {
|
|
241
|
+
pageType = 'landing-page';
|
|
242
|
+
targetKeywords = topic.keywords.slice(0, 3).map(k => k.keyword);
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
// Determine difficulty
|
|
246
|
+
let difficulty: ContentRecommendation['difficulty'];
|
|
247
|
+
if (topic.avgDifficulty < 20) {
|
|
248
|
+
difficulty = 'easy';
|
|
249
|
+
} else if (topic.avgDifficulty < 40) {
|
|
250
|
+
difficulty = 'medium';
|
|
251
|
+
} else {
|
|
252
|
+
difficulty = 'hard';
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
recommendations.push({
|
|
256
|
+
topic: topic.topic,
|
|
257
|
+
pageType,
|
|
258
|
+
targetKeywords,
|
|
259
|
+
potentialTraffic: topic.totalVolume,
|
|
260
|
+
difficulty,
|
|
261
|
+
});
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
return recommendations;
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
/**
|
|
268
|
+
* Format topic grouping report
|
|
269
|
+
*/
|
|
270
|
+
export function formatTopicReport(result: TopicClusterResult): string {
|
|
271
|
+
const lines: string[] = [];
|
|
272
|
+
|
|
273
|
+
lines.push('');
|
|
274
|
+
lines.push('═'.repeat(70));
|
|
275
|
+
lines.push(' KEYWORD TOPIC CLUSTERS');
|
|
276
|
+
lines.push('═'.repeat(70));
|
|
277
|
+
lines.push('');
|
|
278
|
+
|
|
279
|
+
for (const topic of result.topics.slice(0, 8)) {
|
|
280
|
+
const icon = topic.contentType === 'hub' ? '📚' :
|
|
281
|
+
topic.contentType === 'pillar' ? '📄' : '📝';
|
|
282
|
+
|
|
283
|
+
lines.push(`${icon} TOPIC: ${topic.topic.toUpperCase()}`);
|
|
284
|
+
lines.push('─'.repeat(70));
|
|
285
|
+
lines.push(` Type: ${topic.contentType.toUpperCase()} | Keywords: ${topic.keywords.length} | Volume: ${topic.totalVolume} | Avg KD: ${topic.avgDifficulty}`);
|
|
286
|
+
lines.push('');
|
|
287
|
+
lines.push(' Top Keywords:');
|
|
288
|
+
|
|
289
|
+
for (const kw of topic.keywords.slice(0, 5)) {
|
|
290
|
+
const vol = String(kw.searchVolume).padStart(6);
|
|
291
|
+
const kd = String(kw.keywordDifficulty).padStart(3);
|
|
292
|
+
lines.push(` • ${kw.keyword.substring(0, 40).padEnd(42)} Vol: ${vol} KD: ${kd}`);
|
|
293
|
+
}
|
|
294
|
+
|
|
295
|
+
if (topic.keywords.length > 5) {
|
|
296
|
+
lines.push(` ... and ${topic.keywords.length - 5} more keywords`);
|
|
297
|
+
}
|
|
298
|
+
lines.push('');
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
// Content strategy recommendations
|
|
302
|
+
if (result.contentStrategy.length > 0) {
|
|
303
|
+
lines.push('📋 CONTENT STRATEGY RECOMMENDATIONS');
|
|
304
|
+
lines.push('─'.repeat(70));
|
|
305
|
+
lines.push('');
|
|
306
|
+
|
|
307
|
+
for (const rec of result.contentStrategy.slice(0, 5)) {
|
|
308
|
+
const diffIcon = rec.difficulty === 'easy' ? '🟢' :
|
|
309
|
+
rec.difficulty === 'medium' ? '🟡' : '🔴';
|
|
310
|
+
|
|
311
|
+
lines.push(` ${diffIcon} Create a ${rec.pageType.replace('-', ' ').toUpperCase()} for "${rec.topic}"`);
|
|
312
|
+
lines.push(` Potential traffic: ~${rec.potentialTraffic}/month`);
|
|
313
|
+
lines.push(` Target keywords: ${rec.targetKeywords.slice(0, 3).join(', ')}`);
|
|
314
|
+
lines.push('');
|
|
315
|
+
}
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
// Uncategorized
|
|
319
|
+
if (result.uncategorized.length > 0) {
|
|
320
|
+
lines.push(`📌 ${result.uncategorized.length} keywords didn't fit into topic clusters`);
|
|
321
|
+
lines.push('');
|
|
322
|
+
}
|
|
323
|
+
|
|
324
|
+
lines.push('═'.repeat(70));
|
|
325
|
+
|
|
326
|
+
return lines.join('\n');
|
|
327
|
+
}
|
|
@@ -0,0 +1,144 @@
|
|
|
1
|
+
// Keyword Research Types
|
|
2
|
+
|
|
3
|
+
export interface SiteProfile {
|
|
4
|
+
domain: string;
|
|
5
|
+
domainAge: 'new' | 'established' | 'authority';
|
|
6
|
+
backlinkCount: 'none' | 'few' | 'some' | 'many';
|
|
7
|
+
businessGoal: 'signups' | 'purchases' | 'leads' | 'awareness';
|
|
8
|
+
contentCapacity: 'low' | 'medium' | 'high';
|
|
9
|
+
targetGeo: string;
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
export interface KeywordData {
|
|
13
|
+
keyword: string;
|
|
14
|
+
searchVolume: number;
|
|
15
|
+
keywordDifficulty: number;
|
|
16
|
+
cpc?: number;
|
|
17
|
+
trafficPotential?: number;
|
|
18
|
+
intent?: 'informational' | 'commercial' | 'transactional' | 'navigational';
|
|
19
|
+
trend?: 'up' | 'down' | 'stable';
|
|
20
|
+
source: 'gsc' | 'dataforseo' | 'autocomplete' | 'manual' | 'competitor';
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
export interface KeywordOpportunity extends KeywordData {
|
|
24
|
+
priorityScore: number;
|
|
25
|
+
category: 'quick-win' | 'medium-term' | 'long-term';
|
|
26
|
+
suggestedAction: KeywordAction;
|
|
27
|
+
currentRanking?: number;
|
|
28
|
+
impressions?: number;
|
|
29
|
+
clicks?: number;
|
|
30
|
+
ctr?: number;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
export interface KeywordAction {
|
|
34
|
+
type: 'add-to-title' | 'add-to-h1' | 'add-to-meta' | 'create-content' | 'optimize-existing';
|
|
35
|
+
description: string;
|
|
36
|
+
targetElement?: string;
|
|
37
|
+
currentValue?: string;
|
|
38
|
+
suggestedValue?: string;
|
|
39
|
+
file?: string;
|
|
40
|
+
line?: number;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
export interface KeywordResearchResult {
|
|
44
|
+
siteProfile: SiteProfile;
|
|
45
|
+
keywords: KeywordOpportunity[];
|
|
46
|
+
quickWins: KeywordOpportunity[];
|
|
47
|
+
mediumTerm: KeywordOpportunity[];
|
|
48
|
+
longTerm: KeywordOpportunity[];
|
|
49
|
+
recommendations: string[];
|
|
50
|
+
maxKdThreshold: number;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
export interface GSCQueryData {
|
|
54
|
+
query: string;
|
|
55
|
+
clicks: number;
|
|
56
|
+
impressions: number;
|
|
57
|
+
ctr: number;
|
|
58
|
+
position: number;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
// Questions to ask the user
|
|
62
|
+
export const SITE_PROFILE_QUESTIONS = [
|
|
63
|
+
{
|
|
64
|
+
id: 'domainAge',
|
|
65
|
+
question: 'How old is your domain?',
|
|
66
|
+
options: [
|
|
67
|
+
{ value: 'new', label: 'New (< 6 months)', description: 'Just started, building from scratch' },
|
|
68
|
+
{ value: 'established', label: 'Established (6 months - 2 years)', description: 'Has some history and content' },
|
|
69
|
+
{ value: 'authority', label: 'Authority (2+ years)', description: 'Well-established with backlinks' },
|
|
70
|
+
],
|
|
71
|
+
impact: 'Determines the maximum keyword difficulty you should target',
|
|
72
|
+
},
|
|
73
|
+
{
|
|
74
|
+
id: 'backlinkCount',
|
|
75
|
+
question: 'How many backlinks does your site have?',
|
|
76
|
+
options: [
|
|
77
|
+
{ value: 'none', label: 'None or very few (0-10)', description: 'Just starting link building' },
|
|
78
|
+
{ value: 'few', label: 'Some (10-50)', description: 'A few quality backlinks' },
|
|
79
|
+
{ value: 'some', label: 'Moderate (50-200)', description: 'Decent backlink profile' },
|
|
80
|
+
{ value: 'many', label: 'Many (200+)', description: 'Strong backlink profile' },
|
|
81
|
+
],
|
|
82
|
+
impact: 'Affects your ability to rank for competitive keywords',
|
|
83
|
+
},
|
|
84
|
+
{
|
|
85
|
+
id: 'businessGoal',
|
|
86
|
+
question: 'What is your primary business goal?',
|
|
87
|
+
options: [
|
|
88
|
+
{ value: 'signups', label: 'Get signups/registrations', description: 'SaaS, apps, newsletters' },
|
|
89
|
+
{ value: 'purchases', label: 'Generate purchases', description: 'E-commerce, products' },
|
|
90
|
+
{ value: 'leads', label: 'Capture leads', description: 'B2B, services, consulting' },
|
|
91
|
+
{ value: 'awareness', label: 'Build brand awareness', description: 'Content, community, thought leadership' },
|
|
92
|
+
],
|
|
93
|
+
impact: 'Weights commercial vs informational keywords',
|
|
94
|
+
},
|
|
95
|
+
{
|
|
96
|
+
id: 'contentCapacity',
|
|
97
|
+
question: 'How much content can you produce monthly?',
|
|
98
|
+
options: [
|
|
99
|
+
{ value: 'low', label: '1-2 pages/month', description: 'Limited time or resources' },
|
|
100
|
+
{ value: 'medium', label: '3-5 pages/month', description: 'Moderate content output' },
|
|
101
|
+
{ value: 'high', label: '5+ pages/month', description: 'Dedicated content team' },
|
|
102
|
+
],
|
|
103
|
+
impact: 'Determines how many keywords to suggest',
|
|
104
|
+
},
|
|
105
|
+
{
|
|
106
|
+
id: 'targetGeo',
|
|
107
|
+
question: 'Where are your target customers?',
|
|
108
|
+
options: [
|
|
109
|
+
{ value: 'us', label: 'United States', description: 'Primary US audience' },
|
|
110
|
+
{ value: 'uk', label: 'United Kingdom', description: 'Primary UK audience' },
|
|
111
|
+
{ value: 'global', label: 'Global', description: 'Worldwide audience' },
|
|
112
|
+
{ value: 'other', label: 'Other', description: 'Specific country or region' },
|
|
113
|
+
],
|
|
114
|
+
impact: 'Determines which search volume data to use',
|
|
115
|
+
},
|
|
116
|
+
] as const;
|
|
117
|
+
|
|
118
|
+
// KD thresholds based on site profile
|
|
119
|
+
export function getMaxKdThreshold(profile: SiteProfile): number {
|
|
120
|
+
const ageMultiplier = {
|
|
121
|
+
new: 1,
|
|
122
|
+
established: 2,
|
|
123
|
+
authority: 3,
|
|
124
|
+
};
|
|
125
|
+
|
|
126
|
+
const backlinkMultiplier = {
|
|
127
|
+
none: 1,
|
|
128
|
+
few: 1.2,
|
|
129
|
+
some: 1.5,
|
|
130
|
+
many: 2,
|
|
131
|
+
};
|
|
132
|
+
|
|
133
|
+
const baseKd = 10;
|
|
134
|
+
const maxKd = baseKd * ageMultiplier[profile.domainAge] * backlinkMultiplier[profile.backlinkCount];
|
|
135
|
+
|
|
136
|
+
return Math.min(Math.round(maxKd), 50); // Cap at 50
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
// Priority weights
|
|
140
|
+
export const PRIORITY_WEIGHTS = {
|
|
141
|
+
businessValue: 0.40,
|
|
142
|
+
difficulty: 0.35,
|
|
143
|
+
trafficPotential: 0.25,
|
|
144
|
+
};
|