• Logging Activity With The Web Beacon API
  • Original article by Drew
  • The Nuggets translation Project
  • Permanent link to this article: github.com/xitu/gold-m…
  • Translator: Elliott Zhao
  • Proofreader: Eternaldeath, StellaBauhinia

The Beacon API is a lightweight and efficient way to get information from web pages to servers. Let’s take a look at how to use it and how it differs from traditional Ajax techniques.

The Beacon API is a javascript-based Web API for sending small amounts of data from a browser to a Web server without waiting for a response. In this article, we’ll show you where it can be used, how it differs from other similar technologies, such as XMLHTTPRequest(‘ Ajax ‘), and how to get started using it.

If you know why you want to use Beacon, you can always jump right to the getting started section.

What does the Beacon API do?

The Beacon API is used to send small amounts of data to the server without waiting for a response. The latter part is the key and points out why Beacon is so useful — our code never needs to process a response, even if the server sends one. Beacons are designed to send data and then forget about it. I wasn’t expecting a response, and we’re not getting one.

Think of it like a postcard sent home from vacation. You put a little bit of data on it (sort of like “wish you were here” and “What a nice day”), put it in the mailbox, and you don’t expect a response. No one replies to a postcard saying “Yeah, I wish I was there, thank you very much!”

For modern websites and applications, there are many use cases that can be neatly categorized into this send-and-forget processing pattern.

Getting the process just right is no easy task. That’s why we set up a ** “I work this way” session ** — smart cookie sharing works great. This is, of course, part of Smashing members’ service.

Track statistics and data analysis

The first use case that comes to mind for most people is analysis. A large solution like Google Analytics might provide a good overview of things like page visits, but what if we want more personalized content? We could write some JavaScript to keep track of what’s happening on the page (maybe how the user interacts with the build, or which article they read before following the CTA’s advice), but we also need to send this data to the server when the user leaves the page. Beacon does this perfectly, because we just record the data and don’t respond.

There’s no reason why we can’t cover the mundane tasks typically assigned to Google Analytics, reporting data based on the capabilities of the users themselves and their devices and browsers. You can even bind these statistics to known individuals if the user is logged into the session. Whatever data you collect, you can use Beacon to send it back to the server.

Debugging and logging

Another useful use of this behavior is to record information from JavaScript code. Suppose you have a complex interaction component on your page that passes all tests perfectly, but occasionally fails in production. You know there is a failure, but there is no way to see the error message and start debugging. If you can sniff out the failure itself from the code, you can collect diagnostic information and use Beacon to send it all back for recording.

In fact, any logging task can be performed using Beacon, such as creating archive points in a game, gathering information about the use of a feature, or recording the results of a multivariable test. If you want the server to know something is happening in the browser, then Beacon might be a good candidate.

Couldn’t we have done that already?

I know what you’re thinking. There’s nothing new, is there? For more than a decade, we’ve been able to communicate from the browser to the server using XMLHTTPRequest. More recently we have the Fetch API, which uses a more modern, Promise-based interface to do almost the same thing. So why do we need the Beacon API?

The key here is because we don’t get a response, the browser can queue the request and send it, non-blocking any other code. As far as the browser is concerned, it doesn’t matter whether our code is still running or where the code executes, because there’s nothing to return, and it can just move the HTTP request into the background until it’s convenient to send it.

This might mean waiting for the CPU load to be low, or the network to be idle, or sending directly if possible. It is important that the browser queues the Beacon and immediately returns control. It doesn’t miss a thing when it sends a Beacon.

To understand why this is a big deal, we need to look at how and how to make this request from our code. Take our analysis log script for example. Our code might calculate how much time the user spends on the page, so sending the data back to the server at the last minute becomes critical. When the user wants to leave the page, we want to stop the timer and send the data home.

In general, you should use either unload or beforeUnload events to perform logging. These events are triggered when the user does something like click a link to navigate to another page. The problem here is that code running on an Unload event blocks execution and delays the page unload. If page unloading is delayed, then loading the next page is also delayed, so the experience feels sluggish.

Remember how slow HTTP requests really are. If you’re thinking about performance, usually one of the main things you’re trying to reduce is extra HTTP requests, because sending a request to the network and getting a response can be super slow. The last thing you want to do is put this time-consuming operation between activating the link and starting the request for the next page.

Beacon handles this by queueing requests without blocking, instantly returning control to your code. The browser is then responsible for sending the request in the background without blocking. This makes everything much faster, which makes users happier and keeps us all in our jobs.

An introduction to

So, we know what a Beacon is and why we need it, so let’s start with some code. The basics couldn’t be simpler:

let result = navigator.sendBeacon(url, data);
Copy the code

The result is Boolean, returning true if the browser accepts and queues the request, and false if something went wrong in the process.

usenavigator.sendBeacon()

Navigator. sendBeacon takes two arguments. The first parameter is the REQUESTED URL. The request is executed as an HTTP POST, sending any data provided in the second parameter.

Data parameters can be in any of several formats, which are taken directly from the Fetch API. This can be a Blob, a BufferSource, FormData, or URLSearchParams — basically any request body type used to create a request using Fetch.

For basic key-value data, I like to use FormData because it’s uncomplicated and easy to read back.

// Send the data to the destination URLlet url = '/api/my-endpoint'; // Create a new FormData and add a key-value pairlet data = new FormData();
data.append('hello'.'world');
    
let result = navigator.sendBeacon(url, data);
    
if (result) { 
  console.log('Successfully queued! ');
} else {
  console.log('Failure.');
}
Copy the code

Browser support

Beacon support is good in browsers, with the notable exceptions being Internet Explorer (which works with Edge) and Opera Mini. It should work for most uses, but it’s worth testing for support before using navigator.sendbeacon.

It's easy to do:if (navigator.sendBeacon) {
      // Beacon 代码
    } else{// No Beacon maybe back to XHR? }Copy the code

If Beacon is unavailable and the request is important, you can fall back to blocking methods such as XHR. Depending on your audience and target, you can also choose not to.

One example: Keeping track of time spent on a page

To understand this in practice, let’s create a basic system to keep track of how long a user stays on the page. When the page loads, we record the time, and when the user leaves the page, we send the start time and the current time to the server.

Since we only care about the time spent (not the actual time), we can use performing.now () to get the basic timestamp when the page loads.

let startTime = performance.now();
Copy the code

If we put the log in a function, we can call it when the page is unloaded.

let logVisit = function() {// Test we have Beacon supportif(! navigator.sendBeacon)return true; // An example of the URL to which data is sentlet url = '/api/log-visit'; // The data to sendlet data = new FormData();
  data.append('start', startTime);
  data.append('end', performance.now());
  data.append('url', document.URL); / / start! navigator.sendBeacon(url, data); };Copy the code

Finally, we need to call this function when the user leaves the page. My instinct is to use unload events, but Safari on the Mac seems to block requests with security warnings, so we’d be better off using beforeunload here.

window.addEventListener('beforeunload'.logVisit);
Copy the code

When the page unloads (or is about to), our logVisit() function will be called and our Beacon will be sent if the browser supports the Beacon API.

(Note that if there is no Beacon support, we return true and pretend everything is fine. Returning false cancels the event and terminates the page unmount. Bad luck.)

Precautions for tracking

Since many of the potential uses of Beacon revolve around activity tracking, I think it would be rash not to mention the social and legal responsibilities of developers when our logging and tracking may be tied to users.

GDPR

We can consider the recent GDPR laws in Europe, which have to do with email, but of course they also deal with any form of storage of personal data. If you know who your users are and can identify their sessions, you should examine the activity you are recording and how it relates to the user terms you declare.

Often, we don’t need to track as much data as the developers tell us. It’s better to deliberately not store information that can be used to identify users, and then reduce the chances of screwing things up.

DNT: Do Not Track

In addition to legal requirements, most browsers have a setting that allows users to express a wish not to be tracked. Do Not Track sends an HTTP header with the request:

DNT: 1
Copy the code

If you are recording data that can track a particular user, and the user sends a positive DNT header, it is best to follow the user’s wishes and anonymize the data, or not track it at all.

For example, in PHP, you could easily detect this header as follows:

if(! empty($_SERVER['HTTP_DNT'])) {// The user does not want to be tracked... }Copy the code

conclusion

The Beacon API is a very useful way to return data from pages to the server, especially for log content. Browser support is extensive, allowing you to record data seamlessly without negatively impacting the user’s browsing experience and site performance. The non-obstructive nature of the request means much better performance than alternatives such as XHR and Fetch.

If you want to read more articles about the Beacon API, the site below is worth a look.

  • W3C Beacon Specification, W3C alternative recommendation
  • “MDN Beacon Documentation”, MDN Network Documentation, Mozilla
  • “Browser support information”, caniuse.com

If you find any mistakes in your translation or other areas that need to be improved, you are welcome to the Nuggets Translation Program to revise and PR your translation, and you can also get the corresponding reward points. The permanent link to this article at the beginning of this article is the MarkDown link to this article on GitHub.


The Nuggets Translation Project is a community that translates quality Internet technical articles from English sharing articles on nuggets. The content covers Android, iOS, front-end, back-end, blockchain, products, design, artificial intelligence and other fields. If you want to see more high-quality translation, please continue to pay attention to the Translation plan of Digging Gold, the official Weibo, Zhihu column.