19

我们刚刚将单元测试移植到了 JUnit5。意识到这仍然是相当早的采用,对谷歌的提示很少。

最具挑战性的是为我们在 jenkins 上使用的 Junit5 测试获取 jacoco 代码覆盖率。因为这花了我将近一天的时间才弄清楚,所以我想我分享一下。不过,如果您知道更好的解决方案,我很想知道!

buildscript {

    dependencies {
       // dependency needed to run junit 5 tests
       classpath 'org.junit.platform:junit-platform-gradle-plugin:1.0.0-M2'
   }
}

// include the jacoco plugin
plugins {
    id 'jacoco'
}

dependencies {
    testCompile "org.junit.jupiter:junit-jupiter-api:5.0.0-M2"
    runtime "org.junit.jupiter:junit-jupiter-engine:5.0.0-M2"
    runtime "org.junit.vintage:junit-vintage-engine:4.12.0-M2"
}

apply plugin: 'org.junit.platform.gradle.plugin'

那么问题似乎是 org.junit.platform.gradle.plugin 中定义的 junitPlatformTest 在 gradle 生命周期阶段定义得太晚,因此在解析脚本时是未知的。

为了仍然能够定义一个观察 junitPlatformTest 任务的 jacoco 任务,需要以下 hack。

tasks.whenTaskAdded { task ->
    if (task.name.equals('junitPlatformTest')) {
        System.out.println("ADDING TASK " + task.getName() + " to the project!")

    // configure jacoco to analyze the junitPlatformTest task
    jacoco {
        // this tool version is compatible with
        toolVersion = "0.7.6.201602180812"
        applyTo task
    }

    // create junit platform jacoco task
    project.task(type: JacocoReport, "junitPlatformJacocoReport",
            {
                sourceDirectories = files("./src/main")
                classDirectories = files("$buildDir/classes/main")
                executionData task
            })
    }
}

最后需要配置 junitPlatform 插件。以下代码允许命令行配置应运行哪些 junit 5 标签:您可以通过运行以下命令运行所有带有“unit”标签的测试:

gradle clean junitPlatformTest -PincludeTags=unit

您可以运行所有缺少 unit 和 integ 标签的测试

gradle clean junitPlatformTest -PexcludeTags=unit,integ

如果未提供任何标签,则将运行所有测试(默认)。

junitPlatform {

    engines {
        include 'junit-jupiter'
        include 'junit-vintage'
    }

    reportsDir = file("$buildDir/test-results")

    tags {
        if (project.hasProperty('includeTags')) {
            for (String t : includeTags.split(',')) {
                include t
            }
        }

        if (project.hasProperty('excludeTags')) {
            for (String t : excludeTags.split(',')) {
                exclude t
            }
        }
    }

    enableStandardTestTask false
}
4

4 回答 4

5

谢谢,所以黑客现在看起来像这样:

project.afterEvaluate {
    def junitPlatformTestTask = project.tasks.getByName('junitPlatformTest')

    // configure jacoco to analyze the junitPlatformTest task
    jacoco {
        // this tool version is compatible with
        toolVersion = "0.7.6.201602180812"
        applyTo junitPlatformTestTask
    }

    // create junit platform jacoco task
    project.task(type: JacocoReport, "junitPlatformJacocoReport",
            {
                sourceDirectories = files("./src/main")
                classDirectories = files("$buildDir/classes/main")
                executionData junitPlatformTestTask
            })
}
于 2016-09-08T08:58:12.997 回答
4

也可以通过直接注入药剂来解决:

subprojects {
   apply plugin: 'jacoco'

   jacoco {
        toolVersion = "0.7.9"
   }

   configurations {
        testAgent {
            transitive = false
        }
   }

   dependencies {
        testAgent("org.jacoco:org.jacoco.agent:0.7.9:runtime")
   }

   tasks.withType(JavaExec) {
        if (it.name == 'junitPlatformTest') {
            doFirst {
                jvmArgs "-javaagent:${configurations.testAgent.singleFile}=destfile=${project.buildDir.name}/jacoco/test.exec"
            }
        }
    }
}

然后报告将与jacocoTestReport任务一起可用

于 2017-07-04T13:54:33.117 回答
1

要获得对junitPlatformTest任务的引用,另一种选择是在项目中实现一个afterEvaluate块,如下所示:

afterEvaluate {
  def junitPlatformTestTask = tasks.getByName('junitPlatformTest')

  // do something with the junitPlatformTestTask
}

有关更多示例,请参阅我在GitHub 上对 JUnit 5的评论。

于 2016-09-07T12:22:34.970 回答
-3

您只需将 @RunWith(JUnitPlatform.class) 添加到您的包中

@RunWith(JUnitPlatform.class)
public class ClassTest {
}
于 2017-10-12T13:54:06.643 回答