com.autonomousapps.internal.JarExploder.kt Maven / Gradle / Ivy
// Copyright (c) 2024. Tony Robalik.
// SPDX-License-Identifier: Apache-2.0
package com.autonomousapps.internal
import com.autonomousapps.internal.asm.ClassReader
import com.autonomousapps.internal.utils.asSequenceOfClassFiles
import com.autonomousapps.internal.utils.getLogger
import com.autonomousapps.model.KtFile
import com.autonomousapps.model.PhysicalArtifact
import com.autonomousapps.model.PhysicalArtifact.Mode
import com.autonomousapps.model.intermediates.AndroidLinterDependency
import com.autonomousapps.model.intermediates.ExplodedJar
import com.autonomousapps.model.intermediates.ExplodingJar
import com.autonomousapps.services.InMemoryCache
import com.autonomousapps.tasks.ExplodeJarTask
import java.util.zip.ZipFile
internal class JarExploder(
private val artifacts: List,
private val androidLinters: Set,
private val inMemoryCache: InMemoryCache
) {
private val logger = getLogger()
fun explodedJars(): Set {
return artifacts.asSequence()
.filter {
// We know how to analyze jars, and directories containing class files
it.isJar() || it.containsClassFiles()
}
.toExplodedJars()
}
private fun Sequence.toExplodedJars(): Set =
map { artifact ->
val explodedJar = if (artifact.isJar()) {
explode(artifact, Mode.ZIP)
} else {
explode(artifact, Mode.CLASSES)
}
ExplodedJar(
artifact = artifact,
exploding = explodedJar
)
}.toSortedSet()
/**
* Analyzes bytecode in order to extract class names and some basic structural information from
* the jar ([PhysicalArtifact.file]).
*
* With Gradle 8.0+, local java-library project dependencies are provided as a collection of class files rather than
* jars. It seems that the behavior when requesting the "android-classes" artifact view has changed (previously we'd
* get jars, but now we get class files).
*/
private fun explode(artifact: PhysicalArtifact, mode: Mode): ExplodingJar {
val entry = findInCache(artifact)
if (entry != null) return entry
val ktFiles: Set
val visitors = when (mode) {
Mode.ZIP -> {
val zip = ZipFile(artifact.file)
ktFiles = KtFile.fromZip(zip)
zip.asSequenceOfClassFiles()
.map { classEntry ->
ClassNameAndAnnotationsVisitor(logger).apply {
val reader = zip.getInputStream(classEntry).use { ClassReader(it.readBytes()) }
reader.accept(this, 0)
}
}
}
Mode.CLASSES -> {
ktFiles = KtFile.fromDirectory(artifact.file)
artifact.file.walkBottomUp()
.filter { it.isFile && it.name.endsWith(".class") }
.map { classFile ->
ClassNameAndAnnotationsVisitor(logger).apply {
val reader = classFile.inputStream().use { ClassReader(it.readBytes()) }
reader.accept(this, 0)
}
}
}
}
val analyzedClasses = visitors.map { it.getAnalyzedClass() }
.filterNot {
// Filter out `java` packages, but not `javax`
it.className.startsWith("java.")
}
.toSortedSet()
return ExplodingJar(
analyzedClasses = analyzedClasses,
ktFiles = ktFiles,
androidLintRegistry = findAndroidLinter(artifact)
).also { putInCache(artifact, it) }
}
private fun findInCache(artifact: PhysicalArtifact): ExplodingJar? {
return inMemoryCache.explodedJar(artifact.file.absolutePath)
}
private fun putInCache(artifact: PhysicalArtifact, explodingJar: ExplodingJar) {
inMemoryCache.explodedJars(artifact.file.absolutePath, explodingJar)
}
private fun findAndroidLinter(physicalArtifact: PhysicalArtifact): String? {
return androidLinters.find { it.coordinates == physicalArtifact.coordinates }?.lintRegistry
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy