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