From 9dbaf731d325f64208384b02a449fe501ca0bf24 Mon Sep 17 00:00:00 2001 From: Adam Raine <6752989+adamraine@users.noreply.github.com> Date: Wed, 22 Sep 2021 15:13:40 -0400 Subject: [PATCH] core(fr): api for constructing flow result (#13034) --- flow-report/src/sidebar/flow.tsx | 9 +- flow-report/src/sidebar/sidebar.tsx | 2 +- flow-report/src/summary/summary.tsx | 15 +- flow-report/src/util.ts | 67 +- flow-report/test/sidebar/flow-test.tsx | 4 +- flow-report/test/summary/summary-test.tsx | 15 +- flow-report/test/util-test.tsx | 45 +- lighthouse-core/fraggle-rock/user-flow.js | 135 + .../scripts/update-flow-fixtures.js | 41 +- .../fraggle-rock/reports/sample-lhrs.json | 43311 ++++++++-------- types/global-lh.d.ts | 2 +- types/lhr/flow.d.ts | 15 +- 12 files changed, 21688 insertions(+), 21973 deletions(-) create mode 100644 lighthouse-core/fraggle-rock/user-flow.js diff --git a/flow-report/src/sidebar/flow.tsx b/flow-report/src/sidebar/flow.tsx index 16f1713c6ed3..395f30b88d81 100644 --- a/flow-report/src/sidebar/flow.tsx +++ b/flow-report/src/sidebar/flow.tsx @@ -7,7 +7,7 @@ import {FunctionComponent} from 'preact'; import {FlowSegment} from '../common'; -import {classNames, useCurrentLhr, useDerivedStepNames, useFlowResult} from '../util'; +import {classNames, useCurrentLhr, useFlowResult} from '../util'; import {Separator} from '../common'; const SidebarFlowStep: FunctionComponent<{ @@ -42,13 +42,12 @@ const SidebarFlowSeparator: FunctionComponent = () => { export const SidebarFlow: FunctionComponent = () => { const flowResult = useFlowResult(); const currentLhr = useCurrentLhr(); - const stepNames = useDerivedStepNames(); return (
{ - flowResult.lhrs.map((lhr, index) => { - const stepName = stepNames[index]; + flowResult.steps.map((step, index) => { + const {lhr, name} = step; const url = new URL(location.href); url.hash = `#index=${index}`; return <> @@ -61,7 +60,7 @@ export const SidebarFlow: FunctionComponent = () => { key={lhr.fetchTime} mode={lhr.gatherMode} href={url.href} - label={stepName} + label={name} isCurrent={index === (currentLhr && currentLhr.index)} /> ; diff --git a/flow-report/src/sidebar/sidebar.tsx b/flow-report/src/sidebar/sidebar.tsx index 26d967ceb5f4..258da89c385b 100644 --- a/flow-report/src/sidebar/sidebar.tsx +++ b/flow-report/src/sidebar/sidebar.tsx @@ -74,7 +74,7 @@ export const SidebarHeader: FunctionComponent<{title: string, date: string}> = ( export const Sidebar: FunctionComponent = () => { const flowResult = useFlowResult(); - const firstLhr = flowResult.lhrs[0]; + const firstLhr = flowResult.steps[0].lhr; return (
diff --git a/flow-report/src/summary/summary.tsx b/flow-report/src/summary/summary.tsx index c579c056d163..d62e803e0222 100644 --- a/flow-report/src/summary/summary.tsx +++ b/flow-report/src/summary/summary.tsx @@ -8,7 +8,7 @@ import {FunctionComponent} from 'preact'; import {useMemo} from 'preact/hooks'; import {FlowSegment, Separator} from '../common'; -import {getScreenDimensions, getScreenshot, useDerivedStepNames, useFlowResult} from '../util'; +import {getScreenDimensions, getScreenshot, useFlowResult} from '../util'; import {Util} from '../../../report/renderer/util'; import {CategoryScore} from '../wrappers/category-score'; @@ -113,15 +113,14 @@ export const SummaryFlowStep: FunctionComponent<{ */ const SummaryFlow: FunctionComponent = () => { const flowResult = useFlowResult(); - const stepNames = useDerivedStepNames(); return (
{ - flowResult.lhrs.map((lhr, index) => + flowResult.steps.map((step, index) => ) @@ -136,8 +135,8 @@ export const SummaryHeader: FunctionComponent = () => { let numNavigation = 0; let numTimespan = 0; let numSnapshot = 0; - for (const lhr of flowResult.lhrs) { - switch (lhr.gatherMode) { + for (const step of flowResult.steps) { + switch (step.lhr.gatherMode) { case 'navigation': numNavigation++; break; diff --git a/flow-report/src/util.ts b/flow-report/src/util.ts index 4f12f9dadd64..6d52b7787940 100644 --- a/flow-report/src/util.ts +++ b/flow-report/src/util.ts @@ -14,33 +14,6 @@ function getHashParam(param: string): string|null { return params.get(param); } -function shortenUrl(longUrl: string) { - const url = new URL(longUrl); - return `${url.hostname}${url.pathname}`; -} - -/** - * The step label should be enumerated if there is another report of the same gather mode in the same section. - * Navigation reports will never be enumerated. - */ -function shouldEnumerate(flowResult: LH.FlowResult, index: number) { - const {lhrs} = flowResult; - if (lhrs[index].gatherMode === 'navigation') return false; - - for (let i = index + 1; lhrs[i] && lhrs[i].gatherMode !== 'navigation'; ++i) { - if (lhrs[i].gatherMode === lhrs[index].gatherMode) { - return true; - } - } - for (let i = index - 1; lhrs[i] && lhrs[i].gatherMode !== 'navigation'; --i) { - if (lhrs[i].gatherMode === lhrs[index].gatherMode) { - return true; - } - } - return false; -} - - export function classNames(...args: Array>): string { const classes = []; for (const arg of args) { @@ -83,7 +56,7 @@ export function useFlowResult(): LH.FlowResult { export function useLocale(): LH.Locale { const flowResult = useFlowResult(); - return flowResult.lhrs[0].configSettings.locale; + return flowResult.steps[0].lhr.configSettings.locale; } export function useHashParam(param: string) { @@ -118,42 +91,12 @@ export function useCurrentLhr(): {value: LH.Result, index: number}|null { return null; } - const value = flowResult.lhrs[index]; - if (!value) { - console.warn(`No LHR at index ${index}`); + const step = flowResult.steps[index]; + if (!step) { + console.warn(`No flow step at index ${index}`); return null; } - return {value, index}; + return {value: step.lhr, index}; }, [indexString, flowResult]); } - -export function useDerivedStepNames() { - const flowResult = useFlowResult(); - - return useMemo(() => { - let numTimespan = 1; - let numSnapshot = 1; - - return flowResult.lhrs.map((lhr, index) => { - const shortUrl = shortenUrl(lhr.finalUrl); - - switch (lhr.gatherMode) { - case 'navigation': - numTimespan = 1; - numSnapshot = 1; - return `Navigation report (${shortUrl})`; - case 'timespan': - if (shouldEnumerate(flowResult, index)) { - return `Timespan report ${numTimespan++} (${shortUrl})`; - } - return `Timespan report (${shortUrl})`; - case 'snapshot': - if (shouldEnumerate(flowResult, index)) { - return `Snapshot report ${numSnapshot++} (${shortUrl})`; - } - return `Snapshot report (${shortUrl})`; - } - }); - }, [flowResult]); -} diff --git a/flow-report/test/sidebar/flow-test.tsx b/flow-report/test/sidebar/flow-test.tsx index db07180ae26b..11f37f0a5531 100644 --- a/flow-report/test/sidebar/flow-test.tsx +++ b/flow-report/test/sidebar/flow-test.tsx @@ -30,8 +30,8 @@ describe('SidebarFlow', () => { const root = render(, {wrapper}); const navigation = root.getByText('Navigation report (www.mikescerealshack.co/)'); - const timespan = root.getByText('Timespan report (www.mikescerealshack.co/search)'); - const snapshot = root.getByText('Snapshot report (www.mikescerealshack.co/search)'); + const timespan = root.getByText('Search input'); + const snapshot = root.getByText('Search results'); const navigation2 = root.getByText('Navigation report (www.mikescerealshack.co/corrections)'); const links = root.getAllByRole('link') as HTMLAnchorElement[]; diff --git a/flow-report/test/summary/summary-test.tsx b/flow-report/test/summary/summary-test.tsx index 882685ca514e..ba7c9fe1c4f5 100644 --- a/flow-report/test/summary/summary-test.tsx +++ b/flow-report/test/summary/summary-test.tsx @@ -50,7 +50,7 @@ describe('SummaryHeader', () => { describe('SummaryFlowStep', () => { it('renders navigation step', async () => { const root = render(, {wrapper}); @@ -77,7 +77,7 @@ describe('SummaryFlowStep', () => { it('renders timespan step', async () => { const root = render(, {wrapper}); @@ -89,23 +89,24 @@ describe('SummaryFlowStep', () => { const screenshot = root.getByTestId('SummaryFlowStep__screenshot') as HTMLImageElement; expect(screenshot.src).toBeFalsy(); - expect(root.getByTestId('SummaryCategory__null')); + // Accessibility and SEO are missing in timespan. + const nullCategories = root.getAllByTestId('SummaryCategory__null'); + expect(nullCategories).toHaveLength(2); + const gauges = root.getAllByTestId('CategoryScore'); - expect(gauges).toHaveLength(3); + expect(gauges).toHaveLength(2); const links = root.getAllByRole('link') as HTMLAnchorElement[]; expect(links.map(a => a.href)).toEqual([ 'file:///Users/example/report.html/#index=1', 'file:///Users/example/report.html/#index=1&anchor=performance', - // Accessibility is missing in timespan. 'file:///Users/example/report.html/#index=1&anchor=best-practices', - 'file:///Users/example/report.html/#index=1&anchor=seo', ]); }); it('renders snapshot step', async () => { const root = render(, {wrapper}); diff --git a/flow-report/test/util-test.tsx b/flow-report/test/util-test.tsx index a700302b0d3d..bb1cddeaad6f 100644 --- a/flow-report/test/util-test.tsx +++ b/flow-report/test/util-test.tsx @@ -8,11 +8,12 @@ import fs from 'fs'; import {dirname} from 'path'; import {fileURLToPath} from 'url'; +import {jest} from '@jest/globals'; import {renderHook} from '@testing-library/preact-hooks'; import {FunctionComponent} from 'preact'; import {act} from 'preact/test-utils'; -import {FlowResultContext, useCurrentLhr, useDerivedStepNames} from '../src/util'; +import {FlowResultContext, useCurrentLhr} from '../src/util'; const flowResult: LH.FlowResult = JSON.parse( fs.readFileSync( @@ -25,6 +26,7 @@ const flowResult: LH.FlowResult = JSON.parse( let wrapper: FunctionComponent; beforeEach(() => { + global.console.warn = jest.fn(); wrapper = ({children}) => ( {children} ); @@ -34,9 +36,10 @@ describe('useCurrentLhr', () => { it('gets current lhr index from url hash', () => { global.location.hash = '#index=1'; const {result} = renderHook(() => useCurrentLhr(), {wrapper}); + expect(console.warn).not.toHaveBeenCalled(); expect(result.current).toEqual({ index: 1, - value: flowResult.lhrs[1], + value: flowResult.steps[1].lhr, }); }); @@ -46,7 +49,7 @@ describe('useCurrentLhr', () => { expect(render.result.current).toEqual({ index: 1, - value: flowResult.lhrs[1], + value: flowResult.steps[1].lhr, }); await act(() => { @@ -54,56 +57,30 @@ describe('useCurrentLhr', () => { }); await render.waitForNextUpdate(); + expect(console.warn).not.toHaveBeenCalled(); expect(render.result.current).toEqual({ index: 2, - value: flowResult.lhrs[2], + value: flowResult.steps[2].lhr, }); }); it('return null if lhr index is unset', () => { const {result} = renderHook(() => useCurrentLhr(), {wrapper}); + expect(console.warn).not.toHaveBeenCalled(); expect(result.current).toBeNull(); }); it('return null if lhr index is out of bounds', () => { global.location.hash = '#index=5'; const {result} = renderHook(() => useCurrentLhr(), {wrapper}); + expect(console.warn).toHaveBeenCalled(); expect(result.current).toBeNull(); }); it('returns null for invalid value', () => { global.location.hash = '#index=OHNO'; const {result} = renderHook(() => useCurrentLhr(), {wrapper}); + expect(console.warn).toHaveBeenCalled(); expect(result.current).toBeNull(); }); }); - -describe('useDerivedStepNames', () => { - it('counts up for each mode', () => { - const {result} = renderHook(() => useDerivedStepNames(), {wrapper}); - expect(result.current).toEqual([ - 'Navigation report (www.mikescerealshack.co/)', - 'Timespan report (www.mikescerealshack.co/search)', - 'Snapshot report (www.mikescerealshack.co/search)', - 'Navigation report (www.mikescerealshack.co/corrections)', - ]); - }); - - it('enumerates if multiple in same group', () => { - const lhrs = flowResult.lhrs; - lhrs[3] = lhrs[2]; - const newFlowResult = {lhrs}; - const wrapper: FunctionComponent = ({children}) => ( - {children} - ); - - const {result} = renderHook(() => useDerivedStepNames(), {wrapper}); - - expect(result.current).toEqual([ - 'Navigation report (www.mikescerealshack.co/)', - 'Timespan report (www.mikescerealshack.co/search)', - 'Snapshot report 1 (www.mikescerealshack.co/search)', - 'Snapshot report 2 (www.mikescerealshack.co/search)', - ]); - }); -}); diff --git a/lighthouse-core/fraggle-rock/user-flow.js b/lighthouse-core/fraggle-rock/user-flow.js new file mode 100644 index 000000000000..8c16e1ca198c --- /dev/null +++ b/lighthouse-core/fraggle-rock/user-flow.js @@ -0,0 +1,135 @@ +/** + * @license Copyright 2021 The Lighthouse Authors. All Rights Reserved. + * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 + * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. + */ +'use strict'; + +const {generateFlowReportHtml} = require('../../report/generator/report-generator.js'); +const {navigation, startTimespan, snapshot} = require('./api.js'); + +/** @typedef {Parameters[0]} FrOptions */ +/** @typedef {Omit} UserFlowOptions */ +/** @typedef {UserFlowOptions & {stepName?: string}} StepOptions */ + +class UserFlow { + /** + * @param {FrOptions['page']} page + * @param {UserFlowOptions=} options + */ + constructor(page, options) { + /** @type {FrOptions} */ + this.options = {page, ...options}; + /** @type {LH.FlowResult.Step[]} */ + this.steps = []; + } + + /** + * @param {string} longUrl + * @returns {string} + */ + _shortenUrl(longUrl) { + const url = new URL(longUrl); + return `${url.hostname}${url.pathname}`; + } + + /** + * @param {LH.Result} lhr + * @return {string} + */ + _getDefaultStepName(lhr) { + const shortUrl = this._shortenUrl(lhr.finalUrl); + switch (lhr.gatherMode) { + case 'navigation': + return `Navigation report (${shortUrl})`; + case 'timespan': + return `Timespan report (${shortUrl})`; + case 'snapshot': + return `Snapshot report (${shortUrl})`; + } + } + + /** + * @param {string} url + * @param {StepOptions=} stepOptions + */ + async navigate(url, stepOptions) { + if (this.currentTimespan) throw Error('Timespan already in progress'); + + const options = {url, ...this.options, ...stepOptions}; + const result = await navigation(options); + if (!result) throw Error('Navigation returned undefined'); + + const providedName = stepOptions && stepOptions.stepName; + this.steps.push({ + lhr: result.lhr, + name: providedName || this._getDefaultStepName(result.lhr), + }); + + return result; + } + + /** + * @param {StepOptions=} stepOptions + */ + async startTimespan(stepOptions) { + if (this.currentTimespan) throw Error('Timespan already in progress'); + + const options = {...this.options, ...stepOptions}; + const timespan = await startTimespan(options); + this.currentTimespan = {timespan, options}; + } + + async endTimespan() { + if (!this.currentTimespan) throw Error('No timespan in progress'); + + const {timespan, options} = this.currentTimespan; + const result = await timespan.endTimespan(); + this.currentTimespan = undefined; + if (!result) throw Error('Timespan returned undefined'); + + const providedName = options && options.stepName; + this.steps.push({ + lhr: result.lhr, + name: providedName || this._getDefaultStepName(result.lhr), + }); + + return result; + } + + /** + * @param {StepOptions=} stepOptions + */ + async snapshot(stepOptions) { + if (this.currentTimespan) throw Error('Timespan already in progress'); + + const options = {...this.options, ...stepOptions}; + const result = await snapshot(options); + if (!result) throw Error('Snapshot returned undefined'); + + const providedName = stepOptions && stepOptions.stepName; + this.steps.push({ + lhr: result.lhr, + name: providedName || this._getDefaultStepName(result.lhr), + }); + + return result; + } + + /** + * @return {LH.FlowResult} + */ + getFlowResult() { + return {steps: this.steps}; + } + + /** + * @return {string} + */ + generateReport() { + const flowResult = this.getFlowResult(); + return generateFlowReportHtml(flowResult); + } +} + +module.exports = UserFlow; diff --git a/lighthouse-core/scripts/update-flow-fixtures.js b/lighthouse-core/scripts/update-flow-fixtures.js index 1db469167981..706bdd5c2118 100644 --- a/lighthouse-core/scripts/update-flow-fixtures.js +++ b/lighthouse-core/scripts/update-flow-fixtures.js @@ -8,52 +8,37 @@ const fs = require('fs'); const open = require('open'); const puppeteer = require('puppeteer'); -const lighthouse = require('../fraggle-rock/api.js'); -const reportGenerator = require('../../report/generator/report-generator.js'); +const UserFlow = require('../fraggle-rock/user-flow.js'); (async () => { - const browser = await puppeteer.launch({headless: false, slowMo: 50}); + const browser = await puppeteer.launch({headless: false}); try { const page = await browser.newPage(); - const navigationResult1 = await lighthouse.navigation({ - url: 'https://www.mikescerealshack.co', - page, - }); + const flow = new UserFlow(page); - const timespan = await lighthouse.startTimespan({page}); + await flow.navigate('https://www.mikescerealshack.co'); + + await flow.startTimespan({stepName: 'Search input'}); await page.type('input', 'call of duty'); const networkQuietPromise = page.waitForNavigation({waitUntil: ['networkidle0']}); await page.click('button[type=submit]'); await networkQuietPromise; - const timespanResult = await timespan.endTimespan(); - - const snapshotResult = await lighthouse.snapshot({page}); + await flow.endTimespan(); - const navigationResult2 = await lighthouse.navigation({ - url: 'https://www.mikescerealshack.co/corrections', - page, - }); + await flow.snapshot({stepName: 'Search results'}); - if ( - !navigationResult1 || - !navigationResult2 || - !timespanResult || - !snapshotResult - ) throw new Error('No results'); + await flow.navigate('https://www.mikescerealshack.co/corrections'); - const flow = { - lhrs: [navigationResult1.lhr, timespanResult.lhr, snapshotResult.lhr, navigationResult2.lhr], - }; + const flowResult = flow.getFlowResult(); + const report = flow.generateReport(); fs.writeFileSync( `${__dirname}/../test/fixtures/fraggle-rock/reports/sample-lhrs.json`, - JSON.stringify(flow, null, 2) + JSON.stringify(flowResult, null, 2) ); - const htmlReport = reportGenerator.generateFlowReportHtml(flow); - - fs.writeFileSync(`${__dirname}/../../flow.report.html`, htmlReport); + fs.writeFileSync(`${__dirname}/../../flow.report.html`, report); open(`${__dirname}/../../flow.report.html`); process.exit(0); diff --git a/lighthouse-core/test/fixtures/fraggle-rock/reports/sample-lhrs.json b/lighthouse-core/test/fixtures/fraggle-rock/reports/sample-lhrs.json index 36adf92368a7..7056a262ca2e 100644 --- a/lighthouse-core/test/fixtures/fraggle-rock/reports/sample-lhrs.json +++ b/lighthouse-core/test/fixtures/fraggle-rock/reports/sample-lhrs.json @@ -1,22471 +1,22138 @@ { - "lhrs": [ + "steps": [ { - "gatherMode": "navigation", - "userAgent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4469.0 Safari/537.36", - "environment": { - "networkUserAgent": "", - "hostUserAgent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4469.0 Safari/537.36", - "benchmarkIndex": 1716, - "credits": { - "axe-core": "4.2.3" - } - }, - "lighthouseVersion": "8.3.0", - "fetchTime": "2021-08-26T19:23:27.529Z", - "requestedUrl": "https://www.mikescerealshack.co/", - "finalUrl": "https://www.mikescerealshack.co/", - "runWarnings": [], - "audits": { - "is-on-https": { - "id": "is-on-https", - "title": "Uses HTTPS", - "description": "All sites should be protected with HTTPS, even ones that don't handle sensitive data. This includes avoiding [mixed content](https://developers.google.com/web/fundamentals/security/prevent-mixed-content/what-is-mixed-content), where some resources are loaded over HTTP despite the initial request being served over HTTPS. HTTPS prevents intruders from tampering with or passively listening in on the communications between your app and your users, and is a prerequisite for HTTP/2 and many new web platform APIs. [Learn more](https://web.dev/is-on-https/).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "service-worker": { - "id": "service-worker", - "title": "Does not register a service worker that controls page and `start_url`", - "description": "The service worker is the technology that enables your app to use many Progressive Web App features, such as offline, add to homescreen, and push notifications. [Learn more](https://web.dev/service-worker/).", - "score": 0, - "scoreDisplayMode": "binary" - }, - "viewport": { - "id": "viewport", - "title": "Has a `` tag with `width` or `initial-scale`", - "description": "Add a `` tag to optimize your app for mobile screens. [Learn more](https://web.dev/viewport/).", - "score": 1, - "scoreDisplayMode": "binary", - "warnings": [] - }, - "first-contentful-paint": { - "id": "first-contentful-paint", - "title": "First Contentful Paint", - "description": "First Contentful Paint marks the time at which the first text or image is painted. [Learn more](https://web.dev/first-contentful-paint/).", - "score": 0.97, - "scoreDisplayMode": "numeric", - "numericValue": 1411.9320000000002, - "numericUnit": "millisecond", - "displayValue": "1.4 s" - }, - "largest-contentful-paint": { - "id": "largest-contentful-paint", - "title": "Largest Contentful Paint", - "description": "Largest Contentful Paint marks the time at which the largest text or image is painted. [Learn more](https://web.dev/lighthouse-largest-contentful-paint/)", - "score": 0.93, - "scoreDisplayMode": "numeric", - "numericValue": 2333.616, - "numericUnit": "millisecond", - "displayValue": "2.3 s" - }, - "first-meaningful-paint": { - "id": "first-meaningful-paint", - "title": "First Meaningful Paint", - "description": "First Meaningful Paint measures when the primary content of a page is visible. [Learn more](https://web.dev/first-meaningful-paint/).", - "score": 0.99, - "scoreDisplayMode": "numeric", - "numericValue": 1411.9320000000002, - "numericUnit": "millisecond", - "displayValue": "1.4 s" - }, - "speed-index": { - "id": "speed-index", - "title": "Speed Index", - "description": "Speed Index shows how quickly the contents of a page are visibly populated. [Learn more](https://web.dev/speed-index/).", - "score": 1, - "scoreDisplayMode": "numeric", - "numericValue": 1679.2807200018947, - "numericUnit": "millisecond", - "displayValue": "1.7 s" - }, - "screenshot-thumbnails": { - "id": "screenshot-thumbnails", - "title": "Screenshot Thumbnails", - "description": "This is what the load of your site looked like.", - "score": null, - "scoreDisplayMode": "informative", - "details": { - "type": "filmstrip", - "scale": 3000, - "items": [ - { - "timing": 300, - "timestamp": 121805065196, - "data": "" - }, - { - "timing": 600, - "timestamp": 121805365196, - "data": "" - }, - { - "timing": 900, - "timestamp": 121805665196, - "data": "" - }, - { - "timing": 1200, - "timestamp": 121805965196, - "data": "" - }, - { - "timing": 1500, - "timestamp": 121806265196, - "data": "" - }, - { - "timing": 1800, - "timestamp": 121806565196, - "data": "" - }, - { - "timing": 2100, - "timestamp": 121806865196, - "data": "" - }, - { - "timing": 2400, - "timestamp": 121807165196, - "data": "" - }, - { - "timing": 2700, - "timestamp": 121807465196, - "data": "" - }, - { - "timing": 3000, - "timestamp": 121807765196, - "data": "" - } - ] - } - }, - "final-screenshot": { - "id": "final-screenshot", - "title": "Final Screenshot", - "description": "The last screenshot captured of the pageload.", - "score": null, - "scoreDisplayMode": "informative", - "details": { - "type": "screenshot", - "timing": 1029, - "timestamp": 121805794681, - "data": "" - } - }, - "total-blocking-time": { - "id": "total-blocking-time", - "title": "Total Blocking Time", - "description": "Sum of all time periods between FCP and Time to Interactive, when task length exceeded 50ms, expressed in milliseconds. [Learn more](https://web.dev/lighthouse-total-blocking-time/).", - "score": 1, - "scoreDisplayMode": "numeric", - "numericValue": 42, - "numericUnit": "millisecond", - "displayValue": "40 ms" - }, - "max-potential-fid": { - "id": "max-potential-fid", - "title": "Max Potential First Input Delay", - "description": "The maximum potential First Input Delay that your users could experience is the duration of the longest task. [Learn more](https://web.dev/lighthouse-max-potential-fid/).", - "score": 0.98, - "scoreDisplayMode": "numeric", - "numericValue": 86, - "numericUnit": "millisecond", - "displayValue": "90 ms" - }, - "cumulative-layout-shift": { - "id": "cumulative-layout-shift", - "title": "Cumulative Layout Shift", - "description": "Cumulative Layout Shift measures the movement of visible elements within the viewport. [Learn more](https://web.dev/cls/).", - "score": 1, - "scoreDisplayMode": "numeric", - "numericValue": 0.0026205596923828123, - "numericUnit": "unitless", - "displayValue": "0.003", - "details": { - "type": "debugdata", - "items": [ - { - "cumulativeLayoutShiftMainFrame": 0.0026205596923828123, - "totalCumulativeLayoutShift": 0.0026205596923828123 - } - ] - } - }, - "errors-in-console": { - "id": "errors-in-console", - "title": "No browser errors logged to the console", - "description": "Errors logged to the console indicate unresolved problems. They can come from network request failures and other browser concerns. [Learn more](https://web.dev/errors-in-console/)", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "server-response-time": { - "id": "server-response-time", - "title": "Initial server response time was short", - "description": "Keep the server response time for the main document short because all other requests depend on it. [Learn more](https://web.dev/time-to-first-byte/).", - "score": 1, - "scoreDisplayMode": "binary", - "numericValue": 41.55800000000005, - "numericUnit": "millisecond", - "displayValue": "Root document took 40 ms", - "details": { - "type": "opportunity", - "headings": [ - { - "key": "url", - "valueType": "url", - "label": "URL" - }, - { - "key": "responseTime", - "valueType": "timespanMs", - "label": "Time Spent" - } - ], - "items": [ - { - "url": "https://www.mikescerealshack.co/", - "responseTime": 41.55800000000005 - } - ], - "overallSavingsMs": -58.44199999999995 - } - }, - "interactive": { - "id": "interactive", - "title": "Time to Interactive", - "description": "Time to interactive is the amount of time it takes for the page to become fully interactive. [Learn more](https://web.dev/interactive/).", - "score": 0.99, - "scoreDisplayMode": "numeric", - "numericValue": 2027.5810000000001, - "numericUnit": "millisecond", - "displayValue": "2.0 s" - }, - "user-timings": { - "id": "user-timings", - "title": "User Timing marks and measures", - "description": "Consider instrumenting your app with the User Timing API to measure your app's real-world performance during key user experiences. [Learn more](https://web.dev/user-timings/).", - "score": null, - "scoreDisplayMode": "informative", - "displayValue": "4 user timings", - "details": { - "type": "table", - "headings": [ - { - "key": "name", - "itemType": "text", - "text": "Name" - }, - { - "key": "timingType", - "itemType": "text", - "text": "Type" - }, - { - "key": "startTime", - "itemType": "ms", - "granularity": 0.01, - "text": "Start Time" - }, - { - "key": "duration", - "itemType": "ms", - "granularity": 0.01, - "text": "Duration" - } - ], - "items": [ - { - "name": "Next.js-before-hydration", - "startTime": 0, - "duration": 715.62, - "timingType": "Measure" - }, - { - "name": "Next.js-hydration", - "startTime": 715.62, - "duration": 12.017, - "timingType": "Measure" - }, - { - "name": "beforeRender", - "startTime": 715.637, - "timingType": "Mark" - }, - { - "name": "afterHydrate", - "startTime": 727.648, - "timingType": "Mark" - } - ] - } - }, - "critical-request-chains": { - "id": "critical-request-chains", - "title": "Avoid chaining critical requests", - "description": "The Critical Request Chains below show you what resources are loaded with a high priority. Consider reducing the length of chains, reducing the download size of resources, or deferring the download of unnecessary resources to improve page load. [Learn more](https://web.dev/critical-request-chains/).", - "score": null, - "scoreDisplayMode": "notApplicable", - "displayValue": "", - "details": { - "type": "criticalrequestchain", - "chains": { - "86CE0257147E2C95A023EE7493070A28": { - "request": { - "url": "https://www.mikescerealshack.co/", - "startTime": 121804.769144, - "endTime": 121805.293867, - "responseReceivedTime": 121805.286882, - "transferSize": 3550 + "lhr": { + "gatherMode": "navigation", + "userAgent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.0 Safari/537.36", + "environment": { + "networkUserAgent": "", + "hostUserAgent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.0 Safari/537.36", + "benchmarkIndex": 1929, + "credits": { + "axe-core": "4.2.3" + } + }, + "lighthouseVersion": "8.4.0", + "fetchTime": "2021-09-14T22:24:22.462Z", + "requestedUrl": "https://www.mikescerealshack.co/", + "finalUrl": "https://www.mikescerealshack.co/", + "runWarnings": [], + "audits": { + "is-on-https": { + "id": "is-on-https", + "title": "Uses HTTPS", + "description": "All sites should be protected with HTTPS, even ones that don't handle sensitive data. This includes avoiding [mixed content](https://developers.google.com/web/fundamentals/security/prevent-mixed-content/what-is-mixed-content), where some resources are loaded over HTTP despite the initial request being served over HTTPS. HTTPS prevents intruders from tampering with or passively listening in on the communications between your app and your users, and is a prerequisite for HTTP/2 and many new web platform APIs. [Learn more](https://web.dev/is-on-https/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "service-worker": { + "id": "service-worker", + "title": "Does not register a service worker that controls page and `start_url`", + "description": "The service worker is the technology that enables your app to use many Progressive Web App features, such as offline, add to homescreen, and push notifications. [Learn more](https://web.dev/service-worker/).", + "score": 0, + "scoreDisplayMode": "binary" + }, + "viewport": { + "id": "viewport", + "title": "Has a `` tag with `width` or `initial-scale`", + "description": "A `` not only optimizes your app for mobile screen sizes, but also prevents [a 300 millisecond delay to user input](https://developers.google.com/web/updates/2013/12/300ms-tap-delay-gone-away). [Learn more](https://web.dev/viewport/).", + "score": 1, + "scoreDisplayMode": "binary", + "warnings": [] + }, + "first-contentful-paint": { + "id": "first-contentful-paint", + "title": "First Contentful Paint", + "description": "First Contentful Paint marks the time at which the first text or image is painted. [Learn more](https://web.dev/first-contentful-paint/).", + "score": 0.98, + "scoreDisplayMode": "numeric", + "numericValue": 1323.2420000000004, + "numericUnit": "millisecond", + "displayValue": "1.3 s" + }, + "largest-contentful-paint": { + "id": "largest-contentful-paint", + "title": "Largest Contentful Paint", + "description": "Largest Contentful Paint marks the time at which the largest text or image is painted. [Learn more](https://web.dev/lighthouse-largest-contentful-paint/)", + "score": 0.79, + "scoreDisplayMode": "numeric", + "numericValue": 2969.3620000000005, + "numericUnit": "millisecond", + "displayValue": "3.0 s" + }, + "first-meaningful-paint": { + "id": "first-meaningful-paint", + "title": "First Meaningful Paint", + "description": "First Meaningful Paint measures when the primary content of a page is visible. [Learn more](https://web.dev/first-meaningful-paint/).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 1323.2420000000004, + "numericUnit": "millisecond", + "displayValue": "1.3 s" + }, + "speed-index": { + "id": "speed-index", + "title": "Speed Index", + "description": "Speed Index shows how quickly the contents of a page are visibly populated. [Learn more](https://web.dev/speed-index/).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 1323.2420000000004, + "numericUnit": "millisecond", + "displayValue": "1.3 s" + }, + "screenshot-thumbnails": { + "id": "screenshot-thumbnails", + "title": "Screenshot Thumbnails", + "description": "This is what the load of your site looked like.", + "score": null, + "scoreDisplayMode": "informative", + "details": { + "type": "filmstrip", + "scale": 3000, + "items": [ + { + "timing": 300, + "timestamp": 56836623734, + "data": "" + }, + { + "timing": 600, + "timestamp": 56836923734, + "data": "" + }, + { + "timing": 900, + "timestamp": 56837223734, + "data": "" + }, + { + "timing": 1200, + "timestamp": 56837523734, + "data": "" + }, + { + "timing": 1500, + "timestamp": 56837823734, + "data": "" + }, + { + "timing": 1800, + "timestamp": 56838123734, + "data": "" + }, + { + "timing": 2100, + "timestamp": 56838423734, + "data": "" + }, + { + "timing": 2400, + "timestamp": 56838723734, + "data": "" + }, + { + "timing": 2700, + "timestamp": 56839023734, + "data": "" + }, + { + "timing": 3000, + "timestamp": 56839323734, + "data": "" } - } - }, - "longestChain": { - "duration": 524.7229999949923, - "length": 1, - "transferSize": 3550 + ] } - } - }, - "redirects": { - "id": "redirects", - "title": "Avoid multiple page redirects", - "description": "Redirects introduce additional delays before the page can be loaded. [Learn more](https://web.dev/redirects/).", - "score": 1, - "scoreDisplayMode": "numeric", - "numericValue": 0, - "numericUnit": "millisecond", - "displayValue": "", - "details": { - "type": "opportunity", - "headings": [], - "items": [], - "overallSavingsMs": 0 - } - }, - "installable-manifest": { - "id": "installable-manifest", - "title": "Web app manifest or service worker do not meet the installability requirements", - "description": "Service worker is the technology that enables your app to use many Progressive Web App features, such as offline, add to homescreen, and push notifications. With proper service worker and manifest implementations, browsers can proactively prompt users to add your app to their homescreen, which can lead to higher engagement. [Learn more](https://web.dev/installable-manifest/).", - "score": 0, - "scoreDisplayMode": "binary", - "numericValue": 2, - "numericUnit": "element", - "displayValue": "2 reasons", - "warnings": [], - "details": { - "type": "table", - "headings": [ - { - "key": "reason", - "itemType": "text", - "text": "Failure reason" - } - ], - "items": [ - { - "reason": "Page has no manifest URL" - }, - { - "reason": "No manifest was fetched" - } - ], - "debugData": { + }, + "final-screenshot": { + "id": "final-screenshot", + "title": "Final Screenshot", + "description": "The last screenshot captured of the pageload.", + "score": null, + "scoreDisplayMode": "informative", + "details": { + "type": "screenshot", + "timing": 635, + "timestamp": 56836958404, + "data": "" + } + }, + "total-blocking-time": { + "id": "total-blocking-time", + "title": "Total Blocking Time", + "description": "Sum of all time periods between FCP and Time to Interactive, when task length exceeded 50ms, expressed in milliseconds. [Learn more](https://web.dev/lighthouse-total-blocking-time/).", + "score": 0.99, + "scoreDisplayMode": "numeric", + "numericValue": 83.99999999999977, + "numericUnit": "millisecond", + "displayValue": "80 ms" + }, + "max-potential-fid": { + "id": "max-potential-fid", + "title": "Max Potential First Input Delay", + "description": "The maximum potential First Input Delay that your users could experience is the duration of the longest task. [Learn more](https://web.dev/lighthouse-max-potential-fid/).", + "score": 0.89, + "scoreDisplayMode": "numeric", + "numericValue": 134, + "numericUnit": "millisecond", + "displayValue": "130 ms" + }, + "cumulative-layout-shift": { + "id": "cumulative-layout-shift", + "title": "Cumulative Layout Shift", + "description": "Cumulative Layout Shift measures the movement of visible elements within the viewport. [Learn more](https://web.dev/cls/).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 0.004178787231445312, + "numericUnit": "unitless", + "displayValue": "0.004", + "details": { "type": "debugdata", - "manifestUrl": null + "items": [ + { + "cumulativeLayoutShiftMainFrame": 0.004178787231445312, + "totalCumulativeLayoutShift": 0.004178787231445312 + } + ] } - } - }, - "apple-touch-icon": { - "id": "apple-touch-icon", - "title": "Does not provide a valid `apple-touch-icon`", - "description": "For ideal appearance on iOS when users add a progressive web app to the home screen, define an `apple-touch-icon`. It must point to a non-transparent 192px (or 180px) square PNG. [Learn More](https://web.dev/apple-touch-icon/).", - "score": 0, - "scoreDisplayMode": "binary", - "warnings": [] - }, - "splash-screen": { - "id": "splash-screen", - "title": "Is not configured for a custom splash screen", - "description": "A themed splash screen ensures a high-quality experience when users launch your app from their homescreens. [Learn more](https://web.dev/splash-screen/).", - "score": 0, - "scoreDisplayMode": "binary", - "explanation": "Failures: No manifest was fetched.", - "details": { - "type": "debugdata", - "items": [ - { - "failures": [ - "No manifest was fetched" - ], - "isParseFailure": true, - "parseFailureReason": "No manifest was fetched" - } - ] - } - }, - "themed-omnibox": { - "id": "themed-omnibox", - "title": "Does not set a theme color for the address bar.", - "description": "The browser address bar can be themed to match your site. [Learn more](https://web.dev/themed-omnibox/).", - "score": 0, - "scoreDisplayMode": "binary", - "explanation": "Failures: No manifest was fetched,\nNo `` tag found.", - "details": { - "type": "debugdata", - "items": [ - { - "failures": [ - "No manifest was fetched", - "No `` tag found" - ], - "themeColor": null, - "isParseFailure": true, - "parseFailureReason": "No manifest was fetched" - } - ] - } - }, - "maskable-icon": { - "id": "maskable-icon", - "title": "Manifest doesn't have a maskable icon", - "description": "A maskable icon ensures that the image fills the entire shape without being letterboxed when installing the app on a device. [Learn more](https://web.dev/maskable-icon-audit/).", - "score": 0, - "scoreDisplayMode": "binary", - "explanation": "No manifest was fetched" - }, - "content-width": { - "id": "content-width", - "title": "Content is sized correctly for the viewport", - "description": "If the width of your app's content doesn't match the width of the viewport, your app might not be optimized for mobile screens. [Learn more](https://web.dev/content-width/).", - "score": 1, - "scoreDisplayMode": "binary" - }, - "image-aspect-ratio": { - "id": "image-aspect-ratio", - "title": "Displays images with correct aspect ratio", - "description": "Image display dimensions should match natural aspect ratio. [Learn more](https://web.dev/image-aspect-ratio/).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "image-size-responsive": { - "id": "image-size-responsive", - "title": "Serves images with appropriate resolution", - "description": "Image natural dimensions should be proportional to the display size and the pixel ratio to maximize image clarity. [Learn more](https://web.dev/serve-responsive-images/).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "preload-fonts": { - "id": "preload-fonts", - "title": "Fonts with `font-display: optional` are preloaded", - "description": "Preload `optional` fonts so first-time visitors may use them. [Learn more](https://web.dev/preload-optional-fonts/)", - "score": null, - "scoreDisplayMode": "notApplicable" - }, - "deprecations": { - "id": "deprecations", - "title": "Avoids deprecated APIs", - "description": "Deprecated APIs will eventually be removed from the browser. [Learn more](https://web.dev/deprecations/).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "mainthread-work-breakdown": { - "id": "mainthread-work-breakdown", - "title": "Minimizes main-thread work", - "description": "Consider reducing the time spent parsing, compiling and executing JS. You may find delivering smaller JS payloads helps with this. [Learn more](https://web.dev/mainthread-work-breakdown/)", - "score": 1, - "scoreDisplayMode": "numeric", - "numericValue": 746.9080000000002, - "numericUnit": "millisecond", - "displayValue": "0.7 s", - "details": { - "type": "table", - "headings": [ - { - "key": "groupLabel", - "itemType": "text", - "text": "Category" - }, - { - "key": "duration", - "itemType": "ms", - "granularity": 1, - "text": "Time Spent" - } - ], - "items": [ - { - "group": "other", - "groupLabel": "Other", - "duration": 259.4800000000003 - }, - { - "group": "scriptEvaluation", - "groupLabel": "Script Evaluation", - "duration": 230.2920000000001 - }, - { - "group": "styleLayout", - "groupLabel": "Style & Layout", - "duration": 193.21599999999995 - }, - { - "group": "paintCompositeRender", - "groupLabel": "Rendering", - "duration": 26.639999999999993 - }, - { - "group": "scriptParseCompile", - "groupLabel": "Script Parsing & Compilation", - "duration": 20.195999999999987 - }, - { - "group": "parseHTML", - "groupLabel": "Parse HTML & CSS", - "duration": 14.091999999999997 + }, + "errors-in-console": { + "id": "errors-in-console", + "title": "No browser errors logged to the console", + "description": "Errors logged to the console indicate unresolved problems. They can come from network request failures and other browser concerns. [Learn more](https://web.dev/errors-in-console/)", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "server-response-time": { + "id": "server-response-time", + "title": "Initial server response time was short", + "description": "Keep the server response time for the main document short because all other requests depend on it. [Learn more](https://web.dev/time-to-first-byte/).", + "score": 1, + "scoreDisplayMode": "binary", + "numericValue": 48.548, + "numericUnit": "millisecond", + "displayValue": "Root document took 50 ms", + "details": { + "type": "opportunity", + "headings": [ + { + "key": "url", + "valueType": "url", + "label": "URL" + }, + { + "key": "responseTime", + "valueType": "timespanMs", + "label": "Time Spent" + } + ], + "items": [ + { + "url": "https://www.mikescerealshack.co/", + "responseTime": 48.548 + } + ], + "overallSavingsMs": -51.452 + } + }, + "interactive": { + "id": "interactive", + "title": "Time to Interactive", + "description": "Time to interactive is the amount of time it takes for the page to become fully interactive. [Learn more](https://web.dev/interactive/).", + "score": 0.98, + "scoreDisplayMode": "numeric", + "numericValue": 2444.9140000000007, + "numericUnit": "millisecond", + "displayValue": "2.4 s" + }, + "user-timings": { + "id": "user-timings", + "title": "User Timing marks and measures", + "description": "Consider instrumenting your app with the User Timing API to measure your app's real-world performance during key user experiences. [Learn more](https://web.dev/user-timings/).", + "score": null, + "scoreDisplayMode": "informative", + "displayValue": "4 user timings", + "details": { + "type": "table", + "headings": [ + { + "key": "name", + "itemType": "text", + "text": "Name" + }, + { + "key": "timingType", + "itemType": "text", + "text": "Type" + }, + { + "key": "startTime", + "itemType": "ms", + "granularity": 0.01, + "text": "Start Time" + }, + { + "key": "duration", + "itemType": "ms", + "granularity": 0.01, + "text": "Duration" + } + ], + "items": [ + { + "name": "Next.js-before-hydration", + "startTime": 0, + "duration": 406.868, + "timingType": "Measure" + }, + { + "name": "Next.js-hydration", + "startTime": 406.868, + "duration": 12.852, + "timingType": "Measure" + }, + { + "name": "beforeRender", + "startTime": 406.881, + "timingType": "Mark" + }, + { + "name": "afterHydrate", + "startTime": 419.73, + "timingType": "Mark" + } + ] + } + }, + "critical-request-chains": { + "id": "critical-request-chains", + "title": "Avoid chaining critical requests", + "description": "The Critical Request Chains below show you what resources are loaded with a high priority. Consider reducing the length of chains, reducing the download size of resources, or deferring the download of unnecessary resources to improve page load. [Learn more](https://web.dev/critical-request-chains/).", + "score": null, + "scoreDisplayMode": "notApplicable", + "displayValue": "", + "details": { + "type": "criticalrequestchain", + "chains": { + "E7887194466C954A10E49E6F1832C102": { + "request": { + "url": "https://www.mikescerealshack.co/", + "startTime": 56836.325777, + "endTime": 56836.553242, + "responseReceivedTime": 56836.552084999996, + "transferSize": 3551 + } + } }, - { - "group": "garbageCollection", - "groupLabel": "Garbage Collection", - "duration": 2.9920000000000004 + "longestChain": { + "duration": 227.46500000357628, + "length": 1, + "transferSize": 3551 } - ] - } - }, - "bootup-time": { - "id": "bootup-time", - "title": "JavaScript execution time", - "description": "Consider reducing the time spent parsing, compiling, and executing JS. You may find delivering smaller JS payloads helps with this. [Learn more](https://web.dev/bootup-time/).", - "score": 1, - "scoreDisplayMode": "numeric", - "numericValue": 188.04, - "numericUnit": "millisecond", - "displayValue": "0.2 s", - "details": { - "type": "table", - "headings": [ - { - "key": "url", - "itemType": "url", - "text": "URL" - }, - { - "key": "total", - "granularity": 1, - "itemType": "ms", - "text": "Total CPU Time" - }, - { - "key": "scripting", - "granularity": 1, - "itemType": "ms", - "text": "Script Evaluation" - }, - { - "key": "scriptParseCompile", - "granularity": 1, - "itemType": "ms", - "text": "Script Parse" + } + }, + "redirects": { + "id": "redirects", + "title": "Avoid multiple page redirects", + "description": "Redirects introduce additional delays before the page can be loaded. [Learn more](https://web.dev/redirects/).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 0, + "numericUnit": "millisecond", + "displayValue": "", + "details": { + "type": "opportunity", + "headings": [], + "items": [], + "overallSavingsMs": 0 + } + }, + "installable-manifest": { + "id": "installable-manifest", + "title": "Web app manifest or service worker do not meet the installability requirements", + "description": "Service worker is the technology that enables your app to use many Progressive Web App features, such as offline, add to homescreen, and push notifications. With proper service worker and manifest implementations, browsers can proactively prompt users to add your app to their homescreen, which can lead to higher engagement. [Learn more](https://web.dev/installable-manifest/).", + "score": 0, + "scoreDisplayMode": "binary", + "numericValue": 2, + "numericUnit": "element", + "displayValue": "2 reasons", + "warnings": [], + "details": { + "type": "table", + "headings": [ + { + "key": "reason", + "itemType": "text", + "text": "Failure reason" + } + ], + "items": [ + { + "reason": "Page has no manifest URL" + }, + { + "reason": "No manifest was fetched" + } + ], + "debugData": { + "type": "debugdata", + "manifestUrl": null } - ], - "items": [ - { - "url": "https://www.mikescerealshack.co/", - "total": 296.88399999999996, - "scripting": 21.879999999999995, - "scriptParseCompile": 3.26 - }, - { - "url": "Unattributable", - "total": 213.90800000000024, - "scripting": 15.144, - "scriptParseCompile": 0.772 - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/webpack-657a8433bac0aabd564e.js", - "total": 86.21999999999998, - "scripting": 85.36799999999998, - "scriptParseCompile": 0.424 - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/pages/_app-ef508c97234d1af96c47.js", - "total": 62.076, - "scripting": 60.32000000000001, - "scriptParseCompile": 0.8720000000000001 + } + }, + "apple-touch-icon": { + "id": "apple-touch-icon", + "title": "Does not provide a valid `apple-touch-icon`", + "description": "For ideal appearance on iOS when users add a progressive web app to the home screen, define an `apple-touch-icon`. It must point to a non-transparent 192px (or 180px) square PNG. [Learn More](https://web.dev/apple-touch-icon/).", + "score": 0, + "scoreDisplayMode": "binary", + "warnings": [] + }, + "splash-screen": { + "id": "splash-screen", + "title": "Is not configured for a custom splash screen", + "description": "A themed splash screen ensures a high-quality experience when users launch your app from their homescreens. [Learn more](https://web.dev/splash-screen/).", + "score": 0, + "scoreDisplayMode": "binary", + "explanation": "Failures: No manifest was fetched.", + "details": { + "type": "debugdata", + "items": [ + { + "failures": [ + "No manifest was fetched" + ], + "isParseFailure": true, + "parseFailureReason": "No manifest was fetched" + } + ] + } + }, + "themed-omnibox": { + "id": "themed-omnibox", + "title": "Does not set a theme color for the address bar.", + "description": "The browser address bar can be themed to match your site. [Learn more](https://web.dev/themed-omnibox/).", + "score": 0, + "scoreDisplayMode": "binary", + "explanation": "Failures: No manifest was fetched,\nNo `` tag found.", + "details": { + "type": "debugdata", + "items": [ + { + "failures": [ + "No manifest was fetched", + "No `` tag found" + ], + "themeColor": null, + "isParseFailure": true, + "parseFailureReason": "No manifest was fetched" + } + ] + } + }, + "maskable-icon": { + "id": "maskable-icon", + "title": "Manifest doesn't have a maskable icon", + "description": "A maskable icon ensures that the image fills the entire shape without being letterboxed when installing the app on a device. [Learn more](https://web.dev/maskable-icon-audit/).", + "score": 0, + "scoreDisplayMode": "binary", + "explanation": "No manifest was fetched" + }, + "content-width": { + "id": "content-width", + "title": "Content is sized correctly for the viewport", + "description": "If the width of your app's content doesn't match the width of the viewport, your app might not be optimized for mobile screens. [Learn more](https://web.dev/content-width/).", + "score": 1, + "scoreDisplayMode": "binary" + }, + "image-aspect-ratio": { + "id": "image-aspect-ratio", + "title": "Displays images with correct aspect ratio", + "description": "Image display dimensions should match natural aspect ratio. [Learn more](https://web.dev/image-aspect-ratio/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "image-size-responsive": { + "id": "image-size-responsive", + "title": "Serves images with appropriate resolution", + "description": "Image natural dimensions should be proportional to the display size and the pixel ratio to maximize image clarity. [Learn more](https://web.dev/serve-responsive-images/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "preload-fonts": { + "id": "preload-fonts", + "title": "Fonts with `font-display: optional` are preloaded", + "description": "Preload `optional` fonts so first-time visitors may use them. [Learn more](https://web.dev/preload-optional-fonts/)", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "deprecations": { + "id": "deprecations", + "title": "Avoids deprecated APIs", + "description": "Deprecated APIs will eventually be removed from the browser. [Learn more](https://web.dev/deprecations/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "mainthread-work-breakdown": { + "id": "mainthread-work-breakdown", + "title": "Minimizes main-thread work", + "description": "Consider reducing the time spent parsing, compiling and executing JS. You may find delivering smaller JS payloads helps with this. [Learn more](https://web.dev/mainthread-work-breakdown/)", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 682.7280000000007, + "numericUnit": "millisecond", + "displayValue": "0.7 s", + "details": { + "type": "table", + "headings": [ + { + "key": "groupLabel", + "itemType": "text", + "text": "Category" + }, + { + "key": "duration", + "itemType": "ms", + "granularity": 1, + "text": "Time Spent" + } + ], + "items": [ + { + "group": "other", + "groupLabel": "Other", + "duration": 294.0760000000006 + }, + { + "group": "scriptEvaluation", + "groupLabel": "Script Evaluation", + "duration": 202.07200000000017 + }, + { + "group": "styleLayout", + "groupLabel": "Style & Layout", + "duration": 127.72400000000003 + }, + { + "group": "parseHTML", + "groupLabel": "Parse HTML & CSS", + "duration": 22.304 + }, + { + "group": "paintCompositeRender", + "groupLabel": "Rendering", + "duration": 19.044 + }, + { + "group": "scriptParseCompile", + "groupLabel": "Script Parsing & Compilation", + "duration": 17.508 + } + ] + } + }, + "bootup-time": { + "id": "bootup-time", + "title": "JavaScript execution time", + "description": "Consider reducing the time spent parsing, compiling, and executing JS. You may find delivering smaller JS payloads helps with this. [Learn more](https://web.dev/bootup-time/).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 189.2160000000001, + "numericUnit": "millisecond", + "displayValue": "0.2 s", + "details": { + "type": "table", + "headings": [ + { + "key": "url", + "itemType": "url", + "text": "URL" + }, + { + "key": "total", + "granularity": 1, + "itemType": "ms", + "text": "Total CPU Time" + }, + { + "key": "scripting", + "granularity": 1, + "itemType": "ms", + "text": "Script Evaluation" + }, + { + "key": "scriptParseCompile", + "granularity": 1, + "itemType": "ms", + "text": "Script Parse" + } + ], + "items": [ + { + "url": "https://www.mikescerealshack.co/", + "total": 242.848, + "scripting": 20.515999999999988, + "scriptParseCompile": 3.028 + }, + { + "url": "Unattributable", + "total": 234.4400000000001, + "scripting": 11.111999999999998, + "scriptParseCompile": 0.744 + }, + { + "url": "https://www.mikescerealshack.co/_next/static/chunks/framework.9d524150d48315f49e80.js", + "total": 161.63200000000012, + "scripting": 148.85200000000012, + "scriptParseCompile": 4.9639999999999995 + } + ], + "summary": { + "wastedMs": 189.2160000000001 } - ], - "summary": { - "wastedMs": 188.04 } - } - }, - "uses-rel-preload": { - "id": "uses-rel-preload", - "title": "Preload key requests", - "description": "Consider using `` to prioritize fetching resources that are currently requested later in page load. [Learn more](https://web.dev/uses-rel-preload/).", - "score": null, - "scoreDisplayMode": "notApplicable", - "details": { - "type": "opportunity", - "headings": [], - "items": [], - "overallSavingsMs": 0 - } - }, - "uses-rel-preconnect": { - "id": "uses-rel-preconnect", - "title": "Preconnect to required origins", - "description": "Consider adding `preconnect` or `dns-prefetch` resource hints to establish early connections to important third-party origins. [Learn more](https://web.dev/uses-rel-preconnect/).", - "score": 0.87, - "scoreDisplayMode": "numeric", - "numericValue": 161.36899999904634, - "numericUnit": "millisecond", - "displayValue": "Potential savings of 160 ms", - "warnings": [], - "details": { - "type": "opportunity", - "headings": [ - { - "key": "url", - "valueType": "url", - "label": "URL" - }, - { - "key": "wastedMs", - "valueType": "timespanMs", - "label": "Potential Savings" - } - ], - "items": [ - { - "url": "https://fonts.googleapis.com", - "wastedMs": 161.36899999904634 - } - ], - "overallSavingsMs": 161.36899999904634 - } - }, - "font-display": { - "id": "font-display", - "title": "All text remains visible during webfont loads", - "description": "Leverage the font-display CSS feature to ensure text is user-visible while webfonts are loading. [Learn more](https://web.dev/font-display/).", - "score": 1, - "scoreDisplayMode": "binary", - "warnings": [], - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "diagnostics": { - "id": "diagnostics", - "title": "Diagnostics", - "description": "Collection of useful page vitals.", - "score": null, - "scoreDisplayMode": "informative", - "details": { - "type": "debugdata", - "items": [ - { - "numRequests": 23, - "numScripts": 12, - "numStylesheets": 2, - "numFonts": 3, - "numTasks": 507, - "numTasksOver10ms": 4, - "numTasksOver25ms": 1, - "numTasksOver50ms": 0, - "numTasksOver100ms": 0, - "numTasksOver500ms": 0, - "rtt": 22.891000000000005, - "throughput": 17420811.69336929, - "maxRtt": 123.648, - "maxServerLatency": 36.634, - "totalByteWeight": 183611, - "totalTaskTime": 186.72699999999978, - "mainDocumentTransferSize": 3550 - } - ] - } - }, - "network-requests": { - "id": "network-requests", - "title": "Network Requests", - "description": "Lists the network requests that were made during page load.", - "score": null, - "scoreDisplayMode": "informative", - "details": { - "type": "table", - "headings": [ - { - "key": "url", - "itemType": "url", - "text": "URL" - }, - { - "key": "protocol", - "itemType": "text", - "text": "Protocol" - }, - { - "key": "startTime", - "itemType": "ms", - "granularity": 1, - "text": "Start Time" - }, - { - "key": "endTime", - "itemType": "ms", - "granularity": 1, - "text": "End Time" - }, - { - "key": "transferSize", - "itemType": "bytes", - "displayUnit": "kb", - "granularity": 1, - "text": "Transfer Size" - }, - { - "key": "resourceSize", - "itemType": "bytes", - "displayUnit": "kb", - "granularity": 1, - "text": "Resource Size" - }, - { - "key": "statusCode", - "itemType": "text", - "text": "Status Code" - }, - { - "key": "mimeType", - "itemType": "text", - "text": "MIME Type" - }, - { - "key": "resourceType", - "itemType": "text", - "text": "Resource Type" - } - ], - "items": [ - { - "url": "https://www.mikescerealshack.co/", - "protocol": "h2", - "startTime": 0, - "endTime": 524.7229999949923, - "finished": true, - "transferSize": 3550, - "resourceSize": 9301, - "statusCode": 200, - "mimeType": "text/html", - "resourceType": "Document" - }, - { - "url": "https://www.mikescerealshack.co/fonts/danielbd.woff2", - "protocol": "h2", - "startTime": 544.1089999949327, - "endTime": 601.4409999916097, - "finished": true, - "transferSize": 35882, - "resourceSize": 35680, - "statusCode": 200, - "mimeType": "font/woff2", - "resourceType": "Font" - }, - { - "url": "https://events.mikescerealshack.co/js/index.js", - "protocol": "http/1.1", - "startTime": 684.4459999992978, - "endTime": 1247.589999999036, - "finished": true, - "transferSize": 1090, - "resourceSize": 1341, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Script" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/css/08dcb440d7d83b488817.css", - "protocol": "h2", - "startTime": 544.752999994671, - "endTime": 649.755999998888, - "finished": true, - "transferSize": 5000, - "resourceSize": 18126, - "statusCode": 200, - "mimeType": "text/css", - "resourceType": "Stylesheet" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/main-1f8481d632114a408557.js", - "protocol": "h2", - "startTime": 545.3669999988051, - "endTime": 612.6709999953164, - "finished": true, - "transferSize": 7039, - "resourceSize": 17656, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Script" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/webpack-657a8433bac0aabd564e.js", - "protocol": "h2", - "startTime": 546.0339999990538, - "endTime": 662.2449999995297, - "finished": true, - "transferSize": 1403, - "resourceSize": 2351, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Script" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/framework.9d524150d48315f49e80.js", - "protocol": "h2", - "startTime": 546.5750000003027, - "endTime": 631.4679999923101, - "finished": true, - "transferSize": 43880, - "resourceSize": 130277, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Script" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/commons.49455e4fa8cc3f51203f.js", - "protocol": "h2", - "startTime": 547.0149999891873, - "endTime": 633.1199999985984, - "finished": true, - "transferSize": 15117, - "resourceSize": 44060, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Script" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/pages/_app-ef508c97234d1af96c47.js", - "protocol": "h2", - "startTime": 547.482999987551, - "endTime": 662.7099999896018, - "finished": true, - "transferSize": 744, - "resourceSize": 1235, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Script" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/1aeab0175d4c4d823d7a78205bceb5dd9cd36d32.a629f28ec97ae6e480bf.js", - "protocol": "h2", - "startTime": 547.7399999945192, - "endTime": 636.237999991863, - "finished": true, - "transferSize": 23958, - "resourceSize": 67673, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Script" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/pages/index-37980adf97404e76e41a.js", - "protocol": "h2", - "startTime": 548.2369999954244, - "endTime": 633.9289999887114, - "finished": true, - "transferSize": 1115, - "resourceSize": 1856, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Script" - }, - { - "url": "https://www.mikescerealshack.co/logo-text.svg", - "protocol": "h2", - "startTime": 682.0379999990109, - "endTime": 790.871999997762, - "finished": true, - "transferSize": 14329, - "resourceSize": 53947, - "statusCode": 200, - "mimeType": "image/svg+xml", - "resourceType": "Image" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/p2jwxtY4-8WEKSVkE5jXt/_buildManifest.js", - "protocol": "h2", - "startTime": 685.0290000002133, - "endTime": 796.0079999902518, - "finished": true, - "transferSize": 754, - "resourceSize": 1545, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Script" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/p2jwxtY4-8WEKSVkE5jXt/_ssgManifest.js", - "protocol": "h2", - "startTime": 685.4179999936605, - "endTime": 724.6429999941029, - "finished": true, - "transferSize": 234, - "resourceSize": 76, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Script" - }, - { - "url": "https://fonts.googleapis.com/css2?family=Poppins:wght@400;500;700&display=swap", - "protocol": "h2", - "startTime": 685.6489999918267, - "endTime": 848.1409999949392, - "finished": true, - "transferSize": 1006, - "resourceSize": 3282, - "statusCode": 200, - "mimeType": "text/css", - "resourceType": "Stylesheet" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/1aeab0175d4c4d823d7a78205bceb5dd9cd36d32.a629f28ec97ae6e480bf.js", - "protocol": "h2", - "startTime": 805.2739999984624, - "endTime": 807.712999987416, - "finished": true, - "transferSize": 0, - "resourceSize": 0, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Other" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/9ea7d3ba8dd80c65c50028121847762825088b49.dc477066508a83415fce.js", - "protocol": "h2", - "startTime": 805.7770000013988, - "endTime": 910.3459999896586, - "finished": true, - "transferSize": 3857, - "resourceSize": 0, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Other" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/pages/scenes/%5Bseason%5D/%5Bepisode%5D/%5Bscene%5D-526fe33be891a56314a3.js", - "protocol": "h2", - "startTime": 806.2019999924814, - "endTime": 839.295999991009, - "finished": true, - "transferSize": 2971, - "resourceSize": 0, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Other" - }, - { - "url": "https://fonts.gstatic.com/s/poppins/v15/pxiEyp8kv8JHgFVrJJfecnFHGPc.woff2", - "protocol": "h2", - "startTime": 958.8949999888428, - "endTime": 992.2589999914635, - "finished": true, - "transferSize": 8390, - "resourceSize": 7900, - "statusCode": 200, - "mimeType": "font/woff2", - "resourceType": "Font" - }, - { - "url": "https://fonts.gstatic.com/s/poppins/v15/pxiByp8kv8JHgFVrLCz7Z1xlFd2JQEk.woff2", - "protocol": "h2", - "startTime": 959.2489999922691, - "endTime": 999.5939999935217, - "finished": true, - "transferSize": 7947, - "resourceSize": 7832, - "statusCode": 200, - "mimeType": "font/woff2", - "resourceType": "Font" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/9ea7d3ba8dd80c65c50028121847762825088b49.dc477066508a83415fce.js", - "protocol": "h2", - "startTime": 961.658999993233, - "endTime": 963.0139999935636, - "finished": true, - "transferSize": 0, - "resourceSize": 8890, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Script" - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/pages/scenes/%5Bseason%5D/%5Bepisode%5D/%5Bscene%5D-526fe33be891a56314a3.js", - "protocol": "h2", - "startTime": 962.1859999897424, - "endTime": 963.912999999593, - "finished": true, - "transferSize": 0, - "resourceSize": 6837, - "statusCode": 200, - "mimeType": "application/javascript", - "resourceType": "Script" - }, - { - "url": "https://www.mikescerealshack.co/favicon.png", - "protocol": "h2", - "startTime": 1255.0349999946775, - "endTime": 1286.9299999874784, - "finished": true, - "transferSize": 5345, - "resourceSize": 5164, - "statusCode": 200, - "mimeType": "image/png", - "resourceType": "Other" - } - ] - } - }, - "network-rtt": { - "id": "network-rtt", - "title": "Network Round Trip Times", - "description": "Network round trip times (RTT) have a large impact on performance. If the RTT to an origin is high, it's an indication that servers closer to the user could improve performance. [Learn more](https://hpbn.co/primer-on-latency-and-bandwidth/).", - "score": null, - "scoreDisplayMode": "informative", - "numericValue": 123.648, - "numericUnit": "millisecond", - "displayValue": "120 ms", - "details": { - "type": "table", - "headings": [ - { - "key": "origin", - "itemType": "text", - "text": "URL" - }, - { - "key": "rtt", - "itemType": "ms", - "granularity": 1, - "text": "Time Spent" - } - ], - "items": [ - { - "origin": "https://events.mikescerealshack.co", - "rtt": 123.648 - }, - { - "origin": "https://www.mikescerealshack.co", - "rtt": 26.504999999999995 - }, - { - "origin": "https://fonts.googleapis.com", - "rtt": 22.891000000000005 - }, - { - "origin": "https://fonts.gstatic.com", - "rtt": 22.891000000000005 - } - ] - } - }, - "network-server-latency": { - "id": "network-server-latency", - "title": "Server Backend Latencies", - "description": "Server latencies can impact web performance. If the server latency of an origin is high, it's an indication the server is overloaded or has poor backend performance. [Learn more](https://hpbn.co/primer-on-web-performance/#analyzing-the-resource-waterfall).", - "score": null, - "scoreDisplayMode": "informative", - "numericValue": 36.634, - "numericUnit": "millisecond", - "displayValue": "40 ms", - "details": { - "type": "table", - "headings": [ - { - "key": "origin", - "itemType": "text", - "text": "URL" - }, - { - "key": "serverResponseTime", - "itemType": "ms", - "granularity": 1, - "text": "Time Spent" - } - ], - "items": [ - { - "origin": "https://www.mikescerealshack.co", - "serverResponseTime": 36.634 - }, - { - "origin": "https://fonts.googleapis.com", - "serverResponseTime": 12.669999999999987 - }, - { - "origin": "https://events.mikescerealshack.co", - "serverResponseTime": 9.651999999999958 - }, - { - "origin": "https://fonts.gstatic.com", - "serverResponseTime": 1.8199999999999967 - } - ] - } - }, - "main-thread-tasks": { - "id": "main-thread-tasks", - "title": "Tasks", - "description": "Lists the toplevel main thread tasks that executed during page load.", - "score": null, - "scoreDisplayMode": "informative", - "details": { - "type": "table", - "headings": [ - { - "key": "startTime", - "itemType": "ms", - "granularity": 1, - "text": "Start Time" - }, - { - "key": "duration", - "itemType": "ms", - "granularity": 1, - "text": "End Time" - } - ], - "items": [ - { - "duration": 9.507, - "startTime": 968.491 - }, - { - "duration": 6.779, - "startTime": 986.355 - }, - { - "duration": 10.258, - "startTime": 998.71 - }, - { - "duration": 33.281, - "startTime": 1097.202 - }, - { - "duration": 21.555, - "startTime": 1133.379 - }, - { - "duration": 15.519, - "startTime": 1155.003 - }, - { - "duration": 6.533, - "startTime": 1236.791 - }, - { - "duration": 6.091, - "startTime": 1397.301 - } - ] - } - }, - "metrics": { - "id": "metrics", - "title": "Metrics", - "description": "Collects all available metrics.", - "score": null, - "scoreDisplayMode": "informative", - "numericValue": 2028, - "numericUnit": "millisecond", - "details": { - "type": "debugdata", - "items": [ - { - "firstContentfulPaint": 1412, - "firstMeaningfulPaint": 1412, - "largestContentfulPaint": 2334, - "interactive": 2028, - "speedIndex": 1679, - "totalBlockingTime": 42, - "maxPotentialFID": 86, - "cumulativeLayoutShift": 0.0026205596923828123, - "cumulativeLayoutShiftMainFrame": 0.0026205596923828123, - "totalCumulativeLayoutShift": 0.0026205596923828123, - "observedTimeOrigin": 0, - "observedTimeOriginTs": 121804765196, - "observedNavigationStart": 0, - "observedNavigationStartTs": 121804765196, - "observedFirstPaint": 689, - "observedFirstPaintTs": 121805454453, - "observedFirstContentfulPaint": 689, - "observedFirstContentfulPaintTs": 121805454454, - "observedFirstContentfulPaintAllFrames": 689, - "observedFirstContentfulPaintAllFramesTs": 121805454454, - "observedFirstMeaningfulPaint": 689, - "observedFirstMeaningfulPaintTs": 121805454454, - "observedLargestContentfulPaint": 956, - "observedLargestContentfulPaintTs": 121805720788, - "observedLargestContentfulPaintAllFrames": 956, - "observedLargestContentfulPaintAllFramesTs": 121805720788, - "observedTraceEnd": 3829, - "observedTraceEndTs": 121808593777, - "observedLoad": 1256, - "observedLoadTs": 121806020944, - "observedDomContentLoaded": 558, - "observedDomContentLoadedTs": 121805322862, - "observedCumulativeLayoutShift": 0.0026205596923828123, - "observedCumulativeLayoutShiftMainFrame": 0.0026205596923828123, - "observedTotalCumulativeLayoutShift": 0.0026205596923828123, - "observedFirstVisualChange": 712, - "observedFirstVisualChangeTs": 121805477196, - "observedLastVisualChange": 1029, - "observedLastVisualChangeTs": 121805794196, - "observedSpeedIndex": 723, - "observedSpeedIndexTs": 121805487714 - }, - { - "lcpInvalidated": false - } - ] - } - }, - "performance-budget": { - "id": "performance-budget", - "title": "Performance budget", - "description": "Keep the quantity and size of network requests under the targets set by the provided performance budget. [Learn more](https://developers.google.com/web/tools/lighthouse/audits/budgets).", - "score": null, - "scoreDisplayMode": "notApplicable" - }, - "timing-budget": { - "id": "timing-budget", - "title": "Timing budget", - "description": "Set a timing budget to help you keep an eye on the performance of your site. Performant sites load fast and respond to user input events quickly. [Learn more](https://developers.google.com/web/tools/lighthouse/audits/budgets).", - "score": null, - "scoreDisplayMode": "notApplicable" - }, - "resource-summary": { - "id": "resource-summary", - "title": "Keep request counts low and transfer sizes small", - "description": "To set budgets for the quantity and size of page resources, add a budget.json file. [Learn more](https://web.dev/use-lighthouse-for-performance-budgets/).", - "score": null, - "scoreDisplayMode": "informative", - "displayValue": "23 requests • 179 KiB", - "details": { - "type": "table", - "headings": [ - { - "key": "label", - "itemType": "text", - "text": "Resource Type" - }, - { - "key": "requestCount", - "itemType": "numeric", - "text": "Requests" - }, - { - "key": "transferSize", - "itemType": "bytes", - "text": "Transfer Size" - } - ], - "items": [ - { - "resourceType": "total", - "label": "Total", - "requestCount": 23, - "transferSize": 183611 - }, - { - "resourceType": "script", - "label": "Script", - "requestCount": 12, - "transferSize": 95334 - }, - { - "resourceType": "font", - "label": "Font", - "requestCount": 3, - "transferSize": 52219 - }, - { - "resourceType": "image", - "label": "Image", - "requestCount": 1, - "transferSize": 14329 - }, - { - "resourceType": "other", - "label": "Other", - "requestCount": 4, - "transferSize": 12173 - }, - { - "resourceType": "stylesheet", - "label": "Stylesheet", - "requestCount": 2, - "transferSize": 6006 - }, - { - "resourceType": "document", - "label": "Document", - "requestCount": 1, - "transferSize": 3550 - }, - { - "resourceType": "media", - "label": "Media", - "requestCount": 0, - "transferSize": 0 - }, - { - "resourceType": "third-party", - "label": "Third-party", - "requestCount": 3, - "transferSize": 17343 - } - ] - } - }, - "third-party-summary": { - "id": "third-party-summary", - "title": "Minimize third-party usage", - "description": "Third-party code can significantly impact load performance. Limit the number of redundant third-party providers and try to load third-party code after your page has primarily finished loading. [Learn more](https://developers.google.com/web/fundamentals/performance/optimizing-content-efficiency/loading-third-party-javascript/).", - "score": 1, - "scoreDisplayMode": "binary", - "displayValue": "Third-party code blocked the main thread for 0 ms", - "details": { - "type": "table", - "headings": [ - { - "key": "entity", - "itemType": "link", - "text": "Third-Party", - "subItemsHeading": { + }, + "uses-rel-preload": { + "id": "uses-rel-preload", + "title": "Preload key requests", + "description": "Consider using `` to prioritize fetching resources that are currently requested later in page load. [Learn more](https://web.dev/uses-rel-preload/).", + "score": null, + "scoreDisplayMode": "notApplicable", + "details": { + "type": "opportunity", + "headings": [], + "items": [], + "overallSavingsMs": 0 + } + }, + "uses-rel-preconnect": { + "id": "uses-rel-preconnect", + "title": "Preconnect to required origins", + "description": "Consider adding `preconnect` or `dns-prefetch` resource hints to establish early connections to important third-party origins. [Learn more](https://web.dev/uses-rel-preconnect/).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 0, + "numericUnit": "millisecond", + "displayValue": "", + "warnings": [], + "details": { + "type": "opportunity", + "headings": [], + "items": [], + "overallSavingsMs": 0 + } + }, + "font-display": { + "id": "font-display", + "title": "All text remains visible during webfont loads", + "description": "Leverage the font-display CSS feature to ensure text is user-visible while webfonts are loading. [Learn more](https://web.dev/font-display/).", + "score": 1, + "scoreDisplayMode": "binary", + "warnings": [], + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "diagnostics": { + "id": "diagnostics", + "title": "Diagnostics", + "description": "Collection of useful page vitals.", + "score": null, + "scoreDisplayMode": "informative", + "details": { + "type": "debugdata", + "items": [ + { + "numRequests": 23, + "numScripts": 12, + "numStylesheets": 2, + "numFonts": 3, + "numTasks": 410, + "numTasksOver10ms": 4, + "numTasksOver25ms": 2, + "numTasksOver50ms": 0, + "numTasksOver100ms": 0, + "numTasksOver500ms": 0, + "rtt": 8.997, + "throughput": 12934489.892703472, + "maxRtt": 124.113, + "maxServerLatency": 35.56999999999999, + "totalByteWeight": 183597, + "totalTaskTime": 170.6819999999999, + "mainDocumentTransferSize": 3551 + } + ] + } + }, + "network-requests": { + "id": "network-requests", + "title": "Network Requests", + "description": "Lists the network requests that were made during page load.", + "score": null, + "scoreDisplayMode": "informative", + "details": { + "type": "table", + "headings": [ + { "key": "url", - "itemType": "url" + "itemType": "url", + "text": "URL" + }, + { + "key": "protocol", + "itemType": "text", + "text": "Protocol" + }, + { + "key": "startTime", + "itemType": "ms", + "granularity": 1, + "text": "Start Time" + }, + { + "key": "endTime", + "itemType": "ms", + "granularity": 1, + "text": "End Time" + }, + { + "key": "transferSize", + "itemType": "bytes", + "displayUnit": "kb", + "granularity": 1, + "text": "Transfer Size" + }, + { + "key": "resourceSize", + "itemType": "bytes", + "displayUnit": "kb", + "granularity": 1, + "text": "Resource Size" + }, + { + "key": "statusCode", + "itemType": "text", + "text": "Status Code" + }, + { + "key": "mimeType", + "itemType": "text", + "text": "MIME Type" + }, + { + "key": "resourceType", + "itemType": "text", + "text": "Resource Type" } - }, - { - "key": "transferSize", - "granularity": 1, - "itemType": "bytes", - "text": "Transfer Size", - "subItemsHeading": { - "key": "transferSize" + ], + "items": [ + { + "url": "https://www.mikescerealshack.co/", + "protocol": "h2", + "startTime": 0, + "endTime": 227.46500000357628, + "finished": true, + "transferSize": 3551, + "resourceSize": 9301, + "statusCode": 200, + "mimeType": "text/html", + "resourceType": "Document" + }, + { + "url": "https://www.mikescerealshack.co/fonts/danielbd.woff2", + "protocol": "h2", + "startTime": 255.6040000054054, + "endTime": 328.5600000017439, + "finished": true, + "transferSize": 35885, + "resourceSize": 35680, + "statusCode": 200, + "mimeType": "font/woff2", + "resourceType": "Font" + }, + { + "url": "https://events.mikescerealshack.co/js/index.js", + "protocol": "http/1.1", + "startTime": 256.0860000012326, + "endTime": 705.2900000053342, + "finished": true, + "transferSize": 1090, + "resourceSize": 1341, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/css/08dcb440d7d83b488817.css", + "protocol": "h2", + "startTime": 257.4700000041048, + "endTime": 321.94900000467896, + "finished": true, + "transferSize": 5001, + "resourceSize": 18126, + "statusCode": 200, + "mimeType": "text/css", + "resourceType": "Stylesheet" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/chunks/main-1f8481d632114a408557.js", + "protocol": "h2", + "startTime": 258.07000000349944, + "endTime": 324.560000000929, + "finished": true, + "transferSize": 6987, + "resourceSize": 17656, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/chunks/webpack-657a8433bac0aabd564e.js", + "protocol": "h2", + "startTime": 258.4790000037174, + "endTime": 322.65600000391714, + "finished": true, + "transferSize": 1403, + "resourceSize": 2351, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/chunks/framework.9d524150d48315f49e80.js", + "protocol": "h2", + "startTime": 258.88700000359677, + "endTime": 384.0909999998985, + "finished": true, + "transferSize": 43880, + "resourceSize": 130277, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/chunks/commons.49455e4fa8cc3f51203f.js", + "protocol": "h2", + "startTime": 259.4440000029863, + "endTime": 343.19200000027195, + "finished": true, + "transferSize": 15118, + "resourceSize": 44060, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/chunks/pages/_app-ef508c97234d1af96c47.js", + "protocol": "h2", + "startTime": 259.847000001173, + "endTime": 344.1900000034366, + "finished": true, + "transferSize": 743, + "resourceSize": 1235, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/chunks/1aeab0175d4c4d823d7a78205bceb5dd9cd36d32.a629f28ec97ae6e480bf.js", + "protocol": "h2", + "startTime": 260.30500000342727, + "endTime": 329.2860000001383, + "finished": true, + "transferSize": 23958, + "resourceSize": 67673, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/chunks/pages/index-37980adf97404e76e41a.js", + "protocol": "h2", + "startTime": 260.71500000398373, + "endTime": 343.6660000006668, + "finished": true, + "transferSize": 1115, + "resourceSize": 1856, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "https://www.mikescerealshack.co/logo-text.svg", + "protocol": "h2", + "startTime": 263.68200000433717, + "endTime": 471.6820000030566, + "finished": true, + "transferSize": 14333, + "resourceSize": 53947, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/p2jwxtY4-8WEKSVkE5jXt/_buildManifest.js", + "protocol": "h2", + "startTime": 264.2039999991539, + "endTime": 344.6800000019721, + "finished": true, + "transferSize": 754, + "resourceSize": 1545, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/p2jwxtY4-8WEKSVkE5jXt/_ssgManifest.js", + "protocol": "h2", + "startTime": 264.6080000049551, + "endTime": 319.1399999996065, + "finished": true, + "transferSize": 286, + "resourceSize": 76, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "https://fonts.googleapis.com/css2?family=Poppins:wght@400;500;700&display=swap", + "protocol": "h2", + "startTime": 265.0290000019595, + "endTime": 459.92800000385614, + "finished": true, + "transferSize": 1006, + "resourceSize": 3282, + "statusCode": 200, + "mimeType": "text/css", + "resourceType": "Stylesheet" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/chunks/1aeab0175d4c4d823d7a78205bceb5dd9cd36d32.a629f28ec97ae6e480bf.js", + "protocol": "h2", + "startTime": 425.53200000111246, + "endTime": 428.30099999991944, + "finished": true, + "transferSize": 0, + "resourceSize": 0, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Other" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/chunks/9ea7d3ba8dd80c65c50028121847762825088b49.dc477066508a83415fce.js", + "protocol": "h2", + "startTime": 425.9989999991376, + "endTime": 465.41300000535557, + "finished": true, + "transferSize": 3856, + "resourceSize": 0, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Other" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/chunks/pages/scenes/%5Bseason%5D/%5Bepisode%5D/%5Bscene%5D-526fe33be891a56314a3.js", + "protocol": "h2", + "startTime": 426.7210000034538, + "endTime": 466.01799999916693, + "finished": true, + "transferSize": 2970, + "resourceSize": 0, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Other" + }, + { + "url": "https://fonts.gstatic.com/s/poppins/v15/pxiEyp8kv8JHgFVrJJfecnFHGPc.woff2", + "protocol": "h2", + "startTime": 476.93900000012945, + "endTime": 566.8710000027204, + "finished": true, + "transferSize": 8391, + "resourceSize": 7900, + "statusCode": 200, + "mimeType": "font/woff2", + "resourceType": "Font" + }, + { + "url": "https://fonts.gstatic.com/s/poppins/v15/pxiByp8kv8JHgFVrLCz7Z1xlFd2JQEk.woff2", + "protocol": "h2", + "startTime": 477.26500000135275, + "endTime": 577.7100000050268, + "finished": true, + "transferSize": 7947, + "resourceSize": 7832, + "statusCode": 200, + "mimeType": "font/woff2", + "resourceType": "Font" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/chunks/9ea7d3ba8dd80c65c50028121847762825088b49.dc477066508a83415fce.js", + "protocol": "h2", + "startTime": 486.5330000029644, + "endTime": 487.7889999988838, + "finished": true, + "transferSize": 0, + "resourceSize": 8890, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "https://www.mikescerealshack.co/_next/static/chunks/pages/scenes/%5Bseason%5D/%5Bepisode%5D/%5Bscene%5D-526fe33be891a56314a3.js", + "protocol": "h2", + "startTime": 486.7930000036722, + "endTime": 488.29400000249734, + "finished": true, + "transferSize": 0, + "resourceSize": 6837, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "https://www.mikescerealshack.co/favicon.png", + "protocol": "h2", + "startTime": 712.082999998529, + "endTime": 747.9940000048373, + "finished": true, + "transferSize": 5323, + "resourceSize": 5164, + "statusCode": 200, + "mimeType": "image/png", + "resourceType": "Other" } - }, - { - "key": "blockingTime", - "granularity": 1, - "itemType": "ms", - "text": "Main-Thread Blocking Time", - "subItemsHeading": { - "key": "blockingTime" + ] + } + }, + "network-rtt": { + "id": "network-rtt", + "title": "Network Round Trip Times", + "description": "Network round trip times (RTT) have a large impact on performance. If the RTT to an origin is high, it's an indication that servers closer to the user could improve performance. [Learn more](https://hpbn.co/primer-on-latency-and-bandwidth/).", + "score": null, + "scoreDisplayMode": "informative", + "numericValue": 124.113, + "numericUnit": "millisecond", + "displayValue": "120 ms", + "details": { + "type": "table", + "headings": [ + { + "key": "origin", + "itemType": "text", + "text": "URL" + }, + { + "key": "rtt", + "itemType": "ms", + "granularity": 1, + "text": "Time Spent" } - } - ], - "items": [ - { - "mainThreadTime": 3.556, - "blockingTime": 0, - "transferSize": 17343, - "entity": { - "type": "link", - "text": "Google Fonts", - "url": "https://fonts.google.com/" - }, - "subItems": { - "type": "subitems", - "items": [ - { - "url": "https://fonts.gstatic.com/s/poppins/v15/pxiEyp8kv8JHgFVrJJfecnFHGPc.woff2", - "mainThreadTime": 0, - "blockingTime": 0, - "transferSize": 8390 - }, - { - "url": "https://fonts.gstatic.com/s/poppins/v15/pxiByp8kv8JHgFVrLCz7Z1xlFd2JQEk.woff2", - "mainThreadTime": 0, - "blockingTime": 0, - "transferSize": 7947 - } - ] + ], + "items": [ + { + "origin": "https://events.mikescerealshack.co", + "rtt": 124.113 + }, + { + "origin": "https://fonts.googleapis.com", + "rtt": 48.958000000000006 + }, + { + "origin": "https://www.mikescerealshack.co", + "rtt": 23.60900000000001 + }, + { + "origin": "https://fonts.gstatic.com", + "rtt": 8.997 } - } - ], - "summary": { - "wastedBytes": 17343, - "wastedMs": 0 + ] } - } - }, - "third-party-facades": { - "id": "third-party-facades", - "title": "Lazy load third-party resources with facades", - "description": "Some third-party embeds can be lazy loaded. Consider replacing them with a facade until they are required. [Learn more](https://web.dev/third-party-facades/).", - "score": null, - "scoreDisplayMode": "notApplicable" - }, - "largest-contentful-paint-element": { - "id": "largest-contentful-paint-element", - "title": "Largest Contentful Paint element", - "description": "This is the largest contentful element painted within the viewport. [Learn More](https://web.dev/lighthouse-largest-contentful-paint/)", - "score": null, - "scoreDisplayMode": "informative", - "displayValue": "1 element found", - "details": { - "type": "table", - "headings": [ - { - "key": "node", - "itemType": "node", - "text": "Element" - } - ], - "items": [ - { - "node": { - "type": "node", - "lhId": "page-1-IMG", - "path": "1,HTML,1,BODY,0,DIV,0,DIV,4,IMG", - "selector": "body > div#__next > div.flex > img.h-32", - "boundingRect": { - "top": 192, - "bottom": 320, - "left": 38, - "right": 322, - "width": 284, - "height": 128 - }, - "snippet": "\"Mike's", - "nodeLabel": "Mike's Cereal Shack Logo" + }, + "network-server-latency": { + "id": "network-server-latency", + "title": "Server Backend Latencies", + "description": "Server latencies can impact web performance. If the server latency of an origin is high, it's an indication the server is overloaded or has poor backend performance. [Learn more](https://hpbn.co/primer-on-web-performance/#analyzing-the-resource-waterfall).", + "score": null, + "scoreDisplayMode": "informative", + "numericValue": 35.56999999999999, + "numericUnit": "millisecond", + "displayValue": "40 ms", + "details": { + "type": "table", + "headings": [ + { + "key": "origin", + "itemType": "text", + "text": "URL" + }, + { + "key": "serverResponseTime", + "itemType": "ms", + "granularity": 1, + "text": "Time Spent" } - } - ] - } - }, - "layout-shift-elements": { - "id": "layout-shift-elements", - "title": "Avoid large layout shifts", - "description": "These DOM elements contribute most to the CLS of the page.", - "score": null, - "scoreDisplayMode": "informative", - "displayValue": "5 elements found", - "details": { - "type": "table", - "headings": [ - { - "key": "node", - "itemType": "node", - "text": "Element" - }, - { - "key": "score", - "itemType": "numeric", - "granularity": 0.001, - "text": "CLS Contribution" - } - ], - "items": [ - { - "node": { - "type": "node", - "lhId": "page-2-DIV", - "path": "1,HTML,1,BODY,0,DIV,0,DIV,5,FORM,2,DIV", - "selector": "div#__next > div.flex > form.flex > div.font-poppins", - "boundingRect": { - "top": 482, - "bottom": 526, - "left": 58, - "right": 302, - "width": 244, - "height": 44 - }, - "snippet": "
", - "nodeLabel": "SEARCHRANDOM" - }, - "score": 0.0009176013980022934 - }, - { - "node": { - "type": "node", - "lhId": "page-3-DIV", - "path": "1,HTML,1,BODY,0,DIV,0,DIV,5,FORM,0,DIV", - "selector": "div#__next > div.flex > form.flex > div.font-poppins", - "boundingRect": { - "top": 352, - "bottom": 376, - "left": 56, - "right": 304, - "width": 248, - "height": 24 - }, - "snippet": "
", - "nodeLabel": "search any line from The Office" - }, - "score": 0.0005271573688018226 - }, - { - "node": { - "type": "node", - "lhId": "page-3-DIV", - "path": "1,HTML,1,BODY,0,DIV,0,DIV,5,FORM,0,DIV", - "selector": "div#__next > div.flex > form.flex > div.font-poppins", - "boundingRect": { - "top": 352, - "bottom": 376, - "left": 56, - "right": 304, - "width": 248, - "height": 24 - }, - "snippet": "
", - "nodeLabel": "search any line from The Office" - }, - "score": 0.0004925652526072917 - }, - { - "node": { - "type": "node", - "lhId": "page-4-DIV", - "path": "1,HTML,1,BODY,0,DIV,0,DIV,8,FOOTER,0,DIV,0,DIV", - "selector": "div.flex > footer.w-full > div.w-full > div", - "boundingRect": { - "top": 588, - "bottom": 600, - "left": 8, - "right": 352, - "width": 344, - "height": 12 - }, - "snippet": "
", - "nodeLabel": "Code © 2020-2021 Mike's Cereal Shack Authors" - }, - "score": 0.0003416178364857023 - }, - { - "node": { - "type": "node", - "lhId": "page-5-DIV", - "path": "1,HTML,1,BODY,0,DIV,0,DIV,8,FOOTER,0,DIV,1,DIV", - "selector": "div.flex > footer.w-full > div.w-full > div", - "boundingRect": { - "top": 600, - "bottom": 612, - "left": 8, - "right": 352, - "width": 344, - "height": 12 - }, - "snippet": "
", - "nodeLabel": "Content © 2005-2013 NBCUniversal Media, LLC" + ], + "items": [ + { + "origin": "https://www.mikescerealshack.co", + "serverResponseTime": 35.56999999999999 }, - "score": 0.0003416178364857023 - } - ] - } - }, - "long-tasks": { - "id": "long-tasks", - "title": "Avoid long main-thread tasks", - "description": "Lists the longest tasks on the main thread, useful for identifying worst contributors to input delay. [Learn more](https://web.dev/long-tasks-devtools/)", - "score": null, - "scoreDisplayMode": "informative", - "displayValue": "3 long tasks found", - "details": { - "type": "table", - "headings": [ - { - "key": "url", - "itemType": "url", - "text": "URL" - }, - { - "key": "startTime", - "itemType": "ms", - "granularity": 1, - "text": "Start Time" - }, - { - "key": "duration", - "itemType": "ms", - "granularity": 1, - "text": "Duration" - } - ], - "items": [ - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/webpack-657a8433bac0aabd564e.js", - "duration": 86, - "startTime": 1565.546 - }, - { - "url": "https://www.mikescerealshack.co/", - "duration": 67, - "startTime": 851.4760000000001 - }, - { - "url": "https://www.mikescerealshack.co/_next/static/chunks/pages/_app-ef508c97234d1af96c47.js", - "duration": 62, - "startTime": 2341.616 - } - ] - } - }, - "no-unload-listeners": { - "id": "no-unload-listeners", - "title": "Avoids `unload` event listeners", - "description": "The `unload` event does not fire reliably and listening for it can prevent browser optimizations like the Back-Forward Cache. Consider using the `pagehide` or `visibilitychange` events instead. [Learn more](https://developers.google.com/web/updates/2018/07/page-lifecycle-api#the-unload-event)", - "score": 1, - "scoreDisplayMode": "binary" - }, - "non-composited-animations": { - "id": "non-composited-animations", - "title": "Avoid non-composited animations", - "description": "Animations which are not composited can be janky and increase CLS. [Learn more](https://web.dev/non-composited-animations)", - "score": null, - "scoreDisplayMode": "notApplicable", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "unsized-images": { - "id": "unsized-images", - "title": "Image elements do not have explicit `width` and `height`", - "description": "Set an explicit width and height on image elements to reduce layout shifts and improve CLS. [Learn more](https://web.dev/optimize-cls/#images-without-dimensions)", - "score": 0, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [ - { - "key": "url", - "itemType": "thumbnail", - "text": "" - }, - { - "key": "url", - "itemType": "url", - "text": "URL" - }, - { - "key": "node", - "itemType": "node", - "text": "Failing Elements" - } - ], - "items": [ - { - "url": "https://www.mikescerealshack.co/logo-text.svg", - "node": { - "type": "node", - "lhId": "page-1-IMG", - "path": "1,HTML,1,BODY,0,DIV,0,DIV,4,IMG", - "selector": "body > div#__next > div.flex > img.h-32", - "boundingRect": { - "top": 192, - "bottom": 320, - "left": 38, - "right": 322, - "width": 284, - "height": 128 - }, - "snippet": "\"Mike's", - "nodeLabel": "Mike's Cereal Shack Logo" - } - } - ] - } - }, - "valid-source-maps": { - "id": "valid-source-maps", - "title": "Page has valid source maps", - "description": "Source maps translate minified code to the original source code. This helps developers debug in production. In addition, Lighthouse is able to provide further insights. Consider deploying source maps to take advantage of these benefits. [Learn more](https://developers.google.com/web/tools/chrome-devtools/javascript/source-maps).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "preload-lcp-image": { - "id": "preload-lcp-image", - "title": "Preload Largest Contentful Paint image", - "description": "Preload the image used by the LCP element in order to improve your LCP time. [Learn more](https://web.dev/optimize-lcp/#preload-important-resources).", - "score": null, - "scoreDisplayMode": "notApplicable", - "details": { - "type": "opportunity", - "headings": [], - "items": [], - "overallSavingsMs": 0 - } - }, - "csp-xss": { - "id": "csp-xss", - "title": "Ensure CSP is effective against XSS attacks", - "description": "A strong Content Security Policy (CSP) significantly reduces the risk of cross-site scripting (XSS) attacks. [Learn more](https://web.dev/csp-xss/)", - "score": null, - "scoreDisplayMode": "informative", - "details": { - "type": "table", - "headings": [ - { - "key": "description", - "itemType": "text", - "subItemsHeading": { - "key": "description" - }, - "text": "Description" - }, - { - "key": "directive", - "itemType": "code", - "subItemsHeading": { - "key": "directive" + { + "origin": "https://fonts.gstatic.com", + "serverResponseTime": 20.057000000000002 }, - "text": "Directive" - }, - { - "key": "severity", - "itemType": "text", - "subItemsHeading": { - "key": "severity" + { + "origin": "https://events.mikescerealshack.co", + "serverResponseTime": 0 }, - "text": "Severity" - } - ], - "items": [ - { - "severity": "High", - "description": "No CSP found in enforcement mode" - } - ] - } - }, - "full-page-screenshot": { - "id": "full-page-screenshot", - "title": "Full-page screenshot", - "description": "A full-height screenshot of the final rendered page", - "score": null, - "scoreDisplayMode": "informative", - "details": { - "type": "full-page-screenshot", - "screenshot": { - "data": "", - "width": 360, - "height": 640 - }, - "nodes": { - "page-0-IMG": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "page-1-IMG": { - "top": 192, - "bottom": 320, - "left": 38, - "right": 322, - "width": 284, - "height": 128 - }, - "page-2-DIV": { - "top": 482, - "bottom": 526, - "left": 58, - "right": 302, - "width": 244, - "height": 44 - }, - "page-3-DIV": { - "top": 352, - "bottom": 376, - "left": 56, - "right": 304, - "width": 248, - "height": 24 - }, - "page-4-DIV": { - "top": 588, - "bottom": 600, - "left": 8, - "right": 352, - "width": 344, - "height": 12 - }, - "page-5-DIV": { - "top": 600, - "bottom": 612, - "left": 8, - "right": 352, - "width": 344, - "height": 12 - }, - "9-0-A": { - "top": 192, - "bottom": 215, - "left": 24, - "right": 121, - "width": 97, - "height": 23 - }, - "9-1-A": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-2-A": { - "top": 128, - "bottom": 151, - "left": 24, - "right": 82, - "width": 58, - "height": 23 - }, - "9-3-A": { - "top": 160, - "bottom": 183, - "left": 24, - "right": 148, - "width": 124, - "height": 23 - }, - "9-4-A": { - "top": 224, - "bottom": 247, - "left": 24, - "right": 134, - "width": 110, - "height": 23 - }, - "9-5-A": { - "top": 256, - "bottom": 279, - "left": 24, - "right": 114, - "width": 90, - "height": 23 - }, - "9-6-A": { - "top": 288, - "bottom": 311, - "left": 24, - "right": 160, - "width": 136, - "height": 23 - }, - "9-7-A": { - "top": 320, - "bottom": 343, - "left": 24, - "right": 139, - "width": 115, - "height": 23 - }, - "9-8-A": { - "top": 482, - "bottom": 526, - "left": 183, - "right": 302, - "width": 119, - "height": 44 - }, - "9-9-A": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-10-A": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-11-path": { - "top": 20, - "bottom": 44, - "left": 315, - "right": 336, - "width": 21, - "height": 24 - }, - "9-12-BODY": { - "top": 0, - "bottom": 640, - "left": 0, - "right": 360, - "width": 360, - "height": 640 - }, - "9-13-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-14-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-15-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-16-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-17-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-18-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-19-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-20-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-21-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-22-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-23-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-24-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-25-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-26-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-27-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-28-LINK": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-29-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-30-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-31-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-32-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-33-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-34-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-35-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-36-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-37-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-38-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-39-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-40-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-41-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-42-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-43-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-44-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-45-META": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-46-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-47-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-48-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-49-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-50-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-51-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-52-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-53-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-54-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-55-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-56-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-57-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-58-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-59-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-60-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-61-SCRIPT": { - "top": 0, - "bottom": 0, - "left": 0, - "right": 0, - "width": 0, - "height": 0 - }, - "9-62-INPUT": { - "top": 392, - "bottom": 450, - "left": 20, - "right": 340, - "width": 320, - "height": 58 - }, - "9-63-BUTTON": { - "top": 482, - "bottom": 526, - "left": 58, - "right": 167, - "width": 109, - "height": 44 - } - } - } - }, - "script-treemap-data": { - "id": "script-treemap-data", - "title": "Script Treemap Data", - "description": "Used for treemap app", - "score": null, - "scoreDisplayMode": "informative", - "details": { - "type": "treemap-data", - "nodes": [ - { - "name": "https://www.mikescerealshack.co/", - "resourceBytes": 582 - }, - { - "name": "https://events.mikescerealshack.co/js/index.js", - "resourceBytes": 1341, - "unusedBytes": 193 - }, - { - "name": "https://www.mikescerealshack.co/_next/static/chunks/polyfills-e893e522365b6e0e5fa4.js", - "resourceBytes": 85 - }, - { - "name": "https://www.mikescerealshack.co/_next/static/chunks/main-1f8481d632114a408557.js", - "resourceBytes": 17656, - "unusedBytes": 4493 - }, - { - "name": "https://www.mikescerealshack.co/_next/static/chunks/webpack-657a8433bac0aabd564e.js", - "resourceBytes": 2351, - "unusedBytes": 1109 - }, - { - "name": "https://www.mikescerealshack.co/_next/static/chunks/framework.9d524150d48315f49e80.js", - "resourceBytes": 130277, - "unusedBytes": 47403 - }, - { - "name": "https://www.mikescerealshack.co/_next/static/chunks/commons.49455e4fa8cc3f51203f.js", - "resourceBytes": 44060, - "unusedBytes": 20326 - }, - { - "name": "https://www.mikescerealshack.co/_next/static/chunks/pages/_app-ef508c97234d1af96c47.js", - "resourceBytes": 1235, - "unusedBytes": 360 - }, - { - "name": "https://www.mikescerealshack.co/_next/static/chunks/1aeab0175d4c4d823d7a78205bceb5dd9cd36d32.a629f28ec97ae6e480bf.js", - "resourceBytes": 67673, - "unusedBytes": 31755 - }, - { - "name": "https://www.mikescerealshack.co/_next/static/chunks/pages/index-37980adf97404e76e41a.js", - "resourceBytes": 1856, - "unusedBytes": 109 - }, - { - "name": "https://www.mikescerealshack.co/_next/static/p2jwxtY4-8WEKSVkE5jXt/_buildManifest.js", - "resourceBytes": 1545, - "unusedBytes": 0 - }, - { - "name": "https://www.mikescerealshack.co/_next/static/p2jwxtY4-8WEKSVkE5jXt/_ssgManifest.js", - "resourceBytes": 76, - "unusedBytes": 0 - }, - { - "name": "https://www.mikescerealshack.co/_next/static/chunks/9ea7d3ba8dd80c65c50028121847762825088b49.dc477066508a83415fce.js", - "resourceBytes": 8890, - "unusedBytes": 7978 - }, - { - "name": "https://www.mikescerealshack.co/_next/static/chunks/pages/scenes/%5Bseason%5D/%5Bepisode%5D/%5Bscene%5D-526fe33be891a56314a3.js", - "resourceBytes": 6837, - "unusedBytes": 6340 - } - ] - } - }, - "pwa-cross-browser": { - "id": "pwa-cross-browser", - "title": "Site works cross-browser", - "description": "To reach the most number of users, sites should work across every major browser. [Learn more](https://web.dev/pwa-cross-browser/).", - "score": null, - "scoreDisplayMode": "manual" - }, - "pwa-page-transitions": { - "id": "pwa-page-transitions", - "title": "Page transitions don't feel like they block on the network", - "description": "Transitions should feel snappy as you tap around, even on a slow network. This experience is key to a user's perception of performance. [Learn more](https://web.dev/pwa-page-transitions/).", - "score": null, - "scoreDisplayMode": "manual" - }, - "pwa-each-page-has-url": { - "id": "pwa-each-page-has-url", - "title": "Each page has a URL", - "description": "Ensure individual pages are deep linkable via URL and that URLs are unique for the purpose of shareability on social media. [Learn more](https://web.dev/pwa-each-page-has-url/).", - "score": null, - "scoreDisplayMode": "manual" - }, - "accesskeys": { - "id": "accesskeys", - "title": "`[accesskey]` values are unique", - "description": "Access keys let users quickly focus a part of the page. For proper navigation, each access key must be unique. [Learn more](https://web.dev/accesskeys/).", - "score": null, - "scoreDisplayMode": "notApplicable" - }, - "aria-allowed-attr": { - "id": "aria-allowed-attr", - "title": "`[aria-*]` attributes match their roles", - "description": "Each ARIA `role` supports a specific subset of `aria-*` attributes. Mismatching these invalidates the `aria-*` attributes. [Learn more](https://web.dev/aria-allowed-attr/).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "aria-command-name": { - "id": "aria-command-name", - "title": "`button`, `link`, and `menuitem` elements have accessible names", - "description": "When an element doesn't have an accessible name, screen readers announce it with a generic name, making it unusable for users who rely on screen readers. [Learn more](https://web.dev/aria-name/).", - "score": null, - "scoreDisplayMode": "notApplicable" - }, - "aria-hidden-body": { - "id": "aria-hidden-body", - "title": "`[aria-hidden=\"true\"]` is not present on the document ``", - "description": "Assistive technologies, like screen readers, work inconsistently when `aria-hidden=\"true\"` is set on the document ``. [Learn more](https://web.dev/aria-hidden-body/).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "aria-hidden-focus": { - "id": "aria-hidden-focus", - "title": "`[aria-hidden=\"true\"]` elements do not contain focusable descendents", - "description": "Focusable descendents within an `[aria-hidden=\"true\"]` element prevent those interactive elements from being available to users of assistive technologies like screen readers. [Learn more](https://web.dev/aria-hidden-focus/).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "aria-input-field-name": { - "id": "aria-input-field-name", - "title": "ARIA input fields have accessible names", - "description": "When an input field doesn't have an accessible name, screen readers announce it with a generic name, making it unusable for users who rely on screen readers. [Learn more](https://web.dev/aria-name/).", - "score": null, - "scoreDisplayMode": "notApplicable" - }, - "aria-meter-name": { - "id": "aria-meter-name", - "title": "ARIA `meter` elements have accessible names", - "description": "When an element doesn't have an accessible name, screen readers announce it with a generic name, making it unusable for users who rely on screen readers. [Learn more](https://web.dev/aria-name/).", - "score": null, - "scoreDisplayMode": "notApplicable" - }, - "aria-progressbar-name": { - "id": "aria-progressbar-name", - "title": "ARIA `progressbar` elements have accessible names", - "description": "When a `progressbar` element doesn't have an accessible name, screen readers announce it with a generic name, making it unusable for users who rely on screen readers. [Learn more](https://web.dev/aria-name/).", - "score": null, - "scoreDisplayMode": "notApplicable" - }, - "aria-required-attr": { - "id": "aria-required-attr", - "title": "`[role]`s have all required `[aria-*]` attributes", - "description": "Some ARIA roles have required attributes that describe the state of the element to screen readers. [Learn more](https://web.dev/aria-required-attr/).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "aria-required-children": { - "id": "aria-required-children", - "title": "Elements with an ARIA `[role]` that require children to contain a specific `[role]` have all required children.", - "description": "Some ARIA parent roles must contain specific child roles to perform their intended accessibility functions. [Learn more](https://web.dev/aria-required-children/).", - "score": null, - "scoreDisplayMode": "notApplicable" - }, - "aria-required-parent": { - "id": "aria-required-parent", - "title": "`[role]`s are contained by their required parent element", - "description": "Some ARIA child roles must be contained by specific parent roles to properly perform their intended accessibility functions. [Learn more](https://web.dev/aria-required-parent/).", - "score": null, - "scoreDisplayMode": "notApplicable" - }, - "aria-roles": { - "id": "aria-roles", - "title": "`[role]` values are valid", - "description": "ARIA roles must have valid values in order to perform their intended accessibility functions. [Learn more](https://web.dev/aria-roles/).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "aria-toggle-field-name": { - "id": "aria-toggle-field-name", - "title": "ARIA toggle fields have accessible names", - "description": "When a toggle field doesn't have an accessible name, screen readers announce it with a generic name, making it unusable for users who rely on screen readers. [Learn more](https://web.dev/aria-name/).", - "score": null, - "scoreDisplayMode": "notApplicable" - }, - "aria-tooltip-name": { - "id": "aria-tooltip-name", - "title": "ARIA `tooltip` elements have accessible names", - "description": "When an element doesn't have an accessible name, screen readers announce it with a generic name, making it unusable for users who rely on screen readers. [Learn more](https://web.dev/aria-name/).", - "score": null, - "scoreDisplayMode": "notApplicable" - }, - "aria-treeitem-name": { - "id": "aria-treeitem-name", - "title": "ARIA `treeitem` elements have accessible names", - "description": "When an element doesn't have an accessible name, screen readers announce it with a generic name, making it unusable for users who rely on screen readers. [Learn more](https://web.dev/aria-name/).", - "score": null, - "scoreDisplayMode": "notApplicable" - }, - "aria-valid-attr-value": { - "id": "aria-valid-attr-value", - "title": "`[aria-*]` attributes have valid values", - "description": "Assistive technologies, like screen readers, can't interpret ARIA attributes with invalid values. [Learn more](https://web.dev/aria-valid-attr-value/).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "aria-valid-attr": { - "id": "aria-valid-attr", - "title": "`[aria-*]` attributes are valid and not misspelled", - "description": "Assistive technologies, like screen readers, can't interpret ARIA attributes with invalid names. [Learn more](https://web.dev/aria-valid-attr/).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "button-name": { - "id": "button-name", - "title": "Buttons have an accessible name", - "description": "When a button doesn't have an accessible name, screen readers announce it as \"button\", making it unusable for users who rely on screen readers. [Learn more](https://web.dev/button-name/).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "bypass": { - "id": "bypass", - "title": "The page contains a heading, skip link, or landmark region", - "description": "Adding ways to bypass repetitive content lets keyboard users navigate the page more efficiently. [Learn more](https://web.dev/bypass/).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "color-contrast": { - "id": "color-contrast", - "title": "Background and foreground colors have a sufficient contrast ratio", - "description": "Low-contrast text is difficult or impossible for many users to read. [Learn more](https://web.dev/color-contrast/).", - "score": 1, - "scoreDisplayMode": "binary", - "details": { - "type": "table", - "headings": [], - "items": [] - } - }, - "definition-list": { - "id": "definition-list", - "title": "`
`'s contain only properly-ordered `
` and `
` groups, `