jest基础

  • jest官网
  • 匹配器
  • 命令行工具
  • 测试异步
  • 钩子函数
  • mock
  • 快照
  • dom

匹配器

jest默认环境是node

如果想在jest环境使用esmodule,需要借助@babel/core转化工具, @babel/preset-env指明如何转化

{
    "presets": [
        [
            "@babel/preset-env",
            {
                "targets": {
                    "node": "current"
                }
            }
        ]
    ]
}
只需要一个项目中安装jest,然后执行 npx jest --watch,jest就会自动将所有的.test.js结尾的文件中的测试用例执行。

jest提供了很多匹配器,最基本的使用

test('测试加法', ()=>{
     // toBe 类似 Object.is(a,b),适用于普通值
    expect(10).toBe(10)
 })


 test('测试内容', ()=>{
   // 匹配对象使用toEqual,递归遍历使用toBe, 加上.not就是取反
   expect({a: 1}).toEqual({a: 1})
 })

如toBe, toEqual,等等还有很多。当执行npx jest --watch的时候,jest就会执行这些测试用例并返回结果。

PASS  ./9.test.js
  √ 测试加法 (2ms)
  √ 测试内容 (1ms)

Snapshot Summary
 › 1 snapshot file obsolete from 1 test suite. To remove it, press `u`.
   ↳   • __snapshots__学习\demo.test.js.snap

Test Suites: 1 passed, 1 total
Tests:       2 passed, 2 total
Snapshots:   1 file obsolete, 0 total
Time:        0.987s
Ran all test suites related to changed files.

测试通过。

还有更多的适配器如:

expect(10).toBeGreaterThan(9) //大于9
 expect( 0.1 + 0.2).toBeCloseTo(0.3) //解决精度问题
 expect('abcd').toMatch(/ab/) //匹配字符串
 expect([1,2]).toContaine(1) //数组或者集合是否包含某个数
 expect(()=>{throw new Error('123')}).toThrow() //抛出异常, 没有toThrow不会抛出异常
 expect(()=>{throw new Error('123')}).toThrow('123') //是否抛出123异常,是的话测试通过

更多适配器可以查看官网。

命令行工具

Test Suites: 1 passed, 1 total
Tests:       1 passed, 1 total
Snapshots:   1 file obsolete, 0 total
Time:        2.302s, estimated 3s
Ran all test suites related to changed files.

Watch Usage: Press w to show more.

执行完后,下面会提示输入W显示更多指令

Watch Usage
 › Press a to run all tests.
 › Press f to run only failed tests.
 › Press p to filter by a filename regex pattern.
 › Press t to filter by a test name regex pattern.
 › Press u to update failing snapshots.
 › Press q to quit watch mode.
 › Press Enter to trigger a test run.
  • a执行所有测试 默认 jest --watchAll 就是打开了a模式
  • f只测试失败的测试
  • p 根据文件名字正则表达式,匹配更改的test
  • t 根据文件名字,过滤某些test
  • u 更改失败的快照(后面会讲)
  • o 配合git,只执行有修改的文件 默认–watch就是打开了。

测试异步

// 默契请求的方法
const fetchData = () =>
  new Promise((resolve, reject) => {
    setTimeout(() => {
        resolve({ success: true });
    }, 2000);
  });

export { fetchData };

因为默认情况下,Jest 测试在执行结束后完成。

test('the data is peanut butter', () => {
  fetchData().then(res=>{
      expect(res).toEqual({success: true})
  })
});

那么其实不管输入什么,该test就会执行正确。

正确测试异步方法:

1 done
test("fetchData 返回结果为{success: true}", (done) => {
  fetchData()
    .then((res) => {
      expect(res).toEqual({ success: true });
      done();
    })
    .catch((err) => {
      done(err);
    });
});

只有调用done,才会正确测试该test。

2 promise

正常使用promise.then,但是一定要return。而且可以使用expect.assertions(1)表示至少执行一次expect。

因为有些场景,我们只俘获了catch,并无俘获then,那么Promsie成功的时候,这个测试将会没有执行。所以需要这个来确定至少执行一次。否则失败。

test("fetchData 返回结果为{success: true}", () => {
  expect.assertions(1) //至少执行一次expect,如果没跑也会报错
  return fetchData()
    .then((res) => {
      expect(res).toEqual({ success: true });
    })
    .catch((err) => {
      expect(err).toMatch("error");
    });
});
3 resolves/reject
// 可以.resolves在您的期望语句中使用匹配器,Jest 将等待该承诺解决。如果 promise 被拒绝,测试将自动失败。
test("fetchData 返回结果为{success: true}", () => {
  return expect(fetchData()).resolves.toMatchObject({
    success: true,
  });
  // 期望一个 Promise 被拒绝,使用.rejects匹配器。如果promise被拒绝,就执行,如果promise成功执行,则该jest默认失败
  return expect(fetchData()).rejects.toThrow()
});
4 asycn/await

记得错误处理

test("fetchData 返回结果为{success: true}", async () => {
  try {
    const data = await fetchData();
    expect(data).toEqual({
      success: true,
    });
  } catch (err) {
    // 抛出异常
    expect(err).toThrow()
  }
});

钩子函数

jest在执行test用例的时候,提供了很多钩子,用来初始化一些东西或者做一些销毁工作,如执行前调用,执行后调用等等。如

beforeAll(()=>{
    console.log('beforeAll');
})

afterAll(()=>{
    console.log('afterAll');
})

他们分别会在所有的钩子执行前执行和执行后执行。

beforeEach(()=>{
    console.log('beforeEach');
})

afterEach(()=>{
    console.log('afterEach');
})

beforeEach和afterEach分别会在每个test执行之前和之后运行。

分组

相同的意义的测试用例可以写在同一个分组中,相当于ts的namespace,jest提供了describe

describe("测试加法", () => {
  beforeAll(() => {
    console.log("内部 eforeAll");
  });
  afterAll(() => {
    console.log("内部 afterAll");
  });
  afterEach(() => {
    console.log("内部 afterEach");
  });
  beforeEach(() => {
    console.log("测试加法 beforeEach");
    counter = new Counter();
  });

  test("test Counter addOne", () => {
    counter.addOne();
    expect(counter.number).toBe(1);
    console.log("addOne");
  });
});

如,describe内部的钩子,只会在内部执行。而整个test文件,可以看作是外层包装了一个descrbie(‘xx’,()=>{// test文件的内容})

那么desceibe内部的钩子和外部的钩子,执行顺序是怎样呢?

以下面的为例子

beforeAll(() => {
    console.log("外部 beforeAll");
  });
  
  afterAll(() => {
    console.log("afterAll");
  });
  beforeEach(() => {
    console.log("外部 beforeEach");
  });
  afterEach(() => {
    console.log("外部 afterEach");
  });
  describe("测试加法", () => {
    beforeAll(() => {
      console.log("内部 eforeAll");
    });
    afterAll(() => {
      console.log("内部 afterAll");
    });
    afterEach(() => {
      console.log("内部 afterEach");
    });
    beforeEach(() => {
      console.log("测试加法 beforeEach");
    });
  
    test("test Counter addOne", () => {
      console.log("descrbie内部的测试 addOne");
    });
  });

执行顺序就是

外部 beforeAll
内部 beforeAll
外部 beforeEach
测试加法 beforeEach
descrbie内部的测试 addOne
内部 afterEach
外部 afterEach
内部 afterAll
外部 afterAll

技巧就是:外部beforeAl > 内部beforeAll > 外部beforeEach > 内部beforeEach > test执行 > 内部afterEach > 外部 afterEach > 内部 afterAll > 外部 afterAll

mock

jest提供了mock的功能,可以模拟请求数据的函数,模拟返回的数据。

其中

jest.fn 返回一个mock函数,1 捕获函数的调用和返回结果, 2 可以自由地设置返回结果 3模拟axios请求返回特定数据
test("测试", () => {
  const func = jest.fn((a) => {
    return a + 4;
  });

  // func.xxx有很多语法。
  // func.mockImplementtation((a)=>{return a+4})跟jest.fn(fn)的fn一样,函数执行的时候的逻辑
  // func.mockImplementationOnce(()=>{}) 只执行一次。

  // mockReturnValue('a') 每次调用都返回a
  func.mockReturnValueOnce("11111").mockReturnValueOnce("2222"); //第1,2次执行func的时候,返回了1111, 2222
  func(1);
  func(2);
  func(3);
  func(4);

  // func是否被执行过
  //expect(func).toBeCalled()
  //expect(func.mock.calls[0]).toEqual([1])
  expect(func.mock.results);
  console.log(func.mock);

  /** 
     console.log(func.mock);
     {可以通过func.mock.calls.length判断调用多少次,以及传参, fn.results判断返回值
      calls: [ [ 1 ], [ 2 ], [ 3 ], [ 4 ] ],
      instances: [ undefined, undefined, undefined, undefined ], func运行的时候,this指向
      invocationCallOrder: [ 1, 2, 3, 4 ],
      results: [
        { type: 'return', value: '11111' },
        { type: 'return', value: '2222' },
        { type: 'return', value: 7 },
        { type: 'return', value: 8 }
      ]
    }

     
});

func执行的时候,实际上是jest.fn()在执行,func.mock提供了很多内部,包括调用多少次,每次的入参,每次的返回等等。并且可以通过func.mockReturnValueOnce(‘xx’)默契一次func调用的时候的返回。

jest.mock可以改变函数的实现
// 3 改变函数的内部实现
import axios from 'axios'
jest.mock('axios')
test.only('测试 axios',async()=>{
    axios.get.mockResolvedValue({data: '123'}) //模式请求返回数据
    console.log('axios', axios);
    await axios.get('xxx').then(data=>{
        console.log(data);
        expect(data).toEqual({data: '123'}) //断言
    })
})
jest.mock(‘axios’)可以将axios上的所有属性变成jest.fn。打印出来的axios是
console.log 9.test.js:5
    axios <ref *1> [Function: wrap] {
      _isMockFunction: true,
      getMockImplementation: [Function (anonymous)],
      mock: [Getter/Setter],
      mockClear: [Function (anonymous)],
      mockReset: [Function (anonymous)],
      mockRestore: [Function (anonymous)],
      mockReturnValueOnce: [Function (anonymous)],
      mockResolvedValueOnce: [Function (anonymous)],
      mockRejectedValueOnce: [Function (anonymous)],
      ....
    }

axios已经被改造了。可以通过axios.get.mockResolvedValue({data: ‘123’})模式get请求返回的数据,测试如下:

PASS  ./9.test.js
  √ 测试 axios (10ms)

  console.log 9.test.js:6
    { data: '123' }

Snapshot Summary
 › 1 snapshot file obsolete from 1 test suite. To remove it, press `u`.
   ↳   • __snapshots__学习\demo.test.js.snap

Test Suites: 1 passed, 1 total
Tests:       1 passed, 1 total
Snapshots:   1 file obsolete, 0 total
Time:        2.568s, estimated 3s
Ran all test suites related to changed files.

测试通过,get请求返回的data就是{data: ‘123’}

自己改造函数

jest执行测试用例的时候,并不希望真正的发送请求。所以可以通过jest.mock模拟。

假设fetchData是代码中真正发送请求的函数,我们想验证这个函数的正确性,又不想发送请求。可以在根目录下创建__mock__文件夹,命名相同的文件名字,如demo.js然后实现

// __mock__/demo.js
const fetchData = () =>
  new Promise((resolve, reject) => {
    setTimeout(() => {
        resolve({ success: true });
    }, 2000);
  });

export {fetchData  }

接着引入fetchData的时候

import {fetchData} from './demo'
jest.mock('./demo') //该语句的用途是去__mock__下面找demo文件,然后看有没有导出的方法,这样上一条语句导入的fetchData就变成了从__mock__/demo.js中导入的我们自己模拟的fetchData,然后去执行。
// 有多个引入的时候,可以通过const {aa} = jest.requireActual('./demo'),这样,aa就回去真正的demo文件下引入了。
test('xxx',()=>{
    return fetchData().then(res=>.....)
})

其次,如果jest.mock(’./demo’)找不到__mock__下的文件的时候,jest内部就会自动模拟,如上面的axios。

mock模拟类
// util.js
class Util {
    a(){
        setTimeout(()=>{
            console.log(123);
        }, 3000)
    }
    b(){
        setTimeout(()=>{
            console.log(123);
        }, 5000)
    }
}
export default  Util
// test.js
import Util from './util'
jest.mock('./util') //因为我们没有在__mock__下定义该文件, jest.mock发现util是一个类,就自动把类的构造函数和方法变成Jest.fn()
// const Util = jest.fn()
// Util.prototype.a = jest.fn()
// Util.prototype.b = jest.fn()

我们也可以自己定义

// __mock__/util.js
 const Util = jest.fn()
 Util.prototype.a = jest.fn()
 Util.prototype.b = jest.fn()

这样jest.mock(’./util’)就会来这里找。

甚至可以

// test.js
jest.mock('./util', ()=>{
 const Util = jest.fn()
 Util.prototype.a = jest.fn()
 Util.prototype.b = jest.fn()
 return Util
})

作为第二个参数,在里面定义并且返回。这就是jest.mock的用法。

定时器

除了可以模拟方法,数据,还可以模拟定时器,因为真正的测试不可能等代码中所有的定时器都执行完,那会花费大量的时间。

  • jest.useFakeTimers(); //遇到setTimoeut等定时器就是用模拟的timers
  • 在调用有timer的函数的时候加上jest.runAllTimers(),它会立即将timer立即完成。
  • runAllTimers会将所有的timer都立即完成,而runOnlyPedingTimers(),将当前处于队列中的timer完成。
  • jest.advanceTimersByTime(3000) 让时间快进3s
const fetchData = () =>
  new Promise((resolve, reject) => {
    setTimeout(() => {
      resolve({ success: true });
    }, 5000);
  });

// 多个test,每个执行前都初始化,每个test都隔离。定时器之间就不会相互影响。
beforeEach(()=>{
  jest.useFakeTimers(); //遇到setTimoeut等定时器就是用模拟的timers
})

afterEach(() => {
  // 结束后需要手动清理
  jest.useRealTimers();
});

如上,在每一个test执行之前,就需要初始化使用假的timer。然后结束的时候需要手动清理。

test("模拟定时器", (done) => {
  fetchData()
    .then((res) => {
      expect(res).toEqual({ success: true });
      done();
    })
    .catch((err) => {
      done(err);
    });
  jest.runAllTimers() //将所有定时器立即执行完,无需等待,但是如果遇到定时器内嵌定时器的,会将内嵌的定时器也完成
  jest.runOnlyPedingTimers() // 只执行当前在等待队列的定时器,内嵌的定时器不会立即完成。
  jest.advanceTimersByTime(10000); //快进10s,可以定义多个。
});

快照

快照,顾名思义,就是现在拍照,然后过一阵时间再拍照,对比两个的差异。jest的快照也是如此

假设有个方法

const generateConfig = () => {
  return {
    server: "http://localhost",
    port: 8080,
    //time: new Date()
  };
};

传统的测试

test('测试generateConfig',()=>{
    expect(generateConfig()).toEqual({
        server: 'http://localhost',
        port:8080
    })
})

快照测试

test('测试generateConfig',()=>{
     // 匹配快照
     expect(generateConfig()).toMatchSnapshot({
         time: expect.any(Date)
     });
 })

第一次执行这个test的时候,会在根目录下创建_snapshots_文件,保存着你的快照

// __snapshots__/demo.test.js.snap 快照文件

// Jest Snapshot v1, https://goo.gl/fbAQLP

exports[`测试generateConfig 1`] = `
Object {
  "port": 8080,
  "server": "http://localhost",
}
`;

第二次执行的时候,就会跟之前保存的快照进行比较。如果不同就报错。

但是如上,time: new Date()等动态数据怎么执行都不可能跟上次一样,所以jest提供了参数,如

expect(generateConfig()).toMatchSnapshot({
         time: expect.any(Date)
     });

只要time匹配到一个Date的类型就可以。

此外,如果遇到不一样的快照,可以输入命令行工具

u 可更新快照全部
i 交互式的一个一个快照进行更新
toMatchInlineSnapshot行内快照,即不生成snapshot文件夹,而是将快照内嵌在test.js文件中,需要搭配安装prettier@1.18.2。如
test("测试generateConfig", () => {
  // 匹配快照
  expect(generateConfig()).toMatchInlineSnapshot(
    {
      time: expect.any(Date)
    },
  );
});

第一次执行之后

test("测试generateConfig", () => {
  // 匹配快照
  expect(generateConfig()).toMatchInlineSnapshot(
    {
      time: expect.any(Date)
    },
    // 执行test生成的快照
    `
    Object {
      "port": 8080,
      "server": "http://localhost",
      "time": Any<Date>,
    }
  `
  );
});

这就是快照的基本用法。

dom

import * as jsdom from 'jsdom'
var $ = require('jquery')(new jsdom.JSDOM().window);
const addDivToBody = () => {
  $("body").append("<div>123</div>");
};

// node不具备dom,而jest在node环境下自己模拟了一套dom的api, jsdom
test("测试 addDivToBody", () => {
  addDivToBody();
  addDivToBody();
  expect($("body").find('div').length).toBe(2)
});

jest在node环境下自己模拟了一套dom的api, jsdom,所以直接使用dom的一些操作。