问题描述
快到 2021 年了,有没有办法模拟单个函数?我的意思是没有对象的函数。
// demo.js
module.exports = () => {
return 'real func demo';
};
// demo.test.js
const demo = require("./demo");
// ...
sinon.stub(demo).callsFake(() => {
return 'mocked function';
});
expect(demo()).to.eq('mocked function')
解决方法
您可以使用 as
语法导入。
如果您导出为 default
则使用“默认”而不是实际名称
import * as yourModule from '../<path>';
const fake = sinon.fake.returns(<value>);
sinon.replace(yourModule,'default',fake);
,
我就quote myself:
您不能在符合 ES2015 的模块 (ESM) 或 Node.js 中的 CommonJS 模块中存根独立导出的函数。这就是这些模块系统的工作方式。只有在将它们转化为别的东西之后,你可能才能实现你想要的。替换模块的方式完全取决于环境,这就是为什么 Sinon 将自己吹捧为“JavaScript 的独立测试间谍、存根和模拟”而不是不是模块替换工具,因为这最好留给特定于环境的实用程序( proxyquire、各种 webpack 加载器、Jest 等),适用于您所处的任何环境。
要更深入地了解为什么在使用解构时这在 CommonJS 中永远不起作用,请参阅我的回答 here。你需要注入一个拦截加载的模块加载器。
这就是在符合规范的环境中尝试使用 Sinon 来删除 ESM 模块的样子:
$ node esm-stubbing.mjs
/private/tmp/test/node_modules/sinon/lib/sinon/stub.js:72
throw new TypeError("ES Modules cannot be stubbed");
^
TypeError: ES Modules cannot be stubbed
at Function.stub (/private/tmp/test/node_modules/sinon/lib/sinon/stub.js:72:15)
at Sandbox.stub (/private/tmp/test/node_modules/sinon/lib/sinon/sandbox.js:388:37)
at file:///private/tmp/test/esm-stubbing.mjs:4:7
$ cat esm-stubbing.mjs
import sinon from "sinon";
import * as myModule from "./module.mjs";
sinon.stub(myModule,"foo").returns(42);
这只是Sinon在看到模块的命名空间为只读时自己抛出错误,而不是让Node抛出更隐晦的错误信息。