maven-jar-plugin 生成错误的 jar MANIFEST.MF

maven-jar-plugin generates jar with wrong MANIFEST.MF

我想生成一个包含其依赖项的 jar 文件。这是我的 pom.xml:

<?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">
    <parent>
        <artifactId>vehicle</artifactId>
        <groupId>org</groupId>
        <version>0.0.1-SNAPSHOT</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <artifactId>vehicle.cli</artifactId>

    <dependencies>
        <dependency>
            <groupId>org</groupId>
            <artifactId>vehicle.model</artifactId>
            <version>0.0.1-SNAPSHOT</version>
        </dependency>
        <dependency>
            <groupId>commons-cli</groupId>
            <artifactId>commons-cli</artifactId>
        </dependency>
    </dependencies>
    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-dependency-plugin</artifactId>
                <executions>
                    <execution>
                        <id>copy-dependencies</id>
                        <phase>prepare-package</phase>
                        <goals>
                            <goal>copy-dependencies</goal>
                        </goals>
                        <configuration>
                            <outputDirectory>
                                ${project.build.directory}/libs
                            </outputDirectory>
                        </configuration>
                    </execution>
                </executions>
            </plugin>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-jar-plugin</artifactId>
                <version>3.2.0</version>
                <configuration>
                    <archive>
                        <manifest>
                            <addClasspath>true</addClasspath>
                            <classpathPrefix>libs/</classpathPrefix>
                            <mainClass>
                                org.vehicle.cli.Main
                            </mainClass>
                        </manifest>
                        <manifestEntries>
                            <Class-Path>.</Class-Path>
                        </manifestEntries>
                    </archive>
                </configuration>
            </plugin>
        </plugins>
    </build>
</project>

依赖项 org.vehicle.model 我从自定义的关系存储库中获取它。 在 运行 mvn package 之后,我得到了目标文件夹,其中的 libs 文件夹和 jar 文件位于同一层次结构中:

- target
   - libs
     - vehicle.model-0.0.1-SNAPSHOT.jar
     - commons-cli-1.4.jar
   - vehicle.cli-0.0.1-SNAPSHOT.jar

当我执行 vehicle.cli-0.0.1-SNAPSHOT.jar (java -jar vehicle.cli-0.0.1-SNAPSHOT.jar) 我得到一个异常 java.lang.NoClassDefFoundError为类在vehicle.model-0.0.1-SNAPSHOT.jar 这很奇怪,因为我已经在 libs 文件夹中拥有所有依赖项 jar,所以我解压缩 vehicle.cli-0.0.1-SNAPSHOT.jar 以检查 MANIFEST.MF 并查看以下内容:

Manifest-Version: 1.0
Created-By: Maven Jar Plugin 3.2.0
Build-Jdk-Spec: 16
Class-Path: . libs/vehicle.model-0.0.1-20210203.101620-3.jar libs/commons-cli-1.4.jar
Specification-Title: vehicle.cli
Specification-Version: 0.0
Implementation-Title: vehicle.cli
Implementation-Version: 0.0.1-SNAPSHOT
Main-Class: org.vehicle.cli.Main

让我感到困惑的是在 ClassPath 的行中,它正在调用名为 libs/vehicle.model-0.0.1-20210203.101620-3.jar[=33= 的 jar 文件] 而此 jar 文件不存在。我希望它应该调用名称为 libs/vehicle.model-0.0.1-SNAPSHOT.jar 的 jar 文件,因为它是由 maven-dependency-plugin 复制到 libs 文件夹的文件。我不明白版本 -0.0.1-20210203.101620-3 来自哪里。

任何人都可以向我解释这里的问题是什么以及如何解决它?谢谢!

我们以前 运行 做过类似的事情。有一个 bug reported for the jar plugin 没有修复。幸运的是,有一个解决方法。像这样调整 jar 插件中的清单:

<archive>
    <manifest>
        <addClasspath>true</addClasspath>
        <classpathPrefix>libs/</classpathPrefix>
        <mainClass>org.vehicle.cli.Main</mainClass>
        <useUniqueVersions>false</useUniqueVersions>   <!-- important! -->
    </manifest>
    <manifestEntries>
        <Class-Path>.</Class-Path>
    </manifestEntries>
</archive>

如果这还不够,dependency plugin 有一个类似的 <useBaseVersion> 属性 可以启用。