Unit Testing $interval in AngularJS - ngMock Fundamentals

Following on from the post that detailed unit testing $timeout, this post looks into ngMock’s $interval service. To recap, Angular’s $interval service is a wrapper for the browser’s native implementation of setInterval(), which allows us to define an expression to be evaluated at set intervals (in milliseconds).
The API has the following methods:
MethodsDescription
$intervalWe use the service instance itself as a function. It returns an instance of the interval.
cancelWe can cancel an interval, by passing the instance in question to this method.
flush([milliseconds])Executes any interval tasks that are set in the code under test. We can optionally provide a delay in milliseconds to pinpoint the place in time.
When using the $interval function, we can pass the following arguments:
ArgumentDescription
functionThis will be the function we wish to execute at the set interval.
delayThe delay in milliseconds between each execution of our function.
count (optional)We can optionally supply a number indicating how many times we wish to repeat the interval until it is stopped. If we do not supply a value or pass an argument of 0, it will run indefinitely (or until we cancel the interval via the cancel method).
invokeApply (optional)Default value is true. If we set this to be false, any changes made to the $scope object within the function will not be immediately visible. This will be discussed later in the post.
Pass (optional)If our function to be executed required arguments, we can pass these values here in comma separated form e.g. ‘arg 1’, ‘arg 2’, { arg3: ‘value’ }.

Unit Testing an $interval

Example Application using $interval

In this example, we will only make use of the arguments functiondelay and count when creating the $interval. It’s a basic app that has two intervals called every 1 second. The interval calls trivial functions that start with a random number and add 1 upon each execution. The first interval will run indefinitely and the second will only execute 10 times before it stops. We also have a cancel function, that allows us to stop both intervals via a click event.
Here’s the code:
var app = angular.module('calculatorApp', []);

app.controller('CalculatorController', function calculatorController($scope, $interval) {

  $scope.result1 = Math.random();
  $scope.result2 = Math.random();
    
  $scope.sum1 = function sum1() {
    $scope.result1 = $scope.result1 + 1;
  }

  $scope.sum2 = function sum2() {
    $scope.result2 = $scope.result2 + 1;
  }

  /* Sum Interval #1 - run every 1 second */
  var sumInterval = $interval($scope.sum1, 1000); 

  /* Sum Interval #2 - run every 1 second, but stop after 10 iterations */
  var sumIntervalTenTimes = $interval($scope.sum2, 1000, 10); 

  $scope.cancel = function() {
    if (angular.isDefined(sumInterval) || angular.isDefined(sumIntervalTenTimes)) {
      $interval.cancel(sumInterval);
      $interval.cancel(sumIntervalTenTimes);
      sumInterval = undefined;
      sumIntervalTenTimes = undefined;
    }
  }

  /* Note: we should always have this code in place, to ensure we don't leak intervals. */
  $scope.$on('$destroy', function() {
    // ensure that we always close any running intervals when a controller instance is un-loaded.
    $scope.cancel();
  });

});
Here’s how the app would behave in a browser (the starting point is marked in the gif):
unit testing interval angularjs
Note how _Sum Interval #2 stops after 10 iterations._
How would we go about unit testing this application? Since we are not actually unit testing that interval works, we don’t need to concern ourselves with testing behaviour, so in most cases we only need to test that we are calling interval with the correct parameters and using its API in the correct way.

Checking that intervals are setup correctly

The following test code asserts that the interval services are configured correctly, when the controller is initialised:
it('should register the intervals', function () {
  
  // We make use of Jasmine's createSpy functionality
  var $intervalSpy = jasmine.createSpy('$interval', $interval);

  // Create the controller instance and pass in our spy in place of the $interval instance.
  var calculatorController = $controller('CalculatorController', { $scope: $scope, $interval: $intervalSpy });

  /* Assertions */

  // We can ask if the spy was called
  expect($intervalSpy).toHaveBeenCalled();

  // Or to make more accurate assertions, such as the number of times called
  expect($intervalSpy.calls.count()).toBe(2);

  // ... or the arguments passed
  expect($intervalSpy).toHaveBeenCalledWith($scope.sum1, 1000);
  expect($intervalSpy).toHaveBeenCalledWith($scope.sum2, 1000, 10);

  // We can also access the calls to the $interval spy as follows:
  var calls = $intervalSpy.calls.all();
  var args0 = calls[0].args; // first
  var args1 = calls[1].args; // second
  // .. do something interesting here...
  
});
Key points:
We can make use of Jasmine’s createSpy, so that we can track calls, arguments etc to the $interval service. Creating the spy in this way will not call ngMock’s actual implementation of $interval:
var $intervalSpy = jasmine.createSpy('$interval', $interval);
We can make use of expectations to verify properties of our spy, such as whether it’s been called:
expect($intervalSpy).toHaveBeenCalled();
We can also check what arguments have been passed to $interval calls, this is useful to verify that we are calling $interval in the code under test with the correct function, delay etc. The following snippet checks that we called timeout for sum1 with a 1 second delay and no count value, and the second line checks that we setup sum2 with a 1 second delay and count of 10 times:
expect($intervalSpy).toHaveBeenCalledWith($scope.sum1, 1000);
expect($intervalSpy).toHaveBeenCalledWith($scope.sum2, 1000, 10);

Checking that intervals are cancelled correctly

Next we need to consider how we could test the code that cancels the intervals when the ‘cancel’ button is clicked. It’s similar to the test code we just examined, with a few additions:
it('should cancel the intervals on click', function () {

  // Note that we've added .and.callThough();
  var $intervalSpy = jasmine.createSpy('$interval', $interval).and.callThrough();

  // we need to register a spy for $interval's cancel function.
  spyOn($intervalSpy, 'cancel');

  var calculatorController = $controller('CalculatorController', { $scope: $scope, $interval: $intervalSpy });

  // execute the cancel method
  $scope.cancel();

  expect($intervalSpy.cancel.calls.count()).toBe(2);

  // how do we assert that cancel is called with the correct interval instances?
  expect($intervalSpy.cancel.calls.argsFor(0)[0].$$intervalId).toBe(0);
  expect($intervalSpy.cancel.calls.argsFor(1)[0].$$intervalId).toBe(1);
  
});
Key points:
For this test we create the spy with the addition of .and.callThrough(). Doing this means that ngMock’s underlying functionality will be called in addition to the spy:
var $intervalSpy = jasmine.createSpy('$interval', $interval).and.callThrough();
We need to do this, since in our controller code we keep track of the interval instances so that we can cancel them. If we didn’t call the ngMock functionality, the sumInterval variable in the following snippet would be undefined and we wouldn’t be able to assert the cancel function is called correctly. This snippet demonstrates how the instance is assigned, so that it can be cancelled:
var sumInterval = $interval($scope.sum1, 1000); 

// we need to pass the sumInterval instance in order to cancel it.
$interval.cancel(sumInterval);
Lastly, we can check that the expected interval instances are passed as arguments to cancel by their $$intervalId value. The convention in ngMock’s implementation is to increment the id of each new $interval by 1. Here’s how to verify that cancel calls the expected $interval instances:
expect($intervalSpy.cancel.calls.argsFor(0)[0].$$intervalId).toBe(0);
expect($intervalSpy.cancel.calls.argsFor(1)[0].$$intervalId).toBe(1);
Example Code:
The full test code is available via this gist.

Unit Testing $interval behaviour

The test code we just reviewed would cover most unit testing scenarios, however, what if we wanted to test some behaviour that involved an interval? In the example that follows, we will also include how to use the invokeApply and passarguments.

Example Application

In this example application, we have an interval that executes every 1 second. The function will take a starting number and an increment value, and will increment the starting number with the given increment value every 1 second.
Here’s the code:
var app = angular.module('calculatorApp', []);

app.controller('CalculatorController', function calculatorController($scope, $interval) {

  var counterInstance;

  // the main function that increments the number using the function arguments for starting and increment values.
  var counterFunction = function(start, inc) {
    if ($scope.ticker === 0 && start)
      $scope.ticker = start;

    if (!inc)
      inc = 1;

    $scope.ticker += inc;
  }

  // defaults
  $scope.ticker = 0;
  $scope.from = 0;
  $scope.inc = 1;

  $scope.start = function() {
   // When the 'start' button is clicked, the interval is started with the given parameters.
    counterInstance = $interval(counterFunction, 1000, $scope.times, true, $scope.from, $scope.inc);
  }

  /* Cancel functionality removed for brevity (same as last example app)... */

});
Here’s how the app would behave in a browser (the starting point is marked in the gif):
unit testing interval behaviour angularjs
The following line of code demonstrates how we can pass our two function arguments start and inc. We also have to set invokeApply to true (which is the default value). Here’s the line of code:
$interval(
 counterFunction, 
 1000,      // delay (milliseconds)
 0,      // count (zero means no limit)
 true,      // invokeApply (the default is true)
 $scope.from, $scope.inc // pass (we can comma separate args)
);
When would we set invokeApply to false? In the counterFunction example, the scope is updated each time ($scope.ticker is updated), which in turn is reflected in the UI. Doing this is a costly operation, since we are causing Angular’s digest loop to run each time. If you’ve yet to look into the digest loop, put simply it checks each time to see if the scope has changed, and if so, whether the change(s) should be reflected in the HTML. Each time the counterFunction is executed, under the covers the $apply function is called which triggers this cycle of checking for changes.
If a function isn’t doing anything with the $scope object, it would be a waste of resource to trigger this cycle each time, so we would set invokeApply to be false. You can experiment with this by setting the value to false in our example app, but doing so would cause the starting number not to be updated in the view each time the interval executes.

Unit Testing the Behaviour

With this approach we won’t make use of Jasmine’s spy functionality, we will use ngMock’s $interval implementation to control the execution flow and to check values on our $scope object at certain points in time to verify that our code is doing what we expect. The key function we will use is mock.flush([delay]). The tests that follow make use of flush to advance the interval to a specified point in time:
describe('calculator tests', function () {
  
  beforeEach(module('calculatorApp'));

  var $controller;
  var $scope;
  var $interval;
  var $rootScope;

  beforeEach(inject(function(_$controller_,  _$rootScope_, _$interval_) {
    $controller = _$controller_;
    $rootScope = _$rootScope_;
    $scope = _$rootScope_.$new();
    $interval = _$interval_;
  }));

  it('should start the interval on click using defaults', function () {
    var calculatorController = $controller('CalculatorController', { $scope: $scope });

    $scope.start();

    expect($scope.from).toBe(0);
    expect($scope.inc).toBe(1);
    expect($scope.ticker).toBe(0);

    // advance in time by 4 seconds
    $interval.flush(4000);

    expect($scope.ticker).toBe(4);
  });

  it('should start the interval on click with user values', function () {

    var calculatorController = $controller('CalculatorController', { $scope: $scope });

    $scope.from = 5;
    $scope.inc = 2;
    $scope.times = 10;

    $scope.start();

    expect($scope.from).toBe(5);
    expect($scope.inc).toBe(2);
    expect($scope.ticker).toBe(0);

    // advance in time by 1 second from call to start()
    $interval.flush(1100);

    expect($scope.ticker).toBe(7);

    // advance in time by 2 seconds from call to start()
    $interval.flush(1100);

    expect($scope.ticker).toBe(9);

    // this interval stops after 10 iterations, and therefore should not pass 25!
    $interval.flush(10000);
    expect($scope.ticker).toBe(25);

    // Another example to demonstrate the point...
    $interval.flush(10000);
    expect($scope.ticker).toBe(25);
  });

});
NB The angular documentation states that the argument for flush is optional, but keep in mind that calling flush() with no arguments has no effect.
Key Points:
In the first test, we advanced time by 4 seconds from when the interval was registered by calling the flush function with 4 seconds (as milliseconds):
$interval.flush(4000);
Using the default values of starting from zero and incrementing by 1, after 4 seconds, our counterFunction should have been called 4 times, and therefore $scope.ticker should be 4.
In the second test, we call the flush function twice, advancing the interval to 1 second and then 2 seconds. Our arguments state that we should start from 5, and add 2 every 1 second. Checking the state after 1 second we expect $scope.tickerto be 7, and after another second it should be 9.
The interval in the second test should also stop after 10 iterations. The test has an expectation to demonstrate this. We started with 5, and add 2 every 1 second, 10 times. Therefore the result should never exceed 25:
$interval.flush(10000);
expect($scope.ticker).toBe(25);
Example Code:
The full test code is available via this gist.

Deeper Dive into ngMock and $interval

ngMock has it’s own implementation of $interval as opposed to using a decorator as we have discussed in other posts e.g. $timeout.
If you were to examine ngMock’s source code, you will find the module definition of ngMock and see how the $IntervalProvider is registered in place of angular’s main implementation:
angular.module('ngMock', ['ng']).provider({
  // other services removed for brevity...
  $interval: angular.mock.$IntervalProvider
  // ...
});
For reference here is the source code for $interval in ngMock.
Internally, when we create a new $interval instance, the function argument is added to an array of functions. Each function is given a unique id and the point in time at which it was registered is stored. Each time we call flush, all the functions in the array are inspected and if the last time an interval was executed is less than the ‘now time’ + the milliseconds argument passed to flush, the tick function for the interval in question gets called.
The tick function will execute the function passed to the interval when it was created, and each time will check to see if an interval should be cancelled if it reaches the count value. If the count value is reached, the interval in question is removed from the array based on its unique and id and interestingly, a call to deferred.resolve(iteration); is made.
What’s deferred? It’s used by the promise variable returned by the $interval function. In the code snippet below, we see how the call to $interval returns a Promise that’s created by the function:
var instanceOfAPromise = $interval(myFunction, myDelay);
Angular has its own promise service called $q. A promise is an approach to make asynchronous more readable. Rather than passing callbacks as function arguments, we can register the callbacks via functions called thencatch or finally that will be executed when the interval instance completes or is cancelled.
In the code below, we can register the three functions, thencatch, and finally via the promise instance that’s returned during creation:
instanceOfAPromise.then(function(successCallback, errorCallback, notifyCallback) {
  console.log('The interval was executed: ' + successCallback + ' times!');
  console.log('Then status: ' + counterInstance.$$state.status);
  // errorCallback (always undefined for interval)
  // notifyCallback (always undefined for interval)
});

instanceOfAPromise.catch(function(errorCallback) {
  // called when the interval is cancelled
  if (errorCallback === 'canceled') {
    console.log('the interval was cancelled!');
  } else {
    console.log('something else went wrong!');
  }
  console.log('Catch status: ' + counterInstance.$$state.status);
});

instanceOfAPromise.finally(function() {
  // always called regardless
  console.log('Finally status: ' + counterInstance.$$state.status);
});
The key points of this snippet are:
  • then will be called only when an $interval completes e.g. if we specified it should run only 3 times, i.e. this would not be called for an interval that ran indefinitely.
  • catch will be called only if the $interval instance is cancelled.
  • finally is always called, if completed or cancelled.
The arguments passed to the callback functions e.g. successCallbackerrorCallback etc are set within the $IntervalProvider code.
The promise also has the following properties. We used $$state.status to printout the status in the previous code snippet and we’ve made use of $$intervalId in earlier unit tests which is unique per interval:
instance.$$intervalId // an id that links the promise to the interval
instance.$$state // an internal object used by the promise to retain its state.
The possible values of $$state.status are:
  • 0 - running
  • 1 - done
  • 2 - cancelled

All Example Code

All the main applications examples from this article can be found via the following github gists:

See more ngMock Tutorials

This is one of many ngMock tutorials. See the full list of AngularJS Unit Testing Tutorials.

Unit Testing $interval in AngularJS - ngMock Fundamentals Unit Testing $interval in AngularJS - ngMock Fundamentals Reviewed by Nhung Pham on 04:03:00 Rating: 5

Không có nhận xét nào:

Được tạo bởi Blogger.