From 393ee33f3458ea11fe39007d2abaeb6ef2ef89c8 Mon Sep 17 00:00:00 2001 From: Salil Kapoor <49902237+salilkapoor@users.noreply.github.com> Date: Thu, 26 Mar 2020 13:26:01 +0530 Subject: [PATCH 01/19] Initial Commit (#4) --- .gitignore | 23 + .size-limit.js | 6 + README.md | 134 +- build-analyzer/reports.html | 99 + lighthouse/config/config.js | 36 + lighthouse/config/custom-headers.json | 5 + lighthouse/reports/initial.report.html | 6741 +++++++++ lighthouse/reports/initial.report.json | 5795 ++++++++ lighthouse/reports/test.report.html | 6741 +++++++++ lighthouse/reports/test.report.json | 5947 ++++++++ package.json | 45 + public/favicon.ico | Bin 0 -> 1150 bytes public/index.html | 43 + public/logo192.png | Bin 0 -> 5347 bytes public/logo512.png | Bin 0 -> 9664 bytes public/manifest.json | 25 + public/robots.txt | 3 + src/App.js | 44 + src/App.scss | 19 + src/App.test.js | 9 + src/components/atoms/button/Button.jsx | 20 + src/components/atoms/button/index.jsx | 3 + src/components/atoms/heading/Heading.jsx | 19 + src/components/atoms/heading/index.jsx | 3 + src/components/atoms/input/Input.jsx | 20 + src/components/atoms/input/index.jsx | 3 + src/components/atoms/table/Table.jsx | 18 + src/components/atoms/table/Table.scss | 35 + src/components/atoms/table/index.jsx | 3 + src/components/molecules/addtask/AddTask.jsx | 34 + src/components/molecules/addtask/AddTask.scss | 34 + src/components/molecules/addtask/index.jsx | 3 + src/components/molecules/tasks/Tasks.jsx | 92 + src/components/molecules/tasks/Tasks.scss | 29 + src/components/molecules/tasks/index.jsx | 3 + src/components/organisms/add/Add.jsx | 17 + src/components/organisms/add/Add.scss | 6 + src/components/organisms/add/index.jsx | 3 + src/components/organisms/banner/Banner.js | 38 + src/components/organisms/banner/Banner.scss | 23 + src/components/organisms/banner/index.js | 3 + src/components/organisms/list/List.jsx | 17 + src/components/organisms/list/List.scss | 7 + src/components/organisms/list/index.jsx | 3 + src/hooks/useToDoState.js | 149 + src/index.js | 6 + src/setupTests.js | 5 + src/styles/base/variables.scss | 5 + src/styles/main.scss | 8 + src/utils/utils.js | 17 + yarn.lock | 12190 ++++++++++++++++ 51 files changed, 38529 insertions(+), 2 deletions(-) create mode 100644 .gitignore create mode 100644 .size-limit.js create mode 100644 build-analyzer/reports.html create mode 100644 lighthouse/config/config.js create mode 100644 lighthouse/config/custom-headers.json create mode 100644 lighthouse/reports/initial.report.html create mode 100644 lighthouse/reports/initial.report.json create mode 100644 lighthouse/reports/test.report.html create mode 100644 lighthouse/reports/test.report.json create mode 100644 package.json create mode 100644 public/favicon.ico create mode 100644 public/index.html create mode 100644 public/logo192.png create mode 100644 public/logo512.png create mode 100644 public/manifest.json create mode 100644 public/robots.txt create mode 100644 src/App.js create mode 100644 src/App.scss create mode 100644 src/App.test.js create mode 100644 src/components/atoms/button/Button.jsx create mode 100644 src/components/atoms/button/index.jsx create mode 100644 src/components/atoms/heading/Heading.jsx create mode 100644 src/components/atoms/heading/index.jsx create mode 100644 src/components/atoms/input/Input.jsx create mode 100644 src/components/atoms/input/index.jsx create mode 100644 src/components/atoms/table/Table.jsx create mode 100644 src/components/atoms/table/Table.scss create mode 100644 src/components/atoms/table/index.jsx create mode 100644 src/components/molecules/addtask/AddTask.jsx create mode 100644 src/components/molecules/addtask/AddTask.scss create mode 100644 src/components/molecules/addtask/index.jsx create mode 100644 src/components/molecules/tasks/Tasks.jsx create mode 100644 src/components/molecules/tasks/Tasks.scss create mode 100644 src/components/molecules/tasks/index.jsx create mode 100644 src/components/organisms/add/Add.jsx create mode 100644 src/components/organisms/add/Add.scss create mode 100644 src/components/organisms/add/index.jsx create mode 100644 src/components/organisms/banner/Banner.js create mode 100644 src/components/organisms/banner/Banner.scss create mode 100644 src/components/organisms/banner/index.js create mode 100644 src/components/organisms/list/List.jsx create mode 100644 src/components/organisms/list/List.scss create mode 100644 src/components/organisms/list/index.jsx create mode 100644 src/hooks/useToDoState.js create mode 100644 src/index.js create mode 100644 src/setupTests.js create mode 100644 src/styles/base/variables.scss create mode 100644 src/styles/main.scss create mode 100644 src/utils/utils.js create mode 100644 yarn.lock diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..4d29575 --- /dev/null +++ b/.gitignore @@ -0,0 +1,23 @@ +# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. + +# dependencies +/node_modules +/.pnp +.pnp.js + +# testing +/coverage + +# production +/build + +# misc +.DS_Store +.env.local +.env.development.local +.env.test.local +.env.production.local + +npm-debug.log* +yarn-debug.log* +yarn-error.log* diff --git a/.size-limit.js b/.size-limit.js new file mode 100644 index 0000000..0b43704 --- /dev/null +++ b/.size-limit.js @@ -0,0 +1,6 @@ +module.exports = [ + { + path: "build/static/js/*.js", + limit: "1.5 s" + } + ] \ No newline at end of file diff --git a/README.md b/README.md index 782878f..524be76 100644 --- a/README.md +++ b/README.md @@ -1,2 +1,132 @@ -# react-perf-tools -Repo for performance tool integration in React app +# TO-DO App + +This is a TO-DO app build with react, styled with sass and provides audit for Performance, Accessibility
+SEO, Best-practices with Google's Lighthouse CLI integration.
+This app also provides budgeting with size-limit integration of 1.5 s of threshold. Bundle can be analyze
+by source-map-explorer. + +## Features + +This TO-DO react app provides integration with lighthouse and report for audits based on various categories.
+Reports are provided in .json, .html format. + +## App Capabilities +1) Add a task +2) See the list - pending, completed tasks +3) Mark your pending tasks completed +4) removed the completed tasks + +### State Management +App state is managed with useReducer hook, a custom hook is built using useReducer
to manage state at single point +and provides dispatch actions for performing various updates to state. + +### Styling +Resets and variables are created under styles folder. + +## Scripts + +### `yarn start` + +Runs the app in the development mode.
+Open [http://localhost:3000](http://localhost:3000) to view it in the browser. + +# Lighthouse + +Config for lighthouse can be found under lighthouse/config/config.js
+reports can be found under lighthouse/reports/----
+ 1. initial.(all) -> these were the reports generated initially + +## Scripts + +### `yarn run lighthouse` +Runs the lighthouse script
+and opens the report in browser. + +### Script Explanation + +"lighthouse http://localhost:3000 --config-path=./lighthouse/config/config.js --output json --output html --output-path ./lighthouse/reports/initial --quiet --view --chrome-flags=--headless" + +### url + +http://localhost:3000 - url on which we need to see the metrics. + +#### input +Configuration +`--config-path=./lighthouse/config/config.js` -> Default config can be found here + +#### output +--output json --output html --output-path ./lighthouse/reports/initial -> + +`--output json --output html` -> generates report in json and html format
+`--output-path` -> generates report at the path specific + +`--quiet` -> +If you want the report to run silently you can use the following argument
+ +`--view` -> +With the “view” argument, your report opens in a chrome window once after it has finished running
+ +`--chrome-flags=--headless` -> +If you don’t want the Chrome browser pop-up to open we need to provide this argument
+ +`--extra-headers=./lighthouse/config/custom-headers.json` -> +At times, there are some custom headers that are required to be send in http request, these
+can be override by this argument
+ +# Build Analyzer + +`source-map-explorer` is used as a bundle analyzer as it shows tree map visualization. + +## Why source-map-explorer? + +Analyze and debug JavaScript code through source maps. + +### Scripts + +#### `yarn run analyze` +Runs the analyze script
+and opens the tree map in browser. + +# Size Limit + +`size-limit` to budget the app. + +### Scripts + +#### `yarn run size` +Runs the analyze script
+and shows report as below for pass & fail respectively. + +##### Success + +``` + Time limit: 1.5 s + Size: 43.4 KB + Loading time: 868 ms on slow 3G + Running time: 446 ms on Snapdragon 410 + Total time: 1.4 s +``` + + +##### Error + +``` + Total time limit has exceeded + Time limit: 1 s + Size: 43.4 KB + Loading time: 868 ms on slow 3G + Running time: 398 ms on Snapdragon 410 + Total time: 1.3 s +``` + + +## References + +| Type | Description | URL | +| ------- | --- | --- | +| Lighthouse Developer API | This is an API reference for Lighthouse. | [Google Docs](https://developers.google.com/web/tools/lighthouse)| +| Lighthouse CLI Configuration | Configurations explained for Lighthouse CLI. | [Repo's README.md File](https://github.com/GoogleChrome/lighthouse/blob/master/docs/configuration.md) | +| Lighthouse Code repo | GIT repo link to Lighthouse. | [Github Repo](https://github.com/GoogleChrome/lighthouse)| +| Lighthouse Scoring & Terms Explained | Docs link which can explain insights about how scores are calculated. | [Repo's README.md File](https://github.com/GoogleChrome/lighthouse/blob/d2ec9ffbb21de9ad1a0f86ed24575eda32c796f0/docs/scoring.md#how-are-the-scores-weighted)| +| Chrome Flags | List of Chromium Command Line Switches --headless used at the top. | [Link to refer Chrome Flags](http://peter.sh/experiments/chromium-command-line-switches/)| +| source-map-explorer | details about the package, scripts & arguments to refer. | [Link to refer NPM package](https://www.npmjs.com/package/source-map-explorer)| diff --git a/build-analyzer/reports.html b/build-analyzer/reports.html new file mode 100644 index 0000000..59bb60b --- /dev/null +++ b/build-analyzer/reports.html @@ -0,0 +1,99 @@ + + + + + + [combined] - Source Map Explorer + + + + +
+ + + +
+
+ + + + + + + + diff --git a/lighthouse/config/config.js b/lighthouse/config/config.js new file mode 100644 index 0000000..33bd70d --- /dev/null +++ b/lighthouse/config/config.js @@ -0,0 +1,36 @@ +const defaultConfig = { + extends: 'lighthouse:default', + audits: [ + 'byte-efficiency/uses-optimized-images', + ], + categories: { + 'performance': { + title: 'Performance', + description: 'Performance of react app', + auditRefs: [ + { id: 'first-cpu-idle', weight: 3, group: 'metrics' }, + { id: 'interactive', weight: 5, group: 'metrics' }, + ], + }, + 'accessibility': { + title: 'Accessibility', + }, + 'best-practices': { + title: 'Best-Practices', + }, + 'seo': { + title: 'SEO', + }, + 'pwa': { + title: 'PWA', + } + }, + groups: { + 'metrics': { + title: 'Metrics', + description: 'These metrics encapsulate your web app\'s performance.' + }, + } +} + +module.exports = defaultConfig; \ No newline at end of file diff --git a/lighthouse/config/custom-headers.json b/lighthouse/config/custom-headers.json new file mode 100644 index 0000000..6f064ba --- /dev/null +++ b/lighthouse/config/custom-headers.json @@ -0,0 +1,5 @@ +{ + "session": "sm-001", + "api-key": "custom-001", + "Cookie": "clientData=SESSION_HASH;" +} \ No newline at end of file diff --git a/lighthouse/reports/initial.report.html b/lighthouse/reports/initial.report.html new file mode 100644 index 0000000..3ef6e20 --- /dev/null +++ b/lighthouse/reports/initial.report.html @@ -0,0 +1,6741 @@ + + + + + + + + Lighthouse Report + + + + + + +
+ +
+ + + + + + diff --git a/lighthouse/reports/initial.report.json b/lighthouse/reports/initial.report.json new file mode 100644 index 0000000..d83100b --- /dev/null +++ b/lighthouse/reports/initial.report.json @@ -0,0 +1,5795 @@ +{ + "userAgent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) HeadlessChrome/80.0.3987.149 Safari/537.36", + "environment": { + "networkUserAgent": "Mozilla/5.0 (Linux; Android 6.0.1; Nexus 5 Build/MRA58N) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/74.0.3694.0 Mobile Safari/537.36 Chrome-Lighthouse", + "hostUserAgent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) HeadlessChrome/80.0.3987.149 Safari/537.36", + "benchmarkIndex": 1560 + }, + "lighthouseVersion": "5.6.0", + "fetchTime": "2020-03-23T05:20:52.372Z", + "requestedUrl": "http://localhost:3000/", + "finalUrl": "http://localhost:3000/", + "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. 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", + "displayValue": "", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "redirects-http": { + "id": "redirects-http", + "title": "Does not redirect HTTP traffic to HTTPS", + "description": "If you've already set up HTTPS, make sure that you redirect all HTTP traffic to HTTPS in order to enable secure web features for all your users. [Learn more](https://web.dev/redirects-http).", + "score": 0, + "scoreDisplayMode": "binary" + }, + "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" + }, + "works-offline": { + "id": "works-offline", + "title": "Current page does not respond with a 200 when offline", + "description": "If you're building a Progressive Web App, consider using a service worker so that your app can work offline. [Learn more](https://web.dev/works-offline).", + "score": 0, + "scoreDisplayMode": "binary", + "warnings": [] + }, + "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": [] + }, + "without-javascript": { + "id": "without-javascript", + "title": "Contains some content when JavaScript is not available", + "description": "Your app should display some content when JavaScript is disabled, even if it's just a warning to the user that JavaScript is required to use the app. [Learn more](https://web.dev/without-javascript).", + "score": 1, + "scoreDisplayMode": "binary" + }, + "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.71, + "scoreDisplayMode": "numeric", + "numericValue": 3175.3810000000003, + "displayValue": "3.2 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.71, + "scoreDisplayMode": "numeric", + "numericValue": 3175.3810000000003, + "displayValue": "3.2 s" + }, + "load-fast-enough-for-pwa": { + "id": "load-fast-enough-for-pwa", + "title": "Page load is fast enough on mobile networks", + "description": "A fast page load over a cellular network ensures a good mobile user experience. [Learn more](https://web.dev/load-fast-enough-for-pwa).", + "score": 1, + "scoreDisplayMode": "binary", + "numericValue": 3175.3810000000003 + }, + "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": 0.92, + "scoreDisplayMode": "numeric", + "numericValue": 3175.3810000000003, + "displayValue": "3.2 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": 642884977143, + "data": "" + }, + { + "timing": 600, + "timestamp": 642885277143, + "data": "" + }, + { + "timing": 900, + "timestamp": 642885577143, + "data": "" + }, + { + "timing": 1200, + "timestamp": 642885877143, + "data": "" + }, + { + "timing": 1500, + "timestamp": 642886177143, + "data": "" + }, + { + "timing": 1800, + "timestamp": 642886477143, + "data": "" + }, + { + "timing": 2100, + "timestamp": 642886777143, + "data": "" + }, + { + "timing": 2400, + "timestamp": 642887077143, + "data": "" + }, + { + "timing": 2700, + "timestamp": 642887377143, + "data": "" + }, + { + "timing": 3000, + "timestamp": 642887677143, + "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": 956, + "timestamp": 642885633546, + "data": "" + } + }, + "estimated-input-latency": { + "id": "estimated-input-latency", + "title": "Estimated Input Latency", + "description": "Estimated Input Latency is an estimate of how long your app takes to respond to user input, in milliseconds, during the busiest 5s window of page load. If your latency is higher than 50 ms, users may perceive your app as laggy. [Learn more](https://web.dev/estimated-input-latency).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 12.8, + "displayValue": "10 ms" + }, + "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.", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 34.12700000000018, + "displayValue": "30 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, in milliseconds, of the longest task. [Learn more](https://developers.google.com/web/updates/2018/05/first-input-delay).", + "score": 0.92, + "scoreDisplayMode": "numeric", + "numericValue": 123, + "displayValue": "120 ms" + }, + "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", + "numericValue": 0, + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "time-to-first-byte": { + "id": "time-to-first-byte", + "title": "Server response times are low (TTFB)", + "description": "Time To First Byte identifies the time at which your server sends a response. [Learn more](https://web.dev/time-to-first-byte).", + "score": 1, + "scoreDisplayMode": "binary", + "numericValue": 1.3829999999999814, + "displayValue": "Root document took 0 ms", + "details": { + "type": "opportunity", + "overallSavingsMs": -598.617, + "headings": [], + "items": [] + } + }, + "first-cpu-idle": { + "id": "first-cpu-idle", + "title": "First CPU Idle", + "description": "First CPU Idle marks the first time at which the page's main thread is quiet enough to handle input. [Learn more](https://web.dev/first-cpu-idle).", + "score": 0.94, + "scoreDisplayMode": "numeric", + "numericValue": 3175.3810000000003, + "displayValue": "3.2 s" + }, + "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.95, + "scoreDisplayMode": "numeric", + "numericValue": 3175.3810000000003, + "displayValue": "3.2 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": "46 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": "⚛ (React Tree Reconciliation: Completed Root)", + "startTime": 441.559, + "duration": 22.911, + "timingType": "Measure" + }, + { + "name": "⚛ App [mount]", + "startTime": 443.269, + "duration": 21.035, + "timingType": "Measure" + }, + { + "name": "⚛ Heading [mount]", + "startTime": 445.01, + "duration": 2.179, + "timingType": "Measure" + }, + { + "name": "⚛ Add [mount]", + "startTime": 447.339, + "duration": 2.155, + "timingType": "Measure" + }, + { + "name": "⚛ Heading [mount]", + "startTime": 447.554, + "duration": 0.105, + "timingType": "Measure" + }, + { + "name": "⚛ AddTask [mount]", + "startTime": 447.685, + "duration": 1.8, + "timingType": "Measure" + }, + { + "name": "⚛ Input [mount]", + "startTime": 448.095, + "duration": 1.059, + "timingType": "Measure" + }, + { + "name": "⚛ Button [mount]", + "startTime": 449.174, + "duration": 0.29, + "timingType": "Measure" + }, + { + "name": "⚛ List [mount]", + "startTime": 449.509, + "duration": 14.716, + "timingType": "Measure" + }, + { + "name": "⚛ Heading [mount]", + "startTime": 449.634, + "duration": 0.1, + "timingType": "Measure" + }, + { + "name": "⚛ Tasks [mount]", + "startTime": 449.75, + "duration": 14.464, + "timingType": "Measure" + }, + { + "name": "⚛ Table [mount]", + "startTime": 461.659, + "duration": 2.016, + "timingType": "Measure" + }, + { + "name": "⚛ Button [mount]", + "startTime": 462.379, + "duration": 0.205, + "timingType": "Measure" + }, + { + "name": "⚛ Button [mount]", + "startTime": 462.605, + "duration": 0.099, + "timingType": "Measure" + }, + { + "name": "⚛ Button [mount]", + "startTime": 462.874, + "duration": 0.1, + "timingType": "Measure" + }, + { + "name": "⚛ Button [mount]", + "startTime": 462.99, + "duration": 0.09, + "timingType": "Measure" + }, + { + "name": "⚛ Button [mount]", + "startTime": 463.26, + "duration": 0.125, + "timingType": "Measure" + }, + { + "name": "⚛ Button [mount]", + "startTime": 463.4, + "duration": 0.084, + "timingType": "Measure" + }, + { + "name": "⚛ Table [mount]", + "startTime": 463.694, + "duration": 0.505, + "timingType": "Measure" + }, + { + "name": "⚛ (Committing Changes)", + "startTime": 464.96, + "duration": 1.534, + "timingType": "Measure" + }, + { + "name": "⚛ (Committing Snapshot Effects: 0 Total)", + "startTime": 465.039, + "duration": 0.641, + "timingType": "Measure" + }, + { + "name": "⚛ (Committing Host Effects: 1 Total)", + "startTime": 465.734, + "duration": 0.451, + "timingType": "Measure" + }, + { + "name": "⚛ (Calling Lifecycle Methods: 0 Total)", + "startTime": 466.284, + "duration": 0.161, + "timingType": "Measure" + }, + { + "name": "⚛ (React Tree Reconciliation)", + "startTime": 441.584, + "timingType": "Mark" + }, + { + "name": "⚛ App [mount] (#4)", + "startTime": 443.281, + "timingType": "Mark" + }, + { + "name": "⚛ Heading [mount] (#11)", + "startTime": 445.021, + "timingType": "Mark" + }, + { + "name": "⚛ Add [mount] (#16)", + "startTime": 447.348, + "timingType": "Mark" + }, + { + "name": "⚛ Heading [mount] (#19)", + "startTime": 447.563, + "timingType": "Mark" + }, + { + "name": "⚛ AddTask [mount] (#20)", + "startTime": 447.689, + "timingType": "Mark" + }, + { + "name": "⚛ Input [mount] (#25)", + "startTime": 448.102, + "timingType": "Mark" + }, + { + "name": "⚛ Button [mount] (#26)", + "startTime": 449.182, + "timingType": "Mark" + }, + { + "name": "⚛ List [mount] (#17)", + "startTime": 449.52, + "timingType": "Mark" + }, + { + "name": "⚛ Heading [mount] (#34)", + "startTime": 449.641, + "timingType": "Mark" + }, + { + "name": "⚛ Tasks [mount] (#35)", + "startTime": 449.757, + "timingType": "Mark" + }, + { + "name": "⚛ Table [mount] (#40)", + "startTime": 461.672, + "timingType": "Mark" + }, + { + "name": "⚛ Button [mount] (#71)", + "startTime": 462.389, + "timingType": "Mark" + }, + { + "name": "⚛ Button [mount] (#72)", + "startTime": 462.614, + "timingType": "Mark" + }, + { + "name": "⚛ Button [mount] (#88)", + "startTime": 462.884, + "timingType": "Mark" + }, + { + "name": "⚛ Button [mount] (#89)", + "startTime": 462.998, + "timingType": "Mark" + }, + { + "name": "⚛ Button [mount] (#105)", + "startTime": 463.264, + "timingType": "Mark" + }, + { + "name": "⚛ Button [mount] (#106)", + "startTime": 463.409, + "timingType": "Mark" + }, + { + "name": "⚛ Table [mount] (#41)", + "startTime": 463.704, + "timingType": "Mark" + }, + { + "name": "⚛ (Committing Changes)", + "startTime": 464.968, + "timingType": "Mark" + }, + { + "name": "⚛ (Committing Snapshot Effects)", + "startTime": 465.044, + "timingType": "Mark" + }, + { + "name": "⚛ (Committing Host Effects)", + "startTime": 465.746, + "timingType": "Mark" + }, + { + "name": "⚛ (Calling Lifecycle Methods)", + "startTime": 466.296, + "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": "informative", + "displayValue": "3 chains found", + "details": { + "type": "criticalrequestchain", + "chains": { + "A9F7F4F3F03A14D21E3AD9F357322D55": { + "request": { + "url": "http://localhost:3000/", + "startTime": 642884.67916, + "endTime": 642884.993876, + "responseReceivedTime": 642884.992942, + "transferSize": 1161 + }, + "children": { + "12896.2": { + "request": { + "url": "http://localhost:3000/static/js/bundle.js", + "startTime": 642884.999602, + "endTime": 642885.004151, + "responseReceivedTime": 642885.002933, + "transferSize": 6710 + } + }, + "12896.3": { + "request": { + "url": "http://localhost:3000/static/js/0.chunk.js", + "startTime": 642884.999822, + "endTime": 642885.066643, + "responseReceivedTime": 642885.012461, + "transferSize": 388559 + } + }, + "12896.4": { + "request": { + "url": "http://localhost:3000/static/js/main.chunk.js", + "startTime": 642885.000097, + "endTime": 642885.070138, + "responseReceivedTime": 642885.069172, + "transferSize": 5599 + } + } + } + } + }, + "longestChain": { + "duration": 390.97799989394844, + "length": 2, + "transferSize": 5599 + } + } + }, + "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, + "displayValue": "", + "details": { + "type": "opportunity", + "headings": [], + "items": [], + "overallSavingsMs": 0 + } + }, + "installable-manifest": { + "id": "installable-manifest", + "title": "Web app manifest meets the installability requirements", + "description": "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": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "debugdata", + "items": [ + { + "failures": [], + "isParseFailure": false, + "hasStartUrl": true, + "hasIconsAtLeast192px": true, + "hasIconsAtLeast512px": true, + "hasPWADisplayValue": true, + "hasBackgroundColor": true, + "hasThemeColor": true, + "hasShortName": true, + "shortNameLength": true, + "hasName": true + } + ] + } + }, + "apple-touch-icon": { + "id": "apple-touch-icon", + "title": "Provides 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": 1, + "scoreDisplayMode": "binary", + "warnings": [] + }, + "splash-screen": { + "id": "splash-screen", + "title": "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": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "debugdata", + "items": [ + { + "failures": [], + "isParseFailure": false, + "hasStartUrl": true, + "hasIconsAtLeast192px": true, + "hasIconsAtLeast512px": true, + "hasPWADisplayValue": true, + "hasBackgroundColor": true, + "hasThemeColor": true, + "hasShortName": true, + "shortNameLength": true, + "hasName": true + } + ] + } + }, + "themed-omnibox": { + "id": "themed-omnibox", + "title": "Sets 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": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "debugdata", + "items": [ + { + "failures": [], + "themeColor": "#000000", + "isParseFailure": false, + "hasStartUrl": true, + "hasIconsAtLeast192px": true, + "hasIconsAtLeast512px": true, + "hasPWADisplayValue": true, + "hasBackgroundColor": true, + "hasThemeColor": true, + "hasShortName": true, + "shortNameLength": true, + "hasName": true + } + ] + } + }, + "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", + "explanation": "" + }, + "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", + "warnings": [], + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "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", + "displayValue": "", + "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": 479.25200000000007, + "displayValue": "0.5 s", + "details": { + "type": "table", + "headings": [ + { + "key": "groupLabel", + "itemType": "text", + "text": "Category" + }, + { + "key": "duration", + "itemType": "ms", + "granularity": 1, + "text": "Time Spent" + } + ], + "items": [ + { + "group": "scriptEvaluation", + "groupLabel": "Script Evaluation", + "duration": 190.372 + }, + { + "group": "other", + "groupLabel": "Other", + "duration": 105.268 + }, + { + "group": "styleLayout", + "groupLabel": "Style & Layout", + "duration": 97.376 + }, + { + "group": "scriptParseCompile", + "groupLabel": "Script Parsing & Compilation", + "duration": 75.084 + }, + { + "group": "parseHTML", + "groupLabel": "Parse HTML & CSS", + "duration": 6.124 + }, + { + "group": "paintCompositeRender", + "groupLabel": "Rendering", + "duration": 5.028000000000001 + } + ] + } + }, + "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": 262.35999999999996, + "displayValue": "0.3 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": "Other", + "total": 231.81199999999998, + "scripting": 15.044, + "scriptParseCompile": 2.9719999999999995 + }, + { + "url": "http://localhost:3000/static/js/main.chunk.js", + "total": 175.63999999999996, + "scripting": 171.15599999999995, + "scriptParseCompile": 4.484 + }, + { + "url": "http://localhost:3000/static/js/0.chunk.js", + "total": 68.70400000000001, + "scripting": 3.2000000000000006, + "scriptParseCompile": 65.504 + } + ], + "summary": { + "wastedMs": 262.35999999999996 + } + } + }, + "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": 1, + "scoreDisplayMode": "numeric", + "numericValue": 0, + "displayValue": "", + "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, + "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": 4, + "numScripts": 3, + "numStylesheets": 0, + "numFonts": 0, + "numTasks": 203, + "numTasksOver10ms": 3, + "numTasksOver25ms": 2, + "numTasksOver50ms": 1, + "numTasksOver100ms": 0, + "numTasksOver500ms": 0, + "rtt": 1.1389999999999532, + "throughput": 56129703.44378596, + "maxRtt": 1.1389999999999532, + "maxServerLatency": 1.1270000000000522, + "totalByteWeight": 402029, + "totalTaskTime": 119.81299999999996, + "mainDocumentTransferSize": 1161 + } + ] + } + }, + "network-requests": { + "id": "network-requests", + "title": "Network Requests", + "description": "Lists the network requests that were made during page load.", + "score": null, + "scoreDisplayMode": "informative", + "numericValue": 4, + "details": { + "type": "table", + "headings": [ + { + "key": "url", + "itemType": "url", + "text": "URL" + }, + { + "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": "http://localhost:3000/", + "startTime": 0, + "endTime": 314.71599999349564, + "transferSize": 1161, + "resourceSize": 1798, + "statusCode": 200, + "mimeType": "text/html", + "resourceType": "Document" + }, + { + "url": "http://localhost:3000/static/js/bundle.js", + "startTime": 320.441999938339, + "endTime": 324.99099988490343, + "transferSize": 6710, + "resourceSize": 31927, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "http://localhost:3000/static/js/0.chunk.js", + "startTime": 320.66199998371303, + "endTime": 387.4829999404028, + "transferSize": 388559, + "resourceSize": 1625592, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "http://localhost:3000/static/js/main.chunk.js", + "startTime": 320.9369999822229, + "endTime": 390.97799989394844, + "transferSize": 5599, + "resourceSize": 60480, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + } + ] + } + }, + "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": 1.1389999999999532, + "displayValue": "0 ms", + "details": { + "type": "table", + "headings": [ + { + "key": "origin", + "itemType": "text", + "text": "URL" + }, + { + "key": "rtt", + "itemType": "ms", + "granularity": 1, + "text": "Time Spent" + } + ], + "items": [ + { + "origin": "http://localhost:3000", + "rtt": 1.1389999999999532 + } + ] + } + }, + "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": 1.1270000000000522, + "displayValue": "0 ms", + "details": { + "type": "table", + "headings": [ + { + "key": "origin", + "itemType": "text", + "text": "URL" + }, + { + "key": "serverResponseTime", + "itemType": "ms", + "granularity": 1, + "text": "Time Spent" + } + ], + "items": [ + { + "origin": "http://localhost:3000", + "serverResponseTime": 1.1270000000000522 + } + ] + } + }, + "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", + "numericValue": 3, + "details": { + "type": "table", + "headings": [ + { + "key": "startTime", + "itemType": "ms", + "granularity": 1, + "text": "Start Time" + }, + { + "key": "duration", + "itemType": "ms", + "granularity": 1, + "text": "End Time" + } + ], + "items": [ + { + "duration": 28.403, + "startTime": 397.412 + }, + { + "duration": 57.407, + "startTime": 425.968 + }, + { + "duration": 11.801, + "startTime": 483.468 + } + ] + } + }, + "metrics": { + "id": "metrics", + "title": "Metrics", + "description": "Collects all available metrics.", + "score": null, + "scoreDisplayMode": "informative", + "numericValue": 3175.3810000000003, + "details": { + "type": "debugdata", + "items": [ + { + "firstContentfulPaint": 3175, + "firstMeaningfulPaint": 3175, + "firstCPUIdle": 3175, + "interactive": 3175, + "speedIndex": 3175, + "estimatedInputLatency": 13, + "totalBlockingTime": 34, + "observedNavigationStart": 0, + "observedNavigationStartTs": 642884677143, + "observedFirstPaint": 503, + "observedFirstPaintTs": 642885180243, + "observedFirstContentfulPaint": 503, + "observedFirstContentfulPaintTs": 642885180243, + "observedFirstMeaningfulPaint": 503, + "observedFirstMeaningfulPaintTs": 642885180243, + "observedLargestContentfulPaint": 503, + "observedLargestContentfulPaintTs": 642885180243, + "observedTraceEnd": 1559, + "observedTraceEndTs": 642886235884, + "observedLoad": 492, + "observedLoadTs": 642885169148, + "observedDomContentLoaded": 467, + "observedDomContentLoadedTs": 642885144011, + "observedFirstVisualChange": 506, + "observedFirstVisualChangeTs": 642885183143, + "observedLastVisualChange": 506, + "observedLastVisualChangeTs": 642885183143, + "observedSpeedIndex": 512, + "observedSpeedIndexTs": 642885188897 + }, + { + "lcpInvalidated": false + } + ] + } + }, + "offline-start-url": { + "id": "offline-start-url", + "title": "`start_url` does not respond with a 200 when offline", + "description": "A service worker enables your web app to be reliable in unpredictable network conditions. [Learn more](https://web.dev/offline-start-url).", + "score": 0, + "scoreDisplayMode": "binary", + "explanation": "Timed out waiting for start_url to respond.", + "warnings": [] + }, + "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" + }, + "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://developers.google.com/web/tools/lighthouse/audits/budgets).", + "score": null, + "scoreDisplayMode": "informative", + "displayValue": "4 requests • 393 KB", + "details": { + "type": "table", + "headings": [ + { + "key": "label", + "itemType": "text", + "text": "Resource Type" + }, + { + "key": "requestCount", + "itemType": "numeric", + "text": "Requests" + }, + { + "key": "size", + "itemType": "bytes", + "text": "Transfer Size" + } + ], + "items": [ + { + "resourceType": "total", + "label": "Total", + "requestCount": 4, + "size": 402029 + }, + { + "resourceType": "script", + "label": "Script", + "requestCount": 3, + "size": 400868 + }, + { + "resourceType": "document", + "label": "Document", + "requestCount": 1, + "size": 1161 + }, + { + "resourceType": "stylesheet", + "label": "Stylesheet", + "requestCount": 0, + "size": 0 + }, + { + "resourceType": "image", + "label": "Image", + "requestCount": 0, + "size": 0 + }, + { + "resourceType": "media", + "label": "Media", + "requestCount": 0, + "size": 0 + }, + { + "resourceType": "font", + "label": "Font", + "requestCount": 0, + "size": 0 + }, + { + "resourceType": "other", + "label": "Other", + "requestCount": 0, + "size": 0 + }, + { + "resourceType": "third-party", + "label": "Third-party", + "requestCount": 0, + "size": 0 + } + ] + } + }, + "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": null, + "scoreDisplayMode": "notApplicable" + }, + "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": 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": null, + "scoreDisplayMode": "notApplicable" + }, + "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": 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": null, + "scoreDisplayMode": "notApplicable" + }, + "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": null, + "scoreDisplayMode": "notApplicable" + }, + "audio-caption": { + "id": "audio-caption", + "title": "`