在玩笑测试中模拟节点模块

时间:2020-01-20 22:26:29

标签: javascript unit-testing testing jestjs

我有一个Jest测试,我正在编写一个进行API调用的函数。如果API调用返回403,则应调用节点模块中的函数。我正在尝试使用模拟笑话功能对此进行测试,但是我无法通过测试使用所制作模块的模拟版本。

file.spec.js

import file from './file'

const mockedNodeModule = jest.genMockFromModule('nodeModule')
mockedNodeModule.namedExport = { logout: jest.fn() }

it('call returns a 403', async () => {
      await file.apiCall('brand', 'entityType', 'name')
      expect(mockedNodeModule.namedExport.logout).toHaveBeenCalled()
})

file.js

import { namedExport } from './nodeModule';
import api from './api';

const apiCall = () => {
  return api.makeCall().then(
    () => {},
    (error) => {
      if (error.status === 403) {
        namedExport.logout();
      }
    },
  );
};

export default { apiCall };

当我检查是否已调用了kededNodeModule.namedExport.logout时,测试总是失败。当我在调用它的行上放置一个断点时,似乎在测试运行时未使用模拟版本(即它仍在使用我的node_modules中的模块)。我也尝试过使用jest.mock(),但是结果是一样的。我设置测试的方式有问题吗?难道在这种情况下不能嘲笑节点模块吗?

1 个答案:

答案 0 :(得分:1)

jest.mock(moduleName, factory, options)应该可以工作。

例如

file.js

import { namedExport } from './nodeModule';
import api from './api';

const apiCall = () => {
  return api.makeCall().then(
    () => {},
    (error) => {
      if (error.status === 403) {
        namedExport.logout();
      }
    },
  );
};

export default { apiCall };

api.js

function makeCall() {}

export default { makeCall };

nodeModule.js

export const namedExport = {
  logout() {
    console.log('real implementation');
  },
};

file.test.js

import file from './file';
import api from './api';
import { namedExport } from './nodeModule';

jest.mock('./nodeModule', () => {
  const mNamedExport = {
    logout: jest.fn(),
  };
  return { namedExport: mNamedExport };
});

jest.mock('./api', () => {
  return { makeCall: jest.fn() };
});

describe('59831697', () => {
  afterEach(() => {
    jest.clearAllMocks();
  });
  it('should handle error if http status equal 403', async () => {
    const mError = { status: 403 };
    api.makeCall.mockRejectedValueOnce(mError);
    await file.apiCall();
    expect(namedExport.logout).toHaveBeenCalledTimes(1);
  });
});

带有覆盖率报告的单元测试结果:

 PASS  src/stackoverflow/59831697/file.test.js (13.506s)
  59831697
    ✓ should handle error if http status equal 403 (7ms)

----------|----------|----------|----------|----------|-------------------|
File      |  % Stmts | % Branch |  % Funcs |  % Lines | Uncovered Line #s |
----------|----------|----------|----------|----------|-------------------|
All files |      100 |       50 |    66.67 |      100 |                   |
 file.js  |      100 |       50 |    66.67 |      100 |                 8 |
----------|----------|----------|----------|----------|-------------------|
Test Suites: 1 passed, 1 total
Tests:       1 passed, 1 total
Snapshots:   0 total
Time:        15.448s

源代码:https://github.com/mrdulin/jest-codelab/tree/master/src/stackoverflow/59831697