web-dev-qa-db-ja.com

Mavenシェードプラグインによって構築された無効または破損したJARファイル

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

添付1:

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:

問題を特定するために、いくつかの依存関係を削除し始めました。しかし、何が悪いのか明確に理解できていません。

ここで、Apache POI依存関係を最上位に移動すると問題が解決することがわかりました(pom.xmlのヒントBを参照)。しかし、それでもどうして何が問題なのかわかりません。

34
d0x

最後に、pom.xmlを使用してプロジェクトをビルドし、Apache-poijfreechartの後に宣言すると、先ほど述べたように、破損したJARを取得します。これら2つの依存関係の順序を入れ替えると、実際に正しいJARが得られます。

以前にmaven-shade-pluginを使用した経験があり、使用したときに作成されたJARのMETA-INFディレクトリで問題が発生したため、欠陥がないか確認しました。

ローカルファイルシステム上のMETA-INFディレクトリを(Total Commanderを使用して)コピーしようとしましたが、META-INF/licences/のファイルのコピー中にエラーが発生したことに気付きました。それらを個別にどこかにコピーしようとしても機能したが、全部をコピーできなかった場合。私の結論は、JAR/Zipアーカイブが破損しているということでした。

私がやったのは、私がentered Total CommanderのJAR(Ctrl+PgDown over JARファイル)およびI renamedthirdpartylicenses.txt to thirdpartylicenses.txt.wtf。これを行うと、Total Commanderはこれを保存し、JARを再パッケージ化します(Total7Zip Total Commander packerプラグインがインストールされています-誰かがこれを試しても動作しない場合は、インストールされた状態で試してください)。

この後。できます。

(また、Cygwinのunzip/Zipコマンドを使用して、名前を変更せずに全体を再パックしようとしましたが、それでも動作しませんでした。新しいアーカイブはまだ破損していました。 「いくつかの魔法を行うと述べた。)

maven-shade-pluginは単に破損または無効なZip/JARアーカイブを作成するだけだと思います。なぜ私が説明したことが他の人には役に立たないのかわからないが、私はそれを言及するだろうと思ったので多分私は助けることができる。


これをそのままにしておくことはできなかったので、深く掘り下げて、答えを見つけたと思います。

bad JARには65608エントリが含まれています。 good JARには65450エントリが含まれています。

plain Zipファイルのエントリ数 ?の上限は何だと思いますか?うん。 Wikiの記事では、この制限を克服するZip64形式について説明しています。

good JARは、pom.xml内の依存関係宣言の位置により実際の依存関係が変化するため、エントリが少なくなります。 ( この回答で説明されているとおり。

このようなエントリを数えました。

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

新しいZip64形式をサポートしていると思われるJava 7 を使用していたため、誰かがJava 5を使用してbad JARのエントリをカウントしようとしたまたは、6がエラーを受け取ります(確かではありません)。

また、JAR explodedを実行しようとしました。 JAR全体をディレクトリに展開し、次のように全体を実行しました。

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

魅力のように働いた。


一番下の行。 使いすぎるmaven-shade-pluginをしないでください。

私はこのようなプロジェクトを構築する仕事をしています。

  • maven-dependency-plugin を使用して、プロジェクトの依存関係をコピーします。 copy-dependenciesゴール をチェックアウトします。私が正しく覚えていれば、これは依存関係をtarget/dependencyにコピーします。
  • maven-jar-plugin を使用して、プラグイン構成でこれらのオプションを使用して、これらの依存関係を最終的なJARのMANIFEST.MFClass-Pathエントリとして追加します。

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

    したがって、Class-Pathなどのdependency/<artifactId>-<version>.jarエントリがあります。

  • その後、 maven-Assembly-plugin を使用して、最終的なJARとdependency/フォルダー全体を含む配布Zipを作成します。
  • アプリケーションをデプロイすると、Java -jar final.jarのように実行できます。

私のプロジェクトでは、META-INFディレクトリ内にいくつかの贅沢なthis and thatが含まれるBouncy Castle JARを使用しているため、主にこのソリューションの使用を選択しました。 maven-shade-pluginを使用して最終的な実行可能なJARを作成すると、すべてが地獄に落ちて厄介になりましたそのメソッドが見つかりませんでした。このシグネチャは正しくありませんエラー。

このようなこともするべきです。このMavenシェードビジネスは、あまりにも便利ですshady.


私が説明しようとしたプロセス全体に関するブログ投稿 すぐ上( baba に感謝)、おそらく将来的に誰かを助けるでしょう。

45

セットアップでmvn dependency:treeを実行しましたが、依存関係宣言でorg.Apache.poiを上に移動すると違いが生じます。

これは Independduction to the Dependency Mechanism から依存関係の順序に関するものです:

依存性メディエーション-これにより、アーティファクトの複数のバージョンが検出されたときに使用される依存関係のバージョンが決まります。現在、Maven 2.0は「最も近い定義」の使用のみをサポートしています。つまり、依存関係のツリーでプロジェクトに最も近い依存関係のバージョンを使用します。プロジェクトのPOMで明示的に宣言することで、いつでもバージョンを保証できます。 2つの依存関係バージョンが依存関係ツリーの同じ深さにある場合、Maven 2.0.8まではどちらが勝つかが定義されていませんでしたが、、Maven 2.0.9以降はカウントする宣言:最初の宣言が勝ちます

依存関係の解決に競合があり、破損したjarファイルを引き起こしているようです(破損した理由がわかりません)。

とにかく、2つのpomの差分を次に示します(左はOrigin、右はorg.Apache.poiが上)。

enter image description hereenter image description here

写真で見るのは難しいかもしれませんが、ズームインすると表示されます。

大きな違いは、動作していないPOMではorg.Apache.httpcomponents:httpcore:jar:4.2.1commons-codec:commons-codec:jar:1.6に依存していることと、動作しているPOMではその依存関係がcommons-codec:commons-codec:jar:1.5でオーバーライドされていることです。

1.6バージョンを必要とするcommons-codecorg.Apache.poi:poi:jar:3.81.5バージョンに問題があると思います。


編集

この優れた答え の後に、jarファイルが壊れている理由(jarのエントリが多すぎる)を説明しています。少なくとも、特定の問題に対して機能する単純なソリューションを追加したいだけです。

タグ<minimizeJar>true</minimizeJar>maven-shade-plugin構成に追加します。

その後、Java -jar target/com.mycompany.test-1.0.0-SNAPSHOT.jarコマンドラインが機能します。

11
maba