Maven:如何在分叉的 JVM 中执行依赖项?

Posted

技术标签:

【中文标题】Maven:如何在分叉的 JVM 中执行依赖项?【英文标题】:Maven: How to execute a dependency in a forked JVM? 【发布时间】:2019-05-03 01:26:00 【问题描述】:

使用maven-exec-plugin 和一个java 目标,我执行了一个jar 程序来验证我项目中的一些文件。验证失败时调用System.exit返回非零返回码。

问题是它和Maven在同一个JVM中执行,所以当它调用exit时,处理停止,因为它does not fork。

我将其配置为使用 maven-exec-pluginjava 目标执行(如在 here 中)。执行 jar 在我的 Nexus 存储库中,所以我想将其作为 pom.xml 中的依赖项下载。

配置 maven-exec-plugin 依赖项的一个非常好的功能是它会下载 jar 及其所有依赖项,因此无需使用 maven 程序集插件将所有 jar 包含在可执行文件中。

如何配置我的 pom.xml 以执行 jar 依赖并在失败时正确停止?

【问题讨论】:

可以详细说明您的问题到底是什么导致通过 exec-maven-plugin 执行依赖项对我来说听起来很奇怪?如果在构建过程中需要它,请创建它的插件...? @khmarbaise:我想执行一个内部工具来验证代码。我设法运行它 exec:java ,这是一个非常好的解决方案,因为我只需要将它放在我们的包存储库中,maven 会自动下载它及其所有依赖项。问题是它在没有分叉的情况下执行,system.exit 将结束 maven 进程。我将在星期一发布原始解决方案。 我建议通过 maven-plugin 运行您的工具,这将解决您遇到的所有问题... @khmarbaise,哪个插件? 简单地编写它以将其正确集成到您的构建过程中......更干净...... 【参考方案1】:

我解决了我的问题。基本上,我必须使用 exec 目标,而不是使用 java 目标,并运行 java 可执行文件。下面的代码用main方法设置类路径和类。

这个使用 pom.xml 和 Nexus 存储库的解决方案比只为我的用户处理 jar 文件有很多优势:

无需在运行它的机器上安装任何东西,无论是开发机器还是持续集成机器。 验证工具开发者可以发布新版本,并且会自动更新。 开发者可以通过一个简单的参数来关闭它。 也解决了原来的问题:验证工具会在单独的进程中执行,所以maven进程在调用System.exit时不会中止。

这是评论pom.xml

<project xmlns="http://maven.apache.org/POM/4.0.0" 
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <groupId>com.company</groupId>
    <artifactId>yourId</artifactId>
    <version>1.0</version>
    <properties>
                <!-- 
                Skip the validation executing maven setting the parameter below
                mvn integration-test  -Dvalidation.skip
                -->

        <validation.skip>false</validation.skip>
        <java.version>1.8</java.version>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    </properties>

    <build>
        <plugins>
            <plugin>
                <groupId>org.codehaus.mojo</groupId>
                <artifactId>exec-maven-plugin</artifactId>
                <version>1.6.0</version>

                <executions>
                    <execution>
                        <id>MyValidator</id>
                        <phase>integration-test</phase> <!-- you can associate to any maven phase -->
                        <goals>
                            <goal>exec</goal> <!-- forces execution in another process -->
                        </goals>
                    </execution>
                </executions>

                <configuration>
                    <executable>java</executable> <!-- java must be in your PATH -->
                    <includeProjectDependencies>false</includeProjectDependencies>
                    <includePluginDependencies>false</includePluginDependencies>
                    <skip>$validation.skip</skip>
                    <arguments>
                        <argument>-classpath</argument> 
                        <classpath/> <!-- will include your class path -->
                        <mainClass>com.company.yourpackage.AppMain</mainClass> <!-- the class that has your main file -->
                        <argument>argument.xml</argument> <!-- any argument for your executable -->
                    </arguments>
                </configuration>

            </plugin>
        </plugins>
    </build>
    <dependencies>
        <dependency>
            <!-- Specify your executable jar here -->
            <groupId>com.company.validator</groupId>
            <artifactId>validatorId</artifactId>
            <version>RELEASE</version> <!-- you can specify a fixed version here -->
            <type>jar</type>
        </dependency>
    </dependencies>
</project>

您可以运行多个传递其 id 的可执行文件:mvn exec:exec@MyValidator

【讨论】:

arguments xml 在这里坏了 @Vadzim 为什么它坏了?这个对我有用。您必须根据您的系统对其进行调整。也许我可以帮助你适应。发布命令行调用如何。 只是尝试针对 xsd 进行验证。我想,对你有用的并不完全是这里引用的。 mainClass 不是exec:exec 的有效选项:mojohaus.org/exec-maven-plugin/exec-mojo.html

以上是关于Maven:如何在分叉的 JVM 中执行依赖项?的主要内容,如果未能解决你的问题,请参考以下文章

如何从 Maven 存储库中清除旧的依赖项?

如果使用的 JVM 是 x86 或 x64,则以不同方式解决 Maven 依赖项?

Surefire Maven 插件:“通过直接写入分叉 JVM 中的本机流来破坏 STDOUT”

如何将分叉的 lerna 存储库的子包安装为节点依赖项?

如何剔除Maven项目中多余的依赖项?

如何在 Maven 中找出隐藏的依赖项和插件版本?