繁体   English   中英

如何在 Sinon 中存根一系列方法?

[英]How do I stub a chain of methods in Sinon?

我知道如何使用存根替换一个函数。

sandbox.stub(Cars, "findOne",
            () => {return car1 });

但是现在我的函数中有一行我想测试我需要存根看起来像这样

Cars.find().fetch()

所以这里有一个函数链,我不确定我需要做什么。 我如何存根“查找”以返回我可以用来存根“获取”的东西?

恕我直言,我们可以只使用returns来做到这一点。 我们不需要使用callsFake或将其模拟为函数。

// Cars.find().fetch()

sinon.stub(Cars, 'find').returns({
  fetch: sinon.stub().returns(anything);
});

以防万一,如果在fetch()之后还有另一个方法,我们可以使用returnsThis()

// Cars.find().fetch().where()

sinon.stub(Cars, 'find').returns({
  fetch: sinon.stub().returnsThis(),
  where: sinon.stub().returns(anything);
});

参考: https : //sinonjs.org/releases/v6.3.3/

希望它有帮助

试试这个:

sandbox.stub(Cars, "find", () => {
    return {
        fetch: sinon.stub().returns(anything);
    };
});

将函数附加到存根的形式如下所示:

sandbox.stub(Cars, "find", () => {
    return {
        fetch: sinon.stub().returns(anything);
    };
});

已弃用。

现在,从6.3 版开始

sandbox.stub(Cars, "find").callsFake(() => {
    return {
        fetch: sinon.stub().returns(anything);
    };
});

这是另一种允许监视 jQuery 方法链的方法 - 我花了很长时间才弄明白。

在示例中,我试图测试是否清除了电子邮件字段

    //set up stub and spy
    const valSpy = sandbox.spy();
    const jQueryStub = sandbox
      .stub($.prototype, "find")       // this prototype is important
      .withArgs("input[name=email]")
      .returns({ val: valSpy });

    // call function under test
    learnerAlreadyAccepted(inviteDoc);

    // check expectations
    expect(jQueryStub).to.have.been.called;      // not really necessary
    expect(valSpy).to.have.been.calledWith("");

被测函数是(大致):

  learnerAlreadyAccepted = function(doc) {
    $("form").find("input[name=email]").val("");
  }

我遇到了这个问题,虽然我喜欢单个测试的解决方案,但想要一些更动态的东西,以便跨测试重用。 我也更喜欢沙箱方法,因为它使大型套件的恢复变得更加容易。 最终结果:

 export function setupChainedMethodStub(sandbox: sinon.SinonSandbox, obj: any, methodName: string, methodChain: string[], value: any) { return sandbox.stub(obj, methodName).returns(generateReturns(sandbox, methodChain, value)); } function generateReturns(sandbox: sinon.SinonSandbox, methodChain: string[], value: any): any { if (methodChain.length === 1) { return { [methodChain[0]]: sandbox.stub().returns(value), }; } else { return { [methodChain[0]]: sandbox.stub().returns(generateReturns(sandbox, methodChain.slice(1), value)), }; } }

无论我想在何处动态设置存根,我都会传入创建的沙箱和其他参数:

setupChainedMethodStub(sandbox, MyMongooseModel, 'findOne', ['sort', 'exec'], { foo: 'bar' })

然后我在我的最高范围afterEach()有一个sandbox.restore() afterEach()

我有一个简单的解决方案,希望对其他人有用。

假设fetch也是Cars上的一个方法,并且fetchfind支持方法链, Cars可能看起来像这样:

class Cars {
  fetch() {
    // do stuff
    return this;
  }

  find() {
    // do stuff
    return this;
  }
}

[ANSWER] 我们应该能够像这样支持使用存根的方法链接:

  sandbox.stub(Cars, 'fetch').callsFake(function () { return this; }); // optional
  sandbox.stub(Cars, 'findOne').callsFake(function () { return this; });

与 v2.0 相比有一些变化。

更多细节在这里

其中之一是:

stub(obj, 'meth', fn) has been removed, see documentation

您可以降级,但我不推荐它,相反,您可以执行以下操作:

let stub = sinon.stub(obj, "meth").callsFake(() => {
  return {
    meth2: sinon.stub().callsFake(() => {
      return {
        meth3: sinon.stub().returns(yourFixture),
      };
    }),
  };
});

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM