2015-10-06 47 views
0

我试图跳过在我的测试中为我的数据转至Firebase,并返回一些简单的结果。我不想测试Firebase代码的工作原理,但我的工厂可以返回数据。使用摩卡和Sinon进行AngularJS测试 - 嘲笑Firebase依赖关系

我有以下工厂:

// Get Firebase Reference 
angular.module('MyApp').factory('FireBaseData', function(FIREBASE) { 
    var FireBaseReference = new Firebase(FIREBASE.URL); // FIREBASE.URL = xxx.firebaseio.com 
    return FireBaseReference; 
}) 

// Get the Firebase Array of Team Records 
angular.module('MyApp').factory('AllTeams', ["FireBaseData", "$firebaseArray", 
    function(FireBaseData, $firebaseArray) { 
     return $firebaseArray(FireBaseData.child('Teams')); 
    } 
]); 

我创建模拟,从而取代了各个功能,以及我的测试将使用这些。

'use strict'; 

var $MockFirebaseArray = function(ArrayWithData) { 
    return ArrayWithData; 
}; 

var $MockFirebaseObject = function(ObjectWithData) { 
    return ObjectWithData; 
}; 

var MockFirebaseData = function() { 
    return { 
     child: function(StringValue) { 
      return ""; 
     } 
    }; 
}; 

测试与嘲笑:

'use strict'; 

describe('Firebase Mocks', function() { 
    var TestArray = [ 
     { 'aString': 'alpha', 'aNumber': 1, 'aBoolean': false }, 
     { 'aString': 'bravo', 'aNumber': 2, 'aBoolean': true }, 
     { 'aString': 'charlie', 'aNumber': 3, 'aBoolean': true }, 
     { 'aString': 'delta', 'aNumber': 4, 'aBoolean': true }, 
     { 'aString': 'echo', 'aNumber': 5 } 
    ]; 

    describe('MockFirebaseData', function() { 
     var TestFirebase = MockFirebaseData(); 
     it('should return empty text ("") from FireBaseData', function() { 
      assert.equal('', TestFirebase.child('SomeNode')); 
     }); 
    }); 

    describe('$MockFirebaseArray', function() { 
     it('should have the data array passed', function() { 
      var TestData = $MockFirebaseArray(TestArray); 
      assert.equal(TestArray.length, TestData.length); 
     }); 
    }); 

    describe('$MockFirebaseObject', function() { 
     it('should have the data object passed', function() { 
      var TestData = $MockFirebaseObject(TestArray[0]); 
      assert.equal(TestArray[0].length, TestData.length); 
      assert.deepEqual(TestArray[0], TestData); 
     }); 
    }); 
}); 

这表明嘲弄正在返回的数据,这是我想从实际访问火力地堡望而却步。现在,当我尝试在测试中使用我的工厂时,出现错误。

测试厂:

describe('Teams Module', function() { 
    beforeEach(module('MyApp'));  // Load My Application 

    describe('AllTeams Array', function() { 
     // Create Test Data 
     var TeamData = [ 
      { "Key": 1, "Name":"Team 1", "Logo": "Team1.jpg" }, 
      { "Key": 3, "Name":"Team 3", "Logo": "Team3.jpg" }, 
      { "Key": 2, "Name":"Team 2", "Logo": "Team2.jpg" }, 
     ]; 

     beforeEach(function() { 
      module(function($provide) { 
       var MockData = MockFirebaseData(); 
       $provide.value('FireBaseData', MockData); 
       $provide.value('$firebaseArray', $MockFirebaseArray(TeamData)); 
      }); 
     }); 

     it('can get an instance of AllTeams factory', inject(function(AllTeams) { 
      assert.isDefined(AllTeams); 
     })); 
    }); 
}); 

返回错误:

PhantomJS 1.9.8 (Windows 7 0.0.0) 
Teams Module 
AllTeams Array 
can get an instance of AllTeams factory FAILED 
TypeError: '[object Object],[object Object],[object Object]' is not a function (evaluating '$firebaseArray(FireBaseData.child('Teams'))') 
    at app/Team.js:9 
+0

'$ MockFirebaseArray'返回一个数组。但是,在“AllTeams”中,您试图将其用作函数。 –

+0

@AnidMonsur我试图修复我的工厂来验证它返回什么,因为我相信$ firebaseArray()调用通常返回数组数据。我正在努力为我的工厂获得同样的东西。也许我的注射是错误的?我想验证我恢复了我所期望的,所以我可以使用相同的结构化模拟与使用AllTeams显示列表的Controller一起工作。 –

+0

也许你会更好地使用[mockfirebase](https://github.com/katowulf/mockfirebase)模拟Firebase的参考资料,并保留原有的AngularFire方法。 –

回答

1

相反的:

$provide.value('$firebaseArray', $MockFirebaseArray(TeamData)); 

试试这个:

$provide.value('$firebaseArray', $MockFirebaseArray); 

我相信这是你打算首先做的。注入后,您的工厂将能够呼叫$firebaseArray作为功能。

+0

这似乎超过了我目前的错误。那么更快的问题就是如何传递数据来恢复它?在我的原始行中,发送数组将返回。我想现在检查我的AllTeams返回3条记录,以便我可以处理它们。 –

+1

它仍然返回你传入的内容。但是你传入一个空字符串,因为这是'MockData.child()'返回的内容。 –

+0

非常感谢。这正是我所期待的。我的测试现在通过了我的工厂的良好代码覆盖。现在我只需要弄清楚如何实际移动mockfirebase库以允许我测试Firebase添加/删除工厂功能的其余部分。 –