【问题标题】:Invalid or corrupt JAR File built by Maven shade plugin由 Maven shade 插件构建的无效或损坏的 JAR 文件
【发布时间】:2012-10-12 20:27:45
【问题描述】:

将 Maven jFree 依赖项添加到我现有的应用程序后,我无法执行创建的 jar。

我得到的唯一错误信息如下:

java -jar target/com.company.product-1.0.0-SNAPSHOT.jar 
Error: Invalid or corrupt jarfile target/com. company.product-1.0.0-SNAPSHOT.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/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion
<groupId>com.mycompany</groupId>
<artifactId>com.mycompany.test</artifactId>
<name>${project.artifactId}</name>
<version>1.0.0-SNAPSHOT</version>

<properties>
    <java-version>1.7</java-version>
    <org.springframework-version>3.1.1.RELEASE</org.springframework-version>
    <org.springframework.data-version>1.0.3.RELEASE</org.springframework.data-version>
    <org.springframework.ws-version>2.0.4.RELEASE</org.springframework.ws-version>
    <org.springframework.ws.oxm-version>1.5.10</org.springframework.ws.oxm-version>
    <org.aspectj-version>1.6.12</org.aspectj-version>
    <org.slf4j-version>1.5.10</org.slf4j-version>
    <selenium-java-version>2.25.0</selenium-java-version>
    <browser-mob-version>2.0-beta-6</browser-mob-version>
</properties>

<dependencies>

    <!-- Hint A: If we delete this dependency it works -->
    <dependency>
        <groupId>org.jfree</groupId>
        <artifactId>jfreechart</artifactId>
        <version>1.0.14</version>
                    <exclusions>
            <exclusion>
                <artifactId>itext</artifactId>
                <groupId>com.lowagie</groupId>
            </exclusion>
        </exclusions>
    </dependency>

    <dependency>
        <groupId>com.itextpdf</groupId>
        <artifactId>itextpdf</artifactId>
        <version>5.3.2</version>
    </dependency>

    <dependency>
        <groupId>de.schlichtherle.io</groupId>
        <artifactId>truezip</artifactId>
        <version>6.6</version>
    </dependency>

    <dependency>
        <groupId>log4j</groupId>
        <artifactId>apache-log4j-extras</artifactId>
        <version>1.1</version>
    </dependency>

    <!-- Caching with ehcache -->
    <dependency>
        <groupId>net.sf.ehcache</groupId>
        <artifactId>ehcache</artifactId>
        <version>2.5.2</version>
        <type>pom</type>
    </dependency>

    <!-- Spring -->
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-context</artifactId>
        <version>${org.springframework-version}</version>
        <exclusions>
            <!-- Exclude Commons Logging in favor of SLF4j -->
            <exclusion>
                <groupId>commons-logging</groupId>
                <artifactId>commons-logging</artifactId>
            </exclusion>
        </exclusions>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-webmvc</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-core</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-beans</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-web</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-aop</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-context-support</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>

    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-tx</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>

    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-orm</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>

    <dependency>
        <groupId>org.springframework.security</groupId>
        <artifactId>spring-security-core</artifactId>
        <version>3.0.3.RELEASE</version>
    </dependency>

    <dependency>
        <groupId>org.springframework.security</groupId>
        <artifactId>spring-security-web</artifactId>
        <version>3.0.3.RELEASE</version>
    </dependency>

    <dependency>
        <groupId>org.springframework.security</groupId>
        <artifactId>spring-security-config</artifactId>
        <version>3.0.3.RELEASE</version>
    </dependency>

    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-test</artifactId>
        <version>${org.springframework-version}</version>
        <scope>test</scope>
    </dependency>
    <!-- Hibernate -->
    <dependency>
        <groupId>org.hibernate.java-persistence</groupId>
        <artifactId>jpa-api</artifactId>
        <version>2.0-cr-1</version>
    </dependency>

    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-entitymanager</artifactId>
        <version>3.5.1-Final</version>
    </dependency>

    <!-- Database Connectors (HSQL should be removed later) -->
    <dependency>
        <groupId>hsqldb</groupId>
        <artifactId>hsqldb</artifactId>
        <version>1.8.0.7</version>
    </dependency>

    <dependency>
        <groupId>mysql</groupId>
        <artifactId>mysql-connector-java</artifactId>
        <version>5.1.16</version>
    </dependency>

    <!-- AspectJ -->
    <dependency>
        <groupId>org.aspectj</groupId>
        <artifactId>aspectjrt</artifactId>
        <version>${org.aspectj-version}</version>
    </dependency>

    <!-- Logging -->
    <dependency>
        <groupId>org.slf4j</groupId>
        <artifactId>slf4j-api</artifactId>
        <version>${org.slf4j-version}</version>
    </dependency>

    <dependency>
        <groupId>log4j</groupId>
        <artifactId>log4j</artifactId>
        <version>1.2.16</version>
    </dependency>

    <!-- @Inject -->
    <dependency>
        <groupId>javax.inject</groupId>
        <artifactId>javax.inject</artifactId>
        <version>1</version>
    </dependency>

    <dependency>
        <groupId>javax.servlet</groupId>
        <artifactId>jstl</artifactId>
        <version>1.2</version>
    </dependency>

    <!-- Spring Data JPA dependencies -->
    <dependency>
        <groupId>org.springframework.data</groupId>
        <artifactId>spring-data-jpa</artifactId>
        <version>${org.springframework.data-version}</version>
    </dependency>
    <dependency>
        <groupId>javax.ws.rs</groupId>
        <artifactId>jsr311-api</artifactId>
        <version>1.1.1</version>
    </dependency>

    <dependency>
        <groupId>org.springframework.ws</groupId>
        <artifactId>spring-oxm</artifactId>
        <version>${org.springframework.ws.oxm-version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework.ws</groupId>
        <artifactId>spring-xml</artifactId>
        <version>${org.springframework.ws-version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework.ws</groupId>
        <artifactId>spring-ws-core</artifactId>
        <version>${org.springframework.ws-version}</version>
    </dependency>
    <dependency>
        <groupId>javax.xml</groupId>
        <artifactId>jaxb-impl</artifactId>
        <version>2.1</version>
    </dependency>
    <dependency>
        <groupId>javax.xml</groupId>
        <artifactId>jaxb-api</artifactId>
        <version>2.1</version>
    </dependency>

    <!-- Test -->
    <dependency>
        <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>4.7</version>
        <scope>test</scope>
    </dependency>

    <dependency>
        <groupId>org.mockito</groupId>
        <artifactId>mockito-all</artifactId>
        <version>1.9.0</version>
        <scope>test</scope>
    </dependency>

    <!-- Common Utils -->
    <dependency>
        <groupId>commons-lang</groupId>
        <artifactId>commons-lang</artifactId>
        <version>2.6</version>
    </dependency>

    <dependency>
        <groupId>commons-collections</groupId>
        <artifactId>commons-collections</artifactId>
        <version>3.2.1</version>
    </dependency>

    <dependency>
        <groupId>commons-cli</groupId>
        <artifactId>commons-cli</artifactId>
        <version>1.2</version>
    </dependency>       

    <!-- Selenium -->
    <dependency>
        <groupId>org.seleniumhq.selenium</groupId>
        <artifactId>selenium-java</artifactId>
        <version>${selenium-java-version}</version>
    </dependency>

    <dependency>
        <groupId>org.seleniumhq.selenium</groupId>
        <artifactId>selenium-firefox-driver</artifactId>
        <version>${selenium-java-version}</version>
    </dependency>

    <dependency>
        <groupId>org.seleniumhq.selenium</groupId>
        <artifactId>selenium-chrome-driver</artifactId>
        <version>${selenium-java-version}</version>
    </dependency>

    <!-- CSV Lib for Keyword Checker -->
    <dependency>
        <groupId>net.sf.opencsv</groupId>
        <artifactId>opencsv</artifactId>
        <version>2.0</version>
    </dependency>

    <!-- Google Places API -->
    <dependency>
        <groupId>com.google.api-client</groupId>
        <artifactId>google-api-client</artifactId>
        <version>1.10.3-beta</version>
        <exclusions>
            <exclusion>
                <artifactId>jackson-core-asl</artifactId>
                <groupId>org.codehaus.jackson</groupId>
            </exclusion>
        </exclusions>
    </dependency>
    <dependency>
        <groupId>com.google.api-client</groupId>
        <artifactId>google-api-client-appengine</artifactId>
        <version>1.10.3-beta</version>
    </dependency>

    <dependency>
        <groupId>org.codehaus.jackson</groupId>
        <artifactId>jackson-mapper-asl</artifactId>
        <version>1.9.9</version>
    </dependency>

    <!-- Google Geocode -->
    <dependency>
        <groupId>com.google.code.geocoder-java</groupId>
        <artifactId>geocoder-java</artifactId>
        <version>0.9</version>
    </dependency>

    <dependency>
        <groupId>com.googlecode.json-simple</groupId>
        <artifactId>json-simple</artifactId>
        <version>1.1</version>
    </dependency>

    <dependency>
        <groupId>com.googlecode.json-simple</groupId>
        <artifactId>json-simple</artifactId>
        <version>1.1</version>
    </dependency>

    <dependency>
        <groupId>net.sf.jgrapht</groupId>
        <artifactId>jgrapht</artifactId>
        <version>0.8.3</version>
    </dependency>

    <dependency>
        <groupId>jgraph</groupId>
        <artifactId>jgraph</artifactId>
        <version>5.13.0.0</version>
    </dependency>

    <!-- Apache Http Client -->
    <dependency>
        <groupId>org.apache.httpcomponents</groupId>
        <artifactId>httpclient</artifactId>
        <version>4.2.1</version>
    </dependency>

    <!-- Amazon web services client -->
    <dependency>
        <groupId>com.amazonaws</groupId>
        <artifactId>aws-java-sdk</artifactId>
        <version>1.2.15</version>
        <exclusions>
            <exclusion>
                <artifactId>jackson-core-asl</artifactId>
                <groupId>org.codehaus.jackson</groupId>
            </exclusion>
        </exclusions>
    </dependency>

    <!-- Docx4j - reading excel files  -->
    <dependency>
        <groupId>org.docx4j</groupId>
        <artifactId>docx4j</artifactId>
        <version>2.8.0</version>
    </dependency>

    <!-- Browser Mob Proxy -->

    <dependency>
        <groupId>biz.neustar</groupId>
        <artifactId>browsermob-proxy</artifactId>
        <version>${browser-mob-version}</version>
        <exclusions>
            <exclusion>
                <groupId>org.seleniumhq.selenium</groupId>
                <artifactId>selenium-api</artifactId>
            </exclusion>
            <exclusion>
                <artifactId>icu4j</artifactId>
                <groupId>com.ibm.icu</groupId>
            </exclusion>
            <exclusion>
                <artifactId>jackson-mapper-asl</artifactId>
                <groupId>org.codehaus.jackson</groupId>
            </exclusion>
            <exclusion>
                <artifactId>jackson-core-asl</artifactId>
                <groupId>org.codehaus.jackson</groupId>
            </exclusion>
            <exclusion>
                <groupId>org.slf4j</groupId>
                <artifactId>slf4j-jdk14</artifactId>
            </exclusion>
        </exclusions>
    </dependency>

    <!-- Hint B: If we copy this Apache POI dependencies to the top, it works -->
    <!-- Apache POI - for reading xlsx files -->
    <dependency>
        <groupId>org.apache.poi</groupId>
        <artifactId>poi</artifactId>
        <version>3.8</version>
    </dependency>

    <dependency>
        <groupId>org.apache.poi</groupId>
        <artifactId>poi-ooxml</artifactId>
        <version>3.8</version>
    </dependency>

    <dependency>
        <groupId>org.apache.xmlbeans</groupId>
        <artifactId>xmlbeans</artifactId>
        <version>2.5.0</version>
    </dependency>

    <dependency>
        <groupId>org.apache.poi</groupId>
        <artifactId>ooxml-schemas</artifactId>
        <version>1.1</version>
    </dependency>

</dependencies>

<build>
    <plugins>
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-compiler-plugin</artifactId>
            <configuration>
                <source>${java-version}</source>
                <target>${java-version}</target>
            </configuration>
        </plugin>

        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-shade-plugin</artifactId>
            <version>1.7</version>
            <executions>
                <execution>
                    <phase>package</phase>
                    <goals>
                        <goal>shade</goal>
                    </goals>
                    <configuration>
                        <transformers>

                            <!-- must be SURE to do this with both spring.handlers and spring.schemas. 
                                otherwise you won't be able to use them in the spring config files. -->
                            <transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
                                <resource>META-INF/spring.handlers</resource>
                            </transformer>

                            <transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
                                <resource>META-INF/spring.schemas</resource>
                            </transformer>

                            <transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
                                <mainClass>com.mycompany.test.Start</mainClass>
                            </transformer>
                        </transformers>

                        <filters>
                          <filter>
                            <artifact>bouncycastle:bcprov-jdk15</artifact>
                            <excludes>
                              <exclude>META-INF/BCKEY.DSA</exclude>
                              <exclude>META-INF/BCKEY.SF</exclude>
                              <exclude>META-INF/MANIFEST.MF</exclude>
                            </excludes>
                          </filter>
                        </filters>                          

                    </configuration>
                </execution>
            </executions>
        </plugin>

    </plugins>
</build>

如果我执行 jar,我不会得到任何其他调试信息。

是否有任何验证 jarfile 的选项?或者有类似java -verbose...?

重现步骤:

  1. 创建一个新文件夹
  2. 将上面的 Pom.xml 复制到文件夹中。
  3. 保存小 `public static void main(...)。进入 src/main/java/com/mycompany/test/Start.java
  4. 执行mvn clean package
  5. 执行java -jar target/com.mycompany.test-1.0.0-SNAPSHOT.jar

附件一:

package com.mycompany.test;


public class Start
{
    public static void main(final String[] args) 
    {
        System.out.println("If you are able to get this printed with java -jar you made it. Thanks a lot! :)");
    }
}

编辑 1:

我开始删除一些依赖项来识别问题。但是我还没有清楚地了解出了什么问题。

现在我发现(参见 pom.xml 中的提示 B)将 Apache POI 依赖项移到顶部将解决问题。但我仍然不知道为什么以及是什么问题。

【问题讨论】:

  • 有趣的错误包括 target/comcompany.product 而不是 target/com.company.product 还是只是输入问题的错字?
  • 这只是我的问题中的一个错字:(
  • 你能展示你拥有的其他依赖吗?我想用和你一样的设置进行测试,看看我是否也有同样的问题。
  • @maba 我已经添加了完整的 pom.xml

标签: java maven jfreechart maven-shade-plugin


【解决方案1】:

我使用您的设置运行了mvn dependency:tree,当您在依赖项声明中将org.apache.poi 向上移动时,它会有所不同。

这取自Introduction to the Dependency Mechanism关于依赖顺序:

依赖中介 - 这决定了当遇到工件的多个版本时将使用哪个版本的依赖。目前,Maven 2.0 仅支持使用“最近定义”,这意味着它将使用依赖树中与您的项目最接近的依赖版本。您始终可以通过在项目的 POM 中明确声明来保证版本。请注意,如果两个依赖版本在依赖树中的深度相同,则在 Maven 2.0.8 之前还没有定义哪个会获胜,但是从 Maven 2.0.9 开始,声明中的顺序很重要:第一个声明获胜

您的依赖解析似乎存在冲突,这导致您的 jar 文件损坏(不知道为什么它会损坏)。

无论如何,这是两个 pom 之间的差异(左边是原点,右边是 org.apache.poi 更高):

也许在图片上很难看到,但如果你放大你就会看到。

最大的区别在于,在非工作 pom 中,org.apache.httpcomponents:httpcore:jar:4.2.1 依赖于 commons-codec:commons-codec:jar:1.6,而在工作 pom 中,该依赖关系已被 commons-codec:commons-codec:jar:1.5 覆盖。

我猜commons-codec1.6 版本以及需要1.5 版本的org.apache.poi:poi:jar:3.8 存在问题。


编辑

this excellent answer 解释了 jar 文件损坏的原因(jar 中的条目太多)之后,我只想添加一个简单的解决方案,至少可以解决您的特定问题。

将标签 &lt;minimizeJar&gt;true&lt;/minimizeJar&gt; 添加到您的 maven-shade-plugin 配置中。

之后你的java -jar target/com.mycompany.test-1.0.0-SNAPSHOT.jar 命令行就可以工作了。

【讨论】:

  • 感谢您的帖子。正如您所提到的,如果有人能够弄清楚 jar 损坏的原因会很好:(。如果我没记错的话,也可以删除所有 Spring 依赖项以让它运行。但我目前不在我的电脑上进行测试它:(
【解决方案2】:

最后,如果我使用您向我们展示的pom.xml 构建您的项目,并在jfreechart 之后声明apache-poi,那么正如您所提到的,我得到了一个损坏的 JAR。交换这两个依赖项的顺序确实给了我一个正确的 JAR。

我以前对maven-shade-plugin 有过一些经验,当我使用它时,我在创建JAR 的META-INF 目录时遇到了问题,所以我检查了它是否存在缺陷。

我尝试在本地文件系统的某处复制(使用 Total Commander)META-INF 目录,我注意到在复制 META-INF/licences/ 中的文件时出现错误。如果我尝试将它们单独复制到某个地方并且它有效,但我无法复制整个内容。我的结论是 JAR/ZIP 存档已损坏。

我所做的是我在 Total Commander 中输入那个 JAR(Ctrl+PgDown over JAR 文件) 我 重命名 thirdpartylicenses.txtthirdpartylicenses.txt.wtf。执行此 Total Commander 可以保存此内容并重新打包 JAR(我已安装 Total7zip Total Commander 打包插件 - 如果有人尝试此操作但它不起作用,请尝试安装此插件)。

在这之后。它有效。

(我也尝试使用 Cygwin 的 unzip/zip 命令重新打包整个内容而不重命名任何内容,但这不起作用,新存档仍然损坏。Total Commander 或我提到的插件会变魔术。)

我猜maven-shade-plugin 只是创建了一个损坏或无效的 ZIP/JAR 存档。我不确定为什么,也许我所描述的内容对其他人不起作用,但我想我会提到它,所以也许我可以提供帮助。


我不能只管这个,所以我深入挖掘,我想我找到了答案。

bad JAR 包含 65608 个条目。 good JAR 包含 65450 个条目。

猜猜the number of entries for a plain ZIP file的上限是多少?是的。 Wiki 文章谈到了克服此限制的 ZIP64 格式。

good JAR 的条目较少,因为实际依赖项会因pom.xml 中依赖项声明的位置而改变。 (As described by this answer.)

我统计过这样的条目。

Collections.list(new JarFile("...").entries()).size();

我使用的是Java 7 which seems to support the new ZIP64 format,所以如果有人尝试使用 Java 5 或 6 计算 bad JAR 中的条目可能会收到错误(不过我不确定)。

我还尝试运行 JARexploded。我已经将整个 JAR 解压到一个目录中,然后像这样运行整个过程。

java -cp <dir/ com.mycompany.test.Start

工作就像一个魅力。


底线。不要过度使用maven-shade-plugin

我有一个正在工作的项目,我会像这样构建我的项目。

  • 我使用maven-dependency-plugin 复制我的项目的依赖项。查看copy-dependencies goal。如果我没记错的话,这会将您的依赖项复制到 target/dependency
  • 使用maven-jar-plugin,我使用插件配置中的这些选项将这些依赖项添加到我的最终JAR 的MANIFEST.MF 作为Class-Path 条目。

    <classpathPrefix>dependency/</classpathPrefix>
    <addClasspath>true</addClasspath>
    

    所以我会有Class-Path 条目,如dependency/&lt;artifactId&gt;-&lt;version&gt;.jar 等。

  • 在此之后,我使用 maven-assembly-plugin 创建一个分发 ZIP,其中包含我的最终 JAR 和整个 dependency/ 文件夹。
  • 部署应用程序时,我可以像 java -jar final.jar 一样运行它。

我主要选择使用此解决方案,因为在我的项目中,我使用了一些 Bouncy Castle JAR,这些 JAR 在其 META-INF 目录中有一些奢侈的 this 和 that。当我使用 maven-shade-plugin 创建我的最终可运行 JAR 时,一切都崩溃了,我得到了令人讨厌的找不到该方法并且此签名不太正确错误。

你也应该这样做。这个 Maven 遮阳业务太阴暗 有用(双关语)。


这是上面的a blog post on the whole process that I've tried to describe(感谢 baba),也许它会在未来帮助某人。

【讨论】:

  • 您在那儿做得非常好!我同意这是问题所在。 +1;
  • 解决方法:在 java 7 中,当条目数超过 2^16 时,它适用于“java -cp final.jar MainClass”,但不适用于“java -jar final.jar”。
  • 嘿,@Kohányi Róbert,我也有和你一样的结论,具有奇怪 META-INF 或奇怪资源文件夹的复杂多模块项目很难通过 shade 插件构建。这就是为什么我听从你的建议并使用 3 个简单的插件来构建 jar。我写了一篇关于它的博客文章:ath3nd.wordpress.com/2013/12/25/… 也许您可以使用每个插件的完整配置更新您的答案,以便帮助其他偶然发现该问题的人?
  • maven-assembly-plugin 一路走好!这是我个人最喜欢的打包插件! +1 答案,不知道 zip 在条目数量方面的限制!
猜你喜欢
  • 1970-01-01
  • 2020-03-21
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2016-12-01
  • 1970-01-01
  • 2021-12-28
  • 2015-08-20
相关资源
最近更新 更多