com.autonomousapps.subplugin.ProjectPlugin.kt Maven / Gradle / Ivy
// Copyright (c) 2024. Tony Robalik.
// SPDX-License-Identifier: Apache-2.0
package com.autonomousapps.subplugin
import com.android.build.api.artifact.Artifacts
import com.android.build.api.dsl.CommonExtension
import com.android.build.api.variant.AndroidComponentsExtension
import com.android.build.api.variant.HasAndroidTest
import com.android.build.api.variant.Sources
import com.autonomousapps.AbstractExtension
import com.autonomousapps.DependencyAnalysisExtension
import com.autonomousapps.DependencyAnalysisSubExtension
import com.autonomousapps.Flags.androidIgnoredVariants
import com.autonomousapps.Flags.projectPathRegex
import com.autonomousapps.Flags.shouldAnalyzeTests
import com.autonomousapps.internal.*
import com.autonomousapps.internal.GradleVersions.isAtLeastGradle82
import com.autonomousapps.internal.advice.DslKind
import com.autonomousapps.internal.analyzer.*
import com.autonomousapps.internal.android.AgpVersion
import com.autonomousapps.internal.artifacts.DagpArtifacts
import com.autonomousapps.internal.artifacts.Publisher.Companion.interProjectPublisher
import com.autonomousapps.internal.utils.addAll
import com.autonomousapps.internal.utils.log
import com.autonomousapps.internal.utils.toJson
import com.autonomousapps.model.declaration.SourceSetKind
import com.autonomousapps.model.declaration.Variant
import com.autonomousapps.services.GlobalDslService
import com.autonomousapps.services.InMemoryCache
import com.autonomousapps.tasks.*
import org.gradle.api.NamedDomainObjectSet
import org.gradle.api.Project
import org.gradle.api.UnknownTaskException
import org.gradle.api.artifacts.component.ProjectComponentIdentifier
import org.gradle.api.file.RegularFile
import org.gradle.api.provider.Provider
import org.gradle.api.tasks.SourceSet
import org.gradle.api.tasks.SourceSetContainer
import org.gradle.api.tasks.TaskProvider
import org.gradle.kotlin.dsl.get
import org.gradle.kotlin.dsl.property
import org.gradle.kotlin.dsl.register
import org.gradle.kotlin.dsl.the
import org.jetbrains.kotlin.gradle.dsl.KotlinProjectExtension
import java.util.concurrent.atomic.AtomicBoolean
private const val APPLICATION_PLUGIN = "application"
private const val JAVA_LIBRARY_PLUGIN = "java-library"
private const val JAVA_PLUGIN = "java"
private const val SCALA_PLUGIN = "scala"
private const val ANDROID_APP_PLUGIN = "com.android.application"
private const val ANDROID_LIBRARY_PLUGIN = "com.android.library"
private const val KOTLIN_ANDROID_PLUGIN = "org.jetbrains.kotlin.android"
private const val KOTLIN_JVM_PLUGIN = "org.jetbrains.kotlin.jvm"
private const val GRETTY_PLUGIN = "org.gretty"
private const val SPRING_BOOT_PLUGIN = "org.springframework.boot"
/** This "plugin" is applied to every project in a build. */
internal class ProjectPlugin(private val project: Project) {
private val dagpExtension: AbstractExtension = if (project == project.rootProject) {
project.extensions.getByType(DependencyAnalysisExtension::class.java)
} else {
DependencyAnalysisSubExtension.of(project)
}
/**
* Used as a gate to prevent this plugin from configuring a project more than once. If ever
* checked and the value is already `true`, creates and configures the [RedundantJvmPlugin].
*/
private val configuredForKotlinJvmOrJavaLibrary = AtomicBoolean(false)
/**
* Used as a gate to prevent this plugin from configuring an app project more than once. This has
* been added because we now react to the plain ol' `java` plugin, in order to be able to analyze
* Spring Boot projects. However, both the `application` and `java-library` plugins also apply
* `java`, so we have to prevent double-configuration.
*/
private val configuredForJavaProject = AtomicBoolean(false)
/** We only want to register the aggregation tasks if the by-variants tasks are registered. */
private val aggregatorsRegistered = AtomicBoolean(false)
private lateinit var findDeclarationsTask: TaskProvider
private lateinit var redundantJvmPlugin: RedundantJvmPlugin
private lateinit var computeAdviceTask: TaskProvider
private lateinit var reasonTask: TaskProvider
private lateinit var computeResolvedDependenciesTask: TaskProvider
private val isDataBindingEnabled = project.objects.property().convention(false)
private val isViewBindingEnabled = project.objects.property().convention(false)
private val projectHealthPublisher = interProjectPublisher(
project = project,
artifact = DagpArtifacts.Kind.PROJECT_HEALTH
)
private val resolvedDependenciesPublisher = interProjectPublisher(
project = project,
artifact = DagpArtifacts.Kind.RESOLVED_DEPS
)
private val dslService = GlobalDslService.of(project)
fun apply() = project.run {
// Conditionally disable analysis on some projects
val projectPathRegex = projectPathRegex()
if (!projectPathRegex.matches(path)) {
logger.info("Skipping dependency analysis of project '$path'. Does not match regex '$projectPathRegex'.")
return
}
// Hydrate dependencies map with version catalog entries
dslService.get().withVersionCatalogs(this)
maybeConfigureExcludes()
// Android plugins cannot be wrapped in afterEvaluate because of strict lifecycle checks around access to AGP DSL
// objects.
pluginManager.withPlugin(ANDROID_APP_PLUGIN) {
logger.log("Adding Android tasks to $path")
checkAgpOnClasspath()
configureAndroidAppProject()
}
pluginManager.withPlugin(ANDROID_LIBRARY_PLUGIN) {
logger.log("Adding Android tasks to $path")
checkAgpOnClasspath()
configureAndroidLibProject()
}
// Giving up. Wrap the whole thing in afterEvaluate for simplicity and for access to user configuration via
// extension.
afterEvaluate {
pluginManager.withPlugin(APPLICATION_PLUGIN) {
logger.log("Adding JVM tasks to ${project.path}")
configureJavaAppProject()
}
pluginManager.withPlugin(JAVA_LIBRARY_PLUGIN) {
logger.log("Adding JVM tasks to ${project.path}")
configureJavaLibProject()
}
pluginManager.withPlugin(KOTLIN_JVM_PLUGIN) {
logger.log("Adding Kotlin-JVM tasks to ${project.path}")
checkKgpOnClasspath()
configureKotlinJvmProject()
}
pluginManager.withPlugin(JAVA_PLUGIN) {
configureJavaAppProject(maybeAppProject = true)
}
}
}
/**
* Certain plugins expect certain dependencies to be available in a way that limits user ability to change. So, we
* configure DAGP to exclude those dependencies from health reports.
*/
private fun Project.maybeConfigureExcludes() {
/*
* TODO(tsr): user control over the Kotlin stdlib is of a different nature than other dependencies. KGP will add
* this automatically to every `api`-like configuration, unless users add `kotlin.stdlib.default.dependency=false`
* to their `gradle.properties` file. As such, advice regarding this dependency needs to be handled with more care.
* Deal with this in a follow-up. Some kind of DSL opt-in or opt-out.
*/
// If it's a Kotlin project, users have limited ability to make changes to the stdlib.
pluginManager.withPlugin(KOTLIN_JVM_PLUGIN) {
dagpExtension.issueHandler.project(path) {
onAny {
exclude("org.jetbrains.kotlin:kotlin-stdlib")
}
}
}
pluginManager.withPlugin(KOTLIN_ANDROID_PLUGIN) {
dagpExtension.issueHandler.project(path) {
onAny {
exclude("org.jetbrains.kotlin:kotlin-stdlib")
}
}
}
// If it's a Scala project, it needs the scala-library dependency.
pluginManager.withPlugin(SCALA_PLUGIN) {
dagpExtension.issueHandler.project(path) {
onUnusedDependencies {
exclude("org.scala-lang:scala-library")
}
}
}
}
private fun checkAgpOnClasspath() {
try {
@Suppress("UNUSED_VARIABLE")
val a = AndroidComponentsExtension::class.java
} catch (_: Throwable) {
dslService.get().notifyAgpMissing()
}
}
private fun checkKgpOnClasspath() {
try {
@Suppress("UNUSED_VARIABLE")
val k = KotlinProjectExtension::class.java
} catch (_: Throwable) {
dslService.get().notifyKgpMissing()
}
}
/** Has the `com.android.application` plugin applied. */
private fun Project.configureAndroidAppProject() {
val project = this
val ignoredVariantNames = androidIgnoredVariants()
val androidComponents = project.extensions.getByType(AndroidComponentsExtension::class.java)
// val newAgpVersion = androidComponents.pluginVersion.toString().removePrefix("Android Gradle Plugin version ")
val agpVersion = AgpVersion.current().version
androidComponents.onVariants { variant ->
if (variant.name !in ignoredVariantNames) {
val mainSourceSets = variant.sources
val unitTestSourceSets = if (shouldAnalyzeTests()) variant.unitTest?.sources else null
val androidTestSourceSets = if (shouldAnalyzeTests() && variant is HasAndroidTest) {
variant.androidTest?.sources
} else {
null
}
mainSourceSets.let { sourceSets ->
val variantSourceSet = newVariantSourceSet(
variantName = variant.name,
kind = SourceSetKind.MAIN,
variant = variant,
agpArtifacts = variant.artifacts,
sources = sourceSets,
)
val dependencyAnalyzer = AndroidAppAnalyzer(
project = project,
variant = DefaultAndroidVariant(project, variant),
agpVersion = agpVersion,
androidSources = variantSourceSet
)
isDataBindingEnabled.set(dependencyAnalyzer.isDataBindingEnabled)
isViewBindingEnabled.set(dependencyAnalyzer.isViewBindingEnabled)
analyzeDependencies(dependencyAnalyzer)
}
unitTestSourceSets?.let { sourceSets ->
val variantSourceSet = newVariantSourceSet(
variantName = variant.name,
kind = SourceSetKind.TEST,
variant = variant,
agpArtifacts = variant.unitTest!!.artifacts,
sources = sourceSets,
)
val dependencyAnalyzer = AndroidAppAnalyzer(
project = project,
variant = DefaultAndroidVariant(project, variant),
agpVersion = agpVersion,
androidSources = variantSourceSet
)
isDataBindingEnabled.set(dependencyAnalyzer.isDataBindingEnabled)
isViewBindingEnabled.set(dependencyAnalyzer.isViewBindingEnabled)
analyzeDependencies(dependencyAnalyzer)
}
androidTestSourceSets?.let { sourceSets ->
val variantSourceSet = newVariantSourceSet(
variantName = variant.name,
kind = SourceSetKind.ANDROID_TEST,
variant = variant,
agpArtifacts = (variant as HasAndroidTest).androidTest!!.artifacts,
sources = sourceSets,
)
val dependencyAnalyzer = AndroidAppAnalyzer(
project = this@configureAndroidAppProject,
variant = DefaultAndroidVariant(project, variant),
agpVersion = agpVersion,
androidSources = variantSourceSet
)
isDataBindingEnabled.set(dependencyAnalyzer.isDataBindingEnabled)
isViewBindingEnabled.set(dependencyAnalyzer.isViewBindingEnabled)
analyzeDependencies(dependencyAnalyzer)
}
}
}
}
/** Has the `com.android.library` plugin applied. */
private fun Project.configureAndroidLibProject() {
val project = this
val ignoredVariantNames = androidIgnoredVariants()
val androidComponents = project.extensions.getByType(AndroidComponentsExtension::class.java)
// val newAgpVersion = androidComponents.pluginVersion.toString().removePrefix("Android Gradle Plugin version ")
val agpVersion = AgpVersion.current().version
androidComponents.onVariants { variant ->
if (variant.name !in ignoredVariantNames) {
val mainSourceSets = variant.sources
val unitTestSourceSets = if (shouldAnalyzeTests()) variant.unitTest?.sources else null
val androidTestSourceSets = if (shouldAnalyzeTests() && variant is HasAndroidTest) {
variant.androidTest?.sources
} else {
null
}
mainSourceSets.let { sourceSets ->
val variantSourceSet = newVariantSourceSet(
variantName = variant.name,
kind = SourceSetKind.MAIN,
variant = variant,
agpArtifacts = variant.artifacts,
sources = sourceSets,
)
val dependencyAnalyzer = AndroidLibAnalyzer(
project = project,
variant = DefaultAndroidVariant(project, variant),
agpVersion = agpVersion,
androidSources = variantSourceSet,
hasAbi = true,
)
isDataBindingEnabled.set(dependencyAnalyzer.isDataBindingEnabled)
isViewBindingEnabled.set(dependencyAnalyzer.isViewBindingEnabled)
analyzeDependencies(dependencyAnalyzer)
}
unitTestSourceSets?.let { sourceSets ->
val variantSourceSet = newVariantSourceSet(
variantName = variant.name,
kind = SourceSetKind.TEST,
variant = variant,
agpArtifacts = variant.unitTest!!.artifacts,
sources = sourceSets,
)
val dependencyAnalyzer = AndroidLibAnalyzer(
project = project,
variant = DefaultAndroidVariant(project, variant),
agpVersion = agpVersion,
androidSources = variantSourceSet,
hasAbi = false,
)
isDataBindingEnabled.set(dependencyAnalyzer.isDataBindingEnabled)
isViewBindingEnabled.set(dependencyAnalyzer.isViewBindingEnabled)
analyzeDependencies(dependencyAnalyzer)
}
androidTestSourceSets?.let { sourceSets ->
val variantSourceSet = newVariantSourceSet(
variantName = variant.name,
kind = SourceSetKind.ANDROID_TEST,
variant = variant,
agpArtifacts = (variant as HasAndroidTest).androidTest!!.artifacts,
sources = sourceSets,
)
val dependencyAnalyzer = AndroidLibAnalyzer(
project = project,
variant = DefaultAndroidVariant(project, variant),
agpVersion = agpVersion,
androidSources = variantSourceSet,
hasAbi = false,
)
isDataBindingEnabled.set(dependencyAnalyzer.isDataBindingEnabled)
isViewBindingEnabled.set(dependencyAnalyzer.isViewBindingEnabled)
analyzeDependencies(dependencyAnalyzer)
}
}
}
}
private fun newVariantSourceSet(
variantName: String,
kind: SourceSetKind,
variant: com.android.build.api.variant.Variant,
agpArtifacts: Artifacts,
sources: Sources,
): AndroidSources {
// https://github.com/autonomousapps/dependency-analysis-gradle-plugin/issues/1111
// https://issuetracker.google.com/issues/325307775
// if ~/.android/analytics.settings has `hasOptedIn` set to `true`, then
// `./gradlew :app:explodeXmlSourceDebugTest --no-daemon` will fail. This only happens for unit test analysis.
// Running "AndroidTestDependenciesSpec.transitive test dependencies should be declared on testImplementation*" will
// reproduce this error. I don't yet know how to set up a test environment that can reproduce that failure
// hermetically (that is, without having to adjust my user home directory).
return if (kind == SourceSetKind.TEST) {
TestAndroidSources(
project = project,
sources = sources,
primaryAgpVariant = variant,
agpArtifacts = agpArtifacts,
variant = Variant(variantName, kind),
compileClasspathConfigurationName = kind.compileClasspathConfigurationName(variantName),
runtimeClasspathConfigurationName = kind.runtimeClasspathConfigurationName(variantName),
)
} else {
DefaultAndroidSources(
project = project,
sources = sources,
primaryAgpVariant = variant,
agpArtifacts = agpArtifacts,
variant = Variant(variantName, kind),
compileClasspathConfigurationName = kind.compileClasspathConfigurationName(variantName),
runtimeClasspathConfigurationName = kind.runtimeClasspathConfigurationName(variantName),
)
}
}
// Scenarios (this comment is a bit outdated)
// 1. Has application, and then kotlin-jvm applied (in that order):
// - should be a kotlin-jvm-app project
// - must use afterEvaluate to see if kotlin-jvm is applied
// 2. Has kotlin-jvm, and then application applied (in that order):
// - should be a kotlin-jvm-app project
// - must use afterEvaluate to see if app or lib type project
// 3. Has only application applied
// - jvm-app project
// 4. Has only kotlin-jvm applied
// - kotlin-jvm-lib project
// 5. Has kotlin-jvm and java-library applied (any order)
// - kotlin-jvm-lib, and one is redundant (depending on source in project)
// 6. Has kotlin-jvm, application, and java-library applied
// - You're fucked, what are you even doing?
// ***** SPRING BOOT --> Always an app project *****
// 7. Has Spring Boot and java applied
// - jvm-app project
// 8. Has Spring Boot and java-library applied
// - jvm-app project (user is wrong to use java-library)
// 9. Has Spring Boot, java, and java-library applied
// - jvm-app project
// - sigh
// 10. Has Spring Boot and kotlin-jvm applied
// - kotlin-jvm-app project
/**
* Has an application-like plugin applied, such as [APPLICATION_PLUGIN], [SPRING_BOOT_PLUGIN], or [GRETTY_PLUGIN].
*
* The `org.jetbrains.kotlin.jvm` may or may not be applied. If it is applied, this is a kotlin-jvm-app project. If it
* isn't, a java-jvm-app project.
*/
private fun Project.configureJavaAppProject(maybeAppProject: Boolean = false) {
if (maybeAppProject) {
if (!isAppProject()) {
// This means we only discovered the java plugin, which isn't sufficient
return
}
logger.log("Adding JVM tasks to ${project.path}")
}
// If kotlin-jvm is NOT applied, then go ahead and configure the project as a java-jvm-app
// project. If it IS applied, do nothing. We will configure this as a kotlin-jvm-app project
// in `configureKotlinJvmProject()`.
if (!pluginManager.hasPlugin(KOTLIN_JVM_PLUGIN)) {
if (configuredForJavaProject.getAndSet(true)) {
logger.info("(dependency analysis) $path was already configured")
return
}
val j = JavaSources(this, dagpExtension)
j.sourceSets.forEach { sourceSet ->
try {
analyzeDependencies(
JavaWithoutAbiAnalyzer(
project = this,
sourceSet = sourceSet,
kind = sourceSet.jvmSourceSetKind()
)
)
} catch (_: UnknownTaskException) {
logger.warn("Skipping tasks creation for sourceSet `${sourceSet.name}`")
}
}
}
}
/** Has the `java-library` plugin applied. */
private fun Project.configureJavaLibProject() {
val j = JavaSources(this, dagpExtension)
configureRedundantJvmPlugin {
it.withJava(j.hasJava)
}
if (configuredForKotlinJvmOrJavaLibrary.getAndSet(true)) {
logger.info("(dependency analysis) $path was already configured for the kotlin-jvm plugin")
redundantJvmPlugin.configure()
return
}
if (configuredForJavaProject.getAndSet(true)) {
logger.info("(dependency analysis) $path was already configured")
return
}
j.sourceSets.forEach { sourceSet ->
try {
val kind = sourceSet.jvmSourceSetKind()
val hasAbi = hasAbi(sourceSet)
// Regardless of the fact that this is a "java-library" project, the presence of Spring
// Boot indicates an app project.
val dependencyAnalyzer = if (pluginManager.hasPlugin(SPRING_BOOT_PLUGIN)) {
logger.warn(
"(dependency analysis) You have both java-library and org.springframework.boot applied. You probably " +
"want java, not java-library."
)
JavaWithoutAbiAnalyzer(
project = this,
sourceSet = sourceSet,
kind = kind
)
} else {
if (hasAbi) {
JavaWithAbiAnalyzer(
project = this,
sourceSet = sourceSet,
kind = kind,
hasAbi = true
)
} else {
JavaWithoutAbiAnalyzer(
project = this,
sourceSet = sourceSet,
kind = kind
)
}
}
analyzeDependencies(dependencyAnalyzer)
} catch (_: UnknownTaskException) {
logger.warn("Skipping tasks creation for sourceSet `${sourceSet.name}`")
}
}
}
/**
* Has the `org.jetbrains.kotlin.jvm` (aka `kotlin("jvm")`) plugin applied. The `application` (and
* by implication the `java`) plugin may or may not be applied. If it is, this is an app project.
* If it isn't, this is a library project.
*/
private fun Project.configureKotlinJvmProject() {
val k = KotlinSources(this, dagpExtension)
configureRedundantJvmPlugin {
it.withKotlin(k.hasKotlin)
}
if (configuredForKotlinJvmOrJavaLibrary.getAndSet(true)) {
logger.info("(dependency analysis) $path was already configured for the java-library plugin")
redundantJvmPlugin.configure()
return
}
k.sourceSets.forEach { sourceSet ->
try {
val kind = sourceSet.jvmSourceSetKind()
val hasAbi = hasAbi(sourceSet)
val dependencyAnalyzer = if (hasAbi) {
KotlinJvmLibAnalyzer(
project = this,
sourceSet = sourceSet,
kind = kind,
hasAbi = true
)
} else {
KotlinJvmAppAnalyzer(
project = this,
sourceSet = sourceSet,
kind = kind
)
}
analyzeDependencies(dependencyAnalyzer)
} catch (_: UnknownTaskException) {
logger.warn("Skipping tasks creation for sourceSet `${sourceSet.name}`")
}
}
}
private fun Project.hasAbi(sourceSet: SourceSet): Boolean {
if (sourceSet.name in dagpExtension.abiHandler.exclusionsHandler.excludedSourceSets.get()) {
// if this sourceSet is user-excluded, then it doesn't have an ABI
return false
}
val kind = sourceSet.jvmSourceSetKind()
val hasApiConfiguration = configurations.findByName(sourceSet.apiConfigurationName) != null
// The 'test' sourceSet does not have an ABI
val isNotTest = kind != SourceSetKind.TEST
// The 'main' sourceSet for an app project does not have an ABI
val isNotMainApp = !(isAppProject() && kind == SourceSetKind.MAIN)
return hasApiConfiguration && isNotTest && isNotMainApp
}
private fun Project.isAppProject() =
pluginManager.hasPlugin(APPLICATION_PLUGIN) ||
pluginManager.hasPlugin(SPRING_BOOT_PLUGIN) ||
pluginManager.hasPlugin(GRETTY_PLUGIN) ||
pluginManager.hasPlugin(ANDROID_APP_PLUGIN) ||
dagpExtension.forceAppProject
/* ===============================================
* The main work of the plugin happens below here.
* ===============================================
*/
private fun Project.configureRedundantJvmPlugin(block: (RedundantJvmPlugin) -> Unit) {
configureAggregationTasks()
if (!::redundantJvmPlugin.isInitialized) {
val projectPath = [email protected]
redundantJvmPlugin = RedundantJvmPlugin(
project = this,
computeAdviceTask = computeAdviceTask,
redundantPluginsBehavior = dagpExtension.issueHandler.redundantPluginsIssueFor(projectPath)
)
}
block(redundantJvmPlugin)
}
/**
* Subproject tasks are registered here. This function is called in a loop, once for each Android variant & source
* set, or Java source set.
*/
private fun Project.analyzeDependencies(dependencyAnalyzer: DependencyAnalyzer) {
configureAggregationTasks()
val thisProjectPath = path
val variantName = dependencyAnalyzer.variantName
val taskNameSuffix = dependencyAnalyzer.taskNameSuffix
val outputPaths = dependencyAnalyzer.outputPaths
/*
* Metadata about the dependency graph.
*/
// Lists the dependencies declared for building the project, along with their physical artifacts (jars).
val artifactsReport = tasks.register("artifactsReport$taskNameSuffix") {
setClasspath(
configurations[dependencyAnalyzer.compileConfigurationName].artifactsFor(dependencyAnalyzer.attributeValueJar)
)
buildPath.set(buildPath(dependencyAnalyzer.compileConfigurationName))
output.set(outputPaths.compileArtifactsPath)
}
// Lists the dependencies declared for running the project, along with their physical artifacts (jars).
val artifactsReportRuntime = tasks.register("artifactsReportRuntime$taskNameSuffix") {
setClasspath(
configurations[dependencyAnalyzer.runtimeConfigurationName].artifactsFor(dependencyAnalyzer.attributeValueJar)
)
buildPath.set(buildPath(dependencyAnalyzer.runtimeConfigurationName))
output.set(outputPaths.runtimeArtifactsPath)
}
// Produce a DAG of the compile and runtime classpaths rooted on this project.
val graphViewTask = tasks.register("graphView$taskNameSuffix") {
configureTask(
project = this@analyzeDependencies,
compileClasspath = configurations[dependencyAnalyzer.compileConfigurationName],
runtimeClasspath = configurations[dependencyAnalyzer.runtimeConfigurationName],
jarAttr = dependencyAnalyzer.attributeValueJar
)
buildPath.set(buildPath(dependencyAnalyzer.compileConfigurationName))
projectPath.set(thisProjectPath)
variant.set(variantName)
kind.set(dependencyAnalyzer.kind)
declarations.set(findDeclarationsTask.flatMap { it.output })
output.set(outputPaths.compileGraphPath)
outputDot.set(outputPaths.compileGraphDotPath)
outputNodes.set(outputPaths.compileNodesPath)
outputRuntime.set(outputPaths.runtimeGraphPath)
outputRuntimeDot.set(outputPaths.runtimeGraphDotPath)
}
// This is an optional task that only works for Gradle 7.5+
if (GradleVersions.isAtLeastGradle75) {
val resolveExternalDependencies =
tasks.register("resolveExternalDependencies$taskNameSuffix") {
configureTask(
project = this@analyzeDependencies,
compileClasspath = configurations[dependencyAnalyzer.compileConfigurationName],
runtimeClasspath = configurations[dependencyAnalyzer.runtimeConfigurationName],
jarAttr = dependencyAnalyzer.attributeValueJar
)
output.set(outputPaths.externalDependenciesPath)
}
computeResolvedDependenciesTask.configure {
externalDependencies.add(resolveExternalDependencies.flatMap { it.output })
}
}
val computeDominatorCompile =
tasks.register("computeDominatorTreeCompile$taskNameSuffix") {
projectPath.set(thisProjectPath)
physicalArtifacts.set(artifactsReport.flatMap { it.output })
graphView.set(graphViewTask.flatMap { it.output })
outputTxt.set(outputPaths.compileDominatorConsolePath)
outputDot.set(outputPaths.compileDominatorGraphPath)
outputJson.set(outputPaths.compileDominatorJsonPath)
}
val computeDominatorRuntime =
tasks.register("computeDominatorTreeRuntime$taskNameSuffix") {
projectPath.set(thisProjectPath)
physicalArtifacts.set(artifactsReportRuntime.flatMap { it.output })
graphView.set(graphViewTask.flatMap { it.outputRuntime })
outputTxt.set(outputPaths.runtimeDominatorConsolePath)
outputDot.set(outputPaths.runtimeDominatorGraphPath)
outputJson.set(outputPaths.runtimeDominatorJsonPath)
}
// a lifecycle task that computes the dominator tree for both compile and runtime classpaths
tasks.register("computeDominatorTree$taskNameSuffix") {
dependsOn(computeDominatorCompile, computeDominatorRuntime)
}
tasks.register("printDominatorTreeCompile$taskNameSuffix") {
consoleText.set(computeDominatorCompile.flatMap { it.outputTxt })
}
tasks.register("printDominatorTreeRuntime$taskNameSuffix") {
consoleText.set(computeDominatorRuntime.flatMap { it.outputTxt })
}
reasonTask.configure {
dependencyGraphViews.add(graphViewTask.flatMap { it.output /* compile graph */ })
dependencyGraphViews.add(graphViewTask.flatMap { it.outputRuntime })
}
/* ******************************
* Producers. Find the capabilities of all the producers (dependencies). There are many capabilities, including:
* 1. Android linters.
* 2. Classes and constants.
* 3. Inline members from Kotlin libraries.
* 4. Android components (e.g. Services and Providers).
* etc.
*
* And then synthesize the above.
********************************/
// A report of all dependencies that supply Android linters on the compile classpath.
val androidLintTask = dependencyAnalyzer.registerFindAndroidLintersTask()
// A report of all dependencies that supply Android assets on the compile classpath.
val findAndroidAssetsTask = dependencyAnalyzer.registerFindAndroidAssetProvidersTask()
// Explode jars to expose their secrets.
val explodeJarTask = tasks.register("explodeJar$taskNameSuffix") {
inMemoryCache.set(InMemoryCache.register(project))
compileClasspath.setFrom(
configurations[dependencyAnalyzer.compileConfigurationName]
.artifactsFor(dependencyAnalyzer.attributeValueJar)
.artifactFiles
)
physicalArtifacts.set(artifactsReport.flatMap { it.output })
androidLintTask?.let { task ->
androidLinters.set(task.flatMap { it.output })
}
output.set(outputPaths.allDeclaredDepsPath)
}
// Find the inline members of this project's dependencies.
val kotlinMagicTask = tasks.register("findKotlinMagic$taskNameSuffix") {
inMemoryCacheProvider.set(InMemoryCache.register(project))
compileClasspath.setFrom(
configurations[dependencyAnalyzer.compileConfigurationName]
.artifactsFor(dependencyAnalyzer.attributeValueJar)
.artifactFiles
)
artifacts.set(artifactsReport.flatMap { it.output })
outputInlineMembers.set(outputPaths.inlineUsagePath)
outputTypealiases.set(outputPaths.typealiasUsagePath)
outputErrors.set(outputPaths.inlineUsageErrorsPath)
}
// Produces a report of packages from included manifests. Null for java-library projects.
val androidManifestTask = dependencyAnalyzer.registerManifestComponentsExtractionTask()
// Produces a report that lists all dependencies that contribute Android resources. Null for java-library projects.
val findAndroidResTask = dependencyAnalyzer.registerFindAndroidResTask()
// Produces a report of all AAR dependencies with bundled native libs.
val findNativeLibsTask = dependencyAnalyzer.registerFindNativeLibsTask()
// A report of service loaders.
val findServiceLoadersTask = tasks.register("serviceLoader$taskNameSuffix") {
setCompileClasspath(
configurations[dependencyAnalyzer.compileConfigurationName].artifactsFor(dependencyAnalyzer.attributeValueJar)
)
output.set(outputPaths.serviceLoaderDependenciesPath)
}
// A report of declared annotation processors.
val declaredProcsTask = dependencyAnalyzer.registerFindDeclaredProcsTask()
val synthesizeDependenciesTask =
tasks.register("synthesizeDependencies$taskNameSuffix") {
inMemoryCache.set(InMemoryCache.register(project))
projectPath.set(thisProjectPath)
compileDependencies.set(graphViewTask.flatMap { it.outputNodes })
physicalArtifacts.set(artifactsReport.flatMap { it.output })
explodedJars.set(explodeJarTask.flatMap { it.output })
inlineMembers.set(kotlinMagicTask.flatMap { it.outputInlineMembers })
typealiases.set(kotlinMagicTask.flatMap { it.outputTypealiases })
serviceLoaders.set(findServiceLoadersTask.flatMap { it.output })
annotationProcessors.set(declaredProcsTask.flatMap { it.output })
// Optional Android-only inputs
androidManifestTask?.let { task -> manifestComponents.set(task.flatMap { it.output }) }
findAndroidResTask?.let { task -> androidRes.set(task.flatMap { it.output }) }
findNativeLibsTask?.let { task -> nativeLibs.set(task.flatMap { it.output }) }
findAndroidAssetsTask?.let { task -> androidAssets.set(task.flatMap { it.output }) }
outputDir.set(outputPaths.dependenciesDir)
}
/* ******************************
* Consumer. Start with introspection: what can we say about this project itself? There are several elements:
* 1. Source code analysis (the only way to see types used as generic types).
* 2. Bytecode analysis -- all classes used by our class files.
* 3. Bytecode analysis -- all classes exposed as the ABI.
* 4. Android resource analysis -- look for class references and Android resource symbols and IDs.
*
* And then synthesize the above.
********************************/
// Lists all import declarations in the source of the current project.
val explodeCodeSourceTask = tasks.register("explodeCodeSource$taskNameSuffix") {
groovySourceFiles.setFrom(dependencyAnalyzer.groovySourceFiles)
javaSourceFiles.setFrom(dependencyAnalyzer.javaSourceFiles)
kotlinSourceFiles.setFrom(dependencyAnalyzer.kotlinSourceFiles)
scalaSourceFiles.setFrom(dependencyAnalyzer.scalaSourceFiles)
output.set(outputPaths.explodedSourcePath)
}
// Lists all classes _used by_ the given project. Analyzes bytecode and collects all class references.
val explodeBytecodeTask = dependencyAnalyzer.registerByteCodeSourceExploderTask()
// Lists all possibly-external XML resources referenced by this project's Android resources (or null if this isn't
// an Android project).
val explodeXmlSourceTask = dependencyAnalyzer.registerExplodeXmlSourceTask()
// List all assets provided by this library (or null if this isn't an Android project).
val explodeAssetSourceTask = dependencyAnalyzer.registerExplodeAssetSourceTask()
// Describes the project's binary API, or ABI. Null for application projects.
val abiAnalysisTask = dependencyAnalyzer.registerAbiAnalysisTask(provider {
// lazy ABI JSON
with(dagpExtension.abiHandler.exclusionsHandler) {
AbiExclusions(
annotationExclusions = annotationExclusions.get(),
classExclusions = classExclusions.get(),
pathExclusions = pathExclusions.get()
).toJson()
}
})
val usagesExclusionsProvider = provider {
with(dagpExtension.usagesHandler.exclusionsHandler) {
UsagesExclusions(
classExclusions = classExclusions.get(),
).toJson()
}
}
// Synthesizes the above into a single view of this project's usages.
val synthesizeProjectViewTask = tasks.register("synthesizeProjectView$taskNameSuffix") {
projectPath.set(thisProjectPath)
buildType.set(dependencyAnalyzer.buildType)
flavor.set(dependencyAnalyzer.flavorName)
variant.set(variantName)
kind.set(dependencyAnalyzer.kind)
graph.set(graphViewTask.flatMap { it.output })
annotationProcessors.set(declaredProcsTask.flatMap { it.output })
explodedBytecode.set(explodeBytecodeTask.flatMap { it.output })
explodedSourceCode.set(explodeCodeSourceTask.flatMap { it.output })
usagesExclusions.set(usagesExclusionsProvider)
// Optional: only exists for libraries.
abiAnalysisTask?.let { t -> explodingAbi.set(t.flatMap { it.output }) }
// Optional: only exists for Android libraries.
explodeXmlSourceTask?.let { t -> androidResSource.set(t.flatMap { it.output }) }
// Optional: only exists for Android libraries.
explodeAssetSourceTask?.let { t -> androidAssetsSource.set(t.flatMap { it.output }) }
// Optional: only exists for Android projects.
testInstrumentationRunner.set(dependencyAnalyzer.testInstrumentationRunner)
output.set(outputPaths.syntheticProjectPath)
}
// Discover duplicates on compile and runtime classpaths
val duplicateClassesCompile =
tasks.register("discoverDuplicationForCompile$taskNameSuffix") {
description("compile")
setClasspath(
configurations[dependencyAnalyzer.compileConfigurationName].artifactsFor(dependencyAnalyzer.attributeValueJar)
)
syntheticProject.set(synthesizeProjectViewTask.flatMap { it.output })
output.set(outputPaths.duplicateCompileClasspathPath)
}
val duplicateClassesRuntime =
tasks.register("discoverDuplicationForRuntime$taskNameSuffix") {
description("runtime")
setClasspath(
configurations[dependencyAnalyzer.runtimeConfigurationName].artifactsFor(dependencyAnalyzer.attributeValueJar)
)
syntheticProject.set(synthesizeProjectViewTask.flatMap { it.output })
output.set(outputPaths.duplicateCompileRuntimePath)
}
computeAdviceTask.configure {
duplicateClassesReports.add(duplicateClassesCompile.flatMap { it.output })
duplicateClassesReports.add(duplicateClassesRuntime.flatMap { it.output })
}
/* **************************************
* Producers -> Consumer. Bring it all together. How does this project (consumer) use its dependencies (producers)?
****************************************/
// Computes how this project really uses its dependencies, without consideration for user reporting preferences.
val computeUsagesTask = tasks.register("computeActualUsage$taskNameSuffix") {
graph.set(graphViewTask.flatMap { it.output })
declarations.set(findDeclarationsTask.flatMap { it.output })
dependencies.set(synthesizeDependenciesTask.flatMap { it.outputDir })
syntheticProject.set(synthesizeProjectViewTask.flatMap { it.output })
kapt.set(isKaptApplied())
output.set(outputPaths.dependencyTraceReportPath)
}
// Null for JVM projects
val androidScoreTask = dependencyAnalyzer.registerAndroidScoreTask(
synthesizeDependenciesTask, synthesizeProjectViewTask
)
computeAdviceTask.configure {
buildPath.set(buildPath(dependencyAnalyzer.compileConfigurationName))
dependencyGraphViews.add(graphViewTask.flatMap { it.output })
dependencyUsageReports.add(computeUsagesTask.flatMap { it.output })
androidScoreTask?.let { t -> androidScoreReports.add(t.flatMap { it.output }) }
}
// Generates graph view of local (project) dependencies
tasks.register("generateProjectGraph$taskNameSuffix") {
compileClasspath.set(
configurations[dependencyAnalyzer.compileConfigurationName]
.incoming
.resolutionResult
.rootComponent
)
runtimeClasspath.set(
configurations[dependencyAnalyzer.runtimeConfigurationName]
.incoming
.resolutionResult
.rootComponent
)
output.set(outputPaths.projectGraphDir)
}
}
private fun Project.configureAggregationTasks() {
if (aggregatorsRegistered.getAndSet(true)) return
val project = this
val theProjectPath = path
val paths = NoVariantOutputPaths(this)
findDeclarationsTask = tasks.register("findDeclarations") {
FindDeclarationsTask.configureTask(
task = this,
project = project,
outputPaths = paths
)
}
computeAdviceTask = tasks.register("computeAdvice") {
projectPath.set(theProjectPath)
declarations.set(findDeclarationsTask.flatMap { it.output })
bundles.set(dagpExtension.dependenciesHandler.serializableBundles())
supportedSourceSets.set(supportedSourceSetNames())
ignoreKtx.set(dagpExtension.dependenciesHandler.ignoreKtx)
explicitSourceSets.set(dagpExtension.dependenciesHandler.explicitSourceSets)
kapt.set(isKaptApplied())
output.set(paths.unfilteredAdvicePath)
dependencyUsages.set(paths.dependencyUsagesPath)
annotationProcessorUsages.set(paths.annotationProcessorUsagesPath)
bundledTraces.set(paths.bundledTracesPath)
}
val filterAdviceTask = tasks.register("filterAdvice") {
// This information...
projectAdvice.set(computeAdviceTask.flatMap { it.output })
// ...is filtered by these preferences...
dataBindingEnabled.set(isDataBindingEnabled)
viewBindingEnabled.set(isViewBindingEnabled)
with(dagpExtension.issueHandler) {
// These all have sourceSet-specific behaviors
anyBehavior.addAll(anyIssueFor(theProjectPath))
unusedDependenciesBehavior.addAll(unusedDependenciesIssueFor(theProjectPath))
usedTransitiveDependenciesBehavior.addAll(usedTransitiveDependenciesIssueFor(theProjectPath))
incorrectConfigurationBehavior.addAll(incorrectConfigurationIssueFor(theProjectPath))
compileOnlyBehavior.addAll(compileOnlyIssueFor(theProjectPath))
runtimeOnlyBehavior.addAll(runtimeOnlyIssueFor(theProjectPath))
unusedProcsBehavior.addAll(unusedAnnotationProcessorsIssueFor(theProjectPath))
// These don't have sourceSet-specific behaviors
redundantPluginsBehavior.set(redundantPluginsIssueFor(theProjectPath))
moduleStructureBehavior.set(moduleStructureIssueFor(theProjectPath))
}
// ...and produces this output.
output.set(paths.filteredAdvicePath)
}
val generateProjectHealthReport = tasks.register("generateConsoleReport") {
projectAdvice.set(filterAdviceTask.flatMap { it.output })
postscript.set(dagpExtension.reportingHandler.postscript)
dslKind.set(DslKind.from(buildFile))
dependencyMap.set(dagpExtension.dependenciesHandler.map)
output.set(paths.consoleReportPath)
}
tasks.register("projectHealth") {
buildFilePath.set(project.buildFile.path)
projectAdvice.set(filterAdviceTask.flatMap { it.output })
consoleReport.set(generateProjectHealthReport.flatMap { it.output })
}
reasonTask = tasks.register("reason") {
rootProjectName.set(rootProject.name)
projectPath.set(theProjectPath)
dependencyMap.set(dagpExtension.dependenciesHandler.map)
dependencyUsageReport.set(computeAdviceTask.flatMap { it.dependencyUsages })
annotationProcessorUsageReport.set(computeAdviceTask.flatMap { it.annotationProcessorUsages })
unfilteredAdviceReport.set(computeAdviceTask.flatMap { it.output })
finalAdviceReport.set(filterAdviceTask.flatMap { it.output })
bundleTracesReport.set(computeAdviceTask.flatMap { it.bundledTraces })
}
tasks.register("fixDependencies") {
buildScript.set(buildFile)
projectAdvice.set(filterAdviceTask.flatMap { it.output })
dependencyMap.set(dagpExtension.dependenciesHandler.map)
}
computeResolvedDependenciesTask = tasks.register("computeResolvedDependencies") {
output.set(paths.resolvedDepsPath)
}
/*
* Finalizing work.
*/
// Store the main output in the extension for consumption by end-users
storeAdviceOutput(filterAdviceTask.flatMap { it.output })
// Publish our artifacts, and add project dependencies on root project to this project
projectHealthPublisher.publish(filterAdviceTask.flatMap { it.output })
resolvedDependenciesPublisher.publish(computeResolvedDependenciesTask.flatMap { it.output })
}
/** Get the buildPath of the current build from the root component of the resolution result. */
private fun Project.buildPath(configuration: String): Provider {
return configurations[configuration].incoming.resolutionResult.let {
if (isAtLeastGradle82) {
it.rootComponent.map { root -> (root.id as ProjectComponentIdentifier).build.buildPath }
} else {
project.provider { @Suppress("DEPRECATION") (it.root.id as ProjectComponentIdentifier).build.name }
}
}
}
private fun Project.isKaptApplied() = providers.provider { plugins.hasPlugin("org.jetbrains.kotlin.kapt") }
/**
* Returns the names of the 'source sets' that are currently supported by the plugin. Dependencies defined on
* configurations that do not belong to any of these source sets are ignored.
*/
private fun Project.supportedSourceSetNames(): Provider> = provider {
if (pluginManager.hasPlugin(ANDROID_APP_PLUGIN) || pluginManager.hasPlugin(ANDROID_LIBRARY_PLUGIN)) {
extensions.getByType(CommonExtension::class.java)
.sourceSets
.matching { s -> shouldAnalyzeSourceSetForProject(dagpExtension, s.name, project.path) }
.map { it.name }
} else {
// JVM Plugins
the()
.matching { s -> shouldAnalyzeSourceSetForProject(dagpExtension, s.name, project.path) }
.map { it.name }
}
}
private fun SourceSet.jvmSourceSetKind() = when (name) {
SourceSet.MAIN_SOURCE_SET_NAME -> SourceSetKind.MAIN
SourceSet.TEST_SOURCE_SET_NAME -> SourceSetKind.TEST
else -> SourceSetKind.CUSTOM_JVM
}
/** Stores advice output in either root extension or subproject extension. */
private fun storeAdviceOutput(advice: Provider) {
dagpExtension.storeAdviceOutput(advice)
}
private class JavaSources(project: Project, dagpExtension: AbstractExtension) {
val sourceSets: NamedDomainObjectSet = project.the().matching { s ->
project.shouldAnalyzeSourceSetForProject(dagpExtension, s.name, project.path)
}
val hasJava: Provider = project.provider { sourceSets.flatMap { it.java() }.isNotEmpty() }
}
// TODO source set abstractions aren't really working out here.
private class KotlinSources(project: Project, dagpExtension: AbstractExtension) {
private val sourceSetContainer = project.the()
private val kotlinSourceSets = project.the().sourceSets
val sourceSets: NamedDomainObjectSet = sourceSetContainer.matching { s ->
project.shouldAnalyzeSourceSetForProject(dagpExtension, s.name, project.path)
}
val hasKotlin: Provider = project.provider { kotlinSourceSets.flatMap { it.kotlin() }.isNotEmpty() }
}
}
private fun Project.shouldAnalyzeSourceSetForProject(
dagpExtension: AbstractExtension,
sourceSetName: String,
projectPath: String,
): Boolean {
return dagpExtension.issueHandler.shouldAnalyzeSourceSet(sourceSetName, projectPath)
&& (project.shouldAnalyzeTests() || sourceSetName != SourceSet.TEST_SOURCE_SET_NAME)
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy