对指令进行单元测试 - 无法强制使用虚假数据

Unit testing a directive - Unable to force a fake data

我试图了解如何在我的以下情况下对我的指令进行单元测试。

基本上我正在尝试对具有控制器的指令进行单元测试。在加载此指令时,控制器通过服务发出 http 请求,该服务再次将一些数据带到控制器,然后将此数据提供给指令视图。

根据我的理解,下面的场景我应该这样做:

如您所见,到目前为止我一直在尝试的是用虚假数据覆盖服务。到目前为止我无法完成的工作。

现在有一些疑问。

正如您在我的控制器中看到的那样。我正在为视图提供整个服务:

$scope.ItemsDataservice = ItemsDataservice;

是什么让我相信我覆盖服务的方法应该有效。

我的问题:

在下面的场景中,我了解到我可以覆盖服务来操作数据,甚至可以覆盖控制器以按范围操作数据。

在这里做什么是正确的? 我理解错了吗? 我在混合单元测试吗?

在我当前的单元测试代码中,当我应用(或不应用)假数据时,没有任何区别:

ItemsDataservice.items = DATARESULT;

ItemsDataservice.items = null;

控制器:

angular.module('app')
    .controller('ItemsCtrl', function ($scope, $log, ItemsDataservice) {

        $scope.ItemsDataservice = ItemsDataservice;
        $scope.ItemsDataservice.items = null;

        $scope.loadItems = function() {

            var items = [];

            ItemsDataservice.getItems().then(function(resp) {

                if (resp.success != 'false') {

                    for (resp.something ... ) {
                        items.push({ ... });
                    };

                    ItemsDataservice.items = items;

                };

            }, function(e) {
                $log.error('Error', e);
            });
        };

        $scope.loadItems();

    });

服务:

angular.module('app')
    .service('ItemsDataservice', function ItemsDataservice($q, $http) {

        ItemsDataservice.getItems = function() {

            var d = $q.defer();
            var deffered = $q.defer();
            var url = 'http://some-url?someparameters=xxx'

            $http.get(url)
                .success(function (d) {
                    deffered.resolve(d);
                });

            return deffered.promise;

        };

        return ItemsDataservice;

    });

指令:

angular.module('app')
    .directive('items', function () {
        return {
            templateUrl: '/items.html',
            restrict: 'A',
            replace: true,
            controller: 'ItemsCtrl'
        };
    });

单元测试指令:

ddescribe('Directive: Items', function () {

  var element, scope, _ItemsDataservice_, requestHandler, httpBackend;
  var URL = 'http://some-url?someparameters=xxx';
  var DATARESULT = [{ ... }];

  // load the directive's module
  beforeEach(module('app'));
  beforeEach(module('Templates')); // setup in karma to get template from .html

  beforeEach(inject(function ($rootScope, ItemsDataservice) {

    httpBackend = $httpBackend;
    scope = $rootScope.$new();
    _ItemsDataservice_ = ItemsDataservice;

    requestHandler = httpBackend.when('GET', URL).respond(200, 'ok');

  }));

  afterEach(function() {
    //httpBackend.verifyNoOutstandingExpectation();
    //httpBackend.verifyNoOutstandingRequest();
  });

  it('Show "No Items available" when empty result', inject(function ($compile) {

    _ItemsDataservice_.items = null;

    element = angular.element('<div data-items></div>');
    element = $compile(element)(scope);
    scope.$digest();

    element = $(element);
    expect(element.find('.msg_noresult').length).toBe(1);

  }));

  it('Should not show "No Items available" when data available ', inject(function ($compile) {

    _ItemsDataservice_.items = DATARESULT;

    element = angular.element('<div data-items></div>');
    element = $compile(element)(scope);
    scope.$digest();

    element = $(element);
    expect(element.find('.msg_noresult').length).toBe(0);

  }));

});

我解决了问题。

更改了这一行:

element = $compile(element)(scope);

到这一行:

element = $compile(element.contents())(scope);

唯一的区别是 jquery 方法 .contents()

我还没明白为什么。但它解决了。

更新:

我刚刚发现的另一件事对我来说非常有用。 您可以在 httpBackend:

上使用正则表达式
httpBackend.whenGET(/.*NameOfThePageXXX\.aspx.*/).respond(200, 'ok');

因此,如果您只是想避免异常,则无需担心使用完全相同的参数等。