javascript - 如何避免 AngularJS 测试中的 "$digest already in progress"错误?

标签 javascript angularjs unit-testing testing

我在使用 Chrome 在 Karma 中运行我的 AngularJS 测试之一时遇到问题。该测试在 PhantomJS 中有效。

使用 Chrome 运行测试时出现以下错误:

Chrome 46.0.2490 (Windows 7 0.0.0) ERROR
  Uncaught Error: [$rootScope:inprog] $digest already in progress
  http://errors.angularjs.org/1.3.18/$rootScope/inprog?p0=%24digest
  at C:/dev/unionvms_eu/branches/dev/unionvms-web/bower_components/angular-mocks/angular-mocks.js:275

测试:

describe('todoRestService', function() {

    beforeEach(module('todoApp'));

    //MOCK RESOURCE
    beforeEach(function () {
        var mockResource = {
            todo: function() {
                return {
                    update: function(todoDTO, callback) {
                        callback({
                            code : 200,
                            data: {
                                status : 'CLOSED'
                            }
                        });
                    },
                }
            },
        };

        module(function ($provide) {
            $provide.value('todoRestFactory', mockResource);
        });

    });

    it("updateTodoStatus should send request to backend and return received object", inject(function($rootScope, $log, todoRestService, Todo) {
        var todo = new Todo();
        todo.status = "CLOSED";

        var resolved = false;
        $rootScope.$apply(function(){
            todoRestService.updateTodoStatus(todo).then(function(updatedTodo){
                resolved = true;
                expect(updatedTodo.status).toEqual(todo.status);
            });
        });
        $rootScope.$digest();
        expect(resolved).toBe(true);
    }));

});

代码:

angular.module('todoApp')
    .factory('todoRestFactory',function($resource) {
        return {
            todo : function(){
                return $resource('/app/todo', {}, {
                    update: {method: 'PUT'}
                });
            },
        };
    })
.factory('todoRestService', function($q, $log, todoRestFactory, Todo){

    var updateTodoStatus = function(todo){
        var deferred = $q.defer();
        todoRestFactory.todo().update(todo.DTO(), function(response) {
            if(response.code !== 200){
                deferred.reject("Invalid response status");
                return;
            }
            deferred.resolve(Todo.fromDTO(response.data));
        }, function(error) {
            $log.error("Error updating todo status");
            $log.error(error);
            deferred.reject(error);
        });
        return deferred.promise;
    };

    return {
        updateTodoStatus: updateTodoStatus,
    };
});

如果我从测试中删除 $rootScope.$digest();,错误就会消失,但最后一个断言将失败。我怎样才能使测试工作?

最佳答案

您需要检查 $digest 是否已经启动。 你可以通过

 if(!$scope.$$phase) {
 //do smth here
}

关于javascript - 如何避免 AngularJS 测试中的 "$digest already in progress"错误?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33776652/

相关文章:

javascript - Angular UI Bootstrap Popover - 如何使用 ESC 关闭弹出窗口

c# - 模拟 AsNoTracking Entity Framework

javascript - 在 javascript 闭包中测试

javascript - 修复 Can't perform a react state update on an unmounted component React native

javascript - 如何在html中添加更多值选项?

javascript - 如何在 Svelte 中动态绑定(bind)事件?

javascript - 在 JavaScript 中的序列化数组中设置复选框值

javascript - $provide.decorator $controller 返回 throw undefined 不是一个函数 angularjs

javascript - 如何在Angularjs中的字符串插值中获得换行符

unit-testing - 使用谷歌验证器进行单元测试