Cross-platform performance measurements with User Timing API and perf-marks

Perf-Marks NPM Package Logo
Perf-Marks NPM Package Logo

TL;DR; You can start using Perf-Marks package in your Frontend (Angular, React, VueJS, Svelte, Ember among others) and Backend apps right now and get the built-in performance, lightweight, isomorphic integration, and a simple and smooth developer experience benefits. Support the project by starring the Github repository and enjoy! 🎉


** Feel free to play with the online demos powered by Stackblitz https://stackblitz.com/edit/perf-marks-playground

Introduction

Performance bottleneck is a common problem to be facing when working in large frontend application? Problems like memory leaks, and functions having high consume of memory can be real issues in your site and there are some other features there are really useful to improve the performance of your web app... but what if it’s not enough?

Also it's good to mention that browsers such as Google Chrome will identify and label slow websites. That means that performance is not just papercut, having an important role in your website.

Performance is not papercut, this is and always will be about user experience as well.

Google Chrome will identify and label slow websites
Google Chrome will identify and label slow websites

Defining baselines for performance measurement

Making your web app blazing fast 🚀 is crucial for user experience purposes. At first step to improve that, is by identifying and measuring where time is being spent across your pages.

At first you can use common metrics such as page load errors or high memory consuption due to Long tasks running in your browser. In that case, you can use a Long Task API.

Rail Model

The experimental Long Tasks API gives us visibility into tasks that take 50 milliseconds or more. The 50 ms threshold comes from the RAIL Model, in particular the "Response: process events in under 50 ms" section. These tasks that are blocking the main thread for 50 ms or more can raise issues such as delays on "Time to interactive" metric, High/variable input and event handling latency or even janky animations and scrolling in your project.

const observer = new PerformanceObserver((list) => {
const freezeThreshold = 300;
const perfEntries = list.getEntries();
// Get a list of all the `longtask` entries to be added as payload for your Analytics API
const analyticsData = perfEntries.filter(entry => entry.duration > freezeThreshold);
// In case of any longtask found, it sends the data to be logged in your Analytics platform
if (analyticsData.length > 0) {
navigator.sendBeacon('/my-analytics-log-platform', analyticsData);
}
});
// register observer for long task notifications
observer.observe({entryTypes: ['longtask']});
// Long script execution after this will result in queueing
// and receiving 'longtask' entries in the observer.

You can find more details about the RAIL Model on the Web.Dev page for the Model

For that, you can cont on User Timing API

Core Web Vitals

Another metric that can be used is called Core Web Vitals. Web Vitals is an initiative by Google to provide unified guidance for quality signals that are essential to delivering a great user experience on the web.

The current set is focuses on 3 aspects of the user: experience—loading, interactivity, and visual stability, including some thresholds.

Core Web Vitals - the 3 metrics experience—loading, interactivity, and visual stability
Core Web Vitals - the 3 metrics experience—loading, interactivity, and visual stability

Measure User Experience instead of technologies

Of course, these are some of the metrics you can use as baseline for your measurement. In combination of that, you can use some techniques to improve the numbers that you analyse from your instrumentation, such as:

  • Setting performance budget
  • async load components
  • critical path management
  • optimistic preloading

But the question still persists. What if we missed something? Which metric is the best for my app? This is something that will always come up, it can be because of unique scenario your app is facing or something else. For critical situations like that, a common suggestion is to track the user experience instead of technology.

Not sure? Just don't measure technology

User Experience is always the king
User Experience is always the king
At the end, the user won't know anything about Javascript Frameworks, CSS, Client-side or server-side rendering.

On the other hand, they will know if your website image is using async load and it's taking more than 2 seconds to be visible. This might affect their experience drastically!


User Timing API and performance measures

User Timing API is an important API if you want to measure performance in your Javascript API’s, components, and pages. It allows the developer to create application specific timestamps that are part of the browser's performance timeline. There are two types of user defined timing event types: the mark event type and the measure event type.

These events are named by the instrumented application and can be set at any location in an application. measure events are also named by the application but they are placed between two marks thus they are effectively a midpoint between two marks. You can access those data from JavaScript using the API or by viewing them on your Chrome DevTools Timeline Recordings and in your reports via Lighthouse Audition.

Example showing how LightHouse Auditions will show User Timing API instrumentation
Example showing how LightHouse Auditions will show User Timing API instrumentation

Simplifying User Timing API dev experience across Frontend and Backend with Perf-Marks package

Even though we have all built-in support for us to instrument and measure our pages and API's, their integrations are quite different between frontend and backend. It means that they don't have the same Developer Experience - aka DX - between isomorphic applications using Javascript! As an example, this is a simple code showing how to integrate User Timing API on the backend side in a NodeJS application.

const { performance, PerformanceObserver } = require('perf_hooks');
const marker = 'my-marker';
const observer = new PerformanceObserver((list) => {
// Using `console.log()` to display metrics for `my-marker`
// After that, this observer will be disconnected
console.log(list.getEntries().find(i => i.name === marker));
observer.disconnect();
});
observer.observe({ entryTypes: ['measure'] });
// Starting the marker
performance.mark(marker);
setTimeout(() => {
// Calling the marker for comparison
performance.mark(`${marker}-end`);
setTimeout(() => {
performance.measure(marker, marker, `${marker}-end`);
}, 1000);
}, 2000);
view raw perf-node.js hosted with ❤ by GitHub

To solve this and other problems like having a common API between Backend and Frontend, with full support for intrumentation in isomorphic applications and more I created the package called Perf-Marks.

Perf-Marks is the isomorphic, simplest, and lightweight solution for User Timing API in Javascript - 🚀 only 208B 🚀. Tree-shaking and entry points built-in. Simple as that!

Bundle and package size matters

Bundle and package sizes are always important and this is something to keep in mind when thinking about this module.

The module supports multiple animations, themes, supports for better web accessibility and more feature with an impressive bundle size of 208B and the package has 140kB to be installed in an application 🎉

Easy entry points integration to avoid increase in your bundle

By using Perf-Marks you can easily instrument your Frontend Apps, components, libraries and API's without any friction. The integration is the same between layers and it also follows some best practices.

Instead of load all the package methods, you can import what you need via entry points and avoid your app the load unnecessary code at that moment. This example shows how to measure a specific method by exporting perf-marks profiler() function via perf-marks/profiler entry point.

import { profiler } from 'perf-marks/profiler';
const marker = 'my-marker';
const someExpensiveMemoryAndTimeTask = () => 1 + 1;
// Just need to wrap your function with `profiler()` method and that's it!
// Easy peasy, lemon squeezy! 🍋
profiler(async () => {
return new Promise((resolve) => {
setTimeout(() => {
setTimeout(() => {
resolve(someExpensiveMemoryAndTimeTask());
}, 1000);
}, 2000);
});
}, marker).then(result => console.log(result));

Smooth integration between Frontend and Backend

All the logic to manage the integration between frontend and backend, client-side and server-side is handled by perf-marks. So your developers should have confidence on their instrumentation by simply doing something like this! Easy peasy, lemon squeezy 🍋!

import { start, end } from 'perf-marks/marks';
const marker = 'my-marker';
// Starting the marker
start(marker);
setTimeout(() => {
// Calling the marker for comparison
console.log(end(marker));
}, 2000);
view raw perf-marks.js hosted with ❤ by GitHub

Flexible ways to measure what matters in your app

You might also measure things like A/B Testing time to load, interactions and user journey! For example, how long the users are interacting with your galleries, how long your API is responding some data to your consumers or how long your Frontend Application located in one AWS Region is taking to upload some files to a distinct AWS region, understand the delays and solve that in advance, in a proactive way!

User Journey and Perf-Marks in action


That’s all for now

Stay tuned because this is not the end, it's just the beginning. Let's keep building awesome stuff and have some #fun 🚀

I hope you enjoyed this reading as much as I enjoyed writing it. Thank you so much for reading until the end and see you soon!


Cya đź‘‹

Subscribe

To keep up with posts on this blog, you can also subscribe via RSS.