關於SpringBoot單元測試(cobertura生成覆蓋率報告)

demo(SpringBoot 項目)

被測試類:

import org.springframework.stereotype.Service;
@Service
public class TestService {
    public String sayHi() {
        return "hi";
    }
    public int divide(int a, int b) {
        return a / b;
    }
}

測試代碼:

import static org.junit.Assert.*;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
@RunWith(SpringRunner.class)
@SpringBootTest
public class TestServiceTest {
    @Autowired
    TestService testService;
    @Test
    public void testSayHi() {
        TestService testService = new TestService();
        String result = testService.sayHi();
        assertEquals("hi", result);
    }
    @Test
    public void testDivide() {
        TestService testService = new TestService();
        int result = testService.divide(3, 6);
        assertTrue(result > -1);
    }
}

pom.xml 配置文件

![cobertura](C:\Users\jiaflu\Desktop\cobertura.PNG)<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.1.5.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com.jiaflu</groupId>
    <artifactId>learn_springoot</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>learn_springoot</name>
    <description>Demo project for Spring Boot</description>
    <properties>
        <java.version>1.8</java.version>
        <jackson.version>2.9.8</jackson.version>
    </properties>
    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
    </dependencies>
    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <configuration>
                    <source>1.8</source>
                    <target>1.8</target>
                </configuration>
            </plugin>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-surefire-plugin</artifactId>
                <version>2.5</version>
            </plugin>
            <plugin>
                <groupId>org.codehaus.mojo</groupId>
                <artifactId>cobertura-maven-plugin</artifactId>
                <version>2.5.2</version>
                <configuration>
                    <encoding>UTF-8</encoding>
                    <formats>
                        <format>html</format>
                        <format>xml</format>
                    </formats>
                </configuration>
            </plugin>
        </plugins>
    </build>
</project>

運行mvn cobertura:cobertura 查看截圖:

覆蓋率測試報告生成(cobertura)

cobertura 原理

cobertura執行過程大致如下:

  • 使用instrument修改我們編譯後的class文件,位於 target\generated-classes。
  • 執行測試,測試數據輸出到xxx.ser中,位於 target\cobertura\cobertura.ser。
  • 使用report生成覆蓋率報告。

1.instrument

instrument:cobertura使用instrument修改我們編譯後的class文件,在代碼裡面加入cobertura的統計代碼。並生成一個.ser文件(用於覆蓋率數據的輸出)。

使用 instrument 執行的過程中,CoberturaInstrumenter 會首先調用分析監聽器分析給定的編譯好的.class,獲得touchPoint(可以認為對應於源代碼中的待覆蓋行)以及需要的其他信息。然後調用註入監聽器將信息註入到新的.class中,保存到 \target\generated-classes 目錄下。

示例:

//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
//
package com.cisco.webex.cmse.soa.soaservice.service;
import net.sourceforge.cobertura.coveragedata.HasBeenInstrumented;
import net.sourceforge.cobertura.coveragedata.TouchCollector;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.PropertySource;
import org.springframework.stereotype.Service;
@PropertySource({"classpath:application.properties"})
@Service
public class PropertyService implements HasBeenInstrumented {
    private static final Logger logger;
    @Value("${cdp.instance.url}")
    private String cdpInstanUrl;
    @Value("${soa.instance.url}")
    private String soaInstanceUrl;
    @Value("${github.api.token}")
    public String gitApiToken;
    @Value("${github.instance.url}")
    private String githubInstance;
    @Value("${github.repo.template.owner}")
    private String tplRepoOwner;
    @Value("${github.repo.consul.owner}")
    private String consulRepoOwner;
    @Value("${slm.listen.queue.name}")
    private String slmListenQueue;
    public PropertyService() {
        boolean var1 = false;
        int __cobertura__branch__number__ = true;
        TouchCollector.touch("com.cisco.webex.cmse.soa.soaservice.service.PropertyService", 12);
        super();
    }
    public String getCdpInstanUrl() {
        boolean var1 = false;
        int __cobertura__branch__number__ = true;
        TouchCollector.touch("com.cisco.webex.cmse.soa.soaservice.service.PropertyService", 33);
        return this.cdpInstanUrl;
    }
    ...
    public void setSlmListenQueue(String ()V) {
        boolean var2 = false;
        int __cobertura__branch__number__ = true;
        TouchCollector.touch("com.cisco.webex.cmse.soa.soaservice.service.PropertyService", 85);
        this.slmListenQueue = slmListenQueue;
        TouchCollector.touch("com.cisco.webex.cmse.soa.soaservice.service.PropertyService", 86);
    }
    static {
        boolean var0 = false;
        boolean var1 = true;
        TouchCollector.touch("com.cisco.webex.cmse.soa.soaservice.service.PropertyService", 13);
        logger = LoggerFactory.getLogger(PropertyService.class);
    }
}

2.執行測試

在執行測試用例時,引用 cobertura 修改過的.class,測試信息寫入到cobertura.ser檔案文件。

3.生成報告

從cobertura.ser獲取覆蓋率數據,然後結合src中提供的源代碼,生成最終的覆蓋率報告,放到瞭target\site\cobertura路徑下。若配置瞭生成 html 格式的報告,可以通過 index.html 查看覆蓋率測試報告。

SpringBoot pom.xml 配置

添加如下依賴:

       <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <configuration>
                    <source>1.8</source>
                    <target>1.8</target>
                </configuration>
            </plugin>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-surefire-plugin</artifactId>
                <version>2.5</version>
                <configuration>
                    <!-- 此參數用於解決一個坑,下面會說明 -->
                    <argLine>-noverify</argLine>
                </configuration>
            </plugin>
            <plugin>
                <groupId>org.codehaus.mojo</groupId>
                <artifactId>cobertura-maven-plugin</artifactId>
                <version>2.5.2</version>
                <configuration>
                    <formats>
                        <format>xml</format>
                        <format>html</format>
                    </formats>
                </configuration>
            </plugin>

采坑:

在使用 mvn cobertura:cobertura 命令生成測試覆蓋率報告時,出現瞭如下問題(截取部分,報錯原因如下):

Reason:

Expected stackmap frame at this location.

Bytecode:

0x0000000: 2ab4 001b 2bb9 002e 0200 c600 2f2a b400

0x0000010: 1b2b b900 2e02 00c0 0030 b600 34c6 001c

解決方法:

本人使用的是 jdk1.8,添加 jvm 參數 -noverify,可以在 pom 文件中添加配置,配置見上述 pom.xml

網上查資料 jdk1.7 添加 jvm 參數 -XX:-UseSplitVerifier,(1.8沒有 -XX:-UseSplitVerifier 這參數)

命令介紹

  • cobertura:check

根據最新的源碼標記(生成的class文件)校驗測試用例的覆蓋率,如果沒有達到要求,則執行失敗.

  • cobertura:check-integration-test

這個命令和cobertura:check功能是一樣的,區別是二者綁定的maven生命周期不一樣.cobertura:check綁定瞭test, cobertura:check-integration-test綁定瞭verify.再說的明白些,maven生命周期中有一個是test跑得單元測試,還有一個是integration-test跑的集成測試.而verify前就是integration-test.即cobertura:check-integration-test比cobertura:check涵蓋的測試用例更多.

  • cobertura:clean

這個好理解,就是清理掉目錄/target/cobertura/中得文件.目前發現裡面就一個文件cobertura.ser.

  • cobertura:cobertura

這個插件的關鍵命令.標記被編譯的文件,運行單元測試,生成測試報告.

  • cobertura:cobertura-integration-test

和cobertura:cobertura做瞭一樣的事情,區別是包含瞭集成測試用例.

  • cobertura:dump-datafile

在命令行輸出覆蓋率數據.數據依據是生成的class文件.這個命令我沒搞懂他的意義何在.在後面一個有趣的實驗我們會用這個命令來更好的理解cobertura-maven-plugin.

  • cobertura:help
  • cobertura:instrument

標記被編譯的class文件.執行這個命令會在目錄/target/generated-classes/cobertura下生成一套class文件.

maven-surefire-plugin 使用說明

Maven本身並不是一個單元測試框架,它隻是在構建執行到特定生命周期階段的時候,通過插件來執行JUnit或者TestNG的測試用例。這個插件就是maven-surefire-plugin,也可以稱為測試運行器(Test Runner),它能兼容JUnit 3、JUnit 4以及TestNG。

在默認情況下,maven-surefire-plugin的test目標會自動執行測試源碼路徑(默認為src/test/java/)下所有符合一組命名模式的測試類。這組模式為:

  • */Test.java:任何子目錄下所有命名以Test開關的Java類。
  • */Test.java:任何子目錄下所有命名以Test結尾的Java類。
  • */TestCase.java:任何子目錄下所有命名以TestCase結尾的Java類。

maven-surefire-plugin 插件應用:

1.跳過測試

跳過測試運行 mvn package -DskipTests

或者通過 pom 提供該屬性:

<plugin>  
    <groupId>org.apache.maven.plugins</groupId>  
    <artifactId>maven-surefire-plugin</artifactId>  
    <version>2.5</version>  
    <configuration>  
        <skipTests>true</skipTests>  
    </configuration>  
</plugin>

跳過測試代碼的編譯 mvn package -Dmaven.test.skip=true

或者通過 pom 提供該屬性:

<plugin>  
    <groupId>org.apache.maven.plugin</groupId>  
    <artifactId>maven-compiler-plugin</artifactId>  
    <version>2.1</version>  
    <configuration>  
        <skip>true</skip>  
    </configuration>  
</plugin>

2.動態指定要運行的測試用例

mvn test -Dtest=RandomGeneratorTest

也可以使用通配符:

mvn test -Dtest=Random*Test

或者也可以使用“,”號指定多個測試類:

mvn test -Dtest=Random*Test,AccountCaptchaServiceTest

如果沒有指定測試類,那麼會報錯並導致構建失敗:

mvn test -Dtest

這時候可以添加 -DfailIfNoTests=false 參數告訴 maven-surefire-plugin 即使沒有任何測試也不要報錯:

mvn test -Dtest -DfailIfNoTests=false

3.包含與排除測試用例

<plugin>  
    <groupId>org.apache.maven.plugins</groupId>  
    <artifactId>maven-surefire-plugin</artifactId>  
    <version>2.5</version>  
    <configuration>  
        <includes>  
            <include>**/*Tests.java</include>  
        </includes>  
        <excludes>  
            <exclude>**/*ServiceTest.java</exclude>  
            <exclude>**/TempDaoTest.java</exclude>  
        </excludes>  
    </configuration>  
</plugin>

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: