3

我对 sinon 和 proxyquire 还很陌生,我想我已经阅读了 SO 上的所有答案,但我仍然没有找到我需要的东西。无论如何,这是我的代码的净化版本。

const fetch = require('node-fetch');

async function deleteID(id, endpoint) {
  try {
      let url = `${endpoint}/delete/${id}`;
      let res = await fetch(url, { method: 'DELETE' });
      res = await res.json(); // <---- THIS FAILS WHEN fetch IS MOCKED
      // do stuff with res
  } catch (err) {
    logger.error(`Error: ${JSON.stringify(err)}`);
  }
}

这很简单,它使用 node-fetch 来访问一个 url,然后在请求成功或失败时执行一些操作。这是我的测试,让我们为 fetch 设置模拟:

const proxyquire = require('proxyquire').noCallThru();
const sinon = require('sinon');

  beforeEach((done) => {

    const validResponse = {
      status: 200,
      data: 'hello, world\n'
    };
    deleteProxy = proxyquire('./delete', {
      'node-fetch': sinon.stub().returns(Promise.resolve(JSON.stringify(validResponse)))
    });
  });

所以 fetch 调用现在返回validResponse而不是访问服务器。这是我的测试:

    it.only('should delete', async () => {
    try {
      deleteProxy.deleteID('id', 'endpoint');
    } catch (err) {
      expect(err.message).to.have.lengthOf.at.least(0);
    }
  });

这失败了,因为 res 只是一个具有状态和数据的对象,它不是具有 Body 等的正确响应......我们的其余代码使用node-mocks-http但所有使用该模块的测试都直接命中了 url ,而不是间接地,通过获取,就像我在上面所做的那样。

如何创建模拟响应以适应上述测试,或者我应该使用不同的方法?

4

1 回答 1

1

通过查看代码和我对 sinon 的经验,我会说这不是实际的 HTTP 响应,所以你也必须模拟json()。在beforeEach方法:

 const body = {
  status: 200,
  data: 'hello, world\n'
};
var validResponse = { json: () => { return body } };
deleteProxy = proxyquire('./delete', {
  'node-fetch': sinon.stub().returns(Promise.resolve(validResponse))
});

试试看JSON.stringify()

让我知道它是否不起作用。

于 2019-10-30T22:49:42.040 回答