本文档讲解了我们是如何使用JSCover来获得Selenium的测试样例的js代码文件的执行覆盖率的。

事实上网上有挺多博客讲这玩意儿了,不过完全按照网上已有的教程去弄的的话,并无法满足我们的需要。

参考链接:

  1. https://www.cnblogs.com/bhlsheji/p/4
  2. https://stackoverflow.com/questions/9495625/selenium-is-there-any-js-javascript-code-coverage-tool-which-we-can-integrate
  3. http://tntim96.github.io/JSCover/manual/manual.xml

工具介绍

Selenium

浏览器自动化测试工具。可以模拟用户对浏览器的操作,从而起到自动化测试的效果。具体怎么用,网上一堆教程,不赘述了。

selenium会在测试样例的编写中被使用,我们还会使用selenium的executeScript调用JSCover提供的hook来保存覆盖率数据。

JSCover

js代码覆盖率检测工具。原理是插桩js代码,该项目在github上开源,它也有托管在github上的官网。附带一提,根据我的使用经验来看,有问题不要查百度、也不要查它的user manual,而是应该查github上的issue。

JSCover我们只使用它的JSCover-all.jar文件,下下来以后就拿出来这个就行了。

原理性流程

  1. 使用JSCover对js代码进行插桩,使得我们能够获得代码覆盖率。
  2. 对插桩后的代码执行测试样例。
  3. 将各测试样例的代码覆盖率导出保存起来。
  4. 合并各测试样例的代码覆盖率
  5. 使用合并后的代码覆盖率文件生成测试报告

流程示意图

实际流程的分步讲解

0. 预准备jscoverage.js

这一步是因为jscover自身存在的问题,原因之后再讲解。总之先得在不启用local-storage的情况下对代码进行一次插桩,得到jscoverage.js

示例指令:java -Dfile.encoding=UTF-8 -jar JSCover-all.jar -fs D:/code_origin D:/code_instrumented

  • -Dfile.encoding=UTF-8:设定编码,避免网页代码中的中文在插桩后变成乱码。
  • -jar JSCover-all.jar:指定JSCover-all.jar的所在路径。
  • -fs:指定使用文件插桩模式。我们只使用该模式,除此之外还有proxy模式,不过我们不使用,原因见最下。
  • D:/code_originD:/code_instrumented:前者为原来的js代码文件所在的目录,后者为想把插桩后的代码保存到的目录。

执行完上述指令后,应该就能在插桩后的代码的所在目录(也就是示例中的D:/code_instrumented)中找到jscoverage.js文件,将它保存到一个合适的地方,留待之后使用。

1. 代码插桩

代码插桩分为两部分,一部分是给网页的js代码文件插桩,使得执行时能够得到覆盖率变量。另一部分是在测试样例的执行末尾利用selenium提供的executeScript调用JSCover提供的hook,这本质就是个动态插桩的过程,它使得我们能够保存下来覆盖率数据。

网页的js代码文件插桩

示例指令:java -Dfile.encoding=UTF-8 -jar JSCover-all.jar -fs --local-storage D:/code_origin D:/code_instrumented

事实上,它和上一步“预准备”中的指令的差别仅在于--local-storage而已。local-storage的含义是“是否启用HTML5的local-storage功能来保存代码覆盖率变量”。如果不启用的话,JSCover就会用一个js变量来保存代码覆盖率变量,而众所周知,js变量是无法跨页面的,这也就导致每当我们切换页面时,代码覆盖率变量就会丢失。所以我们需要启用local-storage来保存代码覆盖率变量。

不过JSCover在local-storage模式下生成的jscoverage.js文件存在bug,所以我们在上一步“预准备”时需要先在非local-storage下生成一个jscoverage.js文件。这个文件仅仅是用来显示测试报告用的,不会影响到代码覆盖率数据。

hook调用

这一部分可以参考参考链接3(官方手册)中的做法。官网代码的python版:

json_str = driver.execute_script("return jscoverage_serializeCoverageToJSON();")

之后将json_str保存成jscoverage.json,然后放到插桩后的代码所在目录下,覆盖掉原来的jscoverage.json文件就行了。

不过因为官网这种做法是一个类一个jscoverage.json,一个类在selenium中就表示一类测试样例,而我们有好多好多类测试样例,所以就会有好多好多代码覆盖率文件jscoverage.json。所以我们需要在执行时分别保存它们,在全部执行完后再合并它们。合并会在后面再讲解。

假设我们有三个Selenium的测试类,分别将叫TC_A, TC_B, TC_C。那我们可以生成类似以下的一个目录结构

- coverage
- TC_A
- jscoverage.json
- TC_B
- jscoverage.json
- TC_C
- jscoverage.json

这仅仅只需要在保存json_str时操作一下就行了,不再赘述。

2. 执行测试样例

这一部分和平时一模一样,仅仅需要注意要执行插桩后的网页代码文件,而不是插桩前的。

3. 合并代码覆盖率文件

在执行测试样例后,一切正常的话,就能得到类似在第一步中描述的那样的代码覆盖率文件的目录结构了:

- coverage
- TC_A
- jscoverage.json
- TC_B
- jscoverage.json
- TC_C
- jscoverage.json

示例指令: java -cp JSCover-all.jar jscover.report.Main --merge coverage/* D:/code_instrumented

  • coverage/*:也就是上述的目录结构的顶层目录的路径。
  • D:/code_instrumented:合并后的代码覆盖率文件的保存目录。我们直接指定为插桩后的代码的所在目录,省去手动覆盖的麻烦。

注意,如果只有一个jscoverage.json的话,是无法使用上述指令进行合并的,毕竟只有一个,JSCover会报错。这时候请手动复制那唯一一个jscoverage.json,覆盖掉原来的jscoverage.json

4. 生成测试报告

把我们在“预准备”时搞到的jscoverage.js翻出来,然后覆盖掉插桩后的代码所在目录下的jscoverage.js,再修改一下它,将jscoverage_isReport改为true:

var jscoverage_isReport = true;

最后打开jscoverage.html即可。可能会啥都没,这时候你打开浏览器的控制台看看报啥错吧,例如chrome的话,通常而言都是Cross origin requests are only supported for protocol schemes: http, data, chrome, chrome-extension, https.,然后百度百度解决一下,或者换个浏览器再试试,或者自己改改jscoverage.html,都行。我是最后直接用Edge打开了。

最后应该是这样子的:

点开一个文件的话,就能看见哪些行被覆盖了,哪些没有:

不使用JSCover的Proxy模式的原因

Proxy模式要方便很多,可以省去插桩网页代码的步骤。因为proxy模式下,jscover会截获所有浏览器的发包和收包,对于所有发过来的js文件,它都会自动进行插桩。

不过因为JSCover的Proxy模式需要修改浏览器代理。而我们目前的本地测试环境使用了Fiddler作为浏览器代理,这两个代理冲突了,也就只能二选一,所以我们没有使用JSCover的Proxy模式,不过要是你的项目不使用浏览器代理来搭建本地测试环境,可以考虑用用JSCover的proxy模式,真的方便很多。

[技术博客]JSCover+selenium获得js代码覆盖率的更多相关文章

  1. [转]有哪些值得关注的技术博客(Java篇)

    有哪些值得关注的技术博客(Java篇)   大部分程序员在自学的道路上不知道走了多少坑,这个视频那个网站搞得自己晕头转向.对我个人来说我平常在学习的过程中喜欢看一些教程式的博客.这些博客的特点: 1. ...

  2. 创建GitHub技术博客

    创建GitHub技术博客全攻略 githubio技术博客网站生成 说明: 首先,你需要注册一个 github 账号,最好取一个有意义的名字,比如姓名全拼,昵称全拼,如果被占用,可以加上有意义的数字.本 ...

  3. 最值得收藏的java技术博客(Java篇)

    第一个:java_my_life 作者介绍:找不到原作者信息.大概做了翻阅全部是2012年的博客. 博客主要内容:主要内容是关于Java设计模式的一些讲解和学习笔记,在相信对学习设计模式的同学帮助很大 ...

  4. Atitit.如何文章写好 论文 文章 如何写好论文 技术博客 v4

    Atitit.如何文章写好 论文  文章  如何写好论文 技术博客 1. 原则 2 1.1. 有深度, 有广度 2 1.2. 业务通用性有通用性 尽可能向上抽象一俩层..业务通用性与语言通用性. 2 ...

  5. Atitit.如何文章写好 论文 文章 如何写好论文 技术博客

    Atitit.如何文章写好 论文  文章  如何写好论文 技术博客 1. 原则 1 1.1. 有深度, 有广度 1 1.2. 业务通用性有通用性 尽可能向上抽象一俩层..业务通用性与语言通用性. 2 ...

  6. 创建GitHub技术博客全攻略【转】

    本文转载自:http://blog.csdn.net/renfufei/article/details/37725057/ 说明: 首先,你需要注册一个 github 账号,最好取一个有意义的名字,比 ...

  7. 技术人如何利用 github+Jekyll ,搭建一个独立免费的技术博客

    上次有人留言说,技术博客是程序员的标配,但据我所知绝大部分技术同学到现在仍然没有自己的技术博客.原因有很多,有的是懒的写,有的是怕写不好,还有的是一直想憋个大招,幻想做到完美再发出来,结果一直胎死腹中 ...

  8. [技术博客]海报图片生成——小程序canvas画布

    目录 背景介绍 canvas简介 代码实现 难点讲解 圆角矩形裁剪失败之PS的妙用 编码不要过硬 对过长的文字进行截取 真机首次生成时字体不对 drawImage只能使用本地图片 背景介绍 目标:利用 ...

  9. 【技术博客】基于JsPlumb和JQuery-UI的流程图的保存和再生成

    开发组在开发过程中,都不可避免地遇到了一些困难或问题,但都最终想出办法克服了.我们认为这样的经验是有必要记录下来的,因此就有了[技术博客]. 基于JsPlumb和JQuery-UI的流程图的保存和再生 ...

随机推荐

  1. JS 数组克隆方法总结(不可更改原数组)

    ES5 方法总结 1.slice let arr = [2,4,434,43]; let arr1= arr.slice();//let arr1 = arr.slice(0); arr[0] = ' ...

  2. windows ssh远程登录阿里云遇到permissions are too open的错误

    我试图用ssh -i 命令远程登录阿里云时,遇到如下错误: Permissions for 'private-key.ppk' are too open. It is required that yo ...

  3. Solr基础理论【倒排索引,模糊查询】

    一.简介 现有的许多不同类型 的技术系统,如关系型数据库.键值存储.操作磁盘文件的map-reduce[映射-规约]引擎.图数据库等,都是为了帮助用户解决颇具挑战性的数据存储与检索问题而设计的.而搜索 ...

  4. golang测试

    简述 Go语言中自带有一个轻量级的测试框架testing和自带的go test命令来实现单元测试和性能测试. go test [-c] [-i] [build flags] [packages] [f ...

  5. MySQL 自带的4个系统数据库的说明

    自带的4个系统数据库:information_schema.mysql.performance_schema.sys: information_schema:这个数据库保存了mysql服务器所有数据库 ...

  6. nfs实现k8s持久化

    1. 部署nfs服务端 k8s-master 节点上搭建了 NFS 服务器 (1)安装nfs服务: yum install -y nfs-utils rpcbind vim /etc/exports ...

  7. 洛谷 P1162 填涂颜色题解

    题目描述 由数字00组成的方阵中,有一任意形状闭合圈,闭合圈由数字11构成,围圈时只走上下左右44个方向.现要求把闭合圈内的所有空间都填写成22.例如:6 \times 66×6的方阵(n=6n=6) ...

  8. Vue基本用法

    在学习Vue的基本用法之前,我们先简单的了解一些es6的语法 let: 特点:1.局部作用域 2.不会存在变量提升 3.变量不能重复声明 const: 特点:1.局部作用域 2.不会存在变量提升 3. ...

  9. django小知识(1)

    聚合查询 关键字:aggregate from django.db.models import Max,Min,Count,Sum,Avg 分组查询 关键字:annotate 1.最简单的规律 mod ...

  10. Java16周作业

    题目1:编写一个应用程序,利用Java多线程机制,实现时间的同步输出显示. /*使用Runnable接口使用类创建线程对象,重写run()方法**/ 代码 public class timetext ...