@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,396 @@
|
|
|
1
|
+
// Responsive Images Check - Essential for Core Web Vitals
|
|
2
|
+
// Reference: "Technical SEO for Developers - 17 Tips to Rank Higher"
|
|
3
|
+
// "Make sure your images are responsive for their screen size"
|
|
4
|
+
// "You should always, always, always lazy load off screen images"
|
|
5
|
+
// "Make sure you're setting height and width on your images" (for CLS)
|
|
6
|
+
|
|
7
|
+
import * as cheerio from 'cheerio';
|
|
8
|
+
import type { AuditIssue } from '../types.js';
|
|
9
|
+
|
|
10
|
+
export interface ImageAnalysis {
|
|
11
|
+
src: string;
|
|
12
|
+
alt: string | null;
|
|
13
|
+
width: string | null;
|
|
14
|
+
height: string | null;
|
|
15
|
+
loading: 'lazy' | 'eager' | null;
|
|
16
|
+
decoding: 'async' | 'sync' | 'auto' | null;
|
|
17
|
+
srcset: string | null;
|
|
18
|
+
sizes: string | null;
|
|
19
|
+
isAboveFold: boolean; // Estimated based on position
|
|
20
|
+
hasExplicitDimensions: boolean;
|
|
21
|
+
hasResponsiveSrcset: boolean;
|
|
22
|
+
usesModernFormat: boolean;
|
|
23
|
+
issues: string[];
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export interface ResponsiveImageData {
|
|
27
|
+
totalImages: number;
|
|
28
|
+
imagesWithoutDimensions: number;
|
|
29
|
+
imagesWithoutLazyLoad: number;
|
|
30
|
+
imagesWithoutSrcset: number;
|
|
31
|
+
imagesWithoutAlt: number;
|
|
32
|
+
modernFormatCount: number;
|
|
33
|
+
legacyFormatCount: number;
|
|
34
|
+
images: ImageAnalysis[];
|
|
35
|
+
recommendations: string[];
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
// Modern image formats
|
|
39
|
+
const MODERN_FORMATS = ['.webp', '.avif'];
|
|
40
|
+
const LEGACY_FORMATS = ['.jpg', '.jpeg', '.png', '.gif', '.bmp'];
|
|
41
|
+
|
|
42
|
+
/**
|
|
43
|
+
* Analyze individual image element
|
|
44
|
+
*/
|
|
45
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
46
|
+
export function analyzeImage($: cheerio.CheerioAPI, img: any, index: number): ImageAnalysis {
|
|
47
|
+
const $img = $(img);
|
|
48
|
+
const issues: string[] = [];
|
|
49
|
+
|
|
50
|
+
const src = $img.attr('src') || $img.attr('data-src') || '';
|
|
51
|
+
const alt = $img.attr('alt');
|
|
52
|
+
const width = $img.attr('width');
|
|
53
|
+
const height = $img.attr('height');
|
|
54
|
+
const loading = $img.attr('loading') as 'lazy' | 'eager' | null;
|
|
55
|
+
const decoding = $img.attr('decoding') as 'async' | 'sync' | 'auto' | null;
|
|
56
|
+
const srcset = $img.attr('srcset');
|
|
57
|
+
const sizes = $img.attr('sizes');
|
|
58
|
+
|
|
59
|
+
// Check for explicit dimensions
|
|
60
|
+
const hasExplicitDimensions = !!(
|
|
61
|
+
(width && height) ||
|
|
62
|
+
$img.css('width') ||
|
|
63
|
+
$img.css('height') ||
|
|
64
|
+
$img.attr('style')?.includes('width') ||
|
|
65
|
+
$img.attr('style')?.includes('height')
|
|
66
|
+
);
|
|
67
|
+
|
|
68
|
+
// Check for responsive srcset
|
|
69
|
+
const hasResponsiveSrcset = !!(srcset && srcset.includes('w'));
|
|
70
|
+
|
|
71
|
+
// Check for modern format
|
|
72
|
+
const srcLower = src.toLowerCase();
|
|
73
|
+
const usesModernFormat = MODERN_FORMATS.some(fmt => srcLower.includes(fmt));
|
|
74
|
+
|
|
75
|
+
// Estimate if above fold (very rough - based on element order)
|
|
76
|
+
// Images in first 3 <img> tags or in header/hero are likely above fold
|
|
77
|
+
const isInHeader = $img.closest('header, [class*="hero"], [class*="banner"], [class*="jumbotron"]').length > 0;
|
|
78
|
+
const isAboveFold = index < 3 || isInHeader;
|
|
79
|
+
|
|
80
|
+
// Generate issues
|
|
81
|
+
if (!hasExplicitDimensions) {
|
|
82
|
+
issues.push('Missing width/height attributes (causes CLS)');
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
if (alt === undefined || alt === null) {
|
|
86
|
+
issues.push('Missing alt attribute');
|
|
87
|
+
} else if (alt === '') {
|
|
88
|
+
// Empty alt is okay for decorative images but should be noted
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
if (!isAboveFold && loading !== 'lazy') {
|
|
92
|
+
issues.push('Below-fold image without lazy loading');
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
if (isAboveFold && loading === 'lazy') {
|
|
96
|
+
issues.push('Above-fold image with lazy loading (delays LCP)');
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
if (!hasResponsiveSrcset && src && !srcLower.includes('.svg')) {
|
|
100
|
+
issues.push('No responsive srcset for different screen sizes');
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
if (srcset && !sizes) {
|
|
104
|
+
issues.push('Has srcset but missing sizes attribute');
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
if (!usesModernFormat && src && !srcLower.includes('.svg')) {
|
|
108
|
+
issues.push('Using legacy image format instead of WebP/AVIF');
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
return {
|
|
112
|
+
src,
|
|
113
|
+
alt: alt ?? null,
|
|
114
|
+
width: width ?? null,
|
|
115
|
+
height: height ?? null,
|
|
116
|
+
loading: loading ?? null,
|
|
117
|
+
decoding: decoding ?? null,
|
|
118
|
+
srcset: srcset ?? null,
|
|
119
|
+
sizes: sizes ?? null,
|
|
120
|
+
isAboveFold,
|
|
121
|
+
hasExplicitDimensions,
|
|
122
|
+
hasResponsiveSrcset,
|
|
123
|
+
usesModernFormat,
|
|
124
|
+
issues,
|
|
125
|
+
};
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
/**
|
|
129
|
+
* Analyze <picture> elements for responsive patterns
|
|
130
|
+
*/
|
|
131
|
+
export function analyzePictureElements($: cheerio.CheerioAPI): {
|
|
132
|
+
hasPictureElements: boolean;
|
|
133
|
+
pictureCount: number;
|
|
134
|
+
hasFormatFallbacks: boolean;
|
|
135
|
+
} {
|
|
136
|
+
const pictures = $('picture');
|
|
137
|
+
const pictureCount = pictures.length;
|
|
138
|
+
let hasFormatFallbacks = false;
|
|
139
|
+
|
|
140
|
+
pictures.each((_, pic) => {
|
|
141
|
+
const sources = $(pic).find('source');
|
|
142
|
+
const hasWebP = sources.filter((_, s) => {
|
|
143
|
+
const type = $(s).attr('type') || '';
|
|
144
|
+
const srcset = $(s).attr('srcset') || '';
|
|
145
|
+
return type.includes('webp') || srcset.includes('.webp');
|
|
146
|
+
}).length > 0;
|
|
147
|
+
const hasAvif = sources.filter((_, s) => {
|
|
148
|
+
const type = $(s).attr('type') || '';
|
|
149
|
+
const srcset = $(s).attr('srcset') || '';
|
|
150
|
+
return type.includes('avif') || srcset.includes('.avif');
|
|
151
|
+
}).length > 0;
|
|
152
|
+
|
|
153
|
+
if (hasWebP || hasAvif) {
|
|
154
|
+
hasFormatFallbacks = true;
|
|
155
|
+
}
|
|
156
|
+
});
|
|
157
|
+
|
|
158
|
+
return {
|
|
159
|
+
hasPictureElements: pictureCount > 0,
|
|
160
|
+
pictureCount,
|
|
161
|
+
hasFormatFallbacks,
|
|
162
|
+
};
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
/**
|
|
166
|
+
* Check for image CDN usage
|
|
167
|
+
*/
|
|
168
|
+
export function detectImageCDN(html: string): { detected: boolean; cdns: string[] } {
|
|
169
|
+
const cdnPatterns = [
|
|
170
|
+
{ name: 'Cloudinary', pattern: /cloudinary\.com|res\.cloudinary/i },
|
|
171
|
+
{ name: 'imgix', pattern: /imgix\.net/i },
|
|
172
|
+
{ name: 'Cloudflare Images', pattern: /imagedelivery\.net|cloudflare-images/i },
|
|
173
|
+
{ name: 'AWS CloudFront', pattern: /cloudfront\.net/i },
|
|
174
|
+
{ name: 'Fastly', pattern: /fastly\.net/i },
|
|
175
|
+
{ name: 'ImageKit', pattern: /imagekit\.io|ik\.imagekit/i },
|
|
176
|
+
{ name: 'Vercel Image Optimization', pattern: /_next\/image/i },
|
|
177
|
+
{ name: 'Prismic', pattern: /images\.prismic\.io/i },
|
|
178
|
+
{ name: 'Contentful', pattern: /images\.ctfassets\.net/i },
|
|
179
|
+
{ name: 'Sanity', pattern: /cdn\.sanity\.io/i },
|
|
180
|
+
{ name: 'Shopify CDN', pattern: /cdn\.shopify\.com/i },
|
|
181
|
+
{ name: 'WordPress Jetpack', pattern: /i\d\.wp\.com/i },
|
|
182
|
+
];
|
|
183
|
+
|
|
184
|
+
const detectedCDNs: string[] = [];
|
|
185
|
+
|
|
186
|
+
for (const { name, pattern } of cdnPatterns) {
|
|
187
|
+
if (pattern.test(html)) {
|
|
188
|
+
detectedCDNs.push(name);
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
return {
|
|
193
|
+
detected: detectedCDNs.length > 0,
|
|
194
|
+
cdns: detectedCDNs,
|
|
195
|
+
};
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
/**
|
|
199
|
+
* Main function: Analyze responsive image best practices
|
|
200
|
+
*/
|
|
201
|
+
export function analyzeResponsiveImages(
|
|
202
|
+
html: string,
|
|
203
|
+
url: string
|
|
204
|
+
): { issues: AuditIssue[]; data: ResponsiveImageData } {
|
|
205
|
+
const $ = cheerio.load(html);
|
|
206
|
+
const issues: AuditIssue[] = [];
|
|
207
|
+
const images: ImageAnalysis[] = [];
|
|
208
|
+
|
|
209
|
+
// Analyze all images
|
|
210
|
+
const imgElements = $('img');
|
|
211
|
+
imgElements.each((index, img) => {
|
|
212
|
+
const analysis = analyzeImage($, img, index);
|
|
213
|
+
images.push(analysis);
|
|
214
|
+
});
|
|
215
|
+
|
|
216
|
+
// Analyze picture elements
|
|
217
|
+
const pictureData = analyzePictureElements($);
|
|
218
|
+
|
|
219
|
+
// Detect image CDN usage
|
|
220
|
+
const cdnData = detectImageCDN(html);
|
|
221
|
+
|
|
222
|
+
// Calculate statistics
|
|
223
|
+
const totalImages = images.length;
|
|
224
|
+
const imagesWithoutDimensions = images.filter(i => !i.hasExplicitDimensions).length;
|
|
225
|
+
const imagesWithoutLazyLoad = images.filter(i => !i.isAboveFold && i.loading !== 'lazy').length;
|
|
226
|
+
const imagesWithoutSrcset = images.filter(i => !i.hasResponsiveSrcset && i.src && !i.src.includes('.svg')).length;
|
|
227
|
+
const imagesWithoutAlt = images.filter(i => i.alt === null).length;
|
|
228
|
+
const modernFormatCount = images.filter(i => i.usesModernFormat).length;
|
|
229
|
+
const legacyFormatCount = images.filter(i =>
|
|
230
|
+
!i.usesModernFormat && i.src && !i.src.includes('.svg')
|
|
231
|
+
).length;
|
|
232
|
+
|
|
233
|
+
// Generate recommendations
|
|
234
|
+
const recommendations: string[] = [];
|
|
235
|
+
|
|
236
|
+
if (!cdnData.detected && totalImages > 3) {
|
|
237
|
+
recommendations.push('Consider using an image CDN (Cloudinary, imgix, Vercel) for automatic optimization');
|
|
238
|
+
}
|
|
239
|
+
|
|
240
|
+
if (!pictureData.hasPictureElements && legacyFormatCount > 0) {
|
|
241
|
+
recommendations.push('Use <picture> elements with WebP/AVIF sources and JPEG/PNG fallbacks');
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
if (imagesWithoutDimensions > 0) {
|
|
245
|
+
recommendations.push('Add width and height attributes to all images to prevent layout shift');
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
if (imagesWithoutLazyLoad > 0) {
|
|
249
|
+
recommendations.push('Add loading="lazy" to below-fold images');
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
if (imagesWithoutSrcset > 0) {
|
|
253
|
+
recommendations.push('Add srcset with multiple sizes for responsive images');
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
// Generate issues
|
|
257
|
+
|
|
258
|
+
// Critical: Many images without dimensions (CLS impact)
|
|
259
|
+
if (imagesWithoutDimensions >= 3) {
|
|
260
|
+
issues.push({
|
|
261
|
+
code: 'IMAGES_MISSING_DIMENSIONS',
|
|
262
|
+
severity: 'error',
|
|
263
|
+
category: 'performance',
|
|
264
|
+
title: `${imagesWithoutDimensions} images missing width/height attributes`,
|
|
265
|
+
description: `Found ${imagesWithoutDimensions} images without explicit dimensions.`,
|
|
266
|
+
impact: 'Missing dimensions cause Cumulative Layout Shift (CLS) when images load.',
|
|
267
|
+
howToFix: 'Add width and height attributes to all <img> tags, even with CSS sizing.',
|
|
268
|
+
affectedUrls: [url],
|
|
269
|
+
details: {
|
|
270
|
+
count: imagesWithoutDimensions,
|
|
271
|
+
images: images.filter(i => !i.hasExplicitDimensions).slice(0, 5).map(i => i.src),
|
|
272
|
+
},
|
|
273
|
+
});
|
|
274
|
+
} else if (imagesWithoutDimensions > 0) {
|
|
275
|
+
issues.push({
|
|
276
|
+
code: 'IMAGES_MISSING_DIMENSIONS',
|
|
277
|
+
severity: 'warning',
|
|
278
|
+
category: 'performance',
|
|
279
|
+
title: `${imagesWithoutDimensions} image(s) missing width/height attributes`,
|
|
280
|
+
description: `Found ${imagesWithoutDimensions} images without explicit dimensions.`,
|
|
281
|
+
impact: 'Missing dimensions cause Cumulative Layout Shift (CLS) when images load.',
|
|
282
|
+
howToFix: 'Add width and height attributes to all <img> tags.',
|
|
283
|
+
affectedUrls: [url],
|
|
284
|
+
details: {
|
|
285
|
+
images: images.filter(i => !i.hasExplicitDimensions).map(i => i.src),
|
|
286
|
+
},
|
|
287
|
+
});
|
|
288
|
+
}
|
|
289
|
+
|
|
290
|
+
// Warning: Images without lazy loading
|
|
291
|
+
if (imagesWithoutLazyLoad >= 3) {
|
|
292
|
+
issues.push({
|
|
293
|
+
code: 'IMAGES_WITHOUT_LAZY_LOAD',
|
|
294
|
+
severity: 'warning',
|
|
295
|
+
category: 'performance',
|
|
296
|
+
title: `${imagesWithoutLazyLoad} below-fold images without lazy loading`,
|
|
297
|
+
description: `Found ${imagesWithoutLazyLoad} images below the fold that load immediately.`,
|
|
298
|
+
impact: 'Eager-loading all images delays initial page load and wastes bandwidth.',
|
|
299
|
+
howToFix: 'Add loading="lazy" to images below the fold. Keep loading="eager" for hero images.',
|
|
300
|
+
affectedUrls: [url],
|
|
301
|
+
details: {
|
|
302
|
+
count: imagesWithoutLazyLoad,
|
|
303
|
+
images: images.filter(i => !i.isAboveFold && i.loading !== 'lazy').slice(0, 5).map(i => i.src),
|
|
304
|
+
},
|
|
305
|
+
});
|
|
306
|
+
}
|
|
307
|
+
|
|
308
|
+
// Warning: No responsive images
|
|
309
|
+
if (imagesWithoutSrcset >= 5 && totalImages >= 5) {
|
|
310
|
+
issues.push({
|
|
311
|
+
code: 'NO_RESPONSIVE_IMAGES',
|
|
312
|
+
severity: 'warning',
|
|
313
|
+
category: 'performance',
|
|
314
|
+
title: 'Images lack responsive srcset',
|
|
315
|
+
description: `${imagesWithoutSrcset} of ${totalImages} images don't have srcset for different screen sizes.`,
|
|
316
|
+
impact: 'Mobile users download full-size images, wasting bandwidth and slowing load.',
|
|
317
|
+
howToFix: 'Add srcset attribute with multiple image sizes (e.g., 320w, 640w, 1280w).',
|
|
318
|
+
affectedUrls: [url],
|
|
319
|
+
details: {
|
|
320
|
+
withoutSrcset: imagesWithoutSrcset,
|
|
321
|
+
total: totalImages,
|
|
322
|
+
},
|
|
323
|
+
});
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
// Warning: Missing alt text
|
|
327
|
+
if (imagesWithoutAlt > 0) {
|
|
328
|
+
issues.push({
|
|
329
|
+
code: 'IMAGES_MISSING_ALT',
|
|
330
|
+
severity: 'warning',
|
|
331
|
+
category: 'on-page', // Alt text impacts both accessibility and SEO
|
|
332
|
+
title: `${imagesWithoutAlt} image(s) missing alt attribute`,
|
|
333
|
+
description: `Found ${imagesWithoutAlt} images without alt text.`,
|
|
334
|
+
impact: 'Missing alt text hurts accessibility and SEO. Screen readers cannot describe the image.',
|
|
335
|
+
howToFix: 'Add descriptive alt text to all images. Use alt="" only for purely decorative images.',
|
|
336
|
+
affectedUrls: [url],
|
|
337
|
+
details: {
|
|
338
|
+
count: imagesWithoutAlt,
|
|
339
|
+
images: images.filter(i => i.alt === null).slice(0, 5).map(i => i.src),
|
|
340
|
+
},
|
|
341
|
+
});
|
|
342
|
+
}
|
|
343
|
+
|
|
344
|
+
// Notice: Not using modern formats
|
|
345
|
+
if (legacyFormatCount > 3 && modernFormatCount === 0 && !pictureData.hasFormatFallbacks) {
|
|
346
|
+
issues.push({
|
|
347
|
+
code: 'NO_MODERN_IMAGE_FORMATS',
|
|
348
|
+
severity: 'notice',
|
|
349
|
+
category: 'performance',
|
|
350
|
+
title: 'Not using modern image formats (WebP/AVIF)',
|
|
351
|
+
description: `All ${legacyFormatCount} images use legacy formats (JPEG/PNG) instead of WebP/AVIF.`,
|
|
352
|
+
impact: 'WebP is 25-35% smaller than JPEG. AVIF is up to 50% smaller.',
|
|
353
|
+
howToFix: 'Use <picture> element to serve WebP/AVIF with JPEG fallback, or use an image CDN.',
|
|
354
|
+
affectedUrls: [url],
|
|
355
|
+
details: {
|
|
356
|
+
legacyCount: legacyFormatCount,
|
|
357
|
+
suggestion: cdnData.detected
|
|
358
|
+
? `Your CDN (${cdnData.cdns.join(', ')}) likely supports format conversion.`
|
|
359
|
+
: 'Consider using Cloudinary, imgix, or Vercel Image Optimization.',
|
|
360
|
+
},
|
|
361
|
+
});
|
|
362
|
+
}
|
|
363
|
+
|
|
364
|
+
// Notice: Above-fold images with lazy loading
|
|
365
|
+
const aboveFoldLazy = images.filter(i => i.isAboveFold && i.loading === 'lazy');
|
|
366
|
+
if (aboveFoldLazy.length > 0) {
|
|
367
|
+
issues.push({
|
|
368
|
+
code: 'HERO_IMAGE_LAZY',
|
|
369
|
+
severity: 'notice',
|
|
370
|
+
category: 'performance',
|
|
371
|
+
title: 'Above-fold images have lazy loading',
|
|
372
|
+
description: `${aboveFoldLazy.length} hero/above-fold image(s) use loading="lazy" which delays LCP.`,
|
|
373
|
+
impact: 'Lazy loading above-fold images increases Largest Contentful Paint time.',
|
|
374
|
+
howToFix: 'Remove loading="lazy" from images visible on initial page load.',
|
|
375
|
+
affectedUrls: [url],
|
|
376
|
+
details: {
|
|
377
|
+
images: aboveFoldLazy.map(i => i.src),
|
|
378
|
+
},
|
|
379
|
+
});
|
|
380
|
+
}
|
|
381
|
+
|
|
382
|
+
return {
|
|
383
|
+
issues,
|
|
384
|
+
data: {
|
|
385
|
+
totalImages,
|
|
386
|
+
imagesWithoutDimensions,
|
|
387
|
+
imagesWithoutLazyLoad,
|
|
388
|
+
imagesWithoutSrcset,
|
|
389
|
+
imagesWithoutAlt,
|
|
390
|
+
modernFormatCount,
|
|
391
|
+
legacyFormatCount,
|
|
392
|
+
images,
|
|
393
|
+
recommendations,
|
|
394
|
+
},
|
|
395
|
+
};
|
|
396
|
+
}
|