2

我已签入如何使用 Jasmine 验证 jQuery AJAX 事件?以及如何在不模拟 AJAX 调用的情况下使用进行 AJAX 调用的函数来测试 Angular 控制器?但由于某种原因,它们不一定会提供适合我的格式。

我的问题是我正在对一个函数进行单元测试,该函数调用一个触发 AJAX 请求的函数。我想在不触发 AJAX 请求(或停止该请求)的情况下测试外部函数,这样我就不会将一堆错误数据射入数据服务器。

这是我的外部函数(它调用包含 AJAX 的函数):

vm.clickSubmit = function () {
    vm.loading = true; // starts the loading spinner
    vm.updateData(); // makes current data current in the data model
    vm.inputData = {    // build the data to be submitted
        ...
    };
    vm.submit(); // calls the AJAX function
};

这是我的 AJAX 函数:

vm.submit = function () {
    var d = $.ajax({
        type: "POST"
        , data: angular.toJson(vm.inputData)
        , url: "http://submit_the_data"
        , contentType: "application/json"
        , xhrFields: { withCredentials: true }
        , crossDomain: true
    })
    .success(function (resp) {
        DataService.ticketNumber = resp; // returns ticket# for the data model
    })
    .error(function (error) {
        DataService.ticketNumber = DataService.submitError;
    });
    d.then(function (d) {
        vm.loading = false; // stops the loading spinner
        DataService.tickets = []; // empty's the array to be filled anew
        $location.path('/submitted'); // success splash html
        $scope.$apply();
    });
};

我已经编写了所有将读取和验证inputData对象中的值的测试,但我不确定如何包围调用,clickSubmit()因此实际上没有任何内容提交给服务器。我在单元测试中已经到了这一点:

'use strict';

describe('Controller: HomeController', function () {
    beforeEach(module('tickets'));
    var controller, scope, $location, DataService;
    var tests = 0;
    beforeEach(inject(function ($rootScope, $controller, _$location_, _DataService_) {
        $location = _$location_;
        DataService = _DataService_;
        scope = $rootScope.$new();
        controller = $controller('HomeController', {
            $scope: scope
        });
    }));
    afterEach(function () {
        tests += 1;
    });
    describe('clickSubmit should verify data and submit new ticket', function () {
        beforeEach(function () {
            jasmine.Ajax.install();
            controller.loading = false;
                ... // inputData fields filled in with test data
        });
        afterEach(function () {
            jasmine.Ajax.uninstall();
        });
        it('should start the spinner when called', function () {
            controller.clickSubmit();
            expect(controller.loading).toBeTruthy();
        });
        // other it('') tests
    });
    it('should have tests', function () {
        expect(tests).toBeGreaterThan(0);
    });
});

那么在加载微调器的期望之后应该如何取消vm.submit()实际代码中的调用呢?

谢谢,

-C§

4

1 回答 1

2

我建议模拟调用而不是实际调用它。粒度取决于您,您可以存根 ajax 调用,也可以存根整个提交函数。

以下是如何存根提交函数:

spyOn(controller, 'submit').and.callFake(function() {
    DataService.ticketNumber = somevalue;
});

在实际调用 caller.clickSubmit() 的控制器之前放置该代码。

然后,您可以跟进对间谍的期望,例如:

expect(controller.submit).toHaveBeenCalled()

或与 spyOn 相关的任何其他期望。

以下是 jasmine 文档: http: //jasmine.github.io/2.0/introduction.html
在“间谍”区域往下看。

如果你想模拟 ajax 调用,你将不得不模拟这样的承诺:

spyOn($, 'ajax').and.callFake(function() {
    var deferred = $q.defer();
    deferred.resolve(someResponse);
    return deferred.promise;
});

此外,为了让代码等待承诺解决,在提交调用之后,您需要运行 $scope.$digest() 以便 Angular 可以处理承诺解决。然后,您可以检查您对取决于承诺的解决或拒绝的代码的期望。

于 2015-08-12T22:02:03.360 回答