如何使用 Maven 测试 Java 控制台应用程序?

Posted

技术标签:

【中文标题】如何使用 Maven 测试 Java 控制台应用程序?【英文标题】:How to test a java console application with Maven? 【发布时间】:2012-12-13 16:51:11 【问题描述】:

这些测试应该在构建 jar 文件之后运行。 他们应该测试它是否运行, 以及在给定某些输入文件的情况下是否产生正确的控制台输出。

您能否指出一些针对控制台应用程序进行此类测试的示例?这些测试是否必须用 Java 编写才能与 Maven 一起使用?

有一个类似的问题 (Testing console based applications/programs - Java) 但我需要的是黑盒测试,而不是单元测试。而且它没有提到 Maven。


更新:

事实证明,这很容易做到。我的第一个问题是在这里看到这么多定义后对集成测试的误解(What is Unit test, Integration Test, Smoke test, Regression Test?;What's the difference between unit, functional, acceptance, and integration tests?;What are unit testing and integration testing, and what other types of testing should I know about?;What is an integration test exactly?)。第二个问题是我并没有打算用Java编写这些测试,但最终我不得不学习如何使用java.lang.Runtime

首先,我已将此添加到我的 pom.xml

<plugin>
    <artifactId>maven-failsafe-plugin</artifactId>
    <version>2.6</version>
    <executions>
        <execution>
            <goals>
                <goal>integration-test</goal>
                <goal>verify</goal>
            </goals>
        </execution>
    </executions>
</plugin>

然后我在src/test/java中创建了一个*IT.java文件:

public class MyAppIT 

    @Test
    public void test() throws IOException 
        Runtime runtime = Runtime.getRuntime();
        String cmd = "java -cp target/myapp.jar my.app.Main";
        Process process = runtime.exec(cmd);
        InputStreamReader isr = new InputStreamReader(process.getErrorStream());
        BufferedReader br = new BufferedReader(isr);
        Boolean containsUsage = false;
        String line;
        while ((line = br.readLine()) != null) 
            if (line.contains("Usage")) 
                containsUsage = true;
            
        
        assertTrue(containsUsage);
    

现在我使用 mvn verify 而不是 mvn package

Results :

Tests run: 1, Failures: 0, Errors: 0, Skipped: 0

【问题讨论】:

【参考方案1】:

您似乎想要进行集成测试。

在许多情况下,应用程序是由持续集成 (CI) 系统构建的,例如 Hudson、Travis 等。此 CI 在每次提交源代码后编译并运行单元测试。 编译成功后,触发集成测试项目。

集成测试驻留在单独的 Maven 源项目中。您将像在 src/test/java 中的普通单元测试一样创建集成测试类。当然,您不会孤立地测试单个类 - 即单元测试 - 您为集成测试设置系统(连接正确的组件)并执行所需的测试“冒烟测试”。

因此,在您的情况下,您可以为集成测试创建另一个项目,并在构建应用程序后执行此项目。您可以创建一个 maven parent-pom 并使用子模块来避免手动集成测试构建。

测试类需要是 java 并且位于 src/test/java,它们看起来像普通的单元测试并且由 maven 运行。 当然,您可以在该测试中启动一个非 java 程序并断言结果。

控制台应用程序有一个“ma​​in”方法,所以我会在测试中调用 main 方法。否则你必须运行一个shell命令或其他东西。 (通过 java.lang.Runtime)

测试控制台应用程序没有专长。除非您想测试用户界面 - 用户界面的自动化测试并不容易而且更加困难,请说明这是否是您的重点。

【讨论】:

是的,我想要测试程序是否真正运行,而不是“main”方法是否有效。我不会称它为我的程序的接口测试不允许任何用户交互。它只是处理输入文件并输出结果。 命令行调用发出对 main 方法的调用。您可以使用运行时执行“java my.app.Main”命令,您在测试项目中拥有完整的 java 可能性。【参考方案2】:

我建议将maven-invoker-plugin 与exec-maven-plugin 结合使用。 Invoker 创建一个子项目并运行它,其中 exec 插件启动您的 JAR。然后调用者验证其输出。

例如,看看s3auth-relay 项目中的这个pom.xml。该模块本身生成一个 JAR 控制台应用程序,它是一个 HTTP 守护程序。 pom.xmlmaven-invoker-plugin 配置的一部分。它启动控制台应用程序并检查它是否正常关闭。为了防止 JAR 作为守护进程启动,我们使用了可选的 -d 标志。

该示例没有显示测试 JAR 输出的能力,但可以使用 maven-invoker-plugin 的 post build script 机制进行测试。

【讨论】:

不幸的是,maven-invoker-plugin 旨在运行 Maven,作为特别是 maven-plugins 集成测试的一部分,而不是控制台应用程序。 只要您的控制台应用程序不需要用户输入,maven-invoker-plugin 就是完美的选择 你有一些使用例子吗? 据我了解,这种方法只能测试应用程序是否真正运行并正常关闭,仅此而已。无法以这种方式分析其输出。我说的对吗? 是的,只有当它正常关闭时。我更新了我的答案以显示分析其输出的能力。

以上是关于如何使用 Maven 测试 Java 控制台应用程序?的主要内容,如果未能解决你的问题,请参考以下文章

sh 使用Maven创建Java控制台应用程序

SpringBoot集成gRPC微服务工程搭建实践

Maven分模块分工程管理

如何使用Spring + EasyMock做Java单元测试

Exec Maven插件

如何使用 Java 启动和停止 Tomcat 容器?