最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - Replace specific module in testing - Stack Overflow

programmeradmin1浏览0评论

I am testing my React-Redux app with Jest and as part of this in my API calls I am importing a fetch module cross-fetch. I want to override or replace this with fetch-mock. Here is my file structure:

Action.js

import fetch from 'cross-fetch';
export const apiCall = () => {
    return fetch('http://url');

Action.test.js

import fetchMock from 'fetch-mock';
import { apiCall } from './Action';
fetchMock.get('*', { hello: 'world' });
describe('actions', () => {
    apiCall().then(
        response => {
            console.log(response)
        })
})

Obviously at this point I haven't set up the test. Because cross-fetch is imported closer to the function it uses it's implementation of fetch, causing it to do the actual call instead of my mock. Whats the best way of getting the fetch to be mocked (apart from removing the import fetch from 'cross-fetch' line)?

Is there a way to do a conditional import depending on whether the node script called is test or build? Or set the mocked fetch to take priority?

I am testing my React-Redux app with Jest and as part of this in my API calls I am importing a fetch module cross-fetch. I want to override or replace this with fetch-mock. Here is my file structure:

Action.js

import fetch from 'cross-fetch';
export const apiCall = () => {
    return fetch('http://url');

Action.test.js

import fetchMock from 'fetch-mock';
import { apiCall } from './Action';
fetchMock.get('*', { hello: 'world' });
describe('actions', () => {
    apiCall().then(
        response => {
            console.log(response)
        })
})

Obviously at this point I haven't set up the test. Because cross-fetch is imported closer to the function it uses it's implementation of fetch, causing it to do the actual call instead of my mock. Whats the best way of getting the fetch to be mocked (apart from removing the import fetch from 'cross-fetch' line)?

Is there a way to do a conditional import depending on whether the node script called is test or build? Or set the mocked fetch to take priority?

Share Improve this question edited Mar 18, 2022 at 8:46 VLAZ 29.1k9 gold badges63 silver badges84 bronze badges asked Mar 2, 2018 at 4:20 SamSam 7,0988 gold badges37 silver badges64 bronze badges 4
  • What happens when you run your test? – Code-Apprentice Commented Mar 2, 2018 at 4:30
  • When I run my code without removing the import fetchMock it says only absolute urls are supported with the fetch – Sam Commented Mar 2, 2018 at 5:14
  • What code causes that error? And did you try changing it to an absolute URL? – Code-Apprentice Commented Mar 2, 2018 at 15:48
  • The import causes that error. If I put an actual URL in it just calls that URL, which is obviously not what I want. If I remove that import it uses the fetch-mock instead. – Sam Commented Mar 4, 2018 at 21:55
Add a ment  | 

4 Answers 4

Reset to default 5

fetch-mock is not intended to replace the fetch() calls in the code you are testing nor do you need to change or remove any imports. Instead, it provides mock responses during your tests so that requests made with fetch() receive known, reliable responses.

If your project is a webpack project, then https://github./plasticine/inject-loader is very useful. You can simply swap any dependency with a mock in just a few lines of code.

describe('MyModule', () => {
  let myModule;
  let dependencySpy;

  beforeEach(() => {
    dependencySpy= // {a mock/spy};
    myModule = require('inject-loader!./MyModule')({
      'cross-fetch': {dependencySpy},
    });
  });

  it('should call fetch', () => {
    myModule.function1();
    expect(dependencySpy.calls.length).toBe(1);
  });

});

Note: make sure you don't import the module under test at the top of your file. the require call does that part.

There are a few ways you can approach this problem

  1. You can stub modules without dependency injection using Sinon.

  2. Use a lib called rewire to mock imported methods in procedural calls

  3. Re write your original function so that you are not using the import directly

    const apiCall = (url, {fetchFn = fetch}) => fetchFn(url);
    
    describe("apiCall", () => {
      it("should call fetch with url", () => {
        const fetchFn = sinon.spy();
        const EXPECTED_URL = "URL";
    
        apiCall(EXPECTED_URL, {fetchFn});
    
        expect(fetchFn.firstCall.args).to.deep.equal([EXPECTED_URL]);
      })
    });
    
  4. Intercept the request and assert on the response (this is appears to be what fetch-mock does however I prefer nock as the documentation is much better).

    describe("apiCall", () => {
      it("should call fetch with url", () => {
        const EXPECTED_URL = "URL";
        const EXPECTED_RESPONSE = 'domain matched';
    
        var scope = nock(EXPECTED_URL)
        .get('/resource')
        .reply(200, EXPECTED_RESPONSE);
    
        return expect(apiCall(EXPECTED_URL)).to.equal(EXPECTED_RESPONSE);
      })
    });
    

Why not also export a maker function in your Action.js file that will get the fetch method injected and then returns the actual apiCaller:

Action.js

// this export allows you to setup an apiCaller
// with a different fetcher than in the global scope 
export const makeApiCaller = (fetch) => (url, opts) => fetch(url, opts);
// this export is your default apiCaller that uses cross-fetch
export const apiCall = makeApiCaller(fetch);

Then in your tests you can somewhere instantiate your ApiCaller, e.g. in the before:

Action.test.js

import fetchMock from 'fetch-mock';
import { makeapiCaller } from './Action';

fetchMock.get('*', { hello: 'world' });

// ...

let apiCall; 
before(() {
  apiCall = makeApiCaller(fetch); // injects mocked fetch 
});

describe('actions', () => {
    apiCall('/foo/bar').then(
        response => {
            console.log(response)
        })
})

Note: the benefit of doing it this way, is that you don't have to introduce another argument to your apiCall function signature (as proposed in another answer) and thus stay backwards patible.

发布评论

评论列表(0)

  1. 暂无评论