seo-core-web-vitals
seo-core-web-vitals (M15)
Core Web Vitals are a confirmed Google ranking signal and a proxy for page quality. Thresholds, field-vs-lab honesty rules, and LCP decomposition: references/cwv-thresholds.md (follow it exactly).
Audits
Working from the PageSnapshot (rendered_dom if present, else raw_html):
- Field CWV at p75 — LCP ≤ 2.5 s, INP ≤ 200 ms (INP replaced FID), CLS ≤ 0.1, all at the 75th percentile of real-user data. All three must pass for a "good" rating. Use the proactive-warning thresholds (LCP > 2.0 s, INP > 160 ms, CLS > 0.08) to
warnbefore failing. - Render-blocking resources — synchronous
<script>in<head>(nodefer/async/type=module), blocking<link rel="stylesheet">, and@importchains that delay first render (LCP/INP risk). - Unoptimized images — missing
width/height(oraspect-ratio), noloading="lazy"below the fold, no responsivesrcset/sizes, legacy formats where AVIF/WebP would serve, and a non-preloaded LCP image. - Layout-shift sources — images/iframes/ads/embeds without reserved dimensions, web-font swap without
font-display/size-adjust, and content injected above existing content (CLS risk). - LCP decomposition — attribute LCP to TTFB vs resource load delay vs load duration vs render delay (see
references/cwv-thresholds.md) so each finding targets the dominant subpart.
Fixes (fixable: advisory)
Advisory only. Performance fixes touch build config, server, CDN, and runtime JavaScript — high breakage risk — so the tool diagnoses and prioritizes but never auto-edits performance code. For each failing metric, name the specific resource/element causing it and rank fixes by expected leverage (e.g. preload the LCP image, defer non-critical JS, reserve image dimensions). Give framework-specific direction where the stack is known. Never fabricate measured values — if field data is absent, ask the user to supply a PSI/CrUX key or leave a clearly-marked TODO placeholder rather than guessing a metric.
Verification
- Field (Tier 1):
node ${CLAUDE_SKILL_DIR}/../../scripts/psi-client.mjs --url <u>calls the PageSpeed Insights API (loadingExperience, methodpsi_api); CrUX History via the same client (crux_api) for p75 trends. Requires a free API key. - Tier 0 (no key): emit
status: needs_apifor every field-CWV finding, plus clearly-labeled lab heuristics (render-blocking count, missing image dimensions, bundle weight). Label lab signals "lab data — not what Google ranks on"; never present a heuristic as a measured field value. - When the required data tier is unavailable, status is
needs_api— never a falsepass.