org.gradle.architecture.test.ProviderMigrationArchitectureTest 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 2022 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.architecture.test;
import com.tngtech.archunit.base.DescribedPredicate;
import com.tngtech.archunit.base.HasDescription;
import com.tngtech.archunit.core.domain.JavaClass;
import com.tngtech.archunit.core.domain.JavaMethod;
import com.tngtech.archunit.core.domain.properties.HasSourceCodeLocation;
import com.tngtech.archunit.junit.AnalyzeClasses;
import com.tngtech.archunit.junit.ArchTest;
import com.tngtech.archunit.lang.ArchCondition;
import com.tngtech.archunit.lang.ArchRule;
import com.tngtech.archunit.lang.ConditionEvents;
import com.tngtech.archunit.lang.SimpleConditionEvent;
import com.tngtech.archunit.lang.conditions.ArchPredicates;
import org.gradle.StartParameter;
import org.gradle.api.DefaultTask;
import org.gradle.api.Task;
import org.gradle.api.artifacts.Configuration;
import org.gradle.api.file.ConfigurableFileCollection;
import org.gradle.api.file.FileCollection;
import org.gradle.api.launcher.cli.WelcomeMessageConfiguration;
import org.gradle.api.provider.Property;
import org.gradle.api.provider.Provider;
import org.gradle.api.resources.TextResource;
import org.gradle.internal.reflect.PropertyAccessorType;
import javax.inject.Inject;
import static com.tngtech.archunit.base.DescribedPredicate.not;
import static com.tngtech.archunit.core.domain.JavaClass.Predicates.assignableTo;
import static com.tngtech.archunit.core.domain.JavaMember.Predicates.declaredIn;
import static com.tngtech.archunit.core.domain.properties.CanBeAnnotated.Predicates.annotatedWith;
import static com.tngtech.archunit.lang.conditions.ArchPredicates.are;
import static com.tngtech.archunit.lang.syntax.ArchRuleDefinition.methods;
import static org.gradle.architecture.test.ArchUnitFixture.freeze;
import static org.gradle.architecture.test.ArchUnitFixture.public_api_methods;
@AnalyzeClasses(packages = "org.gradle")
public class ProviderMigrationArchitectureTest {
private static final DescribedPredicate getters = new DescribedPredicate("getters") {
@Override
public boolean test(JavaMethod input) {
PropertyAccessorType accessorType = PropertyAccessorType.fromName(input.getName());
return accessorType == PropertyAccessorType.GET_GETTER || accessorType == PropertyAccessorType.IS_GETTER;
}
};
private static final DescribedPredicate class_with_any_mutable_property = new DescribedPredicate("class with any mutable property") {
@Override
public boolean test(JavaClass input) {
return input.getAllMethods().stream()
.filter(getters)
.anyMatch(ProviderMigrationArchitectureTest::hasSetter);
}
};
private static final DescribedPredicate mutable_public_API_properties = ArchPredicates.are(public_api_methods)
.and(not(declaredIn(assignableTo(Task.class))))
.and(not(declaredIn(StartParameter.class)))
.and(not(declaredIn(WelcomeMessageConfiguration.class))) // used in StartParameter
.and(not(declaredIn(Configuration.class)))
.and(not(declaredIn(FileCollection.class)))
.and(not(declaredIn(ConfigurableFileCollection.class)))
.and(are(declaredIn(class_with_any_mutable_property)))
.and(are(getters))
.and(not(annotatedWith(Inject.class)))
.as("mutable public API properties");
@SuppressWarnings("deprecation")
private static final DescribedPredicate task_properties = ArchPredicates.are(public_api_methods)
.and(declaredIn(assignableTo(Task.class)))
.and(are(getters))
.and(not(annotatedWith(Inject.class)))
.and(not(declaredIn(Task.class)))
.and(not(declaredIn(DefaultTask.class)))
.and(not(declaredIn(org.gradle.api.internal.AbstractTask.class)))
.as("task properties");
@ArchTest
public static final ArchRule mutable_public_api_properties_should_be_providers = freeze(methods()
.that(are(mutable_public_API_properties))
.and().doNotHaveRawReturnType(TextResource.class)
.and().doNotHaveRawReturnType(assignableTo(FileCollection.class))
.should(haveProviderReturnType()));
@ArchTest
public static final ArchRule mutable_public_api_properties_should_be_configurable_file_collections = freeze(methods()
.that(are(mutable_public_API_properties))
.and().haveRawReturnType(assignableTo(FileCollection.class))
.should(haveFileCollectionReturnType()));
@ArchTest
public static final ArchRule mutable_public_api_properties_should_not_use_text_resources = freeze(methods()
.that(are(mutable_public_API_properties))
.should().notHaveRawReturnType(TextResource.class));
@ArchTest
public static final ArchRule public_api_task_properties_are_providers = freeze(methods()
.that(are(task_properties))
.and().doNotHaveRawReturnType(TextResource.class)
.and().doNotHaveRawReturnType(assignableTo(FileCollection.class))
.should(haveProviderReturnType()));
@ArchTest
public static final ArchRule public_api_task_file_properties_are_configurable_file_collections = freeze(methods()
.that(are(task_properties))
.and().haveRawReturnType(assignableTo(FileCollection.class))
.should(haveFileCollectionReturnType()));
@ArchTest
public static final ArchRule public_api_task_properties_should_not_use_text_resources = freeze(methods()
.that(are(task_properties))
.should().notHaveRawReturnType(TextResource.class));
private static HaveLazyReturnType haveProviderReturnType() {
return new HaveLazyReturnType(Property.class, Provider.class);
}
private static HaveLazyReturnType haveFileCollectionReturnType() {
return new HaveLazyReturnType(ConfigurableFileCollection.class, FileCollection.class);
}
public static class HaveLazyReturnType extends ArchCondition {
private final Class> mutableType;
private final Class> immutableType;
public HaveLazyReturnType(Class> mutableType, Class> immutableType) {
super("have return type " + immutableType.getSimpleName());
this.mutableType = mutableType;
this.immutableType = immutableType;
}
@Override
public void check(JavaMethod javaMethod, ConditionEvents events) {
boolean hasSetter = hasSetter(javaMethod);
Class> expectedReturnType = hasSetter ? mutableType : immutableType;
boolean satisfied = javaMethod.getRawReturnType().isAssignableTo(expectedReturnType);
String message = createMessage(javaMethod, (satisfied ? "has " : "does not have ") + "raw return type assignable to " + expectedReturnType.getName());
events.add(new SimpleConditionEvent(javaMethod, satisfied, message));
}
private static String createMessage(T object, String message) {
return object.getDescription() + " " + message + " in " + object.getSourceCodeLocation();
}
}
private static boolean hasSetter(JavaMethod input) {
PropertyAccessorType accessorType = PropertyAccessorType.fromName(input.getName());
String propertyNameFromGetter = accessorType.propertyNameFor(input.getName());
return input.getOwner().getAllMethods().stream()
.filter(method -> PropertyAccessorType.fromName(method.getName()) == PropertyAccessorType.SETTER)
.anyMatch(method -> PropertyAccessorType.SETTER.propertyNameFor(method.getName()).equals(propertyNameFromGetter));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy