@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,294 @@
|
|
|
1
|
+
// Free Keyword Data Sources
|
|
2
|
+
// Prioritize these over paid APIs
|
|
3
|
+
|
|
4
|
+
import { httpGet } from '../../utils/http.js';
|
|
5
|
+
import * as cheerio from 'cheerio';
|
|
6
|
+
import type { KeywordData } from '../types.js';
|
|
7
|
+
|
|
8
|
+
const USER_AGENT = 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36';
|
|
9
|
+
|
|
10
|
+
// 1. Google "People Also Ask" Scraper
|
|
11
|
+
export async function getPeopleAlsoAsk(query: string): Promise<string[]> {
|
|
12
|
+
try {
|
|
13
|
+
const response = await httpGet<string>('https://www.google.com/search', {
|
|
14
|
+
params: { q: query },
|
|
15
|
+
headers: { 'User-Agent': USER_AGENT },
|
|
16
|
+
timeout: 10000,
|
|
17
|
+
});
|
|
18
|
+
|
|
19
|
+
const $ = cheerio.load(response.data);
|
|
20
|
+
const questions: string[] = [];
|
|
21
|
+
|
|
22
|
+
// PAA boxes have specific data attributes
|
|
23
|
+
$('[data-sgrd]').each((_, el) => {
|
|
24
|
+
const text = $(el).text().trim();
|
|
25
|
+
if (text.endsWith('?') && text.length > 10) {
|
|
26
|
+
questions.push(text);
|
|
27
|
+
}
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
// Alternative selector
|
|
31
|
+
$('div[jsname="Cpkphb"]').each((_, el) => {
|
|
32
|
+
const text = $(el).text().trim();
|
|
33
|
+
if (text.length > 10 && text.length < 200) {
|
|
34
|
+
questions.push(text);
|
|
35
|
+
}
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
return [...new Set(questions)].slice(0, 10);
|
|
39
|
+
} catch {
|
|
40
|
+
return [];
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
// 2. Google Related Searches Scraper
|
|
45
|
+
export async function getRelatedSearches(query: string): Promise<string[]> {
|
|
46
|
+
try {
|
|
47
|
+
const response = await httpGet<string>('https://www.google.com/search', {
|
|
48
|
+
params: { q: query },
|
|
49
|
+
headers: { 'User-Agent': USER_AGENT },
|
|
50
|
+
timeout: 10000,
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
const $ = cheerio.load(response.data);
|
|
54
|
+
const related: string[] = [];
|
|
55
|
+
|
|
56
|
+
// Related searches at bottom of SERP
|
|
57
|
+
$('div[data-ved] a').each((_, el) => {
|
|
58
|
+
const href = $(el).attr('href');
|
|
59
|
+
if (href?.includes('/search?q=')) {
|
|
60
|
+
const text = $(el).text().trim();
|
|
61
|
+
if (text.length > 3 && text.length < 100 && !text.includes('http')) {
|
|
62
|
+
related.push(text);
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
return [...new Set(related)].slice(0, 8);
|
|
68
|
+
} catch {
|
|
69
|
+
return [];
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
// 3. Wikipedia Topics for Long-Tail Ideas
|
|
74
|
+
export async function getWikipediaTopics(query: string): Promise<string[]> {
|
|
75
|
+
try {
|
|
76
|
+
const response = await httpGet<string>('https://en.wikipedia.org/w/api.php', {
|
|
77
|
+
params: {
|
|
78
|
+
action: 'opensearch',
|
|
79
|
+
search: query,
|
|
80
|
+
limit: 10,
|
|
81
|
+
namespace: 0,
|
|
82
|
+
format: 'json',
|
|
83
|
+
},
|
|
84
|
+
timeout: 5000,
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
// Response: [query, [titles], [descriptions], [urls]]
|
|
88
|
+
return response.data[1] || [];
|
|
89
|
+
} catch {
|
|
90
|
+
return [];
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
// 4. AnswerThePublic-style Question Generator
|
|
95
|
+
export function generateQuestions(keyword: string): string[] {
|
|
96
|
+
const questionPrefixes = [
|
|
97
|
+
'what is', 'what are', 'what does',
|
|
98
|
+
'how to', 'how do', 'how does', 'how can',
|
|
99
|
+
'why is', 'why do', 'why does', 'why are',
|
|
100
|
+
'when to', 'when should', 'when do',
|
|
101
|
+
'where to', 'where can', 'where do',
|
|
102
|
+
'which', 'who', 'can you', 'is it',
|
|
103
|
+
'are there', 'does', 'do',
|
|
104
|
+
];
|
|
105
|
+
|
|
106
|
+
const questions: string[] = [];
|
|
107
|
+
|
|
108
|
+
for (const prefix of questionPrefixes) {
|
|
109
|
+
questions.push(`${prefix} ${keyword}`);
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
return questions;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
// 5. Modifier-based Keyword Expansion
|
|
116
|
+
export function expandWithModifiers(keyword: string): string[] {
|
|
117
|
+
const modifiers = {
|
|
118
|
+
commercial: ['best', 'top', 'review', 'comparison', 'vs', 'alternative', 'pricing'],
|
|
119
|
+
informational: ['how to', 'what is', 'guide', 'tutorial', 'examples', 'tips'],
|
|
120
|
+
temporal: ['2024', '2025', 'new', 'latest', 'updated'],
|
|
121
|
+
intent: ['free', 'online', 'cheap', 'premium', 'professional'],
|
|
122
|
+
action: ['download', 'buy', 'get', 'create', 'make', 'build', 'use'],
|
|
123
|
+
};
|
|
124
|
+
|
|
125
|
+
const expanded: string[] = [];
|
|
126
|
+
|
|
127
|
+
for (const [, mods] of Object.entries(modifiers)) {
|
|
128
|
+
for (const mod of mods) {
|
|
129
|
+
expanded.push(`${mod} ${keyword}`);
|
|
130
|
+
expanded.push(`${keyword} ${mod}`);
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
return expanded;
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
// 6. Competitor Title Analyzer (from SERP)
|
|
138
|
+
export async function analyzeCompetitorTitles(query: string): Promise<{ titles: string[]; keywords: string[] }> {
|
|
139
|
+
try {
|
|
140
|
+
const response = await httpGet<string>('https://www.google.com/search', {
|
|
141
|
+
params: { q: query, num: 10 },
|
|
142
|
+
headers: { 'User-Agent': USER_AGENT },
|
|
143
|
+
timeout: 10000,
|
|
144
|
+
});
|
|
145
|
+
|
|
146
|
+
const $ = cheerio.load(response.data);
|
|
147
|
+
const titles: string[] = [];
|
|
148
|
+
const keywords: Set<string> = new Set();
|
|
149
|
+
|
|
150
|
+
// Extract titles from search results
|
|
151
|
+
$('h3').each((_, el) => {
|
|
152
|
+
const title = $(el).text().trim();
|
|
153
|
+
if (title.length > 10 && title.length < 100) {
|
|
154
|
+
titles.push(title);
|
|
155
|
+
|
|
156
|
+
// Extract potential keywords from titles
|
|
157
|
+
const words = title.toLowerCase().split(/[\s\-|:]+/);
|
|
158
|
+
words.filter(w => w.length > 3).forEach(w => keywords.add(w));
|
|
159
|
+
}
|
|
160
|
+
});
|
|
161
|
+
|
|
162
|
+
return {
|
|
163
|
+
titles: titles.slice(0, 10),
|
|
164
|
+
keywords: Array.from(keywords).slice(0, 20),
|
|
165
|
+
};
|
|
166
|
+
} catch {
|
|
167
|
+
return { titles: [], keywords: [] };
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
// 7. Combine All Free Sources
|
|
172
|
+
export async function getAllFreeKeywordIdeas(
|
|
173
|
+
seedKeyword: string,
|
|
174
|
+
options: { includeQuestions?: boolean; includeModifiers?: boolean } = {}
|
|
175
|
+
): Promise<KeywordData[]> {
|
|
176
|
+
const { includeQuestions = true, includeModifiers = true } = options;
|
|
177
|
+
|
|
178
|
+
const allKeywords: string[] = [];
|
|
179
|
+
const seen = new Set<string>();
|
|
180
|
+
|
|
181
|
+
const addKeyword = (kw: string) => {
|
|
182
|
+
const normalized = kw.toLowerCase().trim();
|
|
183
|
+
if (!seen.has(normalized) && normalized.length > 2) {
|
|
184
|
+
seen.add(normalized);
|
|
185
|
+
allKeywords.push(kw);
|
|
186
|
+
}
|
|
187
|
+
};
|
|
188
|
+
|
|
189
|
+
// Run sources in parallel where possible
|
|
190
|
+
console.log(' → Fetching from Google autocomplete...');
|
|
191
|
+
const [paa, related, wiki, competitors] = await Promise.all([
|
|
192
|
+
getPeopleAlsoAsk(seedKeyword).catch(() => []),
|
|
193
|
+
getRelatedSearches(seedKeyword).catch(() => []),
|
|
194
|
+
getWikipediaTopics(seedKeyword).catch(() => []),
|
|
195
|
+
analyzeCompetitorTitles(seedKeyword).catch(() => ({ titles: [], keywords: [] })),
|
|
196
|
+
]);
|
|
197
|
+
|
|
198
|
+
// Add PAA questions
|
|
199
|
+
paa.forEach(addKeyword);
|
|
200
|
+
|
|
201
|
+
// Add related searches
|
|
202
|
+
related.forEach(addKeyword);
|
|
203
|
+
|
|
204
|
+
// Add Wikipedia topics
|
|
205
|
+
wiki.forEach(addKeyword);
|
|
206
|
+
|
|
207
|
+
// Add competitor keywords
|
|
208
|
+
competitors.keywords.forEach(addKeyword);
|
|
209
|
+
|
|
210
|
+
// Add generated questions
|
|
211
|
+
if (includeQuestions) {
|
|
212
|
+
generateQuestions(seedKeyword).forEach(addKeyword);
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
// Add modifier expansions
|
|
216
|
+
if (includeModifiers) {
|
|
217
|
+
expandWithModifiers(seedKeyword).forEach(addKeyword);
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
// Convert to KeywordData format
|
|
221
|
+
return allKeywords.map((kw) => ({
|
|
222
|
+
keyword: kw,
|
|
223
|
+
searchVolume: 0, // Will be estimated
|
|
224
|
+
keywordDifficulty: 0, // Will be estimated
|
|
225
|
+
source: 'autocomplete' as const,
|
|
226
|
+
}));
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
// 8. Estimate metrics based on keyword characteristics
|
|
230
|
+
export function estimateMetrics(keyword: string): { volume: number; difficulty: number } {
|
|
231
|
+
const words = keyword.split(' ').length;
|
|
232
|
+
const hasQuestion = /^(how|what|why|when|where|who|which|can|is|are|do|does)/i.test(keyword);
|
|
233
|
+
const hasCommercial = /\b(best|top|review|buy|price|cheap|free|vs|alternative)\b/i.test(keyword);
|
|
234
|
+
const hasYear = /\b20\d{2}\b/.test(keyword);
|
|
235
|
+
|
|
236
|
+
// Base estimates
|
|
237
|
+
let volume = 500;
|
|
238
|
+
let difficulty = 30;
|
|
239
|
+
|
|
240
|
+
// Adjust based on word count (longer = less volume, less competition)
|
|
241
|
+
if (words >= 5) {
|
|
242
|
+
volume = 50;
|
|
243
|
+
difficulty = 10;
|
|
244
|
+
} else if (words >= 4) {
|
|
245
|
+
volume = 150;
|
|
246
|
+
difficulty = 15;
|
|
247
|
+
} else if (words >= 3) {
|
|
248
|
+
volume = 300;
|
|
249
|
+
difficulty = 20;
|
|
250
|
+
} else if (words === 2) {
|
|
251
|
+
volume = 800;
|
|
252
|
+
difficulty = 35;
|
|
253
|
+
} else {
|
|
254
|
+
volume = 2000;
|
|
255
|
+
difficulty = 50;
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
// Adjust for question keywords (usually lower competition)
|
|
259
|
+
if (hasQuestion) {
|
|
260
|
+
difficulty -= 5;
|
|
261
|
+
volume *= 0.8;
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
// Adjust for commercial intent (usually higher competition)
|
|
265
|
+
if (hasCommercial) {
|
|
266
|
+
difficulty += 10;
|
|
267
|
+
volume *= 1.2;
|
|
268
|
+
}
|
|
269
|
+
|
|
270
|
+
// Year keywords are often seasonal
|
|
271
|
+
if (hasYear) {
|
|
272
|
+
volume *= 1.5;
|
|
273
|
+
}
|
|
274
|
+
|
|
275
|
+
return {
|
|
276
|
+
volume: Math.round(Math.max(10, volume)),
|
|
277
|
+
difficulty: Math.round(Math.max(1, Math.min(100, difficulty))),
|
|
278
|
+
};
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
// 9. Apply estimates to keyword list
|
|
282
|
+
export function applyEstimates(keywords: KeywordData[]): KeywordData[] {
|
|
283
|
+
return keywords.map((kw) => {
|
|
284
|
+
if (kw.searchVolume === 0 || kw.keywordDifficulty === 0) {
|
|
285
|
+
const estimates = estimateMetrics(kw.keyword);
|
|
286
|
+
return {
|
|
287
|
+
...kw,
|
|
288
|
+
searchVolume: kw.searchVolume || estimates.volume,
|
|
289
|
+
keywordDifficulty: kw.keywordDifficulty || estimates.difficulty,
|
|
290
|
+
};
|
|
291
|
+
}
|
|
292
|
+
return kw;
|
|
293
|
+
});
|
|
294
|
+
}
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
// Google Search Console Integration (Free - Your Own Data)
|
|
2
|
+
|
|
3
|
+
import type { KeywordData, GSCQueryData } from '../types.js';
|
|
4
|
+
|
|
5
|
+
// Note: GSC API requires OAuth2 authentication
|
|
6
|
+
// This module provides the structure and helpers for GSC integration
|
|
7
|
+
|
|
8
|
+
export interface GSCCredentials {
|
|
9
|
+
clientId: string;
|
|
10
|
+
clientSecret: string;
|
|
11
|
+
refreshToken: string;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export interface GSCQueryOptions {
|
|
15
|
+
siteUrl: string;
|
|
16
|
+
startDate: string;
|
|
17
|
+
endDate: string;
|
|
18
|
+
dimensions?: string[];
|
|
19
|
+
rowLimit?: number;
|
|
20
|
+
startRow?: number;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
// Transform GSC data to our KeywordData format
|
|
24
|
+
export function transformGSCData(gscData: GSCQueryData[]): KeywordData[] {
|
|
25
|
+
return gscData.map((row) => ({
|
|
26
|
+
keyword: row.query,
|
|
27
|
+
searchVolume: estimateVolumeFromImpressions(row.impressions),
|
|
28
|
+
keywordDifficulty: estimateKdFromPosition(row.position),
|
|
29
|
+
source: 'gsc' as const,
|
|
30
|
+
currentRanking: row.position,
|
|
31
|
+
impressions: row.impressions,
|
|
32
|
+
clicks: row.clicks,
|
|
33
|
+
ctr: row.ctr,
|
|
34
|
+
}));
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
// Estimate search volume from impressions
|
|
38
|
+
// If you're ranking #1-3, impressions ≈ 60-90% of total volume
|
|
39
|
+
// If you're ranking #4-10, impressions ≈ 10-30% of total volume
|
|
40
|
+
function estimateVolumeFromImpressions(impressions: number): number {
|
|
41
|
+
// Rough estimate: multiply by factor based on assumed position
|
|
42
|
+
// This is a simplified calculation
|
|
43
|
+
return Math.round(impressions * 1.5);
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
// Estimate KD from current ranking position
|
|
47
|
+
// If you're already ranking, KD is within your reach
|
|
48
|
+
function estimateKdFromPosition(position: number): number {
|
|
49
|
+
if (position <= 3) return 5; // Already ranking well = easy for you
|
|
50
|
+
if (position <= 10) return 15; // First page = achievable
|
|
51
|
+
if (position <= 20) return 25; // Second page = moderate
|
|
52
|
+
if (position <= 50) return 35; // You're in the game
|
|
53
|
+
return 45; // Need more effort
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
// Find keywords with high impressions but low clicks (optimization opportunities)
|
|
57
|
+
export function findCTROpportunities(gscData: GSCQueryData[]): GSCQueryData[] {
|
|
58
|
+
return gscData
|
|
59
|
+
.filter((row) => {
|
|
60
|
+
// High impressions, low CTR, good position
|
|
61
|
+
return row.impressions > 100 && row.ctr < 0.03 && row.position <= 20;
|
|
62
|
+
})
|
|
63
|
+
.sort((a, b) => b.impressions - a.impressions);
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
// Find keywords where you're close to page 1
|
|
67
|
+
export function findAlmostPage1Keywords(gscData: GSCQueryData[]): GSCQueryData[] {
|
|
68
|
+
return gscData
|
|
69
|
+
.filter((row) => row.position > 10 && row.position <= 20 && row.impressions > 50)
|
|
70
|
+
.sort((a, b) => a.position - b.position);
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
// Find your best performing keywords
|
|
74
|
+
export function findTopPerformers(gscData: GSCQueryData[]): GSCQueryData[] {
|
|
75
|
+
return gscData
|
|
76
|
+
.filter((row) => row.position <= 10 && row.clicks > 10)
|
|
77
|
+
.sort((a, b) => b.clicks - a.clicks);
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// Generate GSC API request body
|
|
81
|
+
export function buildGSCRequest(options: GSCQueryOptions): object {
|
|
82
|
+
return {
|
|
83
|
+
startDate: options.startDate,
|
|
84
|
+
endDate: options.endDate,
|
|
85
|
+
dimensions: options.dimensions || ['query'],
|
|
86
|
+
rowLimit: options.rowLimit || 1000,
|
|
87
|
+
startRow: options.startRow || 0,
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
// Parse date range for last N days
|
|
92
|
+
export function getDateRange(days: number = 28): { startDate: string; endDate: string } {
|
|
93
|
+
const endDate = new Date();
|
|
94
|
+
endDate.setDate(endDate.getDate() - 3); // GSC data is delayed by ~3 days
|
|
95
|
+
|
|
96
|
+
const startDate = new Date(endDate);
|
|
97
|
+
startDate.setDate(startDate.getDate() - days);
|
|
98
|
+
|
|
99
|
+
return {
|
|
100
|
+
startDate: startDate.toISOString().split('T')[0],
|
|
101
|
+
endDate: endDate.toISOString().split('T')[0],
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
// Example response structure for reference
|
|
106
|
+
export const GSC_EXAMPLE_RESPONSE = {
|
|
107
|
+
rows: [
|
|
108
|
+
{
|
|
109
|
+
keys: ['mermaid animation'],
|
|
110
|
+
clicks: 45,
|
|
111
|
+
impressions: 1200,
|
|
112
|
+
ctr: 0.0375,
|
|
113
|
+
position: 4.2,
|
|
114
|
+
},
|
|
115
|
+
{
|
|
116
|
+
keys: ['animate diagrams'],
|
|
117
|
+
clicks: 12,
|
|
118
|
+
impressions: 800,
|
|
119
|
+
ctr: 0.015,
|
|
120
|
+
position: 8.5,
|
|
121
|
+
},
|
|
122
|
+
],
|
|
123
|
+
};
|