All Downloads are FREE. Search and download functionalities are using the official Maven repository.

ru.vyarus.gradle.plugin.animalsniffer.AnimalSnifferPlugin.groovy Maven / Gradle / Ivy

There is a newer version: 1.7.2
Show newest version
package ru.vyarus.gradle.plugin.animalsniffer

import groovy.transform.CompileStatic
import groovy.transform.Memoized
import groovy.transform.TypeCheckingMode
import org.gradle.api.GradleException
import org.gradle.api.Plugin
import org.gradle.api.Project
import org.gradle.api.artifacts.Configuration
import org.gradle.api.artifacts.Dependency
import org.gradle.api.file.FileCollection
import org.gradle.api.file.RegularFile
import org.gradle.api.plugins.ExtraPropertiesExtension
import org.gradle.api.plugins.JavaBasePlugin
import org.gradle.api.plugins.ReportingBasePlugin
import org.gradle.api.reporting.ReportingExtension
import org.gradle.api.specs.NotSpec
import org.gradle.api.tasks.SourceSet
import org.gradle.api.tasks.SourceSetOutput
import org.gradle.api.tasks.TaskProvider
import org.gradle.util.GradleVersion
import ru.vyarus.gradle.plugin.animalsniffer.info.SignatureInfoTask
import ru.vyarus.gradle.plugin.animalsniffer.signature.AnimalSnifferSignatureExtension
import ru.vyarus.gradle.plugin.animalsniffer.signature.BuildSignatureTask
import ru.vyarus.gradle.plugin.animalsniffer.util.ContainFilesSpec
import ru.vyarus.gradle.plugin.animalsniffer.util.ExcludeFilePatternSpec

/**
 * AnimalSniffer plugin. Implemented the same way as gradle quality plugins (checkstyle, pmd, findbugs):
 * main configuration 'animalsniffer' used to configure custom tasks, registered for each source set.
 * 

* Signatures are expected to be defined with 'signature' configuration. Multiple signatures may be set. * If no signatures defined - no check will be performed. *

* Reports configuration is performed with reports section. Only text report supported. All violations * are always printed to console. *

* To support signature configuration, plugin registered 'animalsnifferSignature' configuration. * If this configuration used then 'animalsnifferSignature' task will be registered to build signature file. * * @author Vyacheslav Rusakov * @since 13.12.2015 */ @CompileStatic class AnimalSnifferPlugin implements Plugin { public static final String SIGNATURE_CONF = 'signature' public static final String ANIMALSNIFFER_CACHE = 'animalsnifferCache' private static final String CHECK_SIGNATURE = 'animalsniffer' private static final String BUILD_SIGNATURE = 'animalsnifferSignature' private Project project private AnimalSnifferExtension extension private AnimalSnifferSignatureExtension buildExtension @Override void apply(Project project) { // activated only when java plugin is enabled project.plugins.withType(JavaBasePlugin) { this.project = project project.plugins.apply(ReportingBasePlugin) checkGradleCompatibility() registerShortcuts() registerConfigurations() registerExtensions() registerCheckTasks() registerBuildTasks() } } private void checkGradleCompatibility() { // due to base internal api changes in gradle 5.0 plugin can't be launched on prior gradle versions GradleVersion version = GradleVersion.current() if (version < GradleVersion.version('5.0')) { throw new GradleException('Animalsniffer plugin requires gradle 5.0 or above, ' + "but your gradle version is: $version.version. Use plugin version 1.4.6.") } } private void registerShortcuts() { ExtraPropertiesExtension props = project.extensions.extraProperties // to allow custom tasks declaration without package props.set(BuildSignatureTask.simpleName, BuildSignatureTask) props.set(SignatureInfoTask.simpleName, SignatureInfoTask) } @CompileStatic(TypeCheckingMode.SKIP) private void registerConfigurations() { project.configurations.create(CHECK_SIGNATURE).with { visible = false transitive = true description = 'The AnimalSniffer libraries to be used for this project.' defaultDependencies { dependencies -> dependencies.add(this.project.dependencies .create("org.codehaus.mojo:animal-sniffer-ant-tasks:${this.extension.toolVersion}")) } } project.configurations.create(SIGNATURE_CONF).with { visible = false transitive = true description = 'AnimalSniffer signatures' } } @CompileStatic(TypeCheckingMode.SKIP) private void registerExtensions() { extension = project.extensions.create(CHECK_SIGNATURE, AnimalSnifferExtension, project) extension.conventionMapping.with { reportsDir = { project.extensions.getByType(ReportingExtension).file(CHECK_SIGNATURE) } sourceSets = { project.sourceSets } } buildExtension = project.extensions.create(BUILD_SIGNATURE, AnimalSnifferSignatureExtension) } @SuppressWarnings(['Indentation', 'NestedBlockDepth']) @CompileStatic(TypeCheckingMode.SKIP) private void registerCheckTasks() { // create tasks for each source set project.sourceSets.all { SourceSet sourceSet -> String sourceSetName = sourceSet.name SourceSetOutput sourceSetOut = sourceSet.output TaskProvider checkTask = project.tasks . register(sourceSet.getTaskName(CHECK_SIGNATURE, null), AnimalSniffer) { description = "Run AnimalSniffer checks for ${ sourceSetName} classes" // task operates on classes instead of sources source = sourceSetOut reports.all { report -> if (GradleVersion.current() >= GradleVersion.version('7.0')) { report.required.convention(true) report.outputLocation.convention(project.provider { { -> new File(extension.reportsDir, "${sourceSetName}.${report.name}") } as RegularFile }) } else { report.conventionMapping.with { it.enabled = { true } it.destination = { new File(extension.reportsDir, "${sourceSetName}.${report.name}") } } } } } configureCheckTask(checkTask, sourceSet) } // include required animalsniffer tasks in check lifecycle project.tasks.named('check').configure { dependsOn { extension.sourceSets*.getTaskName(CHECK_SIGNATURE, null) } } } @SuppressWarnings(['Indentation', 'MethodSize', 'UnnecessaryGetter']) @CompileStatic(TypeCheckingMode.SKIP) private void configureCheckTask(TaskProvider checkTask, SourceSet sourceSet) { Configuration animalsnifferConfiguration = project.configurations[CHECK_SIGNATURE] // build special signature from provided signatures and all jars to be able to cache it // and perform much faster checks after the first run TaskProvider signatureTask = project.tasks . register(sourceSet.getTaskName(ANIMALSNIFFER_CACHE, null), BuildSignatureTask) { // this special task can be skipped if animalsniffer check supposed to be skipped // note that task is still created because signatures could be registered dynamically onlyIf { !extension.signatures.empty && extension.cache.enabled } conventionMapping.with { animalsnifferClasspath = { animalsnifferConfiguration } signatures = { extension.signatures } files = { excludeJars(getClasspathWithoutModules(sourceSet)) } exclude = { extension.cache.exclude as Set } mergeSignatures = { extension.cache.mergeSignatures } // debug for cache tasks controlled by check debug debug = { extension.debug } } } checkTask.configure { dependsOn(sourceSet.classesTaskName) // skip if no signatures configured or no sources to check onlyIf { !getAnimalsnifferSignatures().empty && getSource().size() > 0 } conventionMapping.with { classpath = { extension.cache.enabled ? getModulesFromClasspath(sourceSet) : excludeJars(sourceSet.compileClasspath) } animalsnifferSignatures = { extension.cache.enabled ? signatureTask.get().outputFiles : extension.signatures } animalsnifferClasspath = { animalsnifferConfiguration } sourcesDirs = { sourceSet.allJava.srcDirs } ignoreFailures = { extension.ignoreFailures } annotation = { extension.annotation } ignoreClasses = { extension.ignore } debug = { extension.debug } } } project.afterEvaluate { if (extension.cache.enabled) { // dependency must not be added earlier to avoid cache task appearance in log checkTask.configure { dependsOn(signatureTask) } } else { // cache task should be always registered to simplify usage, but, by default, it is not enabled // cache task is not created at this point (gradle avoidance api) signatureTask.configure { enabled = false } } } } @CompileStatic(TypeCheckingMode.SKIP) private void registerBuildTasks() { project.afterEvaluate { // register build signature task if files specified for signature creation if (buildExtension.configured) { project.tasks.register(BUILD_SIGNATURE, BuildSignatureTask) { task -> buildExtension.files.each { task.files(it) } buildExtension.signatures.each { task.signatures(it) } task.include = buildExtension.include task.exclude = buildExtension.exclude // project name by default to be compatible with maven artifacts task.outputName = buildExtension.outputName ?: project.name // for project signature use hardcoded 'signature' folder instead of task name task.outputDirectory = new File(project.buildDir, '/animalsniffer/signature/') task.conventionMapping.debug = { buildExtension.debug } } } // defaults applied to all tasks (including manually created) project.tasks.withType(BuildSignatureTask).configureEach { task -> if (task.animalsnifferClasspath == null) { task.animalsnifferClasspath = project.configurations[CHECK_SIGNATURE] } if (task.outputName == null) { task.outputName = task.name } } } } /** * In multi-module projects there is high probability that modules could change, so its safer to exclude them * from project specific signature to rebuild it less often (and speed up overall build time). * * @param sourceSet source set * @return source set classpath with excluded other module jars */ @Memoized @CompileStatic(TypeCheckingMode.SKIP) private FileCollection getClasspathWithoutModules(SourceSet sourceSet) { Set excludeJars = moduleJars if (excludeJars.empty) { return sourceSet.compileClasspath } sourceSet.compileClasspath.filter new NotSpec(new ContainFilesSpec(excludeJars)) } /** * As other module dependencies removed from project specific signature, they must be added to check task * classpath. * * @param sourceSet source set * @return file collection containing only module jars or null if no dependency on other modules */ @Memoized @CompileStatic(TypeCheckingMode.SKIP) private FileCollection getModulesFromClasspath(SourceSet sourceSet) { Set includeJars = moduleJars if (includeJars.empty) { return null } sourceSet.compileClasspath.filter new ContainFilesSpec(includeJars) } /** * @return list of all project modules jars as a set of patterns */ @Memoized private Set getModuleJars() { Set patterns = [] project.rootProject.allprojects.each { it.configurations.findByName(Dependency.DEFAULT_CONFIGURATION)?.allArtifacts?.each { patterns << it.file } } patterns } /** * Exclude some jars from the classpath. Required to support small 3rd party library signatures. * * @param classpath classpath file collection * @return filtered or original classpath if nothing configured to exclude */ private FileCollection excludeJars(FileCollection classpath) { return extension.excludeJars ? classpath.filter(new ExcludeFilePatternSpec(extension.excludeJars)) : classpath } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy