原文连接javascript
作了多年的Angular的前端开发,一直没有胆量对前端进行单元测试,缘由一是前端是跟用户打交道,很差测试,缘由二是项目的时间压力没有精力弄单元测试。这也就致使在前端开发时,业务一旦改变,就要人肉进行测试。费时又没有技术含量,直接让我怀疑人生。最近得空,索性就把Angular的单元测试研究了一把。Angular其实本身有单元测试的工具:Karma + Jasmine:html
当建立Angular应用后,在package.json文件中已经添加了Karma和Jasmine的依赖性:前端
"karma": "~1.7.1",
"karma-chrome-launcher": "~2.2.0",
"karma-coverage-istanbul-reporter": "~2.0.0",
"karma-jasmine": "~1.1.1",
"karma-jasmine-html-reporter": "^0.2.2",
复制代码
作事后端测试的同行,估计已经知道这些组件的分工了:java
在src目录下会看到名为:karma.conf.js、test.ts的两个文件。web
karma.conf.js:Karma的配置文件,其中须要重点关注的配置有:chrome
frameworks:使用的测试框架,这里使用Jasmineshell
port:测试使用的端口数据库
autoWatch:是否自动监测测试代码的改变,自动执行测试npm
plugins:测试使用到的插件,与package.json文件保持一致json
browsers:测试运行使用的浏览器,这里使用Chrome,若是你须要使用其余浏览器,须要经过npm安装浏览器发射器,并在plugins和这里设置,例如使用Safari:
npm install karma-safari-launcher --save-dev
plugins: [
require('karma-safari-launcher')
]
browsers: ['Safari'],
复制代码
test.ts:测试入口文件,其中初始化了测试环境以及指定全部测试文件
在app目录下,还会找到一个名为app.component.spec.ts的文件,这就是一个Jasmine的测试,内容以下:
import { TestBed, async } from '@angular/core/testing';
import { AppComponent } from './app.component';
//测试入口,参数为测试名、方法
describe('AppComponent', () => {
//每一个测试用的Setup
beforeEach(async(() => {
TestBed.configureTestingModule({
declarations: [
AppComponent
],
}).compileComponents();
}));
//测试用例
it('should create the app', async(() => {
const fixture = TestBed.createComponent(AppComponent);
const app = fixture.debugElement.componentInstance;
expect(app).toBeTruthy();
}));
it(`should have as title 'test-demo'`, async(() => {
const fixture = TestBed.createComponent(AppComponent);
const app = fixture.debugElement.componentInstance;
//断言,指望值是否知足要求
expect(app.title).toEqual('test-demo');
}));
it('should render title in a h1 tag', async(() => {
const fixture = TestBed.createComponent(AppComponent);
fixture.detectChanges();
const compiled = fixture.debugElement.nativeElement;
//经过querySelector获取页面元素
expect(compiled.querySelector('h1').textContent).toContain('Welcome to test-demo!');
}));
//每一个测试用例的TearDown
afterEach(function() {
//清除测试数据
});
});
复制代码
上述代码使用了Jasmine的语法,关于Jasmine的更详细介绍,参见JavaScript 单元测试框架:Jasmine 初探。这里不赘述。
执行: ng test,就会看到上述文件的测试报告:
另外在测试报告中还可单击某个测试单独执行,报告以下:
对于Pipe、Service、Router等组件的测试,可参见Angular文档,这里重点讲述下在测试中遇到的各类坑。
测试时,若是被测组件须要其余第三方组件、servcie或pipe,没有被引入,就会出现No provider 的错误,解决方法很简单,在beforeEach中使用imports或provider引入便可:
beforeEach(async(() => {
TestBed.configureTestingModule({
declarations: [
//这里声明
],
imports: [
//这里引入
],
providers: [
//这里引入
],
schemas: [CUSTOM_ELEMENTS_SCHEMA],
})
.compileComponents();
}));
复制代码
在开发时,异步请求因为网络缘由常会出现TimeOut的错误,一般的解决方法是设置TimeOut时间的上限,并对TimeOut错误做出人性化的提示。在测试时也一样会发生TimeOut的错误:
it('#loadBalance for BCT should return real value', async () => {
jasmine.DEFAULT_TIMEOUT_INTERVAL = 1000000;
...
})
复制代码
或者在BeforeEach中统一设置TimeOut时间:
describe("my async specs", function() {
var originalTimeout;
beforeEach(function() {
originalTimeout = jasmine.DEFAULT_TIMEOUT_INTERVAL;
jasmine.DEFAULT_TIMEOUT_INTERVAL = 1000000;
});
...
afterEach(function() {
jasmine.DEFAULT_TIMEOUT_INTERVAL = originalTimeout;
});
});
复制代码
Angular缺省针对开发和产品提供了不一样的Environment,对于测试,咱们一样能够设置Enviroment。
在src/environment下建立environment.test.ts,并修改angular.json内容:
"architect":{
"test":{
...
"configurations": {
"test": {
"fileReplacements": [
{
"replace": "src/environments/environment.ts",
"with": "src/environments/environment.test.ts"
}
]
}
}
}
}
复制代码
修改package.json文件:
"scripts": {
"test": "ng test --configuration=test",
}
复制代码
这样执行以下命令:
npm test
//或者
ng test --configuration=test
复制代码
执行测试时,使用的就是environment.test.ts文件中配置的内容。
作过Grails开发的伙计应该知道,单元测试、集成测试后,数据库中的测试数据会经过配置文件清除掉。在前端测试中,测试数据须要自行调用清除代码,对于使用LocalStorage、SessionStorage保持的数据亦是如此,方法很简单,在afterEach添加清除代码:
describe("my async specs", function() {
afterEach(function() {
//在这里清除测试数据
});
});
复制代码
先前我发布了一篇题为《StoryBook实战》的文章,StoryBook也是用来测试组件的,它与Karma+Jasmine有什么区别呢?
两者都能测试的:
StoryBook不能测、Karma + Jasmine可测试的:
Karma + Jasmine不能作的,StoryBook能作的:
从上面能够看出,Storybook进行的是黑盒测试,Karma + Jasmine则注重白盒测试,两者侧重点不一样,没有谁强谁弱之分,只有扬长避短,利用好各自的优势,方可以让前端测试更完美,将前端bug扼杀在开发阶段。
虽然前端开发的工做比较繁琐,也是客户Challenge最多的地方,可是不表明前端只有页面,没有架构。之前之因此以为Angular的单元测试难作,就是以为都是页面的东西怎么测?其实,终其缘由,仍是没有架构,全部的代码都集中在Component中,为了赶进度,经过拷贝、粘贴,怎么快怎么来。结果,悲剧了,后期代码维护困难,一点改动就须要人肉测试。费时不说,开发人员也没有成长。接触Angular前端测试后,个人脑海里又出现了“测试驱动开发”。一段好代码,前提是要易于测试,无论这段代码是用于前端仍是后端。 前端开发人员不只仅要关注页面的易用性、美观性,一样须要关注前端的架构,一个易于测试的架构才是最好的“武器”。