Testing Angular.js
As an Angular.js developer, I want to test my code, so that I can feel reasonably good about it. This sounds like a “User Story” in Scrum but it’s more of a developer quest. So in this blog we will look at ways to test our Angular.js code.
Let’s see what we need -
- We need to write unit tests for our
- Controllers
- Services
- Directives
- Filters
- We need to write end-to-end tests so that we can be sure everything is working well together
To kick off things, lets build a simple app which has a -
- Login page, on which the user enters a username / password
- Main page, which is shown if the user is successfully logged in
- On the main page, we fetch data from a server and show it to the user
Simple enough, this would look something like -
We have 2 controllers and 2 services in this setup -
controllers/login.coffee
controllers/main.coffee
services/webservice.coffee
services/helperservice.coffee
Now, a piece of advice. We must try to keep our controllers lean and move logic in a OO or functional manner to services. As we will see services are easier to test.
Second point worth noting is that to run tests etc. we need a build system. I recommend Yeoman which uses the excellent Grunt task runner. So to setup a Yeoman + Angular project, you can check out the Yeoman site.
From hereon, I assume that we are using the Yeoman setup. The default Yeoman setup gives us a sample controller unit test in Jasmine, which we can modify to this -
spec/controllers/main.coffee
Two main things to note are -
- the use of Angular’s dependency injection services which gives us a reference to the $injector itself which we use to mock the HTTP service (which is used by the MainCtrl’s constructor)
- and a reference to the controller itself
Since we kept our Controllers lean, there is nothing much to test. To test the services is much easier -
spec/services/helper.coffee
Once again the $injector does the hard work of giving us our service’s reference and from there on it’s an easy task.
So we are done with writing some unit tests, you can run “grunt test” to test them.
Time to write some end-to-end tests. Unfortunately at the time of writing this blog, Yeoman does not fully support running end-to-end tests out of the box. It gives us a Karma end-to-end config file but does not provide a mechanism to run it.
To run it, we need to do a few changes to our Gruntfile.js -
Remove the grunt:test task and add these new two tasks
Then modify the karma task like this
In the karma-e2e.conf.js file, uncomment the last two lines -
We now have two Grunt tasks to run Unit and E2E tests respectively - grunt test:unit and grunt test:e2e
Let us write a E2E test -
test/e2e/smoke.coffee
Like I mentioned we can run the E2E tests by running grunt test:e2e, this task first starts the test http server and then the tests are run over that.
That’s it, we have run our Unit and End-to-End tests successfully. Happy hacking with Angular.js.