DEV Community

Cover image for How to create and read QR codes in Node.js
Matt Angelosanto for LogRocket

Posted on • Originally published at blog.logrocket.com

How to create and read QR codes in Node.js

Written by Kayode Adeniyi✏️

QR codes are a convenient and efficient way to store and share information. Reading and writing QR codes can easily be done with Node.js, the popular JavaScript runtime environment. In this article, we will go over the steps for reading and writing QR codes in Node using three popular libraries: qrcode-reader and jimp for reading QR codes, and qrcode for generating QR codes.

To jump ahead:

Writing QR codes in Node

The qrcode library provides a simple API for writing QR codes.

Installing the qrcode library

To install the qrcode library, you will need to have Node.js and npm (the Node.js package manager) installed on your system. If you do not have these tools installed, you can download and install them from the Node.js website.

Once you have Node.js and npm installed, you can install the qrcode library and its dependencies using npm.

Open a terminal window and run the following command:

npm install qrcode
Enter fullscreen mode Exit fullscreen mode

Generating QR code images

To generate a QR code image, you can use the toFile method from the qrcode module. This method takes a file path, the text to encode in the QR code, and an options object as arguments, and generates a QR code image file at the specified file path.

Here is an example of how to use the toFile method to generate a QR code image file:

// __ Importing qrcode __ \\
const QRCode = require('qrcode');

QRCode.toFile('/output-file-path/file.png', 'Encode this text in QR code', {
  errorCorrectionLevel: 'H'
}, function(err) {
  if (err) throw err;
  console.log('QR code saved!');
});
Enter fullscreen mode Exit fullscreen mode

This code will generate a QR code image file with the specified text and error correction level, and save it to the specified output file.

The error correction level specifies the amount of redundancy encoded in the QR code to allow for errors to be corrected when the code is scanned.

In addition to generating QR code image files, the qrcode library also allows you to generate QR code images in other formats, such as JPEG or SVG. To do this, you can use the toString method, which returns a string containing the QR code image data in the specified format.

Here is an example of how to use the toString method to generate a QR code image in SVG format:

// __ Importing qrcode __ \\
const QRCode = require('qrcode');

QRCode.toString('Encode this text in QR code', {
  errorCorrectionLevel: 'H',
  type: 'svg'
}, function(err, data) {
  if (err) throw err;
  console.log(data);
});
Enter fullscreen mode Exit fullscreen mode

This code will generate a QR code image with the specified text and error correction level, and return it as a string of SVG data. You can then use this data to display the QR code image in a browser or other application.

You can scan the QR code below and see the string that you gave as an input to the toString method in the above code snippet.

This is the output:

QR Code Output

Reading QR codes in Node

In this section, you will be reading the QR code file that you generated and saved in the previous section.

To read QR codes in Node, you will use the qrcode-reader and jimp library. These libraries provide simple APIs for parsing and reading QR codes from images and video sources.

Installing the qrcode-reader and jimp libraries

To install the qrcode-reader and jimp libraries, you will once again need to install the required npm packages for reading QR codes.

Open a terminal window and run the following command:

npm install qrcode-reader jimp
Enter fullscreen mode Exit fullscreen mode

Reading QR codes from image files

To read QR codes from image files, you can use the qrcode-reader module's decode method and the jimp module’s read method. The jimp read method parses the image, which will be decoded by the decode method of qrcode-reader module. Here is an example of how to use the read and decode methods to read a QR code from the image we previously saved:

// __ Importing jimp __ \\
const Jimp = require("jimp");

// __ Importing filesystem = __ \\
const fs = require('fs')

// __ Importing qrcode-reader __ \\
const qrCodeReader = require('qrcode-reader');

// __ Read the image and create a buffer __ \\
const buffer = fs.readFileSync('/output-file-path/file.png');

// __ Parse the image using Jimp.read() __ \\
Jimp.read(buffer, function(err, image) {
    if (err) {
        console.error(err);
    }
// __ Creating an instance of qrcode-reader __ \\

    const qrCodeInstance = new qrCodeReader();

    qrCodeInstance.callback = function(err, value) {
        if (err) {
            console.error(err);
        }
// __ Printing the decrypted value __ \\
        console.log(value.result);
    };

// __ Decoding the QR code __ \\
    qrCodeInstance.decode(image.bitmap);
});
Enter fullscreen mode Exit fullscreen mode

This code will read the QR code from the image file and log the result to the console. If the QR code is successfully decoded, the result will be a string containing the QR code's text. If the QR code cannot be decoded, the result will be null.

In the output below, you can see the string that we encoded using the qrcode library.

This is the output:

Encode this text in QR code
Enter fullscreen mode Exit fullscreen mode

You can see that the qrcode library is very useful and can be used with different parameters to create QR codes. The best thing about this library is that it can work with both React.js apps and Node.js servers.

You might be wondering what could be a potential use case for using QR code in a Node.js backend service. Let’s demonstrate an example for creating a simple login route and authenticating a user on the basis of a QR code coming from the frontend.

Creating backend in Node

First, run the following code:

mkdir my-server
Enter fullscreen mode Exit fullscreen mode

Then, we initialize the backend server:

npm init
Enter fullscreen mode Exit fullscreen mode

Next, we have to create the controller and route files:

touch controller.js route.js
Enter fullscreen mode Exit fullscreen mode

Now, install the Node packages:

npm install express jsonwebtoken nodemon
Enter fullscreen mode Exit fullscreen mode

Express: This library is used to create a server and listen on a port in Node.js

Jsonwebtoken: This library is used to create a JWT for sending back to the client

Nodemon: This library is used for detecting live changes in the server code

This will prompt you to enter some metadata and create a package-lock.json file.

After we initialize the backend, the next step is to install the necessary dependencies:

Npm install express jsonwebtoken qrcode
Enter fullscreen mode Exit fullscreen mode

When our server is configured, let's create a controller function for login. Copy and paste the following code into your controller.js file:

// __ controller.js __\\

exports.login = async (req, res) => {
  try {

// __ Extract the secret from the request body. __ \\
    const secret = req.body.secret;
// __ Validate the secret. __ \\

    if (secret === VALID_SECRET) {
// __ Generate a JWT token. __ \\
      const token = jwt.sign({ user: 'example' }, JWT_SECRET, { expiresIn: '1h' });

// __ Send the JWT token to the client. __ \\
      res.json({ token });
    } else {

// __ If the secret is invalid, return an error. __ \\
      res.status(401).json({ error: 'Invalid QR code' });
    }
  } catch (error) {
    console.error(error);
    res.status(500).json({ error: 'Server error' });
  }
};

exports.test = async (req, res) => {

console.log("Hitting the test route");
return res.status(200).json( {message: "Hitting the test route"} );
}
Enter fullscreen mode Exit fullscreen mode

In the code above, we created a login controller that extracts the secret from the request body and validates it. If the validation is successful, it creates a JWT token, signs it with this secret, and sends it back to the client. If the validation fails, it responds with a status code of 401.

N.B., We have added a test function in the file to check the liveness of our server.

After the controller is completed, we will now create a login route to expose our resource. Copy and paste the following code into the route.js file:

// __ route.js __ \\

const app = require("express");
const router = app.Router();

const authController = require("./controller");
router.post("/login", authController.login);
router.get("/test", authController.test);

module.exports = router;
Enter fullscreen mode Exit fullscreen mode

The last step is to write code for index.js and our server will be up and running:

// __ index.js __ \\
const express = require("express");
const app = express();

const auth = require("./route");

app.use("/auth", auth);

const port = process.env.PORT || 3001;
app.listen(port, () => {
console.log(`Listening to port ${port}...`);
});
Enter fullscreen mode Exit fullscreen mode

Running our server

To run the server, first add the following code in the scripts objects located in package.json:

"start": "nodemon index.js",
Enter fullscreen mode Exit fullscreen mode

Now, run:

npm start
Enter fullscreen mode Exit fullscreen mode

Your output should look like this:

Output After Running The Server

Let's test our server by hitting the test route that we previously configured. We can test it in many ways, such as with Postman or Thunder Client, but I will use a curl request.

Open your terminal and run the following command:

curl -I http://localhost:3001/auth/test
Enter fullscreen mode Exit fullscreen mode

The output should look like this:

Output After Using Curl Request

Now that we have tested our server’s liveness, we can send requests to the login route from the frontend we built and authenticate our users.

Conclusion

In this tutorial, we learned how to store and share information in a convenient and efficient way by using QR codes. Whether we need to encode data for a web or mobile app, it can easily be achieved using the Node runtime environment. We also learned how to encode data in different forms in a QR code, such as an SVG file. Finally, we created a Node.js server for authenticating our user with a JWT token by signing it using the QR code secret coming from the frontend.

To further explore Node modules, visit their documentation page.

Happy coding! 😇


200’s only ✔️ Monitor failed and slow network requests in production

Deploying a Node-based web app or website is the easy part. Making sure your Node instance continues to serve resources to your app is where things get tougher. If you’re interested in ensuring requests to the backend or third party services are successful, try LogRocket.

LogRocket Network Request Monitoring

LogRocket is like a DVR for web apps, recording literally everything that happens on your site. Instead of guessing why problems happen, you can aggregate and report on problematic network requests to quickly understand the root cause.

LogRocket instruments your app to record baseline performance timings such as page load time, time to first byte, slow network requests, and also logs Redux, NgRx, and Vuex actions/state. Start monitoring for free.

Top comments (1)

Collapse
 
naucode profile image
Al - Naucode

Great article, you got my follow, keep writing!