Skip to main content

Utilizing HTTP/2 Push for Faster Page Load in Node.js

HTTP/2 has several advantages over HTTP/1 that I've mention in my earlier post. In this post, I want to show how push-request can be performed using Node.js to create an HTTP/2 server. Push request is used to push static files such as scripts and styles so that the client can consume those static files as soon as possible without the need to request them first.


In this example, several built-in Node modules are required and an external module for ease of content-type setting named mime. Let's install it first.

npm init
npm i --save mime

HTTP/2 encodes all headers of a request and it presents several new headers for identifying a request such as :method and :path. For more clarity, I call some constants related to the HTTP/2 header from the http2.constants property. Let's create the server.js file.

const http2 = require('http2');
const { 
  HTTP2_HEADER_PATH, 
  HTTP2_HEADER_METHOD, 
  HTTP2_HEADER_CONTENT_TYPE,
  HTTP2_HEADER_CONTENT_LENGTH,
  HTTP2_HEADER_LAST_MODIFIED,
  HTTP2_HEADER_AUTHORITY,
  HTTP2_HEADER_STATUS,
  HTTP_STATUS_INTERNAL_SERVER_ERROR
} = http2.constants;

const mime = require('mime');
const path = require('path');
const fs = require('fs');
const fsp = require('fs/promises');
const { O_RDONLY } = fs.constants;

Currently, most browsers require TLS encrypted communication for HTTP/2 so that for this demo, we need to generate a self-signed certificate and include the certificate as the server parameter.

const serverPort = 3000;
const publicLocation = 'public'; // directory to store static files
const serverOptions = {
  key: fs.readFileSync('./your-selfsigned-key.pem'),
  cert: fs.readFileSync('./your-selfsigned-cert.pem')
}

We need to create a public directory. Then, create several static files including index.html, app.js, and style.css inside the directory. We can write any methods or declarations inside those files for demo purposes. The index.html file should include app.js and style.css on the head or body.

In server.js, we create a function that will handle file sending through the HTTP/2 stream. In Node.js, the stream is an instance of the http2.ServerHttp2Stream object.

function sendFile(stream, fileLocation) {
  let fileHandle;

  fsp.open(fileLocation, O_RDONLY)
    .then((fh) => {
      fileHandle = fh;
      return fileHandle.stat();
    })
    .then((stats) => {
      // setup file sending header
      const headers = {
        [HTTP2_HEADER_CONTENT_LENGTH]: stats.size,
        [HTTP2_HEADER_LAST_MODIFIED]: stats.mtime.toUTCString(),
        [HTTP2_HEADER_CONTENT_TYPE]: mime.getType(fileLocation)
      };

      // close the file in 'close' event of the stream
      stream.on('close', () => {
        fileHandle.close();
      });

      // send response with file descriptor
      stream.respondWithFD(fileHandle.fd, headers);
    })
    .catch((reason) => {
      stream.respond({
        [HTTP2_HEADER_STATUS]: HTTP_STATUS_INTERNAL_SERVER_ERROR
      });

      stream.end();
    });
}

Last, we define an HTTP/2 server object that will handle file requests. For this demo, the server only accepts any request to index.html file. Other requests will be responded with a plain text message.

const server = http2.createSecureServer(serverOptions);

server.on('stream', (stream, headers) => {

  // get some headers
  const method = headers[HTTP2_HEADER_METHOD].toLowerCase();
  const url = new URL(headers[HTTP2_HEADER_PATH], 'https://' + headers[HTTP2_HEADER_AUTHORITY]);
  const pathname = url.pathname.replace(/^\/+|\/+$/g, '');

  // handle root or index.html file request
  if (pathname==='' || pathname==='index.html') {
    if (stream.pushAllowed) {
      
      // push app.js
      stream.pushStream({
        [HTTP2_HEADER_PATH]: '/app.js'
      }, (err, pushStream) => {
        if (!err) {
          sendFile(pushStream, path.join(__dirname, publicLocation, 'app.js'));
        }
      });

      // push style.css
      stream.pushStream({
        [HTTP2_HEADER_PATH]: '/style.css'
      }, (err, pushStream) => {
        if (!err) {
          sendFile(pushStream, path.join(__dirname, publicLocation, 'style.css'));
        }
      });
    }

    // send index.html
    let indexFileLocation = path.join(__dirname, publicLocation, 'index.html');
    sendFile(stream, indexFileLocation);

  } else { // handle other requests
    stream.respond({
      'content-type': 'text/plain; charset=utf-8',
      ':status': 200
    });
    stream.end('hello world');
  }
});

server.listen(serverPort, () => {
  console.log('HTTP2 server listen to port ' + serverPort);
});

Now we can start the server and open the website in a browser with the address https://localhost:3000/index.html. If we open the browser inspection tool, we can see on the network panel that the initiator of app.js and style.css requests are called "Push". In other words, those files have already been cached by the browser, and the browser isn't required to make additional HTTP requests to the server.



Comments

Popular posts from this blog

Deploying a Web Server on UpCloud using Terraform Modules

In my earlier post , I shared an example of deploying UpCloud infrastructure using Terraform from scratch. In this post, I want to share how to deploy the infrastructure using available Terraform modules to speed up the set-up process, especially for common use cases like preparing a web server. For instance, our need is to deploy a website with some conditions as follows. The website can be accessed through HTTPS. If the request is HTTP, it will be redirected to HTTPS. There are 2 domains, web1.yourdomain.com and web2.yourdomain.com . But, users should be redirected to "web2" if they are visiting "web1". There are 4 main modules that we need to set up the environment. Private network. It allows the load balancer to connect with the server and pass the traffic. Server. It is used to host the website. Load balancer. It includes backend and frontend configuration. Dynamic certificate. It is requ...

Armin or Commander Erwin

In the moment of conflict in the scout team, who will be revived?

What's Good About Strapi, a Headless CMS

Recently, I've been revisiting Strapi as a solution for building backend systems. I still think this headless CMS can be quite useful in certain cases, especially for faster prototyping or creating common websites like company profiles or e-commerce platforms . It might even have the potential to handle more complex systems. With the release of version 5, I'm curious to know what updates it brings. Strapi has launched a new documentation page, and it already feels like an improvement in navigation and content structure compared to the previous version. That said, there's still room for improvement, particularly when it comes to use cases and best practices for working with Strapi. In my opinion, Strapi stands out with some compelling features that could catch developers' attention. I believe three key aspects of Strapi offer notable advantages. First, the content-type builder feature lets us design the data structure of an entity or database model , including ...

Kenshin VS The Assassin

It is an assassin versus assassin.

Rangkaian Sensor Infrared dengan Photo Dioda

Keunggulan photodioda dibandingkan LDR adalah photodioda lebih tidak rentan terhadap noise karena hanya menerima sinar infrared, sedangkan LDR menerima seluruh cahaya yang ada termasuk infrared. Rangkaian yang akan kita gunakan adalah seperti gambar di bawah ini. Pada saat intensitas Infrared yang diterima Photodiode besar maka tahanan Photodiode menjadi kecil, sedangkan jika intensitas Infrared yang diterima Photodiode kecil maka tahanan yang dimiliki photodiode besar. Jika  tahanan photodiode kecil  maka tegangan  V- akan kecil . Misal tahanan photodiode mengecil menjadi 10kOhm. Maka dengan teorema pembagi tegangan: V- = Rrx/(Rrx + R2) x Vcc V- = 10 / (10+10) x Vcc V- = (1/2) x 5 Volt V- = 2.5 Volt Sedangkan jika  tahanan photodiode besar  maka tegangan  V- akan besar  (mendekati nilai Vcc). Misal tahanan photodiode menjadi 150kOhm. Maka dengan teorema pembagi tegangan: V- = Rrx/(Rrx + R2) x Vcc V- = 150 / (1...

How To Verify Phone Number for Free Using WhatsApp

If you have a product or business that maintains user information like phone numbers, verifying the validity or ownership of the phone number could become important, as the phone number can be used as an authentication method or targeted marketing channel. The typical phone verification procedure is by generating a code or OTP in our application, sending that OTP to the user's phone, and then the user should insert the OTP in our application for verification. The OTP can be sent to the users through services like SMS or WhatsApp that require a valid phone number. For internet-based communication, WhatsApp has become the de facto standard for sending the OTP. WhatsApp requires its users to have a valid phone number during account creation, and it already has a huge number of users, approximately 3 billion in 2025. Using that common procedure, WhatsApp will charge us for each OTP sent. The cost depends on the country of the target phone number. For Indonesia...