Android-библиотека Gradle релиз JAR

Как я могу освободить упаковку Jar проекта android-library?
Я нашел, classes.jar находится под build/bundles/release/classes.jar, и я полагаю, что это правильный пакет Jar (содержит *.class файлы).

Есть ли какой-то официальный способ, чтобы выпустить библиотеку как JAR вместо AAR?

Edit
Я использую Gradle для выпуска артефактов Maven, и я хотел бы выпустить JAR вместе с пакетом AAR. Итак, JAR с подписью, md5, манифест,...
на основе https://chris.banes.me/2013/08/27/pushing-aars-to-maven-central/

apply plugin: 'maven'
apply plugin: 'signing'

configurations {
    archives {
        extendsFrom configurations.default
    }
}

def sonatypeRepositoryUrl
if (isReleaseBuild()) {
    println 'RELEASE BUILD'
    sonatypeRepositoryUrl = "https://oss.sonatype.org/service/local/staging/deploy/maven2/"
} else {
    println 'DEBUG BUILD'
    sonatypeRepositoryUrl = "https://oss.sonatype.org/content/repositories/snapshots/"
}

if(!hasProperty('nexusPassword')) {
    ext.set('nexusPassword', System.console().readPassword("\n\$ Type in password for Sonatype nexus account " + nexusUsername + ": "))
}

if(!signing.hasProperty('password')) {
    ext.set('signing.password', System.console().readPassword("\n\$ Type in GPG key password: "))
}

afterEvaluate { project ->
    uploadArchives {
        repositories {
            mavenDeployer {
                beforeDeployment { MavenDeployment deployment -> signing.signPom(deployment) }

                pom.artifactId = POM_ARTIFACT_ID

                repository(url: sonatypeRepositoryUrl) {
                    authentication(userName: nexusUsername, password: nexusPassword)
                }

                pom.project {
                    name POM_NAME
                    packaging POM_PACKAGING
                    description POM_DESCRIPTION
                    url POM_URL

                    scm {
                        url POM_SCM_URL
                        connection POM_SCM_CONNECTION
                        developerConnection POM_SCM_DEV_CONNECTION
                    }

                    licenses {
                        license {
                            name POM_LICENCE_NAME
                            url POM_LICENCE_URL
                            distribution POM_LICENCE_DIST
                        }
                    }

                    developers {
                        developer {
                            id "loopj"
                            name "James Smith"
                        }
                        developer {
                            id "smarek"
                            name "Marek Sebera"
                        }
                    }
                }
            }
        }
    }

    signing {
        required { isReleaseBuild() && gradle.taskGraph.hasTask("uploadArchives") }
        sign configurations.archives
    }

    task androidJavadocs(type: Javadoc) {
        source = android.sourceSets.main.java.srcDirs
    }

    task androidJavadocsJar(type: Jar) {
        classifier = 'javadoc'
        from androidJavadocs.destinationDir
    }

    task androidSourcesJar(type: Jar) {
        classifier = 'sources'
        from android.sourceSets.main.java.srcDirs
    }

    artifacts {
        archives androidSourcesJar
        archives androidJavadocsJar
    }
}

используя

task androidJar(type: Jar) {
    from android.sourceSets.main.java.srcDirs
}

будет упаковывать только java файлы, не скомпилированные и связанные с android sdk

Ответ 1

Пока я не пытался загружать артефакты с развертыванием в Sonatype (или даже локальное репо), вот что мне удалось придумать несколько недель назад при попытке решить ту же проблему.

android.libraryVariants.all { variant ->
  def name = variant.buildType.name
  if (name.equals(com.android.builder.core.BuilderConstants.DEBUG)) {
    return; // Skip debug builds.
  }
  def task = project.tasks.create "jar${name.capitalize()}", Jar
  task.dependsOn variant.javaCompile
  task.from variant.javaCompile.destinationDir
  artifacts.add('archives', task);
}

Затем запустите следующее:

./gradlew jarRelease

Ответ 2

Еще один способ генерации jar из проекта библиотеки через gradle выглядит следующим образом:

В вашей библиотеке build.gradle:

def jarName = 'someJarName.jar'

task clearJar(type: Delete) {
    delete "${project.buildDir}/libs/" + jarName
}

task makeJar(type: Copy) {
    from("${project.buildDir}/intermediates/bundles/release/")
    into("${project.buildDir}/libs/")
    include('classes.jar')
    rename('classes.jar', jarName)
}

makeJar.dependsOn(clearJar, build)

То, что мы здесь делаем, это просто копирование классов .jar, сгенерированных плагином Android gradle. Обязательно загляните в свой каталог сборки для этого файла и посмотрите, находится ли его содержимое так, как вы хотите.

Затем запустите задачу makeJar, и полученный jar будет в библиотеке /build/libs/ ${jarName}.jar

У вас будут файлы классов в соответствии с вашей конфигурацией для выпуска. Если вы запутываете его, тогда файлы в банке будут запутаны.

Ответ 3

Просто потому, что предыдущие ответы не соответствовали моим потребностям, я делюсь своей сложной версией, чтобы создать банку с классами проекта и зависимостями aar/jar.

// A tricky jar operation, we want to generate a jar files that contains only the required classes used.
// Dependencies are in jar and aar, so we need to open the aar to extract the classes and put all at the same level
// (aar is a zip that contains classes.jar, the latter is a zip that contains .class files)
task jar(type: Jar) {
    from {
        List<File> allFiles = new ArrayList<>();
        configurations.compile.collect {
            for (File f : zipTree(it).getFiles()) {
                if (f.getName().equals("classes.jar")) {
                    allFiles.addAll(zipTree(f).getAt("asFileTrees").get(0).getDir())
                }
            }
        }
        allFiles.add(new File('build/intermediates/classes/release'))
        allFiles // To return the result inside a lambda
    }
    archiveName( project.ext.appName + '.jar' )
}

Это НЕ управляет типами/ароматами сборки, но может быть адаптировано (он просто подходит для сборки без релиза).

Если у вас есть более умное или более элегантное решение, пожалуйста, поделитесь!