npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

@opencensus/web-instrumentation-zone-peer-dep

v0.0.7

Published

OpenCensus Web instrumentation zone is used to generate user interaction spans for interactions after the initial page loads. Add Zone.js as peer dependency.

Downloads

32

Readme

OpenCensus Web instrumentation zone for user interaction tracing

Gitter chat

For overview and usage info see the main OpenCensus Web readme.

This package generates OpenCensus Web model spans based on user interactions coming after initial page loads. This uses the library Zone.js to generate those spans.

The library is in alpha stage and the API is subject to change.

Overview

This package adds Zone.js as a peer dependency as your application might already be dependent on it. This is the case, for example, with application on Angular as this already imports it.

How it works?

In general, Zone.js is in charge of intercepting all the tasks running in an application, then, OC Web mokey-patches the functions runTask, scheduleTask and cancelTask to intercept and know when a task is running, being scheduled or canceled. To start a new interaction, in the runTask monkey-patch, OC Web checks if the task.eventName corresponds to a click and the element that triggered the task either has a data-ocweb-id attibute or can be named with a CSS selector. In case those conditions are true, a new Zone and rootSpan are started to measure the interaction.

Detecting user interaction stability

To be able to know when the interactions finishes, the number of tasks the interaction triggers has to be tracked. The count of tasks is incremented when a task is scheduled (e.g. setTimeout is called) or the interaction just started, and the count decrements when a task finishes running (e.g. setTimeout finished running the callback) or is canceled. When the count of tasks is 0, the interaction is maybe complete, it is maybe because a setTimeout of 0 ms is queued to actually complete the interaction in case there are no more scheduled tasks ahead. In case there are more tasks ahead, this task is canceled to allow keep counting the tasks. If the interaction is complete, the rootSpan ends and is exported to the OC Agent.

Concurrent interactions

To keep track of concurrent interactions, that is interactions triggered by either different DOM elements or the same element (e.g. click several times the same button). A Map is used to keep track of all interactions.

Handling with multiple event handlers for the same interaction

Also, as in some cases, elements have multiple event handlers or some frameworks might trigger several events for the same interaction. This is the case for React, that triggers several click events for a button if the user only clicked once. OC Web creates a flag that is set when a new interaction is detected and reset it after 50 ms using setTimeout. Having this flag means that all handled events happening within 50 milliseconds will be associated to the same interaction. After 50 milliseconds, if a handled event is detected, this will cause the creation of a new user interaction.

For more details see the src/interaction_tracker.ts and src/on-page-interaction-stop-watch.ts files containing the source code.

What kind of tasks are tracked during the user interaction?

There are three kinds of tasks:

  • MicroTask: used for doing work right after the current task. In this case the Promises are an example of microtasks.
  • MacroTask: used for doing work later. Such as setTimeout or setInterval.
  • EventTask: used for listening on some future event. This may execute zero or more times, with an unknown delay.

Several tasks might run after a user interaction has started, such as a setTimeout or HTTP requests. In order to detect the stability of an interaction, OC web only counts the MicroTasks and MacroTasks as these tasks tell the actual work the interaction might have. We do not mind about EventTasks as they only correspond to events happening and do not do network calls or long code executions. However, OC Web does not count periodic tasks such as setInterval as they will be called repeatedly, which would cause the interaction to never finish and would be impossible to measure the stability of the interaction.

How are route transitions detected and named?

Another kind of traced user interactions are the route transitions within the web page. Route transitions are named as 'Navigation /path/to/page'. For this case, OC Web monkey-patches the browser’s History API. This allows to OC Web detecting all route transitions in a user interaction and give the right name to the interaction. The way it works is identifying the currently running interaction by getting the current interaction from the active Zone when the History API methods (e.g. pushState) are called.

See the src/history-api-patch.ts file.

Intercepting XHR calls and joing browser data

OC Web intercepts XHR calls generated by a user interaction. A new childSpan is created to measure the XHR and the Trace Context Header in the W3C Trace Context format is sent along the XHR.

To intercept XHRs, OC Web checks if the task.target is an instance of XMLHttpRequest. If that is the case, uses the method setRequestHeader() to send the traceparent header as long as the XHR URL is same origin or matches the regular expression given in ocTraceHeaderHostRegex global variable. The XHR span has to start once send() is called but the actual send has not run yet. For that, this method is monkey-patched to dispatch an event telling the interaction tracker the span has to start a new XHR span, the name for the span will be the URL pathname.

When the XHR has ended (readyState is DONE) the XHR span is ended to get the total duration of the XHR. Additionally, some attributes are added to the span, such as the HTTP status (e.g. 200) code and HTTP method (e.g. POST) used for the request. To obtain the HTTP method, the XMLHttpRequest.open() method is monkey-patched and adds this data to the XHR object.

See the src/monkey-patching.ts and src/xhr-interceptor.ts files for details.

In addition to the childSpan for the XHR, OC Web attaches the Resource Performamce Timing API data to the span. This data is retrieved by performance.getEntriesByType(‘resource’) where the resource timing entries will provide that kind of data. However, this data does not correlate every entry with a specific XHR object and some of those XHRs may trigger an additional request related to CORS preflight. Also, there might be several user interactions being tracked at the same time sending requests to the same URL, creating several entries in the Performance Resource Array with the same name. To solve this, an algorithm is created which in general filters out this data and looks for the best entries fitting within the XHR span.

For details in the algorithm see src/perf-resource-timing-selector.ts.

Usage

Custom spans

In addition to automatic user interaction tracing, it is possible to create your own spans for the tasks or code involved in a user interaction. Here is an example for JavaScript

import { tracing } from '@opencensus/web-instrumentation-zone-peer-dep';

function handleClick() {
  // Start child span which will be child of the current root span on the current interaction.
  // To make sure the span is attached to the root span, add this in code that the button is running.
  const childSpan = tracing.tracer.startChildSpan({
    name: 'name of your child span'
  });
  // Do some operation...
  // Finish the child span at the end of it's operation
  childSpan.end();
}

// Create a fake button to point out the custom span is created in the click handler.
const button = document.createElement('button');
button.onclick = handleClick;

Check the user interaction client example which instruments the package and create some custom spans.

For more features and ways to use it see the main OpenCensus Web readme.

Useful links

License

Apache 2.0 - See LICENSE for more information.