Spock单测利器,用了都说好

简介: 参考Spock单元测试框架介绍以及在美团优选的实践最近发现了一种写法简洁高效,一个单测方法可以测试多组测试数据,且测试结果一目了然的单测框架Spock。Spock国外的测试框架,其设计灵感来自JUnit、Mockito、Groovy,可以用于Java和Groovy应用的测试。尽管Spock写单测,需要使用groovy语言,但是groovy语言是一种弱类型,写法超级简单,我也是零基础的groovy新

参考

Spock单元测试框架介绍以及在美团优选的实践

最近发现了一种写法简洁高效,一个单测方法可以测试多组测试数据,且测试结果一目了然的单测框架Spock。Spock国外的测试框架,其设计灵感来自JUnit、Mockito、Groovy,可以用于Java和Groovy应用的测试。尽管Spock写单测,需要使用groovy语言,但是groovy语言是一种弱类型,写法超级简单,我也是零基础的groovy新手,相信你看过这篇文档,就会用groovy写单测啦。

环境配置

引入jar包

 			  <!--groovy单测框架-->
        <dependency>
            <groupId>org.codehaus.groovy</groupId>
            <artifactId>groovy-all-tests</artifactId>
            <version>2.0.0-rc-3</version>
        </dependency>
        <!-- Mandatory dependencies for using Spock test framework -->
        <dependency>
            <groupId>org.spockframework</groupId>
            <artifactId>spock-core</artifactId>
            <version>1.3-groovy-2.4</version>
        </dependency>
        <dependency>
            <groupId>org.spockframework</groupId>
            <artifactId>spock-spring</artifactId>
            <version>1.3-groovy-2.4</version>
            <scope>test</scope>
        </dependency>

配置插件

						<plugin>
                <!--groovy plugin-->
                <groupId>org.codehaus.gmavenplus</groupId>
                <artifactId>gmavenplus-plugin</artifactId>
                <version>1.4</version>
                <extensions>true</extensions>
                <executions>
                    <execution>
                        <goals>
                            <goal>compile</goal>
                            <goal>testCompile</goal>
                        </goals>
                    </execution>
                </executions>
                <configuration>
                		<!-- spock单测文件路径 -->
                    <testSources>
                        <testSource>
                            <directory>${project.basedir}/src/test/java</directory>
                            <includes>
                                <include>**/*.groovy</include>
                            </includes>
                        </testSource>
                        <testSource>
                            <directory>${project.basedir}/src/test/groovy</directory>
                            <includes>
                                <include>**/*.groovy</include>
                            </includes>
                        </testSource>
                    </testSources>
                </configuration>
            </plugin>

Spock用法

0、被测试类

public class TaskService {
    /**
     * 环境
     */
    @Value("${spring.current.env}")
    private String env;

    /**
     * 任务 服务类
     */
    @Resource
    private ITaskRepository taskRepository;

    /**
     * 智能配置 管理类
     */
    @Resource
    private IntelligentConfigManager configManager;

    /**
     * 查询任务信息(根据环境,查询任务)
     *
     * @return 任务
     */
    public Result<Task> getTask() {
        if (EnvEnum.isDaily(env)) {
            // 日常环境,任务取值 本方法直接new
            Task task = new Task();
            task.setInput(EnvEnum.DAILY.name() + " 任务");
            return Result.isOk(task);
        }
        if (EnvEnum.isPre(env)) {
            // 预发环境,任务取值于 本类的方法的返回值
            return getPreTask();
        }
        try {
            // 线上环境,任务取值于 另一个类的方法的返回值
            return Result.isOk(taskRepository.getTask(1L));
        } catch (Exception ex) {
            // 异常
            return Result.onError("异常任务");
        }
    }

    /**
     * 查询智能配置信息
     *
     * @param query 查询智能配置信息的query
     * @return 智能配置信息集合
     */
    public Result<List<IntelligentConfigDTO>> getAllIntelligentConfigDTOList(IntelligentConfigQuery query) {
        List<IntelligentConfigDTO> allConfigDTOList = Lists.newArrayList();

        // 查询全部的智能配置信息
        query.setPage(1);
        PageResult<IntelligentConfigDTO> intelligentConfigDTOPageResult = configManager.queryList(query);
        while (intelligentConfigDTOPageResult.isSuccessful()
            && !CollectionUtils.isEmpty(intelligentConfigDTOPageResult.getList())) {
            allConfigDTOList.addAll(Lists.newArrayList(intelligentConfigDTOPageResult.getList()));

            query.setPage(query.getPage() + 1);
            intelligentConfigDTOPageResult = configManager.queryList(query);
        }
        return Result.isOk(allConfigDTOList);
    }

    /**
     * 查询预发环境的任务
     *
     * @return 任务
     */
    public Result<Task> getPreTask() {
        Task task = new Task();
        task.setInput("TaskService getInternalTask 任务" + EnvEnum.PRE.name());
        return Result.isOk(task);
    }
}

1、given-expect-where

given块:用于写测试前的准备工作,例如mock方法的返回值等

expect块:只能写判断式,如a==b

where块:用于写断言(测试数据、及期望返回值),where块可以写多组测试数据、和期望返回值

@Unroll:通过“#”可以动态获取where块的数据,测试结果一目了然,看例子,更容易理解

import com.google.common.collect.Lists
import spock.lang.Specification
import spock.lang.Unroll

class TaskServiceSpockTest extends Specification {

    // mock TaskService的属性
    ITaskRepository taskRepository = Mock()
    IntelligentConfigManager configManager = Mock()

    // 要测试的类
    TaskService taskService = new TaskService(taskRepository: taskRepository, configManager: configManager)

    void setup() {
        // 也可以在setup中,给TaskService的属性赋值
        // taskTestService.taskRepository = taskRepository
        // taskTestService.configManager = configManager
    }

    @Unroll
    def "testGetTask 环境=#env, 任务包含关键字=#keyWord, 任务是否包含关键字=#result"() {
        given: "测试前的准备:给taskService的env赋值"
        taskService.env = env

        and: "mock taskRepository.getTask(_) 的返回值"
        Task task = new Task();
        task.setInput(EnvEnum.PRODUCT.name())
        taskRepository.getTask(_) >> task

        and: "执行taskService.getTask()"
        Result<Task> taskResult = taskService.getTask()
        println(taskResult)

        expect: "expect只能写判断式,断言测试结果"
        result == taskResult.getData().getInput().contains(keyWord)

        where: "测试数据、及测试结果"
        env                      | keyWord                | result
        EnvEnum.DAILY.getVal()   | EnvEnum.DAILY.name()   | true
        EnvEnum.PRE.getVal()     | EnvEnum.PRE.name()     | true
        EnvEnum.PRODUCT.getVal() | EnvEnum.PRODUCT.name() | true
    }
}

执行结果

看到这里,你是不是很兴奋,一个单测就能覆盖到被测方法的各个逻辑,且测试结果一目了然。

2、given-when-then

除了given-expect-where,还可以使用given-when-then。

given块:用于写测试前的准备工作,例如mock方法的返回值等

when块:执行被测试方法

then块:用于写断言,如==、不会抛出异常noExceptionThrown()、方法被调用的次数0 * taskRepository.getTask(_)

    def "testGetTaskWhen"() {
        given: "测试前的准备: mock taskRepository.getTask(_)的返回值"
        Task task = new Task();
        task.setInput(EnvEnum.PRODUCT.name())
        taskRepository.getTask(_) >> task

        and: "给taskService的env赋值"
        taskService.env = EnvEnum.PRODUCT.getVal()

        when: "执行被测试方法"
        Result<Task> result = taskService.getTask()
        println(result)

        then: "断言"
        // 断言:返回结果是true
        result.isSuccessful() == true
        // 断言:不会抛出异常
        noExceptionThrown()
    }

3、mock异常

com.alibaba.polystar.service.intelligent.service.TaskService#getTask有try-catch,那么怎么覆盖掉catch的逻辑呢?下面讲下,如何mock异常。


    @Unroll
    def "testGetTaskException 环境=#env, 结果=#result"() {
        given: "测试前的准备:给taskService的env赋值"
        taskService.env = env

        and: "mock taskRepository.getTask(_) 抛出异常"
        taskRepository.getTask(_) >> { throw new Exception() }

        and: "执行被测试方法"
        Result<Task> result1 = taskService.getTask()
        println(result1)

        expect: "expect只能是判断式:断言 测试结果"
        result == result1.isSuccessful()

        // 测试数据、测试结果断言
        where:
        env                      | result
        EnvEnum.DAILY.getVal()   | true
        EnvEnum.PRE.getVal()     | true
        EnvEnum.PRODUCT.getVal() | false
    }

通过given-when-then测试抛出异常,不能像where能测试多组测试数据。

    def "testGetTaskWhen 异常"() {
        given: "测试前的准备: mock taskRepository.getTask(_)抛出运行时异常"
        taskRepository.getTask(_) >> { throw new RuntimeException() }

        and: "给taskService的env赋值"
        taskService.env = EnvEnum.PRODUCT.getVal()

        when: "执行被测试方法"
        Result<Task> result1 = taskService.getTask()
        println(result1)

        then: "断言测试结果"
        result1.isSuccessful() == false
    }

4、mock方法每次的返回值不一样

在日常开发中,可能会遇到while查询某个方法,直到某种条件,才会break,如TaskService.getAllIntelligentConfigDTOList。为了测试这样的逻辑,就需要使每次mock方法的返回值不同。

 def "testGetAllIntelligentConfigDTOList"() {

        given: "测试前的准备"

        // 第一次调,返回 长度=1的集合
        IntelligentConfigDTO configDTO = new IntelligentConfigDTO();
        configDTO.setId(1L)
        com.alibaba.polystar.common.PageResult<IntelligentConfigDTO> pageResult =
                PageResult.build(1, 1, 1, Lists.newArrayList(configDTO))

        // 第二次调,返回 空集合,使while循环结束
        com.alibaba.polystar.common.PageResult<IntelligentConfigDTO> pageResult2 =
                PageResult.build(2, 1, 0, Lists.newArrayList())

        // 模拟方法调多次时,返回的结果
        configManager.queryList(_) >> pageResult >> pageResult2

        // 执行被测试方法
        IntelligentConfigQuery query = new IntelligentConfigQuery();
        Result<List<IntelligentConfigDTO>> result = taskService.getAllIntelligentConfigDTOList(query)
        println(result)
        // 智能配置的总条数
        def size = result.getData().size()

        expect: "expect只能是判断式:断言 测试结果,断言智能配置size=1"
        size == 1
    }

5、mock本类方法

在日常开发中,被测试方法A调用了同类的方法B,而B方法逻辑复杂,如getPreTask()方法,会调用本类的getInternalTask(),这时可以通过spy来mock本类方法getInternalTask(),来编写getPreTask()方法的单测。TaskService taskService = Spy()的作用是,如果TaskService的方法没有mock的话,则会执行方法;如果TaskService的方法被mock的话,则不会执行方法。


    /**
     * 查询预发环境的任务
     *
     * @return 任务
     */
    public Result<Task> getPreTask() {
        return Result.isOk(getInternalTask());
    }

    /**
     * 查询 内部 任务
     *
     * @return 任务
     */
    public Task getInternalTask() {
        Task task = new Task();
        task.setInput("TaskService getInternalTask 任务" + EnvEnum.PRE.name());
        return task;
    }

单测


    def "testGetPreTask"() {
        given: "测试前的准备"
        // 通过spy创建TaskService,TaskService的方法如果没有mock的话,则会执行方法;如果TaskService的方法被mock的话,则不会执行方法
        TaskService taskService = Spy();

        and: "mock 本类的的方法"
        Task task = new Task();
        task.setInput("spy getInternalTask 任务");
        taskService.getInternalTask() >> task

        and: "执行被测试方法"
        Result<Task> result = taskService.getPreTask()
        println(result)

        expect: "expect只能是判断式:断言测试结果"
        result.getData().getInput().contains("spy") == true
    }

最后

看到这里,是不是你也觉得spock语法非常简洁、功能非常强大,那就快快使用起来吧。

相关文章
|
3月前
|
设计模式 关系型数据库 测试技术
进阶技巧:提高单元测试覆盖率与代码质量
【10月更文挑战第14天】随着软件复杂性的不断增加,确保代码质量的重要性日益凸显。单元测试作为软件开发过程中的一个重要环节,对于提高代码质量、减少bug以及加快开发速度都有着不可替代的作用。本文将探讨如何优化单元测试以达到更高的测试覆盖率,并确保代码质量。我们将从编写有效的测试用例策略入手,讨论如何避免常见的测试陷阱,使用mocking工具模拟依赖项,以及如何重构难以测试的代码。
93 4
|
5月前
|
监控 jenkins 测试技术
自动化测试中的“守护神”: 持续集成与代码质量监控
【8月更文挑战第31天】在软件开发的海洋里,自动化测试犹如一座灯塔,指引着项目向着高质量和高效率的方向前进。本文将深入探讨如何通过持续集成(CI)和代码质量监控相结合的方式,构建起一道坚固的防线,保障软件项目在快速迭代中不失方向。我们将一起探索这一过程中的关键实践,以及它们是如何相互作用,共同提升软件项目的可靠性和稳定性。
|
6月前
|
测试技术
单元测试问题之过去的软件测试如何解决
单元测试问题之过去的软件测试如何解决
|
5月前
|
测试技术 开发者
单元测试问题之单元测试想提高协同效率与质量,如何实现
单元测试问题之单元测试想提高协同效率与质量,如何实现
|
8月前
|
Java 测试技术 开发者
深入理解与应用单元测试:软件质量的守护者
【4月更文挑战第30天】 在现代软件开发过程中,单元测试作为保障代码健康的重要环节,其地位日益凸显。本文将探讨单元测试的核心概念、实施单元测试的重要性以及如何高效地设计并执行单元测试。通过实例分析,我们将揭示单元测试在确保软件产品质量和加速开发周期中的关键作用。
|
Java 测试技术 数据安全/隐私保护
软件测试小白如何实施单元测试?
软件测试小白如何实施单元测试?
129 0
|
人工智能 自然语言处理 Java
提升函数代码质量的利器有哪些?
全栈式全自动软件开发工具SoFlu软件机器人结合当下AI技术今年重磅上线函数AI生成器——FuncGPT(慧函数)。FuncGPT(慧函数)采用代码编写最佳实践及大规模机器联合训练的方式,可以显著提高代码的质量、可维护性、健壮性,为中国软件开发者提供全栈式全流程软件开发的最佳体验。
|
存储 Java 测试技术
【C#编程最佳实践 一】单元测试实践
【C#编程最佳实践 一】单元测试实践
125 0
|
自然语言处理 Java 测试技术
告别祈祷式编程|单元测试在项目里的正确落地姿势
单元测试(unit testing),是指对软件中的最小可测试单元进行检查和验证。对于单元测试中单元的含义,一般来说,要根据实际情况去判定其具体含义,如C语言中单元指一个函数,Jav...
183 1
|
XML Java 测试技术
告别加班/解放双手提高单测覆盖率之Java 自动生成单测代码神器推荐
很多公司对分支单测覆盖率会有一定的要求,比如 单测覆盖率要达到 60% 或者 80%才可以发布。 有时候工期相对紧张,就优先开发功能,测试功能,然后再去补单元测试。 ![在这里插入图片描述](https://img-blog.csdnimg.cn/e9e8ea7d35ca4830bce7929774471207.jpg) 但是编写单元测试又比较浪费时间,有没有能够很大程度上自动化生成单元测试的插件,自己简单改改即可呢? 自己尝试在 Idea 插件库里搜索相关插件并去尝试使用,发现 `TestMe` 还可以。后面和其他同学交流,谎伴 同学推荐他一直在用的 `Squaretest`,我试用
7109 1
告别加班/解放双手提高单测覆盖率之Java 自动生成单测代码神器推荐