@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,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
+ }