Why Most Angular 2+ Testing Tuorials Are Wrong

Most Angular 2+ testing tutorials (including the official tutorial on angular.io) recommend the wrong way to test Components.

Consider the following component:

// System Under Test Component
@Component({
template: `<h1>{{title}}</h1>`,
})
export class SutComponent {
@Input() title;
}

and its test code:

describe('SutComponent', () => {
beforeEach(() => {
TestBed.configureTestingModule({
declarations: [SutComponent],
});
TestBed.compileComponents();
});
it('should render default title in a h1 tag', async(() => {
const fixture = TestBed.createComponent(SutComponent);
// reach into component to set an input value
fixture.componentInstance.title = 'Hello world!';
fixture.detectChanges();
const compiled = fixture.debugElement.nativeElement;
expect(compiled.querySelector('h1').textContent).toContain('Hello world!');
}));
});

The test code reaches into the component to modify the value of title. This is a common practice you’ll find in most tutorials, and it is problematic in a couple of ways.

Problem 1: The test is fragile, and can break even when the implementation is correct.

This test makes assumptions into the implementation of SutComponent, i.e. it assumes that the component has a field called title. However, SutComponent was intended to have a property title that a parent could bind to, and not a public field named title.

If for some reason we need the field to be someOtherName, but the property name remains as title, as illustrated below:

@Component({
template: `<h1>{{someOtherName}}</h1>`,
})
export class SutComponent {
@Input('title') someOtherName;
}
view raw prob-1.ts hosted with ❤ by GitHub

This implementation would still work correctly in the application but the test would now fail.

Problem 2: The test is incomplete, it can pass even if implementation is broken.

For the same reason, the original test does not exercise the property binding title.

If we remove the property binding but keep the field title, as illustrated in the code below, the component would now not work as intended, but the test above would still pass and not catch the error.

@Component({
template: `<h1>{{title}}</h1>`,
})
export class SutComponent {
title; // without @Input()
}
view raw prob-2.ts hosted with ❤ by GitHub

The Right Way

To test Angular 2+ Components correctly, our tests must be written without knowledge or assumption of the component’s implementation details. Tests must be written to test a component’s public interfaces and contracts and only its public interfaces and contracts.

What are the public interfaces of Angular 2+ Components?

A well-encapsulated component can have the following types of public interfaces:

  • Property binding, as illustrated above
  • Event bindings, through which a component can emit an event to its parent
  • UI rendering and interactions
  • Interactions with models, business logic, and external systems through dependency-injected services

These must be the only interfaces used in the Angular 2+ component test code.

Consider the following component ArticleComponent that contains all these types of public interfaces:

@Component({
selector: 'app-article',
// UI rendering
template: `
<h2>{{article.title}}</h2>
<p>{{text}}<p>`
})
export class ArticleComponent {
// Property Binding
@Input() private article;
// Event Binding
@Output() private select = new EventEmitter<number>();
// UI input
@HostListener('click') private onClick() {
this.select.emit(this.article.id);
}
// Service to model, logic and external systems
constructor(private articlesService: ArticlesService) { }
private get text(): string {
return this.articlesService.getText(this.article.id);
}
}

You can see that we declare all the fields and methods as private. This would not affect the application, and it helps enforcing the strict encapsulation during development. (Update 2017-04-19: Private fields are no longer accepted for production build. However, to preserve modularity, fields should not be accessed directly from outside its component.)

To test ArticleComponent, we start by creating a mock parent component that binds to the properties and events of ArticleComponent. We also create a mock instance of ArticlesService, mockArticlesService, with the interfaces we intend to use in the component:

@Component({
template: `<app-article
[article]='article'
(select)='onSelect($event)'>
</app-article>`
})
export class TestParent {
public article = {id: ARTICLE_ID, title: ARTICLE_TITLE};
public onSelect(id: number) { }
}
// Mock service
let mockArticleService = {
getText(id: number) { return ARTICLE_TEXT; }
};

Instead of instantiating ArticleComponent, we instantiate the TestParent component which binds to the properties and events of ArticleComponent. The ArticleComponent test fixture is then extracted from the TestParent fixture.

To test the interactions between ArticleComponent and ArticlesService, mockArticlesService is injected as a provider for ArticlesService. We then spy on this mock service instance using jasmine’s spyOn.

Note that spyOn must be called on the ArticlesService instance that provided by TestBed, not on the mockArticlesService object we created.

beforeEach(async(() => {
TestBed.configureTestingModule({
declarations: [TestParent, ArticleComponent],
providers: [{provide: ArticlesService, useValue: mockArticleService}],
schemas: [NO_ERRORS_SCHEMA]
})
.compileComponents();
}));
beforeEach(() => {
parentFixture = TestBed.createComponent(TestParent);
// spy on the injected articlesService
let articleService = TestBed.get(ArticlesService);
spy = spyOn(articleService, 'getText').and.callThrough();
parentFixture.detectChanges();
debugElement = parentFixture.debugElement.query(By.css('app-article'));
compiled = debugElement.nativeElement;
});

The tests are then performed on these instances:

it('should display title property', () => {
expect(compiled.querySelector('h2').textContent).toContain(ARTICLE_TITLE);
});
it('should retrieve and display article text', () => {
// check that correct test is displayed
expect(compiled.querySelector('p').textContent).toContain(ARTICLE_TEXT);
// check that getText is called with the correct param
expect(serviceGetTextSpy).toHaveBeenCalledWith(ARTICLE_ID);
});
it('should emit select event with article id when clicked', () => {
let parentOnSelectSpy = spyOn(parentFixture.componentInstance, 'onSelect');
debugElement.triggerEventHandler('click', '');
// check parent event binding with correct event param emitted
expect(parentOnSelectSpy).toHaveBeenCalledWith(ARTICLE_ID);
});

Putting it all together, the test code reads:

// faker node module for randomising test data
import * as faker from 'faker';
const ARTICLE_ID = faker.random.number;
const ARTICLE_TITLE = faker.lorem.sentence;
const ARTICLE_TEXT = faker.lorem.paragraph;
// Test Parent for property and event bindings
@Component({
template: `<app-article
[article]='article'
(select)='onSelect($event)'>
</app-article>`
})
export class TestParent {
public article = {id: ARTICLE_ID, title: ARTICLE_TITLE};
public onSelect(id: number) { }
}
// Mock service
let mockArticleService = {
getText(id: number) { return ARTICLE_TEXT; }
};
describe('ArticleComponent', () => {
let parentFixture: ComponentFixture<TestParent>;
let debugElement: DebugElement;
let compiled: any;
let serviceGetTextSpy: any;
beforeEach(async(() => {
TestBed.configureTestingModule({
declarations: [TestParent, ArticleComponent],
providers: [{provide: ArticlesService, useValue: mockArticleService}],
schemas: [NO_ERRORS_SCHEMA]
})
.compileComponents();
}));
beforeEach(() => {
parentFixture = TestBed.createComponent(TestParent);
let articleService = TestBed.get(ArticlesService);
serviceGetTextSpy = spyOn(articleService, 'getText').and.callThrough();
parentFixture.detectChanges();
debugElement = parentFixture.debugElement.query(By.css('app-article'));
compiled = debugElement.nativeElement;
});
it('should display title property', () => {
expect(compiled.querySelector('h2').textContent).toContain(ARTICLE_TITLE);
});
it('should retrieve and display article text', () => {
// check that correct test is displayed
expect(compiled.querySelector('p').textContent).toContain(ARTICLE_TEXT);
// check that getText is called with the correct param
expect(serviceGetTextSpy).toHaveBeenCalledWith(ARTICLE_ID);
});
it('should emit select event with article id when clicked', () => {
let parentOnSelectSpy = spyOn(parentFixture.componentInstance, 'onSelect');
debugElement.triggerEventHandler('click', '');
// check parent event binding with correct event param emitted
expect(parentOnSelectSpy).toHaveBeenCalledWith(ARTICLE_ID);
});
});

These tests exercise the intended interfaces and contracts of the component, in the ways that the component would be used in the application.

Summing Up

In this article, you should have learned:

  • Why most Angular 2+ Component testing tutorial are wrong
  • What are the public interfaces of an Angular 2+ Component and how to test them