Error de tecleado:No se puede leer la propiedad 'entonces' de la prueba angularjs-grunt indefinida
-
21-12-2019 - |
Pregunta
Estoy usando el servicio $q para realizar llamadas asíncronas.No puedo resolver "entonces" y "aplazar" en pruebas unitarias usando karma.
El siguiente es mi código de controlador.
scope.filterUrls = [{url:'page1'}, {url: 'page2'}, {url:'page-error'}];
scope.bindFilters = function () {
angular.forEach(scope.filterUrls, function (data) {
scope.getFilterData(data.url, '').then(function (result) {
if (data.url === 'page1') {
scope.moduleData.index = result.data;
} else if (data.url === 'page2') {
scope.moduleData.page2 = result.data;
}
});
});
}
scope.getFilterData = function (filterUrls, params) {
// $q service object
var deferred = q.defer();
// regular ajax request
http({
method: 'GET',
url: app.api.root + filterUrls,
params: params
})
.success(function (result) {
// promise resolve
deferred.resolve(result);
})
.error(function (result) {
// called asynchronously if an error occurs
// or server returns response with an error status.
deferred.reject('Erreur request : ' + result);
});
return deferred.promise;
};
especificaciones de prueba:
it('should call getFilterData() in bindFilters()', function () {
spyOn(scope, 'getFilterData');
scope.bindFilters();
expect(scope.getFilterData).toHaveBeenCalled();
});
Recibo un error llamado "TypeError:No se puede leer la propiedad 'entonces' de indefinido".
¿Cómo puedo escribir una prueba unitaria para esos dos métodos usando karma?
Actualizar:
1. ¿Cómo podemos probar el éxito y el error de alcance.getFilterData()?
2. Luego funcione en la función alcance.bindFilters().
Por favor ayuda..
Solución
Si usted solo necesita saber si getFilterData
se llama o no, intenta devolver una promesa falsa falsificando la función:
Con Jasmine 1.3, podríamos usar andCallFake
:
it('should call getFilterData() in bindFilters()', function () {
spyOn(scope, 'getFilterData').andCallFake(function(){//replace with a fake function
var deferred = $q.defer(); //assume that you already inject $q service in beforeEach and save it as a variable.
return deferred.promise; //returns a fake promise
});
scope.bindFilters();
expect(scope.getFilterData).toHaveBeenCalled();
});
Con Jasmine 2.0, podríamos usar and.callFake
en cambio.
Otra solución es utilizar andReturnValue
y $q.when()
:
it('should call getFilterData() in bindFilters()', function () {
spyOn(scope, 'getFilterData').andReturnValue($q.when());
scope.bindFilters();
expect(scope.getFilterData).toHaveBeenCalled();
});
Con Jasmine 2.0, podríamos usar and.returnValue
en cambio.