ru.vyarus.gradle.plugin.animalsniffer.signature.BuildSignatureTask.groovy Maven / Gradle / Ivy
package ru.vyarus.gradle.plugin.animalsniffer.signature
import groovy.transform.CompileStatic
import groovy.transform.TypeCheckingMode
import org.gradle.api.GradleException
import org.gradle.api.file.FileCollection
import org.gradle.api.internal.ConventionTask
import org.gradle.api.internal.project.IsolatedAntBuilder
import org.gradle.api.tasks.*
import org.gradle.util.GFileUtils
import ru.vyarus.gradle.plugin.animalsniffer.AnimalSnifferPlugin
import javax.inject.Inject
/**
* Task for building animalsniffer signature. Compiled classes, jars and other signatures may be used as source.
*
* Task may be used directly or through registered `animalsnifferSignature' configuration closure.
*
* AnimalsnifferClasspath will be set from 'animalsniffer' configuration. By default, output file
* is `build/animalsniffer/${task.name}/${task.name}.sig`. In some cases multiple signatures could be produces,
* so use {@code task.outputFiles} method to get correct task output.
*
* Properties may be used for configuration directly, but it will be more convenient to use provided helper
* methods instead.
*
* Animalsniffer ant task does not allow empty files, but, in some cases, it could happen.
* For example, when two or more signatures must be merged. In this case plugin will
* specify fake file together with it's exclusion from resulted signature to overcome ant task limitation.
*
* @author Vyacheslav Rusakov
* @since 12.04.2017
* @see AnimalSnifferSignatureExtension for configuration extension
* @see
* ant task docks
*/
@CompileStatic
@CacheableTask
@SuppressWarnings(['ConfusingMethodName', 'Println'])
class BuildSignatureTask extends ConventionTask {
/**
* Symbol used when {@link #mergeSignatures} disabled and so multiple signatures produced (according to input
* signatures count). This is used for signature caching for check task so it could print real signature name
* in output without cache signature name.
*/
public static final String SIGNATURE_DELIMITER = '_!'
private static final String DOT = '.'
private static final String NL = '\n'
/**
* The class path containing the Animal Sniffer library to be used.
* Default set by plugin.
*/
@Classpath
@InputFiles
FileCollection animalsnifferClasspath
/**
* Signature files to extend.
*/
@InputFiles
@Optional
@PathSensitive(PathSensitivity.ABSOLUTE)
FileCollection signatures
/**
* Compiled classes and jars.
* Ant task requires files, but plugin will overcome this with fake files
* ({@see # allowEmptyFiles)
*/
@InputFiles
@Optional
@PathSensitive(PathSensitivity.ABSOLUTE)
FileCollection files = project.files()
/**
* Include only packages ('com.java.*')
*/
@Input
@Optional
Set include = []
/**
* Exclude packages ('com.sun.*')
*/
@Input
@Optional
Set exclude = []
/**
* Output signatures directory.
*/
@OutputDirectory
File outputDirectory
/**
* Output signature name. If extension is not set then '.sig' extension will be used. By default, equal to
* task name (set by plugin).
*
* Note: when {@link #mergeSignatures} is false, provided name is used as base to create multiple names
* (by adding _!signatureName postfix).
*/
@Input
String outputName
/**
* Animalsniffer ant task does not allow empty files, but, in some cases, it could happen.
* For example, when two or more signatures must be merged. In this case plugin could
* specify fake file together with it's exclusion from resulted signature to overcome ant task limitation.
*
* Option exists only to be able to switch off empty files workaround. It may be required due to
* security restrictions which could lead to source jar resolution failure (not the case for most environments).
*/
@Input
boolean allowEmptyFiles = true
/**
* When multiple signatures provided they will be merged. If merge disabled, multiple signatures will be provided
* for each incoming signature.
*
* This is very special case used internally for classpath caching when multiple signatures used for check task.
*/
@Input
boolean mergeSignatures = true
/**
* Configuration debug output
*/
@Console
boolean debug
@SuppressWarnings('UnnecessaryGetter')
BuildSignatureTask() {
// task-specific output directory used to avoid clashes when multiple build tasks used
// (output files are taken from directory, so if multiple tasks used, last task output will include
// all tasks outputs, which is wrong).
conventionMapping.map('outputDirectory') { getDefaultTaskOutputDirectory() }
}
@Inject
IsolatedAntBuilder getAntBuilder() {
throw new UnsupportedOperationException()
}
@TaskAction
void run() {
if (handleSimpleCase()) {
return
}
applyFakeFilesIfRequired()
if (!getInclude().empty || !getExclude().empty) {
logger.info("Building signature with includes=${getInclude()} and excludes=${getExclude()}")
}
Set signs = getSignatures() ? getSignatures().files : [] as Set
if (getMergeSignatures() || signs.size() <= 1) {
runSignatureBuild(signs, targetFile)
} else {
// use each provided signature as base (provide multiple signatures)
signs.each {
runSignatureBuild([it], getPerSignatureTargetFile(it))
}
}
}
/**
* Files to use for signature generation. Includes classes (compiled) or jars.
* Internally {@link org.gradle.api.Project#files(java.lang.Object ...)} is used so any definition may be used,
* supported by gradle method.
*
* Method may be called multiple times - all inputs will aggregate.
*
* Common cases:
*
* - sourceSets.main.output - all compiled classes of source set
* - configurations.mycustom - all jars, defined in 'mycustom' configuration (of course, standard
* configurations like 'compile' or 'test' could also be used)
*
*
* @param input input files (classes ot jars)
*/
void files(Object input) {
if (getFiles() == null) {
setFiles(project.files(input))
} else {
setFiles(getFiles() + project.files(input))
}
}
/**
* Signatures to extend from (use all definitions from signature). May be used only together with files definition
* (for example, you can't just merge multiple signatures; this is ant task limitation).
*
* Method may be called multiple times - all inputs will aggregate.
*
* Plugin register special 'signature' configuration to use for signatures definition and it makes sense to use
* it for signatures definition for build task. In this case, use {@code configurations.signature} to add
* all defined signatures.
*
* Internally {@link org.gradle.api.Project#files(java.lang.Object ...)} is used so any definition may be used,
* supported by gradle method.
*
* @param signature base signatures
*/
void signatures(Object signature) {
if (getSignatures() == null) {
setSignatures(project.files(signature))
} else {
setSignatures(getSignatures() + project.files(signature))
}
}
/**
* Use to include only defined classes in resulted signature.
* For example: specify 'com.java.*' to include only 'com.java' package and all it's sub packages.
*
* Method may be called multiple times - all inputs will aggregate.
*
* @param path class names or package mask
*/
void include(String... path) {
Set include = getInclude()
include.addAll(path)
setInclude(include)
}
/**
* Use to exclude classes in resulted signature.
* For example: specify 'com.sun.*' to exclude 'com.sun' package and all it's sub packages.
*
* Method may be called multiple times - all inputs will aggregate.
*
* @param path class names or package mask
*/
void exclude(String... path) {
Set exclude = getExclude()
exclude.addAll(path)
setExclude(exclude)
}
/**
* It is intentional that property is not annotated with {@code @OutputFiles}!
*
* Task output depends on inputs (number of produced signatures and their names rely on
* configured signatures). Since gradle 4 task output property can't force dependency resolution,
* but in most cases input signatures are configured with 'signature' configuration. This makes impossible
* computation of exact output file names before execution (such computation was implemented in plugin version 1.4).
*
* Due to this limitation, task declares only output directory as task output. But that means that
* {@code task.outputs.files} can't be used, because it will contain only output directory (itself).
* But it's a very handy to rely on task output files, and this method should be used instead of
* outputs mechanism ({@code task.outputFiles}) to configure produced signatures in other tasks.
*
* @return lazy collection of resulted signatures to use as input for other tasks
*/
@Internal
FileCollection getOutputFiles() {
return project.fileTree(getOutputDirectory()).builtBy(this)
}
/**
* Note that in case of check task, this will never be true because exclusions are always applied for cache
* task (see {@link ru.vyarus.gradle.plugin.animalsniffer.CheckCacheExtension#exclude}).
*
* @return true when no processing required for configured signatures
*/
private boolean handleSimpleCase() {
if (getFiles().empty
&& (getSignatures().size() == 1 || !getMergeSignatures())
&& getExclude().empty && getInclude().empty) {
getSignatures().each {
File target = getMergeSignatures() ? targetFile : getPerSignatureTargetFile(it)
if (getDebug()) {
println 'No signature build required, simply copying signature:\n' +
"\t$it.name -> ${project.relativePath(target)}"
}
GFileUtils.copyFile(it, target)
}
return true
}
return false
}
private void applyFakeFilesIfRequired() {
boolean signaturesConfigured = getSignatures() != null && !getSignatures().empty
// situation: files specified, but collection is empty (for example, empty configuration)
if (getFiles().empty && signaturesConfigured && isAllowEmptyFiles()) {
logger.info('Workaround empty files for signature build by adding plugin jar as file ' +
'with package exclusion')
// if this will fail due to security restrictions, simply disable allowEmptyFiles config
files(BuildSignatureTask.protectionDomain.codeSource.location)
exclude('ru.vyarus.gradle.plugin.animalsniffer.*')
} else if (getFiles().empty) {
throw new GradleException("No files found in: ${getFiles()}")
}
}
/**
* By default task name used for output directory.
* For cache tasks, 'animalsniffer' prefix cut off.
*
* @return default output directory
*/
private File getDefaultTaskOutputDirectory() {
String subDir = name
if (subDir.startsWith(AnimalSnifferPlugin.ANIMALSNIFFER_CACHE)) {
subDir = "cache${subDir[AnimalSnifferPlugin.ANIMALSNIFFER_CACHE.length()..-1]}"
}
return new File(project.buildDir, "animalsniffer/$subDir")
}
private String getSignatureFileName() {
String out = getOutputName()
if (outputName.indexOf(DOT) < 0) {
out += '.sig'
}
return out
}
/**
* @return output signature file for merge mode ({@link #mergeSignatures} enabled)
*/
private File getTargetFile() {
return new File(getOutputDirectory(), signatureFileName)
}
/**
* When {@link #mergeSignatures} is disabled, task may produce multiple signatures
* for each source signature. Output file name is build by convention:
* configured signature name + special delimiter (to be able to recognize it) +
* source signature name.
*
* @param sig signature file
* @return output signature file name for specific source signature
*/
private File getPerSignatureTargetFile(File sig) {
// use each provided signature as base (provide multiple signatures)
String out = signatureFileName
String baseName = out[0..out.lastIndexOf(DOT) - 1]
String ext = out[out.lastIndexOf(DOT) + 1..-1]
String name = sig.name[0..sig.name.lastIndexOf(DOT) - 1]
return new File(getOutputDirectory(), "${baseName}${SIGNATURE_DELIMITER}$name.$ext")
}
@CompileStatic(TypeCheckingMode.SKIP)
private void runSignatureBuild(Collection signatures, File dest) {
String filesPath = getFiles() && !getFiles().empty ? getFiles().asPath : null
if (getDebug()) {
printTaskConfig(signatures, dest, filesPath)
}
antBuilder.withClasspath(getAnimalsnifferClasspath()).execute { a ->
ant.taskdef(name: 'buildSignature', classname: 'org.codehaus.mojo.animal_sniffer.ant.BuildSignaturesTask')
ant.buildSignature(destfile: dest.absolutePath) {
if (filesPath) {
path(path: filesPath)
}
signatures.each {
signature(src: it.absolutePath)
}
getInclude().each {
includeClasses(className: it)
}
getExclude().each {
excludeClasses(className: it)
}
}
}
}
private void printTaskConfig(Collection signatures, File dest, String path) {
StringBuilder res = new StringBuilder("$dest.name\n")
String rootDir = "${project.rootDir.canonicalPath}${File.separator}"
if (!signatures.empty) {
res.append('\n\tsignatures:\n')
.append(signatures.sort().collect { "\t\t$it.name" }.join(NL)).append(NL)
}
if (!getFiles().empty) {
res.append('\n\tfiles:\n')
.append(path.split(File.pathSeparator)
.collect { "\t\t${it.replace(rootDir, '')}" }
.join(NL))
.append(NL)
}
if (!getInclude().empty) {
res.append('\n\tinclude:\n')
.append(getInclude().collect { "\t\t$it" }.join(NL)).append(NL)
}
if (!getExclude().empty) {
res.append('\n\texclude:\n')
.append(getExclude().collect { "\t\t$it" }.join(NL)).append(NL)
}
println "$res\n"
}
}