io.kubernetes.client.fluent.Visitors Maven / Gradle / Ivy
/*
Copyright 2022 The Kubernetes 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 io.kubernetes.client.fluent;
import java.lang.reflect.Array;
import java.lang.reflect.GenericArrayType;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.lang.reflect.TypeVariable;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
public final class Visitors {
private Visitors() {
// Utility Class
}
public static Visitor newVisitor(
Class type, io.kubernetes.client.fluent.Visitor visitor) {
return new DelegatingVisitor(type, visitor);
}
protected static List getTypeArguments(
java.lang.Class baseClass, java.lang.Class extends T> childClass) {
Map resolvedTypes = new LinkedHashMap();
Type type = childClass;
// start walking up the inheritance hierarchy until we hit baseClass
for (Class cl = getClass(type);
cl != null && cl != Object.class && !baseClass.getName().equals(cl.getName());
cl = getClass(type)) {
if (type instanceof Class) {
Class c = (Class) type;
Optional nextInterface =
baseClass.isInterface()
? getMatchingInterface(baseClass, c.getGenericInterfaces())
: Optional.empty();
if (nextInterface.isPresent()) {
type = nextInterface.get();
} else {
// there is no useful information for us in raw types, so just keep going.
type = ((Class) type).getGenericSuperclass();
}
} else if (type instanceof ParameterizedType) {
ParameterizedType parameterizedType = (ParameterizedType) type;
Type t = parameterizedType.getRawType();
if (t instanceof Class) {
Class> rawType = (Class) parameterizedType.getRawType();
Type[] actualTypeArguments = parameterizedType.getActualTypeArguments();
TypeVariable>[] typeParameters = rawType.getTypeParameters();
for (int i = 0; i < actualTypeArguments.length; i++) {
resolvedTypes.put(typeParameters[i], actualTypeArguments[i]);
}
if (!baseClass.equals(rawType)) {
type = rawType.getGenericSuperclass();
}
} else {
break;
}
}
}
// finally, for each actual type argument provided to baseClass, determine (if possible)
// the raw class for that type argument.
Type[] actualTypeArguments;
if (type instanceof Class) {
actualTypeArguments = ((Class) type).getTypeParameters();
} else {
actualTypeArguments = ((ParameterizedType) type).getActualTypeArguments();
}
List typeArgumentsAsClasses = new ArrayList();
// resolve types by chasing down type variables.
for (Type baseType : actualTypeArguments) {
while (resolvedTypes.containsKey(baseType)) {
baseType = resolvedTypes.get(baseType);
}
typeArgumentsAsClasses.add(getClass(baseType));
}
return typeArgumentsAsClasses;
}
private static String getRawName(Type type) {
return type instanceof ParameterizedType
? ((ParameterizedType) type).getRawType().getTypeName()
: type.getTypeName();
}
private static java.lang.Class> getClass(java.lang.reflect.Type type) {
if (type instanceof Class) {
return (Class) type;
} else if (type instanceof ParameterizedType) {
return getClass(((ParameterizedType) type).getRawType());
} else if (type instanceof GenericArrayType) {
Type componentType = ((GenericArrayType) type).getGenericComponentType();
Class> componentClass = getClass(componentType);
if (componentClass != null) {
return Array.newInstance(componentClass, 0).getClass();
} else {
return null;
}
} else {
return null;
}
}
private static Optional getMatchingInterface(
java.lang.Class> targetInterface, java.lang.reflect.Type... candidates) {
if (candidates == null || candidates.length == 0) {
return Optional.empty();
}
Optional match =
Arrays.stream(candidates)
.filter(c -> getRawName(c).equals(targetInterface.getTypeName()))
.findFirst();
if (match.isPresent()) {
return match;
} else {
for (Type candidate : candidates) {
if (candidate instanceof Class) {
Class c = (Class) candidate;
Optional next = getMatchingInterface(targetInterface, c.getGenericInterfaces());
if (next.isPresent()) {
return Optional.of(c);
}
}
}
return Optional.empty();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy