@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.
Files changed (178) hide show
  1. package/README.md +242 -0
  2. package/dist/analyzer-2CSWIQGD.mjs +6 -0
  3. package/dist/chunk-YNZYHEYM.mjs +774 -0
  4. package/dist/index.d.mts +4012 -0
  5. package/dist/index.d.ts +4012 -0
  6. package/dist/index.js +29672 -0
  7. package/dist/index.mjs +28602 -0
  8. package/package.json +53 -0
  9. package/scripts/build-deno.ts +134 -0
  10. package/src/audit/ai/analyzer.ts +347 -0
  11. package/src/audit/ai/index.ts +29 -0
  12. package/src/audit/ai/prompts/content-analysis.ts +271 -0
  13. package/src/audit/ai/types.ts +179 -0
  14. package/src/audit/checks/additional-checks.ts +439 -0
  15. package/src/audit/checks/ai-citation-worthiness.ts +399 -0
  16. package/src/audit/checks/ai-content-structure.ts +325 -0
  17. package/src/audit/checks/ai-readiness.ts +339 -0
  18. package/src/audit/checks/anchor-text.ts +179 -0
  19. package/src/audit/checks/answer-conciseness.ts +322 -0
  20. package/src/audit/checks/asset-minification.ts +270 -0
  21. package/src/audit/checks/bing-optimization.ts +206 -0
  22. package/src/audit/checks/brand-mention-optimization.ts +349 -0
  23. package/src/audit/checks/caching-headers.ts +305 -0
  24. package/src/audit/checks/canonical-advanced.ts +150 -0
  25. package/src/audit/checks/canonical-domain.ts +196 -0
  26. package/src/audit/checks/citation-quality.ts +358 -0
  27. package/src/audit/checks/client-rendering.ts +542 -0
  28. package/src/audit/checks/color-contrast.ts +342 -0
  29. package/src/audit/checks/content-freshness.ts +170 -0
  30. package/src/audit/checks/content-science.ts +589 -0
  31. package/src/audit/checks/conversion-elements.ts +526 -0
  32. package/src/audit/checks/crawlability.ts +220 -0
  33. package/src/audit/checks/directory-listing.ts +172 -0
  34. package/src/audit/checks/dom-analysis.ts +191 -0
  35. package/src/audit/checks/dom-size.ts +246 -0
  36. package/src/audit/checks/duplicate-content.ts +194 -0
  37. package/src/audit/checks/eeat-signals.ts +990 -0
  38. package/src/audit/checks/entity-seo.ts +396 -0
  39. package/src/audit/checks/featured-snippet.ts +473 -0
  40. package/src/audit/checks/freshness-signals.ts +443 -0
  41. package/src/audit/checks/funnel-intent.ts +463 -0
  42. package/src/audit/checks/hreflang.ts +174 -0
  43. package/src/audit/checks/html-compliance.ts +302 -0
  44. package/src/audit/checks/image-dimensions.ts +167 -0
  45. package/src/audit/checks/images.ts +160 -0
  46. package/src/audit/checks/indexnow.ts +275 -0
  47. package/src/audit/checks/interactive-tools.ts +475 -0
  48. package/src/audit/checks/internal-link-graph.ts +436 -0
  49. package/src/audit/checks/keyword-analysis.ts +239 -0
  50. package/src/audit/checks/keyword-cannibalization.ts +385 -0
  51. package/src/audit/checks/keyword-placement.ts +471 -0
  52. package/src/audit/checks/links.ts +203 -0
  53. package/src/audit/checks/llms-txt.ts +224 -0
  54. package/src/audit/checks/local-seo.ts +296 -0
  55. package/src/audit/checks/mobile.ts +167 -0
  56. package/src/audit/checks/modern-images.ts +226 -0
  57. package/src/audit/checks/navboost-signals.ts +395 -0
  58. package/src/audit/checks/on-page.ts +209 -0
  59. package/src/audit/checks/page-resources.ts +285 -0
  60. package/src/audit/checks/pagination.ts +180 -0
  61. package/src/audit/checks/performance.ts +153 -0
  62. package/src/audit/checks/platform-presence.ts +580 -0
  63. package/src/audit/checks/redirect-analysis.ts +153 -0
  64. package/src/audit/checks/redirect-chain.ts +389 -0
  65. package/src/audit/checks/resource-hints.ts +420 -0
  66. package/src/audit/checks/responsive-css.ts +247 -0
  67. package/src/audit/checks/responsive-images.ts +396 -0
  68. package/src/audit/checks/review-ecosystem.ts +415 -0
  69. package/src/audit/checks/robots-validation.ts +373 -0
  70. package/src/audit/checks/security-headers.ts +172 -0
  71. package/src/audit/checks/security.ts +144 -0
  72. package/src/audit/checks/serp-preview.ts +251 -0
  73. package/src/audit/checks/site-maturity.ts +444 -0
  74. package/src/audit/checks/social-meta.test.ts +275 -0
  75. package/src/audit/checks/social-meta.ts +134 -0
  76. package/src/audit/checks/soft-404.ts +151 -0
  77. package/src/audit/checks/structured-data.ts +238 -0
  78. package/src/audit/checks/tech-detection.ts +496 -0
  79. package/src/audit/checks/topical-clusters.ts +435 -0
  80. package/src/audit/checks/tracker-bloat.ts +462 -0
  81. package/src/audit/checks/tracking-verification.test.ts +371 -0
  82. package/src/audit/checks/tracking-verification.ts +636 -0
  83. package/src/audit/checks/url-safety.ts +682 -0
  84. package/src/audit/deno-entry.ts +66 -0
  85. package/src/audit/discovery/index.ts +15 -0
  86. package/src/audit/discovery/link-crawler.ts +232 -0
  87. package/src/audit/discovery/repo-routes.ts +347 -0
  88. package/src/audit/engine.ts +620 -0
  89. package/src/audit/fixes/index.ts +209 -0
  90. package/src/audit/fixes/social-meta-fixes.test.ts +329 -0
  91. package/src/audit/fixes/social-meta-fixes.ts +463 -0
  92. package/src/audit/index.ts +74 -0
  93. package/src/audit/runner.test.ts +299 -0
  94. package/src/audit/runner.ts +130 -0
  95. package/src/audit/types.ts +1953 -0
  96. package/src/content/featured-snippet.ts +367 -0
  97. package/src/content/generator.test.ts +534 -0
  98. package/src/content/generator.ts +501 -0
  99. package/src/content/headline.ts +317 -0
  100. package/src/content/index.ts +62 -0
  101. package/src/content/intent.ts +258 -0
  102. package/src/content/keyword-density.ts +349 -0
  103. package/src/content/readability.ts +262 -0
  104. package/src/executor.ts +336 -0
  105. package/src/fixer.ts +416 -0
  106. package/src/frameworks/detector.test.ts +248 -0
  107. package/src/frameworks/detector.ts +371 -0
  108. package/src/frameworks/index.ts +68 -0
  109. package/src/frameworks/recipes/angular.yaml +171 -0
  110. package/src/frameworks/recipes/astro.yaml +206 -0
  111. package/src/frameworks/recipes/django.yaml +180 -0
  112. package/src/frameworks/recipes/laravel.yaml +137 -0
  113. package/src/frameworks/recipes/nextjs.yaml +268 -0
  114. package/src/frameworks/recipes/nuxt.yaml +175 -0
  115. package/src/frameworks/recipes/rails.yaml +188 -0
  116. package/src/frameworks/recipes/react.yaml +202 -0
  117. package/src/frameworks/recipes/sveltekit.yaml +154 -0
  118. package/src/frameworks/recipes/vue.yaml +137 -0
  119. package/src/frameworks/recipes/wordpress.yaml +209 -0
  120. package/src/frameworks/suggestion-engine.ts +320 -0
  121. package/src/geo/geo-content.test.ts +305 -0
  122. package/src/geo/geo-content.ts +266 -0
  123. package/src/geo/geo-history.test.ts +473 -0
  124. package/src/geo/geo-history.ts +433 -0
  125. package/src/geo/geo-tracker.test.ts +359 -0
  126. package/src/geo/geo-tracker.ts +411 -0
  127. package/src/geo/index.ts +10 -0
  128. package/src/git/commit-helper.test.ts +261 -0
  129. package/src/git/commit-helper.ts +329 -0
  130. package/src/git/index.ts +12 -0
  131. package/src/git/pr-helper.test.ts +284 -0
  132. package/src/git/pr-helper.ts +307 -0
  133. package/src/index.ts +66 -0
  134. package/src/keywords/ai-keyword-engine.ts +1062 -0
  135. package/src/keywords/ai-summarizer.ts +387 -0
  136. package/src/keywords/ci-mode.ts +555 -0
  137. package/src/keywords/engine.ts +359 -0
  138. package/src/keywords/index.ts +151 -0
  139. package/src/keywords/llm-judge.ts +357 -0
  140. package/src/keywords/nlp-analysis.ts +706 -0
  141. package/src/keywords/prioritizer.ts +295 -0
  142. package/src/keywords/site-crawler.ts +342 -0
  143. package/src/keywords/sources/autocomplete.ts +139 -0
  144. package/src/keywords/sources/competitive-search.ts +450 -0
  145. package/src/keywords/sources/competitor-analysis.ts +374 -0
  146. package/src/keywords/sources/dataforseo.ts +206 -0
  147. package/src/keywords/sources/free-sources.ts +294 -0
  148. package/src/keywords/sources/gsc.ts +123 -0
  149. package/src/keywords/topic-grouping.ts +327 -0
  150. package/src/keywords/types.ts +144 -0
  151. package/src/keywords/wizard.ts +457 -0
  152. package/src/loader.ts +40 -0
  153. package/src/reports/index.ts +7 -0
  154. package/src/reports/report-generator.test.ts +293 -0
  155. package/src/reports/report-generator.ts +713 -0
  156. package/src/scheduler/alerts.test.ts +458 -0
  157. package/src/scheduler/alerts.ts +328 -0
  158. package/src/scheduler/index.ts +8 -0
  159. package/src/scheduler/scheduled-audit.test.ts +377 -0
  160. package/src/scheduler/scheduled-audit.ts +149 -0
  161. package/src/test/integration-test.ts +325 -0
  162. package/src/tools/analyzer.ts +373 -0
  163. package/src/tools/crawl.ts +293 -0
  164. package/src/tools/files.ts +301 -0
  165. package/src/tools/h1-fixer.ts +249 -0
  166. package/src/tools/index.ts +67 -0
  167. package/src/tracking/github-action.ts +326 -0
  168. package/src/tracking/google-analytics.ts +265 -0
  169. package/src/tracking/index.ts +45 -0
  170. package/src/tracking/report-generator.ts +386 -0
  171. package/src/tracking/search-console.ts +335 -0
  172. package/src/types.ts +134 -0
  173. package/src/utils/http.ts +302 -0
  174. package/src/wasm-adapter.ts +297 -0
  175. package/src/wasm-entry.ts +14 -0
  176. package/tsconfig.json +17 -0
  177. package/tsup.wasm.config.ts +26 -0
  178. 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';