Costruire vaso eseguibile con Maven?


122

Sto cercando di generare un jar eseguibile per un piccolo progetto domestico chiamato "logmanager" usando Maven, proprio come questo:

Come posso creare un JAR eseguibile con dipendenze utilizzando Maven?

Ho aggiunto lo snippet mostrato lì al pom.xml e ho eseguito mvn assembly: assembly. Genera due file jar in logmanager / target: logmanager-0.1.0.jar e logmanager-0.1.0-jar-with-dependencies.jar. Ricevo un errore quando faccio doppio clic sul primo barattolo:

Could not find the main class: com.gorkwobble.logmanager.LogManager. Program will exit.

Un errore leggermente diverso quando faccio doppio clic su jar-with-dependencies.jar:

Failed to load Main-Class manifest attribute from: C:\EclipseProjects\logmanager\target\logmanager-0.1.0-jar-with-dependencies.jar

Ho copiato e incollato il percorso e il nome della classe e ho controllato l'ortografia nel POM. La mia classe principale si avvia bene da una configurazione di lancio di Eclipse. Qualcuno può aiutarmi a capire perché il mio file jar non funziona? Inoltre, perché ci sono due barattoli per cominciare? Fammi sapere se hai bisogno di maggiori informazioni.

Ecco il completo pom.xml, per riferimento:

<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.gorkwobble</groupId>
  <artifactId>logmanager</artifactId>
  <name>LogManager</name>
  <version>0.1.0</version>
  <description>Systematically renames specified log files on a scheduled basis. Designed to help manage MUSHClient logging and prevent long, continuous log files.</description>
  <build>
    <plugins>
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-jar-plugin</artifactId>
            <version>2.2</version>
            <!-- nothing here -->
        </plugin>
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-assembly-plugin</artifactId>
            <version>2.2-beta-4</version>
            <configuration>
              <descriptorRefs>
                <descriptorRef>jar-with-dependencies</descriptorRef>
              </descriptorRefs>
              <archive>
                <manifest>
                  <mainClass>com.gorkwobble.logmanager.LogManager</mainClass>
                </manifest>
              </archive>
            </configuration>
            <executions>
              <execution>
                <phase>package</phase>
                <goals>
                  <goal>single</goal>
                </goals>
              </execution>
            </executions>
          </plugin>
          <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-compiler-plugin</artifactId>
            <configuration>
              <source>1.6</source>
              <target>1.6</target>
            </configuration>
          </plugin>
    </plugins>
  </build>
  <dependencies>
    <!-- commons-lang -->
    <dependency>
        <groupId>commons-lang</groupId>
        <artifactId>commons-lang</artifactId>
        <version>2.4</version>
    </dependency> 

    <!-- Quartz scheduler -->
    <dependency>
        <groupId>opensymphony</groupId>
        <artifactId>quartz</artifactId>
        <version>1.6.3</version>
    </dependency>
    <!-- Quartz 1.6.0 depends on commons collections -->
    <dependency>
      <groupId>commons-collections</groupId>
      <artifactId>commons-collections</artifactId>
      <version>3.1</version>
    </dependency>
    <!-- Quartz 1.6.0 depends on commons logging -->
    <dependency>
      <groupId>commons-logging</groupId>
      <artifactId>commons-logging</artifactId>
      <version>1.1</version>
    </dependency>
    <!-- Quartz 1.6.0 requires JTA in non J2EE environments -->
    <dependency>
      <groupId>javax.transaction</groupId>
      <artifactId>jta</artifactId>
      <version>1.1</version>
      <scope>runtime</scope>
    </dependency>

    <!-- junitx test assertions -->
    <dependency>
        <groupId>junit-addons</groupId>
        <artifactId>junit-addons</artifactId>
        <version>1.4</version>
        <scope>test</scope>
    </dependency>

    <!-- junit dependency; FIXME: make this a separate POM -->
    <dependency>
      <groupId>junit</groupId>
      <artifactId>junit</artifactId>
      <version>4.1</version>
    </dependency>

  </dependencies>
  <dependencyManagement>
  </dependencyManagement>
</project>

Risposte:


244

In realtà, penso che la risposta data nella domanda che hai citato sia semplicemente sbagliata ( UPDATE - 20101106: qualcuno l'ha aggiustata, questa risposta si riferisce alla versione precedente la modifica ) e questo spiega, almeno in parte, perché incappi nei guai.


Genera due file jar in logmanager / target: logmanager-0.1.0.jar e logmanager-0.1.0-jar-with-dependencies.jar.

Il primo è il JAR del modulo logmanager generato durante la packagefase da jar:jar(perché il modulo ha un packaging di tipo jar). Il secondo è l'assembly generato da assembly:assemblye dovrebbe contenere le classi del modulo corrente e le sue dipendenze (se hai usato il descrittore jar-with-dependencies).

Ricevo un errore quando faccio doppio clic sul primo barattolo:

Could not find the main class: com.gorkwobble.logmanager.LogManager. Program will exit.

Se hai applicato la configurazione suggerita del collegamento pubblicato come riferimento, hai configurato il plugin jar per produrre un artefatto eseguibile, qualcosa del genere:

  <plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-jar-plugin</artifactId>
    <configuration>
      <archive>
        <manifest>
          <addClasspath>true</addClasspath>
          <mainClass>com.gorkwobble.logmanager.LogManager</mainClass>
        </manifest>
      </archive>
    </configuration>
  </plugin>

Quindi logmanager-0.1.0.jarè effettivamente eseguibile ma 1. questo non è quello che vuoi (perché non ha tutte le dipendenze) e 2. non contiene com.gorkwobble.logmanager.LogManager(questo è ciò che sta dicendo l'errore, controlla il contenuto del vaso).

Un errore leggermente diverso quando faccio doppio clic su jar-with-dependencies.jar:

Failed to load Main-Class manifest attribute from: C:\EclipseProjects\logmanager\target\logmanager-0.1.0-jar-with-dependencies.jar

Di nuovo, se hai configurato il plug-in assembly come suggerito, hai qualcosa del genere:

  <plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-assembly-plugin</artifactId>
    <configuration>
      <descriptorRefs>
        <descriptorRef>jar-with-dependencies</descriptorRef>
      </descriptorRefs>
    </configuration>
  </plugin>

Con questa configurazione, logmanager-0.1.0-jar-with-dependencies.jarcontiene le classi del modulo corrente e le sue dipendenze ma, in base all'errore, META-INF/MANIFEST.MF non contiene una Main-Classvoce (probabilmente non è lo stesso MANIFEST.MF di logmanager-0.1.0.jar). Il jar in realtà non è eseguibile, il che di nuovo non è quello che vuoi.


Quindi, il mio suggerimento sarebbe quello di rimuovere l' configurationelemento dal plug-in maven-jar-plug e di configurare il plug-in di assemblaggio maven in questo modo:

  <plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-jar-plugin</artifactId>
    <version>2.2</version>
    <!-- nothing here -->
  </plugin>
  <plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-assembly-plugin</artifactId>
    <version>2.2-beta-4</version>
    <configuration>
      <descriptorRefs>
        <descriptorRef>jar-with-dependencies</descriptorRef>
      </descriptorRefs>
      <archive>
        <manifest>
          <mainClass>org.sample.App</mainClass>
        </manifest>
      </archive>
    </configuration>
    <executions>
      <execution>
        <phase>package</phase>
        <goals>
          <goal>single</goal>
        </goals>
      </execution>
    </executions>
  </plugin>

Ovviamente, sostituisci org.sample.Appcon la classe che desideri eseguire. Piccolo bonus, sono vincolato assembly:singlealla packagefase quindi non devi assembly:assemblypiù correre . Basta correre mvn installe l'assemblaggio verrà prodotto durante la build standard.

Quindi, aggiorna il tuo pom.xml con la configurazione data sopra ed esegui mvn clean install. Quindi, cd nella targetdirectory e riprova:

java -jar logmanager-0.1.0-jar-with-dependencies.jar

Se ricevi un errore, aggiorna la tua domanda con essa e pubblica il contenuto del META-INF/MANIFEST.MFfile e la parte pertinente del tuo pom.xml(le parti di configurazione dei plugin). Inoltre si prega di pubblicare il risultato di:

java -cp logmanager-0.1.0-jar-with-dependencies.jar com.gorkwobble.logmanager.LogManager

per dimostrare che funziona bene sulla riga di comando (indipendentemente da ciò che sta dicendo eclipse).

EDIT: per Java 6, è necessario configurare il plug-in del compilatore maven. Aggiungi questo al tuo pom.xml:

  <plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-compiler-plugin</artifactId>
    <configuration>
      <source>1.6</source>
      <target>1.6</target>
    </configuration>
  </plugin>

4
Grazie per i tuoi commenti! Ho cambiato il mio pom.xml come hai specificato. Quando eseguo mvn clean install, ottengo un sacco di errori di compilazione dal mio codice, dicendo che le annotazioni e così via non sono supportate in -source 1.3. Sto usando jdk1.6 e si compila in eclipse; Non sono sicuro di come sia stato introdotto il 1.3. Forse una delle versioni della libreria nello snippet del pom è più vecchia?
RMorrisey

Grazie! Ho superato il problema 1.3. Ho anche dovuto aggiungere la dipendenza junit4 al mio POM. Sto lavorando alla risoluzione dei problemi di altre cose; se rimango bloccato posterò di nuovo! Se faccio funzionare il barattolo, lo contrassegno come risposta. Il mio attuale POM è aggiornato nella domanda sopra.
RMorrisey

C'è un modo per escludere le risorse dal file jar generato?

2
È possibile che il jar risultante abbia un nome "normale"?
Daniil Shevelev,

1
Ho trovato anche questo Sonatype post del blog utile

15

La risposta di Pascal Thivent ha aiutato anche me. Ma se gestisci i tuoi plugin all'interno <pluginManagement>dell'elemento, devi definire di nuovo l'assembly al di fuori della gestione dei plugin, altrimenti le dipendenze non vengono impacchettate nel jar se esegui mvn install.

<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>
    <version>1.0.0-SNAPSHOT</version>
    <packaging>jar</packaging>


    <build>
        <pluginManagement>
            <plugins>

                <plugin>
                    <groupId>org.apache.maven.plugins</groupId>
                    <artifactId>maven-compiler-plugin</artifactId>
                    <version>3.1</version>
                    <configuration>
                        <source>1.6</source>
                        <target>1.6</target>
                    </configuration>
                </plugin>

                <plugin>
                    <groupId>org.apache.maven.plugins</groupId>
                    <artifactId>maven-assembly-plugin</artifactId>
                    <version>2.4</version>
                    <configuration>
                        <archive>
                            <manifest>
                                <mainClass>main.App</mainClass>
                            </manifest>
                        </archive>
                        <descriptorRefs>
                            <descriptorRef>jar-with-dependencies</descriptorRef>
                        </descriptorRefs>
                    </configuration>
                    <executions>
                        <execution>
                            <id>make-assembly</id>
                            <phase>package</phase>
                            <goals>
                                <goal>single</goal>
                            </goals>
                        </execution>
                    </executions>
                </plugin>

            </plugins>

        </pluginManagement>

        <plugins> <!-- did NOT work without this  -->
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-assembly-plugin</artifactId>
            </plugin>
        </plugins>

    </build>


    <dependencies>
       <!--  dependencies commented out to shorten example -->
    </dependencies>

</project>

1
Grazie Mike, mi ha aiutato. Inizialmente il mio pacchetto veniva generato senza utilizzare <pluginManagement>. Ma Eclipse dava qualche errore in pom.xml "maven - L'esecuzione del plugin non è coperta dal ciclo di vita". Che mi distrae. Quindi per risolvere questo problema ho aggiunto <pluginManagement>, ora l'errore di eclipse è sparito ma il mio pacchetto ha smesso di essere compilato. Il tuo frammento di pom sopra ha funzionato per me. :)
shashaDenovo

2
Questo è stato utile .. durante l'utilizzo di <pluginManagement>, la risposta migliore non funzionerà.
ininprsr

5

Se non vuoi eseguire l'obiettivo di assemblaggio sul pacchetto, puoi usare il comando successivo:

mvn package assembly:single

Qui il pacchetto è la parola chiave.


-1

Fare clic con il pulsante destro del mouse sul progetto e dare a maven build, maven clean, maven generate resource e maven install.Il file jar verrà generato automaticamente.

Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.