@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,342 @@
1
+ /**
2
+ * Color Contrast Accessibility Check
3
+ *
4
+ * Analyzes color contrast for WCAG 2.1 compliance.
5
+ *
6
+ * WCAG Requirements:
7
+ * - AA: 4.5:1 for normal text, 3:1 for large text (18pt+ or 14pt bold)
8
+ * - AAA: 7:1 for normal text, 4.5:1 for large text
9
+ *
10
+ * Note: Full contrast checking requires rendering the page and computing
11
+ * styles. This check analyzes inline styles and common patterns as a
12
+ * heuristic approach.
13
+ */
14
+
15
+ import * as cheerio from 'cheerio';
16
+ import type { AuditIssue } from '../types.js';
17
+
18
+ export interface ColorContrastData {
19
+ elementsAnalyzed: number;
20
+ potentialIssues: ContrastIssue[];
21
+ passedChecks: number;
22
+ failedChecks: number;
23
+ cannotDetermine: number;
24
+ }
25
+
26
+ interface ContrastIssue {
27
+ selector: string;
28
+ foreground: string;
29
+ background: string;
30
+ contrastRatio: number;
31
+ wcagLevel: 'AA' | 'AAA' | 'none';
32
+ isLargeText: boolean;
33
+ }
34
+
35
+ // Common problematic color combinations
36
+ const KNOWN_LOW_CONTRAST_PAIRS: Array<{ fg: string; bg: string; ratio: number }> = [
37
+ { fg: '#999999', bg: '#ffffff', ratio: 2.85 },
38
+ { fg: '#888888', bg: '#ffffff', ratio: 3.54 },
39
+ { fg: '#aaaaaa', bg: '#ffffff', ratio: 2.32 },
40
+ { fg: '#cccccc', bg: '#ffffff', ratio: 1.61 },
41
+ { fg: '#666666', bg: '#000000', ratio: 5.74 },
42
+ { fg: '#777777', bg: '#000000', ratio: 4.48 },
43
+ { fg: 'gray', bg: 'white', ratio: 3.95 },
44
+ { fg: 'silver', bg: 'white', ratio: 1.77 },
45
+ { fg: 'lightgray', bg: 'white', ratio: 1.46 },
46
+ ];
47
+
48
+ // Named colors to RGB mapping (common ones)
49
+ const NAMED_COLORS: Record<string, [number, number, number]> = {
50
+ black: [0, 0, 0],
51
+ white: [255, 255, 255],
52
+ red: [255, 0, 0],
53
+ green: [0, 128, 0],
54
+ blue: [0, 0, 255],
55
+ gray: [128, 128, 128],
56
+ grey: [128, 128, 128],
57
+ silver: [192, 192, 192],
58
+ navy: [0, 0, 128],
59
+ yellow: [255, 255, 0],
60
+ orange: [255, 165, 0],
61
+ purple: [128, 0, 128],
62
+ pink: [255, 192, 203],
63
+ lightgray: [211, 211, 211],
64
+ lightgrey: [211, 211, 211],
65
+ darkgray: [169, 169, 169],
66
+ darkgrey: [169, 169, 169],
67
+ transparent: [255, 255, 255], // Treat as white for calculation
68
+ };
69
+
70
+ export function analyzeColorContrast(
71
+ html: string,
72
+ url: string
73
+ ): { issues: AuditIssue[]; data: ColorContrastData } {
74
+ const issues: AuditIssue[] = [];
75
+ const $ = cheerio.load(html);
76
+
77
+ const potentialIssues: ContrastIssue[] = [];
78
+ let elementsAnalyzed = 0;
79
+ let passedChecks = 0;
80
+ let failedChecks = 0;
81
+ let cannotDetermine = 0;
82
+
83
+ // Elements that typically contain text
84
+ const textElements = $('p, span, a, button, h1, h2, h3, h4, h5, h6, li, td, th, label, div');
85
+
86
+ textElements.each((_, element) => {
87
+ const $el = $(element);
88
+ const style = $el.attr('style') || '';
89
+
90
+ // Skip elements without inline styles (we can't determine their computed styles)
91
+ if (!style) {
92
+ cannotDetermine++;
93
+ return;
94
+ }
95
+
96
+ elementsAnalyzed++;
97
+
98
+ // Extract colors from inline style
99
+ const colorMatch = style.match(/(?:^|;)\s*color\s*:\s*([^;]+)/i);
100
+ const bgMatch = style.match(/(?:^|;)\s*background(?:-color)?\s*:\s*([^;]+)/i);
101
+
102
+ const foreground = colorMatch ? colorMatch[1].trim() : null;
103
+ const background = bgMatch ? bgMatch[1].trim() : null;
104
+
105
+ // Skip if we can't determine both colors
106
+ if (!foreground) {
107
+ cannotDetermine++;
108
+ return;
109
+ }
110
+
111
+ // Default background to white if not specified
112
+ const bgColor = background || 'white';
113
+
114
+ // Parse colors
115
+ const fgRgb = parseColor(foreground);
116
+ const bgRgb = parseColor(bgColor);
117
+
118
+ if (!fgRgb || !bgRgb) {
119
+ cannotDetermine++;
120
+ return;
121
+ }
122
+
123
+ // Calculate contrast ratio
124
+ const ratio = calculateContrastRatio(fgRgb, bgRgb);
125
+
126
+ // Determine if large text
127
+ const fontSize = style.match(/font-size\s*:\s*(\d+)(?:px|pt)?/i);
128
+ const fontWeight = style.match(/font-weight\s*:\s*(\d+|bold)/i);
129
+ const isLargeText =
130
+ (fontSize && parseInt(fontSize[1]) >= 18) ||
131
+ (fontSize && parseInt(fontSize[1]) >= 14 && fontWeight && (fontWeight[1] === 'bold' || parseInt(fontWeight[1]) >= 700));
132
+
133
+ // Check WCAG compliance
134
+ const aaThreshold = isLargeText ? 3.0 : 4.5;
135
+ const aaaThreshold = isLargeText ? 4.5 : 7.0;
136
+
137
+ let wcagLevel: 'AA' | 'AAA' | 'none' = 'none';
138
+ if (ratio >= aaaThreshold) {
139
+ wcagLevel = 'AAA';
140
+ passedChecks++;
141
+ } else if (ratio >= aaThreshold) {
142
+ wcagLevel = 'AA';
143
+ passedChecks++;
144
+ } else {
145
+ failedChecks++;
146
+
147
+ // Create selector for identification
148
+ const tagName = (element as cheerio.Element).name;
149
+ const id = $el.attr('id');
150
+ const className = $el.attr('class')?.split(' ')[0];
151
+ const selector = id ? `#${id}` : (className ? `${tagName}.${className}` : tagName);
152
+
153
+ potentialIssues.push({
154
+ selector,
155
+ foreground,
156
+ background: bgColor,
157
+ contrastRatio: Math.round(ratio * 100) / 100,
158
+ wcagLevel,
159
+ isLargeText: isLargeText || false,
160
+ });
161
+ }
162
+ });
163
+
164
+ // Also check for known problematic color classes
165
+ const styleTag = $('style').text();
166
+ for (const pair of KNOWN_LOW_CONTRAST_PAIRS) {
167
+ if (styleTag.includes(pair.fg) && styleTag.includes(pair.bg)) {
168
+ // Potential issue in stylesheet
169
+ }
170
+ }
171
+
172
+ // Generate issues
173
+ if (potentialIssues.length > 0) {
174
+ const criticalIssues = potentialIssues.filter(i => i.contrastRatio < 3);
175
+ const moderateIssues = potentialIssues.filter(i => i.contrastRatio >= 3 && i.contrastRatio < 4.5);
176
+
177
+ if (criticalIssues.length > 0) {
178
+ issues.push({
179
+ code: 'A11Y_CONTRAST_CRITICAL',
180
+ severity: 'error',
181
+ category: 'accessibility',
182
+ title: 'Critical color contrast issues',
183
+ description: `Found ${criticalIssues.length} element(s) with very low contrast (below 3:1). This text may be unreadable for many users.`,
184
+ impact: 'Users with low vision or color blindness cannot read this content. Fails WCAG accessibility requirements.',
185
+ howToFix: 'Increase contrast by using darker text on light backgrounds or lighter text on dark backgrounds. Use a contrast checker tool to verify 4.5:1 ratio minimum.',
186
+ affectedUrls: [url],
187
+ details: {
188
+ issues: criticalIssues.slice(0, 5),
189
+ recommendation: 'WCAG AA requires 4.5:1 contrast for normal text, 3:1 for large text.',
190
+ },
191
+ });
192
+ }
193
+
194
+ if (moderateIssues.length > 0) {
195
+ issues.push({
196
+ code: 'A11Y_CONTRAST_LOW',
197
+ severity: 'warning',
198
+ category: 'accessibility',
199
+ title: 'Low color contrast detected',
200
+ description: `Found ${moderateIssues.length} element(s) with contrast below WCAG AA standard (4.5:1).`,
201
+ impact: 'May be difficult to read for users with visual impairments.',
202
+ howToFix: 'Adjust colors to achieve at least 4.5:1 contrast ratio for normal text.',
203
+ affectedUrls: [url],
204
+ details: {
205
+ issues: moderateIssues.slice(0, 5),
206
+ },
207
+ });
208
+ }
209
+ }
210
+
211
+ // Note about limitations
212
+ if (cannotDetermine > elementsAnalyzed * 0.8) {
213
+ issues.push({
214
+ code: 'A11Y_CONTRAST_UNDETERMINED',
215
+ severity: 'notice',
216
+ category: 'accessibility',
217
+ title: 'Color contrast could not be fully analyzed',
218
+ description: 'Most text colors are set via CSS stylesheets rather than inline styles. Full contrast checking requires browser rendering.',
219
+ impact: 'Potential contrast issues may exist that were not detected.',
220
+ howToFix: 'Use browser DevTools (Lighthouse accessibility audit) or dedicated tools like axe-core for comprehensive contrast checking.',
221
+ affectedUrls: [url],
222
+ details: {
223
+ elementsAnalyzed,
224
+ cannotDetermine,
225
+ recommendation: 'Run Lighthouse accessibility audit for complete contrast analysis.',
226
+ },
227
+ });
228
+ }
229
+
230
+ return {
231
+ issues,
232
+ data: {
233
+ elementsAnalyzed,
234
+ potentialIssues,
235
+ passedChecks,
236
+ failedChecks,
237
+ cannotDetermine,
238
+ },
239
+ };
240
+ }
241
+
242
+ /**
243
+ * Parse a CSS color value to RGB
244
+ */
245
+ function parseColor(color: string): [number, number, number] | null {
246
+ const trimmed = color.trim().toLowerCase();
247
+
248
+ // Named colors
249
+ if (NAMED_COLORS[trimmed]) {
250
+ return NAMED_COLORS[trimmed];
251
+ }
252
+
253
+ // Hex colors
254
+ const hexMatch = trimmed.match(/^#?([a-f0-9]{3}|[a-f0-9]{6})$/i);
255
+ if (hexMatch) {
256
+ let hex = hexMatch[1];
257
+ if (hex.length === 3) {
258
+ hex = hex[0] + hex[0] + hex[1] + hex[1] + hex[2] + hex[2];
259
+ }
260
+ return [
261
+ parseInt(hex.slice(0, 2), 16),
262
+ parseInt(hex.slice(2, 4), 16),
263
+ parseInt(hex.slice(4, 6), 16),
264
+ ];
265
+ }
266
+
267
+ // RGB/RGBA
268
+ const rgbMatch = trimmed.match(/rgba?\s*\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)/i);
269
+ if (rgbMatch) {
270
+ return [
271
+ parseInt(rgbMatch[1]),
272
+ parseInt(rgbMatch[2]),
273
+ parseInt(rgbMatch[3]),
274
+ ];
275
+ }
276
+
277
+ // HSL (simplified conversion)
278
+ const hslMatch = trimmed.match(/hsla?\s*\(\s*(\d+)\s*,\s*(\d+)%\s*,\s*(\d+)%/i);
279
+ if (hslMatch) {
280
+ const h = parseInt(hslMatch[1]) / 360;
281
+ const s = parseInt(hslMatch[2]) / 100;
282
+ const l = parseInt(hslMatch[3]) / 100;
283
+ return hslToRgb(h, s, l);
284
+ }
285
+
286
+ return null;
287
+ }
288
+
289
+ /**
290
+ * Convert HSL to RGB
291
+ */
292
+ function hslToRgb(h: number, s: number, l: number): [number, number, number] {
293
+ let r: number, g: number, b: number;
294
+
295
+ if (s === 0) {
296
+ r = g = b = l;
297
+ } else {
298
+ const hue2rgb = (p: number, q: number, t: number): number => {
299
+ if (t < 0) t += 1;
300
+ if (t > 1) t -= 1;
301
+ if (t < 1/6) return p + (q - p) * 6 * t;
302
+ if (t < 1/2) return q;
303
+ if (t < 2/3) return p + (q - p) * (2/3 - t) * 6;
304
+ return p;
305
+ };
306
+
307
+ const q = l < 0.5 ? l * (1 + s) : l + s - l * s;
308
+ const p = 2 * l - q;
309
+ r = hue2rgb(p, q, h + 1/3);
310
+ g = hue2rgb(p, q, h);
311
+ b = hue2rgb(p, q, h - 1/3);
312
+ }
313
+
314
+ return [Math.round(r * 255), Math.round(g * 255), Math.round(b * 255)];
315
+ }
316
+
317
+ /**
318
+ * Calculate relative luminance according to WCAG 2.1
319
+ */
320
+ function getRelativeLuminance(rgb: [number, number, number]): number {
321
+ const [r, g, b] = rgb.map(c => {
322
+ const sRGB = c / 255;
323
+ return sRGB <= 0.03928
324
+ ? sRGB / 12.92
325
+ : Math.pow((sRGB + 0.055) / 1.055, 2.4);
326
+ });
327
+
328
+ return 0.2126 * r + 0.7152 * g + 0.0722 * b;
329
+ }
330
+
331
+ /**
332
+ * Calculate contrast ratio according to WCAG 2.1
333
+ */
334
+ function calculateContrastRatio(fg: [number, number, number], bg: [number, number, number]): number {
335
+ const l1 = getRelativeLuminance(fg);
336
+ const l2 = getRelativeLuminance(bg);
337
+
338
+ const lighter = Math.max(l1, l2);
339
+ const darker = Math.min(l1, l2);
340
+
341
+ return (lighter + 0.05) / (darker + 0.05);
342
+ }
@@ -0,0 +1,170 @@
1
+ // Content Freshness Checks
2
+ // Checks for Last-Modified header, article:modified_time, and content staleness
3
+
4
+ import { httpHead } from '../../utils/http.js';
5
+ import * as cheerio from 'cheerio';
6
+ import type { AuditIssue } from '../types.js';
7
+ import { ISSUE_DEFINITIONS } from '../types.js';
8
+ const ONE_YEAR_MS = 365 * 24 * 60 * 60 * 1000;
9
+
10
+ export interface ContentFreshnessData {
11
+ lastModified: Date | null;
12
+ articleModifiedTime: Date | null;
13
+ articlePublishedTime: Date | null;
14
+ ogUpdatedTime: Date | null;
15
+ ageInDays: number | null;
16
+ isStale: boolean;
17
+ source: 'last-modified' | 'og-updated-time' | 'article-modified-time' | 'none';
18
+ }
19
+
20
+ /**
21
+ * Parse date from various formats
22
+ */
23
+ function parseDate(dateStr: string | undefined | null): Date | null {
24
+ if (!dateStr) return null;
25
+
26
+ try {
27
+ const date = new Date(dateStr);
28
+ if (!isNaN(date.getTime())) {
29
+ return date;
30
+ }
31
+ } catch {
32
+ // Invalid date
33
+ }
34
+ return null;
35
+ }
36
+
37
+ /**
38
+ * Calculate age in days from a date
39
+ */
40
+ function calculateAgeInDays(date: Date | null): number | null {
41
+ if (!date) return null;
42
+ const now = new Date();
43
+ const diff = now.getTime() - date.getTime();
44
+ return Math.floor(diff / (24 * 60 * 60 * 1000));
45
+ }
46
+
47
+ /**
48
+ * Extract date signals from HTML
49
+ */
50
+ function extractDatesFromHtml(
51
+ html: string
52
+ ): Pick<ContentFreshnessData, 'articleModifiedTime' | 'articlePublishedTime' | 'ogUpdatedTime'> {
53
+ const $ = cheerio.load(html);
54
+
55
+ const articleModifiedTime = parseDate(
56
+ $('meta[property="article:modified_time"]').attr('content') ||
57
+ $('meta[name="article:modified_time"]').attr('content')
58
+ );
59
+
60
+ const articlePublishedTime = parseDate(
61
+ $('meta[property="article:published_time"]').attr('content') ||
62
+ $('meta[name="article:published_time"]').attr('content')
63
+ );
64
+
65
+ const ogUpdatedTime = parseDate(
66
+ $('meta[property="og:updated_time"]').attr('content') || $('meta[name="og:updated_time"]').attr('content')
67
+ );
68
+
69
+ return {
70
+ articleModifiedTime,
71
+ articlePublishedTime,
72
+ ogUpdatedTime,
73
+ };
74
+ }
75
+
76
+ /**
77
+ * Analyze content freshness
78
+ */
79
+ export async function analyzeContentFreshness(
80
+ url: string,
81
+ html: string
82
+ ): Promise<{ issues: AuditIssue[]; data: ContentFreshnessData }> {
83
+ const issues: AuditIssue[] = [];
84
+
85
+ // Get Last-Modified from headers
86
+ let lastModified: Date | null = null;
87
+
88
+ try {
89
+ const response = await httpHead(url, {
90
+
91
+ timeout: 10000,
92
+ validateStatus: () => true,
93
+ });
94
+
95
+ const lastModifiedHeader = response.headers['last-modified'];
96
+ lastModified = parseDate(lastModifiedHeader);
97
+ } catch {
98
+ // Couldn't fetch headers
99
+ }
100
+
101
+ // Extract dates from HTML
102
+ const htmlDates = extractDatesFromHtml(html);
103
+
104
+ // Determine the most recent date and its source
105
+ const dates: { date: Date; source: ContentFreshnessData['source'] }[] = [];
106
+
107
+ if (lastModified) {
108
+ dates.push({ date: lastModified, source: 'last-modified' });
109
+ }
110
+ if (htmlDates.articleModifiedTime) {
111
+ dates.push({ date: htmlDates.articleModifiedTime, source: 'article-modified-time' });
112
+ }
113
+ if (htmlDates.ogUpdatedTime) {
114
+ dates.push({ date: htmlDates.ogUpdatedTime, source: 'og-updated-time' });
115
+ }
116
+
117
+ // Sort by date descending
118
+ dates.sort((a, b) => b.date.getTime() - a.date.getTime());
119
+
120
+ const mostRecent = dates[0] || null;
121
+ const ageInDays = mostRecent ? calculateAgeInDays(mostRecent.date) : null;
122
+ const isStale = ageInDays !== null && ageInDays > 365;
123
+
124
+ // Generate issues
125
+ if (!lastModified) {
126
+ issues.push({
127
+ ...ISSUE_DEFINITIONS.NO_LAST_MODIFIED,
128
+ affectedUrls: [url],
129
+ });
130
+ }
131
+
132
+ if (isStale) {
133
+ issues.push({
134
+ ...ISSUE_DEFINITIONS.CONTENT_STALE,
135
+ affectedUrls: [url],
136
+ details: {
137
+ ageInDays,
138
+ lastModified: mostRecent?.date.toISOString(),
139
+ source: mostRecent?.source,
140
+ },
141
+ });
142
+ }
143
+
144
+ // Check for article:modified_time on article pages
145
+ const $ = cheerio.load(html);
146
+ const isArticle =
147
+ $('article').length > 0 ||
148
+ $('meta[property="og:type"][content="article"]').length > 0 ||
149
+ $('meta[property="article:published_time"]').length > 0;
150
+
151
+ if (isArticle && !htmlDates.articleModifiedTime && !htmlDates.ogUpdatedTime) {
152
+ issues.push({
153
+ ...ISSUE_DEFINITIONS.OG_UPDATED_TIME_MISSING,
154
+ affectedUrls: [url],
155
+ });
156
+ }
157
+
158
+ return {
159
+ issues,
160
+ data: {
161
+ lastModified,
162
+ articleModifiedTime: htmlDates.articleModifiedTime,
163
+ articlePublishedTime: htmlDates.articlePublishedTime,
164
+ ogUpdatedTime: htmlDates.ogUpdatedTime,
165
+ ageInDays,
166
+ isStale,
167
+ source: mostRecent?.source || 'none',
168
+ },
169
+ };
170
+ }