@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,359 @@
|
|
|
1
|
+
// Main Keyword Research Engine
|
|
2
|
+
|
|
3
|
+
import * as cheerio from 'cheerio';
|
|
4
|
+
import { httpGet } from '../utils/http.js';
|
|
5
|
+
import type {
|
|
6
|
+
SiteProfile,
|
|
7
|
+
KeywordData,
|
|
8
|
+
KeywordResearchResult,
|
|
9
|
+
} from './types.js';
|
|
10
|
+
import { prioritizeKeywords } from './prioritizer.js';
|
|
11
|
+
import {
|
|
12
|
+
getExpandedSuggestions,
|
|
13
|
+
enrichKeywordsWithEstimates,
|
|
14
|
+
} from './sources/autocomplete.js';
|
|
15
|
+
import {
|
|
16
|
+
getAllFreeKeywordIdeas,
|
|
17
|
+
applyEstimates,
|
|
18
|
+
} from './sources/free-sources.js';
|
|
19
|
+
import {
|
|
20
|
+
getKeywordSuggestions,
|
|
21
|
+
getRelatedKeywords,
|
|
22
|
+
type DataForSEOCredentials,
|
|
23
|
+
} from './sources/dataforseo.js';
|
|
24
|
+
|
|
25
|
+
const USER_AGENT = 'SEO-Autopilot/1.0 (+https://seo-autopilot.dev)';
|
|
26
|
+
|
|
27
|
+
export interface KeywordResearchOptions {
|
|
28
|
+
seedKeywords: string[];
|
|
29
|
+
siteProfile: SiteProfile;
|
|
30
|
+
url?: string; // To fetch current meta for suggestions
|
|
31
|
+
dataForSEOCredentials?: DataForSEOCredentials;
|
|
32
|
+
maxKeywords?: number;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
export interface PageMeta {
|
|
36
|
+
title?: string;
|
|
37
|
+
description?: string;
|
|
38
|
+
h1?: string;
|
|
39
|
+
url: string;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
export async function runKeywordResearch(
|
|
43
|
+
options: KeywordResearchOptions
|
|
44
|
+
): Promise<KeywordResearchResult> {
|
|
45
|
+
const {
|
|
46
|
+
seedKeywords,
|
|
47
|
+
siteProfile,
|
|
48
|
+
url,
|
|
49
|
+
dataForSEOCredentials,
|
|
50
|
+
maxKeywords = 50,
|
|
51
|
+
} = options;
|
|
52
|
+
|
|
53
|
+
console.log('\n🔍 Starting keyword research...\n');
|
|
54
|
+
|
|
55
|
+
// Fetch current page meta if URL provided
|
|
56
|
+
let pageMeta: PageMeta | undefined;
|
|
57
|
+
if (url) {
|
|
58
|
+
console.log('📥 Fetching current page meta...');
|
|
59
|
+
pageMeta = await fetchPageMeta(url);
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
// Collect keywords from different sources
|
|
63
|
+
let allKeywords: KeywordData[] = [];
|
|
64
|
+
|
|
65
|
+
// Source 1: Free Sources (Autocomplete, PAA, Related, Wikipedia, Competitors)
|
|
66
|
+
console.log('🔤 Getting keyword ideas from free sources...');
|
|
67
|
+
for (const seed of seedKeywords) {
|
|
68
|
+
// Expanded autocomplete (alphabet soup technique)
|
|
69
|
+
const autocompleteSuggestions = await getExpandedSuggestions(seed, {
|
|
70
|
+
language: 'en',
|
|
71
|
+
country: siteProfile.targetGeo === 'us' ? 'us' : siteProfile.targetGeo,
|
|
72
|
+
});
|
|
73
|
+
allKeywords.push(...autocompleteSuggestions);
|
|
74
|
+
|
|
75
|
+
// All other free sources (PAA, Related, Wikipedia, Modifiers)
|
|
76
|
+
const freeIdeas = await getAllFreeKeywordIdeas(seed, {
|
|
77
|
+
includeQuestions: true,
|
|
78
|
+
includeModifiers: true,
|
|
79
|
+
});
|
|
80
|
+
allKeywords.push(...freeIdeas);
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
// Source 2: DataForSEO (Premium - only if credentials provided)
|
|
84
|
+
if (dataForSEOCredentials) {
|
|
85
|
+
console.log('💎 Getting premium keyword data from DataForSEO...');
|
|
86
|
+
try {
|
|
87
|
+
for (const seed of seedKeywords) {
|
|
88
|
+
const suggestions = await getKeywordSuggestions(
|
|
89
|
+
seed,
|
|
90
|
+
dataForSEOCredentials,
|
|
91
|
+
2840, // US
|
|
92
|
+
30
|
|
93
|
+
);
|
|
94
|
+
allKeywords.push(...suggestions);
|
|
95
|
+
|
|
96
|
+
const related = await getRelatedKeywords(
|
|
97
|
+
seed,
|
|
98
|
+
dataForSEOCredentials,
|
|
99
|
+
2840,
|
|
100
|
+
20
|
|
101
|
+
);
|
|
102
|
+
allKeywords.push(...related);
|
|
103
|
+
}
|
|
104
|
+
} catch (error) {
|
|
105
|
+
console.log('⚠️ DataForSEO fetch failed, using free estimates');
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
// Remove duplicates
|
|
110
|
+
const uniqueKeywords = deduplicateKeywords(allKeywords);
|
|
111
|
+
|
|
112
|
+
// Enrich keywords without data with estimates (free estimation)
|
|
113
|
+
const enrichedKeywords = applyEstimates(enrichKeywordsWithEstimates(uniqueKeywords));
|
|
114
|
+
|
|
115
|
+
// Filter to relevant keywords
|
|
116
|
+
const relevantKeywords = filterRelevantKeywords(enrichedKeywords, seedKeywords);
|
|
117
|
+
|
|
118
|
+
// Limit to max keywords
|
|
119
|
+
const limitedKeywords = relevantKeywords.slice(0, maxKeywords);
|
|
120
|
+
|
|
121
|
+
console.log(`\n📋 Found ${limitedKeywords.length} keyword opportunities\n`);
|
|
122
|
+
|
|
123
|
+
// Prioritize and categorize
|
|
124
|
+
const result = prioritizeKeywords(
|
|
125
|
+
limitedKeywords,
|
|
126
|
+
siteProfile,
|
|
127
|
+
pageMeta
|
|
128
|
+
? {
|
|
129
|
+
title: pageMeta.title,
|
|
130
|
+
description: pageMeta.description,
|
|
131
|
+
h1: pageMeta.h1,
|
|
132
|
+
}
|
|
133
|
+
: undefined
|
|
134
|
+
);
|
|
135
|
+
|
|
136
|
+
return result;
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
async function fetchPageMeta(url: string): Promise<PageMeta> {
|
|
140
|
+
try {
|
|
141
|
+
const response = await httpGet<string>(url, {
|
|
142
|
+
|
|
143
|
+
timeout: 10000,
|
|
144
|
+
});
|
|
145
|
+
|
|
146
|
+
const $ = cheerio.load(response.data);
|
|
147
|
+
|
|
148
|
+
return {
|
|
149
|
+
url,
|
|
150
|
+
title: $('title').text().trim() || undefined,
|
|
151
|
+
description: $('meta[name="description"]').attr('content')?.trim() || undefined,
|
|
152
|
+
h1: $('h1').first().text().trim() || undefined,
|
|
153
|
+
};
|
|
154
|
+
} catch {
|
|
155
|
+
return { url };
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
function deduplicateKeywords(keywords: KeywordData[]): KeywordData[] {
|
|
160
|
+
const seen = new Map<string, KeywordData>();
|
|
161
|
+
|
|
162
|
+
for (const kw of keywords) {
|
|
163
|
+
const key = kw.keyword.toLowerCase().trim();
|
|
164
|
+
const existing = seen.get(key);
|
|
165
|
+
|
|
166
|
+
// Keep the one with more data (higher volume or from better source)
|
|
167
|
+
if (!existing || kw.searchVolume > existing.searchVolume) {
|
|
168
|
+
seen.set(key, kw);
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
return Array.from(seen.values());
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
function filterRelevantKeywords(
|
|
176
|
+
keywords: KeywordData[],
|
|
177
|
+
seedKeywords: string[]
|
|
178
|
+
): KeywordData[] {
|
|
179
|
+
const seeds = seedKeywords.map((s) => s.toLowerCase());
|
|
180
|
+
|
|
181
|
+
return keywords.filter((kw) => {
|
|
182
|
+
const kwLower = kw.keyword.toLowerCase();
|
|
183
|
+
|
|
184
|
+
// Keep if contains any seed keyword
|
|
185
|
+
if (seeds.some((seed) => kwLower.includes(seed) || seed.includes(kwLower))) {
|
|
186
|
+
return true;
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
// Keep if related to common patterns
|
|
190
|
+
const relatedPatterns = [
|
|
191
|
+
'how to',
|
|
192
|
+
'best',
|
|
193
|
+
'free',
|
|
194
|
+
'online',
|
|
195
|
+
'tool',
|
|
196
|
+
'app',
|
|
197
|
+
'software',
|
|
198
|
+
'tutorial',
|
|
199
|
+
'guide',
|
|
200
|
+
];
|
|
201
|
+
|
|
202
|
+
return relatedPatterns.some((pattern) => kwLower.includes(pattern));
|
|
203
|
+
});
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
// Format results for console output
|
|
207
|
+
export function formatKeywordReport(result: KeywordResearchResult): string {
|
|
208
|
+
const lines: string[] = [];
|
|
209
|
+
|
|
210
|
+
lines.push('');
|
|
211
|
+
lines.push('═'.repeat(70));
|
|
212
|
+
lines.push(' KEYWORD RESEARCH REPORT');
|
|
213
|
+
lines.push('═'.repeat(70));
|
|
214
|
+
lines.push('');
|
|
215
|
+
|
|
216
|
+
// Site profile summary
|
|
217
|
+
lines.push('📋 SITE PROFILE');
|
|
218
|
+
lines.push('─'.repeat(70));
|
|
219
|
+
lines.push(` Domain: ${result.siteProfile.domain}`);
|
|
220
|
+
lines.push(` Age: ${result.siteProfile.domainAge}`);
|
|
221
|
+
lines.push(` Backlinks: ${result.siteProfile.backlinkCount}`);
|
|
222
|
+
lines.push(` Goal: ${result.siteProfile.businessGoal}`);
|
|
223
|
+
lines.push(` Max KD: ${result.maxKdThreshold} (based on your profile)`);
|
|
224
|
+
lines.push('');
|
|
225
|
+
|
|
226
|
+
// Quick wins
|
|
227
|
+
if (result.quickWins.length > 0) {
|
|
228
|
+
lines.push('🟢 QUICK WINS (Target First!)');
|
|
229
|
+
lines.push('─'.repeat(70));
|
|
230
|
+
lines.push(' These keywords have low competition and you can rank quickly.');
|
|
231
|
+
lines.push('');
|
|
232
|
+
|
|
233
|
+
const table = formatKeywordTable(result.quickWins.slice(0, 10));
|
|
234
|
+
lines.push(table);
|
|
235
|
+
lines.push('');
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
// Medium term
|
|
239
|
+
if (result.mediumTerm.length > 0) {
|
|
240
|
+
lines.push('🟡 MEDIUM TERM OPPORTUNITIES');
|
|
241
|
+
lines.push('─'.repeat(70));
|
|
242
|
+
lines.push(' Build some authority first, then target these keywords.');
|
|
243
|
+
lines.push('');
|
|
244
|
+
|
|
245
|
+
const table = formatKeywordTable(result.mediumTerm.slice(0, 5));
|
|
246
|
+
lines.push(table);
|
|
247
|
+
lines.push('');
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
// Long term
|
|
251
|
+
if (result.longTerm.length > 0) {
|
|
252
|
+
lines.push('🔴 LONG TERM (High Competition)');
|
|
253
|
+
lines.push('─'.repeat(70));
|
|
254
|
+
lines.push(' Save these for when you have more domain authority.');
|
|
255
|
+
lines.push('');
|
|
256
|
+
|
|
257
|
+
const table = formatKeywordTable(result.longTerm.slice(0, 3));
|
|
258
|
+
lines.push(table);
|
|
259
|
+
lines.push('');
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
// Recommendations
|
|
263
|
+
if (result.recommendations.length > 0) {
|
|
264
|
+
lines.push('💡 RECOMMENDATIONS');
|
|
265
|
+
lines.push('─'.repeat(70));
|
|
266
|
+
for (const rec of result.recommendations) {
|
|
267
|
+
lines.push(` • ${rec}`);
|
|
268
|
+
}
|
|
269
|
+
lines.push('');
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
// Top actions
|
|
273
|
+
const topActions = result.quickWins
|
|
274
|
+
.slice(0, 3)
|
|
275
|
+
.filter((kw) => kw.suggestedAction.suggestedValue);
|
|
276
|
+
|
|
277
|
+
if (topActions.length > 0) {
|
|
278
|
+
lines.push('🎯 SUGGESTED CHANGES');
|
|
279
|
+
lines.push('─'.repeat(70));
|
|
280
|
+
|
|
281
|
+
for (const kw of topActions) {
|
|
282
|
+
lines.push(` Keyword: "${kw.keyword}"`);
|
|
283
|
+
lines.push(` Action: ${kw.suggestedAction.description}`);
|
|
284
|
+
if (kw.suggestedAction.currentValue) {
|
|
285
|
+
lines.push(` Current: ${kw.suggestedAction.currentValue}`);
|
|
286
|
+
}
|
|
287
|
+
if (kw.suggestedAction.suggestedValue) {
|
|
288
|
+
lines.push(` Suggest: ${kw.suggestedAction.suggestedValue}`);
|
|
289
|
+
}
|
|
290
|
+
lines.push('');
|
|
291
|
+
}
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
lines.push('═'.repeat(70));
|
|
295
|
+
|
|
296
|
+
return lines.join('\n');
|
|
297
|
+
}
|
|
298
|
+
|
|
299
|
+
function formatKeywordTable(keywords: KeywordData[]): string {
|
|
300
|
+
const lines: string[] = [];
|
|
301
|
+
|
|
302
|
+
// Header
|
|
303
|
+
lines.push(' Keyword Vol KD Score Action');
|
|
304
|
+
lines.push(' ' + '─'.repeat(65));
|
|
305
|
+
|
|
306
|
+
for (const kw of keywords) {
|
|
307
|
+
const keyword = kw.keyword.substring(0, 30).padEnd(32);
|
|
308
|
+
const volume = String(kw.searchVolume).padStart(6);
|
|
309
|
+
const kd = String(kw.keywordDifficulty).padStart(4);
|
|
310
|
+
const score = 'priorityScore' in kw ? String((kw as any).priorityScore).padStart(5) : ' - ';
|
|
311
|
+
const action =
|
|
312
|
+
'suggestedAction' in kw
|
|
313
|
+
? (kw as any).suggestedAction.type.substring(0, 15)
|
|
314
|
+
: '';
|
|
315
|
+
|
|
316
|
+
lines.push(` ${keyword} ${volume} ${kd} ${score} ${action}`);
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
return lines.join('\n');
|
|
320
|
+
}
|
|
321
|
+
|
|
322
|
+
// Extract seed keywords from page content
|
|
323
|
+
export async function extractSeedKeywords(url: string): Promise<string[]> {
|
|
324
|
+
try {
|
|
325
|
+
const response = await httpGet<string>(url, {
|
|
326
|
+
|
|
327
|
+
timeout: 10000,
|
|
328
|
+
});
|
|
329
|
+
|
|
330
|
+
const $ = cheerio.load(response.data);
|
|
331
|
+
|
|
332
|
+
const seeds: Set<string> = new Set();
|
|
333
|
+
|
|
334
|
+
// From title
|
|
335
|
+
const title = $('title').text().toLowerCase();
|
|
336
|
+
const titleWords = title.split(/[\s\-|:]+/).filter((w) => w.length > 3);
|
|
337
|
+
titleWords.forEach((w) => seeds.add(w));
|
|
338
|
+
|
|
339
|
+
// From H1
|
|
340
|
+
const h1 = $('h1').first().text().toLowerCase();
|
|
341
|
+
const h1Words = h1.split(/[\s\-|:]+/).filter((w) => w.length > 3);
|
|
342
|
+
h1Words.forEach((w) => seeds.add(w));
|
|
343
|
+
|
|
344
|
+
// From meta keywords (if present)
|
|
345
|
+
const metaKeywords = $('meta[name="keywords"]').attr('content');
|
|
346
|
+
if (metaKeywords) {
|
|
347
|
+
metaKeywords.split(',').forEach((k) => seeds.add(k.trim().toLowerCase()));
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
// From description
|
|
351
|
+
const desc = $('meta[name="description"]').attr('content')?.toLowerCase() || '';
|
|
352
|
+
const descWords = desc.split(/\s+/).filter((w) => w.length > 5);
|
|
353
|
+
descWords.slice(0, 5).forEach((w) => seeds.add(w));
|
|
354
|
+
|
|
355
|
+
return Array.from(seeds).slice(0, 10);
|
|
356
|
+
} catch {
|
|
357
|
+
return [];
|
|
358
|
+
}
|
|
359
|
+
}
|
|
@@ -0,0 +1,151 @@
|
|
|
1
|
+
// Keyword Research Module
|
|
2
|
+
|
|
3
|
+
// Types
|
|
4
|
+
export * from './types.js';
|
|
5
|
+
|
|
6
|
+
// Main engine (legacy)
|
|
7
|
+
export {
|
|
8
|
+
runKeywordResearch,
|
|
9
|
+
formatKeywordReport,
|
|
10
|
+
extractSeedKeywords,
|
|
11
|
+
type KeywordResearchOptions,
|
|
12
|
+
type PageMeta,
|
|
13
|
+
} from './engine.js';
|
|
14
|
+
|
|
15
|
+
// AI-Powered Keyword Research (new)
|
|
16
|
+
export {
|
|
17
|
+
runAIKeywordResearch,
|
|
18
|
+
type AIKeywordResearchOptions,
|
|
19
|
+
type PlanTier,
|
|
20
|
+
type FreeKeywordResult,
|
|
21
|
+
type PaidKeywordResult,
|
|
22
|
+
type KeywordOpportunity,
|
|
23
|
+
type FreeToolIdea,
|
|
24
|
+
type CompetitiveInsight,
|
|
25
|
+
type UncertaintyAssessment,
|
|
26
|
+
type KeywordRecommendation,
|
|
27
|
+
} from './ai-keyword-engine.js';
|
|
28
|
+
|
|
29
|
+
// Site Crawler
|
|
30
|
+
export {
|
|
31
|
+
crawlSite,
|
|
32
|
+
extractKeyPhrases,
|
|
33
|
+
type CrawledPage,
|
|
34
|
+
type SiteCrawlResult,
|
|
35
|
+
} from './site-crawler.js';
|
|
36
|
+
|
|
37
|
+
// AI Summarizer
|
|
38
|
+
export {
|
|
39
|
+
summarizeSite,
|
|
40
|
+
createFallbackSummary,
|
|
41
|
+
enhanceSummaryWithCompetitors,
|
|
42
|
+
generateUncertaintyQuestions,
|
|
43
|
+
type SiteSummary,
|
|
44
|
+
type SummarizerOptions,
|
|
45
|
+
} from './ai-summarizer.js';
|
|
46
|
+
|
|
47
|
+
// NLP Analysis
|
|
48
|
+
export {
|
|
49
|
+
runNLPAnalysis,
|
|
50
|
+
calculateTFIDF as calculateKeywordTFIDF,
|
|
51
|
+
extractNgrams,
|
|
52
|
+
calculateBM25,
|
|
53
|
+
clusterKeywordsByEmbedding,
|
|
54
|
+
extractEntityPhrases,
|
|
55
|
+
extractTopics,
|
|
56
|
+
tokenize,
|
|
57
|
+
type TFIDFResult,
|
|
58
|
+
type NGram,
|
|
59
|
+
type KeywordCluster,
|
|
60
|
+
type TopicModel,
|
|
61
|
+
type NLPAnalysisResult,
|
|
62
|
+
} from './nlp-analysis.js';
|
|
63
|
+
|
|
64
|
+
// Wizard
|
|
65
|
+
export {
|
|
66
|
+
startWizardSession,
|
|
67
|
+
generateWizardQuestions,
|
|
68
|
+
processWizardResponse,
|
|
69
|
+
getNextQuestion,
|
|
70
|
+
completeWizard,
|
|
71
|
+
wizardResponsesToContext,
|
|
72
|
+
formatWizardProgress,
|
|
73
|
+
type WizardQuestion,
|
|
74
|
+
type WizardResponse,
|
|
75
|
+
type WizardSession,
|
|
76
|
+
type WizardResult,
|
|
77
|
+
} from './wizard.js';
|
|
78
|
+
|
|
79
|
+
// CI Mode
|
|
80
|
+
export {
|
|
81
|
+
runCIKeywordResearch,
|
|
82
|
+
formatCIResult,
|
|
83
|
+
type CIKeywordOptions,
|
|
84
|
+
type CIKeywordResult,
|
|
85
|
+
type CIAction,
|
|
86
|
+
} from './ci-mode.js';
|
|
87
|
+
|
|
88
|
+
// Prioritization
|
|
89
|
+
export { prioritizeKeywords } from './prioritizer.js';
|
|
90
|
+
|
|
91
|
+
// Data sources
|
|
92
|
+
export {
|
|
93
|
+
getAutocompleteSuggestions,
|
|
94
|
+
getExpandedSuggestions,
|
|
95
|
+
enrichKeywordsWithEstimates,
|
|
96
|
+
} from './sources/autocomplete.js';
|
|
97
|
+
|
|
98
|
+
export {
|
|
99
|
+
transformGSCData,
|
|
100
|
+
findCTROpportunities,
|
|
101
|
+
findAlmostPage1Keywords,
|
|
102
|
+
findTopPerformers,
|
|
103
|
+
buildGSCRequest,
|
|
104
|
+
getDateRange,
|
|
105
|
+
} from './sources/gsc.js';
|
|
106
|
+
|
|
107
|
+
export {
|
|
108
|
+
getKeywordData,
|
|
109
|
+
getKeywordSuggestions,
|
|
110
|
+
getRelatedKeywords,
|
|
111
|
+
checkBalance,
|
|
112
|
+
LOCATION_CODES,
|
|
113
|
+
type DataForSEOCredentials,
|
|
114
|
+
} from './sources/dataforseo.js';
|
|
115
|
+
|
|
116
|
+
// Competitor Analysis (SpyFu Kombat-style)
|
|
117
|
+
export {
|
|
118
|
+
discoverCompetitorKeywords,
|
|
119
|
+
formatCompetitorReport,
|
|
120
|
+
type CompetitorKeywordResult,
|
|
121
|
+
type CompetitorOverlap,
|
|
122
|
+
} from './sources/competitor-analysis.js';
|
|
123
|
+
|
|
124
|
+
// Topic Grouping/Clustering
|
|
125
|
+
export {
|
|
126
|
+
groupKeywordsByTopic,
|
|
127
|
+
formatTopicReport,
|
|
128
|
+
type KeywordTopic,
|
|
129
|
+
type TopicClusterResult,
|
|
130
|
+
type ContentRecommendation,
|
|
131
|
+
} from './topic-grouping.js';
|
|
132
|
+
|
|
133
|
+
// Competitive Search (free APIs: Brave, Serper, GitHub, npm)
|
|
134
|
+
export {
|
|
135
|
+
searchCompetitors,
|
|
136
|
+
searchFormatConverters,
|
|
137
|
+
searchHackerNews,
|
|
138
|
+
type CompetitiveSearchResult,
|
|
139
|
+
type CompetitorTool,
|
|
140
|
+
type CompetitiveSearchOptions,
|
|
141
|
+
} from './sources/competitive-search.js';
|
|
142
|
+
|
|
143
|
+
// LLM Judge for Tool Feasibility
|
|
144
|
+
export {
|
|
145
|
+
evaluateToolFeasibility,
|
|
146
|
+
enhanceToolIdea,
|
|
147
|
+
evaluateAndEnhanceToolIdeas,
|
|
148
|
+
type ToolFeasibilityScore,
|
|
149
|
+
type EnhancedToolIdea,
|
|
150
|
+
type LLMJudgeOptions,
|
|
151
|
+
} from './llm-judge.js';
|