在构建工具中运行测试
1. Maven
Maven默认在test
阶段启动maven-surefire-plugin
插件,执行项目中的所有测试。
1.1 插件和依赖
要支持JUnit平台,maven-surefire-plugin
插件和maven-failsafe-plugin
插件的版本要在2.20.0以上,同时,测试类路径中必须包含至少一个TestEngine
实现。
如果只需要支持JUnit Jupiter,可以这样配置pom.xml
:
<build>
<plugins>
<plugin>
<artifactId>maven-surefire-plugin</artifactId>
<version>2.22.2</version>
</plugin>
<plugin>
<artifactId>maven-failsafe-plugin</artifactId>
<version>2.22.2</version>
</plugin>
</plugins>
</build>
<!-- ... -->
<dependencies>
<!-- ... -->
<dependency>
<groupId>org.junit.jupiter</groupId>
<artifactId>junit-jupiter-api</artifactId>
<version>5.6.2</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.junit.jupiter</groupId>
<artifactId>junit-jupiter-engine</artifactId>
<version>5.6.2</version>
<scope>test</scope>
</dependency>
<!-- ... -->
</dependencies>
<!-- ... -->
如果要支持JUnit 4编写的测试,要添加junit-vintage-engine
测试依赖项:
<!-- ... -->
<build>
<plugins>
<plugin>
<artifactId>maven-surefire-plugin</artifactId>
<version>2.22.2</version>
</plugin>
<plugin>
<artifactId>maven-failsafe-plugin</artifactId>
<version>2.22.2</version>
</plugin>
</plugins>
</build>
<!-- ... -->
<dependencies>
<!-- ... -->
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.13</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.junit.vintage</groupId>
<artifactId>junit-vintage-engine</artifactId>
<version>5.6.2</version>
<scope>test</scope>
</dependency>
<!-- ... -->
</dependencies>
<!-- ... -->
1.2 根据测试类文件名过滤测试
Maven Surefire插件默认会在项目测试类路径中扫描测试类,将全限定类名符合下面模式的测试类包含到测试中:
**/Test*.java
**/*Test.java
**/*Tests.java
**/*TestCase.java
默认情况下,会忽略掉嵌套测试类。
可以配置maven-surefire-plugin
插件的include
和exclude
规则,改变上面的默认值。例如:
<!-- ... -->
<build>
<plugins>
<plugin>
<artifactId>maven-surefire-plugin</artifactId>
<version>2.22.2</version>
<configuration>
<includes>
<include>**/*.Test.java</include>
<include>**/*.Spec.java</include>
</includes>
<excludes>
<exclude>../Abstract*.java</excluse>
</excludes>
</configuration>
</plugin>
</plugins>
</build>
<!-- ... -->
1.3 根据@Tag
过滤测试
可以根据测试类或测试方法的@Tag
注解来包含或排除某些测试。可以通过Tag表达式来组合Tag。
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-surefire-plugin</artifactId>
<version>2.22.2</version>
<executions>
<execution>
<id>unit-test</id>
<goals>
<goal>test</goal>
</goals>
<phase>test</phase>
<configuration>
<excludedGroups>integration, acceptance</excludedGroups>
</configuration>
</execution>
<execution>
<id>integration-test</id>
<goals>
<goal>test</goal>
</goals>
<phase>integration-test</phase>
<configuration>
<groups>integration | acceptance</groups>
</configuration>
</execution>
</executions>
<configuration>
<includes>
<include>**/*</include>
</includes>
</configuration>
</plugin>
1.4 配置测试参数
可以像下面的示例那样配置测试平台的全局测试参数,其效果等价于在junit-platform.properties
文件中设置全局测试参数:
<!-- ... -->
<build>
<plugins>
<plugin>
<artifactId>maven-surefire-plugin</artifactId>
<version>2.22.2</version>
<configuration>
<properties>
<configurationParameters>
junit.jupiter.conditions.deactivate = *
junit.jupiter.extensions.autodetection.enabled = true
junit.jupiter.testinstance.lifecycle.default = per_class
</configurationParameters>
</properties>
</configuration>
</plugin>
</plugins>
</build>
<!-- ... -->
1.5 执行测试
在maven
构建生命周期的test
阶段,默认绑定了maven-surefire-plugin
插件,会自动执行项目中的单元测试。在进入终端命令解释器之后,在项目的根目录下执行:
mvn clean test
就会自动根据pom.xml
中的配置,执行项目中的全部或部分单元测试。
2. Gradle
从Gradle 4.6
版开始,提供了对JUnit
平台的直接支持。
2.1 配置依赖项
在build.gradle
文件中添加JUnit
依赖项:
dependencies {
testImplementation("org.junit.jupiter:junit-jupiter-api:5.6.2")
testRuntimeOnly("org.junit.jupiter:junit-jupiter-engine:5.6.2")
}
2.2 配置测试引擎
在test
节中指定使用JUnit
:
test {
useJUnitPlatform()
}
如果要根据@Tag
注解对测试进行过滤,需要在useJUnitPlatform()
内部指定includeTags
和/或excludeTags
:
test {
useJUnitPlatform {
includeTags 'fast', 'smoke & feature-a'
excludeTags 'slow', 'ci'
}
}
还可以在useJUnitPlatform()
内部指定包含或排除特定的测试引擎:
test {
useJUnitPlatform {
includeEngines 'junit-jupiter'
excludeEngines 'junit-vintage'
}
}
2.3 配置全局测试参数
还可以指定配置测试平台的全局测试参数:
test {
// ...
systemProperty 'junit.jupiter.conditions.deactivate', '*'
systemProperties = [
'junit.jupiter.extensions.autodetection.enabled': 'true',
'junit.jupiter.testinstance.lifecycle.default': 'per_class'
]
// ...
}
2.4 执行测试
在终端窗口中,进入gradle
项目的根目录,执行下面的命令:
gradle build
会根据build.gradle
中的配置,自动执行项目中全部或部分单元测试。
3. Ant
从1.10.3
版本开始,Ant
通过junitlauncher
任务可以执行启动JUnit
平台。从1.10.6
版开始,支持将测试fork
到独立的JVM
中执行。
build.xml
文件示例配置如下:
<path id="test.classpath">
<!-- The location where you have your compiled test classes -->
<pathelement location="${build.classes.dir}" />
</path>
<!-- ... -->
<junitlauncher>
<classpath refid="test.classpath" />
</junitlauncher>
junitlauncher
任务可以在类路径中找到所有测试类,交给测试引擎执行测试。
可以通过在junitlauncher
中定义test
子任务的方式,让junitlauncher
只执行指定的测试:
<junitlauncher>
<classpath refid="test.classpath" />
<test name="org.myapp.test.MyFirstJUnit5Test" />
</junitlauncher>
可以通过在junitlauncher
中定义testclasses
子任务的方式,让junitlauncher
搜索单元测试:
<junitlauncher>
<classpath refid="test.classpath" />
<testclasses outputdir="${output.dir}">
<fileset dir="${build.classes.dir}">
<include name="org/example/**/demo/**/" />
</fileset>
<fileset dir="${some.other.dir}">
<include name="org/myapp/**/" />
</fileset>
</testclasses>
</junitlauncher>
完整build.xml
文件例子:
<project>
<property name="output.dir" value="${basedir}/build"/>
<property name="src.test.dir" value="${basedir}/src/test"/>
<property name="build.classes.dir" value="${output.dir}/classes"/>
<target name="init">
<mkdir dir="${output.dir}"/>
</target>
<path id="junit.platform.libs.classpath">
<fileset dir="${basedir}/src/lib/junit-platform/"/>
</path>
<path id="junit.engine.jupiter.classpath">
<fileset dir="${basedir}/src/lib/jupiter/"/>
</path>
<target name="compile-test" depends="init">
<mkdir dir="${build.classes.dir}"/>
<javac srcdir="${src.test.dir}"
destdir="${build.classes.dir}">
<!-- our tests only need JUnit Jupiter engine
libraries in our compile classpath for the tests -->
<classpath refid="junit.engine.jupiter.classpath"/>
</javac>
</target>
<target name="test" depends="compile-test">
<junitlauncher>
<!-- include the JUnit platform related libraries
required to run the tests -->
<classpath refid="junit.platform.libs.classpath"/>
<!-- include the JUnit Jupiter engine libraries -->
<classpath refid="junit.engine.jupiter.classpath"/>
<classpath>
<!-- the test classes themselves -->
<pathelement location="${build.classes.dir}"/>
</classpath>
<testclasses outputdir="${output.dir}">
<fileset dir="${build.classes.dir}"/>
<listener type="legacy-brief" sendSysOut="true"/>
<listener type="legacy-xml" sendSysErr="true" sendSysOut="true"/>
</testclasses>
</junitlauncher>
</target>
</project>
运行下面的命令执行测试:
ant test