Saturday, 6 February 2016

Unit testing with Karma and Jasmine for AngularJS

So you’ve started to build applications with AngularJS; or, maybe you’ve done unit testing before but haven’t used it with AngularJS; or, maybe it’s all new to you, but either way this article should help to orient you to the value of using Karma with Jasmine and offer some tips on displaying functionalities of Angular JS. Let’s get started!
My app is small it doesn’t need any tests, right?
Wrong!
A small app can have errors and it’s bad practice to have an untested app.
Besides, a small app is the perfect place to get started in using proper unit testing because there won’t be many tests and the functionality will be less complex.
First of all, what is Unit Testing?
Unit testing is very important for both the quality of the code and the well being of the developer. It allows for them to freely change their code and not just hope that it’s not a breaking change, as the test will recognize the break straight away. This will minimize the number of regressions in the code.
Unit tests are all about splitting your code into small testable modules with each module having its own functionality and level of abstraction. This is where AngularJS shines.
An added bonus is that the unit tests will act as a type of documentation because it describes the expected behavior of the functions and modules. Unit tests don’t care how your code does the task, it just cares about the results.
Okay, but when should I write unit tests?
Short answer – the start of the project is the ideal time. You will have all your tests and then can work toward writing your code with passing them in mind.
Sadly it is unrealistic to expect complete coverage with unit testing, so my suggestion is write as many as you can possibly think of at the start, and if new components appear, then they should be tested too; a more iterative process of revisiting tests should be used.
What testing framework should I use?
AngularJS can be tested in a lot of ways, by lots of testing frameworks, including but certainly not limited to mocha, jasmine, qunit, and sin on.
There are a lot of choices, but here’s the secret: your choice really doesn’t matter because there is no “best” option; they are all very similar.
If you are torn and love brilliant documentation (who doesn’t) I would choose Jasmine (used proudly by the Logentries team btw).
After you pick your testing framework, you can now choose a task runner. The most well known include Karma and Protractor (both by the Angular team). For our examples we will be using Karma.
Now that you know some basics on why you should be using unit test, and you have an Angular project ready, lets get our hands on an example.
We’ll need to install some dependencies for running unit tests with Karma and Jasmine.
To do this, you should run the following lines on the command line:
npm install karma --save-dev
npm install karma-phantomjs-launcher --save-dev
npm install karma-jasmine --save-dev
Now configure your karma.conf.js file using this guide.
How to design a test?
Scenario
  • “A calculator that adds and subtracts”
This scenario can be broken into features, which will in turn be broken into units of testable code.
The calculator has two main features, adding and subtracting numbers. These features can be split into units, such as the methods for addition and subtraction. These methods need to be tested and there may be many different cases that need to be taken into account. A possible example is below, but you may come up with some more tests.
describe('Calculator ', function() {

// necessary
it('should add two numbers correctly', function() {});
it('should subtract two numbers correctly', function() {});

// helpful but not needed
it('should add negative numbers', function() { });
it('should reject non numbers', function() { });
});

The “describe()” method defines our test suite. This takes a string with the convention that it describes the module which is being tested.
An “it()” is a proper test spec. Put what you want to be tested here, the convention is that it should read as a sentence, e.g. ‘it is expected to do this’ reads as “it is expected to do this”
How do I use it with my AngularJS app?
Here is an example of a very basic controller. It only has a scope variable and we will test if this value is correct.
 var myApp = angular.module('myApp',[]);
myApp.controller('HelloWorldController', ['$scope', function($scope) {
$scope.greeting = 'Hello World!';
}]);
Jasmine Test
describe(‘Hello World example ’, function() {

beforeEach(module(‘myApp’));

var HelloWorldController,
scope;

beforeEach(inject(function ($rootScope, $controller) {
scope = $rootScope.$new();
HelloWorldController = $controller('HelloWorldController', {
$scope: scope
});
}));
it('says hello world!', function () {
expect(scope.greeting).toEqual("Hello world!”);
});

});

Run your test with:
karma start karma.conf.js

Every “beforeEach()” function is executed right before the tests are conducted. In the above case the function gets access to your angular module and then injects the controller and scope, so that it can be tested. This allows you to set up for each assertion, allowing the same clean set up for all your tests.
Some of the possible expectations include:
  • toEqual, toBe, toBeTruthy, toBeFalsy
  • toBe for negation
Jasmine has you covered if you want to ignore some of the tests temporarily. “xdescribe” will ignore a “describe” and “xit” will ignore a specific test.
Conclusion
So now you can run a basic unit test for your AngularJS app and slowly begin to build out to more complex tests. Once you’ve got your unit tests in place you have the foundations for a great application!

No comments:

Post a Comment

Angular Tutorial (Update to Angular 7)

As Angular 7 has just been released a few days ago. This tutorial is updated to show you how to create an Angular 7 project and the new fe...