Recently doing one of the more common front-end performance monitoring platform, from front abnormal monitoring, the front end performance monitoring need report and shows the main front end performance data, including home page rendering time, each page white during time, each page load time of all resources and each page so request response time, and so on.

This article describes how to design a general purpose JSSDK that can automatically report front-end performance data with less intrusion. Performance API and sendBeacon method are mainly used. The main reference is the practice of Google Analytics and Ali Cloud front-end performance monitoring platform.

I use NestJS as a backend framework in my projects. Nestjs is a perfectly typescript-enabled, Java Spring-like Node backend framework based on Express. This article mainly focuses on how to report performance data. The back-end processing logic is relatively simple and will not be introduced in detail, so there is no need to know how to use NestJS. The main contents of this paper include:

  • Obtain front-end Performance data based on the Performance API
  • When performance data should be reported
  • How do I report Performance Data

Originally in my blog, welcome star

Github.com/fortheallli…


1. Obtain front-end Performance data based on the Performance API

The front-end Performance data reported in this document consists of two parts: the Performance data obtained through the Performance API and the user-defined data that should be reported on each page.

Let’s start by looking at the data obtained through the Performance API. This data also consists of two parts: Performance data for the current page and data related to the current page resource load and asynchronous request.

(1) Performance data provided by the Performance API

Window. The performance. The timing will return an object, the object contains a variety of data associated with the page rendering. This article will not introduce this object in detail, but only give a method to calculate performance data based on this object:

let times = {}; let t = window.performance.timing; // Redirection time times. RedirectTime = t.retend - t.redirectStart; // DNS query time times.dnsTime = t. domainlookupend -t. domainlookupstart; //TTFB read the first byte of the page time. ttfbTime = t.esponsestart - t.navigationStart; //DNS cache time times.appcacheTime = t. domainlookupStart -t. setchstart; // Time to uninstall the page times. UnloadTime = t. loadeventend - t. loadeventstart; // TCP connection time times. TcpTime = t.connectend - t.connectstart; // Request request time times.reqTime = t.esponseend -t.esponsestart; // Time to parse dom tree times. AnalysisTime = t. domcomplete - t. dominteractive; // blankTime times. BlankTime = t.blankLoading - t.fetchstart; //domReadyTime times.domReadyTime = t.domContentLoadedEventEnd - t.fetchStart;Copy the code

The Times object above contains performance related properties, which can be calculated based on performance. Timing. Here we consider domReadyTime to be the first screen loading time. In addition, we can also customize the method to report the first screen time:

For example, some scenarios can be considered to be the point at which the dom increment is greatest, and some scenarios can define the time at which the visible DOM increment is greatest.

(2) Resource loading and request data provided by the Performance API

Through the window. The performance. However, () to obtain resource loading and request the relevant data. In each page, there are many resources to load, such as JS, CSS, etc., and there are some asynchronous requests in the page. Through the window. The performance. However, () can get these resource loading and the data related to the asynchronous request. We can obtain load and asynchronous request data as follows:

let entryTimesList = []; let entryList = window.performance.getEntries(); entryList.forEach((item,index)=>{ let templeObj = {}; let usefulType = ['navigation','script','css','fetch','xmlhttprequest','link','img']; if(usefulType.indexOf(item.initiatorType)>-1){ templeObj.name = item.name; templeObj.nextHopProtocol = item.nextHopProtocol; // DNS query time templeobj.dnsTime = item.domainLookupEnd - item.domainLookupStart; Templeobj. tcpTime = item.connectend - item.connectStart; // Request time templeobj.reqtime = item.responseEnd - item.responseStart; // Redirection time templeobj.redirectTime = item.redirectend - item.redirectStart; entryTimesList.push(templeObj); }});Copy the code

We through the window. The performance. However, () to obtain an asynchronous request with resource loading and the correlation data array, Then filter out the element data with one of the attributes [‘navigation’,’script’,’ CSS ‘,’fetch’,’ XMLHttprequest ‘,’link’,’img’] based on the initiatorType attribute of each element in the array.

(3) Pay attention

  • Through the window. The performance. Timing of the page rendering of the relevant data, and applied in a single page to change the url but without refreshing the page is not updated. Therefore, it is not possible to obtain the page rendering time of each subroute simply by using the API. If you need to report the rerender time of each sub-page in the case of route switchover, you need to customize the report.

  • Through the window. The performance. However, () to obtain resource loading and asynchronous requests the related data, and will return when page switching routing calculation, can realize automatic report.

2. When to report performance data

Next, determine when to report performance data, since pv and UV are to be dealt with and it is generally considered that a report is a visit, so when to report performance data. In my system, the following scenarios are selected to report front-end performance data:

  • Pages load and refresh
  • Page Switching route
  • The TAB of the page becomes visible again

For the above three scenarios, especially for route switching, if route switching is implemented by changing hash value, then only need to listen for hashchange event, if url is changed through HTML5 history API, The pushState and ReplacEstate events need to be redefined. See my last article on how to gracefully listen for URL changes in a single page application.

The solution to monitor URL changes in the routing scenario of history is given directly:

var _wr = function(type) {
   var orig = history[type];
   return function() {
       var rv = orig.apply(this, arguments);
      var e = new Event(type);
       e.arguments = arguments;
       window.dispatchEvent(e);
       return rv;
   };
};
 history.pushState = _wr('pushState');
 history.replaceState = _wr('replaceState');
Copy the code

Then we can monitor corresponding events according to the above scenarios, so as to realize the reporting of front-end performance data:

addEvent(window,'load',function(e){ ... deal with something }); AddEvent (window,'replaceState', function(e) {... deal with something }); addEvent(window,'pushState', function(e) { ... deal with something }); AddEvent (window,'hashchange',function(e){... deal with something }); AddEvent (' document ', 'visibilitychang', function (e) {... deal with something })Copy the code

AddEvent is an event that is compatible with IE and the standard DOM event flow model.

3. How to report performance Data

So how do we report performance data? Our first reaction is to report front-end performance data in the form of Ajax requests. This approach has some drawbacks, such as the need for special handling across domains and the fact that the corresponding Ajax method does not always succeed if the page is destroyed.

Among them, cross-domain problems are easier to deal with, and the most difficult problem to solve is the second point:

If the page is destroyed, the corresponding Ajax method may not be sent successfully.

We can use different methods to report performance data according to the methods in Google Analytics (GA), based on browser compatibility and URL length. The main principles are as follows:

Requests are sent by dynamically creating img tags and concatenating urls in img. SRC. There is no cross-domain restriction. If the URL is too long, send the request as sendBeacon. If the sendBeacon method is not compatible, send the Ajax POST synchronization request

(1) sendBeacon method

To solve the problem of not being able to complete asynchronous Ajax requests after the document is unloaded or the page is closed, in many cases we turn asynchrony into synchronization. Perform synchronous method calls in the UNLOAD or beforeUnload event of a page unload.

The problem with synchronous method calls, however, is that they delay the transition from page A to page B. The sendBeacon method solves this problem in a nutshell:

The sendBeacon method sends data asynchronously during page destruction, so it does not cause blocking problems like synchronous Ajax requests and does not affect the rendering of the next page

SendBeacon is called as follows:

navigator.sendBeacon(url [, data]);
Copy the code

Data can be ArrayBufferView, Blob, DOMString, or FormData

To send parameters, we typically specify data as a Blob. Also note that in sendBeacon’s request header, content-type “Application /json; Charset = utf-8 “.

In the header of sendBeacon, only three forms of content-type are supported:

  • application/x-www-form-urlencoded
  • multipart/form-data
  • text/plain

Generally formulated as Application/X-www-form-urlencoded, complete examples of sending requests through sendBeacon are as follows:

Function sendBeacon(url,data){// Check whether navigator is supported. SendBeacon let headers = {type: 'application/x-www-form-urlencoded' }; let blob = new Blob([JSON.stringify(data)], headers); navigator.sendBeacon(url,blob); }Copy the code

How does the back end handle the sendBeacon request? SendBeacon sends a post-like request in the request header, so it can handle the sendBeacon request as if it were POST.

The content-type of an Ajax request is usually: “Application /json; Charset = UTF-8 “and the content-Type of the sendBeacon request is: “Application/X-www-form-urlencoded” so that a normal Ajax POST request can be distinguished from a sendBeacon request in back-end processing.

In addition, if there is a cross-domain problem in the request processing, use the cross-domain mode of CORS to handle the request. The back-end configuration needs to be: allow-Control-Allow-Origin, etc. You can use the Express CORS package to simplify the configuration:

async function bootstrap() {
  const app = await NestFactory.create(ApplicationModule,instance);
  app.use(cors());

  await app.listen(3000)
}
bootstrap();
Copy the code

(2) Dynamically create the form of img tags

The request can be sent by dynamically creating the IMG tag and specifying the URL specified by the SRC attribute. Firstly, there is no cross-domain restriction. Secondly, the dynamic insertion of the IMG tag will delay the unloading of the page and ensure the insertion of the image.

Here is an example of dynamically creating an IMG tag:

function imgReport(url, data) { if (! url || ! data) { return; } let image = document.createElement('img'); let items = []; items = JSON.Parse(data); let name = 'img_' + (+new Date()); image.onload = image.onerror = function () { }; let newUrl = url + (url.indexOf('? ') < 0? '? ' : '&') + items.join('&'); image.src = newUrl; }Copy the code

In addition, when we dynamically create the IMG tag to send the request, the request is an image. In the back-end process, the image should be returned at the end so that the front-end image.onload method is triggered. Example: localhost:8080/1.jpg

@Controller('1.jpg') export class AppUploadController { constructor(private readonly appService: AppService) {} @Get() getUpload(@Req() req,@Res() res): void { ... deal with some thing res.sendFile(join(__dirname, '.. ', 'public/1.jpg')) } }Copy the code

SendFile (join(__dirname, ‘.. ‘, ‘public/1.jpg’)) will return the image so that the onload method of the front image will be called.

(3) Synchronize ajax POST requests

The dynamic creation of img tags has some problems when concatenating urls because browsers have limits on the length of urls. The sendBeacon method is not very compatible, and the last resort is to send a synchronous Ajax request, which, as mentioned earlier, will be executed before the page destruction period, although it will block the rendering of the next page to some extent.

function xmlLoadData(url,data) {
  var client = new XMLHttpRequest();
  client.open("POST", url,false);
  client.setRequestHeader("Content-Type", "application/json; charset=utf-8");
  client.send(JSON.stringify(data));
}
Copy the code

(4) Comprehensive solutions

If the LENGTH of THE URL is less than the maximum length allowed by the browser, then the front-end performance data is sent in the form of dynamic creation of IMG tags. If the URL is too long, it is determined whether the browser supports sendBeacon method. If so, Send the request through the sendBeacon method, otherwise send a synchronous Ajax request.

function dealWithUrl(url,appId){
      let times = performanceInfo(appId);
      let items = decoupling(times);
      let urlLength = (url + (url.indexOf('?') < 0 ? '?' : '&') + items.join('&')).length;
      if(urlLength<2083){
        imgReport(url,times);
      }else if(navigator.sendBeacon){
        sendBeacon(url,times);
      }else{
        xmlLoadData(url,times);
      }
    }
Copy the code