前端自动化单元测试初探

前言

本篇文章是我在学习前端自动化单元测试时的一些思路整理,之前也从未接触过单元测试相关工具,如有错漏,请读者斧正。要是觉得不专业...你打我呀~~~
示例代码的github地址:https://github.com/BboyAwey/auto-unit-test-testing

1. 什么是单元测试

单元测试(unit testing),是指对软件中的最小可测试单元进行检查和验证。对于单元测试中单元的含义,一般来说,要根据实际情况去判定其具体含义,如C语言中单元指一个函数,Java里单元指一个类,图形化的软件中可以指一个窗口或一个菜单等。总的来说,单元就是人为规定的最小的被测功能模块。单元测试是在软件开发过程中要进行的最低级别的测试活动,软件的独立单元将在与程序的其他部分相隔离的情况下进行测试。——百度百科

对于JavaScript来说,通常也是针对函数、对象和模块的测试

2. 为什么要进行单元测试

经验表明一个尽责的单元测试方法将会在软件开发的某个阶段发现很多的Bug,并且修改它们的成本也很低。在软件开发的后期阶段,Bug的发现并修改将会变得更加困难,并要消耗大量的时间和开发费用。无论什么时候作出修改都要进行完整的回归测试,在生命周期中尽早地对软件产品进行测试将使效率和质量得到最好的保证。在提供了经过测试的单元的情况下,系统集成过程将会大大地简化。开发人员可以将精力集中在单元之间的交互作用和全局的功能实现上,而不是陷入充满很多Bug的单元之中不能自拔。——百度百科

3. 如何进行单元测试

3.1 选择测试工具

在JavaScript世界中,我们需要至少三个工具来进行单元测试,这意味着每个工具都需要你进行选择:

  • 测试管理工具
    测试管理工具是用来组织和运行整个测试的工具,它能够将测试框架、断言库、测试浏览器、测试代码和被测试代码组织起来,并运行被测试代码进行测试。测试工具有很多选择,Selenium、WebDriver/Selenium 2、Mocha[1]、JsTestDriver、HTML Runners和Karma,我这里选择使用Karma。(关于它们的对比,可以看看这篇文章:karma 测试框架的前世今生
  • 测试框架
    测是框架是单元测试的核心,它提供了单元测试所需的各种API,你可以使用它们来对你的代码进行单元测试。JavaScript的测试框架可谓百花齐放,选择太多了(可以参考List of unit testing frameworks),我这里选择使用Mocha(关于它们中一些框架的对比,可以参考javascript单元测试
  • 断言库
    断言库提供了用于描述你的具体测试的API,有了它们你的测试代码便能简单直接,也更为语义化,理想状态下你甚至可以让非开发人员来撰写单元测试。当然,你也完全可以不使用断言库,而是用自己的测试代码去测试,不过几乎没有人会这么干,除非你自己实现了一个测试断言库。测试断言库的选择也不少:better-assert、should.js、expect.js、chai.js等等(有关它们的对比,可以参考几款前端测试断言库(Assertions lib)的选型总结我这里选择chai.js

有了上面的三个工具,你就可以开始对你的node代码进行测试了。但是如果想要对前端代码进行测试,还需要另外一个工具:

  • 测试浏览器
    前端代码是运行在浏览器中的,要对其进行单元测试,只能将其运行在浏览器上。目前大部分测试工具都支持调用和运行本地浏览器来进行测试,但如果你的测试仅仅是针对函数和模块的单元测试,则完全可以使用一款无界面的浏览器:PhantomJs

另外,还有一个很重要的事情就是测试覆盖率的统计。一般情况下你的测试管理工具会提供相关的覆盖率统计工具,但是有些情况下它们提供的工具未必是你想要的。比如当被测试的代码是经过了某些打包工具打包完了且被压缩和混淆了,同时打包工具还混入了很多自己的代码,这时覆盖率的统计就容易不准确。所以为了避免这种情况,测试覆盖率统计工具需要谨慎选择,至少你得确认它支持你的打包工具已经打包好的代码。

  • 测试覆盖率统计工具
    Karma-Coverage是Karma官方提供的覆盖率统计插件,自然成为项目的首选。

至此,我们需要的工具已经完备了,下面是选择好的工具清单:

  • 测试管理工具:Karma
  • 测试框架:Mocha
  • 断言库:Chai
  • 测试浏览器:PhantomJs
  • 测试覆盖率统计工具:Karma-Coverage

3.2 构建一个最基本的测试

3.2.1 配置好你的npm

初始化项目的package.json,并将需要的工具安装到项目中,安装完成后pakeage.json的devDependencies中应当出现下面的这些工具

"devDependencies": {
  "chai": "^3.5.0",
  "karma": "^1.3.0",
  "karma-chai": "^0.1.0",
  "karma-coverage": "^1.1.1",
  "karma-mocha": "^1.1.1",
  "karma-phantomjs-launcher": "^1.0.2",
  "mocha": "^3.0.2"
}
3.2.2 初始化Karma配置文件

在项目的根目录运行

karma init

初始化Karma配置文件:


初始化Karma配置文件

生成的karma.conf.js只是最基本的karma配置,我们还需要手动修改一些地方。
在其中的frameworks一项中增加chai

frameworks: ['mocha','chai'],

然后在config.set({})中添加:

plugins : [
  'karma-mocha',
  'karma-chai',
  'karma-phantomjs-launcher'
],
3.2.3 提供需要测试的代码并编写测试代码

在上文的初始化Karma配置时,已经告诉Karma,需要被测试的代码和测试代码放在src/test/文件夹中,所以我们应该在src/文件夹下提供要被测试的代码,在test/文件夹下提供测试代码:

项目文件结构

src/文件夹中新建index.js文件,在这个文件中添加两个非常简单的函数:

function isNum (num) {
  return typeof num === 'number'
}
function isString (str) {
  return typeof str === 'string'
}

然后在test/文件夹中新建index.test.js文件。通常,测试脚本与所要测试的源码脚本同名,但是后缀名为.test.js(表示测试)或者.spec.js(表示规格)。在该文件中开始编写测试代码:

describe('index.js的测试', function () {
  it('1应该是数字', function() {
      // expect(isNum(1)).to.be.true
      isNum(1).should.equal(true)
  })
  it('"1" 应该是字符', function() {
      // expect(isString('1')).to.be.true
      isString('1').should.equal(true)
  })
})

编写测试文件时,describeit都是由mocha提供的测试用api:

describe块称为"测试套件"(test suite),表示一组相关的测试。它是一个函数,第一个参数是测试套件的名称("index.js的测试"),第二个参数是一个实际执行的函数。

it块称为"测试用例"(test case),表示一个单独的测试,是测试的最小单位。它也是一个函数,第一个参数是测试用例的名称("1应该是数字"),第二个参数是一个实际执行的函数。
——测试框架 Mocha 实例教程

如果测试不通过,测试套件和测试用例的描述都会在命令行输出,告诉你哪里测试失败了。
被测试代码和测试代码编写完毕后,在项目根目录输入:

karma start

运行成功后,测试结果便会显示在命令行中。并且这时你修改任意代码,单元测试便会在你保存后自动运行。

3.2.4 统计测试覆盖率

单元测试很多时候需要统计测试覆盖率。使用karma-coverage来统计你的单元测试覆盖率。修改karma.conf.js

preprocessors: {
  'src/*.js': ['coverage']
},
reporters: ['progress', 'coverage'],

然后再运行karma start,你的项目中便会多出一个coverage文件夹,文件夹中按照浏览器分了覆盖率统计结果,我们使用的是PhantomJs,自然会有一个PhantomJs ..*文件夹,用浏览器打开index.html便可查看测试覆盖率。

3.3 集成webpack

很多时候,项目中会用到webpack来进行打包,有了Webpack我们可以使用ES6甚至ES7语法,可以轻松打包Vue、React、Angular等主流框架,可以有Eslint代码检查。所以将Webpack集成进Karma后,我们可以使用最新的JS语法来编写测试代码,也可以对使用了主流框架的代码进行单元测试了。

这里我们以使用ES6语法为目的,来演示如何集成Webpack。

3.3.1 安装Webpack和Babel

首先安装Webpack和karma-webpack插件

npm install webpack karma-webpack --save-dev

然后安装babel

npm i --save-dev babel-loader babel-core babel-preset-es2015

3.3.2 在Karma中配置和使用Webpack

修改karma.conf.js,将webpack添加进去。

3.3.2.1 设置需要Webpack打包的文件

preprocessors中告诉karma需要Webpack打包的文件所在位置,这里我想同时在被测试代码和测试代码中使用ES6语法,那么理论上我除了将被测试代码位置告诉Webpack之外,还需要将测试代码的位置也告诉Webpack。

但如果你的代码是模块化的,使用了ES6的模块系统,那么即使你将已经模块化的index.js打包并好并注入到浏览器也是没有用的,所以正确的做法应该是在你的测试代码也就是index.test.js中引入index.js模块进行测试。然而Webpack在处理index.test.js时会查找它的引用并自动打包过来,所以如果你的被测试代码是模块化的,Karma配置中的preprocessors中就应当去掉Webpack对被测试代码的处理,同时files中也不需要让Karma将被测试代码放到浏览器了,这一切应当都交给Webpack来做:

preprocessors: {
  // 'src/*.js': ['webpack', 'coverage'],
  'src/*.js': ['coverage'],
  'test/*.js': ['webpack']
},
files: [
  // './src/*.js',
  './test/*.js'
],
3.3.2.2 配置好Webpack

在Karma中写好Webpack的配置:

// webpack config
    webpack: {
      module: {
        loaders: [{
          test: /\\.js$/,
          loader: 'babel',
          exclude: /node_modules/,
          query: {
            presets: ['es2015']
          }
        }]
      }
    },

这一步有三点需要注意:

  1. 上面这些配置,完全可以独立出来成为一个webpack.test.config.js,怎么样,是不是很眼熟?
  2. 你可能已经注意到Webpack的配置中没有entry,也没有output,因为在Karma的preprocessors中已经告诉了Webpack需要打包哪些文件了,同时Karma也会处理好打包后文件的去向(当然是注入浏览器了,还能去哪,别忘记了还有karma-webpack这个插件在起作用)
  3. 测试的Webpack配置除了上面说的入口和出口,其余的配置跟普通使用Webpack没有本质区别,所以从这里你完全可以发散思维,用Webpack去做你想做的~
3.3.2.3 添加karma-webpack插件

别忘记新版的Karma几乎所有的工具都需要插件支持,这在老版本中是不需要的。所以得把karma-webpack添加到Karma的plugins中去

plugins : [
  'karma-mocha',
  'karma-chai',
  'karma-phantomjs-launcher',
  'karma-coverage',
  'karma-webpack'
],
3.3.2.4 在你的被测试代码和测试代码中使用ES6语法

首先是被测试代码

function isNum (num) {
  return typeof num === 'number'
}
function isString (str) {
  return typeof str === 'string'
}
export default {
  isNum,
  isString
}

然后是测试代码

import Index from '../src/index'
console.log('开始测试')
describe('index.js的测试', function () {
  it('1应该是数字', function() {
      // expect(isNum(1)).to.be.true
      Index.isNum(1).should.equal(true)
  })
  it('"1" 应该是字符', function() {
      // expect(isString('1')).to.be.true
      Index.isString('1').should.equal(true)
  })
})

我这里使用了ES6 中的模块写法,在index.js中输出了一个带有两个方法的模块,这时测试代码中就需要引入这个模块了,因为仅仅是简单地将index.js输出到浏览器是不会起任何作用的(webpack打包后,两个需要测试的函数已经是私有变量了,前文也有所提及)。

这时再运行karma start,便能看到测试通过的结果,说明我们成功使用了webpack编译了ES6。现在检查一下代码覆盖率:

异常的代码覆盖率

会发现代码覆盖率无法正常检测了。即使你注释掉某个函数的测试用例,代码覆盖率仍旧是100%。这就是前文提到的,如果使用karma-coverage检测Webpack打包后的代码,就会出现这种情况。所以这里我们需要使用其它办法来检测代码覆盖率。

一般代码覆盖率的检测是需要统计被测试代码中需要测试的量,比如函数、行数等信息,然而打包后的代码因为被混入了很多别的代码,或者是变量被私有化了,这些统计就会出问题。所以最好的办法是在打包之前进行统计。

方案其实有很多,比如isparta、isparta-instrumenter-loader、istanbul。这里选择istanbul,因为karma-coverage用的就是它。同时,babel提供了一个插件babel-plugin-istanbul,能够在babel编译之前instrument你的ES6代码,可以像下面这样使用(参考babel-plugin-istanbul):
首先安装babel-plugin-istanbul

npm install babel-plugin-istanbul --save-dev

然后将其放入到babel的插件选项中:

loaders: [{
  test: /\\.js$/,
  loader: 'babel',
  exclude: /node_modules/,
  query: {
    presets: ['es2015'],
    plugins: ['istanbul']
  }
}]

这里需要注意的是:

Note: This plugin does not generate any report or save any data to any file;it only adds instrumenting code to your JavaScript source code.To integrate with testing tools, please see the Integrations section.
—— cnpm babel-plugin-istanbul

这个插件的功能仅仅是instrument,不生成报告,所以报告的生成还是需要karma-coverage来完成的,所以之前有关karma-coverage的设置只需要将instrument部分也就是karma.conf.js中的preprocessors中的coverage去掉即可:

preprocessors: {
  // 'src/*.js': ['webpack', 'coverage'],
  // 'src/*.js': ['coverage'],
  'test/*.js': ['webpack']
},

这时再次运行karma start,便能看到测试通过的结果,说明我们成功使用了webpack编译了ES6。现在检查一下代码覆盖率:

正常的代码覆盖率

已经恢复正常了!按照上面的思路,我们完成了将Webpack配置到Karma中的工作,所以现在你可以使用Webpack来统一管理你的被测试代码和测试代码了。

4. 其它注意事项

4.1 关于npm

正常情况下国内是需要翻墙才能使用npm的,但你有两个选择:翻墙或者使用cnpm。我建议使用cnpm,安装使用可以去cnpm官网查看详细教程(非常简单)

4.2 关于测试框架mocha和断言库chai.js

一个咖啡一个茶,虽然你已经能够将它们运用到你的构建体系中去了,但这两者的详细API还是需要去熟悉和了解的,否则也没办法写出高质量的测试代码
mocha除了可以去mocha官方网站看英文文档之外,还可以参考我翻译的中文文档:Mocha.js官方文档翻译 —— 简单、灵活、有趣
chai.js则只需要去chai.js官方网站看API文档,我也翻译了TDD部分的API文档:Chai.js断言库API中文文档

4.3 关于BDD与TDD

BDD是行为驱动开发,TDD是测试驱动开发。但其实可以认为BDD是TDD的一个子集或分支,是测试驱动开发的升级版。具体可以参考这几篇文章:


  1. Mocha既是测试工具,也是测试框架,其实有不少测试工具既是管理工具,也是测试框架

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352

推荐阅读更多精彩内容