<< Part 1 – Angular Test Basic
<< Part 2 – Jasmine Grammar Introduction
Before you start writing Unit tests, you need to be clear about one question: What is a Unit Test?
Unit tests: Unit tests are code written by developers to check the correctness of object code under certain conditions. They are checks and validations of the smallest testable units in software. Unit testing is the lowest level of testing activity to be performed during software development, where individual units of software are tested in isolation from the rest of the program.
Highlight: Unit tests focus only on functional code, not databases, background requests, etc. If a service or router is involved, you can use fakeService and fakeRouter to handle it.
Here are some simple demos to understand the writing method and ideas of unit tests in detail. The following examples start from the test of method classes and temporarily do not involve the test of template rendering, Component dependency, interaction and other contents. They are only for reviewing their own learning process and accepting criticism humbly.
Note: The project code is only to understand the specific writing method and basic ideas of unit test, SRC /app directory (01-07) corresponding to the functional code and test code, do not need to pay attention to the business logic of the functional code itself. Download or clone the code locally
NPM I // Installation depends on ngtest// Run unit testsCopy the code
1. Start by testing simple calculation methods
The following is a simple calculation. There is an if judgment in the functional code, so there are two test branches, the case of number<0 and number>=0, so at least two test cases are required to override the method.
// compute.ts
export function compute (number) {
if (number < 0) {
number = 0;
return number;
} else {
returnnumber + 1; }}Copy the code
Unit test:
// compute.spec.ts
describe('compute', () => {
it('should return 0 if input is negative', () => {
const result = compute(-1);
expect(result).toBe(0);
});
it('should return 1 if input is 0', () => {
const result = compute(0);
expect(result).toBe(1);
});
});
Copy the code
In the above test code, the Describe method defines a set of test cases, the IT method defines a specific test case, and the Expect and toBe() methods make assertions, all of which are Jasmine’s basic syntax and won’t be described here. See Angular Test Basic, Jasmine Introduction to Grammar Part I.
Two test cases are constructed above. Through the difference of input, the test conditions meet the two branches in the functional code, namely number<0 and number>=0, to judge whether the output meets the expectation.
A complete unit test must cover all branches of the functional code. In particular, if or switch conditions exist, to consider all the circumstances of the test.
Generally, a test case can be divided into three parts: Arrange=>Act=>Assert. The above test cases are tests for a specific method, so the Arrange part is omitted. Act can be called and Assert can be asserted directly.
2. Unit test for string and array methods
Define a simple method that returns an array of three currencies:
// getCurrency.ts
export function getCurrency() {
return ['CHY'.'USD'.'EUR'];
}
Copy the code
Corresponding test code:
// getCurrency.spec.ts
describe('getCUrrency', () => {
it('should contain the correct currency', () => {
// Act
const result = getCurrency();
// Asset
expect(result).toContain('CHY');
expect(result).toContain('USD');
expect(result).toContain('EUR');
});
});
Copy the code
ToContain,toBe(Length),index, etc.
Next, define a simple method that takes an argument and returns a fixed-format string.
// sayHello.ts
export function sayHello(name) {
return `Hello ${name}!!!!! `; }Copy the code
Test code:
// sayHello.spec.ts
describe('sayHello', () => {
it('should contains the input name', () => {
// Act
const result = sayHello('Reina');
// Asset
expect(result).toContain('Reina');
// expect(result).toBe('Hello Reina !! ');
});
});
Copy the code
In the above test code, expect(result).tobe (‘Hello Reina!! ‘), if toBe’s judgment is used, any changes will cause the test code to fail, which is too fragile.
3. Unit tests for simple Components
The following defines a simple Vote component (template,stylesheet, and so on are not considered here) that contains both upVote and downVote methods that control the addition and subtracting of totalVote, respectively.
// vote.component.ts
export class VoteComponent {
totalVotes = 0;
upVote() {
this.totalVotes++;
}
downVote() { this.totalVotes--; }}Copy the code
For upVote() and downVote(), define two test cases respectively. In each test case, we need to initialize a voteComponent to ensure that the two test cases are independent of each other and totalVotes does not affect each other:
describe('vote', () => {
it('should increment the totalvotes when upvote', () => {
// Arrange
const component = new VoteComponent();
// Act
component.upVote();
// Assert
expect(component.totalVotes).toBe(1);
});
it('should decrement the totalvotes when downvote', () => {
// Arrange
const component = new VoteComponent();
// Act
component.downVote();
// Assert
expect(component.totalVotes).toBe(-1);
});
});
Copy the code
As the volume of tests increases, components may need to be initialized in multiple test cases, resulting in a lot of repetitive code, in which case you can use beforeEach for refactoring.
describe('vote', () = > {let component: VoteComponent;
beforeEach(() => {
component = new VoteComponent();
});
it('should increment the totalvotes when upvote', () => {
// Act
component.upVote();
// Assert
expect(component.totalVotes).toBe(1);
});
it('should decrement the totalvotes when downvote', () => {
// Act
component.downVote();
// Assert
expect(component.totalVotes).toBe(-1);
});
});
Copy the code
In the above test code, a component variable is declared but not initialized, and the initialization code is placed in beforeEach, which is executed beforeEach test case is run. Jasmine also provides beforeAll, except beforeEach afterEach, afterAll. Literally, beforeEach and afterEach are executed before and afterEach test case execution, respectively; BeforeAll and afterAll are executed before and afterAll test cases, respectively.
4. Unit test of the form
First, initialize a FormGroup with The FormBuilder and add two fields name and email to it, making name mandatory.
// form.component.ts
exportclass FormComponent { formGroup: FormGroup; Constructor (fb: FormBuilder) {// Initialize a formGroup and set name to required this. formgroup = fb.group({name: [' ', Validators.required],
email: [' ']}); }}Copy the code
Test each of the three conditions:
- The formGroup should contain the name and email fields
- The name field is empty and the form is invalid
- If a valid value is entered in the name field, the form becomes valid
describe('form', () = > {letcomponent: FormComponent; BeforeEach (() => {// To initialize formComponent, pass in an instance of formBuilder component = new formComponent (new formBuilder ()); }); it('should create the formGroup with 2 fields', () => {
expect(component.formGroup.contains('name')).toBeTruthy();
expect(component.formGroup.contains('email')).toBeTruthy();
});
it('should make the name control required', () => {
const control = component.formGroup.get('name');
control.setValue(' ');
expect(control.valid).toBeFalsy();
});
it('should make the name control valid if input acceptable name', () => {
const control = component.formGroup.get('name');
control.setValue('Reina');
expect(control.valid).toBeTruthy();
});
});
Copy the code
The above test code first initializes the formComponent in beforeEach, passing new formBuilder () as an argument because it needs an instance of formBuilder in its constructor.
In addition, in the above test code, formControl and formGroup attributes and methods are used, and Boolean values are judged by toBeFalsy(), toBeTruthy() and other methods.
5. EventEmitter Unit Test
Again using the VoteComponent method as an example, in this component, a voteChanged method is defined as output, and this.votechanged. Emit (this.totalVotes) is fired in upVote.
// eventEmitter/vote.component.ts
export class VoteComponent {
totalVotes = 0;
@Output() voteChanged = new EventEmitter();
upVote() {
this.totalVotes ++;
this.voteChanged.emit(this.totalVotes);
}
Copy the code
When testing eventEmitter, first ensure that the corresponding output method will be triggered, and determine whether the obtained parameters are correct. In addition, the EventEmitter class is derived from Observable, so you can subscribe to it. For example:
describe('voteComponent', () = > {let component: VoteComponent;
beforeEach(() => {
component = new VoteComponent();
});
it('should trigger voteChanged with totalVotes when upVotes', () => {
// Arrange
let totalVotes = null;
component.voteChanged.subscribe(tv => totalVotes = tv);
// Act
component.upVote();
// Assert
expect(totalVotes).toBe(1);
});
});
Copy the code
In the above test case, we first subscribe to VoteChanged and get an assignment to totalVotes in its callback; The UPDATE method is then triggered, making assertions based on the value of totalVotes.
6. Unit test for service
First, to recall from the beginning, unit testing is only about functional code, not about databases, background requests, etc. Therefore, when unit testing a Service, the request should not actually be triggered, but rather through fakeService. Whether a service request returns the correct value is the domain of integration testing and is beyond this consideration.
First, we construct a heroComponent and a heroService, respectively. HeroService provides the data request for heroComponent, and we inject heroService into heroComponent.
In service, three methods are defined to push back the request, regardless of the implementation and its actual return value:
// hero.service.ts
export class HeroService {
constructor(private http: Http) { }
getHeroes() {
return this.http.get('... ');
}
addHero(hero) {
return this.http.post('... ', hero);
}
deleteHero(id) {
return this.http.get('... '+ id); }}Copy the code
In the heroComponent, the getHeros method of the Service is called at initialization, and the add and delete methods call the addHero and deleteHero methods of the Service, respectively.
// hero.component.ts
export class HeroComponent implements OnInit {
heroList: any;
message: string;
constructor(private heroService: HeroService ) {}
ngOnInit() {
this.heroService.getHeroes().subscribe((res) => {
this.heroList = res;
});
}
add() {
let hero = { id: 1, name: 'hero'}; this.heroService.addHero(hero).subscribe( res => { this.heroList.push(res) }, err => { this.message = err; }); } delete(id) {if ( confirm('Are you sure to delete? ')) { this.heroService.deleteHero(id); }}}Copy the code
Because the service method is called in three cases in the component, look at the test cases for each case:
- The getHeros method is called in ngOnInit and the return value is assigned to heroList
- The Add method calls addHeros and handles the successful and failed requests, respectively
- The delete method first uses window.confirm and then calls the deleteHero method
First, initialize component and service in beforeEach, since no real request is sent when testing, use NULL instead of HTTP to initialize service, and inject service into Component:
describe('heroComponent', () = > {let service: HeroService;
letcomponent: HeroComponent; beforeEach(() => { service = new HeroService(null); component = new HeroComponent(service); }); / /... });Copy the code
After initialization, test the first function point (ngOnInit calls the getHeros method and assigns the return value to heroList) :
it('should set heroList with the heros returned from service', () => {
// Arrange
const heroes = ['Reina'.'David'];
spyOn(service, 'getHeroes').and.callFake(() => {
return of(heroes);
});
// Act
component.ngOnInit();
// Assert
expect(component.heroList).toBe(heroes);
});
Copy the code
In the above test case,spyOn () is used to add monitoring of the execution of a function on an object. The spyOn method takes two parameters, the monitored object and the method name of the monitored function. SpyOn (service, ‘getHeroes’) monitors the getHeroes method in the service object; The listening method return value can be forged with the and.callfake method, where a custom function returns the heroes array.
NgOnInit is then called to trigger the getHeroes method and the result is determined by the return values of component.heroList and mock.
In addition to the and.callfake method, jasmine provides an and.returnvalue method to specify the method returnValue to listen on; The and.throwError method allows the listener to return an error message after execution, which can be adapted using toThrowError, etc.
Next, for the second function point (addHeros is called in the Add method and the request is processed successfully and failed, respectively), we test the following three cases:
- The addHero method in the service can be called successfully;
it('should call the service to save changes when new hero is added', () => {
const spy = spyOn(service, 'addHero').and.returnValue(empty());
component.add();
expect(spy).toHaveBeenCalled();
});
Copy the code
In the above test case, the addHero method in the service is listened on, so it only tests whether the method is called properly, and therefore does not process the returned value. An empty data stream is returned using the empty operator of RXJS. ToHaveBeenCalled () checks whether the listener is called.
- When the addHero request succeeds, its return value can be added to the heroLists;
it('should add the new hero to herolist return from the server', () => {
spyOn(service, 'addHero').and.returnValue(of('Lee'));
component.add();
expect(component.heroList.indexOf('Lee')).toBeGreaterThan(-1);
});
Copy the code
In the above test case, we also listen for the addHero method in the service, return ‘Lee’ through and.returnValue(of(‘Lee’)), and determine whether Lee was successfully added to the heroList array. Here we use the OF operator of RXJS and the toBeGreaterThan method provided in Jasmine.
- When addHero reports an error, Message gets the error message.
it('should set error message with the msg return from server when add new hero throw error', () => {
const mockError = 'Add Hero Error';
spyOn(service, 'addHero').and.returnValue(throwError(mockError));
component.add();
expect(component.message).toBe(mockError);
});
Copy the code
In this test case, we also listen for the addHero method in the service and return an error message with the throwError operator of RXJS via the AND.returnValue method and determine whether the Component’s message was successfully assigned.
Finally, the third function point is tested (the delete method first uses window.confirm and then calls the deleteHero method after confirmation). Test both confirm deletion and cancel deletion respectively:
it('should delete the hero with id when user confirm to delete', () => {
spyOn(window, 'confirm').and.returnValue(true);
const spy = spyOn(service, 'deleteHero').and.returnValue(empty());
component.delete(1);
expect(spy).toHaveBeenCalledWith(1);
});
Copy the code
In the preceding test case, spyOn(window, ‘confirm’).and.returnvalue (true) is used to listen on the confirm operation and set the returnValue to true, that is, the confirm operation is deleted. We then listen for the deleteHero method in the service and return null with AND.returnValue to determine whether the deleteHero method was successfully called and check that the parameter information is consistent with that passed in.
Similarly, verify that the delete method is not called by spyOn(window, ‘confirm’).and.returnValue(false) to simulate a user cancellation scenario. Jasmine’s common assertion method, which precedes any assertion method with not, means the opposite. For example:
it('should NOT delete the hero with id when user cancel to delete', () => {
spyOn(window, 'confirm').and.returnValue(false);
const spy = spyOn(service, 'deleteHero').and.returnValue(empty());
component.delete(1);
expect(spy).not.toHaveBeenCalledWith(1);
});
Copy the code
Above, a simple example documents the basics of getting started with Angular unit testing. Of course, in real projects, it’s much more complicated than that, so keep digging
The demo source code