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

org.perfectable.introspection.query.MethodQuery Maven / Gradle / Ivy

package org.perfectable.introspection.query;

import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.Type;
import java.util.Arrays;
import java.util.Comparator;
import java.util.HashSet;
import java.util.Objects;
import java.util.Set;
import java.util.function.Predicate;
import java.util.regex.Pattern;
import java.util.stream.Stream;

import org.checkerframework.checker.nullness.qual.Nullable;

import static java.util.Objects.requireNonNull;

/**
 * Iterable-like container that searches for methods in class.
 *
 * 

This is straightforward way to search for methods that have specified characteristics in provided class. * *

Instances of this class are immutable, each filtering produces new, modified instance. To obtain unrestricted * query, use {@link #of}. * *

To obtain results either iterate this class with {@link #iterator} (or in enhanced-for loop) or use one of * {@link #stream()}, {@link #unique()}, {@link #option()} or {@link #isPresent()}. * *

All methods present in class are searched, either declared in it directly, or inherited from parent classes. This * means that even methods that are overridden in the inheritance chain will be returned, so methods with the same * signature but different declaring class might be present multiple times. To avoid this behavior, use * {@link #notOverridden}. * *

Example usage, which injects all public instance methods in class "UserService" that have annotation "Inject" on * them, have one parameter, and does not return value (is void). Before passing field to the method, it is marked * as {@link Method#setAccessible accessible}. *

 *     MethodQuery.of(UserService.class)
 *         .requiringModifier(Modifier.PUBLIC)
 *         .excludingModifier(Modifier.STATIC)
 *         .annotatedBy(Inject.class)
 *         .parameters(ParametersFilter.count(1))
 *         .returningVoid()
 *         .asAccessible()
 *         .stream()
 *         .forEach(this::inject);
 * 
*/ @SuppressWarnings({ "DesignForExtension", // class is closed because of package-private constructor "ClassDataAbstractionCoupling" }) public abstract class MethodQuery extends ExecutableQuery { /** * Queries for methods in specified class. * * @param type class to search methods in * @return query that returns all methods in specified class. */ public static MethodQuery of(Class type) { return of(InheritanceQuery.of(type)); } /** * Queries for methods in classes contained in specified inheritance chain. * * @param type classes to search methods in * @return query that returns all methods in specified classes. */ public static MethodQuery of(InheritanceQuery type) { requireNonNull(type); return new InClasses<>(type); } @Override public MethodQuery named(String name) { requireNonNull(name); return new Named(this, name); } @Override public MethodQuery nameMatching(Pattern namePattern) { requireNonNull(namePattern); return new NameMatching(this, namePattern); } @Override public MethodQuery filter(Predicate filter) { requireNonNull(filter); return new Predicated(this, filter); } @Override public MethodQuery sorted(Comparator comparator) { requireNonNull(comparator); return new Sorted(this, comparator); } @Override public MethodQuery parameters(ParametersFilter parametersFilter) { requireNonNull(parametersFilter); return new Parameters(this, parametersFilter); } /** * Restricts query to methods that has a return type that is subtype of specified type. * *

This methods accepts every type, even {@link Void#TYPE}. In latter case, it will filter for methods that * does not return a value, i.e. void methods. For readability, use {@link #returningVoid}. * * @param type return type to restrict * @return copy of this query that will also filter by return type */ public MethodQuery returning(Type type) { return returning(TypeFilter.subtypeOf(type)); } /** * Restricts query to methods that has a return type that match the filter. * * @param typeFilter filter to restrict method return type with * @return copy of this query that will also filter by return type */ public MethodQuery returning(TypeFilter typeFilter) { requireNonNull(typeFilter); return new Returning(this, typeFilter); } /** * Restricts query to methods that doesn't return a value, i.e. are void. * * @apiNote This method could be named "notReturning", but its a matter of opinion if it would be more confusing * than current name. The void method actually returns, ends normally. * * @return copy of this query that will filter methods that have void return */ public MethodQuery returningVoid() { return returning(Void.TYPE); } /** * Restricts query for methods that are not overridden by other in the inheritance chain. * *

Normally, the query returns methods that are overridden in the inheritance chain - methods with the same * signature but different declaring class might be present multiple times. Using this method will change that * behavior. * * @return copy of this query that will filter methods that are not overridden */ public MethodQuery notOverridden() { return new NotOverriden(this); } @Override public MethodQuery annotatedWith(AnnotationFilter annotationFilter) { requireNonNull(annotationFilter); return new Annotated(this, annotationFilter); } @Override public MethodQuery requiringModifier(int requiredModifier) { return new RequiringModifier(this, requiredModifier); } @Override public MethodQuery excludingModifier(int excludedModifier) { return new ExcludingModifier(this, excludedModifier); } @Override public MethodQuery asAccessible() { return new AccessibleMarking(this); } MethodQuery() { // package extension only } private static final class InClasses extends MethodQuery { private final InheritanceQuery chain; InClasses(InheritanceQuery chain) { this.chain = chain; } @Override public Stream stream() { return this.chain.stream() .flatMap(testedClass -> Stream.of(testedClass.getDeclaredMethods())); } @Override public boolean contains(@Nullable Object candidate) { if (!(candidate instanceof Method)) { return false; } Method candidateMethod = (Method) candidate; @SuppressWarnings("unchecked") Class declaringClass = (Class) candidateMethod.getDeclaringClass(); return chain.contains(declaringClass); } } private abstract static class Filtered extends MethodQuery { private final MethodQuery parent; Filtered(MethodQuery parent) { this.parent = parent; } protected abstract boolean matches(Method candidate); @Override public Stream stream() { return this.parent.stream() .filter(this::matches); } @Override public boolean contains(@Nullable Object candidate) { if (!(candidate instanceof Method)) { return false; } Method candidateMethod = (Method) candidate; return matches(candidateMethod) && parent.contains(candidate); } } private static final class Predicated extends Filtered { private final Predicate filter; Predicated(MethodQuery parent, Predicate filter) { super(parent); this.filter = filter; } @Override protected boolean matches(Method candidate) { return this.filter.test(candidate); } } private static final class Sorted extends MethodQuery { private final MethodQuery parent; private final Comparator comparator; Sorted(MethodQuery parent, Comparator comparator) { this.parent = parent; this.comparator = comparator; } @Override public MethodQuery sorted(Comparator nextComparator) { @SuppressWarnings("unchecked") Comparator<@Nullable Object> casted = (Comparator<@Nullable Object>) nextComparator; return new Sorted(parent, this.comparator.thenComparing(casted)); } @Override public Stream stream() { return parent.stream().sorted(comparator); } @Override public boolean contains(@Nullable Object candidate) { return parent.contains(candidate); } } private static final class Named extends Filtered { private final String name; Named(MethodQuery parent, String name) { super(parent); this.name = name; } @Override protected boolean matches(Method candidate) { return this.name.equals(candidate.getName()); } } private static final class NameMatching extends Filtered { private final Pattern namePattern; NameMatching(MethodQuery parent, Pattern namePattern) { super(parent); this.namePattern = namePattern; } @Override protected boolean matches(Method candidate) { return this.namePattern.matcher(candidate.getName()).matches(); } } private static final class Parameters extends Filtered { private final ParametersFilter parametersFilter; Parameters(MethodQuery parent, ParametersFilter parametersFilter) { super(parent); this.parametersFilter = parametersFilter; } @Override protected boolean matches(Method candidate) { return parametersFilter.matches(candidate); } } private static final class Returning extends Filtered { private final TypeFilter typeFilter; Returning(MethodQuery parent, TypeFilter typeFilter) { super(parent); this.typeFilter = typeFilter; } @Override protected boolean matches(Method candidate) { return this.typeFilter.matches(candidate.getReturnType()); } } private static final class Annotated extends Filtered { private final AnnotationFilter annotationFilter; Annotated(MethodQuery parent, AnnotationFilter annotationFilter) { super(parent); this.annotationFilter = annotationFilter; } @Override protected boolean matches(Method candidate) { return this.annotationFilter.matches(candidate); } } private static final class RequiringModifier extends Filtered { private final int requiredModifier; RequiringModifier(MethodQuery parent, int requiredModifier) { super(parent); this.requiredModifier = requiredModifier; } @Override protected boolean matches(Method candidate) { return (candidate.getModifiers() & this.requiredModifier) != 0; } } private static final class ExcludingModifier extends Filtered { private final int excludedModifier; ExcludingModifier(MethodQuery parent, int excludedModifier) { super(parent); this.excludedModifier = excludedModifier; } @Override protected boolean matches(Method candidate) { return (candidate.getModifiers() & this.excludedModifier) == 0; } } private static final class AccessibleMarking extends MethodQuery { private final MethodQuery parent; AccessibleMarking(MethodQuery parent) { this.parent = parent; } @Override public Stream stream() { return parent.stream() .peek(element -> element.setAccessible(true)); } @Override public boolean contains(@Nullable Object candidate) { return parent.contains(candidate); } } private static final class NotOverriden extends MethodQuery { private final MethodQuery parent; NotOverriden(MethodQuery parent) { this.parent = parent; } @Override public Stream stream() { Set processedMethods = new HashSet<>(); return parent.stream() .filter(candidate -> processedMethods.stream() .filter(processed -> hasEquivalentSignature(candidate, processed)) .noneMatch(processed -> isOverriddenByAssumingSignature(candidate, processed))) .peek(processedMethods::add); } @Override public boolean contains(@Nullable Object candidate) { if (!(candidate instanceof Method)) { return false; } if (!parent.contains(candidate)) { return false; } Method candidateMethod = (Method) candidate; MethodQuery overriding = parent .parameters(ParametersFilter.typesExact(candidateMethod.getParameterTypes())) .named(candidateMethod.getName()) .filter(method -> !candidate.equals(method)) .filter(method -> isOverriddenByAssumingSignature(candidateMethod, method)); return !overriding.isPresent(); } private static boolean hasEquivalentSignature(Method left, Method right) { return left.getName().equals(right.getName()) && Arrays.equals(left.getParameterTypes(), right.getParameterTypes()); } private static boolean isOverriddenByAssumingSignature(Method method, Method potentialOverride) { int modifiers = method.getModifiers(); if (Modifier.isPrivate(modifiers)) { return false; } Class declaringClass = method.getDeclaringClass(); Class potentialOverrideDeclaringClass = potentialOverride.getDeclaringClass(); @Nullable Package methodPackage = declaringClass.getPackage(); @Nullable Package potentialOverridePackage = potentialOverrideDeclaringClass.getPackage(); boolean samePackage = Objects.equals(methodPackage, potentialOverridePackage); if (!Modifier.isProtected(modifiers) && !Modifier.isPublic(modifiers) && !samePackage) { return false; } return declaringClass.isAssignableFrom(potentialOverrideDeclaringClass); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy