All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.gradle.model.internal.manage.schema.extract.ModelSchemaUtils Maven / Gradle / Ivy

There is a newer version: 8.11.1
Show newest version
/*
 * Copyright 2015 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.model.internal.manage.schema.extract;

import com.google.common.base.Equivalence;
import com.google.common.base.Function;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableListMultimap;
import com.google.common.collect.ImmutableSortedMap;
import com.google.common.collect.ListMultimap;
import com.google.common.collect.Multimaps;
import com.google.common.collect.Ordering;
import groovy.lang.GroovyObject;
import org.gradle.internal.reflect.GroovyMethods;
import org.gradle.internal.reflect.Types.TypeVisitor;
import org.gradle.model.Managed;

import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.Proxy;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.Map;

import static org.gradle.internal.reflect.Methods.SIGNATURE_EQUIVALENCE;
import static org.gradle.internal.reflect.Types.walkTypeHierarchy;

public class ModelSchemaUtils {
    public static final List> IGNORED_OBJECT_TYPES = ImmutableList.of(Object.class, GroovyObject.class);

    /**
     * Returns all candidate methods for schema generation declared by the given type and its super-types indexed by name.
     *
     * 

Overriding methods are not folded like in the case of {@link Class#getMethods()}. This allows * the caller to identify annotations declared at different levels in the hierarchy, and also to identify all * the classes declaring a certain method.

* *

Method candidates exclude:

*
    *
  • methods defined by {@link Object} and their overrides
  • *
  • methods defined by {@link GroovyObject} and their overrides
  • *
  • synthetic methods
  • *
* *

Methods are returned in the order of their specialization, most specialized methods first.

*/ public static CandidateMethods getCandidateMethods(Class clazz) { final ImmutableListMultimap.Builder methodsByNameBuilder = ImmutableListMultimap.builder(); walkTypeHierarchy(clazz, IGNORED_OBJECT_TYPES, new TypeVisitor() { @Override public void visitType(Class type) { Method[] declaredMethods = type.getDeclaredMethods(); // Sort of determinism Arrays.sort(declaredMethods, Ordering.usingToString()); for (Method method : declaredMethods) { if (ModelSchemaUtils.isIgnoredMethod(method)) { continue; } methodsByNameBuilder.put(method.getName(), method); } } }); ImmutableListMultimap methodsByName = methodsByNameBuilder.build(); ImmutableSortedMap.Builder, Collection>> candidatesBuilder = ImmutableSortedMap.naturalOrder(); for (String methodName : methodsByName.keySet()) { ImmutableList methodsWithSameName = methodsByName.get(methodName); ListMultimap, Method> equivalenceIndex = Multimaps.index(methodsWithSameName, new Function>() { @Override public Equivalence.Wrapper apply(Method method) { return SIGNATURE_EQUIVALENCE.wrap(method); } }); candidatesBuilder.put(methodName, equivalenceIndex.asMap()); } return new CandidateMethods(candidatesBuilder.build()); } private static boolean isIgnoredMethod(Method method) { int modifiers = method.getModifiers(); if (method.isSynthetic() || Modifier.isStatic(modifiers)) { return true; } // Ignore overrides of Object and GroovyObject methods return GroovyMethods.isObjectMethod(method); } /** * Tries to find the most specific declaration of a method that is not declared in a {@link Proxy} class. * Mock objects generated via {@link Proxy#newProxyInstance(ClassLoader, Class[], java.lang.reflect.InvocationHandler)} * lose their generic type parameters and can confuse schema extraction. This way we can ignore these * declarations, and use the ones from the proxied interfaces instead. * * @param declaringMethods declarations of the same method from different types in the type hierarchy. They are * expected to be in order of specificity, i.e. overrides preceding overridden declarations. * @return the most specific declaration of the method. * @throws IllegalArgumentException if no declaration can be found. */ public static Method findMostSpecificMethod(Iterable declaringMethods) { for (Method method : declaringMethods) { if (Proxy.isProxyClass(method.getDeclaringClass())) { continue; } return method; } throw new IllegalArgumentException("Cannot find most-specific declaration of method. Declarations checked: " + declaringMethods); } /** * Returns whether the most specific of the given methods has been declared in a @{@link Managed} type or not. */ public static boolean isMethodDeclaredInManagedType(Iterable declarations) { Method mostSpecificDeclaration = findMostSpecificMethod(declarations); return isMethodDeclaredInManagedType(mostSpecificDeclaration); } /** * Returns whether the method has been declared in a @{@link Managed} type or not. */ public static boolean isMethodDeclaredInManagedType(Method method) { return method.getDeclaringClass().isAnnotationPresent(Managed.class); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy