org.gradle.api.internal.tasks.execution.ResolveTaskOutputCachingStateExecuterTest.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 2017 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.api.internal.tasks.execution
import org.gradle.api.GradleException
import org.gradle.api.internal.OverlappingOutputs
import org.gradle.api.internal.TaskInternal
import org.gradle.api.internal.TaskOutputCachingState
import org.gradle.api.internal.tasks.CacheableTaskOutputFilePropertySpec
import org.gradle.api.internal.tasks.TaskExecuter
import org.gradle.api.internal.tasks.TaskExecuterResult
import org.gradle.api.internal.tasks.TaskExecutionContext
import org.gradle.api.internal.tasks.TaskOutputCachingDisabledReasonCategory
import org.gradle.api.internal.tasks.TaskOutputFilePropertySpec
import org.gradle.api.internal.tasks.TaskStateInternal
import org.gradle.api.specs.Spec
import org.gradle.caching.internal.tasks.DefaultTaskOutputCachingBuildCacheKeyBuilder
import org.gradle.caching.internal.tasks.TaskOutputCachingBuildCacheKey
import org.gradle.internal.file.RelativeFilePathResolver
import org.gradle.internal.hash.HashCode
import org.gradle.internal.snapshot.impl.ImplementationSnapshot
import org.gradle.testing.internal.util.Specification
import javax.annotation.Nullable
class ResolveTaskOutputCachingStateExecuterTest extends Specification {
def task = Stub(TaskInternal)
def cacheableOutputProperty = Mock(CacheableTaskOutputFilePropertySpec)
def cacheKey = Stub(TaskOutputCachingBuildCacheKey) {
isValid() >> true
}
def relativeFilePathResolver = Mock(RelativeFilePathResolver)
def "error message contains which cacheIf spec failed to evaluate"() {
when:
resolveCachingState(
[cacheableOutputProperty],
cacheKey,
task,
[spec({ throw new RuntimeException() }, "Exception is thrown")],
[],
null
)
then:
GradleException e = thrown()
e.message.contains("Could not evaluate spec for 'Exception is thrown'.")
}
def "error message contains which doNotCacheIf spec failed to evaluate"() {
when:
resolveCachingState(
[cacheableOutputProperty],
cacheKey,
task,
[spec({ true })],
[spec({ "throw new RuntimeException()" }, "Exception is thrown")],
null
)
then:
GradleException e = thrown()
e.message.contains("Could not evaluate spec for 'Exception is thrown'.")
}
def "report no reason if the task is cacheable"() {
when:
def state = resolveCachingState(
[cacheableOutputProperty],
cacheKey,
task,
[spec({ true })],
[],
null
)
then:
state.enabled
state.disabledReasonCategory == null
state.disabledReason == null
}
def "caching is disabled with no outputs"() {
when:
def state = resolveCachingState(
[],
cacheKey,
task,
[spec({ true })],
[],
null
)
then:
!state.enabled
state.disabledReasonCategory == TaskOutputCachingDisabledReasonCategory.NO_OUTPUTS_DECLARED
state.disabledReason == "No outputs declared"
}
def "no cacheIf() means no caching"() {
when:
def state = resolveCachingState(
[cacheableOutputProperty],
cacheKey,
task,
[],
[],
null
)
then:
!state.enabled
state.disabledReasonCategory == TaskOutputCachingDisabledReasonCategory.NOT_ENABLED_FOR_TASK
state.disabledReason == "Caching has not been enabled for the task"
}
def "can turn caching off via cacheIf()"() {
when:
def state = resolveCachingState(
[cacheableOutputProperty],
cacheKey,
task,
[spec({ false }, "Cacheable test")],
[],
null
)
then:
!state.enabled
state.disabledReasonCategory == TaskOutputCachingDisabledReasonCategory.CACHE_IF_SPEC_NOT_SATISFIED
state.disabledReason == "'Cacheable test' not satisfied"
}
def "can turn caching off via doNotCacheIf()"() {
when:
def state = resolveCachingState(
[cacheableOutputProperty],
cacheKey,
task,
[spec({ true })],
[spec({ true }, "Uncacheable test")],
null
)
then:
!state.enabled
state.disabledReasonCategory == TaskOutputCachingDisabledReasonCategory.DO_NOT_CACHE_IF_SPEC_SATISFIED
state.disabledReason == "'Uncacheable test' satisfied"
}
def "caching is disabled for non-cacheable file outputs is reported"() {
when:
def state = resolveCachingState(
[Stub(TaskOutputFilePropertySpec) {
getPropertyName() >> "non-cacheable property"
}],
cacheKey,
task,
[spec({ true })],
[],
null
)
then:
!state.enabled
state.disabledReason == "Output property 'non-cacheable property' contains a file tree"
state.disabledReasonCategory == TaskOutputCachingDisabledReasonCategory.NON_CACHEABLE_TREE_OUTPUT
}
def "caching is disabled when cache key is invalid because of invalid task implementation"() {
def builder = new DefaultTaskOutputCachingBuildCacheKeyBuilder()
builder.appendTaskImplementation(ImplementationSnapshot.of("org.gradle.TaskType", null))
def invalidBuildCacheKey = builder.build()
when:
def state = resolveCachingState(
[cacheableOutputProperty],
invalidBuildCacheKey,
task,
[spec({ true })],
[],
null
)
then:
!state.enabled
state.disabledReason == "Task class was loaded with an unknown classloader (class 'org.gradle.TaskType')."
state.disabledReasonCategory == TaskOutputCachingDisabledReasonCategory.NON_CACHEABLE_TASK_IMPLEMENTATION
}
def "caching is disabled when cache key is invalid because of invalid task action"() {
def builder = new DefaultTaskOutputCachingBuildCacheKeyBuilder()
builder.appendTaskActionImplementations([ImplementationSnapshot.of('org.my.package.MyPlugin$$Lambda$1/23246642345', HashCode.fromInt(12345))])
def invalidBuildCacheKey = builder.build()
when:
def state = resolveCachingState(
[cacheableOutputProperty],
invalidBuildCacheKey,
task,
[spec({ true })],
[],
null
)
then:
!state.enabled
state.disabledReason == 'Task action was implemented by the Java lambda \'org.my.package.MyPlugin$$Lambda$1/23246642345\'. Using Java lambdas is not supported, use an (anonymous) inner class instead.'
state.disabledReasonCategory == TaskOutputCachingDisabledReasonCategory.NON_CACHEABLE_TASK_ACTION
}
def "caching is disabled when cache key is invalid because of invalid input"() {
def builder = new DefaultTaskOutputCachingBuildCacheKeyBuilder()
builder.inputPropertyNotCacheable("someProperty", 'was implemented by the Java lambda \'org.my.package.MyPlugin$$Lambda$5/342523421\'')
def invalidBuildCacheKey = builder.build()
when:
def state = resolveCachingState(
[cacheableOutputProperty],
invalidBuildCacheKey,
task,
[spec({ true })],
[],
null
)
then:
!state.enabled
state.disabledReason == 'Non-cacheable inputs: property \'someProperty\' was implemented by the Java lambda \'org.my.package.MyPlugin$$Lambda$5/342523421\''
state.disabledReasonCategory == TaskOutputCachingDisabledReasonCategory.NON_CACHEABLE_INPUTS
}
def "caching is disabled when cache key is invalid because of overlapping outputs"() {
def overlappingOutputs = new OverlappingOutputs("someProperty", "path/to/outputFile")
when:
def state = resolveCachingState(
[cacheableOutputProperty],
cacheKey,
task,
[spec({ true })],
[],
overlappingOutputs
)
then:
1 * relativeFilePathResolver.resolveAsRelativePath(overlappingOutputs.overlappedFilePath) >> "relative/path"
!state.enabled
state.disabledReason == "Gradle does not know how file 'relative/path' was created (output property 'someProperty'). Task output caching requires exclusive access to output paths to guarantee correctness."
state.disabledReasonCategory == TaskOutputCachingDisabledReasonCategory.OVERLAPPING_OUTPUTS
}
def "when build cache is disabled, state is DISABLED"() {
def taskState = Mock(TaskStateInternal)
def taskContext = Mock(TaskExecutionContext)
def delegate = Mock(TaskExecuter)
def executer = new ResolveTaskOutputCachingStateExecuter(false, relativeFilePathResolver, delegate)
when:
executer.execute(task, taskState, taskContext)
then:
1 * taskState.setTaskOutputCaching({ !it.enabled } as TaskOutputCachingState)
then:
1 * delegate.execute(task, taskState, taskContext) >> TaskExecuterResult.NO_REUSED_OUTPUT
0 * _
}
static def spec(Spec spec, String description = "test cacheIf()") {
new SelfDescribingSpec(spec, description)
}
TaskOutputCachingState resolveCachingState(
Collection outputFileProperties,
TaskOutputCachingBuildCacheKey buildCacheKey,
TaskInternal task,
Collection> cacheIfSpecs,
Collection> doNotCacheIfSpecs,
@Nullable OverlappingOutputs overlappingOutputs
) {
return ResolveTaskOutputCachingStateExecuter.resolveCachingState(
!outputFileProperties.isEmpty(),
outputFileProperties,
buildCacheKey,
task,
cacheIfSpecs,
doNotCacheIfSpecs,
overlappingOutputs,
relativeFilePathResolver
)
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy