// Generated by delombok at Sun Feb 26 12:31:38 KST 2017
package scouter.bytebuddy.matcher;
import scouter.bytebuddy.description.method.MethodDescription;
import scouter.bytebuddy.description.type.TypeDefinition;
import scouter.bytebuddy.description.type.TypeDescription;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
/**
* A matcher that checks if any super type of a type declares a method with the same shape of a matched method.
*
* @param <T> The type of the matched entity.
*/
public class MethodOverrideMatcher<T extends MethodDescription> extends ElementMatcher.Junction.AbstractBase<T> {
/**
* The matcher that is to be applied to the type that declares a method of the same shape.
*/
private final ElementMatcher<? super TypeDescription.Generic> matcher;
/**
* Creates a new method override matcher.
*
* @param matcher The matcher that is to be applied to the type that declares a method of the same shape.
*/
public MethodOverrideMatcher(ElementMatcher<? super TypeDescription.Generic> matcher) {
this.matcher = matcher;
}
@Override
public boolean matches(T target) {
Set<TypeDescription> duplicates = new HashSet<TypeDescription>();
for (TypeDefinition typeDefinition : target.getDeclaringType()) {
if (matches(target, typeDefinition) || matches(target, typeDefinition.getInterfaces(), duplicates)) {
return true;
}
}
return false;
}
/**
* Matches a method against a list of types.
*
* @param target The method that is matched as a target.
* @param typeDefinitions The type definitions to check if they declare a method with the same signature as {@code target}.
* @param duplicates A set containing duplicate interfaces that do not need to be revisited.
* @return {@code true} if any type defines a method with the same signature as the {@code target} method.
*/
private boolean matches(MethodDescription target, List<? extends TypeDefinition> typeDefinitions, Set<TypeDescription> duplicates) {
for (TypeDefinition anInterface : typeDefinitions) {
if (duplicates.add(anInterface.asErasure()) && (matches(target, anInterface) || matches(target, anInterface.getInterfaces(), duplicates))) {
return true;
}
}
return false;
}
/**
* Checks if a type declares a method with the same signature as {@code target}.
*
* @param target The method to be checked.
* @param typeDefinition The type to check for declaring a method with the same signature as {@code target}.
* @return {@code true} if the supplied type declares a compatible method.
*/
private boolean matches(MethodDescription target, TypeDefinition typeDefinition) {
for (MethodDescription methodDescription : typeDefinition.getDeclaredMethods().filter(ElementMatchers.isVirtual())) {
if (methodDescription.asSignatureToken().equals(target.asSignatureToken())) {
if (matcher.matches(typeDefinition.asGenericType())) {
return true;
} else {
break;
}
}
}
return false;
}
@Override
public String toString() {
return "isOverriddenFrom(" + matcher + ")";
}
@java.lang.Override
@java.lang.SuppressWarnings("all")
@javax.annotation.Generated("lombok")
public boolean equals(final java.lang.Object o) {
if (o == this) return true;
if (!(o instanceof MethodOverrideMatcher)) return false;
final MethodOverrideMatcher<?> other = (MethodOverrideMatcher<?>) o;
if (!other.canEqual((java.lang.Object) this)) return false;
final java.lang.Object this$matcher = this.matcher;
final java.lang.Object other$matcher = other.matcher;
if (this$matcher == null ? other$matcher != null : !this$matcher.equals(other$matcher)) return false;
return true;
}
@java.lang.SuppressWarnings("all")
@javax.annotation.Generated("lombok")
protected boolean canEqual(final java.lang.Object other) {
return other instanceof MethodOverrideMatcher;
}
@java.lang.Override
@java.lang.SuppressWarnings("all")
@javax.annotation.Generated("lombok")
public int hashCode() {
final int PRIME = 59;
int result = 1;
final java.lang.Object $matcher = this.matcher;
result = result * PRIME + ($matcher == null ? 43 : $matcher.hashCode());
return result;
}
}