参考网上的教程,用jacoco统计代码覆盖率,需要用到jacoco和ant,我原来的计划也是jacoco和ant的方案,后来研究发现其实不用ant也可以实现,省去了ant的安装和配置,更加灵活方便。
相关软件下载:
sudo wget http://search.maven.org/remotecontent?filepath=org/jacoco/jacoco/0.8.5/jacoco-0.8.5.zip -O jacoco-0.8.5.zip
Sudo wget https://mirrors.tuna.tsinghua.edu.cn/apache//ant/binaries/apache-ant-1.9.14-bin.tar.gz
关于jacoco和ant的具体安装和环境变量配置过程这里就不多说了,自行百度
jacoco统计代码覆盖率过程:
以springboot工程为例主要步骤为:
- 启动jar包时,指定jacocoagent.jar做为javaagent(output模式有file/tcpserver/tcpclient)
- dump出统计数据.exec文件
- 下载源码并编译生成class文件
- 通过.exec文件,class文件和源码文件生成测试报告
关键:在启动jar包时,启动命令指定jacocoagent.jar做为javaagent,实现对原jar包的动态插桩和收集代码执行数据
java - javaagent:${JACOCO_HOME}/lib/jacocoagent.jar=includes=*,output=tcpserver,port=6000,address=127.0.0.1 -jar demo2-0.0.1-SNAPSHOT.jar
原计划:jacoco+ant方案
服务器部署过程:
- 安装jacoco和ant到指定的服务器上
- 可选:为jacoco和ant配置环境变量
- 修改服务启动脚本,增加javaagent相关参数
- 为ant配置build.xml并放入指定位置
用jenkin收集代码覆盖率:
- dump出代码覆盖统计数据:ant dump
- 下载工程源码,用mvn命令打包:mvn compile
- 生成代码覆盖率统计报告要用到class文件,需要指定classfiles,必需
- 如果想到在报告中看到源码,需要指定sourcefiles,非必需
- 生成报告:ant report
ant需要用到的build.xml模板
<?xml version="1.0" encoding="UTF-8"?>
<project name="test" xmlns:jacoco="antlib:org.jacoco.ant" >
<!--Jacoco的安装路径-->
<property name="jacocoantPath" value="/opt/jacoco/lib/jacocoant.jar"/>
<!--最终生成.exec文件的路径,Jacoco就是根据这个文件生成最终的报告的-->
<property name="jacocoexecPath" value="./exec"/>
<!--生成覆盖率报告的路径-->
<property name="reportfolderPath" value="./report"/>
<!--JaCoCo的tcpserver的ip地址-->
<property name="server_ip" value="192.168.0.201"/>
<!--JaCoCo的tcpserver的端口,要跟启动时jacocoagent中的参数一样-->
<!--property name="server_port" value="6001"/-->
<!--源代码路径可以包含多个源代码-->
<property name="webSrcpath" value="./push-center" />
<!--.class文件路径可以包含多个-->
<property name="webClasspath" value="./push-center"/>
<!--让ant知道去哪儿找Jacoco-->
<taskdef uri="antlib:org.jacoco.ant" resource="org/jacoco/ant/antlib.xml">
<classpath path="${jacocoantPath}" />
</taskdef>
<!--dump任务: 根据前面配置的ip地址,和端口号,访问目标tomcat服务,并生成.exec文件。
可以配置多个服务和端口同时dump,适用一台服务器上跑多个服务的情况-->
<target name="dump">
<!-- jacoco:dump address="${server_ip}" reset="true" destfile="${jacocoexecPath}" port="${server_port}" append="false"/-->
<jacoco:dump address="${server_ip}" reset="true" destfile="${jacocoexecPath}/push_center_core_provider.exec" port="6001" append="false"/>
<jacoco:dump address="${server_ip}" reset="true" destfile="${jacocoexecPath}/push_center_rest_api.exec" port="6002" append="false"/>
<jacoco:dump address="${server_ip}" reset="true" destfile="${jacocoexecPath}/push-center-validation-provider.exec" port="6003" append="false"/>
</target>
<!--jacoco任务:
根据前面配置的源代码路径和.class文件路径,
根据dump后,生成的.exec文件,生成最终的html覆盖率报告。-->
<target name="report">
<delete dir="${reportfolderPath}" />
<mkdir dir="${reportfolderPath}" />
<jacoco:report>
<executiondata>
<!--fileset dir="${jacocoexecPath}" includes="*.exec" /-->
<file file="${jacocoexecPath}/push_center_core_provider.exec" />
<file file="${jacocoexecPath}/push_center_rest_api.exec" />
<file file="${jacocoexecPath}/push-center-validation-provider.exec" />
</executiondata>
<structure name="JaCoCo Report">
<group name='push_center_core_provider'>
<!--此处配置classes文件地址 -->
<classfiles>
<fileset dir="${webSrcpath}/push-center-core/push-center-core-provider/target/classes" />
</classfiles>
<!--此处配置源码地址-->
<sourcefiles encoding="utf-8">
<fileset dir="${webClasspath}/push-center-core/push-center-core-provider/src/main/java" />
</sourcefiles>
</group>
<group name='push_center_rest_api'>
<!--此处配置classes文件地址 -->
<classfiles>
<fileset dir="${webSrcpath}/push-center-rest-api/target/classes" />
</classfiles>
<!--此处配置源码地址-->
<sourcefiles encoding="utf-8">
<fileset dir="${webClasspath}/push-center-rest-api/src/main/java" />
</sourcefiles>
</group>
<group name='push-center-validation-provider'>
<!--此处配置classes文件地址 -->
<classfiles>
<fileset dir="${webClasspath}/push-center-validation/push-center-validation-provider/target/classes" />
</classfiles>
<!--此处配置源码地址-->
<sourcefiles encoding="utf-8">
<fileset dir="${webSrcpath}/push-center-validation/push-center-validation-provider/src/main/java" />
</sourcefiles>
</group>
</structure>
<html destdir="${reportfolderPath}" encoding="utf-8" />
</jacoco:report>
</target>
</project>
此方案假设所有的步骤都在服务所运行的服务器上执行,其实收集数据和生成报告可以分离
- 服务器端只需安装jacoco,收集统计信息,并使用tcpserver模式提供dump服务
- 生成报告可以在客户机上完成,客户机需安装ant,通过服务端提供的tcpserver来dump出exec,然后生成报告
为了方便运维部署操作,我打算写两个脚本
- jacoco和ant安装脚本:jacoco_ant_install.sh
- 自动生成ant的build.xml的脚本:ant_build_gen.sh
- 这个脚本需要一个build.xml模板
- 为指定的工程生成build.xml需要如下参数
- jar包名:dump数据会用到
- serverip:获取当前机器的ip:如果获取不到,默认为127.0.01
- port:固定,如6000
- Jacocoant.jar路径:固定,如/usr/local/jacoco-0.8.5/lib/jacocoant.jar
- exec路径:./exec/jar包名.exec
- report路径:./report
- Classfiles: 难点
- Sourcefiles: 难点
将下载下来的jacoco和ant软件包,以及2个安装脚本,build.xml模板组装成一个包,叫:jacoco-setup.tar,然后只要将这个包交给运维,经过简单的几步就可以完成jacoco在服务端的部署了
此方案的局限性和不足
- 严重依赖ant,需要配置build.xml
- 虽然可以用脚本生成build.xml,但写脚本也需要考虑各种情况,我司情况是,同一服务器上部署多个服务,采用output=tcpserver模式需要提供多个未使用的端口,不能用固定端口,而且有可能需要在一个build.xml中收集多个服务,每一个服务又可能包含多个classfiles和sourcefiles路径,增加了编写脚本的难度,考虑到部署环境的复杂性,脚本不可能完全兼顾,最终可能还是需要人工维护build.xml
新方案:不依赖ant
那么能不能不依赖ant?
生成报告摆脱ant
进一步调研发现生成报告还有第三种方式
- 用ant命令生成report
- jenkins的jacoco插件可以直接收集exec,class和src(推荐)
- 用jacococli.jar生成report
用jenkins的jacoco插件生成报告非常简单,这也是推荐用这种方式生成报告的原因,具体操作自行百度吧
用jacococli.jar生成report示例:
java -jar /usr/local/jacoco-0.8.5/lib/jacococli.jar report exec/*.exec --classfiles push-center/push-center-core/push-center-core-api/target/classes/ --html report
非常好,生成报告可以不依赖ant了,但是dump还是要依赖ant啊。
dump统计数据摆脱ant
继续探索,jacoco支持三种方式output=file/tcpserver/tcpclient
- 如果是output=tcpserver模式,用ant命令dump(最常用,推荐,无需重启服务)
- 如果是output=file模式,kill服务时会dump统计数据到本地文件(不依赖ant,更简单,需重启服务)
- 如果是output=tcpclient,未做研究
考虑过采用output=file模式,输出到指定文件的方式,这种方式的优点是不依赖ant,但是缺点也很明显,需要kill服务进程来收集代码覆盖率数据。而kill服务进程又需要在服务器上执行,不利于将生成报告与采集数据解耦。
想到即然可以用用jacococli.jar生成report,那么用jacococli.jar是不是也有dump的命令,查了一下果真支持:
java -jar /usr/local/jacoco-0.8.5/lib/jacococli.jar --help
java -jar /usr/local/jacoco-0.8.5/lib/jacococli.jar dump --help
好了,如此一来完全不用再依赖ant就可以愉快的玩耍了。
最终实现
启动服务:
java -javaagent:/opt/jacoco/lib/jacocoagent.jar=includes=*,output=tcpserver,port=6001,address=192.168.0.201 -jar push-center-core-provider-1.0.1-SNAPSHOT.jar
java -javaagent:/opt/jacoco/lib/jacocoagent.jar=includes=*,output=tcpserver,port=6002,address=192.168.0.201 -jar push-center-validation-provider-1.0.1-SNAPSHOT.jar
java -javaagent:/opt/jacoco/lib/jacocoagent.jar=includes=*,output=tcpserver,port=6003,address=192.168.0.201 -jar push-center-rest-api-1.0.1-SNAPSHOT.jar
dump数据:
java -jar /opt/jacoco/lib/jacococli.jar dump --address 192.168.0.201 --port 6001 --destfile exec/6001.exec --reset
java -jar /opt/jacoco/lib/jacococli.jar dump --address 192.168.0.201 --port 6002 --destfile exec/6002.exec --reset
java -jar /opt/jacoco/lib/jacococli.jar dump --address 192.168.0.201 --port 6003 --destfile exec/6003.exec --reset
生成报告:
# 编译源码,生成class文件
mvn compile -f push-center/pom.xml
# 删除以前的report
report_dir=report
if [ -d $report_dir ]; then
echo '删除之前的report:'$target_dir
rm -r $report_dir
fi
# 可以指定多个classfiles路径
classfiles=''
classfiles=$classfiles' --classfiles push-center/push-center-core/push-center-core-provider/target/classes'
classfiles=$classfiles' --classfiles push-center/push-center-validation/push-center-validation-provider/target/classes'
classfiles=$classfiles' --classfiles push-center/push-center-rest-api/target/classes'
# 可以指定多个sourcefiles路径
sourcefiles=''
sourcefiles=$sourcefiles' --sourcefiles push-center/push-center-core/push-center-core-provider/src/main/java'
sourcefiles=$sourcefiles' --sourcefiles push-center/push-center-validation/push-center-validation-provider/src/main/java'
sourcefiles=$sourcefiles' --sourcefiles push-center/push-center-rest-api/src/main/java'
# 使用jacococli.jar生成报告
java -jar /opt/jacoco/lib/jacococli.jar report exec/*.exec $classfiles $sourcefiles --encoding utf-8 --html report
另外,可以使用cli查看exec文件内容:
java -jar D:\jacococli.jar execinfo jacoco.exec
更多jacococli.jar的使用参考:https://www.eclemma.org/jacoco/trunk/doc/cli.html
来源:CSDN
作者:boweiqiang
链接:https://blog.csdn.net/boweiqiang/article/details/103530818