Measuring Performance

Blitz has a built-in relayer that allows you to analyze and measure the performance of pages using different metrics.

To measure any of the supported metrics, you need export a

reportWebVitals function from the App component:

  1. // app/pages/_app.jsexport function reportWebVitals(metric) { console.log(metric)}function MyApp({Component, pageProps}) { return <Component {...pageProps} />}export default MyApp

This function is fired when the final values for any of the metrics have finished calculating on the page. You can use to log any of the results to the console or send to a particular endpoint.

The

metric object returned to the function consists of a number of properties:

  • id: Unique identifier for the metric in the context of the current page load
  • name: Metric name
  • startTime: First recorded timestamp of the performance entry (if applicable)
  • value: Value, or duration, of performance entry
  • label: Type of metric (web-vital or custom)

There are two types of metrics that are tracked:

  • Web Vitals
  • Custom metrics

Web Vitals

Web Vitals are a set of useful metrics that aim to capture the user experience of a web page. The following web vitals are all included:

You can handle all the results of these metrics using the

web-vital label:

  1. export function reportWebVitals(metric) { if (metric.label === "web-vital") { console.log(metric) // The metric object ({ id, name, startTime, value, label }) is logged to the console }}

There’s also the option of handling each of the metrics separately:

  1. export function reportWebVitals(metric) { switch (metric.name) { case "FCP": // handle FCP results break case "LCP": // handle LCP results break case "CLS": // handle CLS results break case "FID": // handle FID results break case "TTFB": // handle TTFB results break default: break }}

A third-party library,

web-vitals, is used to measure these metrics. Browser compatibility depends on the particular metric, so refer to the Browser Support section to find out which browsers are supported.

Custom metrics

In addition to the core metrics listed above, there are some additional custom metrics that measure the time it takes for the page to hydrate and render:

  • Next.js-hydration: Length of time it takes for the page to start and finish hydrating (in ms)
  • Next.js-route-change-to-render: Length of time it takes for a page to start rendering after a route change (in ms)
  • Next.js-render: Length of time it takes for a page to finish render after a route change (in ms)

You can handle all the results of these metrics using the

custom label:

  1. export function reportWebVitals(metric) { if (metric.label === "custom") { console.log(metric) // The metric object ({ id, name, startTime, value, label }) is logged to the console }}

There’s also the option of handling each of the metrics separately:

  1. export function reportWebVitals(metric) { switch (metric.name) { case "Next.js-hydration": // handle hydration results break case "Next.js-route-change-to-render": // handle route-change to render results break case "Next.js-render": // handle render results break default: break }}

These metrics work in all browsers that support the

User Timing API.

Sending results to analytics

With the relay function, you can send any of results to an analytics endpoint to measure and track real user performance on your site. For example:

  1. export function reportWebVitals(metric) { const body = JSON.stringify(metric) const url = "https://example.com/analytics" // Use `navigator.sendBeacon()` if available, falling back to `fetch()`. if (navigator.sendBeacon) { navigator.sendBeacon(url, body) } else { fetch(url, {body, method: "POST", keepalive: true}) }}

Note: If you use

Google Analytics, using the id value can allow you to construct metric distributions manually (to calculate percentiles, etc…).

  1. export function reportWebVitals({id, name, label, value}) { ga("send", "event", { eventCategory: label === "web-vital" ? "Web Vitals" : "Blitz custom metric", eventAction: name, eventValue: Math.round(name === "CLS" ? value * 1000 : value), // values must be integers eventLabel: id, // id unique to current page load nonInteraction: true, // avoids affecting bounce rate. })}

Read more about sending results to Google Analytics here.