如何将测试类包含到 Maven jar 中并执行它们?

问题描述 投票:0回答:2

在 Maven 项目中,我的测试类和源类位于同一个包中,但位于不同的物理位置。

.../src/main/java/package/** <-- application code
.../src/test/java/package/** <-- test code

测试类中访问源类是没有问题的, 但我想在 main 方法中运行一个测试运行程序并访问

AllTest.class
,以便我可以创建 jar 并执行我的测试。

 public static void main(String[] args) {
    // AllTest not found
    Result result = JUnitCore.runClasses(AllTest.class);
    for (Failure failure : result.getFailures()) {
        System.out.println(failure.toString());
    }
    System.out.println(result.wasSuccessful());
}

但它不起作用,因为我无权访问测试代码。我不明白,因为它们在同一个包中。

问题:如何从应用程序类访问测试类?或者,Maven 如何打包包含测试类的 fat jar 并执行测试?

java maven unit-testing maven-3 maven-jar-plugin
2个回答
66
投票

您不应从应用程序代码中访问测试类,而应在测试范围中创建一个 main(相同的 main),并为您的项目创建一个附加工件。

但是,在这个额外的工件(jar)中,您需要:

  • 测试班
  • 应用程序代码类
  • 应用程序代码所需的外部依赖项(在
    compile
    范围内)
  • 测试代码所需的外部依赖项(在
    test
    范围内)

这基本上意味着一个胖罐子,添加了测试类(及其依赖项)。 Maven Jar Plugin 及其

test-jar
目标无法满足这种需求。 Maven Shade Plugin 及其
shadeTestJar
选项也无济于事。

那么,如何在 Maven 中创建一个包含测试类和外部依赖项的 fat jar?

在这种情况下,Maven Assembly Plugin 是完美的候选者。

这是一个最小的 POM 示例:

<project>
    <modelVersion>4.0.0</modelVersion>
    <groupId>com.sample</groupId>
    <artifactId>sample-project</artifactId>
    <version>1.0-SNAPSHOT</version>

    <build>
        <plugins>
            <plugin>
                <artifactId>maven-assembly-plugin</artifactId>
                <version>2.3</version>
                <configuration>
                    <descriptor>src/main/assembly/assembly.xml</descriptor>
                </configuration>
                <executions>
                    <execution>
                        <id>make-assembly</id>
                        <phase>package</phase>
                        <goals>
                            <goal>single</goal>
                        </goals>
                        <configuration>
                            <archive>
                                <manifest>
                                    <mainClass>com.sample.TestMain</mainClass>
                                </manifest>
                            </archive>
                        </configuration>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>

    <dependencies>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.11</version>
            <scope>test</scope>
        </dependency>
    </dependencies>
</project>

上面的配置是设置您在测试类中定义的主类。但这还不够。

您还需要创建一个 descriptor 文件,在

src\main\assembly
文件夹中创建一个
assembly.xml
文件,其中包含以下内容:

<assembly
    xmlns="http://maven.apache.org/plugins/maven-assembly-plugin/assembly/1.1.3"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/plugins/maven-assembly-plugin/assembly/1.1.3 http://maven.apache.org/xsd/assembly-1.1.3.xsd">
    <id>fat-tests</id>
    <formats>
        <format>jar</format>
    </formats>
    <includeBaseDirectory>false</includeBaseDirectory>
    <dependencySets>
        <dependencySet>
            <outputDirectory>/</outputDirectory>
            <useProjectArtifact>true</useProjectArtifact>
            <unpack>true</unpack>
            <scope>test</scope>
        </dependencySet>
    </dependencySets>
    <fileSets>
        <fileSet>
            <directory>${project.build.directory}/test-classes</directory>
            <outputDirectory>/</outputDirectory>
            <includes>
                <include>**/*.class</include>
            </includes>
            <useDefaultExcludes>true</useDefaultExcludes>
        </fileSet>
    </fileSets>
</assembly>

上面的配置是:

  • 设置从
    test
    范围获取的外部依赖项(这也将获取
    compile
    范围)
  • 设置
    fileset
    以将编译的测试类包含在打包的 fat jar 中
  • 使用
    fat-tests
    分类器设置最终的 jar(因此你的最终文件将类似于
    sampleproject-1.0-SNAPSHOT-fat-tests.jar
    )。

然后您可以按如下方式调用 main(从

target
文件夹):

java -jar sampleproject-1.0-SNAPSHOT-fat-tests.jar

从这样的主程序中,您还可以调用所有测试用例,如下所示:

  • 创建 JUnit 测试套件
  • 将相关测试添加到测试套件中
  • 从普通 Java 主程序中调用测试套件

测试套件示例:

import org.junit.runner.RunWith;
import org.junit.runners.Suite;
import org.junit.runners.Suite.SuiteClasses;

@RunWith(Suite.class)
@SuiteClasses({ AppTest.class })
public class AllTests {

}

注意:在这种情况下,测试套件仅涉及

AppTest
示例测试。

然后你可以有一个主类如下:

import org.junit.internal.TextListener;
import org.junit.runner.JUnitCore;

public class MainAppTest {

    public static void main(String[] args) {
        System.out.println("Running tests!");
        
        JUnitCore engine = new JUnitCore();
        engine.addListener(new TextListener(System.out)); // required to print reports
        engine.run(AllTests.class);
    }
}

上面的 main 将执行测试套件,该套件将链式执行所有配置的测试。


0
投票

线路中的预期内容

<mainClass>com.sample.TestMain</mainClass>
类名之前? 我的意思是我应该输入什么而不是 com.sample ? 我已经决定了groupId 就像这里
<groupId>com.sample</groupId>
但如果我把我的
<groupId>project</groupId>
,

<mainClass>project.TestMain</mainClass>

它在我的情况下不起作用。 我收到错误“无法找到或加载主类项目.Main”

© www.soinside.com 2019 - 2024. All rights reserved.