One, the introduction

Front-end development is very rapid in recent years, the function of our system is becoming more and more complex, which puts forward higher requirements for our front-end engineering ability, hear engineering, everyone’s first reaction must be high quality code design and high quality code implementation.

But in fact, front-end automation test is also a very important part of front-end engineering.

Two, Jest basic introduction

When a typical front-end hears about automated tests, the first reaction might be: I’ve worked for years and never written a test. Does this work?

A: Very useful

If you open GitHub and take a look at the source code for popular open source libraries or frameworks, you’ll find that all of them contain a lot of automated test code. Like ANTD, Lodash, vue, React, Echarts, Redux…

Open source tools need stability, and there’s no better way to introduce front-end automation to provide stability for open source projects.

Three, learning the premise

To read this article, you need the following knowledge:

  • Js, ES6 basic syntax
  • Knowledge of Node and NPM
  • Git related operations
  • React or Vue. Know at least one
  • Status management tools. Know at least one

Iv. Background and principle

Start by creating a math.js file in any directory. Suppose this file is a math library that defines two functions, addition and subtraction:

// math.js

function add(a, b) {
  return a + b;
}

function minus(a, b) {
  return a - b;
}
Copy the code

At this point we can use the math library in the business code.

However, if we make a mistake in the minus function, and replace the subtraction with multiplication, it will cause unexpected bugs if we directly use this method in business code.

In this case, we need to automate the testing of math.js, a common library, and make sure it works before the business components call it, so that we don’t have too many bugs.

We can do this:

Create a math.test.js file in this directory and write a little test code:

const result = add(3.7);
const expect = 10;

if(result ! == expect) {throw new Error(3 plus 7 is going to be equal to${expect}It turned out to be${result}`);
}
Copy the code
const result = minus(3.3);
const expect = 0;

if(result ! == expect) {throw new Error(3 minus 3 is going to be equal to${expect}It turned out to be${result}`);
}
Copy the code

When we run this code, we see that no exceptions are thrown, indicating that both test cases pass.

This is the original prototype of automated testing.

Then we thought about how to simplify the pile of code into a common function, like this:

// Test whether 3 + 3 equals 6
expect(add(3.3)).toBe(6);

// Test whether 3-3 equals 0
expect(minus(3.3)).toBe(0);
Copy the code

Expect method implementation:

function expect(result) {
  return {
    toBe(actual) {
      if(result ! == actual) {throw new Error("The expected value is not equal to the actual value"); }}}; }Copy the code

When we run this code, we see that no exceptions are thrown, indicating that both test cases pass.

We implement the Expect function, but the errors are always the same. We don’t know which method is wrong, and then we think we need to improve the Expect method further. If we can wrap another layer around the Expect method, we can pass some extra content. Like creating something like this:

test("Test addition 3 + 3".() = > {
  expect(add(3.3)).toBe(6);
});

test("Test subtraction 3-3".() = > {
  expect(minus(3.3)).toBe(0);
});
Copy the code

After this encapsulation, we can both test and get a description of the test.

Test method implementation:

function test(desc, fn) {
  try {
    fn();
    console.log(`${desc}Pass the test ');
  } catch {
    console.log(`${desc}Failed the test); }}Copy the code

So what exactly is front-end automation testing?

Answer: actually is to write a section of other JS code for testing, through the test code to run the business code, judge whether the actual results meet the expected results, if yes, there is no problem, if not, there is a problem.

The expect method and test method implemented above are in fact identical to the syntax in the mainstream front-end automation test framework JEST. So the sample code above can be understood as the underlying implementation of JEST.

Introduction to Jest framework

In the above section, we implemented the Expect and Test methods.

In the process of actual project automation test, if only these two methods, obviously, is not enough, at this time, we need to expand the previous method, at the same time, there are many automatic mechanisms need to be integrated.

At that time! Jest debuts!

5.1 Introduction to Jest Framework

A good automated testing framework should stand out in three areas:

  • Performance is good
  • The function is all ready
  • Good ease of use

Jest does all three very well (as do mainstream front-end testing frameworks like Jasmine and Mocha, of course).

All of the major front-end testing frameworks use similar methods and principles. In fact, once we learn one, we can easily move on to the others, so this article will focus on Jest.

So what exactly is Jest useful for?

Here are the answers:

  • Fast (can automatically monitor the modified code, does not repeat the test)
  • Simple API and few in number
  • Easy to configure
  • Good isolation,
  • Monitoring mode
  • IDE integration (e.g. Vs Code)
  • Snapshot test
  • Multi-project parallelism
  • coverage
  • The Mock rich
  • Good support for new technology

5.2 Modifying an Automated Test Example Using Jest

In the above section, we wrote some test code ourselves to test the Math library, and now we’re going to retest Math using Jest.

First open the editor and go to the console (to avoid unnecessary trouble, I suggest you use VS Code as I did) :

Step 1: Initialize the NPM environment:

npm init

If you see a package.json file in your directory, this is a standard NPM package.

Step 2: Install dependencies

npm install jest -D
Copy the code

(Note: -d stands for — save-dev)

If you see Jest in the package.json file and the node_modules folder in the directory, the installation is successful.

Step 3: Modify the previous code

First, delete the test and Expect methods we wrote earlier, because Jest comes with both methods and we don’t need to implement them manually. Then, using modular standards, export the methods in Math.js:

// math.js

function add(a, b) {
  return a + b;
}

function minus(a, b) {
  return a - b;
}

function multi(a, b) {
  return a * b;
}

module.exports = { add, minus, multi };
Copy the code

Then introduce these methods in math.test.js:

// math.test.js

const { add, minus, multi } = require("./math");

test("Test addition 3 + 3".() = > {
  expect(add(3.3)).toBe(6);
});

test("Test subtraction 3-3".() = > {
  expect(minus(3.3)).toBe(0);
});

test("Test multiplication 3 by 3".() = > {
  expect(multi(3.3)).toBe(9);
});
Copy the code

Step 3: Configure the package.json file

Add the following code to package.json:

"scripts": {
  "test": "jest"
},
Copy the code

Step 4: Execute the test code

After the configuration is complete, run YARN test or NPM run test. The console displays the following information:

Suppose at this point, we change the result of the third test case to an error value:

// math.test.js

const { add, minus, multi } = require("./math");

test("Test addition 3 + 3".() = > {
  expect(add(3.3)).toBe(6);
});

test("Test subtraction 3-3".() = > {
  expect(minus(3.3)).toBe(0);
});

test("Test multiplication 3 by 3".() = > {
  expect(multi(3.3)).toBe(10); // Error result
});
Copy the code

Run yarn test or NPM run test again. The following information is displayed:

As you can see, the console reported an error:

The multiplication test failed. Expected was 10, but returned was Received 9.

We then change the test result to the correct value and re-execute the command line to get the correct test result.

At this point, the statementJestThis framework has been used correctly.

5.3 Simple Configuration of Jest

Like Webpack, Jest has a default configuration, which can be initialized by running the NPX Jest –init command. When you run the command line, some options pop up, such as:

  • Whether to enabletypescript
  • Whether to generateCoverage report
  • chooseThe node environmentorBrowser environment
  • After the testing is done, if you need to do somecleanup

You can select yes or no based on your actual situation.

When all the selections are complete, you can see that there is an additional jest. Config.js file in the directory, indicating that the initial configuration is complete. Click here to view the configuration documentation.

1. Code test coverage

When we type NPX jest –coverage in the console, the console prints the following:

This is a description of test coverage, this is a graph that tells us that all files are tested 100%.

If executing the NPX jest –coverage command is too much of a hassle, you can also add the following code to package.json:

"scripts": {
  "test": "jest"."coverage": "jest --coverage"
},
Copy the code

You can then execute NPM run coverage, which has the same effect as NPX jest –coverage.

In addition to the chart information available to the console, you can also see a generated Coverage directory under the current directory:

Inside there is an lCOV-Report folder with an index. HTML. If we open it, we’ll see:

As you can see, this is a nice HTML page generated by Jest that tells us exactly how much code our tests cover for each file. As you can see, the code in math.js is 100% covered.

For example, code coverage can mean a lot of things.

To sum up:Test coverageThat’s what we wroteThe test codeTo the originalFunction codeThe proportion of.

Cover Directory in the configuration file is the name of the folder where the code test coverage report is generated:

module.exports = {
  coverageDirectory: "abc"
};
Copy the code

For example, if we give cover Directory a value of ABC, the generated test report will be under directory ABC.

2. Change commonJS to ES Module

Math.js and math.test.js to the es Module specification:

// math.js

export function add(a, b) {
  return a + b;
}

export function minus(a, b) {
  return a - b;
}

export function multi(a, b) {
  return a * b;
}
Copy the code
// math.test.js

import { add, minus, multi } from "./math";

test("Test addition 3 + 3".() = > {
  expect(add(3.3)).toBe(6);
});

test("Test subtraction 3-3".() = > {
  expect(minus(3.3)).toBe(0);
});

test("Test multiplication 3 by 3".() = > {
  expect(multi(3.3)).toBe(9);
});

Copy the code

Then run the NPM run test and you will find that the console reported an error.

To solve this problem, we need to use Babel to convert the code. If we want Babel to support ESM, we need three packages:

  • @babel/core: The Babel core library
  • @babel/preset-env: library for ES syntax conversions
  • babel-jest: a library that communicates with Jest to check whether the above two dependencies are installed

But since babel-Jest is installed by default when we install Jest, we only need to install the remaining two packages.

Install dependencies:

npm install @babel/core @babel/preset-env -D
Copy the code

(Note: -d stands for — save-dev)

After the installation is complete, create a.babelrc configuration file in the root directory of the project:

Once created, add the following code to.babelrc:

{
  "presets": [["@babel/preset-env", { "targets": { "node": "current"}}}]]Copy the code

After saving, rerun the NPM run test to execute the test case.

5.4 Matchers in Jest

As we all know, in the previous code, we used test, Expect, and toBe methods when writing a test case.

Expect the result of one value to match another value.

Such as:

expect(1 + 1).toBe(2);
Copy the code

This example can be translated as: the result of expecting 1 + 1 is 2, where the toBe is a matcher that determines whether the received value is equal to the expected value.

In fact, there are many matchers in Jest besides toBe:

  • toBe: tests whether the values of two objects are equal, similar to the === operator in JS
// toBe

test("Testing the commutative law of addition.".() = > {
  for (let x = 0; x <= 10; x++) {
    for (let y = 0; y <= 10; y++) { expect(x + y).toBe(y + x); }}});Copy the code
  • toEqual: tests whether the original value of two objects is equal, only checking the content, not the reference
// toEqual

const can1 = { value: "hello" };
const can2 = { value: "hello" };

test("Test whether the contents of CAN1 and can2 are equal".() = > {
  expect(can1).toEqual(can2);
});
Copy the code
  • toBeNull: Tests whether the value of the object is null.toBe(null)
// toBeNull

const value = null;

test("Test value null".() = > {
  expect(value).toBeNull();
});
Copy the code
  • toBeUndefined: Tests whether the value of the object is undefined.toBe(undefined)
// toBeUndefined

const value = undefined;

test("Test value undefined".() = > {
  expect(value).toBeUndefined();
});
Copy the code
  • toBeDefined: Tests whether a value is defined, except undefined
// toBeDefined

const value = 1;

test("Is the test value defined?".() = > {
  expect(value).toBeDefined();
});
Copy the code
  • toBeTruthy: Checks if the value is true after being converted to a Boolean value
  • toBeFalsy: Checks if the value is false after being converted to a Boolean value
/ / toBeTruthy, toBeFalsy

test("Test for true".() = > {
  expect(0).toBeTruthy(); / / not through
  expect("").toBeTruthy(); / / not through
  expect(null).toBeTruthy(); / / not through
  expect(false).toBeTruthy(); / / not through
  expect(undefined).toBeTruthy(); / / not through
});

test("Test for false value".() = > {
  expect(0).toBeFalsy(); / / by
  expect("").toBeFalsy(); / / by
  expect(null).toBeFalsy(); / / by
  expect(false).toBeFalsy(); / / by
  expect(undefined).toBeFalsy(); / / by
});
Copy the code
  • not: fetch the reverse matcher, equivalent to js! The operator
// not

test("Is the test not aaa?".() = > {
  expect("hello").not.toBe("aaa");
});

test("Test value not null".() = > {
  expect([]).not.toBeNull();
});

test("Test value not undefined".() = > {
  expect({}).not.toBeUndefined();
});
Copy the code
  • toBeGreaterThan: Checks whether the received value is greater than the expected value
// toBeGreaterThan

test("Test if 10 is greater than 9.".() = > {
  expect(10).toBeGreaterThan(9);
});
Copy the code
  • toBeLessThan: Checks whether the received value is less than the expected value
// toBeLessThan

test("Test if 10 is less than 20".() = > {
  expect(10).toBeLessThan(20);
});
Copy the code
  • toBeGreaterThanOrEqual: Checks whether the received value is greater than or equal to the expected value
// toBeGreaterThanOrEqual

test("Test if 10 is greater than or equal to 10.".() = > {
  expect(10).toBeGreaterThanOrEqual(10);
});
Copy the code
  • toBeLessThanOrEqual: Checks whether the received value is less than or equal to the expected value
// toBeLessThanOrEqual

test("Test if 10 is less than or equal to 10.".() = > {
  expect(10).toBeLessThanOrEqual(10);
});
Copy the code
  • toBeCloseTo: Checks whether floating point numbers are close (approximately equal)
// toBeCloseTo

test("Test whether 0.1 + 0.2 equals 0.3.".() = > {
  expect(0.1 + 0.2).toBe(0.3); / / not through
  expect(0.1 + 0.2).toBeCloseTo(0.3); / / by
});
Copy the code
  • toMatch: Checks whether the value matches a string or a re
// toMatch

test("Test if string contains baidu".() = > {
  expect("www.baidu.com").toMatch("baidu");
  expect("www.baidu.com").toMatch(/baidu/);
});
Copy the code
  • toContainCheck if an item is included in the array (similar to the includes method in JS)
// toContain

test("Test if the list contains 3".() = > {
  const list = [1.2.3];
  expect(list).toContain(3);
});
Copy the code
  • toThrow: tests whether an exception is thrown when the function is called
// toThrow

const fn1 = () = > {
  console.log("hello");
};

const fn2 = () = > {
  throw new Error("this is a new err");
};

test("Test fn1, fn2 call for exception".() = > {
  expect(fn1).toThrow(); / / not through
  expect(fn2).toThrow(); / / by
});
Copy the code

In Jest, there are many matchers available in addition to the common matchers listed above. You don’t need to memorize all of them, just know a few that are commonly used. If you want to learn more, you can check out the official documentation here.

5.5 Using the Jest Command Line Tool

Json file and add the –watchAll argument to the test command line:

"scripts": {
  "test": "jest --watchAll"."coverage": "jest --coverage"
},
Copy the code

The advantage of this is that we don’t need to execute the NPM run test every time. After the first execution, the process will automatically listen for changes in the test case, and if the test case code changes, it will be executed automatically.

Run NPM run test and you will see the following options:

As you can see, there are many shortcut keys when entering the Watch mode. The functions of these shortcut keys are as follows:

  • According to thefKey: Run only failed test cases
  • According to theoKey: Run only the test cases that have changed
  • According to thepKey: Press file name regex mode to filter
  • According to thetKey: Filter by test name regex mode
  • According to theqKey: Exit the monitoring mode
  • According to theEnterKey: Triggers the test case

There are many command line tools in Jest, the reasonable use of command line, will make our test become more flexible, easy to use.

5.6 Testing asynchronous code

To test asynchronous code, we need to write asynchronous code and install AXIos. Once axiOS is installed, we can send requests from both node and browser environments:

npm install axios --save
Copy the code

Start by preparing an interface that returns a JSON object:

{
  "success"true
}
Copy the code

Then import Axios, write an asynchronous function, and export:

import axios from "axios";

export const getData = () = > {
  return axios.get("http://www.dell-lee.com/react/api/demo.json");
};
Copy the code

When testing asynchronous functions, write:

Method 1: Promises

Return value for successful test:

import { getData } from "./index";

test("Test getData returns {success: true}".() = > {
  return getData().then(res= > {
    expect(res.data).toEqual({ success: true });
  });
});
Copy the code

When the test returns a successful value, the test case needs to be executed in.then, and must be preceded by a return that returns the entire asynchronous code, otherwise the test is meaningless.

If you omit this return, your test will be completed before the promise resolution returned by the asynchronous function.

Return value for test failure:

import { getData } from "./index";

test("Test getData return value containing 404".() = > {
  return getData().catch(err= > {
    expect.assertions(1);
    expect(err.toString()).toMatch("404");
  });
});
Copy the code

If the test fails and returns a value, you need to execute the test case inside the.catch, but if the request succeeds and the asynchronous function walks in, then the.catch callback is not executed, so the test case does nothing and still passes the test.

To fix this, preceded by expect. Assertions (1); , to assert that the test case calls a certain amount of Expect. If not called enough times, the test case will not pass.

? Aloft /. Rejects

Return value for successful test:

import { getData } from "./index";

test("Test getData returns {success: true}".() = > {
  return expect(getData()).resolves.toEqual({ success: true });
});
Copy the code

At this point, Jest will wait for Promises function to be resolved.

If Promises function is rejected, the test case automatically fails.

Return value for test failure:

import { getData } from "./index";

test("Test getData return value containing 404".() = > {
  return expect(getData()).rejects.toMatch("404");
});
Copy the code

In the same way, you can use one in expect statements. Rejects, Jest will wait for Promises to be rejected.

If Promises functions are resolved, the test case will automatically fail.

Async/Await

Return value for successful test:

import { getData } from "./index";

test("Test getData returns {success: true}".async() = > {const { data } = await getData();
  expect(data).toEqual({ success: true });
});
Copy the code

You can also pass the async keyword to the callback function of the test method and call the asynchronous function to get the return value using the await keyword before the Expect statement, just as you would in normal JS code.

Return value for test failure:

import { getData } from "./index";

test("Test getData return value containing 404".async () => {
  expect.assertions(1);
  try {
    await getData();
  } catch (err) {
    expect(err.message).toMatch("404"); }});Copy the code

Similarly, if you need to test for a failed return value, you’ll need to use the try/catch statement in native JS to catch the exception, along with expect. Assertions (1); A certain amount of Expect is invoked to verify this test case. If not enough calls are made, no exceptions are caught and the test case fails.

Convergency: Async, Await, and.rejects are considered together

import { getData } from "./index";

test("Test getData returned successfully".async() = > {await expect(getData()).resolves.toEqual({ success: true });
});

test("Failed to return test getData".async() = > {await expect(getData()).rejects.toMatch("404");
});
Copy the code

In this case, async and await use the same syntactic sugar as the Promise example.

5.7 Hook functions in Jest

Often, when writing tests, you need to do some initialization before the test runs and some completion after the test runs. Jest provides hook functions to handle this problem.

We use a counter to learn about hook functions.

First define a class in index.js and export it:

// index.js

class Counter {
  constructor() {
    this.number = 0;
  }
  add() {
    this.number++;
  }
  minus() {
    this.number--; }}export default Counter;
Copy the code

As you can see, this is a class defined using ES6 syntax:

  • Defined at instantiation timenumber, and the initial value is0
  • Static methods are definedaddThe result isnumberadd1
  • Static methods are definedminusThe result isnumberReduction of1

Then add this class to index.test.js and test the add method:

// index.test.js

import Counter from "./index";

const counter = new Counter();

test("Test the add method for Counter".() = > {
  expect(counter.number).toBe(0);
  counter.add();
  expect(counter.number).toBe(1);
});
Copy the code

Then run the test case and pass with flying colors.

Then continue adding the test case for the minus method:

// index.test.js

import Counter from "./index";

const counter = new Counter();

test("Test the add method for Counter".() = > {
  expect(counter.number).toBe(0);
  counter.add();
  expect(counter.number).toBe(1);
});

test("Test Counter minus method".() = > {
  expect(counter.number).toBe(0);
  counter.minus();
  expect(counter.number).toBe(-1);
});
Copy the code

Then run the test case with the result:

  • testaddMethod: Through
  • testminusMethod: Do not pass

The reasons for this are as follows:

Our instantiation process is written on the test exception side. All test cases share one counter instance, so we have modified the number attribute of the instance when testing the Add method. The test case of the minus method failed.

The solution:

We can write the instantiation in each test case, creating a new counter instance for each test. So you don’t share a counter, and you don’t affect other instances.

But in general, we don’t do that, because it would be a hassle to create one counter instance at a time if we have a lot of test cases, so if we have 1000 counter test cases in the current file, then we need to create 1000 counter instances.

This time! Hook functions come in handy!

// index.test.js

import Counter from "./index";

let counter = null;

beforeEach(() = > {
  counter = new Counter();
});

test("Test the add method for counter".() = > {
  expect(counter.number).toBe(0);
  counter.add();
  expect(counter.number).toBe(1);
});

test("Test counter minus method".() = > {
  expect(counter.number).toBe(0);
  counter.minus();
  expect(counter.number).toBe(-1);
});
Copy the code

The test case is then run and the result passes.

BeforeEach () executes the callback function beforeEach test case is executed. You can use this if you need to do repetitive work on multiple tests before the test starts, such as initialize state.

Jest actually has four hook functions:

  • beforeAll: called before all test cases are executed (called once)
  • afterAll: called after all test cases are executed (called once)
  • beforeEach: called before each test case execution (called multiple times)
  • afterEach: called after each test case execution (called multiple times)

5.8 Scope of hook functions

Before we get to scope, we need to know a little bit about Scoping

By default, beforeAll and afterAll are applied to each test case in the file.

In fact, test cases can also be grouped using the Describe method. When they are in describe, beforeAll and afterAll apply only to the test cases in the current group.

// index.test.js

describe("Test Group 1".() = > {
  beforeAll(() = > {
    console.log("Test Group 1 - beforeAll");
  });
  afterAll(() = > {
    console.log("Test Group 1 - afterAll");
  });
  test("Test".() = > {
    console.log("Test Group 1 Test");
    expect(1 + 1).toBe(2);
  });
});

describe("Test Group 2".() = > {
  beforeAll(() = > {
    console.log("Test Group 2 - beforeAll");
  });
  afterAll(() = > {
    console.log("Test Group 2 - afterAll");
  });
  test("Test".() = > {
    console.log("Test Group 2 Test");
    expect(1 + 1).toBe(2);
  });
});
Copy the code

Execute the above code:

By default, Jest will run all test groups consecutively in the order described, waiting for each test to complete before continuing.

Note that:

  • If we don’t group it, we’re writing a layer on the outsidedescribe.
  • In addition to that, actually, indescribeYou can nest it insidedescribe, like the following:
// index.test.js

describe("First floor".() = > {
  beforeAll(() = > console.log("Layer 1 - beforeAll"));
  describe("The second layer".() = > {
    beforeAll(() = > console.log("Layer 2 - beforeAll"));
    describe("The third layer".() = > {
      beforeAll(() = > console.log("Layer 3 - beforeAll"));
      test("Test".() = > {
        console.log("Test");
        expect("hello" + "" + "world").toBe("hello world");
      });
    });
  });
});
Copy the code

Run the test and you can see the following:

So some conclusions can be drawn:

  • eachdescribeYou can have your ownHook function
  • eachdescribeThey have their ownscope
  • eachHook functionIt also has its own scope, which is the current onedescribe
  • eachdescribeThe inside of theHook functionTo oneselfscopeAll of the followingThe test caseIs to take effect
  • ifdescribeIs multi-level nested, so the order of test case execution isFrom outside to inside

One last point:

If there are multiple test cases, but only one test case is to run, it is often not the best choice to comment out the other test cases, we can use the. Only syntax to execute:

// index.test.js

test.only("This test will be performed.".() = > {
  expect("A").toBe("A");
});

test("This test will be skipped.".() = > {
  expect("B").toBe("B");
});
Copy the code

This allows you to run a single test case, and other tests are skipped.

5.9 Mock in Jest

1. Mock functions

In our project, methods in one module usually call methods in another module.

When you’re testing, you don’t need to care about the execution and results of a method, you just need to know that it’s called correctly, and that’s where Jest’s Mock function comes in.

Mock functions provide three features that are useful when writing test code:

  • Capture function calls
  • Sets the return value of the function
  • Change the internal implementation of a function

1.1 Test whether the function is called normally

First define a function to execute the incoming callback, then export:

// index.js

export const runCallback = callback= > {
  callback();
};
Copy the code

Then we need to test it like this:

// index.test.js

import { runCallback } from "./index";

test("Test runCallback".() = > {
  const func = jest.fn(); // Generate mock functions to capture function calls
  runCallback(func); // Call the mock function
  expect(func).toBeCalled(); // The toBeCalled match is used to check whether the function is called
});
Copy the code

1.2 Test whether the number of function calls is correct

First define a function to execute the incoming callback, then export:

// index.js

export const runCallback = callback= > {
  callback();
};
Copy the code

Then we need to test it like this:

// index.test.js

import { runCallback } from "./index";

test("Number of test calls".() = > {
  const func = jest.fn(); // Generate mock functions to capture function calls
  runCallback(func); // Call the mock function for the first time
  runCallback(func); // Call the mock function for the second time
  runCallback(func); // Call the mock function for the third time
  expect(func.mock.calls.length).toBe(3); // Check if the function is called three times
});
Copy the code

1.3 Test whether the function returns undefined

First define a function to execute the incoming callback, then export:

// index.js

export const runCallback = callback= > {
  callback();
};
Copy the code

Then we need to test it like this:

// index.test.js

import { runCallback } from "./index";

test("Test Return value".() = > {
 const func = jest.fn(); // Generate mock functions to capture function calls
  expect(runCallback(func)).toBeUndefined(); // Check whether the function returns undefined
});
Copy the code

Chapter Summary

This article covers some of the basics of JEST.

You can read the official documentation to learn jEST. After learning the basics, you are ready to learn JEST.

If I have time later, I’ll write an article on advanced uses of JEST, such as snapshot snapshot testing, Mock Timers, testing ES6 classes, and testing DOM nodes.