org.gradle.integtests.fixtures.ToBeFixedForConfigurationCacheExtension.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 2019 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.fixtures
import org.gradle.integtests.fixtures.executer.GradleContextualExecuter
import org.gradle.internal.reflect.ClassInspector
import org.gradle.test.fixtures.ResettableExpectations
import org.opentest4j.TestAbortedException
import org.spockframework.runtime.extension.IAnnotationDrivenExtension
import org.spockframework.runtime.extension.IMethodInterceptor
import org.spockframework.runtime.extension.IMethodInvocation
import org.spockframework.runtime.model.FeatureInfo
import java.lang.reflect.Field
import java.util.concurrent.atomic.AtomicBoolean
import java.util.function.Predicate
import static org.gradle.integtests.fixtures.ToBeFixedForConfigurationCache.Skip.DO_NOT_SKIP
class ToBeFixedForConfigurationCacheExtension implements IAnnotationDrivenExtension {
@Override
void visitFeatureAnnotation(ToBeFixedForConfigurationCache annotation, FeatureInfo feature) {
if (GradleContextualExecuter.isNotConfigCache()) {
return
}
if (!isEnabledSpec(annotation, feature)) {
return
}
if (annotation.skip() != DO_NOT_SKIP) {
feature.skipped = true
return
}
if (feature.isParameterized()) {
feature.addInterceptor(new ToBeFixedIterationInterceptor(annotation.iterationMatchers()))
} else {
feature.getFeatureMethod().addInterceptor(new ToBeFixedInterceptor())
}
}
private static class ToBeFixedInterceptor implements IMethodInterceptor {
@Override
void intercept(IMethodInvocation invocation) throws Throwable {
if (failsAsExpected(invocation)) {
return
}
throw new UnexpectedSuccessException()
}
}
private static class ToBeFixedIterationInterceptor implements IMethodInterceptor {
private final String[] iterationMatchers
ToBeFixedIterationInterceptor(String[] iterationMatchers) {
this.iterationMatchers = iterationMatchers
}
@Override
void intercept(IMethodInvocation invocation) throws Throwable {
final AtomicBoolean pass = new AtomicBoolean()
invocation.getFeature().getFeatureMethod().interceptors.add(
0,
new InnerIterationInterceptor(pass, iterationMatchers)
)
try {
invocation.proceed()
} catch (Throwable ex) {
expectedFailure(ex)
pass.set(true)
}
if (pass.get()) {
throw new TestAbortedException("Failed as expected.")
} else {
throw new UnexpectedSuccessException()
}
}
private static class InnerIterationInterceptor implements IMethodInterceptor {
private final AtomicBoolean pass
private final String[] iterationMatchers
InnerIterationInterceptor(AtomicBoolean pass, String[] iterationMatchers) {
this.pass = pass
this.iterationMatchers = iterationMatchers
}
@Override
void intercept(IMethodInvocation invocation) throws Throwable {
if (iterationMatches(iterationMatchers, invocation.iteration.displayName)) {
if (failsAsExpected(invocation)) {
pass.set(true)
}
} else {
invocation.proceed()
}
}
}
}
private static boolean failsAsExpected(IMethodInvocation invocation) {
try {
invocation.proceed()
} catch (Throwable ex) {
expectedFailure(ex)
ignoreCleanupAssertionsOf(invocation)
return true
}
// Trigger validation failures early so they can still fail the test the usual way
try {
allResettableExpectationsOf(invocation.instance).forEach { expectations ->
expectations.resetExpectations()
}
} catch (Throwable ex) {
expectedFailure(ex)
ignoreCleanupAssertionsOf(invocation)
return true
}
return false
}
private static ignoreCleanupAssertionsOf(IMethodInvocation invocation) {
def instance = invocation.instance
if (instance instanceof AbstractIntegrationSpec) {
instance.ignoreCleanupAssertions()
}
allResettableExpectationsOf(instance).forEach { expectations ->
try {
expectations.resetExpectations()
} catch (Throwable error) {
error.printStackTrace()
}
}
}
private static List allResettableExpectationsOf(instance) {
allInstanceFieldsOf(instance).findResults { field ->
try {
def fieldValue = field.tap { accessible = true }.get(instance)
fieldValue instanceof ResettableExpectations ? fieldValue : null
} catch (Exception ignored) {
null
}
}
}
private static Collection allInstanceFieldsOf(instance) {
ClassInspector.inspect(instance.getClass()).instanceFields
}
private static void expectedFailure(Throwable ex) {
System.err.println("Failed with configuration cache as expected:")
ex.printStackTrace()
}
private static boolean isEnabledSpec(ToBeFixedForConfigurationCache annotation, FeatureInfo feature) {
isEnabledBottomSpec(annotation.bottomSpecs(), { it == feature.spec.bottomSpec.name })
}
static boolean isEnabledBottomSpec(String[] bottomSpecs, Predicate specNamePredicate) {
bottomSpecs.length == 0 || bottomSpecs.any { specNamePredicate.test(it) }
}
static boolean iterationMatches(String[] iterationMatchers, String iterationName) {
isAllIterations(iterationMatchers) || iterationMatchers.any { iterationName.matches(it) }
}
static boolean isAllIterations(String[] iterationMatchers) {
iterationMatchers.length == 0
}
static class UnexpectedSuccessException extends Exception {
UnexpectedSuccessException() {
super("Expected to fail with configuration cache, but succeeded!")
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy