How to write tests in full-stack MERN web application
Only by writing tests can you understand the significance of testing. No coding Bootcamp or course, as far as I know, teaches you how to write tests. Writing tests may not be necessary for demo projects or coursework projects, but it is critical for real-world applications. So, in this Part 3 of the "Let's build and deploy a full stack MERN web application" series, I'll demonstrate how to write tests for ReactJs components and ExpressJs RESTAPI.
Let's get started
What is testing?
During the testing stage of software development, the application's actual outcomes are compared to those that were expected. Before deploying your application, this phase is used to ensure that everything is functioning as it should. Beginner developers frequently believe that writing tests are a time-consuming and unimportant process. But they are ignorant of the fact that testing actually buys you a lot of time in the long run. You can think of testing as a one-time investment.
Software development uses a variety of testing methods, including Unit Testing, Integration Testing, Beta Testing, Regression Testing, and others. We will learn about Unit Testing in this lesson and how to create tests for applications built with ReactJs and NodeJs.
What is Unit Testing?
As the name implies, unit testing involves testing individual pieces of code or an application. For instance, in REST API, we are able to create distinct tests for each endpoint, and in a ReactJs application, we can test each component separately.
Let's begin creating tests for our "Productivity Tracker" application. (If you don't know what this is, read the previous article - Adding Authentication to full stack MERN web application)
Writing tests in NodeJs application with Jest and SuperTest
Note: I recommend setting up a separate database for testing purposes. But for this tutorial, it's not necessary.
Step 1: Separate application and server
First, we have to separate the server from the application because we are using SuperTest to test the application, not the server.
So create an app.js
file in the root and paste this code.
javascript1const express = require("express"); 2const cors = require("cors"); 3 4const ActivityRouter = require("./routes/activity.route"); 5const AuthRouter = require("./routes/auth.route"); 6 7const app = express(); 8 9/* Telling the application to use the express.json() middleware. This middleware will parse the body of 10any request that has a Content-Type of application/json. */ 11app.use(express.json()); 12 13/* Allowing the frontend to access the backend. */ 14app.use(cors()); 15 16/* This is a route handler. It is listening for a GET request to the root route of the application. 17When it receives a request, it will send back a response with the string "Hello World!". */ 18app.get("/", (req, res) => { 19 res.send("Hello World!"); 20}); 21 22/* Telling the application to use the ActivityRouter for any requests that start with "/api". */ 23app.use("/api", ActivityRouter); 24 25/* Telling the application to use the AuthRouter for any requests that start with "/api/auth". */ 26app.use("/api/auth", AuthRouter); 27 28module.exports = app;
... and in server.js
.
javascript1const express = require("express"); 2const mongoose = require("mongoose"); 3 4const app = require("./app"); 5 6/* Loading the environment variables from the .env file. */ 7require("dotenv").config(); 8 9const PORT = process.env.PORT || 5000; 10const MONGODB_URI = process.env.MONGODB_URI || "mongodb://localhost/todoapiDB"; 11 12/* Connecting to the database and then starting the server. */ 13mongoose 14 .connect(MONGODB_URI, { useNewUrlParser: true }) 15 .then(() => { 16 app.listen(PORT, console.log("Server stated on port 5000")); 17 }) 18 .catch((err) => { 19 console.log(err); 20 });
Step 2: Install packages
To start writing tests, you need three npm packages: jest
, supertest
, and cross-env
.
bash1npm i --save-dev jest supertest cross-env
jest: Jest is a framework for testing JavaScript code. Unit testing is its main usage of it. supertest: Using Supertest, we can test endpoints and routes on HTTP servers. cross-env: You can set environmental variables inline within a command using cross-env.
Step 3: Add test script
Open your package.json
file and add the test script to the scripts.
json1"scripts": { 2 "test": "cross-env NODE_ENV=test jest --testTimeout=5000", 3 "start": "node server.js", 4 "dev": "nodemon server.js" 5},
In this case, testTimeout
is set to 5000
because it's possible for certain requests to take a while to complete, and cross-env
is being used to set environment variables and jest
to run test suites.
Step 4: Start writing tests
First, create a folder called tests/
at the application's root, and then create a file there called activity.test.js
. Jest searches for the folder tests/
at the project's root when you do npm run test
. As a result, you must place your test files in the tests/
folder.
Next, import the supertest
and mongoose
packages into the test file.
javascript1const mongoose = require("mongoose"); 2const request = require("supertest");
Import dotenv
to load environment variables, and import app.js
as that is where our application starts.
javascript1const mongoose = require("mongoose"); 2const request = require("supertest"); 3 4const app = require("../app"); 5 6require("dotenv").config();
You'll need to connect and disconnect the database before and after each test (because we don't require the database once testing is complete).
javascript1/* Connecting to the database before each test. */ 2beforeEach(async () => { 3 await mongoose.connect(process.env.MONGODB_URI); 4}); 5 6/* Closing database connection after each test. */ 7afterEach(async () => { 8 await mongoose.connection.close(); 9});
Now write your first unit test.
javascript1describe("GET /api/activities", () => { 2 it("should get all the activities", async () => { 3 const token = await request(app).post("/api/auth/login").send({ 4 email: process.env.EMAIL, 5 password: process.env.PASSWORD, 6 }); 7 8 const response = await request(app) 9 .get("/api/activities") 10 .set({ 11 Authorization: "bearer " + token.body.token, 12 "Content-Type": "application/json", 13 }); 14 15 expect(response.statusCode).toBe(200); 16 expect(response.body.length).toBeGreaterThan(0); 17 }); 18});
In the above code,
- We use
describe
to describe the unit test. Even though it is not required, it will be useful to identify tests in test results. - In
it
, we write the actual test code. Write the expected result in the first argument, and then in the second argument, write a callback function that contains the test code. - In the callback function, the request is sent to the endpoint first, and the expected and actual responses are then compared. The test passes if both answers match, else, it fails.
Note: Since we have authenticated routes in our application, we need to get the token by logging in to the application and using that token to access the route.
You can write tests for all the endpoints in the same manner.
javascript1describe("POST /api/activity", () => { 2 it("should add an activity to the database", async () => { 3 const token = await request(app).post("/api/auth/login").send({ 4 email: process.env.EMAIL, 5 password: process.env.PASSWORD, 6 }); 7 8 const response = await request(app) 9 .post("/api/activity") 10 .send({ 11 name: "Jogging", 12 time: "3:00 PM", 13 }) 14 .set({ 15 Authorization: "bearer " + token.body.token, 16 "Content-Type": "application/json", 17 }); 18 19 expect(response.statusCode).toBe(201); 20 }); 21});
Then run npm run test
to run the test suites (suite - test file).
Writing tests in ReactJs application with React Testing Library
In the same way, let's write tests for React applications. For this, we will be using React Testing Library.
The
@testing-library
family of packages helps you test UI components in a user-centric way.
Step 1: Install packages
To start writing tests in React, you need React Testing Library. But these packages come pre-installed with create-react-app (CRA). If you haven't used CRA you can install them like this.
bash1npm install --save-dev @testing-library/react @testing-library/jest-dom @testing-library/user-event
Step 2: Create test files
It's a good practice to create a test file for every component.
We have 2 components that need to be tested - <App />
and <Login />
. So create App.test.js
and Login.test.js
in the src/
folder. But for now create only the App.test.js
file.
Step 3: Start writing tests
Write your first test.
In the App.jsx
file add the data-testid={"app-header-heading"}
attribute to the main heading element. Later, while writing tests we can refer to this element with this id.
html1<h1 data-testid="app-header-heading">Productivity Tracker</h1>
Now, open the App.test.js
file and paste the below code.
javascript1import { render, screen } from "@testing-library/react"; 2 3import App from "./App"; 4 5/* This is a test that is testing the App component. 6 * It is testing that the heading is correct. */ 7describe("App", () => { 8 it("should have exact heading", () => { 9 /* Rendering the App component. */ 10 render(<App />); 11 12 /* Getting the element with the test id of "app-header-heading". */ 13 const mainHeading = screen.getByTestId("app-header-heading"); 14 15 /* Checking that the innerHTML of the element with the test id of "app-header-heading" is equal to 16 "Productivity Tracker". */ 17 expect(mainHeading.innerHTML).toBe("Productivity Tracker"); 18 }); 19});
In the above code,
- We first imported the required packages -
render
andscreen
from@testing-library/react
and then imported the component that needs to be tested. - To describe the unit test, we use the same syntax as in express. Most React developers, by convention, put the component name in the first argument. The actual test is written as a callback function in the second argument.
- In
it
, we write the actual test code. Write the expected result in the first argument, and then in the second argument, write a callback function that contains the test code. - Rest of the code is self-explanatory and you can also understand from the comments. We are expecting the App component should contain a heading called "Productivity Tracker".
This is the general flow of writing a test for a React component:
- Render the component -> Write elements you want to interact with -> Interact with those elements -> Assert that the outcomes are as expected.
Now run the npm test
command. You will see the results.
Please refer to official docs to learn more about React Testing Library and write your own tests.
As a challenge, try to implement the "Add activity" flow. In the next article, we will learn some DevOps stuff.
Also read,
- Part 1: Let's build and deploy a full stack MERN web application
- Part 2: Adding Authentication to full stack MERN web application
Feel free to ask your doubts in the comments.