I'm just starting out writing tests for my AngularJS app and am doing so in Jasmine.
Here are the relevant code snippets
ClientController:
'use strict';
adminConsoleApp.controller('ClientController',
function ClientController($scope, Client) {
//Get list of clients
$scope.clients = Client.query(function () {
//preselect first client in array
$scope.selected.client = $scope.clients[0];
});
//necessary for data-binding so that it is accessible in child scopes.
$scope.selected = {};
//Current page
$scope.currentPage = 'start.html';
//For Client nav bar
$scope.clientNavItems = [
{destination: 'features.html', title: 'Features'},
];
//Set current page
$scope.setCurrent = function (title, destination) {
if (destination !== '') {
$scope.currentPage = destination;
}
};
//Return path to current page
$scope.getCurrent = function () {
return 'partials/clients/' + $scope.currentPage;
};
//For nav bar highlighting of active page
$scope.isActive = function (destination) {
return $scope.currentPage === destination ? true : false;
};
//Reset current page on client change
$scope.clientChange = function () {
$scope.currentPage = 'start.html';
};
});
ClientControllerSpec:
'use strict';
var RESPONSE = [
{
"id": 10,
"name": "Client Plus",
"ref": "client-plus"
},
{
"id": 13,
"name": "Client Minus",
"ref": "client-minus"
},
{
"id": 23805,
"name": "Shaun QA",
"ref": "saqa"
}
];
describe('ClientController', function() {
var scope;
beforeEach(inject(function($controller, $httpBackend, $rootScope) {
scope = $rootScope;
$httpBackend.whenGET('http://localhost:3001/clients').respond(RESPONSE);
$controller('ClientController', {$scope: scope});
$httpBackend.flush();
}));
it('should preselect first client in array', function() {
//this fails.
expect(scope.selected.client).toEqual(RESPONSE[0]);
});
it('should set current page to start.html', function() {
expect(scope.currentPage).toEqual('start.html');
});
});
The test fails:
Chrome 25.0 (Mac) ClientController should preselect first client in array FAILED
Expected { id : 10, name : 'Client Plus', ref : 'client-plus' } to equal { id : 10, name : 'Client Plus', ref : 'client-plus' }.
Error: Expected { id : 10, name : 'Client Plus', ref : 'client-plus' } to equal { id : 10, name : 'Client Plus', ref : 'client-plus' }.
at null.<anonymous> (/Users/shaun/sandbox/zong-admin-console-app/test/unit/controllers/ClientControllerSpec.js:43:39)
Does anyone have any ideas on why this might be happening?
Also .. as I'm new to writing AngularJS tests, any comments on whether I'm setting up my test wrong or whether it can be improved will be welcome.
Update:
Including ClientService:
'use strict';
AdminConsoleApp.services.factory('Client', function ($resource) {
//API is set up such that if clientId is passed in, will retrieve client by clientId, else retrieve all.
return $resource('http://localhost:port/clients/:clientId', {port: ':3001', clientId: '@clientId'}, {
});
});
Also, I got around the problem by comparing ids instead:
it('should preselect first client in array', function () {
expect(scope.selected.client.id).toEqual(RESPONSE[0].id);
});
toEqual
makes a deep equality comparison. Which means that when all the properties of the objects' values are equal, the objects are considered to be equal.As you said, you are using resource which adds a couple of properties to the objects in the array.
So this
{id:12}
becomes this{id:12, $then: function, $resolved: true}
which are not equal. Id checking should be fine if you are just testing if you set the values properly.I had the same problem so I just called
JSON.stringify()
on the objects to be compared.A little bit verbose, but produces helpful message when expectation fails:
Explanation:
angular.toJson
will strip the resource of all angular specific properties like$promise
JSON.parse
will convert JSON string back to normal Object (or Array), which can now be compared to another Object (or Array).Short answer:
The existing answers all recommend either stringifying your objects, or creating a custom matcher/comparison function. But, there is an easier way: use
angular.equals()
in your Jasmineexpect
call, instead of using Jasmine's built-intoEqual
matcher.angular.equals()
will ignore the additional properties added to your objects by Angular, whereastoEqual
will fail the comparison for, say,$promise
being on one of the objects.Longer explanation:
I ran across this same problem in my AngularJS application. Let's set the scenario:
In my test, I created a local object and a local array, and expected them as responses to two GET requests. Afterwards, I compared the result of the GET's with the original object and array. I tested this using four different methods, and only one gave proper results.
Here's a portion of foobar-controller-spec.js:
For reference, here's the output from
console.log
usingJSON.stringify()
and.toString()
:Notice how the stringified object has extra properties, and how
toString
yields invalid data which will give a false positive.From looking at the above, here's a summary of the different methods:
expect(scope.foobar).toEqual(foobar)
: This fails both ways. When comparing objects, toString reveals that Angular has added extra properties. When comparing arrays, the contents seem identical, but this method still claims they are different.expect(scope.foo.toString()).toEqual(myFooObject.toString())
: This passes both ways. However, this is a false positive, since the objects are not being translated fully. The only assertion this makes is that the two arguments have the same number of objects.expect(JSON.stringify(scope.foo)).toEqual(JSON.stringify(myFooObject))
: This method gives the proper response when comparing arrays, but the object comparison has a similar fault to the raw comparison.expect(angular.equals(scope.foo, myFooObject)).toBe(true)
: This is the correct way to make the assertion. By letting Angular do the comparison, it knows to ignore any properties which were added in the backend, and gives the proper result.If it matters to anyone, I'm using AngularJS 1.2.14 and Karma 0.10.10, and testing on PhantomJS 1.9.7.
I just had a similar problem and implemented a custom matcher as follows, based on many approaches:
and then used this way:
Of course, it's a very simple matcher and doesn't support many cases, but I didn't need anything more complex than that. You can wrap the matchers in a config file.
Check this answer for a similar implementation.
Long story short: add
angular.equals
as a jasmine matcher.So, then you can use it as follows: