执行集成测试后,Maven Exec 插件被阻止并且 HTTP 服务器启动

问题描述

我一直在尝试将我们的集成测试转换为使用 Maven Exec 插件来启动服务器,但它只会在集成测试执行后启动 HTTP 服务器,即使我指定了 pre-integration-test

问题

第一季度。有没有其他方法可以让集成测试在之后开始?

第二季度。有没有办法让集成阶段等待服务器启动?

Q3。 Maven Exec 和 Bazaar Maven Exec 插件有什么区别?

注意:这以前使用 bazaar maven exec plugin 工作过,但是这已经坏了,似乎无法让它在 Jenkins 上为 JDK 11.0.9 恢复工作。

以下是我们 POM 的一个片段,我们有各种其他插件来创建 jar 并启动 MockServer。

        <plugins>
            <plugin>
                <groupId>org.codehaus.mojo</groupId>
                <artifactId>exec-maven-plugin</artifactId>
                <version>3.0.0</version>
                <executions>
                    <execution>
                        <id>exec</id>
                        <phase>pre-integration-test</phase>
                        <goals>
                            <goal>exec</goal>
                        </goals>
                    </execution>
                </executions>
                <configuration>
                    <asyncDestroyOnShutdown>true</asyncDestroyOnShutdown>
                    <executable>java</executable>
                    <commandlineArgs>-Dserver.port=8089 -jar /gitRepos/public-api/target/api-jar-with-dependencies.jar</commandlineArgs>
                    <async>true</async>
                </configuration>
            </plugin>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-failsafe-plugin</artifactId>
                <version>3.0.0-M5</version>
                <executions>
                    <execution>
                        <goals>
                            <goal>verify</goal>
                        </goals>
                    </execution>
                </executions>
                <configuration>
                    <excludedGroups>${it.excluded}</excludedGroups>
                    <argLine>--add-opens=java.base/jdk.internal.misc=ALL-UNNAMED</argLine>
                    <argLine>
                        --illegal-access=permit
                    </argLine>
                </configuration>
            </plugin>
          </plugins>


<!--            this what was prevIoUsly working. -->
<!--            <plugin>-->
<!--                <groupId>com.bazaarvoice.maven.plugins</groupId>-->
<!--                <artifactId>process-exec-maven-plugin</artifactId>-->
<!--                <version>0.9</version>-->
<!--                <configuration>-->
<!--                    <processLogFile>${project.build.directory}/my-log.log</processLogFile>-->
<!--                </configuration>-->
<!--            </plugin>-->

对我来说奇怪的是 HTTP 服务器在集成测试之后启动,就像其他插件阻止服务器启动一样。请参阅下面的日志。

[DEBUG] Freed 4 thread-local buffer(s) from thread: nioEventLoopGroup-3-20
[DEBUG] Freed 4 thread-local buffer(s) from thread: nioEventLoopGroup-3-19
[DEBUG] Freed 5 thread-local buffer(s) from thread: nioEventLoopGroup-3-28
[INFO] [main] 13:13:37.594 [main] INFO  c.i.p.s.Server - Grizzly ThreadPool for listener grizzly set to 24 worker threads.
[INFO] [main] 13:13:37.724 [main] INFO  o.g.g.http.server.NetworkListener - Started listener bound to [0.0.0.0:8089]
[INFO] [main] 13:13:37.726 [main] INFO  o.g.grizzly.http.server.HttpServer - [HttpServer] Started.

解决方法

感谢您在问题下方发表评论。

在深入了解您的问题后,我决定重现您的怀疑,并得出结论,使用 exec-maven-plugin 一切正常。

让我解释一下 - 我没有改变你的 pom.xml 片段,只实现了一个小的 Java 主类,它每 100 毫秒打印一行。

public class Ocp {
    public static void main(String[] args) {
        IntStream.range(1,100).forEach(i -> {
            System.err.println("====== " + i);
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
    }
}

结果如下:

~/dev/git/so-mvn$ mvn clean verify
[INFO] Scanning for projects...
[INFO] 
[INFO] ---------------------------< de.mle:so-mvn >----------------------------
[INFO] Building stackoverflow 0.0.1-SNAPSHOT
[INFO] --------------------------------[ jar ]---------------------------------
[INFO] 

***intentionally left out***

[INFO] --- maven-jar-plugin:2.4:jar (default-jar) @ so-mvn ---
[INFO] Building jar: /home/marco/dev/git/so-mvn/target/so-mvn-0.0.1-SNAPSHOT.jar
[INFO] 
[INFO] --- exec-maven-plugin:3.0.0:exec (exec) @ so-mvn ---
[INFO] 
[INFO] --- maven-failsafe-plugin:3.0.0-M5:integration-test (default) @ so-mvn ---
====== 1
====== 2
====== 3
====== 4
[INFO] 
[INFO] -------------------------------------------------------
[INFO]  T E S T S
[INFO] -------------------------------------------------------
====== 5
====== 6
====== 7
====== 8
[INFO] Running so.mvn.Ocp11IT
====== 9
[INFO] Tests run: 1,Failures: 0,Errors: 0,Skipped: 0,Time elapsed: 0.057 s - in so.mvn.Ocp11IT
[INFO] 
[INFO] Results:
[INFO] 
[INFO] Tests run: 1,Skipped: 0
[INFO] 
[INFO] 
[INFO] --- maven-failsafe-plugin:3.0.0-M5:verify (default) @ so-mvn ---
[INFO] ------------------------------------------------------------------------
[INFO] BUILD SUCCESS
[INFO] ------------------------------------------------------------------------
[INFO] Total time:  2.398 s
[INFO] Finished at: 2021-02-12T19:17:24+01:00
[INFO] ------------------------------------------------------------------------

如您所见,该流程在集成测试阶段(Q1 和 Q2)之前就开始了。仅由于您的 async 配置,集成测试不会等待我的存根服务器完成启动。所以在这里使用 Maven 一切正常。

您提到的 bazaarvoice plugin 只是做了一个小技巧来缓解这种情况。它有两个功能,在它们为集成测试提供“开始”之前稍等片刻。它们是:

  • healthcheckUrl:推荐,但可选。您应该提供一个健康检查 url,以便插件等待您的进程的健康检查全部为绿色。如果未提供,插件会等待 waitAfterLaunch 秒后继续。
  • waitAfterLaunch:可选。这指定启动进程后等待的最长时间(以秒为单位)。如果指定了 healthcheckUrl,则一旦健康检查通过,它将继续前进。默认值为 30 秒。

总结 在您的情况下,您的 bazaarvoice 插件只是等待了默认的 30 秒,这似乎足以让您的外部服务器启动。 exec-maven-plugin 中没有这样的“等待”功能,但您可以例如在您的集成测试本身(Q3)中以编程方式移动这个但缺少的健康检查,就像在依赖(微)服务的现实生活中的健康检查一样。另一种选择(我的首选)是,将所有外部服务器依赖项或进程移动到 docker 容器中,然后再次对 (Docker) plugin level 进行健康检查。

如果您查看我的示例项目及其 little wait for an ES container to come upcorresponding integration test,可能是一个很好的入门。

另一个无需在插件级别等待时间的选项是非常复杂的 Awaitility lib,它在您的断言中带有很好的 waiting featurespolling intervals

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...