Использование Gradle для создания банки с зависимостями

У меня есть сборка мультипроектов, и я поставил задачу построить живую банку в одном из подпроектов. Я создал задачу, подобную той, что описана в кулинарной книге Я также сообщил об этом в Gradle JIRA, если это ошибка.

Ответ 1

Я опубликовал решение в JIRA против Gradle:

// Include dependent libraries in archive.
mainClassName = "com.company.application.Main"

jar {
  manifest { 
    attributes "Main-Class": "$mainClassName"
  }  

  from {
    configurations.compile.collect { it.isDirectory() ? it : zipTree(it) }
  }
}

Обратите внимание, что mainClassName должно появиться ДО jar {.

Ответ 2

Если вы хотите, чтобы задача jar работала нормально, и дополнительная задача fatJar, используйте следующее:

task fatJar(type: Jar) {
    classifier = 'all'
    from { configurations.compile.collect { it.isDirectory() ? it : zipTree(it) } }
    with jar
}

Важная часть with jar. Без этого классы этого проекта не включены.

Ответ 3

Ответ @felix почти привел меня туда. У меня было две проблемы:

  1. В Gradle 1.5 тег манифеста не распознавался внутри задачи fatJar, поэтому атрибут Main-Class не мог быть установлен напрямую
  2. в банке были конфликтующие внешние файлы META-INF.

Следующая настройка решает эту проблему

jar {
  manifest {
    attributes(
      'Main-Class': 'my.project.main',
    )
  }
}

task fatJar(type: Jar) {
  manifest.from jar.manifest
  classifier = 'all'
  from {
    configurations.runtimeClasspath.collect { it.isDirectory() ? it : zipTree(it) }
  } {
    exclude "META-INF/*.SF"
    exclude "META-INF/*.DSA"
    exclude "META-INF/*.RSA"
  }
  with jar
}

Чтобы добавить это к стандартной задаче сборки или сборки, добавьте:

artifacts {
    archives fatJar
}

Изменение: благодаря @mjaggard: в последних версиях Gradle измените configurations.runtime на configurations.runtimeClasspath

Ответ 4

Это отлично работает для меня.

Мой основной класс:

package com.curso.online.gradle;

import org.apache.commons.lang3.StringUtils;
import org.apache.log4j.Logger;

public class Main {

    public static void main(String[] args) {
        Logger logger = Logger.getLogger(Main.class);
        logger.debug("Starting demo");

        String s = "Some Value";

        if (!StringUtils.isEmpty(s)) {
            System.out.println("Welcome ");
        }

        logger.debug("End of demo");
    }

}

И это содержимое моего файла build.gradle:

apply plugin: 'java'

apply plugin: 'eclipse'

repositories {
    mavenCentral()
}

dependencies {
    compile group: 'commons-collections', name: 'commons-collections', version: '3.2'
    testCompile group: 'junit', name: 'junit', version: '4.+'
    compile  'org.apache.commons:commons-lang3:3.0'
    compile  'log4j:log4j:1.2.16'
}

task fatJar(type: Jar) {
    manifest {
        attributes 'Main-Class': 'com.curso.online.gradle.Main'
    }
    baseName = project.name + '-all'
    from { configurations.compile.collect { it.isDirectory() ? it : zipTree(it) } }
    with jar
}

И я пишу следующее на моей консоли:

java -jar ProyectoEclipseTest-all.jar

И результат отличный:

log4j:WARN No appenders could be found for logger (com.curso.online.gradle.Main)
.
log4j:WARN Please initialize the log4j system properly.
log4j:WARN See http://logging.apache.org/log4j/1.2/faq.html#noconfig for more in
fo.
Welcome

Ответ 5

Чтобы генерировать полный JAR с основным исполняемым классом, избегая проблем с подписанными JAR, я предлагаю gradle -one-jar plugin. Простой плагин, который использует One-JAR project.

Прост в использовании:

apply plugin: 'gradle-one-jar'

buildscript {
    repositories {
        mavenCentral()
    }
    dependencies {
        classpath 'com.github.rholder:gradle-one-jar:1.0.4'
    }
}

task myjar(type: OneJar) {
    mainClass = 'com.benmccann.gradle.test.WebServer'
}

Ответ 6

Ответ от @ben почти работает для меня, за исключением того, что мои зависимости слишком велики, и я получил следующую ошибку

Execution failed for task ':jar'.
> archive contains more than 65535 entries.

  To build this archive, please enable the zip64 extension.

Чтобы решить эту проблему, я должен использовать следующий код

mainClassName = "com.company.application.Main"

jar {
  manifest { 
    attributes "Main-Class": "$mainClassName"
  }  
  zip64 = true
  from {
    configurations.compile.collect { it.isDirectory() ? it : zipTree(it) }
  }
}

Ответ 7

Простая сушка

jar {
    manifest {
        attributes 'Main-Class': 'cova2.Main'
    } 
    doFirst {
        from { configurations.runtime.collect { it.isDirectory() ? it : zipTree(it) } }
    }
}

Ответ 8

Если вы привыкли к ant, вы можете попробовать то же самое с Gradle тоже:

task bundlemyjava{
    ant.jar(destfile: "build/cookmyjar.jar"){
        fileset(dir:"path to your source", includes:'**/*.class,*.class', excludes:'if any')
        } 
}