Construindo jar executável com maven?

122

Estou tentando gerar um jar executável para um pequeno projeto doméstico chamado "logmanager" usando o maven, assim:

Como posso criar um JAR executável com dependências usando o Maven?

Adicionei o trecho mostrado lá ao pom.xml e executei o mvn assembly: assembly. Ele gera dois arquivos jar no logmanager / target: logmanager-0.1.0.jar e logmanager-0.1.0-jar-with-dependencies.jar. Eu recebo um erro ao clicar duas vezes no primeiro frasco:

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

Um erro ligeiramente diferente quando clico duas vezes no jar-with-dependencies.jar:

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

Copiei e colei o caminho e o nome da classe e verifiquei a ortografia no POM. Minha classe principal inicia bem a partir de uma configuração de inicialização do eclipse. Alguém pode me ajudar a descobrir por que meu arquivo jar não será executado? Além disso, por que existem dois jarros para começar? Entre em contato se precisar de mais informações.

Aqui está o completo pom.xml, para referência:

<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>
RMorrisey
fonte

Respostas:

244

Na verdade, acho que a resposta dada na pergunta que você mencionou está errada ( UPDATE - 20101106: alguém consertou, isso resposta se refere à versão anterior à edição ) e isso explica, pelo menos parcialmente, por que você encontra problemas.


Ele gera dois arquivos jar no logmanager / target: logmanager-0.1.0.jar e logmanager-0.1.0-jar-with-dependencies.jar.

O primeiro é o JAR do módulo gerenciador de log gerado durante a packagefase por jar:jar(porque o módulo possui uma embalagem do tipo jar). O segundo é a montagem gerada peloassembly:assembly e deve conter as classes do módulo atual e suas dependências (se você usou o descritor jar-with-dependencies).

Eu recebo um erro ao clicar duas vezes no primeiro frasco:

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

Se você aplicou a configuração sugerida do link publicado como referência, configurou o plug-in jar para produzir um artefato executável, algo como isto:

  <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>

Portanto, logmanager-0.1.0.jaré realmente executável, mas 1. não é isso que você deseja (porque não possui todas as dependências) e 2. não contémcom.gorkwobble.logmanager.LogManager (é o que o erro está dizendo, verifique o conteúdo do jar).

Um erro ligeiramente diferente quando clico duas vezes no jar-with-dependencies.jar:

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

Novamente, se você configurou o plug-in de montagem conforme sugerido, terá algo parecido com isto:

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

Com essa configuração, logmanager-0.1.0-jar-with-dependencies.jarcontém as classes do módulo atual e suas dependências, mas, de acordo com o erro, META-INF/MANIFEST.MF ele não contém uma Main-Classentrada (provavelmente não é o mesmo MANIFEST.MF do logmanager-0.1.0.jar). O frasco é realmente não executável, o que novamente não é o que você deseja.


Portanto, minha sugestão seria remover o configurationelemento do maven-jar-plugin e configurar o maven-assembly-plugin assim:

  <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>

Obviamente, substitua org.sample.Apppela classe que você deseja executar. Pouco bônus, eu estou ligado assembly:singleà packagefase para que você não precise correrassembly:assembly mais . Basta executar mvn installe a montagem será produzida durante a compilação padrão.

Portanto, atualize seu pom.xml com a configuração fornecida acima e execute mvn clean install. Em seguida, cd no targetdiretório e tente novamente:

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

Se você receber um erro, atualize sua pergunta e publique o conteúdo do META-INF/MANIFEST.MFarquivo e a parte relevante do seu pom.xml(as partes de configuração dos plug-ins). Publique também o resultado de:

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

para demonstrar que está funcionando bem na linha de comando (independentemente do que o eclipse esteja dizendo).

EDIT: Para Java 6, você precisa configurar o maven-compiler-plugin. Adicione isso ao seu 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>
Pascal Thivent
fonte
4
Obrigado por seus comentários! Mudei meu pom.xml conforme você especificou. Quando executo o mvn clean install, recebo vários erros de compilação do meu código, dizendo que anotações e assim por diante não são suportadas no -source 1.3. Estou usando o jdk1.6 e ele compila no eclipse; Não tenho certeza de como o 1.3 foi introduzido. Talvez uma das versões da biblioteca no trecho pom seja uma mais antiga?
RMorrisey
Obrigado! Passei a questão 1.3. Eu também tive que adicionar a dependência junit4 ao meu POM. Estou trabalhando na solução de outros problemas; se eu ficar preso vou postar de novo! Se eu colocar o jarro em funcionamento, vou marcar isso como resposta. Meu POM atual é atualizado na pergunta acima.
RMorrisey
Existe alguma maneira de excluir os recursos do arquivo jar gerado?
2
É possível que o jar resultante tenha um nome "normal"?
Daniil Shevelev
1
Eu também achei este post do Sonatype útil
15

A resposta de Pascal Thivent também me ajudou. Mas se você gerenciar seus plug-ins dentro do <pluginManagement>elemento, precisará definir o assembly novamente fora do gerenciamento de plug-ins, caso contrário, as dependências não serão compactadas no jar se você executar 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>
Mike
fonte
1
Obrigado Mike, Isso me ajudou. Inicialmente, meu pacote estava sendo gerado sem o uso de <pluginManagement>. Mas o Eclipse estava dando algum erro no pom.xml "maven - Execução de plug-in não coberta pelo ciclo de vida". O que me distrai. Então, para resolver isso, adicionei <pluginManagement>, agora o erro do eclipse desapareceu, mas meu pacote parou de ser compilado. Seu trecho de pom acima funcionou para mim. :)
shashaDenovo
2
Isso foi útil. Ao usar <pluginManagement>, a resposta principal não funcionará.
ininprsr
5

Se você não deseja executar a meta de montagem no pacote, pode usar o próximo comando:

mvn package assembly:single

Aqui pacote é palavra-chave.

leonidv
fonte
-1

Clique com o botão direito do mouse no projeto e forneça o maven build, o maven clean, o maven gerar recursos e o maven install. O arquivo jar será gerado automaticamente.

boneca de boneca
fonte