@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,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
+ };