org.gradle.integtests.resolve.transform.ArtifactTransformValuesInjectionIntegrationTest.groovy Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-api Show documentation
Show all versions of gradle-api Show documentation
Gradle 6.9.1 API redistribution.
/*
* Copyright 2018 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.gradle.integtests.resolve.transform
import com.google.common.reflect.TypeToken
import org.gradle.api.artifacts.transform.InputArtifact
import org.gradle.api.artifacts.transform.InputArtifactDependencies
import org.gradle.api.file.FileCollection
import org.gradle.api.provider.Provider
import org.gradle.api.tasks.Console
import org.gradle.api.tasks.Destroys
import org.gradle.api.tasks.Input
import org.gradle.api.tasks.InputDirectory
import org.gradle.api.tasks.InputFile
import org.gradle.api.tasks.Internal
import org.gradle.api.tasks.LocalState
import org.gradle.api.tasks.OutputDirectories
import org.gradle.api.tasks.OutputDirectory
import org.gradle.api.tasks.OutputFile
import org.gradle.api.tasks.OutputFiles
import org.gradle.api.tasks.options.OptionValues
import org.gradle.integtests.fixtures.AbstractDependencyResolutionTest
import spock.lang.Unroll
import static org.gradle.util.Matchers.matchesRegexp
class ArtifactTransformValuesInjectionIntegrationTest extends AbstractDependencyResolutionTest implements ArtifactTransformTestFixture {
@Unroll
def "transform can receive parameters, workspace and input artifact (#inputArtifactType) via abstract getter"() {
settingsFile << """
include 'a', 'b', 'c'
"""
setupBuildWithColorAttributes()
buildFile << """
allprojects {
dependencies {
registerTransform(MakeGreen) {
from.attribute(color, 'blue')
to.attribute(color, 'green')
parameters {
extension = 'green'
}
}
}
}
project(':a') {
dependencies {
implementation project(':b')
implementation project(':c')
}
}
abstract class MakeGreen implements TransformAction {
interface Parameters extends TransformParameters {
@Input
String getExtension()
void setExtension(String value)
}
@InputArtifact
abstract ${inputArtifactType} getInput()
void transform(TransformOutputs outputs) {
File inputFile = input${convertToFile}
println "processing \${inputFile.name}"
def output = outputs.file(inputFile.name + "." + parameters.extension)
output.text = "ok"
}
}
"""
when:
if (expectedDeprecation) {
executer.expectDeprecationWarning()
}
run(":a:resolve")
then:
outputContains("processing b.jar")
outputContains("processing c.jar")
outputContains("result = [b.jar.green, c.jar.green]")
if (expectedDeprecation) {
outputContains(expectedDeprecation)
}
where:
inputArtifactType | convertToFile | expectedDeprecation
'File' | '' | 'Injecting the input artifact of a transform as a File has been deprecated. This is scheduled to be removed in Gradle 7.0. Declare the input artifact as Provider instead.'
'Provider' | '.get().asFile' | null
}
@Unroll
def "transform can receive parameter of type #type"() {
settingsFile << """
include 'a', 'b', 'c'
"""
setupBuildWithColorTransform {
params("""
prop = ${value}
nested.nestedProp = ${value}
""")
}
buildFile << """
project(':a') {
dependencies {
implementation project(':b')
implementation project(':c')
}
}
interface NestedType {
@Input
${type} getNestedProp()
}
abstract class MakeGreen implements TransformAction {
interface Parameters extends TransformParameters {
@Input
${type} getProp()
@Input @Optional
${type} getOtherProp()
@Nested
NestedType getNested()
}
void transform(TransformOutputs outputs) {
println "processing using prop: \${parameters.prop.get()}, nested: \${parameters.nested.nestedProp.get()}"
assert parameters.otherProp.getOrNull() == ${expectedNullValue}
}
}
"""
when:
run("a:resolve")
then:
outputContains("processing using prop: ${expected}, nested: ${expected}")
where:
type | value | expected | expectedNullValue
"Property" | "'value'" | 'value' | null
"ListProperty" | "['a', 'b']" | "[a, b]" | "[]"
"SetProperty" | "['a', 'b']" | "[a, b]" | "[] as Set"
"MapProperty" | "[a: 1, b: 2]" | "[a:1, b:2]" | "[:]"
}
def "transform parameters are validated for input output annotations"() {
settingsFile << """
include 'a', 'b'
"""
buildFile << """
interface NestedType {
@InputFile
RegularFileProperty getInputFile()
@OutputDirectory
DirectoryProperty getOutputDirectory()
Property getStringProperty()
}
"""
setupBuildWithColorTransform {
params("""
extension = 'green'
nested.inputFile = file("some")
""")
}
buildFile << """
project(':a') {
dependencies {
implementation project(':b')
}
}
@CacheableTransform
abstract class MakeGreen implements TransformAction {
interface Parameters extends TransformParameters {
String getExtension()
void setExtension(String value)
@OutputDirectory
File getOutputDir()
void setOutputDir(File outputDir)
@Input
String getMissingInput()
void setMissingInput(String missing)
@Input
File getFileInput()
void setFileInput(File file)
@InputFiles
ConfigurableFileCollection getNoPathSensitivity()
@InputFile
File getNoPathSensitivityFile()
void setNoPathSensitivityFile(File file)
@InputDirectory
File getNoPathSensitivityDir()
void setNoPathSensitivityDir(File file)
@PathSensitive(PathSensitivity.ABSOLUTE)
@InputFiles
ConfigurableFileCollection getAbsolutePathSensitivity()
@Incremental
@Input
String getIncrementalNonFileInput()
void setIncrementalNonFileInput(String value)
@Nested
NestedType getNested()
}
void transform(TransformOutputs outputs) {
throw new RuntimeException()
}
}
"""
when:
fails(":a:resolve")
then:
failure.assertThatDescription(matchesRegexp('Cannot isolate parameters MakeGreen\\$Parameters_Decorated@.* of artifact transform MakeGreen'))
failure.assertHasCause('Some problems were found with the configuration of the artifact transform parameter MakeGreen.Parameters.')
assertPropertyValidationErrors(
absolutePathSensitivity: 'is declared to be sensitive to absolute paths. This is not allowed for cacheable transforms',
extension: 'is not annotated with an input annotation',
fileInput: [
'does not have a value specified',
'has @Input annotation used on property of type \'File\'',
],
incrementalNonFileInput: [
'does not have a value specified',
'is annotated with @Incremental that is not allowed for @Input properties',
],
missingInput: 'does not have a value specified',
'nested.outputDirectory': 'is annotated with invalid property type @OutputDirectory',
'nested.inputFile': 'is declared without normalization specified. Properties of cacheable work must declare their normalization via @PathSensitive, @Classpath or @CompileClasspath. Defaulting to PathSensitivity.ABSOLUTE',
'nested.stringProperty': 'is not annotated with an input annotation',
noPathSensitivity: 'is declared without normalization specified. Properties of cacheable work must declare their normalization via @PathSensitive, @Classpath or @CompileClasspath. Defaulting to PathSensitivity.ABSOLUTE',
noPathSensitivityDir: 'is declared without normalization specified. Properties of cacheable work must declare their normalization via @PathSensitive, @Classpath or @CompileClasspath. Defaulting to PathSensitivity.ABSOLUTE',
noPathSensitivityFile: 'is declared without normalization specified. Properties of cacheable work must declare their normalization via @PathSensitive, @Classpath or @CompileClasspath. Defaulting to PathSensitivity.ABSOLUTE',
outputDir: 'is annotated with invalid property type @OutputDirectory',
)
}
def "cannot query parameters for transform without parameters"() {
settingsFile << """
include 'a', 'b', 'c'
"""
setupBuildWithColorTransform()
buildFile << """
project(':a') {
dependencies {
implementation project(':b')
implementation project(':c')
}
}
abstract class MakeGreen implements TransformAction {
void transform(TransformOutputs outputs) {
println getParameters()
}
}
"""
when:
fails(":a:resolve")
then:
failure.assertResolutionFailure(':a:implementation')
failure.assertHasCause("Cannot query parameters for artifact transform without parameters.")
}
def "transform parameters type cannot use caching annotations"() {
settingsFile << """
include 'a', 'b', 'c'
"""
setupBuildWithColorTransform {
params("""
extension = 'green'
""")
}
buildFile << """
project(':a') {
dependencies {
implementation project(':b')
implementation project(':c')
}
}
abstract class MakeGreen implements TransformAction {
@CacheableTask @CacheableTransform
interface Parameters extends TransformParameters {
@Input
String getExtension()
void setExtension(String value)
}
void transform(TransformOutputs outputs) {
throw new RuntimeException()
}
}
"""
when:
fails(":a:resolve")
then:
failure.assertThatDescription(matchesRegexp('Cannot isolate parameters MakeGreen\\$Parameters_Decorated@.* of artifact transform MakeGreen'))
failure.assertHasCause('Some problems were found with the configuration of the artifact transform parameter MakeGreen.Parameters.')
failure.assertHasCause("Type 'MakeGreen.Parameters': Cannot use @CacheableTask on type. This annotation can only be used with Task types.")
failure.assertHasCause("Type 'MakeGreen.Parameters': Cannot use @CacheableTransform on type. This annotation can only be used with TransformAction types.")
}
@Unroll
def "transform parameters type cannot use annotation @#annotation.simpleName"() {
settingsFile << """
include 'a', 'b'
"""
setupBuildWithColorTransform {
params("""
extension = 'green'
""")
}
buildFile << """
project(':a') {
dependencies {
implementation project(':b')
}
}
abstract class MakeGreen implements TransformAction {
interface Parameters extends TransformParameters {
@Input
String getExtension()
void setExtension(String value)
@${annotation.simpleName}
String getBad()
void setBad(String value)
}
void transform(TransformOutputs outputs) {
throw new RuntimeException()
}
}
"""
when:
fails(":a:resolve")
then:
failure.assertThatDescription(matchesRegexp('Cannot isolate parameters MakeGreen\\$Parameters_Decorated@.* of artifact transform MakeGreen'))
failure.assertHasCause('A problem was found with the configuration of the artifact transform parameter MakeGreen.Parameters.')
assertPropertyValidationErrors(bad: "is annotated with invalid property type @${annotation.simpleName}")
where:
annotation << [OutputFile, OutputFiles, OutputDirectory, OutputDirectories, Destroys, LocalState, OptionValues]
}
@Unroll
def "transform parameters type cannot use injection annotation @#annotation.simpleName"() {
settingsFile << """
include 'a', 'b', 'c'
"""
setupBuildWithColorTransform {
params("""
extension = 'green'
""")
}
buildFile << """
project(':a') {
dependencies {
implementation project(':b')
implementation project(':c')
}
}
abstract class MakeGreen implements TransformAction {
interface Parameters extends TransformParameters {
String getExtension()
void setExtension(String value)
@${annotation.simpleName}
String getBad()
void setBad(String value)
}
void transform(TransformOutputs outputs) {
throw new RuntimeException()
}
}
"""
when:
fails(":a:resolve")
then:
failure.assertHasDescription('A problem occurred evaluating root project')
failure.assertHasCause('Could not create an instance of type MakeGreen$Parameters.')
failure.assertHasCause('Could not generate a decorated class for interface MakeGreen$Parameters.')
failure.assertHasCause("Cannot use @${annotation.simpleName} annotation on method Parameters.getBad().")
where:
annotation << [InputArtifact, InputArtifactDependencies]
}
def "transform action is validated for input output annotations"() {
settingsFile << """
include 'a', 'b', 'c'
"""
setupBuildWithColorTransform {
params("""
extension = 'green'
""")
}
buildFile << """
project(':a') {
dependencies {
implementation project(':b')
implementation project(':c')
}
}
@CacheableTransform
abstract class MakeGreen implements TransformAction {
interface Parameters extends TransformParameters {
@Input
String getExtension()
void setExtension(String value)
}
@InputFile
File inputFile
File notAnnotated
@InputArtifact
abstract Provider getNoPathSensitivity()
@PathSensitive(PathSensitivity.ABSOLUTE)
@InputArtifactDependencies
abstract FileCollection getAbsolutePathSensitivityDependencies()
@PathSensitive(PathSensitivity.NAME_ONLY)
@InputFile @InputArtifact @InputArtifactDependencies
Provider getConflictingAnnotations() { }
void transform(TransformOutputs outputs) {
throw new RuntimeException()
}
}
"""
when:
fails(":a:resolve")
then:
failure.assertHasDescription('A problem occurred evaluating root project')
failure.assertHasCause('Some problems were found with the configuration of MakeGreen.')
assertPropertyValidationErrors(
absolutePathSensitivityDependencies: 'is declared to be sensitive to absolute paths. This is not allowed for cacheable transforms',
'conflictingAnnotations': [
'has conflicting type annotations declared: @InputFile, @InputArtifact, @InputArtifactDependencies; assuming @InputFile',
'is annotated with invalid property type @InputFile'
],
inputFile: 'is annotated with invalid property type @InputFile',
noPathSensitivity: 'is declared without normalization specified. Properties of cacheable work must declare their normalization via @PathSensitive, @Classpath or @CompileClasspath. Defaulting to PathSensitivity.ABSOLUTE',
notAnnotated: 'is not annotated with an input annotation',
)
}
def "transform action type cannot use cacheable task annotation"() {
settingsFile << """
include 'a', 'b', 'c'
"""
setupBuildWithColorTransform()
buildFile << """
project(':a') {
dependencies {
implementation project(':b')
implementation project(':c')
}
}
@CacheableTask
abstract class MakeGreen implements TransformAction {
void transform(TransformOutputs outputs) {
throw new RuntimeException()
}
}
"""
when:
fails(":a:resolve")
then:
failure.assertHasDescription('A problem occurred evaluating root project')
failure.assertHasCause('A problem was found with the configuration of MakeGreen.')
failure.assertHasCause("Type 'MakeGreen': Cannot use @CacheableTask on type. This annotation can only be used with Task types.")
}
@Unroll
def "transform action type cannot use annotation @#annotation.simpleName"() {
settingsFile << """
include 'a', 'b', 'c'
"""
setupBuildWithColorTransform {
params("""
extension = 'green'
""")
}
buildFile << """
project(':a') {
dependencies {
implementation project(':b')
implementation project(':c')
}
}
abstract class MakeGreen implements TransformAction {
interface Parameters extends TransformParameters {
@Input
String getExtension()
void setExtension(String value)
}
@${annotation.simpleName}
String getBad() { }
void transform(TransformOutputs outputs) {
throw new RuntimeException()
}
}
"""
when:
fails(":a:resolve")
then:
failure.assertHasDescription('A problem occurred evaluating root project')
failure.assertHasCause('A problem was found with the configuration of MakeGreen.')
assertPropertyValidationErrors(bad: "is annotated with invalid property type @${annotation.simpleName}")
where:
annotation << [Input, InputFile, InputDirectory, OutputFile, OutputFiles, OutputDirectory, OutputDirectories, Destroys, LocalState, OptionValues, Console, Internal]
}
@Unroll
def "transform can receive dependencies via abstract getter of type #targetType"() {
settingsFile << """
include 'a', 'b', 'c'
"""
setupBuildWithColorTransformAction()
buildFile << """
project(':a') {
dependencies {
implementation project(':b')
}
}
project(':b') {
dependencies {
implementation project(':c')
}
}
abstract class MakeGreen implements TransformAction {
@InputArtifactDependencies
abstract ${targetType} getDependencies()
@InputArtifact
abstract Provider getInputArtifact()
void transform(TransformOutputs outputs) {
def input = inputArtifact.get().asFile
println "received dependencies files \${dependencies*.name} for processing \${input.name}"
def output = outputs.file(input.name + ".green")
output.text = "ok"
}
}
"""
when:
run(":a:resolve")
then:
outputContains("received dependencies files [] for processing c.jar")
outputContains("received dependencies files [c.jar] for processing b.jar")
outputContains("result = [b.jar.green, c.jar.green]")
where:
targetType << ["FileCollection", "Iterable"]
}
@Unroll
def "old style transform cannot use @#annotation.name"() {
settingsFile << """
include 'a', 'b', 'c'
"""
setupBuildWithColorAttributes()
buildFile << """
allprojects {
dependencies {
registerTransform {
from.attribute(color, 'blue')
to.attribute(color, 'green')
artifactTransform(MakeGreen)
}
}
}
project(':a') {
dependencies {
implementation project(':b')
implementation project(':c')
}
}
abstract class MakeGreen extends ArtifactTransform {
@${annotation.name}
abstract File getInputFile()
List transform(File input) {
println "processing \${input.name}"
def output = new File(outputDirectory, input.name + ".green")
output.text = "ok"
return [output]
}
}
"""
when:
fails(":a:resolve")
then:
failure.assertHasCause("Cannot use @${annotation.simpleName} annotation on method MakeGreen.getInputFile().")
where:
annotation << [InputArtifact, InputArtifactDependencies]
}
def "transform can receive parameter object via constructor parameter"() {
settingsFile << """
include 'a', 'b', 'c'
"""
setupBuildWithColorTransform {
params("""
extension = 'green'
""")
}
buildFile << """
project(':a') {
dependencies {
implementation project(':b')
implementation project(':c')
}
}
abstract class MakeGreen implements TransformAction {
interface Parameters extends TransformParameters {
@Input
String getExtension()
void setExtension(String value)
}
private Parameters conf
@Inject
MakeGreen(Parameters conf) {
this.conf = conf
}
void transform(TransformOutputs outputs) {
}
}
"""
expect:
succeeds(":a:resolve")
}
@Unroll
def "transform cannot use @InputArtifact to receive #propertyType"() {
settingsFile << """
include 'a', 'b'
"""
setupBuildWithColorTransformAction()
def typeName = propertyType instanceof Class ? propertyType.name : propertyType.toString()
buildFile << """
project(':a') {
dependencies {
implementation project(':b')
}
}
abstract class MakeGreen implements TransformAction {
@InputArtifact
abstract ${typeName} getInput()
void transform(TransformOutputs outputs) {
input
throw new RuntimeException("broken")
}
}
"""
when:
fails(":a:resolve")
then:
failure.assertHasDescription("A problem occurred evaluating root project")
failure.assertHasCause("Cannot register artifact transform MakeGreen (from {color=blue} to {color=green})")
failure.assertHasCause("Cannot use @InputArtifact annotation on property MakeGreen.getInput() of type ${typeName}. Allowed property types: java.io.File, org.gradle.api.provider.Provider.")
where:
propertyType << [FileCollection, new TypeToken>() {}.getType(), new TypeToken>() {}.getType()]
}
@Unroll
def "transform cannot use @InputArtifactDependencies to receive #propertyType"() {
settingsFile << """
include 'a', 'b'
"""
setupBuildWithColorTransformAction()
buildFile << """
project(':a') {
dependencies {
implementation project(':b')
}
}
abstract class MakeGreen implements TransformAction {
@${annotation.name}
abstract ${propertyType.name} getDependencies()
void transform(TransformOutputs outputs) {
dependencies
throw new RuntimeException("broken")
}
}
"""
when:
fails(":a:resolve")
then:
failure.assertHasDescription("A problem occurred evaluating root project")
failure.assertHasCause("Cannot register artifact transform MakeGreen (from {color=blue} to {color=green})")
failure.assertHasCause("Cannot use @InputArtifactDependencies annotation on property MakeGreen.getDependencies() of type ${propertyType.name}. Allowed property types: org.gradle.api.file.FileCollection.")
where:
annotation | propertyType
InputArtifactDependencies | File
InputArtifactDependencies | String
}
def "transform cannot use @Inject to receive input file"() {
settingsFile << """
include 'a', 'b', 'c'
"""
setupBuildWithColorTransformAction()
buildFile << """
project(':a') {
dependencies {
implementation project(':b')
}
}
abstract class MakeGreen implements TransformAction {
@Inject
abstract File getWorkspace()
void transform(TransformOutputs outputs) {
workspace
throw new RuntimeException("broken")
}
}
"""
when:
fails(":a:resolve")
then:
// Documents existing behaviour. Should fail eagerly and with a better error message
failure.assertHasDescription("Execution failed for task ':a:resolve'.")
failure.assertHasCause("Execution failed for MakeGreen: ${file('b/build/b.jar')}.")
failure.assertHasCause("No service of type class ${File.name} available.")
}
def "task implementation cannot use cacheable transform annotation"() {
buildFile << """
@CacheableTransform
class MyTask extends DefaultTask {
@TaskAction void execute() {}
}
tasks.create('broken', MyTask)
"""
expect:
fails('broken')
failure.assertHasDescription("A problem was found with the configuration of task ':broken' (type 'MyTask').")
failure.assertHasCause("Type 'MyTask': Cannot use @CacheableTransform on type. This annotation can only be used with TransformAction types.")
}
def "task @Nested bean cannot use cacheable annotations"() {
buildFile << """
class MyTask extends DefaultTask {
@Nested
Options getThing() { new Options() }
@TaskAction
void go() { }
}
@CacheableTransform @CacheableTask
class Options {
}
tasks.create('broken', MyTask)
"""
expect:
// Probably should be eager
fails('broken')
failure.assertHasDescription("Some problems were found with the configuration of task ':broken' (type 'MyTask').")
failure.assertHasCause("Type 'Options': Cannot use @CacheableTask on type. This annotation can only be used with Task types.")
failure.assertHasCause("Type 'Options': Cannot use @CacheableTransform on type. This annotation can only be used with TransformAction types.")
}
@Unroll
def "task implementation cannot use injection annotation @#annotation.simpleName"() {
buildFile << """
class MyTask extends DefaultTask {
@${annotation.name}
File getThing() { null }
}
tasks.create('broken', MyTask)
"""
expect:
fails('broken')
failure.assertHasDescription("A problem occurred evaluating root project")
failure.assertHasCause("Could not create task of type 'MyTask'.")
failure.assertHasCause("Could not generate a decorated class for class MyTask.")
failure.assertHasCause("Cannot use @${annotation.simpleName} annotation on method MyTask.getThing().")
where:
annotation << [InputArtifact, InputArtifactDependencies]
}
void assertPropertyValidationErrors(Map validationErrors) {
int count = 0
validationErrors.each { propertyName, errorMessageOrMessages ->
def errorMessages = errorMessageOrMessages instanceof Iterable ? [*errorMessageOrMessages] : [errorMessageOrMessages]
errorMessages.each { errorMessage ->
count++
failure.assertHasCause("Property '${propertyName}' ${errorMessage}.")
}
}
assert errorOutput.count("> Property") == count
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy