如何在Jest中的单个测试中模拟ES6模块?

时间:2018-04-15 06:37:19

标签: javascript unit-testing jest es6-modules

说我有以下模块:

的src / validators.js

export const isPositiveNumber = value => value > 0 && value < Number.MAX_SAFE_INTEGER;

我在另一个模块中使用它:

的src /计算器/ volume.js

import { isPositiveNumber } from '../validators';
import { InvalidArgumentException } from '../exceptions';

export const sphere = radius => {
    if (!isPositiveNumber(radius)) {
        throw new InvalidArgumentException('radius must be a positive number');
    }

    return (4/3) * Math.PI * Math.pow(radius, 3);
};

然后在我的测试中:

测试/计算器/ volume.test.js

import { volume } from '../../src/calculators/volume';
import { InvalidArgumentException } from '../../src/exceptions';
import { isPositiveNumber } from '../../src/validators';

jest.mock('../../src/validators');

describe('Volume calculations', () => {
    describe('sphere()', () => {
        it('should throw an exception if the radius is invalid', () => {
            isPositiveNumber.mockReturnValue(false);
            expect(() => volume()).toThrow(InvalidArgumentException);
        });

        it('should compute the volume', () => {
            isPositiveNumber.mockReturnValue(true);
            expect(volume(3)).toBeCloseTo(113,3);
        });
    });
});

这是有效的,除了,我不想在实际计算音量的第二个测试中使用模拟isPositiveNumber

我希望isPositiveNumber模拟仅在验证测试中。

我不知道如何使用我正在使用的ES6模块设置。它似乎要求模拟拦截发生在测试范围之外,这意味着我必须在套件中的每个测试中模拟返回值。

这只是一个简单测试的例子,但稍后会有更复杂的测试,我想知道如何在每个测试的基础上更精确地模拟ES6模块。

任何帮助都将不胜感激。

1 个答案:

答案 0 :(得分:1)

这可能是因为您正在使用browser-sync start --server "a/b/c/" --files "index.html" according to the documentation会将babel-jest提升到顶级,无论其实际放置在何处在您的ES6模块代码中。

  

使用jest.mock()时,对babel-jest的调用将自动提升到代码块的顶部。

相反,您可以分别在mockbefore()挂钩中使用jest.doMock()jest.dontMock()方法,用于需要模拟模块定义中的函数的单个测试