AngularJS - 如何测试是否从另一个函数中调用函数?

时间:2015-01-07 15:36:06

标签: angularjs jasmine karma-runner karma-jasmine

我正在尝试使用karma-jasmine,我想知道为什么这个测试会失败:

it("should call fakeFunction", function() {
    spyOn(controller, 'addNew');
    spyOn(controller, 'fakeFunction');
    controller.addNew();
    expect(controller.fakeFunction).toHaveBeenCalled();
});

在我之前为此测试设置的控制器中,我有以下内容:

function addNew() {
    fakeFunction(3);
}

function fakeFunction(number) {
    return number;
}

使用以下内容公开addNewfakeFunction

vm.addNew = addNew;
vm.fakeFunction = fakeFunction;

然而,测试失败了以下内容:

Expected spy fakeFunction to have been called.

如果我在测试中调用该函数,我可以通过测试。我希望,但是,我可以测试fakeFunction是否被另一个函数调用。实现这个目标的正确方法是什么?

更新

//test.js

beforeEach(function() {

    module("app");

    inject(function(_$rootScope_, $controller) {

        $scope = _$rootScope_.$new();
        controller = $controller("CreateInvoiceController", {$scope: $scope});

    });

});

如果我测试类似的东西:

it('should say hello', function() {
    expect(controller.message).toBe('Hello');
});

如果我将以下内容放入我的控制器中,则测试通过:

var vm = this;
vm.message = 'Hello';

我只想知道如何测试是否从另一个函数调用了公共函数。

2 个答案:

答案 0 :(得分:10)

您的addNew方法正在调用fakeFunction。但是,它没有调用controller.fakeFunction,这是您的期望。

您需要更改代码才能使用控制器,而不是使用这些独立功能。

编辑:您还需要不监视addNew功能。这导致该功能被间谍取代。另一种方法是将其更改为:

spyOn(controller, 'addNew').and.callThrough()

答案 1 :(得分:8)

我自己就是遇到过这个问题。 @Vadim先前的回答有正确的原则,但我不认为一切都很清楚。在我的情况下,我试图从另一个函数内调用服务上的公共函数。以下是相关摘要:

服务:

angular.module('myApp').factory('myService', function() {

    function doSomething() {
      service.publicMethod();
    }

    function publicMethod(){
      // Do stuff
    }

    var service = {
      publicMethod: publicMethod
    };

    return service;
});

测试:

it('calls the public method when doing something', function(){
  spyOn(service, 'publicMethod');

  // Run stuff to trigger doSomething()

  expect(service.publicMethod).toHaveBeenCalled();
});

这里的关键是被测试的函数需要调用与被窥探的公共函数相同的引用。