Jose Antonio Pio Gil

Software Engineering

Jasmin SpyOn andResolve andReject

After working a bit with angular 1 I've found some issues to automate promises testing. I had figure out that testing HTTP communication is not always a simple endevour and it could take some effort to test a controller that handles a couple of API calls and reacts on them.

So here is a simple case on a controller injecting a User $resource and its test spec. I would dive the case in 2 ways: Canonical Way and Testing Concerns Way.

 

Canonical Way


  @newco.controller 'SignUpController', ( $scope, User) ->

  $scope.saveUser = (form) ->
    new User($scope.account).$save().then (response) ->
      # Do something when Success
    , (response) ->
      # Do something else when fail


 

So the way to test this code would be to Mock the http request from User $resource and retuning 200 or 401 in case there is an error.


  # Simple Jasmine test to ensure the test suit is able to run

describe 'SignUpController', ->
  controller = null
  scope = null

  beforeEach(module('myModule'))

  beforeEach inject ($controller, $rootScope, User) ->
    scope = $rootScope.$new()

    controller = $controller 'SignUpController',
      $scope: scope
      User: User


  describe 'when Success', ->
    beforeEach inject ($httpBackend) ->
      $httpBackend.expectPOST("User/api/url").respond 200, { content: "some content" }

    it 'Should so something', ->
      scope.saveUser({})
      $httpBackend.flush()
      expect("#Do Something").toBeTruthy()

  describe 'when Fail', ->
    beforeEach inject ($httpBackend) ->
      $httpBackend.expectPOST("User/api/url").respond 422, { data: { errors: "A bunch of errors" } }

    it 'Should so something else', ->
      scope.saveUser({})
      $httpBackend.flush()
      expect("#Do Something else").toBeTruthy()

 

Testing Concerns Way

So in order to test actually what the controller does and not what the $resource object does I suggest to move the mocking one level up in the onion skins layers and mock the promise returned by the $resource instead of what the resource does. For doing this I would like to modify this controller in order to have a more clear vision of the mocked promised.


  @newco.controller 'SignUpController', ( $scope, User) ->

  $scope.saveUserObect= ()->
    new User($scope.account).$save()

  $scope.saveUser = (form) ->
    $scope.saveUserObect().then (response) ->
      # Do something when Success
    , (response) ->
      # Do something else when fail

and the spec could look like this:


  # Simple Jasmine test to ensure the test suit is able to run

describe 'SignUpController', ->
  controller = null
  scope = null

  beforeEach(module('myModule'))

  beforeEach inject ($controller, $rootScope, User) ->
    scope = $rootScope.$new()

    controller = $controller 'SignUpController',
      $scope: scope
      User: User


  describe 'when Success', ->
    beforeEach inject ($q) ->
      defferred = $q.defer()
      defferred.resolve({ content: "some content" })
      spyOn(scope, "saveUserObect").andReturn (defferred.promise)->

    it 'Should so something', ->
      scope.saveUser({})
      scope.$apply()
      expect("#Do Something").toBeTruthy()

  describe 'when Fail', ->
    beforeEach inject ($q) ->
      defferred = $q.defer()
      defferred.resolve({ data: { errors: "A bunch of errors" } })
      spyOn(scope, "saveUserObect").andReturn (defferred.promise)->

    it 'Should so something else', ->
      scope.saveUser({})
      scope.$apply()
      expect("#Do Something else").toBeTruthy()

After using this spec for a while I realize that I can extend the promise generation to a Jasmine Spy prototype method and reused as needed in other tests. So I came up whit this solution:

 


  # Simple Jasmine test to ensure the test suit is able to run

describe 'SignUpController', ->
  controller = null
  scope = null

  beforeEach(module('myModule'))

  beforeEach inject ($controller, $rootScope, User, $q) ->
    scope = $rootScope.$new()

    jasmine.Spy.prototype.andResolve = (value)->
      @plan = ()->
        defferred = $q.defer();
        defferred.resolve(value);
        defferred.promise;
      @

    jasmine.Spy.prototype.andReject = (value)->
      @plan = ()->
        defferred = $q.defer();
        defferred.reject(value);
        defferred.promise;
      @


    controller = $controller 'SignUpController',
      $scope: scope
      User: User


  describe 'when Success', ->

    it 'Should so something', ->
      spyOn(scope, "saveUserObect").andResolve({ content: "some content" })
      scope.saveUser({})
      scope.$apply()
      expect("#Do Something").toBeTruthy()

  describe 'when Fail', ->

    it 'Should so something else', ->
      spyOn(scope, "saveUserObect").andReject({ data: { errors: "A bunch of errors" } })
      scope.saveUser({})
      scope.$apply()
      expect("#Do Something else").toBeTruthy()
    

 

I would like to know how to extend this prototype to the whole test environment so I can reuse them in other test. So far here is my idea of how you can improve a littler your tests readability.

 

Do you know how to convert the beforeEach inside one describe to a general before each. Or do you know how to extend jasmine.Spy with and AngularJs dependency?


  # Extend this for whole test suite or in general as dependent on Angularjs $q.

    jasmine.Spy.prototype.andResolve = (value)->
      @plan = ()->
        defferred = $q.defer();
        defferred.resolve(value);
        defferred.promise;
      @

    jasmine.Spy.prototype.andReject = (value)->
      @plan = ()->
        defferred = $q.defer();
        defferred.reject(value);
        defferred.promise;
      @