docs(testing): from Igor Minar feedback (Part 1) (#2450)

This commit is contained in:
Ward Bell 2016-09-23 02:03:20 -07:00 committed by GitHub
parent 37cebe9746
commit b478aaf7a2
11 changed files with 296 additions and 277 deletions

View File

@ -11,7 +11,8 @@ import { BannerComponent } from './banner.component';
// #docregion setup // #docregion setup
let comp: BannerComponent; let comp: BannerComponent;
let fixture: ComponentFixture<BannerComponent>; let fixture: ComponentFixture<BannerComponent>;
let el: DebugElement; let de: DebugElement;
let el: HTMLElement;
describe('BannerComponent', () => { describe('BannerComponent', () => {
beforeEach(() => { beforeEach(() => {
@ -23,25 +24,27 @@ describe('BannerComponent', () => {
comp = fixture.componentInstance; // BannerComponent test instance comp = fixture.componentInstance; // BannerComponent test instance
// get title DebugElement by element name // query for the title <h1> by CSS element selector
el = fixture.debugElement.query(By.css('h1')); de = fixture.debugElement.query(By.css('h1'));
el = de.nativeElement;
}); });
// #enddocregion setup // #enddocregion setup
// #docregion tests // #docregion tests
it('should display original title', () => { it('should display original title', () => {
fixture.detectChanges(); // trigger data binding fixture.detectChanges();
expect(el.nativeElement.textContent).toContain(comp.title); expect(el.textContent).toContain(comp.title);
}); });
it('should display a different test title', () => { it('should display a different test title', () => {
comp.title = 'Test Title'; comp.title = 'Test Title';
fixture.detectChanges(); // trigger data binding fixture.detectChanges();
expect(el.nativeElement.textContent).toContain('Test Title'); expect(el.textContent).toContain('Test Title');
}); });
// #enddocregion tests // #enddocregion tests
// #docregion test-w-o-detect-changes // #docregion test-w-o-detect-changes
it('no title in the DOM until manually call `detectChanges`', () => { it('no title in the DOM until manually call `detectChanges`', () => {
expect(el.nativeElement.textContent).toEqual(''); expect(el.textContent).toEqual('');
}); });
// #enddocregion test-w-o-detect-changes // #enddocregion test-w-o-detect-changes
@ -59,8 +62,7 @@ describe('BannerComponent with AutoChangeDetect', () => {
fixture = TestBed.configureTestingModule({ fixture = TestBed.configureTestingModule({
declarations: [ BannerComponent ], declarations: [ BannerComponent ],
providers: [ providers: [
{ provide: ComponentFixtureAutoDetect, { provide: ComponentFixtureAutoDetect, useValue: true }
useValue: true }
] ]
}) })
// #enddocregion auto-detect // #enddocregion auto-detect
@ -68,27 +70,28 @@ describe('BannerComponent with AutoChangeDetect', () => {
comp = fixture.componentInstance; // BannerComponent test instance comp = fixture.componentInstance; // BannerComponent test instance
// find title DebugElement by element name // query for the title <h1> by CSS element selector
el = fixture.debugElement.query(By.css('h1')); de = fixture.debugElement.query(By.css('h1'));
el = de.nativeElement;
}); });
// #docregion auto-detect-tests // #docregion auto-detect-tests
it('should display original title', () => { it('should display original title', () => {
// Hooray! No `fixture.detectChanges()` needed // Hooray! No `fixture.detectChanges()` needed
expect(el.nativeElement.textContent).toContain(comp.title); expect(el.textContent).toContain(comp.title);
}); });
it('should still see original title after comp.title change', () => { it('should still see original title after comp.title change', () => {
const oldTitle = comp.title; const oldTitle = comp.title;
comp.title = 'Test Title'; comp.title = 'Test Title';
// Displayed title is old because Angular didn't hear the change :( // Displayed title is old because Angular didn't hear the change :(
expect(el.nativeElement.textContent).toContain(oldTitle); expect(el.textContent).toContain(oldTitle);
}); });
it('should display updated title after detectChanges', () => { it('should display updated title after detectChanges', () => {
comp.title = 'Test Title'; comp.title = 'Test Title';
fixture.detectChanges(); // detect changes explicitly fixture.detectChanges(); // detect changes explicitly
expect(el.nativeElement.textContent).toContain(comp.title); expect(el.textContent).toContain(comp.title);
}); });
// #enddocregion auto-detect-tests // #enddocregion auto-detect-tests
}); });
@ -114,14 +117,14 @@ describe('BannerComponent (simpified)', () => {
// #docregion simple-example-it // #docregion simple-example-it
it('should display original title', () => { it('should display original title', () => {
// trigger data binding to update the view // trigger change detection to update the view
fixture.detectChanges(); fixture.detectChanges();
// find the title element in the DOM using a CSS selector // query for the title <h1> by CSS element selector
el = fixture.debugElement.query(By.css('h1')); de = fixture.debugElement.query(By.css('h1'));
// confirm the element's content // confirm the element's content
expect(el.nativeElement.textContent).toContain(comp.title); expect(de.nativeElement.textContent).toContain(comp.title);
}); });
// #enddocregion simple-example-it // #enddocregion simple-example-it
}); });

View File

@ -2,14 +2,15 @@
// #docregion // #docregion
import { TitleCasePipe } from './title-case.pipe'; import { TitleCasePipe } from './title-case.pipe';
// #docregion excerpt // #docregion excerpt, mini-excerpt
describe('TitleCasePipe', () => { describe('TitleCasePipe', () => {
// This pipe is a pure function so no need for BeforeEach // This pipe is a pure, stateless function so no need for BeforeEach
let pipe = new TitleCasePipe(); let pipe = new TitleCasePipe();
it('transforms "abc" to "Abc"', () => { it('transforms "abc" to "Abc"', () => {
expect(pipe.transform('abc')).toBe('Abc'); expect(pipe.transform('abc')).toBe('Abc');
}); });
// #enddocregion mini-excerpt
it('transforms "abc def" to "Abc Def"', () => { it('transforms "abc def" to "Abc Def"', () => {
expect(pipe.transform('abc def')).toBe('Abc Def'); expect(pipe.transform('abc def')).toBe('Abc Def');
@ -28,6 +29,6 @@ describe('TitleCasePipe', () => {
it('transforms " abc def" to " Abc Def" (preserves spaces) ', () => { it('transforms " abc def" to " Abc Def" (preserves spaces) ', () => {
expect(pipe.transform(' abc def')).toBe(' Abc Def'); expect(pipe.transform(' abc def')).toBe(' Abc Def');
}); });
// #docregion excerpt // #docregion excerpt, mini-excerpt
}); });
// #enddocregion excerpt // #enddocregion excerpt, mini-excerpt

View File

@ -12,7 +12,8 @@ describe('TwainComponent', () => {
let fixture: ComponentFixture<TwainComponent>; let fixture: ComponentFixture<TwainComponent>;
let spy: jasmine.Spy; let spy: jasmine.Spy;
let twainEl: DebugElement; // the element with the Twain quote let de: DebugElement;
let el: HTMLElement;
let twainService: TwainService; // the actually injected service let twainService: TwainService; // the actually injected service
const testQuote = 'Test Quote'; const testQuote = 'Test Quote';
@ -37,54 +38,53 @@ describe('TwainComponent', () => {
// #enddocregion spy // #enddocregion spy
// Get the Twain quote element by CSS selector (e.g., by class name) // Get the Twain quote element by CSS selector (e.g., by class name)
twainEl = fixture.debugElement.query(By.css('.twain')); de = fixture.debugElement.query(By.css('.twain'));
el = de.nativeElement;
}); });
// #enddocregion setup // #enddocregion setup
// #docregion tests // #docregion tests
function getQuote() { return twainEl.nativeElement.textContent; }
it('should not show quote before OnInit', () => { it('should not show quote before OnInit', () => {
expect(getQuote()).toBe('', 'nothing displayed'); expect(el.textContent).toBe('', 'nothing displayed');
expect(spy.calls.any()).toBe(false, 'getQuote not yet called'); expect(spy.calls.any()).toBe(false, 'getQuote not yet called');
}); });
it('should still not show quote after component initialized', () => { it('should still not show quote after component initialized', () => {
fixture.detectChanges(); // trigger data binding fixture.detectChanges();
// getQuote service is async => still has not returned with quote // getQuote service is async => still has not returned with quote
expect(getQuote()).toBe('...', 'no quote yet'); expect(el.textContent).toBe('...', 'no quote yet');
expect(spy.calls.any()).toBe(true, 'getQuote called'); expect(spy.calls.any()).toBe(true, 'getQuote called');
}); });
// #docregion async-test // #docregion async-test
it('should show quote after getQuote promise (async)', async(() => { it('should show quote after getQuote promise (async)', async(() => {
fixture.detectChanges(); // trigger data binding fixture.detectChanges();
fixture.whenStable().then(() => { // wait for async getQuote fixture.whenStable().then(() => { // wait for async getQuote
fixture.detectChanges(); // update view with quote fixture.detectChanges(); // update view with quote
expect(getQuote()).toBe(testQuote); expect(el.textContent).toBe(testQuote);
}); });
})); }));
// #enddocregion async-test // #enddocregion async-test
// #docregion fake-async-test // #docregion fake-async-test
it('should show quote after getQuote promise (fakeAsync)', fakeAsync(() => { it('should show quote after getQuote promise (fakeAsync)', fakeAsync(() => {
fixture.detectChanges(); // trigger data binding fixture.detectChanges();
tick(); // wait for async getQuote tick(); // wait for async getQuote
fixture.detectChanges(); // update view with quote fixture.detectChanges(); // update view with quote
expect(getQuote()).toBe(testQuote); expect(el.textContent).toBe(testQuote);
})); }));
// #enddocregion fake-async-test // #enddocregion fake-async-test
// #enddocregion tests // #enddocregion tests
// #docregion done-test // #docregion done-test
it('should show quote after getQuote promise (done)', done => { it('should show quote after getQuote promise (done)', done => {
fixture.detectChanges(); // trigger data binding fixture.detectChanges();
// get the spy promise and wait for it to resolve // get the spy promise and wait for it to resolve
spy.calls.mostRecent().returnValue.then(() => { spy.calls.mostRecent().returnValue.then(() => {
fixture.detectChanges(); // update view with quote fixture.detectChanges(); // update view with quote
expect(getQuote()).toBe(testQuote); expect(el.textContent).toBe(testQuote);
done(); done();
}); });
}); });

View File

@ -1,7 +1,7 @@
// #docplaster // #docplaster
import { ComponentFixture, TestBed } from '@angular/core/testing'; import { ComponentFixture, inject, TestBed } from '@angular/core/testing';
import { By } from '@angular/platform-browser'; import { By } from '@angular/platform-browser';
import { DebugElement } from '@angular/core'; import { DebugElement } from '@angular/core';
import { UserService } from './model'; import { UserService } from './model';
import { WelcomeComponent } from './welcome.component'; import { WelcomeComponent } from './welcome.component';
@ -10,8 +10,10 @@ describe('WelcomeComponent', () => {
let comp: WelcomeComponent; let comp: WelcomeComponent;
let fixture: ComponentFixture<WelcomeComponent>; let fixture: ComponentFixture<WelcomeComponent>;
let userService: UserService; // the actually injected service let componentUserService: UserService; // the actually injected service
let welcomeEl: DebugElement; // the element with the welcome message let userService: UserService; // the TestBed injected service
let de: DebugElement; // the DebugElement with the welcome message
let el: HTMLElement; // the DOM element with the welcome message
let userServiceStub: { let userServiceStub: {
isLoggedIn: boolean; isLoggedIn: boolean;
@ -32,7 +34,8 @@ describe('WelcomeComponent', () => {
TestBed.configureTestingModule({ TestBed.configureTestingModule({
declarations: [ WelcomeComponent ], declarations: [ WelcomeComponent ],
// #enddocregion setup // #enddocregion setup
// providers: [ UserService ] // a real service would be a problem! // providers: [ UserService ] // NO! Don't provide the real service!
// Provide a test-double instead
// #docregion setup // #docregion setup
providers: [ {provide: UserService, useValue: userServiceStub } ] providers: [ {provide: UserService, useValue: userServiceStub } ]
}); });
@ -42,51 +45,64 @@ describe('WelcomeComponent', () => {
comp = fixture.componentInstance; comp = fixture.componentInstance;
// #enddocregion setup // #enddocregion setup
// #docregion inject-from-testbed // #docregion injected-service
// UserService provided to the TestBed
userService = TestBed.get(UserService);
// #enddocregion inject-from-testbed
// #docregion setup
// #docregion injected-service
// UserService actually injected into the component // UserService actually injected into the component
userService = fixture.debugElement.injector.get(UserService); userService = fixture.debugElement.injector.get(UserService);
// #enddocregion injected-service // #enddocregion injected-service
componentUserService = userService;
// #docregion setup
// #docregion inject-from-testbed
// UserService from the root injector
userService = TestBed.get(UserService);
// #enddocregion inject-from-testbed
// get the "welcome" element by CSS selector (e.g., by class name) // get the "welcome" element by CSS selector (e.g., by class name)
welcomeEl = fixture.debugElement.query(By.css('.welcome')); de = fixture.debugElement.query(By.css('.welcome'));
el = de.nativeElement;
}); });
// #enddocregion setup // #enddocregion setup
// #docregion tests // #docregion tests
it('should welcome the user', () => { it('should welcome the user', () => {
fixture.detectChanges(); // trigger data binding fixture.detectChanges();
const content = el.textContent;
let content = welcomeEl.nativeElement.textContent;
expect(content).toContain('Welcome', '"Welcome ..."'); expect(content).toContain('Welcome', '"Welcome ..."');
expect(content).toContain('Test User', 'expected name'); expect(content).toContain('Test User', 'expected name');
}); });
it('should welcome "Bubba"', () => { it('should welcome "Bubba"', () => {
userService.user.name = 'Bubba'; // welcome message hasn't been shown yet userService.user.name = 'Bubba'; // welcome message hasn't been shown yet
fixture.detectChanges();
fixture.detectChanges(); // trigger data binding expect(el.textContent).toContain('Bubba');
let content = welcomeEl.nativeElement.textContent;
expect(content).toContain('Bubba');
}); });
it('should request login if not logged in', () => { it('should request login if not logged in', () => {
userService.isLoggedIn = false; // welcome message hasn't been shown yet userService.isLoggedIn = false; // welcome message hasn't been shown yet
fixture.detectChanges();
fixture.detectChanges(); // trigger data binding const content = el.textContent;
let content = welcomeEl.nativeElement.textContent;
expect(content).not.toContain('Welcome', 'not welcomed'); expect(content).not.toContain('Welcome', 'not welcomed');
expect(content).toMatch(/log in/i, '"log in"'); expect(content).toMatch(/log in/i, '"log in"');
}); });
// #enddocregion tests // #enddocregion tests
it('orig stub and injected UserService are not the same object', () => { // #docregion inject-it
expect(userServiceStub === userService).toBe(false); it('should inject the component\'s UserService instance',
inject([UserService], (service: UserService) => {
expect(service).toBe(componentUserService);
}));
// #enddocregion inject-it
it('TestBed and Component UserService should be the same', () => {
expect(userService === componentUserService).toBe(true);
}); });
// #docregion stub-not-injected
it('stub object and injected UserService should not be the same', () => {
expect(userServiceStub === userService).toBe(false);
// Changing the stub object has no effect on the injected service
userServiceStub.isLoggedIn = false;
expect(userService.isLoggedIn).toBe(true);
});
// #enddocregion stub-not-injected
}); });

View File

@ -60,7 +60,7 @@
"ts-to-js": { "ts-to-js": {
"title": "TypeScript to JavaScript", "title": "TypeScript to JavaScript",
"intro": "Convert Angular 2 TypeScript examples into ES5 JavaScript", "intro": "Convert Angular TypeScript examples into ES5 JavaScript",
"hide": true "hide": true
}, },

View File

@ -10,7 +10,7 @@
"architecture": { "architecture": {
"title": "Architecture Overview", "title": "Architecture Overview",
"navTitle": "Architecture", "navTitle": "Architecture",
"intro": "The basic building blocks of Angular 2 applications", "intro": "The basic building blocks of Angular applications",
"nextable": true, "nextable": true,
"basics": true "basics": true
}, },
@ -59,7 +59,7 @@
"style-guide": { "style-guide": {
"title": "Style Guide", "title": "Style Guide",
"intro": "Write Angular 2 with style.", "intro": "Write Angular with style.",
"basics": true "basics": true
}, },
@ -92,7 +92,7 @@
"glossary": { "glossary": {
"title": "Glossary", "title": "Glossary",
"intro": "Brief definitions of the most important words in the Angular 2 vocabulary", "intro": "Brief definitions of the most important words in the Angular vocabulary",
"basics": true "basics": true
}, },
@ -125,7 +125,7 @@
"router": { "router": {
"title": "Routing & Navigation", "title": "Routing & Navigation",
"intro": "Discover the basics of screen navigation with the Angular 2 Component Router." "intro": "Discover the basics of screen navigation with the Angular Component Router."
}, },
"security": { "security": {
@ -140,13 +140,13 @@
"testing": { "testing": {
"title": "Testing", "title": "Testing",
"intro": "Techniques and practices for testing an Angular 2 app", "intro": "Techniques and practices for testing an Angular app",
"hide": true "hide": true
}, },
"typescript-configuration": { "typescript-configuration": {
"title": "TypeScript Configuration", "title": "TypeScript Configuration",
"intro": "TypeScript configuration for Angular 2 developers", "intro": "TypeScript configuration for Angular developers",
"hide": true "hide": true
}, },
@ -158,7 +158,7 @@
"webpack": { "webpack": {
"title": "Webpack: an introduction", "title": "Webpack: an introduction",
"intro": "Create your Angular 2 applications with a Webpack based tooling", "intro": "Create your Angular applications with a Webpack based tooling",
"hide": true "hide": true
} }
} }

View File

@ -55,7 +55,7 @@
"ts-to-js": { "ts-to-js": {
"title": "TypeScript to JavaScript", "title": "TypeScript to JavaScript",
"intro": "Convert Angular 2 TypeScript examples into ES5 JavaScript" "intro": "Convert Angular TypeScript examples into ES5 JavaScript"
}, },
"visual-studio-2015": { "visual-studio-2015": {

View File

@ -10,7 +10,7 @@
"architecture": { "architecture": {
"title": "Architecture Overview", "title": "Architecture Overview",
"navTitle": "Architecture", "navTitle": "Architecture",
"intro": "The basic building blocks of Angular 2 applications", "intro": "The basic building blocks of Angular applications",
"nextable": true, "nextable": true,
"basics": true "basics": true
}, },
@ -59,7 +59,7 @@
"style-guide": { "style-guide": {
"title": "Style Guide", "title": "Style Guide",
"intro": "Write Angular 2 with style.", "intro": "Write Angular with style.",
"basics": true "basics": true
}, },
@ -86,7 +86,7 @@
"glossary": { "glossary": {
"title": "Glossary", "title": "Glossary",
"intro": "Brief definitions of the most important words in the Angular 2 vocabulary", "intro": "Brief definitions of the most important words in the Angular vocabulary",
"basics": true "basics": true
}, },
@ -118,7 +118,7 @@
"router": { "router": {
"title": "Routing & Navigation", "title": "Routing & Navigation",
"intro": "Discover the basics of screen navigation with the Angular 2 router." "intro": "Discover the basics of screen navigation with the Angular router."
}, },
"security": { "security": {
@ -133,13 +133,13 @@
"testing": { "testing": {
"title": "Testing", "title": "Testing",
"intro": "Techniques and practices for testing an Angular 2 app", "intro": "Techniques and practices for testing an Angular app",
"hide": true "hide": true
}, },
"typescript-configuration": { "typescript-configuration": {
"title": "TypeScript Configuration", "title": "TypeScript Configuration",
"intro": "TypeScript configuration for Angular 2 developers", "intro": "TypeScript configuration for Angular developers",
"hide": true "hide": true
}, },
@ -150,7 +150,7 @@
"webpack": { "webpack": {
"title": "Webpack: an introduction", "title": "Webpack: an introduction",
"intro": "Create your Angular 2 applications with a Webpack based tooling", "intro": "Create your Angular applications with a Webpack based tooling",
"hide": true "hide": true
} }
} }

View File

@ -65,7 +65,7 @@
"ts-to-js": { "ts-to-js": {
"title": "TypeScript to JavaScript", "title": "TypeScript to JavaScript",
"intro": "Convert Angular 2 TypeScript examples into ES5 JavaScript" "intro": "Convert Angular TypeScript examples into ES5 JavaScript"
}, },
"visual-studio-2015": { "visual-studio-2015": {

View File

@ -10,7 +10,7 @@
"architecture": { "architecture": {
"title": "Architecture Overview", "title": "Architecture Overview",
"navTitle": "Architecture", "navTitle": "Architecture",
"intro": "The basic building blocks of Angular 2 applications", "intro": "The basic building blocks of Angular applications",
"nextable": true, "nextable": true,
"basics": true "basics": true
}, },
@ -59,7 +59,7 @@
"style-guide": { "style-guide": {
"title": "Style Guide", "title": "Style Guide",
"intro": "Write Angular 2 with style.", "intro": "Write Angular with style.",
"basics": true "basics": true
}, },
@ -90,7 +90,7 @@
"glossary": { "glossary": {
"title": "Glossary", "title": "Glossary",
"intro": "Brief definitions of the most important words in the Angular 2 vocabulary", "intro": "Brief definitions of the most important words in the Angular vocabulary",
"basics": true "basics": true
}, },
@ -122,7 +122,7 @@
"router": { "router": {
"title": "Routing & Navigation", "title": "Routing & Navigation",
"intro": "Discover the basics of screen navigation with the Angular 2 Router." "intro": "Discover the basics of screen navigation with the Angular Router."
}, },
"security": { "security": {
@ -137,12 +137,12 @@
"testing": { "testing": {
"title": "Testing", "title": "Testing",
"intro": "Techniques and practices for testing an Angular 2 app" "intro": "Techniques and practices for testing an Angular app"
}, },
"typescript-configuration": { "typescript-configuration": {
"title": "TypeScript Configuration", "title": "TypeScript Configuration",
"intro": "TypeScript configuration for Angular 2 developers" "intro": "TypeScript configuration for Angular developers"
}, },
"upgrade": { "upgrade": {
@ -152,6 +152,6 @@
"webpack": { "webpack": {
"title": "Webpack: an introduction", "title": "Webpack: an introduction",
"intro": "Create your Angular 2 applications with a Webpack based tooling" "intro": "Create your Angular applications with a Webpack based tooling"
} }
} }

View File

@ -9,7 +9,7 @@ block includes
:marked :marked
This chapter offers tips and techniques for testing Angular applications. This chapter offers tips and techniques for testing Angular applications.
Along the way you will learn some general testing principles and techniques but the focus is on Along the way you will learn some general testing principles and techniques but the focus is on
Angular testing. testing applications written with Angular
#top #top
:marked :marked
@ -21,17 +21,20 @@ block includes
- [npm packages](#npm-packages) - [npm packages](#npm-packages)
1. [The first karma test](#1st-karma-test) 1. [The first karma test](#1st-karma-test)
<br><br> <br><br>
1. [The Angular Testing Platform (ATP) ](#atp-intro) 1. [Introduction to the Angular testing utilities](#atu-intro)
<br><br> <br><br>
1. [The sample application and its tests](#sample-app) 1. [The sample application and its tests](#sample-app)
<br><br> <br><br>
1. [A simple component test](#simple-component-test) 1. [A simple component test](#simple-component-test)
- [_configureTestingModule_](#configure-testing-module) - [_configureTestingModule_](#configure-testing-module)
- [_createComponent_](#create-component) - [_createComponent_](#create-component)
- [_ComponentFixture_, _DebugElement_, _query(By.css)_](#component-fixture)
- [_detectChanges_](#detect-changes) - [_detectChanges_](#detect-changes)
- [_autoDetectChanges_](#auto-detect-changes) - [_autoDetectChanges_](#auto-detect-changes)
1. [Test a component with a service dependency](#component-with-dependency) 1. [Test a component with a service dependency](#component-with-dependency)
<br><br> - [test doubles](#service-test-doubles)
- [get the injected service](#get-injected-service)
- [_TestBed.get_](#testbed-get)
1. [Test a component with an async service](#component-with-async-service) 1. [Test a component with an async service](#component-with-async-service)
- [spies](#service-spy) - [spies](#service-spy)
- [_async_](#async) - [_async_](#async)
@ -64,12 +67,12 @@ block includes
<br><br> <br><br>
1. [Test an attribute directive](#attribute-directive) 1. [Test an attribute directive](#attribute-directive)
<br><br> <br><br>
1. [Isolated tests](#testing-without-atp "Testing without the Angular Testing Platform") 1. [Isolated unit tests](#isolated-unit-tests "Unit testing without the Angular testing utilities")
- [Services](#isolated-service-tests) - [Services](#isolated-service-tests)
- [Pipes](#isolated-pipe-tests) - [Pipes](#isolated-pipe-tests)
- [Components](#isolated-component-tests) - [Components](#isolated-component-tests)
1. [_Angular Testing Platform_ API](#atp-api) 1. [Angular testing utility APIs](#atu-apis)
- [Stand-alone functions](#atp-api): `async`, `fakeAsync`, etc. - [Stand-alone functions](#atu-apis): `async`, `fakeAsync`, etc.
- [_TestBed_](#testbed-class-summary) - [_TestBed_](#testbed-class-summary)
- [_ComponentFixture_](#component-fixture-class-summary) - [_ComponentFixture_](#component-fixture-class-summary)
- [_DebugElement_](#debug-element-details) - [_DebugElement_](#debug-element-details)
@ -102,9 +105,6 @@ a(href="#top").to-top Back to top
When a part of the application seems hard to test, the root cause is often a design flaw, When a part of the application seems hard to test, the root cause is often a design flaw,
something to cure now rather than later when it becomes expensive to fix. something to cure now rather than later when it becomes expensive to fix.
This chapter assumes that you know something about testing. Don't worry if you don't.
There are plenty of books and online resources to get up to speed.
<!-- TODO <!-- TODO
:marked :marked
## Learn more ## Learn more
@ -132,20 +132,20 @@ table(width="100%")
provides everything needed to write basic tests. provides everything needed to write basic tests.
It ships with an HTML test runner that executes tests in the browser. It ships with an HTML test runner that executes tests in the browser.
tr(style=top) tr(style=top)
td(style="vertical-align: top") Angular Testing Platform td(style="vertical-align: top") Angular Testing Utilities
td td
:marked :marked
The Angular Testing Platform creates a test environment and harness The Angular testing utilities create a test environment
for the application code under test. for the Angular application code under test.
Use it to condition and control parts of the application as they Use them to condition and control parts of the application as they
interact _within_ the Angular environment. interact _within_ the Angular environment.
tr(style=top) tr(style=top)
td(style="vertical-align: top") Karma td(style="vertical-align: top") Karma
td td
:marked :marked
The [karma test runner](https://karma-runner.github.io/1.0/index.html) The [karma test runner](https://karma-runner.github.io/1.0/index.html)
is ideal for writing and running tests while developing the application. is ideal for writing and running unit tests while developing the application.
It can be an integral part of the application build process. It can be an integral part of the project's development and continuous integration processes.
This chapter describes how to setup and run tests with karma. This chapter describes how to setup and run tests with karma.
tr(style=top) tr(style=top)
td(style="vertical-align: top") Protractor td(style="vertical-align: top") Protractor
@ -274,46 +274,28 @@ table(width="100%")
+makeExample('testing/ts/app/1st.spec.ts', '', 'app/1st.spec.ts')(format='.') +makeExample('testing/ts/app/1st.spec.ts', '', 'app/1st.spec.ts')(format='.')
:marked :marked
## Run karma ## Run karma
Compile and run it in karma from the command line. Compile and run it in karma from the command line with this command:
.l-sub-section
:marked
The QuickStart repo adds the following command to the `scripts` section in `package.json`.
code-example(format="." language="bash").
"test": "tsc && concurrently \"tsc -w\" \"karma start karma.conf.js\"",
:marked
Add that to your `package.json` if it's not there already.
:marked
Open a terminal or command window and enter
code-example(format="." language="bash"). code-example(format="." language="bash").
npm test npm test
:marked :marked
The command compiles the application and test code a first time. The command compiles the application and test code and starts karma.
If the compile fails, the command aborts. Both processes watch pertinent files, write messages to the console, and re-run when they detect changes.
.l-sub-section
If it succeeds, the command re-compiles (this time in watch mode) in one process :marked
and starts karma in another. The QuickStart development path defined the `test` command in the `scripts` section of npm's `package.json`.
Both processes watch pertinent files and re-run when they detect changes. The Angular CLI has different commands to do the same thing. Adjust accordingly.
:marked
After a few moments, karma opens a browser ... After a few moments, karma opens a browser and starts writing to the console.
figure.image-display figure.image-display
img(src='/resources/images/devguide/testing/karma-browser.png' style="width:400px;" alt="Karma browser") img(src='/resources/images/devguide/testing/karma-browser.png' style="width:400px;" alt="Karma browser")
:marked :marked
... and starts writing to the console.
Hide (don't close!) the browser and focus on the console output which should look something like this. Hide (don't close!) the browser and focus on the console output which should look something like this.
code-example(format="." language="bash"). code-example(format="." language="bash").
> npm test > npm test
> tsc && concurrently "tsc -w" "karma start karma.conf.js" ...
[0] 1:37:03 PM - Compilation complete. Watching for file changes. [0] 1:37:03 PM - Compilation complete. Watching for file changes.
[1] 24 07 2016 13:37:09.310:WARN [karma]: No captured browser, open http://localhost:9876/ ...
[1] 24 07 2016 13:37:09.361:INFO [karma]: Karma v0.13.22 server started at http://localhost:9876/
[1] 24 07 2016 13:37:09.370:INFO [launcher]: Starting browser Chrome
[1] 24 07 2016 13:37:10.974:INFO [Chrome 51.0.2704]: Connected on socket /#Cf6A5PkvMzjbbtn1AAAA with id 24600087
[1] Chrome 51.0.2704: Executed 0 of 0 SUCCESS [1] Chrome 51.0.2704: Executed 0 of 0 SUCCESS
Chrome 51.0.2704: Executed 1 of 1 SUCCESS Chrome 51.0.2704: Executed 1 of 1 SUCCESS
SUCCESS (0.005 secs / 0.005 secs) SUCCESS (0.005 secs / 0.005 secs)
@ -333,8 +315,7 @@ code-example(format="." language="bash").
:marked :marked
The _karma_ watcher detects the change to the compilation output and re-runs the test. The _karma_ watcher detects the change to the compilation output and re-runs the test.
code-example(format="." language="bash"). code-example(format="." language="bash").
[1] Chrome 51.0.2704: Executed 0 of 1 SUCCESS [1] Chrome 51.0.2704 1st tests true is true FAILED
Chrome 51.0.2704 1st tests true is true FAILED
[1] Expected false to equal true. [1] Expected false to equal true.
[1] Chrome 51.0.2704: Executed 1 of 1 (1 FAILED) (0.005 secs / 0.005 secs) [1] Chrome 51.0.2704: Executed 1 of 1 (1 FAILED) (0.005 secs / 0.005 secs)
@ -348,20 +329,15 @@ code-example(format="." language="bash").
:marked :marked
The console log can be quite long. Keep your eye on the last line. The console log can be quite long. Keep your eye on the last line.
It says `SUCCESS` when all is well. It says `SUCCESS` when all is well.
If it says `FAILED`, scroll up to look for the error or, if that's too painful,
pipe the console output to a file and inspect with your favorite editor.
code-example(format="." language="json").
npm test > spec-output.txt
:marked :marked
## Test debugging ## Test debugging
Debug specs in the browser in the same way you debug an application. Debug specs in the browser in the same way you debug an application.
- Reveal the karma browser window (hidden earlier). - Reveal the karma browser window (hidden earlier).
- Click the "DEBUG" button; it opens a new browser tab and re-runs the tests
- Open the browser's “Developer Tools” (F12 or Ctrl-Shift-I). - Open the browser's “Developer Tools” (F12 or Ctrl-Shift-I).
- Pick the “sources” section - Pick the "sources" section
- Open the `1st.spec.ts` test file (Ctrl-P, then start typing the name of the file). - Open the `1st.spec.ts` test file (Ctrl-P, then start typing the name of the file).
- Set a breakpoint in the test - Set a breakpoint in the test
- Refresh the browser … and it stops at the breakpoint. - Refresh the browser … and it stops at the breakpoint.
@ -372,71 +348,66 @@ figure.image-display
a(href="#top").to-top Back to top a(href="#top").to-top Back to top
.l-hr .l-hr
#atp-intro #atu-intro
:marked :marked
# The Angular Testing Platform (ATP) # Introduction to the Angular Testing Utilities
Many tests explore how applications classes interact with Angular and the DOM while under Angular's control. Many tests explore how applications classes interact with Angular and the DOM while under Angular's control.
Such tests are easy to write with the help of the _Angular Testing Platform_ (ATP) Such tests are easy to write with the help of the Angular testing utilities
which consists of the `TestBed` class and some helper functions. which include the `TestBed` class and some helper functions.
Tests written with the _Angular Testing Platform_ are the main focus of this chapter. Tests written with these utilities are the main focus of this chapter.
But they are not the only tests you should write. But they are not the only tests you should write.
### Isolated unit tests ### Isolated unit tests
You can and should write [isolated unit tests](#testing-without-atp "Testing without the Angular Testing Platform") [Isolated unit tests](#isolated-unit-tests "Unit testing without the Angular testing utilities")
for components, directives, pipes, and services. examine an instance of a class all by itself without any dependence on Angular or any injected values.
Isolated unit tests examine an instance of a class all by itself without
any dependence on Angular or any injected values.
The tester creates a test instance of the class with new, supplying test doubles for the constructor parameters as needed, and The tester creates a test instance of the class with new, supplying test doubles for the constructor parameters as needed, and
then probes the test instance API surface. then probes the test instance API surface.
Isolated tests don't reveal how the class interacts with Angular. You can and should write isolated unit tests for pipes and services.
+makeExample('testing/ts/app/shared/title-case.pipe.spec.ts', 'mini-excerpt', 'app/shared/title-case.pipe.spec.ts (excerpt)')
:marked
Components can be tested in isolation as well.
However, isolated unit tests don't reveal how these classes interact with Angular.
In particular, they can't reveal how a component class interacts with its own template or with other components. In particular, they can't reveal how a component class interacts with its own template or with other components.
Such tests require the Angular testing utilities.
Those tests require the Angular Testing Platform. ### Testing with the Angular Testing Utilities
### Testing with the _ Angular Testing Platform_ The Angular testing utilities include the `TestBed` class and several helper functions from `@angular/core/testing`.
The _Angular Testing Platform_ consists of the `TestBed` class and some helper functions from `@angular/core/testing`.
The `TestBed` creates an Angular testing module &mdash; an `@NgModule` class &mdash; The `TestBed` creates an Angular testing module &mdash; an `@NgModule` class &mdash;
that you configure to produce the module environment for the class you want to test. that you configure to produce the module environment for the class you want to test.
You tell the `TestBed` to create an instance of the test component and probe that instance with tests. You tell the `TestBed` to create an instance of the _component-under-test_ and probe that instance with tests.
That's the `TestBed` in a nutshell. Before each spec, the `TestBed` resets itself to a base state.
The base state includes a default testing module configuration consisting of the
In practice, you work with the static methods of the `TestBed` class. declarables (components, directives, and pipes) and providers (some of them mocked)
These static methods create and update a fresh hidden `TestBed` instance before each Jasmine `it`. that almost everyone needs.
.l-sub-section .l-sub-section
:marked :marked
You can access that hidden instance anytime by calling `getTestBed()`; The testing shims mentioned [earlier](#setup) initialize the testing module configuration
to something like the `BrowserModule` from `@angular/platform-browser`.
:marked :marked
Thanks to initialization in the [testing shims](#setup), This default configuration is merely a _foundation_ for testing an app.
the default `TestBed` instance is pre-configured with a baseline of default providers and declarables (components, directives, and pipes)
that almost everyone needs.
The shims in this chapter are designed for testing a browser application so the default configuration includes the `CommonModule` declarables from `@angular/common`
and the `BrowserModule` providers (some of them mocked) from `@angular/platform-browser`.
This default testing module configuration is a _foundation_ for testing _any_ browser app.
You call `TestBed.configureTestingModule` with an object that defines additional imports, declarations, providers and schemas You call `TestBed.configureTestingModule` with an object that defines additional imports, declarations, providers and schemas
to reshape the testing module to fit your application tests. to fit your application tests.
Optional `override...` methods can fine-tune aspects of the configuration. Optional `override...` methods can fine-tune aspects of the configuration.
After configuring the `TestBed`, tell it to create an instance of the test component and the test fixture After configuring the `TestBed`, tell it to create an instance of the _component-under-test_ and the test fixture
that you'll need to inspect and control the component's immediate environment. that you'll need to inspect and control the component's immediate environment.
+makeExample('testing/ts/app/banner.component.spec.ts', 'simple-example-before-each', 'app/banner.component.spec.ts (simplified)')(format='.') +makeExample('testing/ts/app/banner.component.spec.ts', 'simple-example-before-each', 'app/banner.component.spec.ts (simplified)')(format='.')
:marked :marked
Angular tests can interact with the HTML in the test DOM, Angular tests can interact with the HTML in the test DOM,
simulate user activity, tell Angular to perform specific task (such as change detection), simulate user activity, tell Angular to perform specific task (such as change detection),
and see the effects of these actions both in the test component and in the test DOM. and see the effects of these actions both in the _component-under-test_ and in the test DOM.
+makeExample('testing/ts/app/banner.component.spec.ts', 'simple-example-it', 'app/banner.component.spec.ts (simplified)')(format='.') +makeExample('testing/ts/app/banner.component.spec.ts', 'simple-example-it', 'app/banner.component.spec.ts (simplified)')(format='.')
:marked :marked
A comprehensive review of the _Angular Testing Platform_ APIs appears [later in the chapter](#atp-api). A comprehensive review of the Angular testing utilities appears [later in the chapter](#atu-apis).
Let's dive right into Angular testing, starting with with the components of a sample application. Let's dive right into Angular testing, starting with with the components of a sample application.
a(href="#top").to-top Back to top a(href="#top").to-top Back to top
@ -450,7 +421,6 @@ a(href="#top").to-top Back to top
This chapter tests a cut-down version of the _Tour of Heroes_ [tutorial app](../tutorial). This chapter tests a cut-down version of the _Tour of Heroes_ [tutorial app](../tutorial).
The following live example shows how it works and provides the complete source code. The following live example shows how it works and provides the complete source code.
Give it some time to load and warm up.
<live-example embedded img="devguide/testing/app-plunker.png"></live-example> <live-example embedded img="devguide/testing/app-plunker.png"></live-example>
<br><br> <br><br>
:marked :marked
@ -459,7 +429,7 @@ a(href="#top").to-top Back to top
It includes the tests discussed in this chapter and additional tests for you to explore. It includes the tests discussed in this chapter and additional tests for you to explore.
This live example contains both application and test code. This live example contains both application and test code.
It is large and can take up to a minute to start. Please be patient. Give it some time to load and warm up.
<live-example plnkr="app-specs" embedded img="devguide/testing/app-specs-plunker.png"></live-example> <live-example plnkr="app-specs" embedded img="devguide/testing/app-specs-plunker.png"></live-example>
a(href="#top").to-top Back to top a(href="#top").to-top Back to top
@ -472,8 +442,9 @@ a(href="#top").to-top Back to top
The top of the screen displays application title, presented by the `BannerComponent` in `app/banner.component.ts`. The top of the screen displays application title, presented by the `BannerComponent` in `app/banner.component.ts`.
+makeExample('testing/ts/app/banner.component.ts', '', 'app/banner.component.ts')(format='.') +makeExample('testing/ts/app/banner.component.ts', '', 'app/banner.component.ts')(format='.')
:marked :marked
`BannerComponent` has an inline template and an interpolation binding, about as simple as it gets. `BannerComponent` has an inline template and an interpolation binding.
Probably too simple to be worth testing in real life but perfect for a first encounter with the `TestBed`. The component is probably too simple to be worth testing in real life but
it's perfect for a first encounter with the `TestBed`.
The corresponding `app/banner-component.spec.ts` sits in the same folder as the component, The corresponding `app/banner-component.spec.ts` sits in the same folder as the component,
for reasons explained [here](#q-spec-file-location); for reasons explained [here](#q-spec-file-location);
@ -493,37 +464,58 @@ a(href="#top").to-top Back to top
already has what `BannerComponent` needs already has what `BannerComponent` needs
and (b) `BannerComponent` doesn't interact with any other components. and (b) `BannerComponent` doesn't interact with any other components.
The configuration could have imported `AppModule` (which declares `BannerComponent`).
But that would lead to tons more configuration in order to support the other components within `AppModule`
that have nothing to do with `BannerComponent`.
#create-component #create-component
:marked :marked
`TestBed.createComponent` creates an instance of `BannerComponent` to test. ### _createComponent_
The method returns a `ComponentFixture`, a handle on the test environment surrounding the created component. `TestBed.createComponent` creates an instance of `BannerComponent` to test and returns a [fixture](#component-fixture).
The fixture provides access to the component instance itself and
to the `DebugElement` which is a handle on the component's DOM element.
Query the `DebugElement` by CSS selector for the `<h1>` sub-element that holds the actual title.
### _createComponent_ closes configuration
`TestBed.createComponent` closes the current `TestBed` instance to further configuration. `TestBed.createComponent` closes the current `TestBed` instance to further configuration.
You cannot call any more `TestBed` configuration methods, not `configureTestModule` You cannot call any more `TestBed` configuration methods, not `configureTestModule`
nor any of the `override...` methods. The `TestBed` throws an error if you try. nor any of the `override...` methods. The `TestBed` throws an error if you try.
.alert.is-important .alert.is-important
:marked :marked
Do not configure the `TestBed` after calling `createComponent`. Do not configure the `TestBed` after calling `createComponent`.
#component-fixture
:marked :marked
### _ComponentFixture_, _DebugElement_, and _query(By.css)_
The `createComponent` method returns a **`ComponentFixture`**, a handle on the test environment surrounding the created component.
The fixture provides access to the component instance itself and
to the **`DebugElement`** which is a handle on the component's DOM element.
The `title` property value was interpolated into the DOM within `<h1>` tags.
Use the fixture's `DebugElement` to `query` for the `<h1>` element by CSS selector.
The **`query`** method takes a predicate function and searches the fixture's entire DOM tree for the
_first_ element that satisfies the predicate.
The result is a _different_ `DebugElement`, one associated with the matching DOM element.
.l-sub-section
:marked
The `queryAll` method returns an array of _all_ `DebugElements` that satisfy the predicate.
A _predicate_ is a function that returns a boolean.
A query predicate receives a `DebugElement` and returns `true` if the element meets the selection criteria.
:marked
The **`By`** class is an Angular testing utility that produces useful predicates.
Its `By.css` static method produces a
<a href="https://developer.mozilla.org/en-US/docs/Web/Guide/CSS/Getting_started/Selectors" target="_blank">standard CSS selector</a>
predicate that filters the same way as a jQuery selector.
Finally, the setup assigns the DOM element from the `DebugElement` **`nativeElement`** property to `el`.
The tests will assert that `el` contains the expected title text.
### The tests ### The tests
Jasmine runs this `beforeEach` before each test of which there are two Jasmine runs the `beforeEach` function before each of these tests
+makeExample('testing/ts/app/banner.component.spec.ts', 'tests', 'app/banner.component.spec.ts (tests)')(format='.') +makeExample('testing/ts/app/banner.component.spec.ts', 'tests', 'app/banner.component.spec.ts (tests)')(format='.')
:marked :marked
These tests ask the `DebugElement` for the native HTML element to satisfy their expectations. These tests ask the `DebugElement` for the native HTML element to satisfy their expectations.
#detect-changes #detect-changes
:marked :marked
### _detectChanges_: Angular change detection under test ### _detectChanges_: Angular change detection within a test
Each test tells Angular when to perform change detection by calling `fixture.detectChanges()`. Each test tells Angular when to perform change detection by calling `fixture.detectChanges()`.
The first test does so immediately, triggering data binding and propagation of the `title` property The first test does so immediately, triggering data binding and propagation of the `title` property
@ -542,34 +534,33 @@ a(href="#top").to-top Back to top
+makeExample('testing/ts/app/banner.component.spec.ts', 'test-w-o-detect-changes', 'app/banner.component.spec.ts (no detectChanges)')(format='.') +makeExample('testing/ts/app/banner.component.spec.ts', 'test-w-o-detect-changes', 'app/banner.component.spec.ts (no detectChanges)')(format='.')
:marked :marked
This behavior (or lack of it) is intentional. This behavior (or lack of it) is intentional.
It gives the tester an opportunity to investigate the state of It gives the tester an opportunity to inspect or change the state of
the component _before Angular initiates data binding or calls lifecycle hooks_. the component _before Angular initiates data binding or calls lifecycle hooks_.
#auto-detect-changes #auto-detect-changes
:marked :marked
### Automatic change detection ### Automatic change detection
Some testers prefer that the Angular test environment run change detection automatically. Some testers prefer that the Angular test environment run change detection automatically.
That's possible by configuring the `TestBed` with the _AutoDetect_ provider: That's possible by configuring the `TestBed` with the _AutoDetect_ provider:
+makeExample('testing/ts/app/banner.component.spec.ts', 'auto-detect', 'app/banner.component.spec.ts (AutoDetect)')(format='.') +makeExample('testing/ts/app/banner.component.spec.ts', 'auto-detect', 'app/banner.component.spec.ts (AutoDetect)')(format='.')
:marked :marked
Here are three tests that illustrate how _auto-detect_ works. Here are three tests that illustrate how _AutoDetect_ works.
+makeExample('testing/ts/app/banner.component.spec.ts', 'auto-detect-tests', 'app/banner.component.spec.ts (AutoDetect Tests)')(format='.') +makeExample('testing/ts/app/banner.component.spec.ts', 'auto-detect-tests', 'app/banner.component.spec.ts (AutoDetect Tests)')(format='.')
:marked :marked
The first test shows the benefit of automatic change detection. The first test shows the benefit of automatic change detection.
The second and third test remind us that Angular does _not_ know about changes to component property The second and third test reveal an important limitation.
values unless Angular itself (or some asynchronous process) makes the change. The Angular testing environment does _not_ know that the test changed the component's `title`.
This is as true in production as it is in test. _AutoDetect_ responds to _asynchronous activities_ such as promise resolution, timers, and DOM events.
But a direct, synchronous update of the component property is invisible to _AutoDetect_.
In production, external forces rarely change component properties like this, The test must call `fixture.detectChanges()` manually to trigger another cycle of change detection.
whereas these kinds of probing changes are typical in unit tests.
The tester will have to call `fixture.detectChanges()` quite often
despite having opted into auto detect.
.alert.is-helpful .alert.is-helpful
:marked :marked
Rather than wonder when the test fixture will or won't perform change detection, Rather than wonder when the test fixture will or won't perform change detection,
the samples in this chapter _always call_ `detectChanges()` _explicitly_. the samples in this chapter _always call_ `detectChanges()` _explicitly_.
There is no harm in calling `detectChanges()` more often than is strictly necessary.
a(href="#top").to-top Back to top a(href="#top").to-top Back to top
@ -583,19 +574,19 @@ a(href="#top").to-top Back to top
It knows who the user is based on a property of the injected `UserService`: It knows who the user is based on a property of the injected `UserService`:
+makeExample('testing/ts/app/welcome.component.ts', '', 'app/welcome.component.ts')(format='.') +makeExample('testing/ts/app/welcome.component.ts', '', 'app/welcome.component.ts')(format='.')
:marked :marked
The `WelcomeComponent` has decision logic that interacts with the service; The `WelcomeComponent` has decision logic that interacts with the service, logic that makes this component worth testing.
such logic makes this component worth testing.
Here's the testing module configuration for the spec file, `app/welcome.component.spec.ts`: Here's the testing module configuration for the spec file, `app/welcome.component.spec.ts`:
+makeExample('testing/ts/app/welcome.component.spec.ts', 'config-test-module', 'app/welcome.component.spec.ts')(format='.') +makeExample('testing/ts/app/welcome.component.spec.ts', 'config-test-module', 'app/welcome.component.spec.ts')(format='.')
:marked :marked
This time, in addition to declaring the component under test, This time, in addition to declaring the _component-under-test_,
the configurations sets the `providers` list with the dependent `UserService`. the configuration adds a `UserService` provider to the `providers` list.
But not the real `UserService`.
This example configures the testing module with a stub `UserService`.
#get-injected-service
:marked
## Provide service test doubles ## Provide service test doubles
A component under test doesn't have to be injected with real services. A _component-under-test_ doesn't have to be injected with real services.
In fact, it is usually better if they are test doubles (stubs, fakes, spies, or mocks). In fact, it is usually better if they are test doubles (stubs, fakes, spies, or mocks).
The purpose of the spec is to test the component, not the service, The purpose of the spec is to test the component, not the service,
and real services can be trouble. and real services can be trouble.
@ -604,57 +595,68 @@ a(href="#top").to-top Back to top
The real service might try to ask the user for login credentials and The real service might try to ask the user for login credentials and
try to reach an authentication server. try to reach an authentication server.
These behaviors could be hard to intercept. These behaviors could be hard to intercept.
It is far easier to create and register a test double in place of the real `UserService`. It is far easier and safer to create and register a test double in place of the real `UserService`.
This particular test suite supplies a minimal `UserService` stub that satisfies the needs of the `WelcomeComponent` This particular test suite supplies a minimal `UserService` stub that satisfies the needs of the `WelcomeComponent`
and its tests: and its tests:
+makeExample('testing/ts/app/welcome.component.spec.ts', 'user-service-stub')(format='.') +makeExample('testing/ts/app/welcome.component.spec.ts', 'user-service-stub')(format='.')
#injected-service-reference #get-injected-service
:marked :marked
## Referencing injected services ## Get injected services
The tests need access to the injected (stubbed) `UserService`. The tests need access to the (stub) `UserService` injected into the `WelcomeComponent`.
You cannot reference the `userServiceStub` object provided to the testing module. Angular has a hierarchical injection system.
**It does not work!** There can be injectors at multiple levels, from the root injector created by the `TestBed`
Surprisingly, the instance actually injected into the component is _not the same_ object down through the component tree.
as the provided `userServiceStub`.
.alert.is-important The safest way to get the injected service, the way that **_always works_**,
:marked is to **get it from the injector of the _component-under-test_**.
Always use an injector to get a reference to an injected service. The component injector is a property of the fixture's `DebugElement`.
+makeExample('testing/ts/app/welcome.component.spec.ts', 'injected-service', 'WelcomeComponent\'s injector')(format='.')
#testbed-get
:marked :marked
Where do you get the injector? ### _TestBed.get_
Angular has an hierarchical injection system.
In a test there can be injectors at multiple levels.
The current `TestBed` injector creates a top-level injector.
The `WelcomeComponent` injector is a child of that injector created specifically for the component.
You can get a `UserService` from the current `TestBed` injector by calling `TestBed.get`. You _may_ also be able to get the service from the root injector via `TestBed.get`.
This is easier to remember and less verbose.
But it only works when Angular injects the component with the service instance in the test's root injector.
Fortunately, in this test suite, the _only_ provider of `UserService` is the root testing module,
so it is safe to call `TestBed.get` as follows:
+makeExample('testing/ts/app/welcome.component.spec.ts', 'inject-from-testbed', 'TestBed injector')(format='.') +makeExample('testing/ts/app/welcome.component.spec.ts', 'inject-from-testbed', 'TestBed injector')(format='.')
.l-sub-section .l-sub-section
:marked :marked
The [inject](#inject) function is another way to inject one or more services into a test. The [`inject`](#inject) utility function is another way to get one or more services from the test root injector.
See the section "[_Override Component Providers_](#component-override)" for a use case
in which `inject` and `TestBed.get` do not work and you must get the service from the component's injector.
:marked :marked
That happens to work for testing the `WelcomeComponent` because the `UserService` instance from the `TestBed` ### Always get the service from an injector
is the same as the `UserService` instance injected into the component. Surprisingly, you dare not reference the `userServiceStub` object
that was provided to the testing module in the body of your test.
**It does not work!**
The `userService` instance injected into the component is a completely _different_ object,
a clone of the provided `userServiceStub`.
+makeExample('testing/ts/app/welcome.component.spec.ts', 'stub-not-injected')(format='.')
That won't always be the case.
Be absolutely sure to reference the service instance that the component is _actually receiving_,
Call `get` on the component's injector which is `fixture.debugElement.injector`:
+makeExample('testing/ts/app/welcome.component.spec.ts', 'injected-service', 'Component\'s injector')(format='.')
.alert.is-important
:marked
Use the component's own injector to get the component's injected service.
#welcome-spec-setup #welcome-spec-setup
:marked :marked
Here's the complete, preferred `beforeEach`: ### Final setup and tests
Here's the complete `beforeEach` using `TestBed.get`:
+makeExample('testing/ts/app/welcome.component.spec.ts', 'setup', 'app/welcome.component.spec.ts')(format='.') +makeExample('testing/ts/app/welcome.component.spec.ts', 'setup', 'app/welcome.component.spec.ts')(format='.')
:marked :marked
And here are some tests: And here are some tests:
+makeExample('testing/ts/app/welcome.component.spec.ts', 'tests', 'app/welcome.component.spec.ts')(format='.') +makeExample('testing/ts/app/welcome.component.spec.ts', 'tests', 'app/welcome.component.spec.ts')(format='.')
:marked :marked
The first is a sanity test; it confirms that the stubbed `UserService` is called and working. The first is a sanity test; it confirms that the stubbed `UserService` is called and working.
.l-sub-section
:marked
The second parameter to the Jasmine `it` (e.g., `'expected name'`) is an optional addendum.
If the expectation fails, Jasmine displays this addendum after the expectation failure message.
It can help clarify what went wrong and which expectation failed in a spec with multiple expectations.
:marked
The remaining tests confirm the logic of the component when the service returns different values. The remaining tests confirm the logic of the component when the service returns different values.
The second test validates the effect of changing the user name. The second test validates the effect of changing the user name.
The third test checks that the component displays the proper message when there is no logged-in user. The third test checks that the component displays the proper message when there is no logged-in user.
@ -721,7 +723,7 @@ a(href="#top").to-top Back to top
Notice the `async` in the third test. Notice the `async` in the third test.
+makeExample('testing/ts/app/shared/twain.component.spec.ts', 'async-test', 'app/shared/twain.component.spec.ts (async test)')(format='.') +makeExample('testing/ts/app/shared/twain.component.spec.ts', 'async-test', 'app/shared/twain.component.spec.ts (async test)')(format='.')
:marked :marked
The `async` function is an independent feature of the _Angular Testing Platform_. The `async` function is one of the Angular testing utilities.
It simplifyies coding of asynchronous tests by arranging for the tester's code to run in a special _async test zone_. It simplifyies coding of asynchronous tests by arranging for the tester's code to run in a special _async test zone_.
@ -768,7 +770,7 @@ a(href="#top").to-top Back to top
+makeExample('testing/ts/app/shared/twain.component.spec.ts', 'fake-async-test', 'app/shared/twain.component.spec.ts (fakeAsync test)')(format='.') +makeExample('testing/ts/app/shared/twain.component.spec.ts', 'fake-async-test', 'app/shared/twain.component.spec.ts (fakeAsync test)')(format='.')
:marked :marked
Notice that `fakeAsync` replaces `async` as the `it` argument. Notice that `fakeAsync` replaces `async` as the `it` argument.
The `fakeAsync` function is another, independent feature of the _Angular Testing Platform_. The `fakeAsync` function is another of the Angular testing utilities.
Like [async](#), it _takes_ a parameterless function and _returns_ a parameterless function Like [async](#), it _takes_ a parameterless function and _returns_ a parameterless function
which becomes the argument to the Jasmine `it` call. which becomes the argument to the Jasmine `it` call.
@ -787,7 +789,7 @@ a(href="#top").to-top Back to top
## The _tick_ function ## The _tick_ function
Compare the third and fourth tests. Notice that `fixture.whenStable` is gone, replaced by `tick()`. Compare the third and fourth tests. Notice that `fixture.whenStable` is gone, replaced by `tick()`.
The `tick` function is a part of the _Angular Testing Platform_ and a companion to `fakeAsync`. The `tick` function is one of the Angular testing utilities and a companion to `fakeAsync`.
It can only be called within a `fakeAsync` body. It can only be called within a `fakeAsync` body.
Calling `tick()` simulates the passage of time until all pending asynchronous activities complete, Calling `tick()` simulates the passage of time until all pending asynchronous activities complete,
@ -882,7 +884,7 @@ a(href="#top").to-top Back to top
:marked :marked
Do not configure the `TestBed` after calling `compileComponents`. Do not configure the `TestBed` after calling `compileComponents`.
Make `compileComponents` the last step Make `compileComponents` the last step
before calling `TestBed.createInstance` to instantiate the test component. before calling `TestBed.createInstance` to instantiate the _component-under-test_.
:marked :marked
The `DashboardHeroComponent` spec follows the asynchonous `beforeEach` with a The `DashboardHeroComponent` spec follows the asynchonous `beforeEach` with a
_synchronous_ `beforeEach` that completes the setup steps and runs tests ... as described in the next section. _synchronous_ `beforeEach` that completes the setup steps and runs tests ... as described in the next section.
@ -1001,9 +1003,9 @@ a(href="#top").to-top Back to top
accepted by many handlers including the `RouterLink` directive. accepted by many handlers including the `RouterLink` directive.
.callout.is-critical .callout.is-critical
header click() is not an ATP function header click() is not an Angular testing utility
:marked :marked
The `click()` helper function is **not** part of the _Angular Testing Platform_. The `click()` helper function is **not** one of the Angular testing utilities.
It's a function defined in _this chapter's sample code_ and used by all of the sample tests. It's a function defined in _this chapter's sample code_ and used by all of the sample tests.
If you like it, add it to your own collection of helpers. If you like it, add it to your own collection of helpers.
:marked :marked
@ -1090,7 +1092,7 @@ a(href="#top").to-top Back to top
Notice the `inject` function in the second `it` argument. Notice the `inject` function in the second `it` argument.
+makeExample('testing/ts/app/dashboard/dashboard.component.spec.ts', 'inject')(format='.') +makeExample('testing/ts/app/dashboard/dashboard.component.spec.ts', 'inject')(format='.')
:marked :marked
The `inject` function is an independent feature of the _Angular Testing Platform_. The `inject` function is one of the Angular testing utilities.
It injects services into the test function where you can alter, spy on, and manipulate them. It injects services into the test function where you can alter, spy on, and manipulate them.
The `inject` function has two parameters The `inject` function has two parameters
@ -1557,12 +1559,12 @@ a(href="#top").to-top Back to top
However, testing a single use case is unlikely to explore the full range of a directive's capabilities. However, testing a single use case is unlikely to explore the full range of a directive's capabilities.
Finding and testing all components that use the directive is tedious, brittle, and almost as unlikely to afford full coverage. Finding and testing all components that use the directive is tedious, brittle, and almost as unlikely to afford full coverage.
[Isolated unit tests](#isolated-tests) might be helpful. [Isolated unit tests](#isolated-unit-tests) might be helpful.
But attribute directives like this one tend to manipulate the DOM. But attribute directives like this one tend to manipulate the DOM.
Isolated tests don't and therefore don't inspire confidence in the directive's efficacy. Isolated unit tests don't and therefore don't inspire confidence in the directive's efficacy.
A better solution is to create an artificial test component that demonstrates all ways to apply the directive. A better solution is to create an artificial test component that demonstrates all ways to apply the directive.
+makeExample('testing/ts/app/shared/highlight.directive.spec.ts', 'test-component', 'app/shared/highlight.directive.spec.ts (test component)')(format='.') +makeExample('testing/ts/app/shared/highlight.directive.spec.ts', 'test-component', 'app/shared/highlight.directive.spec.ts (TestComponent)')(format='.')
figure.image-display figure.image-display
img(src='/resources/images/devguide/testing/highlight-directive-spec.png' width="200px" alt="HighlightDirective spec in action") img(src='/resources/images/devguide/testing/highlight-directive-spec.png' width="200px" alt="HighlightDirective spec in action")
.l-sub-section .l-sub-section
@ -1594,17 +1596,15 @@ a(href="#top").to-top Back to top
.l-hr .l-hr
#isolated-tests #isolated-unit-tests
#testing-without-atp
:marked :marked
# Testing without the Angular Testing Platform # Isolated Unit Tests
Testing applications with the help of the Angular Testing Platform (ATP) is the main focus of this chapter. Testing applications with the help of the Angular testing utilities is the main focus of this chapter.
However, it's often more productive to explore the inner logic of application classes However, it's often more productive to explore the inner logic of application classes
with _isolated_ unit tests that don't use the ATP. with _isolated_ unit tests that don't depend upon Angular.
Such tests are often smaller, easier to read, Such tests are often smaller and easier to read, write and maintain.
and easier to write and maintain.
They don't They don't
* import from the Angular test libraries * import from the Angular test libraries
@ -1630,11 +1630,11 @@ a(href="#top").to-top Back to top
## Services ## Services
Services are good candidates for isolated unit testing. Services are good candidates for isolated unit testing.
Here are some synchronous and asynchronous unit tests of the `FancyService` Here are some synchronous and asynchronous unit tests of the `FancyService`
written without assistance from Angular Testing Platform. written without assistance from Angular testing utilities.
+makeExample('testing/ts/app/bag/bag.no-testbed.spec.ts', 'FancyService', 'app/bag/bag.no-testbed.spec.ts') +makeExample('testing/ts/app/bag/bag.no-testbed.spec.ts', 'FancyService', 'app/bag/bag.no-testbed.spec.ts')
:marked :marked
A rough line count suggests that these tests are about 25% smaller than equivalent ATP tests. A rough line count suggests that these isolated unit tests are about 25% smaller than equivalent Angular tests.
That's telling but not decisive. That's telling but not decisive.
The benefit comes from reduced setup and code complexity. The benefit comes from reduced setup and code complexity.
@ -1642,12 +1642,12 @@ a(href="#top").to-top Back to top
+makeTabs( +makeTabs(
`testing/ts/app/bag/bag.no-testbed.spec.ts, testing/ts/app/bag/bag.spec.ts`, `testing/ts/app/bag/bag.no-testbed.spec.ts, testing/ts/app/bag/bag.spec.ts`,
'getTimeoutValue, getTimeoutValue', 'getTimeoutValue, getTimeoutValue',
`app/bag/bag.no-testbed.spec.ts, app/bag/bag.spec.ts (with ATP)`) `app/bag/bag.no-testbed.spec.ts (Isolated), app/bag/bag.spec.ts (with Angular testing utilities)`)
:marked :marked
They have about the same line-count. They have about the same line-count.
The ATP version has more moving parts, including a couple of helper functions (`async` and `inject`). But the Angular-dependent version has more moving parts, including a couple of utility functions (`async` and `inject`).
Both work and it's not much of an issue if you're using the Angular Testing Platform nearby for other reasons. Both approaches work and it's not much of an issue if you're using the Angular testing utilities nearby for other reasons.
On the other hand, why burden simple service tests with ATP complexity? On the other hand, why burden simple service tests with added complexity?
Pick the approach that suits you. Pick the approach that suits you.
@ -1673,13 +1673,13 @@ a(href="#top").to-top Back to top
These _isolated_ unit testing techniques are great for exploring the inner logic of a service or its These _isolated_ unit testing techniques are great for exploring the inner logic of a service or its
simple integration with a component class. simple integration with a component class.
Use the Angular Testing Platform when writing tests that validate how a service interacts with components Use the Angular testing utilities when writing tests that validate how a service interacts with components
_within the Angular runtime environment_. _within the Angular runtime environment_.
#isolated-pipe-tests #isolated-pipe-tests
:marked :marked
## Pipes ## Pipes
Pipes are easy to test without the Angular Testing Platform (ATP). Pipes are easy to test without the Angular testing utilities.
A pipe class has one method, `transform`, that turns an input to an output. A pipe class has one method, `transform`, that turns an input to an output.
The `transform` implementation rarely interacts with the DOM. The `transform` implementation rarely interacts with the DOM.
@ -1694,12 +1694,11 @@ a(href="#top").to-top Back to top
Use simple Jasmine to explore the expected cases and the edge cases. Use simple Jasmine to explore the expected cases and the edge cases.
+makeExample('testing/ts/app/shared/title-case.pipe.spec.ts', 'excerpt', 'app/shared/title-case.pipe.spec.ts') +makeExample('testing/ts/app/shared/title-case.pipe.spec.ts', 'excerpt', 'app/shared/title-case.pipe.spec.ts')
:marked :marked
### Write ATP tests too ### Write Angular tests too
These are tests of the pipe _in isolation_. These are tests of the pipe _in isolation_.
They can't tell if the `TitleCasePipe` is working properly They can't tell if the `TitleCasePipe` is working properly as applied in the application components.
as applied in the application components.
Consider adding ATP component tests such as this one. Consider adding component tests such as this one:
+makeExample('testing/ts/app/hero/hero-detail.component.spec.ts', 'title-case-pipe', 'app/hero/hero-detail.component.spec.ts (pipe test)') +makeExample('testing/ts/app/hero/hero-detail.component.spec.ts', 'title-case-pipe', 'app/hero/hero-detail.component.spec.ts (pipe test)')
#isolated-component-tests #isolated-component-tests
@ -1707,12 +1706,12 @@ a(href="#top").to-top Back to top
## Components ## Components
Component tests typically examine how a component class interacts with its own template or with collaborating components. Component tests typically examine how a component class interacts with its own template or with collaborating components.
The Angular Testing Platform is specifically designed to facilitate such tests. The Angular testing utilities are specifically designed to facilitate such tests.
Consider this `ButtonComp` component. Consider this `ButtonComp` component.
+makeExample('testing/ts/app/bag/bag.ts', 'ButtonComp', 'app/bag/bag.ts (ButtonComp)')(format='.') +makeExample('testing/ts/app/bag/bag.ts', 'ButtonComp', 'app/bag/bag.ts (ButtonComp)')(format='.')
:marked :marked
The following ATP test demonstrates that clicking a button in the template leads The following Angular test demonstrates that clicking a button in the template leads
to an update of the on-screen message. to an update of the on-screen message.
+makeExample('testing/ts/app/bag/bag.spec.ts', 'ButtonComp', 'app/bag/bag.spec.ts (ButtonComp)')(format='.') +makeExample('testing/ts/app/bag/bag.spec.ts', 'ButtonComp', 'app/bag/bag.spec.ts (ButtonComp)')(format='.')
:marked :marked
@ -1720,32 +1719,32 @@ a(href="#top").to-top Back to top
from the component back to a _different_ HTML control (the `<span>`). from the component back to a _different_ HTML control (the `<span>`).
A passing test means the component and its template are wired up correctly. A passing test means the component and its template are wired up correctly.
Tests _without_ the ATP can more rapidly probe a component at its API boundary, Isolated unit tests can more rapidly probe a component at its API boundary,
exploring many more conditions with less effort. exploring many more conditions with less effort.
Here are a set of _unit tests_ that verify the component's outputs in the face of a variety of Here are a set of unit tests that verify the component's outputs in the face of a variety of
component inputs. component inputs.
+makeExample('testing/ts/app/bag/bag.no-testbed.spec.ts', 'ButtonComp', 'app/bag/bag.no-testbed.spec.ts (ButtonComp)')(format='.') +makeExample('testing/ts/app/bag/bag.no-testbed.spec.ts', 'ButtonComp', 'app/bag/bag.no-testbed.spec.ts (ButtonComp)')(format='.')
:marked :marked
Isolated component tests offer a lot of test coverage with less code and almost no setup. Isolated component tests offer a lot of test coverage with less code and almost no setup.
This advantage is even more pronounced with complex components that This advantage is even more pronounced with complex components that
require meticulous preparation with the Angular Testing Platform. may require meticulous preparation with the Angular testing utilities.
On the other hand, isolated unit tests can't confirm that the `ButtonComp` is On the other hand, isolated unit tests can't confirm that the `ButtonComp` is
properly bound to its template or even data bound at all. properly bound to its template or even data bound at all.
Use ATP tests for that. Use Angular tests for that.
a(href="#top").to-top Back to top a(href="#top").to-top Back to top
.l-hr .l-hr
#atp-api #atu-apis
:marked :marked
# Angular Testing Platform APIs # Angular Testing Utility APIs
This section takes inventory of the most useful _Angular Testing Platform_ features and summarizes what they do. This section takes inventory of the most useful Angular testing features and summarizes what they do.
The _Angular Testing Platform_ consists of the `TestBed` and `ComponentFixture` classes plus a handful of functions in the test environment. The Angular testing utilities include the `TestBed`, the `ComponentFixture`, and a handful of functions that control the test environment.
The [_TestBed_](#testbed-api-summary) and [_ComponentFixture_](#component-fixture-api-summary) classes are covered separately. The [_TestBed_](#testbed-api-summary) and [_ComponentFixture_](#component-fixture-api-summary) classes are covered separately.
Here's a summary of the stand-alone functions, in order of likely utility: Here's a summary of the stand-alone functions, in order of likely utility:
@ -1834,7 +1833,7 @@ table
#testbed-class-summary #testbed-class-summary
:marked :marked
# _TestBed_ Class Summary # _TestBed_ Class Summary
The `TestBed` class is a principle feature of the _Angular Testing Platform_. The `TestBed` class is one of the principal Angular testing utilities.
Its API is quite large and can be overwhelming until you've explored it first Its API is quite large and can be overwhelming until you've explored it first
a little at a time. Read the early part of this chapter first a little at a time. Read the early part of this chapter first
to get the basics before trying to absorb the full API. to get the basics before trying to absorb the full API.