@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,275 @@
1
+ // IndexNow Protocol Implementation Check
2
+ // IndexNow enables instant indexing by Bing, Yandex, Seznam, Naver, and other participating search engines
3
+
4
+ import { httpGet } from '../../utils/http.js';
5
+ import * as cheerio from 'cheerio';
6
+ import type { AuditIssue } from '../types.js';
7
+
8
+ export interface IndexNowData {
9
+ hasKey: boolean;
10
+ keyFileUrl: string | null;
11
+ keyValue: string | null;
12
+ keyValid: boolean;
13
+ hasLinkTag: boolean;
14
+ inRobotsTxt: boolean;
15
+ cmsPlugin: string | null;
16
+ }
17
+
18
+ // Common IndexNow key file locations
19
+ const KEY_FILE_PATTERNS = [
20
+ // Standard pattern: /{key}.txt
21
+ /^[a-f0-9]{32}\.txt$/i,
22
+ /^[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}\.txt$/i,
23
+ ];
24
+
25
+ // CMS plugins that implement IndexNow
26
+ const CMS_PLUGINS: Record<string, string[]> = {
27
+ WordPress: ['IndexNow', 'Yoast SEO Premium', 'Rank Math SEO', 'All in One SEO'],
28
+ Wix: ['IndexNow built-in'],
29
+ Cloudflare: ['Cloudflare IndexNow'],
30
+ };
31
+
32
+ /**
33
+ * Check for IndexNow key in robots.txt
34
+ */
35
+ export async function checkRobotsTxtForIndexNow(
36
+ baseUrl: string
37
+ ): Promise<{ found: boolean; key: string | null }> {
38
+ try {
39
+ const response = await httpGet<string>(`${new URL(baseUrl).origin}/robots.txt`, {
40
+
41
+ timeout: 10000,
42
+ validateStatus: () => true,
43
+ });
44
+
45
+ if (response.status !== 200) {
46
+ return { found: false, key: null };
47
+ }
48
+
49
+ const content = response.data as string;
50
+
51
+ // Look for IndexNow comment or directive
52
+ const indexNowPattern = /indexnow[:\s]+([a-f0-9-]+)/i;
53
+ const match = content.match(indexNowPattern);
54
+
55
+ if (match) {
56
+ return { found: true, key: match[1] };
57
+ }
58
+
59
+ return { found: false, key: null };
60
+ } catch {
61
+ return { found: false, key: null };
62
+ }
63
+ }
64
+
65
+ /**
66
+ * Validate IndexNow key format
67
+ */
68
+ export function isValidIndexNowKey(key: string): boolean {
69
+ // Key should be 8-128 characters, hexadecimal or UUID format
70
+ if (key.length < 8 || key.length > 128) return false;
71
+
72
+ // UUID format
73
+ if (/^[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}$/i.test(key)) {
74
+ return true;
75
+ }
76
+
77
+ // Hex format (32 chars)
78
+ if (/^[a-f0-9]{32}$/i.test(key)) {
79
+ return true;
80
+ }
81
+
82
+ // Other alphanumeric formats
83
+ if (/^[a-zA-Z0-9-]{8,128}$/.test(key)) {
84
+ return true;
85
+ }
86
+
87
+ return false;
88
+ }
89
+
90
+ /**
91
+ * Try to find and validate IndexNow key file
92
+ */
93
+ export async function findIndexNowKeyFile(
94
+ baseUrl: string,
95
+ possibleKeys: string[]
96
+ ): Promise<{ found: boolean; url: string | null; key: string | null }> {
97
+ const origin = new URL(baseUrl).origin;
98
+
99
+ for (const key of possibleKeys) {
100
+ if (!isValidIndexNowKey(key)) continue;
101
+
102
+ const keyUrl = `${origin}/${key}.txt`;
103
+
104
+ try {
105
+ const response = await httpGet<string>(keyUrl, {
106
+
107
+ timeout: 5000,
108
+ validateStatus: () => true,
109
+ });
110
+
111
+ if (response.status === 200) {
112
+ const content = (response.data as string).trim();
113
+ if (content === key) {
114
+ return { found: true, url: keyUrl, key };
115
+ }
116
+ }
117
+ } catch {
118
+ // Key file not found
119
+ }
120
+ }
121
+
122
+ return { found: false, url: null, key: null };
123
+ }
124
+
125
+ /**
126
+ * Check for IndexNow link tag in HTML
127
+ */
128
+ export function checkIndexNowLinkTag(html: string): { found: boolean; href: string | null } {
129
+ const $ = cheerio.load(html);
130
+
131
+ // Check for indexnow link tag (non-standard but sometimes used)
132
+ const indexNowLink =
133
+ $('link[rel="indexnow"]').attr('href') ||
134
+ $('link[rel="index-now"]').attr('href') ||
135
+ $('meta[name="indexnow-key"]').attr('content') ||
136
+ null;
137
+
138
+ return { found: !!indexNowLink, href: indexNowLink };
139
+ }
140
+
141
+ /**
142
+ * Detect CMS/framework and potential IndexNow plugin
143
+ */
144
+ export function detectIndexNowCMSSupport(html: string, headers: Record<string, string>): string | null {
145
+ const $ = cheerio.load(html);
146
+
147
+ // Check for WordPress
148
+ const isWordPress =
149
+ html.includes('wp-content') ||
150
+ html.includes('wp-includes') ||
151
+ $('meta[name="generator"][content*="WordPress"]').length > 0;
152
+
153
+ if (isWordPress) {
154
+ // Check for known IndexNow plugins
155
+ if (html.includes('yoast-seo') || html.includes('wordpress-seo')) {
156
+ return 'WordPress + Yoast SEO (may have IndexNow)';
157
+ }
158
+ if (html.includes('rank-math') || html.includes('rankmath')) {
159
+ return 'WordPress + Rank Math (has IndexNow)';
160
+ }
161
+ return 'WordPress (IndexNow plugins available)';
162
+ }
163
+
164
+ // Check for Wix
165
+ if (html.includes('wix.com') || html.includes('_wix_browser_sess')) {
166
+ return 'Wix (IndexNow built-in)';
167
+ }
168
+
169
+ // Check for Cloudflare
170
+ if (headers['cf-ray'] || headers['server']?.includes('cloudflare')) {
171
+ return 'Cloudflare (IndexNow integration available)';
172
+ }
173
+
174
+ return null;
175
+ }
176
+
177
+ /**
178
+ * Main function: Analyze IndexNow implementation
179
+ */
180
+ export async function analyzeIndexNow(
181
+ html: string,
182
+ url: string,
183
+ headers: Record<string, string> = {}
184
+ ): Promise<{ issues: AuditIssue[]; data: IndexNowData }> {
185
+ const issues: AuditIssue[] = [];
186
+ const origin = new URL(url).origin;
187
+
188
+ // Check for link tag
189
+ const linkTag = checkIndexNowLinkTag(html);
190
+
191
+ // Check robots.txt
192
+ const robotsCheck = await checkRobotsTxtForIndexNow(url);
193
+
194
+ // Try to find key file with any discovered keys
195
+ const possibleKeys: string[] = [];
196
+ if (linkTag.href) possibleKeys.push(linkTag.href);
197
+ if (robotsCheck.key) possibleKeys.push(robotsCheck.key);
198
+
199
+ // Also try common key patterns
200
+ // Note: In a real implementation, you might crawl the root directory
201
+ // For now, we'll just check if the key file exists with discovered keys
202
+
203
+ const keyFile = await findIndexNowKeyFile(url, possibleKeys);
204
+
205
+ // Detect CMS support
206
+ const cmsPlugin = detectIndexNowCMSSupport(html, headers);
207
+
208
+ const data: IndexNowData = {
209
+ hasKey: keyFile.found,
210
+ keyFileUrl: keyFile.url,
211
+ keyValue: keyFile.key,
212
+ keyValid: keyFile.key ? isValidIndexNowKey(keyFile.key) : false,
213
+ hasLinkTag: linkTag.found,
214
+ inRobotsTxt: robotsCheck.found,
215
+ cmsPlugin,
216
+ };
217
+
218
+ // Generate issues
219
+ if (!keyFile.found && !linkTag.found && !robotsCheck.found) {
220
+ const suggestion = cmsPlugin
221
+ ? `Your platform (${cmsPlugin}) may support IndexNow. Check for built-in support or plugins.`
222
+ : 'Create a key at indexnow.org, add the key file to your root directory, and submit URLs via the IndexNow API.';
223
+
224
+ issues.push({
225
+ code: 'INDEXNOW_NOT_IMPLEMENTED',
226
+ severity: 'notice',
227
+ category: 'crawlability',
228
+ title: 'IndexNow not implemented',
229
+ description: 'Site does not appear to have IndexNow instant indexing set up.',
230
+ impact: 'Content changes may take longer to appear in Bing, Yandex, and other IndexNow-supporting search engines.',
231
+ howToFix: suggestion,
232
+ affectedUrls: [url],
233
+ details: { cmsPlugin },
234
+ });
235
+ } else if (keyFile.found && !data.keyValid) {
236
+ issues.push({
237
+ code: 'INDEXNOW_INVALID_KEY',
238
+ severity: 'warning',
239
+ category: 'crawlability',
240
+ title: 'IndexNow key format invalid',
241
+ description: `The IndexNow key "${keyFile.key}" does not match expected formats.`,
242
+ impact: 'IndexNow submissions may fail with invalid key format.',
243
+ howToFix: 'Use a valid key format: 32-character hex or UUID format.',
244
+ affectedUrls: [keyFile.url || url],
245
+ });
246
+ }
247
+
248
+ // Informational notice if IndexNow is implemented
249
+ if (keyFile.found && data.keyValid) {
250
+ // This is good! No issue needed, but we track the data
251
+ }
252
+
253
+ return { issues, data };
254
+ }
255
+
256
+ /**
257
+ * Generate IndexNow submission URL for a page
258
+ */
259
+ export function generateIndexNowSubmitUrl(
260
+ pageUrl: string,
261
+ key: string,
262
+ searchEngine: 'bing' | 'yandex' | 'seznam' | 'naver' = 'bing'
263
+ ): string {
264
+ const endpoints: Record<string, string> = {
265
+ bing: 'https://www.bing.com/indexnow',
266
+ yandex: 'https://yandex.com/indexnow',
267
+ seznam: 'https://search.seznam.cz/indexnow',
268
+ naver: 'https://searchadvisor.naver.com/indexnow',
269
+ };
270
+
271
+ const endpoint = endpoints[searchEngine];
272
+ const parsedUrl = new URL(pageUrl);
273
+
274
+ return `${endpoint}?url=${encodeURIComponent(pageUrl)}&key=${key}&keyLocation=${encodeURIComponent(`${parsedUrl.origin}/${key}.txt`)}`;
275
+ }