@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.
- package/README.md +242 -0
- package/dist/analyzer-2CSWIQGD.mjs +6 -0
- package/dist/chunk-YNZYHEYM.mjs +774 -0
- package/dist/index.d.mts +4012 -0
- package/dist/index.d.ts +4012 -0
- package/dist/index.js +29672 -0
- package/dist/index.mjs +28602 -0
- package/package.json +53 -0
- package/scripts/build-deno.ts +134 -0
- package/src/audit/ai/analyzer.ts +347 -0
- package/src/audit/ai/index.ts +29 -0
- package/src/audit/ai/prompts/content-analysis.ts +271 -0
- package/src/audit/ai/types.ts +179 -0
- package/src/audit/checks/additional-checks.ts +439 -0
- package/src/audit/checks/ai-citation-worthiness.ts +399 -0
- package/src/audit/checks/ai-content-structure.ts +325 -0
- package/src/audit/checks/ai-readiness.ts +339 -0
- package/src/audit/checks/anchor-text.ts +179 -0
- package/src/audit/checks/answer-conciseness.ts +322 -0
- package/src/audit/checks/asset-minification.ts +270 -0
- package/src/audit/checks/bing-optimization.ts +206 -0
- package/src/audit/checks/brand-mention-optimization.ts +349 -0
- package/src/audit/checks/caching-headers.ts +305 -0
- package/src/audit/checks/canonical-advanced.ts +150 -0
- package/src/audit/checks/canonical-domain.ts +196 -0
- package/src/audit/checks/citation-quality.ts +358 -0
- package/src/audit/checks/client-rendering.ts +542 -0
- package/src/audit/checks/color-contrast.ts +342 -0
- package/src/audit/checks/content-freshness.ts +170 -0
- package/src/audit/checks/content-science.ts +589 -0
- package/src/audit/checks/conversion-elements.ts +526 -0
- package/src/audit/checks/crawlability.ts +220 -0
- package/src/audit/checks/directory-listing.ts +172 -0
- package/src/audit/checks/dom-analysis.ts +191 -0
- package/src/audit/checks/dom-size.ts +246 -0
- package/src/audit/checks/duplicate-content.ts +194 -0
- package/src/audit/checks/eeat-signals.ts +990 -0
- package/src/audit/checks/entity-seo.ts +396 -0
- package/src/audit/checks/featured-snippet.ts +473 -0
- package/src/audit/checks/freshness-signals.ts +443 -0
- package/src/audit/checks/funnel-intent.ts +463 -0
- package/src/audit/checks/hreflang.ts +174 -0
- package/src/audit/checks/html-compliance.ts +302 -0
- package/src/audit/checks/image-dimensions.ts +167 -0
- package/src/audit/checks/images.ts +160 -0
- package/src/audit/checks/indexnow.ts +275 -0
- package/src/audit/checks/interactive-tools.ts +475 -0
- package/src/audit/checks/internal-link-graph.ts +436 -0
- package/src/audit/checks/keyword-analysis.ts +239 -0
- package/src/audit/checks/keyword-cannibalization.ts +385 -0
- package/src/audit/checks/keyword-placement.ts +471 -0
- package/src/audit/checks/links.ts +203 -0
- package/src/audit/checks/llms-txt.ts +224 -0
- package/src/audit/checks/local-seo.ts +296 -0
- package/src/audit/checks/mobile.ts +167 -0
- package/src/audit/checks/modern-images.ts +226 -0
- package/src/audit/checks/navboost-signals.ts +395 -0
- package/src/audit/checks/on-page.ts +209 -0
- package/src/audit/checks/page-resources.ts +285 -0
- package/src/audit/checks/pagination.ts +180 -0
- package/src/audit/checks/performance.ts +153 -0
- package/src/audit/checks/platform-presence.ts +580 -0
- package/src/audit/checks/redirect-analysis.ts +153 -0
- package/src/audit/checks/redirect-chain.ts +389 -0
- package/src/audit/checks/resource-hints.ts +420 -0
- package/src/audit/checks/responsive-css.ts +247 -0
- package/src/audit/checks/responsive-images.ts +396 -0
- package/src/audit/checks/review-ecosystem.ts +415 -0
- package/src/audit/checks/robots-validation.ts +373 -0
- package/src/audit/checks/security-headers.ts +172 -0
- package/src/audit/checks/security.ts +144 -0
- package/src/audit/checks/serp-preview.ts +251 -0
- package/src/audit/checks/site-maturity.ts +444 -0
- package/src/audit/checks/social-meta.test.ts +275 -0
- package/src/audit/checks/social-meta.ts +134 -0
- package/src/audit/checks/soft-404.ts +151 -0
- package/src/audit/checks/structured-data.ts +238 -0
- package/src/audit/checks/tech-detection.ts +496 -0
- package/src/audit/checks/topical-clusters.ts +435 -0
- package/src/audit/checks/tracker-bloat.ts +462 -0
- package/src/audit/checks/tracking-verification.test.ts +371 -0
- package/src/audit/checks/tracking-verification.ts +636 -0
- package/src/audit/checks/url-safety.ts +682 -0
- package/src/audit/deno-entry.ts +66 -0
- package/src/audit/discovery/index.ts +15 -0
- package/src/audit/discovery/link-crawler.ts +232 -0
- package/src/audit/discovery/repo-routes.ts +347 -0
- package/src/audit/engine.ts +620 -0
- package/src/audit/fixes/index.ts +209 -0
- package/src/audit/fixes/social-meta-fixes.test.ts +329 -0
- package/src/audit/fixes/social-meta-fixes.ts +463 -0
- package/src/audit/index.ts +74 -0
- package/src/audit/runner.test.ts +299 -0
- package/src/audit/runner.ts +130 -0
- package/src/audit/types.ts +1953 -0
- package/src/content/featured-snippet.ts +367 -0
- package/src/content/generator.test.ts +534 -0
- package/src/content/generator.ts +501 -0
- package/src/content/headline.ts +317 -0
- package/src/content/index.ts +62 -0
- package/src/content/intent.ts +258 -0
- package/src/content/keyword-density.ts +349 -0
- package/src/content/readability.ts +262 -0
- package/src/executor.ts +336 -0
- package/src/fixer.ts +416 -0
- package/src/frameworks/detector.test.ts +248 -0
- package/src/frameworks/detector.ts +371 -0
- package/src/frameworks/index.ts +68 -0
- package/src/frameworks/recipes/angular.yaml +171 -0
- package/src/frameworks/recipes/astro.yaml +206 -0
- package/src/frameworks/recipes/django.yaml +180 -0
- package/src/frameworks/recipes/laravel.yaml +137 -0
- package/src/frameworks/recipes/nextjs.yaml +268 -0
- package/src/frameworks/recipes/nuxt.yaml +175 -0
- package/src/frameworks/recipes/rails.yaml +188 -0
- package/src/frameworks/recipes/react.yaml +202 -0
- package/src/frameworks/recipes/sveltekit.yaml +154 -0
- package/src/frameworks/recipes/vue.yaml +137 -0
- package/src/frameworks/recipes/wordpress.yaml +209 -0
- package/src/frameworks/suggestion-engine.ts +320 -0
- package/src/geo/geo-content.test.ts +305 -0
- package/src/geo/geo-content.ts +266 -0
- package/src/geo/geo-history.test.ts +473 -0
- package/src/geo/geo-history.ts +433 -0
- package/src/geo/geo-tracker.test.ts +359 -0
- package/src/geo/geo-tracker.ts +411 -0
- package/src/geo/index.ts +10 -0
- package/src/git/commit-helper.test.ts +261 -0
- package/src/git/commit-helper.ts +329 -0
- package/src/git/index.ts +12 -0
- package/src/git/pr-helper.test.ts +284 -0
- package/src/git/pr-helper.ts +307 -0
- package/src/index.ts +66 -0
- package/src/keywords/ai-keyword-engine.ts +1062 -0
- package/src/keywords/ai-summarizer.ts +387 -0
- package/src/keywords/ci-mode.ts +555 -0
- package/src/keywords/engine.ts +359 -0
- package/src/keywords/index.ts +151 -0
- package/src/keywords/llm-judge.ts +357 -0
- package/src/keywords/nlp-analysis.ts +706 -0
- package/src/keywords/prioritizer.ts +295 -0
- package/src/keywords/site-crawler.ts +342 -0
- package/src/keywords/sources/autocomplete.ts +139 -0
- package/src/keywords/sources/competitive-search.ts +450 -0
- package/src/keywords/sources/competitor-analysis.ts +374 -0
- package/src/keywords/sources/dataforseo.ts +206 -0
- package/src/keywords/sources/free-sources.ts +294 -0
- package/src/keywords/sources/gsc.ts +123 -0
- package/src/keywords/topic-grouping.ts +327 -0
- package/src/keywords/types.ts +144 -0
- package/src/keywords/wizard.ts +457 -0
- package/src/loader.ts +40 -0
- package/src/reports/index.ts +7 -0
- package/src/reports/report-generator.test.ts +293 -0
- package/src/reports/report-generator.ts +713 -0
- package/src/scheduler/alerts.test.ts +458 -0
- package/src/scheduler/alerts.ts +328 -0
- package/src/scheduler/index.ts +8 -0
- package/src/scheduler/scheduled-audit.test.ts +377 -0
- package/src/scheduler/scheduled-audit.ts +149 -0
- package/src/test/integration-test.ts +325 -0
- package/src/tools/analyzer.ts +373 -0
- package/src/tools/crawl.ts +293 -0
- package/src/tools/files.ts +301 -0
- package/src/tools/h1-fixer.ts +249 -0
- package/src/tools/index.ts +67 -0
- package/src/tracking/github-action.ts +326 -0
- package/src/tracking/google-analytics.ts +265 -0
- package/src/tracking/index.ts +45 -0
- package/src/tracking/report-generator.ts +386 -0
- package/src/tracking/search-console.ts +335 -0
- package/src/types.ts +134 -0
- package/src/utils/http.ts +302 -0
- package/src/wasm-adapter.ts +297 -0
- package/src/wasm-entry.ts +14 -0
- package/tsconfig.json +17 -0
- package/tsup.wasm.config.ts +26 -0
- 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
|
+
}
|