proguard.gradle.plugin.android.ProGuardTransform.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of proguard-gradle Show documentation
Show all versions of proguard-gradle Show documentation
Gradle plugin for ProGuard, the free shrinker, optimizer, obfuscator, and preverifier for Java bytecode
The newest version!
/*
* ProGuard -- shrinking, optimization, obfuscation, and preverification
* of Java bytecode.
*
* Copyright (c) 2002-2021 Guardsquare NV
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the Free
* Software Foundation; either version 2 of the License, or (at your option)
* any later version.
*
* This program is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
* more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
package proguard.gradle.plugin.android
import com.android.build.api.transform.Format
import com.android.build.api.transform.Format.DIRECTORY
import com.android.build.api.transform.Format.JAR
import com.android.build.api.transform.QualifiedContent
import com.android.build.api.transform.QualifiedContent.DefaultContentType
import com.android.build.api.transform.QualifiedContent.DefaultContentType.CLASSES
import com.android.build.api.transform.QualifiedContent.DefaultContentType.RESOURCES
import com.android.build.api.transform.QualifiedContent.Scope
import com.android.build.api.transform.QualifiedContent.Scope.EXTERNAL_LIBRARIES
import com.android.build.api.transform.QualifiedContent.Scope.PROJECT
import com.android.build.api.transform.QualifiedContent.Scope.PROVIDED_ONLY
import com.android.build.api.transform.QualifiedContent.Scope.SUB_PROJECTS
import com.android.build.api.transform.SecondaryFile
import com.android.build.api.transform.Transform
import com.android.build.api.transform.TransformInput
import com.android.build.api.transform.TransformInvocation
import com.android.build.api.transform.TransformOutputProvider
import com.android.build.api.variant.VariantInfo
import com.android.build.gradle.BaseExtension
import org.gradle.api.Project
import proguard.gradle.ProGuardTask
import proguard.gradle.plugin.android.AndroidPlugin.Companion.COLLECT_CONSUMER_RULES_TASK_NAME
import proguard.gradle.plugin.android.AndroidProjectType.ANDROID_APPLICATION
import proguard.gradle.plugin.android.AndroidProjectType.ANDROID_LIBRARY
import proguard.gradle.plugin.android.dsl.ProGuardAndroidExtension
import proguard.gradle.plugin.android.dsl.UserProGuardConfiguration
import java.io.File
class ProGuardTransform(
private val project: Project,
private val proguardBlock: ProGuardAndroidExtension,
private val projectType: AndroidProjectType,
private val androidExtension: BaseExtension,
) : Transform() {
override fun transform(transformInvocation: TransformInvocation) {
val variantName: String = transformInvocation.context.variantName
val variantBlock =
proguardBlock.configurations.findVariantConfiguration(variantName)
?: throw RuntimeException("Invalid configuration: $variantName")
val proguardTask = project.tasks.create("proguardTask${variantName.capitalize()}", ProGuardTask::class.java)
createIOEntries(transformInvocation.inputs, transformInvocation.outputProvider).forEach {
proguardTask.injars(it.first)
proguardTask.outjars(it.second)
}
proguardTask.extraJar(
transformInvocation
.outputProvider
.getContentLocation("extra.jar", setOf(CLASSES, RESOURCES), mutableSetOf(PROJECT), JAR),
)
proguardTask.libraryjars(createLibraryJars(transformInvocation.referencedInputs))
proguardTask.configuration(project.tasks.getByPath(COLLECT_CONSUMER_RULES_TASK_NAME + variantName.capitalize()).outputs.files)
proguardTask.configuration(variantBlock.configurations.map { project.file(it.path) })
val aaptRulesFile = getAaptRulesFile()
if (aaptRulesFile != null && File(aaptRulesFile).exists()) {
proguardTask.configuration(aaptRulesFile)
} else {
project.logger.warn(
"AAPT rules file not found: you may need to apply some extra keep rules for classes referenced from " +
"resources in your own ProGuard configuration.",
)
}
val mappingDir = project.buildDir.resolve("outputs/proguard/$variantName/mapping")
if (!mappingDir.exists()) mappingDir.mkdirs()
proguardTask.printmapping(File(mappingDir, "mapping.txt"))
proguardTask.printseeds(File(mappingDir, "seeds.txt"))
proguardTask.printusage(File(mappingDir, "usage.txt"))
proguardTask.android()
proguardTask.proguard()
}
override fun getName(): String = "ProguardTransform"
override fun getInputTypes(): Set = setOf(CLASSES, RESOURCES)
override fun getScopes(): MutableSet =
when (projectType) {
ANDROID_APPLICATION -> mutableSetOf(PROJECT, SUB_PROJECTS, EXTERNAL_LIBRARIES)
ANDROID_LIBRARY -> mutableSetOf(PROJECT)
}
override fun getReferencedScopes(): MutableSet =
when (projectType) {
ANDROID_APPLICATION -> mutableSetOf(PROVIDED_ONLY)
ANDROID_LIBRARY -> mutableSetOf(PROVIDED_ONLY, EXTERNAL_LIBRARIES, SUB_PROJECTS)
}
override fun isIncremental(): Boolean = false
override fun applyToVariant(variant: VariantInfo?): Boolean =
variant?.let { proguardBlock.configurations.findVariantConfiguration(it) } != null
override fun getSecondaryFiles(): MutableCollection =
proguardBlock
.configurations
.flatMap { it.configurations }
.filterIsInstance()
.map { SecondaryFile(project.file(it.path), false) }
.toMutableSet().apply {
getAaptRulesFile()?.let { this.add(SecondaryFile(project.file(it), false)) }
}
private fun createIOEntries(
inputs: Collection,
outputProvider: TransformOutputProvider,
): List {
fun createEntry(
input: QualifiedContent,
format: Format,
): ProGuardIOEntry {
return ProGuardIOEntry(
input.file,
outputProvider.getContentLocation(input.name, input.contentTypes, input.scopes, format).canonicalFile,
)
}
return inputs.flatMap { input ->
input.directoryInputs.map { createEntry(it, DIRECTORY) } + input.jarInputs.map { createEntry(it, JAR) }
}
}
private fun createLibraryJars(inputs: Collection): List =
inputs.flatMap { input -> input.directoryInputs.map { it.file } + input.jarInputs.map { it.file } } +
listOf(androidExtension.sdkDirectory.resolve("platforms/${androidExtension.compileSdkVersion}/android.jar")) +
androidExtension.libraryRequests.map {
androidExtension.sdkDirectory.resolve("platforms/${androidExtension.compileSdkVersion}/optional/${it.name}.jar")
}
private fun getAaptRulesFile() =
androidExtension.aaptAdditionalParameters
.zipWithNext { cmd, param -> if (cmd == "--proguard") param else null }
.filterNotNull()
.firstOrNull { File(it).exists() }
}
typealias ProGuardIOEntry = Pair
© 2015 - 2025 Weber Informatics LLC | Privacy Policy