@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,636 @@
1
+ /**
2
+ * Tracking Verification Checks
3
+ *
4
+ * Verifies that essential tracking and verification codes are present:
5
+ * - Google Analytics 4 (gtag.js)
6
+ * - Google Search Console verification
7
+ * - Bing Webmaster Tools verification
8
+ * - Google Tag Manager
9
+ * - Schema.org structured data
10
+ */
11
+
12
+ import * as cheerio from 'cheerio';
13
+ import type { AuditIssue } from '../types.js';
14
+
15
+ // =============================================================================
16
+ // Local Check Types (for compatibility with check runner)
17
+ // =============================================================================
18
+ interface PageContext {
19
+ url: string;
20
+ html: string;
21
+ $: cheerio.CheerioAPI;
22
+ }
23
+
24
+ interface CheckResult {
25
+ id: string;
26
+ checkId: string;
27
+ type: string;
28
+ category: string;
29
+ title: string;
30
+ description: string;
31
+ affectedElement?: string;
32
+ recommendation: string;
33
+ impact: string;
34
+ effort: string;
35
+ learnMoreUrl?: string;
36
+ autoFixable?: boolean;
37
+ fixCode?: string;
38
+ }
39
+
40
+ interface AuditCheck {
41
+ id: string;
42
+ name: string;
43
+ description: string;
44
+ category: string;
45
+ severity: string;
46
+ impact: string;
47
+ check(context: PageContext): Promise<CheckResult | null>;
48
+ }
49
+
50
+ // =============================================================================
51
+ // Tracking Data Interface
52
+ // =============================================================================
53
+ export interface TrackingData {
54
+ ga4: {
55
+ detected: boolean;
56
+ measurementId?: string;
57
+ };
58
+ gtm: {
59
+ detected: boolean;
60
+ containerId?: string;
61
+ hasNoscript: boolean;
62
+ };
63
+ gsc: {
64
+ verified: boolean;
65
+ verificationCode?: string;
66
+ };
67
+ bing: {
68
+ verified: boolean;
69
+ verificationCode?: string;
70
+ };
71
+ schemaOrg: {
72
+ detected: boolean;
73
+ types: string[];
74
+ hasValidJson: boolean;
75
+ };
76
+ sitemap: {
77
+ inRobotsTxt: boolean;
78
+ };
79
+ }
80
+
81
+ // =============================================================================
82
+ // Main Analyze Function
83
+ // =============================================================================
84
+ export function analyzeTrackingVerification(
85
+ html: string,
86
+ url: string
87
+ ): { issues: AuditIssue[]; data: TrackingData } {
88
+ const issues: AuditIssue[] = [];
89
+ const $ = cheerio.load(html);
90
+
91
+ // Detect GA4
92
+ const ga4Match = html.match(/gtag\s*\(\s*['"]config['"]\s*,\s*['"](G-[A-Z0-9]+)['"]/i) ||
93
+ html.match(/googletagmanager\.com\/gtag\/js\?id=(G-[A-Z0-9]+)/i);
94
+ const hasGa4 = !!ga4Match;
95
+ const ga4MeasurementId = ga4Match ? ga4Match[1] : undefined;
96
+
97
+ // Detect GTM
98
+ const gtmMatch = html.match(/googletagmanager\.com\/gtm\.js\?id=(GTM-[A-Z0-9]+)/i);
99
+ const hasGtm = !!gtmMatch;
100
+ const gtmContainerId = gtmMatch ? gtmMatch[1] : undefined;
101
+ const hasGtmNoscript = /googletagmanager\.com\/ns\.html\?id=GTM-[A-Z0-9]+/i.test(html);
102
+
103
+ // Detect GSC verification
104
+ const gscMeta = $('meta[name="google-site-verification"]').attr('content');
105
+ const hasGscVerification = !!(gscMeta && gscMeta.length > 10);
106
+
107
+ // Detect Bing verification
108
+ const bingMeta = $('meta[name="msvalidate.01"]').attr('content');
109
+ const hasBingVerification = !!(bingMeta && bingMeta.length > 10);
110
+
111
+ // Detect Schema.org
112
+ const jsonLdScripts = $('script[type="application/ld+json"]');
113
+ const hasMicrodata = $('[itemtype*="schema.org"]').length > 0;
114
+ const hasJsonLd = jsonLdScripts.length > 0;
115
+
116
+ const schemaTypes: string[] = [];
117
+ let hasValidJson = true;
118
+
119
+ jsonLdScripts.each((_, el) => {
120
+ try {
121
+ const content = $(el).html();
122
+ if (content) {
123
+ const parsed = JSON.parse(content);
124
+ if (parsed['@type']) {
125
+ schemaTypes.push(parsed['@type']);
126
+ }
127
+ }
128
+ } catch {
129
+ hasValidJson = false;
130
+ }
131
+ });
132
+
133
+ const data: TrackingData = {
134
+ ga4: { detected: hasGa4, measurementId: ga4MeasurementId },
135
+ gtm: { detected: hasGtm, containerId: gtmContainerId, hasNoscript: hasGtmNoscript },
136
+ gsc: { verified: hasGscVerification, verificationCode: gscMeta },
137
+ bing: { verified: hasBingVerification, verificationCode: bingMeta },
138
+ schemaOrg: { detected: hasJsonLd || hasMicrodata, types: schemaTypes, hasValidJson },
139
+ sitemap: { inRobotsTxt: false }, // Set later if we check robots.txt
140
+ };
141
+
142
+ // ==================== GA4 Check ====================
143
+ if (!hasGa4 && !hasGtm) {
144
+ issues.push({
145
+ code: 'GA4_MISSING',
146
+ severity: 'warning',
147
+ category: 'on-page',
148
+ title: 'Google Analytics 4 not detected',
149
+ description:
150
+ 'No GA4 tracking code found. GA4 is essential for measuring organic traffic and conversions.',
151
+ impact: 'Cannot measure SEO impact without analytics tracking.',
152
+ howToFix:
153
+ 'Add GA4 tracking code to your site. Get your measurement ID from analytics.google.com.',
154
+ affectedUrls: [url],
155
+ });
156
+ }
157
+
158
+ // ==================== GSC Verification Check ====================
159
+ if (!hasGscVerification) {
160
+ issues.push({
161
+ code: 'GSC_VERIFICATION_MISSING',
162
+ severity: 'warning',
163
+ category: 'on-page',
164
+ title: 'Google Search Console verification not found',
165
+ description:
166
+ 'No GSC verification meta tag detected. GSC provides essential data about your search performance.',
167
+ impact: 'Missing critical insights into Google search visibility, impressions, and clicks.',
168
+ howToFix:
169
+ 'Add your GSC verification meta tag to the <head> section. Get it from search.google.com/search-console.',
170
+ affectedUrls: [url],
171
+ });
172
+ }
173
+
174
+ // ==================== Bing Verification Check ====================
175
+ if (!hasBingVerification) {
176
+ issues.push({
177
+ code: 'BING_VERIFICATION_MISSING',
178
+ severity: 'notice',
179
+ category: 'on-page',
180
+ title: 'Bing Webmaster Tools verification not found',
181
+ description:
182
+ 'No Bing verification meta tag detected. Bing/Yahoo account for 5-8% of search traffic.',
183
+ impact: 'Missing insights on Bing/Yahoo search traffic.',
184
+ howToFix:
185
+ 'Add Bing Webmaster Tools verification for insights on Bing/Yahoo traffic. Get it from bing.com/webmasters.',
186
+ affectedUrls: [url],
187
+ });
188
+ }
189
+
190
+ // ==================== GTM Check (informational) ====================
191
+ if (!hasGtm) {
192
+ issues.push({
193
+ code: 'GTM_MISSING',
194
+ severity: 'notice',
195
+ category: 'on-page',
196
+ title: 'Google Tag Manager not detected',
197
+ description:
198
+ 'GTM not found. While not required, GTM makes it easier to manage tracking tags without code changes.',
199
+ impact: 'Tag management requires developer involvement for each change.',
200
+ howToFix:
201
+ 'Consider using GTM for easier tag management. It allows marketing to add tags without developer help.',
202
+ affectedUrls: [url],
203
+ });
204
+ } else if (!hasGtmNoscript) {
205
+ issues.push({
206
+ code: 'GTM_NOSCRIPT_MISSING',
207
+ severity: 'notice',
208
+ category: 'on-page',
209
+ title: 'GTM noscript fallback missing',
210
+ description:
211
+ 'GTM is installed but the noscript fallback is missing. This is needed for users with JavaScript disabled.',
212
+ impact: 'Users with JS disabled won\'t be tracked.',
213
+ howToFix: 'Add the GTM noscript tag immediately after the opening <body> tag.',
214
+ affectedUrls: [url],
215
+ });
216
+ }
217
+
218
+ // ==================== Schema.org Check ====================
219
+ if (!hasJsonLd && !hasMicrodata) {
220
+ issues.push({
221
+ code: 'SCHEMA_ORG_MISSING',
222
+ severity: 'warning',
223
+ category: 'structured-data',
224
+ title: 'No structured data found',
225
+ description:
226
+ 'No Schema.org structured data detected. Structured data helps search engines understand your content and can enable rich snippets.',
227
+ impact: 'Missing opportunity for rich results in search (stars, images, FAQs, etc.).',
228
+ howToFix:
229
+ 'Add JSON-LD structured data for your content type (Organization, Product, Article, etc.).',
230
+ affectedUrls: [url],
231
+ });
232
+ } else if (hasJsonLd && !hasValidJson) {
233
+ issues.push({
234
+ code: 'SCHEMA_ORG_INVALID',
235
+ severity: 'error',
236
+ category: 'structured-data',
237
+ title: 'Invalid JSON-LD structured data',
238
+ description:
239
+ 'One or more JSON-LD scripts contain invalid JSON. This will prevent search engines from parsing your structured data.',
240
+ impact: 'Structured data is being ignored due to syntax errors.',
241
+ howToFix:
242
+ 'Fix the JSON syntax errors in your structured data. Use the Rich Results Test to validate.',
243
+ affectedUrls: [url],
244
+ });
245
+ }
246
+
247
+ return { issues, data };
248
+ }
249
+
250
+ // =============================================================================
251
+ // Google Analytics 4 Check
252
+ // =============================================================================
253
+ export const ga4TrackingCheck: AuditCheck = {
254
+ id: 'ga4-tracking',
255
+ name: 'Google Analytics 4 Tracking',
256
+ description: 'Checks if GA4 tracking code is properly installed',
257
+ category: 'on-page',
258
+ severity: 'warning',
259
+ impact: 'medium',
260
+
261
+ async check(context: PageContext): Promise<CheckResult | null> {
262
+ const { html } = context;
263
+
264
+ // Look for GA4 patterns
265
+ const ga4Patterns = [
266
+ // gtag.js with G- measurement ID
267
+ /gtag\s*\(\s*['"]config['"]\s*,\s*['"]G-[A-Z0-9]+['"]/i,
268
+ // gtag.js script
269
+ /googletagmanager\.com\/gtag\/js\?id=G-[A-Z0-9]+/i,
270
+ // GA4 measurement protocol
271
+ /G-[A-Z0-9]{10,}/,
272
+ ];
273
+
274
+ const hasGa4 = ga4Patterns.some((pattern) => pattern.test(html));
275
+
276
+ // Also check for GTM (which might contain GA4)
277
+ const hasGtm = /googletagmanager\.com\/gtm\.js/.test(html);
278
+
279
+ if (!hasGa4 && !hasGtm) {
280
+ return {
281
+ id: 'ga4-tracking',
282
+ checkId: 'ga4-tracking',
283
+ type: 'warning',
284
+ category: 'on-page',
285
+ title: 'Google Analytics 4 not detected',
286
+ description:
287
+ 'No GA4 tracking code found. GA4 is essential for measuring organic traffic and conversions.',
288
+ affectedElement: '<head>',
289
+ recommendation:
290
+ 'Add GA4 tracking code to your site. Get your measurement ID from analytics.google.com.',
291
+ impact: 'medium',
292
+ effort: 'low',
293
+ learnMoreUrl: 'https://support.google.com/analytics/answer/9304153',
294
+ autoFixable: true,
295
+ fixCode: `<!-- Add to <head> -->
296
+ <script async src="https://www.googletagmanager.com/gtag/js?id=G-XXXXXXXXXX"></script>
297
+ <script>
298
+ window.dataLayer = window.dataLayer || [];
299
+ function gtag(){dataLayer.push(arguments);}
300
+ gtag('js', new Date());
301
+ gtag('config', 'G-XXXXXXXXXX');
302
+ </script>`,
303
+ };
304
+ }
305
+
306
+ return null;
307
+ },
308
+ };
309
+
310
+ // =============================================================================
311
+ // Google Search Console Verification Check
312
+ // =============================================================================
313
+ export const gscVerificationCheck: AuditCheck = {
314
+ id: 'gsc-verification',
315
+ name: 'Google Search Console Verification',
316
+ description: 'Checks if GSC verification meta tag is present',
317
+ category: 'on-page',
318
+ severity: 'warning',
319
+ impact: 'high',
320
+
321
+ async check(context: PageContext): Promise<CheckResult | null> {
322
+ const { $ } = context;
323
+
324
+ // Check for GSC verification meta tag
325
+ const gscMeta = $('meta[name="google-site-verification"]').attr('content');
326
+
327
+ // Also check for other verification methods mentioned in HTML
328
+ const hasGscVerification = gscMeta && gscMeta.length > 10;
329
+
330
+ if (!hasGscVerification) {
331
+ return {
332
+ id: 'gsc-verification',
333
+ checkId: 'gsc-verification',
334
+ type: 'warning',
335
+ category: 'on-page',
336
+ title: 'Google Search Console verification not found',
337
+ description:
338
+ 'No GSC verification meta tag detected. GSC provides essential data about your search performance.',
339
+ affectedElement: '<head>',
340
+ recommendation:
341
+ 'Add your GSC verification meta tag to the <head> section. Get it from search.google.com/search-console.',
342
+ impact: 'high',
343
+ effort: 'low',
344
+ learnMoreUrl: 'https://support.google.com/webmasters/answer/9008080',
345
+ autoFixable: true,
346
+ fixCode: `<!-- Add to <head> -->
347
+ <meta name="google-site-verification" content="YOUR_VERIFICATION_CODE" />`,
348
+ };
349
+ }
350
+
351
+ return null;
352
+ },
353
+ };
354
+
355
+ // =============================================================================
356
+ // Bing Webmaster Tools Verification Check
357
+ // =============================================================================
358
+ export const bingVerificationCheck: AuditCheck = {
359
+ id: 'bing-verification',
360
+ name: 'Bing Webmaster Tools Verification',
361
+ description: 'Checks if Bing Webmaster Tools verification is present',
362
+ category: 'on-page',
363
+ severity: 'notice',
364
+ impact: 'low',
365
+
366
+ async check(context: PageContext): Promise<CheckResult | null> {
367
+ const { $ } = context;
368
+
369
+ // Check for Bing verification meta tag
370
+ const bingMeta = $('meta[name="msvalidate.01"]').attr('content');
371
+
372
+ const hasBingVerification = bingMeta && bingMeta.length > 10;
373
+
374
+ if (!hasBingVerification) {
375
+ return {
376
+ id: 'bing-verification',
377
+ checkId: 'bing-verification',
378
+ type: 'notice',
379
+ category: 'on-page',
380
+ title: 'Bing Webmaster Tools verification not found',
381
+ description:
382
+ 'No Bing verification meta tag detected. Bing/Yahoo account for 5-8% of search traffic.',
383
+ affectedElement: '<head>',
384
+ recommendation:
385
+ 'Add Bing Webmaster Tools verification for insights on Bing/Yahoo traffic. Get it from bing.com/webmasters.',
386
+ impact: 'low',
387
+ effort: 'low',
388
+ learnMoreUrl: 'https://www.bing.com/webmasters/help/how-to-verify-ownership-of-your-site-afcfefc6',
389
+ autoFixable: true,
390
+ fixCode: `<!-- Add to <head> -->
391
+ <meta name="msvalidate.01" content="YOUR_BING_VERIFICATION_CODE" />`,
392
+ };
393
+ }
394
+
395
+ return null;
396
+ },
397
+ };
398
+
399
+ // =============================================================================
400
+ // Google Tag Manager Check
401
+ // =============================================================================
402
+ export const gtmCheck: AuditCheck = {
403
+ id: 'gtm-installed',
404
+ name: 'Google Tag Manager',
405
+ description: 'Checks if GTM is installed for flexible tag management',
406
+ category: 'on-page',
407
+ severity: 'notice',
408
+ impact: 'low',
409
+
410
+ async check(context: PageContext): Promise<CheckResult | null> {
411
+ const { html } = context;
412
+
413
+ // Check for GTM
414
+ const gtmPattern = /googletagmanager\.com\/gtm\.js\?id=GTM-[A-Z0-9]+/i;
415
+ const hasGtm = gtmPattern.test(html);
416
+
417
+ // Also check for noscript GTM
418
+ const gtmNoscript = /googletagmanager\.com\/ns\.html\?id=GTM-[A-Z0-9]+/i;
419
+ const hasGtmNoscript = gtmNoscript.test(html);
420
+
421
+ if (!hasGtm) {
422
+ return {
423
+ id: 'gtm-installed',
424
+ checkId: 'gtm-installed',
425
+ type: 'notice',
426
+ category: 'on-page',
427
+ title: 'Google Tag Manager not detected',
428
+ description:
429
+ 'GTM not found. While not required, GTM makes it easier to manage tracking tags without code changes.',
430
+ affectedElement: '<head>',
431
+ recommendation:
432
+ 'Consider using GTM for easier tag management. It allows marketing to add tags without developer help.',
433
+ impact: 'low',
434
+ effort: 'medium',
435
+ learnMoreUrl: 'https://tagmanager.google.com/',
436
+ autoFixable: true,
437
+ fixCode: `<!-- Add to <head> -->
438
+ <script>(function(w,d,s,l,i){w[l]=w[l]||[];w[l].push({'gtm.start':
439
+ new Date().getTime(),event:'gtm.js'});var f=d.getElementsByTagName(s)[0],
440
+ j=d.createElement(s),dl=l!='dataLayer'?'&l='+l:'';j.async=true;j.src=
441
+ 'https://www.googletagmanager.com/gtm.js?id='+i+dl;f.parentNode.insertBefore(j,f);
442
+ })(window,document,'script','dataLayer','GTM-XXXXXXX');</script>
443
+
444
+ <!-- Add immediately after opening <body> -->
445
+ <noscript><iframe src="https://www.googletagmanager.com/ns.html?id=GTM-XXXXXXX"
446
+ height="0" width="0" style="display:none;visibility:hidden"></iframe></noscript>`,
447
+ };
448
+ }
449
+
450
+ // Check if noscript fallback is present
451
+ if (hasGtm && !hasGtmNoscript) {
452
+ return {
453
+ id: 'gtm-noscript-missing',
454
+ checkId: 'gtm-installed',
455
+ type: 'notice',
456
+ category: 'on-page',
457
+ title: 'GTM noscript fallback missing',
458
+ description:
459
+ 'GTM is installed but the noscript fallback is missing. This is needed for users with JavaScript disabled.',
460
+ affectedElement: '<body>',
461
+ recommendation: 'Add the GTM noscript tag immediately after the opening <body> tag.',
462
+ impact: 'low',
463
+ effort: 'low',
464
+ learnMoreUrl: 'https://developers.google.com/tag-manager/quickstart',
465
+ };
466
+ }
467
+
468
+ return null;
469
+ },
470
+ };
471
+
472
+ // =============================================================================
473
+ // Schema.org Structured Data Check
474
+ // =============================================================================
475
+ export const schemaOrgCheck: AuditCheck = {
476
+ id: 'schema-org',
477
+ name: 'Schema.org Structured Data',
478
+ description: 'Checks for structured data markup',
479
+ category: 'structured-data',
480
+ severity: 'warning',
481
+ impact: 'medium',
482
+
483
+ async check(context: PageContext): Promise<CheckResult | null> {
484
+ const { $ } = context;
485
+
486
+ // Check for JSON-LD
487
+ const jsonLdScripts = $('script[type="application/ld+json"]');
488
+ const hasJsonLd = jsonLdScripts.length > 0;
489
+
490
+ // Check for microdata
491
+ const hasMicrodata = $('[itemtype*="schema.org"]').length > 0;
492
+
493
+ // Check for RDFa
494
+ const hasRdfa = $('[typeof]').length > 0 || $('[property]').length > 0;
495
+
496
+ if (!hasJsonLd && !hasMicrodata && !hasRdfa) {
497
+ return {
498
+ id: 'schema-org-missing',
499
+ checkId: 'schema-org',
500
+ type: 'warning',
501
+ category: 'structured-data',
502
+ title: 'No structured data found',
503
+ description:
504
+ 'No Schema.org structured data detected. Structured data helps search engines understand your content and can enable rich snippets.',
505
+ affectedElement: '<head>',
506
+ recommendation:
507
+ 'Add JSON-LD structured data for your content type (Organization, Product, Article, etc.).',
508
+ impact: 'medium',
509
+ effort: 'medium',
510
+ learnMoreUrl: 'https://developers.google.com/search/docs/appearance/structured-data',
511
+ autoFixable: true,
512
+ fixCode: `<!-- Add to <head> - Example Organization schema -->
513
+ <script type="application/ld+json">
514
+ {
515
+ "@context": "https://schema.org",
516
+ "@type": "Organization",
517
+ "name": "Your Company Name",
518
+ "url": "https://your-domain.com",
519
+ "logo": "https://your-domain.com/logo.png",
520
+ "sameAs": [
521
+ "https://twitter.com/yourcompany",
522
+ "https://linkedin.com/company/yourcompany"
523
+ ]
524
+ }
525
+ </script>`,
526
+ };
527
+ }
528
+
529
+ // Validate JSON-LD syntax
530
+ if (hasJsonLd) {
531
+ let hasInvalidJsonLd = false;
532
+ jsonLdScripts.each((_, el) => {
533
+ try {
534
+ const content = $(el).html();
535
+ if (content) {
536
+ JSON.parse(content);
537
+ }
538
+ } catch {
539
+ hasInvalidJsonLd = true;
540
+ }
541
+ });
542
+
543
+ if (hasInvalidJsonLd) {
544
+ return {
545
+ id: 'schema-org-invalid',
546
+ checkId: 'schema-org',
547
+ type: 'error',
548
+ category: 'structured-data',
549
+ title: 'Invalid JSON-LD structured data',
550
+ description:
551
+ 'One or more JSON-LD scripts contain invalid JSON. This will prevent search engines from parsing your structured data.',
552
+ affectedElement: 'script[type="application/ld+json"]',
553
+ recommendation:
554
+ 'Fix the JSON syntax errors in your structured data. Use the Rich Results Test to validate.',
555
+ impact: 'high',
556
+ effort: 'low',
557
+ learnMoreUrl: 'https://search.google.com/test/rich-results',
558
+ };
559
+ }
560
+ }
561
+
562
+ return null;
563
+ },
564
+ };
565
+
566
+ // =============================================================================
567
+ // Sitemap Reference Check
568
+ // =============================================================================
569
+ export const sitemapReferenceCheck: AuditCheck = {
570
+ id: 'sitemap-reference',
571
+ name: 'Sitemap Reference',
572
+ description: 'Checks if sitemap is referenced for search engines',
573
+ category: 'crawlability',
574
+ severity: 'warning',
575
+ impact: 'medium',
576
+
577
+ async check(context: PageContext): Promise<CheckResult | null> {
578
+ const { url } = context;
579
+
580
+ // Only check on homepage
581
+ const parsedUrl = new URL(url);
582
+ if (parsedUrl.pathname !== '/' && parsedUrl.pathname !== '') {
583
+ return null;
584
+ }
585
+
586
+ // Try to fetch robots.txt to check for sitemap reference
587
+ try {
588
+ const robotsUrl = new URL('/robots.txt', url).href;
589
+ const response = await fetch(robotsUrl, {
590
+ headers: { 'User-Agent': 'RankCLI-Audit/1.0' },
591
+ signal: AbortSignal.timeout(5000),
592
+ });
593
+
594
+ if (response.ok) {
595
+ const robotsTxt = await response.text();
596
+ const hasSitemapRef = /sitemap:/i.test(robotsTxt);
597
+
598
+ if (!hasSitemapRef) {
599
+ return {
600
+ id: 'sitemap-not-in-robots',
601
+ checkId: 'sitemap-reference',
602
+ type: 'warning',
603
+ category: 'crawlability',
604
+ title: 'Sitemap not referenced in robots.txt',
605
+ description:
606
+ 'Your robots.txt does not reference a sitemap. Adding a sitemap reference helps search engines discover all your pages.',
607
+ affectedElement: '/robots.txt',
608
+ recommendation: 'Add a Sitemap directive to your robots.txt file.',
609
+ impact: 'medium',
610
+ effort: 'low',
611
+ learnMoreUrl: 'https://developers.google.com/search/docs/crawling-indexing/sitemaps/build-sitemap',
612
+ autoFixable: true,
613
+ fixCode: `# Add to robots.txt
614
+ Sitemap: ${new URL('/sitemap.xml', url).href}`,
615
+ };
616
+ }
617
+ }
618
+ } catch {
619
+ // Ignore fetch errors
620
+ }
621
+
622
+ return null;
623
+ },
624
+ };
625
+
626
+ // =============================================================================
627
+ // Export all tracking checks
628
+ // =============================================================================
629
+ export const trackingChecks: AuditCheck[] = [
630
+ ga4TrackingCheck,
631
+ gscVerificationCheck,
632
+ bingVerificationCheck,
633
+ gtmCheck,
634
+ schemaOrgCheck,
635
+ sitemapReferenceCheck,
636
+ ];