KEMBAR78
Slaven tomac unit testing in angular js | PPTX
Slaven Tomac (Amphinicy Technologies)
slaven.tomac@amphinicy.com
@slaventomac
 JavaScript unit testing
 AngularJS application testing tools
 Karma
 Jasmine
 Angular Mock
 What to test
 Example
 JavaScript testing problems
ā—¦ Mixed JS/DOM
ā—¦ Mixed backend calls into JS functions
 AngularJS tries to fight it
ā—¦ DOM manipulations only in directives
ā—¦ Business logic in services
ā—¦ Flow of application in controllers
ā—¦ . . .
 Karma – test launcher
 Assertion frameworks (Jasmine,
Mocha, QUnit …)
 AngularMock library (optional)
 Prerequsitions:
ā—¦ Node.js
ā—¦ NPM (Node Package Manager)
 Test runner
ā—¦ launches HTTP server
ā—¦ loads needed files
ā—¦ runs test
 All major browsers supported
 Available for testing on continuous integration server
 Configuration driven
npm install karma -g
 Assertion library
 Karma supports it via plugin
 (Custom) Matchers
 Setup and Teardown (beforeEach, afterEach)
 Spies
describe("One testing suite", function() {
it("contains spec with an expectation", function() {
var javaCroAtendees = 12;
expect(javaCroAtendees).toBe(12, "number of javaCroAtendees
should be 12ā€);
});
npm install karma-jasmine -g
 AngularJS mocking library
 Injecting and mocking Angular services
 Included mocks
ā—¦ $exceptionHandler
ā—¦ $log
ā—¦ $interval
ā—¦ $timeout
ā—¦ $httpBackend
$httpBackend
.when('GET', '/api/1.0/event')
.respond(
[
{
ā€˜event_name’ : ’javaCro’,
ā€˜location’ : ’Porec’
},
. . .
]
)
 Directives
 Services
 Controllers
 Filters
 Interceptors
 Resources
 . . .
 JavaScript
 Layout
 Functionalities
 Model changes ($apply required)
 Scope
Note:
($compile required)
element = $compile('<card-deck cards="myCards"></card-deck>');
<ul>
<li>Ace of Spades</li>
<li>Queen of Hearts</li>
</ul>
 Function definition
 Function testing
Note:
(if includes backend requests, $httpBackend mock required)
 Scope – variable instantiation
 Scope – function definitions and functionality
 Application workflow
Note:
($controller needed) beforeEach (inject(function ($controller) {
MainCtrl = $controller('MainCtrl', {
$scope: scope
});
}));
 Functionality
 DOM changes ($compile needed)
Note:
(dependency injection with suffix ā€˜Filter’)
beforeEach (inject(function(ageRangeFilter) {
myAgeRangeFilter = ageRangeFilter;
}));
 Application for displaying JavaCro atendees and filtering them
by age
 E2E tests
 How to make testing/developing more standard/automated?
ā—¦ Use Yeoman for scaffolding your app structure
ā—¦ Use Grunt for building, deploying and automated testing
yo angular
grunt serve
15%
60%
75% 75%
30%
90%
0%
10%
20%
30%
40%
50%
60%
70%
80%
90%
100%
2y ago 6m ago 3m ago EOY 2014
WebUI unit testing strategy
Web GUIs Unit tests Automated tests
Hey, let’s use AngularJS
 Use Karma as test runner
 Write your tests in Jasmine
 Integrate Karma on continuous integration server and
TEST, TEST, TEST, TEST… you’ll be happier later 
Thank you!

Slaven tomac unit testing in angular js

  • 1.
    Slaven Tomac (AmphinicyTechnologies) slaven.tomac@amphinicy.com @slaventomac
  • 2.
     JavaScript unittesting  AngularJS application testing tools  Karma  Jasmine  Angular Mock  What to test  Example
  • 3.
     JavaScript testingproblems ā—¦ Mixed JS/DOM ā—¦ Mixed backend calls into JS functions  AngularJS tries to fight it ā—¦ DOM manipulations only in directives ā—¦ Business logic in services ā—¦ Flow of application in controllers ā—¦ . . .
  • 4.
     Karma –test launcher  Assertion frameworks (Jasmine, Mocha, QUnit …)  AngularMock library (optional)  Prerequsitions: ā—¦ Node.js ā—¦ NPM (Node Package Manager)
  • 5.
     Test runner ā—¦launches HTTP server ā—¦ loads needed files ā—¦ runs test  All major browsers supported  Available for testing on continuous integration server  Configuration driven npm install karma -g
  • 6.
     Assertion library Karma supports it via plugin  (Custom) Matchers  Setup and Teardown (beforeEach, afterEach)  Spies describe("One testing suite", function() { it("contains spec with an expectation", function() { var javaCroAtendees = 12; expect(javaCroAtendees).toBe(12, "number of javaCroAtendees should be 12ā€); }); npm install karma-jasmine -g
  • 7.
     AngularJS mockinglibrary  Injecting and mocking Angular services  Included mocks ā—¦ $exceptionHandler ā—¦ $log ā—¦ $interval ā—¦ $timeout ā—¦ $httpBackend $httpBackend .when('GET', '/api/1.0/event') .respond( [ { ā€˜event_name’ : ’javaCro’, ā€˜location’ : ’Porec’ }, . . . ] )
  • 8.
     Directives  Services Controllers  Filters  Interceptors  Resources  . . .  JavaScript
  • 9.
     Layout  Functionalities Model changes ($apply required)  Scope Note: ($compile required) element = $compile('<card-deck cards="myCards"></card-deck>'); <ul> <li>Ace of Spades</li> <li>Queen of Hearts</li> </ul>
  • 10.
     Function definition Function testing Note: (if includes backend requests, $httpBackend mock required)
  • 11.
     Scope –variable instantiation  Scope – function definitions and functionality  Application workflow Note: ($controller needed) beforeEach (inject(function ($controller) { MainCtrl = $controller('MainCtrl', { $scope: scope }); }));
  • 12.
     Functionality  DOMchanges ($compile needed) Note: (dependency injection with suffix ā€˜Filter’) beforeEach (inject(function(ageRangeFilter) { myAgeRangeFilter = ageRangeFilter; }));
  • 13.
     Application fordisplaying JavaCro atendees and filtering them by age
  • 14.
     E2E tests How to make testing/developing more standard/automated? ā—¦ Use Yeoman for scaffolding your app structure ā—¦ Use Grunt for building, deploying and automated testing yo angular grunt serve
  • 15.
    15% 60% 75% 75% 30% 90% 0% 10% 20% 30% 40% 50% 60% 70% 80% 90% 100% 2y ago6m ago 3m ago EOY 2014 WebUI unit testing strategy Web GUIs Unit tests Automated tests Hey, let’s use AngularJS
  • 16.
     Use Karmaas test runner  Write your tests in Jasmine  Integrate Karma on continuous integration server and TEST, TEST, TEST, TEST… you’ll be happier later 
  • 17.

Editor's Notes

  • #4Ā no unit to testdon’t know which js file does what and what to expect from itStandardsMVCDOM manipulation only in directivesMaintainable and easy to test codeAngular philosophy encourages developers to create their own directives, turning the HTML into aDSLĀ suited to building their kind of app. The result significantly reduces the amount and complexity of JavaScript needed to build web applications.The Angular philosophy is that UI is best described in declarative form (HTML), and that behavior is best described in imperative form (JavaScript) and that the two should never meet.
  • #7Ā A test suite begins with a call to the global Jasmine functiondescribeĀ with two parameters: a string and a function. The string is a name or title for a spec suite – usually what is being tested. The function is a block of code that implements the suite.Specs are defined by calling the global Jasmine functionĀ it, which, likeĀ describeĀ takes a string and a function. The string is the title of the spec and the function is the spec, or test. A spec contains one or more expectations that test the state of the code. An expectation in Jasmine is an assertion that is either true or false. A spec with all true expectations is a passing spec. A spec with one or more false expectations is a failing spec.Each matcher implements a boolean comparison between the actual value and the expected value. It is responsible for reporting to Jasmine if the expectation is true or false. Jasmine will then pass or fail the spec.
  • #8Ā a) Original service b) Mocked service$exceptionHandler –Any uncaught exception in angular expressions is delegated to this service. The default implementation simply delegates toĀ $log.errorĀ which logs it into the browser console.This service is overridden byĀ mock $exceptionHandlerĀ which aids in testing.$log –Simple service for logging. Default implementation safely writes the message into the browser&apos;s console (if present).Mock implementation ofĀ $logĀ that gathers all logged messages in arrays (one array per logging level). These arrays are exposed asĀ logsĀ property of each of the level-specific log function, e.g. for levelĀ errorĀ the array is exposed asĀ $log.error.logs.$interval – Angular&apos;swrapper forĀ window.setInterval. TheĀ fnĀ function is executed everyĀ delayĀ milliseconds.In tests you can useĀ $interval.flush(millis)Ā to move forward byĀ millisĀ milliseconds and trigger any functions scheduled to run in that time.$timeout –Angular&apos;s wrapper forĀ window.setTimeout. TheĀ fnĀ function is wrapped into a try/catch block and delegates any exceptions toĀ $exceptionHandlerĀ serviceIn tests you can useĀ $timeout.flush()Ā to synchronously flush the queue of deferred functions.$httpBackend – Fake HTTP backend implementation suitable for unit testing applications that use theĀ $http service.During unit testing, we want our unit tests to run quickly and have no external dependencies so we don’t want to sendĀ XHRĀ orĀ JSONPrequests to a real server. All we really need is to verify whether a certain request has been sent or not, or alternatively just let the application make requests, respond with pre-trained responses and assert that the end result is what we expect it to be.
  • #11Ā If you don’t mock $httpBackend karma will throw exception about unexpected request
  • #15Ā Yeoman – your tests always in same (standard place)Grunt – it will deploy your application in production without all unnecessary testsBower – you don’t need dependencies on your integration server (your tests are not using them)