Istanbul

课堂笔记

代码覆盖率:一个程序的代码被运行程度的量度

基本准则:函数覆盖(一函数为单位)、语句覆盖(以每行代码为单位)、分支覆盖(if判断的不同分支-trueORfalse都是不同分支)、条件覆盖(会检测到条件中任一条件是否覆盖到)

Istanbuljs程序的代码覆盖率测试工具

1可检查包括语句、分支的函数覆盖

2命令行工具

3可生成html和lcov报表

istanbul

第一章 代码覆盖率简介

1.1 代码覆盖率介绍

代码覆盖(Code coverage)是用来描述在特定的测试套件(条件)下,一个程序的代码被运行程度的量度(一般指比例),具有高的 代码覆盖率(以百分比为单位)的程序,表示在测试期间执行了更多的程序源代码;这表明与具有低代码覆盖率的程序相比,该程序具有较低的可能性检测到程序的bug。

1.2 代码覆盖率基本准则

  • 函数覆盖(Function Coverage)

程序中每一个function或者子程序是否被调用到

  • 语句覆盖(Statement Coverage)

最常用也是最常见的一种覆盖方式,就是度量被测代码中每个可执行语句是否被执行到了,也称作行覆盖

  • 分支覆盖(Branch Coverag)

度量程序中每一个判定的分支是否被检测到了

  • 条件覆盖

每一个字表达是结果为true或者false是否被检测到了

参考以下的JavaScript的函数

  function test(a, b){
    var c =0;
    if(a
>
0 
&
&
 b
>
0){
      c=a;
    }
    return c;
  }

假设这个方法使我们某个项目的一个function,并且我们使用一些测试套件来测试该函数:

  • 测试过程中,该函数至少被调用一次;

  • 如调用

    test(1, 1)

    ,则满足该函数的语句覆盖,这种情况下执行函数的每一行;

  • 如调用

    test(1, 1)

    test(0, 1)

    ,那么将满足分支覆盖,在第一种情况下,将满足if条件分支,第二种情况,第一个条件

    a

    >

    0

    不被满足, 这会阻止执行

    c=a

    ;

  • 如果调用

    test(1,0)

    test(0,1)

    ,那么将满足条件覆盖,第一种情况下

    a

    >

    0

    将被满足,第二种情况下

    b

    >

    0

    将被满足

满足了条件覆盖并代表满足了分支覆盖;上面的第四种情况,满足了a和b均大于零,但是并不能满足if的true和false两种条件;

第二章 istanbul简介

Istanbul 是 JavaScript 程序的代码覆盖率检查工具;

特性

  • 可检查包括语句、分支和函数覆盖,以及反向工程的代码行覆盖

  • 命令行工具 可运行带覆盖率检查的 node 单元测试,不需要对测试运行进行协作

  • 可生成 HTML 和 LCOV 报表

  • 在浏览器和 node 0.4.x, 0.6.x, 0.8.x 上测试通过

第三章 Istanbul安装使用

3.1 Istanbul安装

Istanbul 是一个 npm 模块,基于node,使用npm安装即可:

$ npm install -g istanbul

3.2 Istanbul简明实例

通过以下实例来了解使用Istanbul来测试代码覆盖率:

有需要测试的脚本文件:test.js, 代码如下:

var a = 'istanbul';
function b(){
  console.log('b');
}
function c(i){
  if(i == 'istanbul'){
    return true;
  }else if(i == '123'){
    return false;
  }
}
c(a);

命令行使用istanbul cover就可以得到覆盖率:

$ istanbul cover test.js

终端输出如下图所示:

返回结果显示,simple.js 有9个语句(statement),执行了6个;有4个分支(branch), 执行了1个;有2个函数,调用了1个;有9行代码,执行了6行。

这条命令同时还生成了一个 coverage 子目录,其中的 coverage.json 文件包含覆盖率的原始数据,coverage/lcov-report 是可以在浏览器打开的覆盖率报告,其中有详细信息,到底哪些代码没有覆盖到。

3.3 覆盖率标准

完美的覆盖率当然是 100%,但是现实中很难达到。一般会设置一个标准来衡量覆盖率是否达标。

istanbul check-coverage 命令用来设置门槛,同时检查当前代码是否达标。

上面命令设置语句覆盖率的门槛是 90% ,结果就报错了,因为实际覆盖率只有75%。

除了百分比门槛,我们还可以设置绝对值门槛,比如只允许有一个语句没有被覆盖到。

上面命令使用负数,表示绝对值门槛。

百分比门槛和绝对值门槛,可以结合使用。

$ istanbul check-coverage --statement -5 --branch -3 --function 100

上面命令设置了3个覆盖率门槛:5个语句、3个 if 代码块、100%的函数。 注意,这三个门槛是"与"(and)的关系,只要有一个没有达标,就会报错。

3.4 与测试框架的结合

实际开发时,istanbul 总是与测试框架结合使用,下面以常用的 Mocha 框架为例。

sqrt.js 是一个计算平方根的脚本:

var My = {
   sqrt: function(x) {
     if (x 
<
 0) throw new Error("负值没有平方根");
       return Math.exp(Math.log(x)/2);
     }
 };

 module.exports = My;

它的测试脚本 test.sqrt.js 放在 test 子目录。

var chai = require('chai');
 var expect = chai.expect;
 var My = require('../sqrt.js');

 describe("sqrt", function() {

   it("4的平方根应该等于2", function() {
     expect(My.sqrt(4)).to.equal(2);
   });

   it("参数为负值时应该报错", function() {
     expect(function(){ My.sqrt(-1); }).to.throw("负值没有平方根");
   });

 });

然后,执行下面的命令得到代码覆盖率。

上面命令中,istanbul cover 命令后面跟的是 _mocha 命令,前面的下划线是不能省略的。

因为,mocha 和 _mocha 是两个不同的命令,前者会新建一个进程执行测试,而后者是在当前进程(即 istanbul 所在的进程)执行测试,只有这样, istanbul 才会捕捉到覆盖率数据。其他测试框架也是如此,必须在同一个进程执行测试。

如果要向 mocha 传入参数,可以写成下面的样子。

$ istanbul cover _mocha -- tests/test.sqrt.js -R spec

上面命令中,两根连词线后面的部分,都会被当作参数传入 Mocha 。如果不加那两根连词线,它们就会被当作 istanbul 的参数

3.5 忽略某些代码

istanbul 提供注释语法,允许某些代码不计入覆盖率。

var object = parameter || /* istanbul ignore next */ {};

上面代码是为 object 指定默认值(一个空对象)。如果由于种种原因,没有为 object 为空对象的情况写测试,可以用注释,不将这种情况计入覆盖率。注意,注释要写在"或"运算符的后面。

/* istanbul ignore if  */
 if (hardToReproduceError)) {
     return callback(hardToReproduceError);
 }

上面代码的 if 语句块,在计算覆盖率的时候会被忽略。

参考文档

Last updated