Jest mock inner function

匿名 (未验证) 提交于 2019-12-03 01:38:01

问题:

I have one file called helper.js that consist of two functions

export const funcA = (key) => {    return funcB(key) };  export const funcB = (key,prop) => {    return someObj; }; 

I have my helper.spec.js to test the helper.js file functions.

import {funcA,funcB} from 'helper';  describe('helper', () => {    test('testFuncB', () => {     }    test('testFuncA', () => {     } } 

The test for funcB is pretty simple i just call it and expect someObj
The problem is to test funcA, in order to test it i want to mock the response of funcB.

I want testFuncB call the actual funcB and testFuncA call mocked funcB

How can i achieve funcB to be mocked and original in my two tests?

This is not a duplicate. It is a different case: they mock inner called functions only, if I remove the testFuncB then it will be the same but I must perform test on testFuncB too.

回答1:

If an ES6 module directly exports two functions (not within a class, object, etc., just directly exports the functions like in the question) and one directly calls the other, then that call cannot be mocked.

In this case, funcB cannot be mocked within funcA the way the code is currently written.

A mock replaces the module export for funcB, but funcA doesn't call the module export for funcB, it just calls funcB directly.


Mocking funcB within funcA requires that funcA call the module export for funcB.

That can be done in one of two ways:


Move funcB to its own module

funcB.js

export const funcB = () => {   return 'original'; }; 

helper.js

import { funcB } from './funcB';  export const funcA = () => {   return funcB(); }; 

helper.spec.js

import * as funcBModule from './funcB'; import { funcA } from './helper';  describe('helper', () => {    test('test funcB', () => {     expect(funcBModule.funcB()).toBe('original');  // Success!   });    test('test funcA', () => {     const spy = jest.spyOn(funcBModule, 'funcB');     spy.mockReturnValue('mocked');      expect(funcA()).toBe('mocked');  // Success!      spy.mockRestore();   }); }); 

Import the module into itself

"ES6 modules support cyclic dependencies automatically" so it is perfectly valid to import a module into itself so that functions within the module can call the module export for other functions in the module:

helper.js

import * as helper from './helper';  export const funcA = () => {   return helper.funcB(); };  export const funcB = () => {   return 'original'; }; 

helper.spec.js

import * as helper from './helper';  describe('helper', () => {    test('test funcB', () => {     expect(helper.funcB()).toBe('original');  // Success!   });    test('test funcA', () => {     const spy = jest.spyOn(helper, 'funcB');     spy.mockReturnValue('mocked');      expect(helper.funcA()).toBe('mocked');  // Success!      spy.mockRestore();   }); }); 


回答2:

I think this might work

import * as helper from 'helper';  describe('helper', () => {    test('testFuncB', () => {     }    test('testFuncA', () => {       const mockTestFuncB = jest.mock();       // spy on calls to testFuncB and respond with a mock function       jest.spyOn(helper, 'testFuncB').mockImplementationOnce(mockTestFuncB);        // Do the testing ...        // Restore helper.testFuncB to it's original function       helper.testFuncB.mockRestore();    } } 


回答3:

import * as helper from 'helper';      describe('helper', () => {        it('should test testFuncA', () => {           const mockTestFuncB = jest.mock();           // spy on calls to testFuncB and respond with a mock function             mockTestFuncB.spyOn(helper, 'testFuncB').mockReturnValue(/*your expected return value*/);            // test logic            // Restore helper.testFuncB to it's original function           helper.testFuncB.mockRestore();        }     } 


标签
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!