Maven exec:多模块项目的java目标

Posted

技术标签:

【中文标题】Maven exec:多模块项目的java目标【英文标题】:Maven exec:java goal on a multi-module project 【发布时间】:2012-06-20 22:08:14 【问题描述】:

我正在尝试在一个简单的两模块项目上运行exec-maven-pluginexec:java 目标,其中一个模块依赖于另一个模块。到目前为止,我找不到有效的配置。这是一个简化的测试用例:

+ exec-multi-module-test/
    + pom.xml
    + module1/
        + pom.xml
        + src/
            + main/
                + java/
                    + HelloPrinter.java
    + module2/
        + pom.xml
        + src/
            + main/
                + java/
                    + MyMain.java

这是父 pom:

<?xml version="1.0" encoding="UTF-8"?>
<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.mkscrg.sandbox</groupId>
    <artifactId>exec-multi-module-test</artifactId>
    <version>1.0</version>
    <packaging>pom</packaging>

    <modules>
        <module>module1</module>
        <module>module2</module>
    </modules>

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

module1的pom:

<?xml version="1.0" encoding="UTF-8"?>
<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>

    <parent>
        <artifactId>exec-multi-module-test</artifactId>
        <groupId>com.mkscrg.sandbox</groupId>
        <version>1.0</version>
    </parent>

    <artifactId>module1</artifactId>
</project>

module2's pom: 4.0.0

    <parent>
        <artifactId>exec-multi-module-test</artifactId>
        <groupId>com.mkscrg.sandbox</groupId>
        <version>1.0</version>
    </parent>

    <artifactId>module2</artifactId>

    <dependencies>
        <dependency>
            <groupId>com.mkscrg.sandbox</groupId>
            <artifactId>module1</artifactId>
            <version>$project.version</version>
        </dependency>
    </dependencies>
</project>

这个项目从顶部编译成功,但是运行mvn exec:java -Dexec.mainClass=myMain失败:

[INFO] Scanning for projects...
[INFO] ------------------------------------------------------------------------
[INFO] Reactor Build Order:
[INFO] 
[INFO] exec-multi-module-test
[INFO] module1
[INFO] module2
[INFO]                                                                         
[INFO] ------------------------------------------------------------------------
[INFO] Building exec-multi-module-test 1.0
[INFO] ------------------------------------------------------------------------
[INFO] 
[INFO] >>> exec-maven-plugin:1.2.1:java (default-cli) @ exec-multi-module-test >>>
[INFO] 
[INFO] <<< exec-maven-plugin:1.2.1:java (default-cli) @ exec-multi-module-test <<<
[INFO] 
[INFO] --- exec-maven-plugin:1.2.1:java (default-cli) @ exec-multi-module-test ---
[WARNING] 
java.lang.ClassNotFoundException: MyMain
        at java.net.URLClassLoader$1.run(URLClassLoader.java:202)
        at java.security.AccessController.doPrivileged(Native Method)
        at java.net.URLClassLoader.findClass(URLClassLoader.java:190)
        at java.lang.ClassLoader.loadClass(ClassLoader.java:306)
        at java.lang.ClassLoader.loadClass(ClassLoader.java:247)
        at org.codehaus.mojo.exec.ExecJavaMojo$1.run(ExecJavaMojo.java:285)
        at java.lang.Thread.run(Thread.java:680)
[INFO] ------------------------------------------------------------------------
[INFO] Reactor Summary:
[INFO] 
[INFO] exec-multi-module-test ............................ FAILURE [0.363s]
[INFO] module1 ........................................... SKIPPED
[INFO] module2 ........................................... SKIPPED
[INFO] ------------------------------------------------------------------------
[INFO] BUILD FAILURE
[INFO] ------------------------------------------------------------------------
[INFO] Total time: 0.566s
[INFO] Finished at: Mon Jun 18 14:11:54 PDT 2012
[INFO] Final Memory: 3M/81M
[INFO] ------------------------------------------------------------------------
[ERROR] Failed to execute goal org.codehaus.mojo:exec-maven-plugin:1.2.1:java (default-cli) on project exec-multi-module-test: An exception occured while executing the Java class. MyMain -> [Help 1]
[ERROR] 
[ERROR] To see the full stack trace of the errors, re-run Maven with the -e switch.
[ERROR] Re-run Maven using the -X switch to enable full debug logging.
[ERROR] 
[ERROR] For more information about the errors and possible solutions, please read the following articles:
[ERROR] [Help 1] http://cwiki.apache.org/confluence/display/MAVEN/MojoExecutionException

配置此项目以允许exec:java 看到MyMain 的正确方法是什么?

编辑:如果您想自己尝试一下,这里有一个要点:https://gist.github.com/2950830

编辑:澄清:我知道可以mvn install 然后从module2 的目录运行exec:java 或使用顶部的-pl 标志。但是,我想避免运行mvn install。无需修改我的本地存储库即可在多模块项目中运行此目标。就像mvn compile“只适用于”多模块项目一样,其他目标/阶段也应该如此。

【问题讨论】:

【参考方案1】:

多模块项目中的目标在从父级运行时,将针对所有模块运行。我不认为那是你想要的。你可以试试:

mvn exec:java -pl module2 -Dexec.mainClass=MyMain

这可能有用吗?更多信息:

Running a specific Maven plugin goal from the command line in a sub-module of a multi-module reactor project

但是,我认为在运行之前将目录更改为包含可执行文件的子模块更直观。

【讨论】:

这(以及 cding 到子模块)只有在您从顶部运行 mvn install 时才有效。否则它会失败,因为它在任何存储库中都找不到module1mvn compile 成功解析了模块,即使它们没有安装在本地 repo 中。我正在寻找一种通过mvn exec:java 获得这种行为的方法。 很抱歉造成混乱;我在问题中添加了说明。 @MikeCraig -- 这个答案不起作用。你为什么接受它?我认为没有“安装”,Maven 无法做到这一点。 SBT 可以做到这一点,但它有自己的一堆问题。 通常您可以添加标志-am 来制作依赖模块,但这会导致插件尝试在该模块中运行 这个答案是错误的。仅在模块之间没有依赖关系时才有效,这不是多模块上下文中的典型场景。在所有其他情况下都会导致“无法解决依赖关系”错误。【参考方案2】: 您应该将exec-maven-plugin 绑定到一个maven 生命周期目标,比如verify。 由于您希望插件仅针对module2 执行,请在pluginManagement 的父pom 中定义插件配置。仅在module 2 中使用相同的。 然后运行以下命令:

mvn verify -Dexec.mainClass=MyMain.

父pom

<build>
    <pluginManagement>
        <plugins>
            <plugin>
                <groupId>org.codehaus.mojo</groupId>
                <artifactId>exec-maven-plugin</artifactId>
                <version>1.2.1</version>
                <executions>
                 <execution>
                     <phase>verify</phase>
                     <goals>
                         <goal>java</goal>
                     </goals>
                 </execution>
                </executions>
            </plugin>
        </plugins>
    </pluginManagement>
</build>

模块 2

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

【讨论】:

我验证了这个作品。凉爽的!仅在运行 exec:java 时使用特殊配置文件来配置此目标会更干净吗?否则,将在经过验证生命周期的任何活动上调用 exec。 @kaliatech。确实是的,根据要求。 @Raghuram - 我尝试了这个解决方案,它非常适合运行验证阶段。但是,当我尝试运行mvn exec:java 时,maven 会抛出一个异常,告诉我父项目缺少参数“mainClass”(而不是只尝试查找子项目的主类)。任何想法我做错了什么? 对我来说这行不通。在我的情况下,不需要每次在验证阶段执行模块。我只需要在手动启动应用程序时运行它,而不是在每次编译/打包/安装/部署时自动运行。【参考方案3】:

请参阅此答案以获取不带mvn install 的单命令替代方案:

https://***.com/a/26448447/1235281

通过在父 pom.xml 中使用 skip,您可以告诉 Maven 仅在特定子模块上运行 exec:java

GitHub: https://github.com/Oduig/mavenfiddle

【讨论】:

以上是关于Maven exec:多模块项目的java目标的主要内容,如果未能解决你的问题,请参考以下文章

无法在项目 mavenproject2 上执行目标 org.codehaus.mojo:exec-maven-plugin:1.3.2:exec (默认):

未能执行目标 org.codehaus.mojo:exec-maven-plugin:1.2:java (default-cli)

使用 Exec Maven 插件分叉 Java,而不使用 `exec` 目标

Maven 不会运行我的项目:无法执行目标 org.codehaus.mojo:exec-maven-plugin:1.2.1:exec

Maven 和 Java:目标 org.codehaus.mojo:exec-maven-plugin:1.2.1:java 的参数“mainClass”丢失或无效

未能执行目标 org.codehaus.mojo:exec-maven-plugin:1.6.0:java