io.kubernetes.client.util.ModelMapper Maven / Gradle / Ivy
/*
Copyright 2020 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.util;
import io.kubernetes.client.Discovery;
import io.kubernetes.client.apimachinery.GroupVersionKind;
import io.kubernetes.client.apimachinery.GroupVersionResource;
import io.kubernetes.client.common.KubernetesListObject;
import io.kubernetes.client.common.KubernetesObject;
import io.kubernetes.client.openapi.ApiException;
import io.kubernetes.client.util.exception.IncompleteDiscoveryException;
import java.io.File;
import java.io.IOException;
import java.net.JarURLConnection;
import java.net.URI;
import java.net.URL;
import java.net.URLDecoder;
import java.time.Duration;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.Optional;
import java.util.concurrent.ConcurrentHashMap;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;
import org.apache.commons.lang3.tuple.ImmutablePair;
import org.apache.commons.lang3.tuple.Pair;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Model mapper loads pre-built kubernetes models from classpath and manages their mapping between
* their apiVersion and kind.
*/
public class ModelMapper {
private static final Logger logger = LoggerFactory.getLogger(ModelMapper.class);
public static final Duration DEFAULT_DISCOVERY_REFRESH_INTERVAL = Duration.ofMinutes(30);
// initially loaded from classpath, can be incorrect
private static Map> preBuiltClassesByGVK = new HashMap<>();
// Model's api-group prefix to kubernetes api-group
private static Map preBuiltApiGroups = new HashMap<>();
// Model's api-version midfix to kubernetes api-version
private static List preBuiltApiVersions = new ArrayList<>();
private static BiDirectionalMap> classesByGVK =
new BiDirectionalMap<>();
private static BiDirectionalMap> classesByGVR =
new BiDirectionalMap<>();
private static Map, Boolean> isNamespacedByClasses = new ConcurrentHashMap<>();
private static Set lastAPIDiscovery = new HashSet<>();
// zero means api-discovery never done
private static volatile long nextDiscoveryRefreshTimeMillis = 0;
static {
try {
initModelMap();
} catch (Exception ex) {
logger.error("Unexpected exception while loading classes", ex);
}
}
/**
* Registering concrete model classes by its apiGroupVersion (e.g. "apps/v1") and kind (e.g.
* "Deployment").
*
* Note that the the so-called apiGroupVersion equals to the "apiVersion" in the kubenretes
* resource yamls.
*
* @deprecated This method is deprecated. Please use the alternative method with the correct
* parameters. For example, use {@link #addModelMap(String, String, String, Class)}
* instead.
*/
@Deprecated
public static void addModelMap(String apiGroupVersion, String kind, Class> clazz) {
String[] parts = apiGroupVersion.split("/");
if (parts.length <= 1) { // legacy api group
addModelMap("", apiGroupVersion, kind, clazz);
} else {
addModelMap(parts[0], parts[1], kind, clazz);
}
}
/**
* Registering concrete model classes by its group, version and kind (e.g. "apps", "v1",
* "Deployment")
*
* @param group the group
* @param version the version
* @param kind the kind
* @param clazz the clazz
* @deprecated This method is deprecated and will be removed in a future version.
* Use the non-deprecated method(s) instead.
*/
@Deprecated
public static void addModelMap(String group, String version, String kind, Class> clazz) {
preBuiltAddModelMap(group, version, kind, clazz);
}
/**
* Registering concrete model classes by its group, version and kind (e.g. "apps", "v1",
* "Deployment")
*
* @param group the group
* @param version the version
* @param kind the kind
* @param resourceNamePlural the resource name plural
* @param objClass the clazz
*/
public static void addModelMap(
String group,
String version,
String kind,
String resourceNamePlural,
Class extends KubernetesObject> objClass,
Class extends KubernetesListObject> objListClass) {
// TODO(yue9944882): consistency between bi-directional maps
classesByGVK.add(new GroupVersionKind(group, version, kind), objClass);
classesByGVR.add(new GroupVersionResource(group, version, resourceNamePlural), objClass);
if (objListClass != null) {
classesByGVK.add(new GroupVersionKind(group, version, kind + "List"), objListClass);
}
}
/**
* Registering concrete model classes by its group, version, kind and isNamespaced (e.g. "apps",
* "v1", "Deployment", true).
*
* @param group the group
* @param version the version
* @param kind the kind
* @param resourceNamePlural the resource name plural
* @param isNamespacedResource the is namespaced resource
* @param objClass the clazz
*/
public static void addModelMap(
String group,
String version,
String kind,
String resourceNamePlural,
Boolean isNamespacedResource,
Class extends KubernetesObject> objClass) {
addModelMap(group, version, kind, resourceNamePlural, objClass, null);
isNamespacedByClasses.put(objClass, isNamespacedResource);
}
public static void addModelMap(
String group,
String version,
String kind,
String resourceNamePlural,
Boolean isNamespacedResource,
Class extends KubernetesObject> objClass,
Class extends KubernetesListObject> objListClass) {
addModelMap(group, version, kind, resourceNamePlural, objClass, objListClass);
isNamespacedByClasses.put(objClass, isNamespacedResource);
}
/**
* Gets the model classes by given apiGroupVersion (e.g. "apps/v1") and kind (e.g. "Deployment").
*
* @param apiGroupVersion the api version
* @param kind the kind
* @return the api type class
*/
public static Class> getApiTypeClass(String apiGroupVersion, String kind) {
String[] parts = apiGroupVersion.split("/");
String apiGroup;
String apiVersion;
if (parts.length == 1) {
// legacy group
apiGroup = "";
apiVersion = apiGroupVersion;
} else {
apiGroup = parts[0];
apiVersion = parts[1];
}
return getApiTypeClass(apiGroup, apiVersion, kind);
}
/**
* Gets the model classes by given group, version and kind (e.g. "apps", ""v1", "Deployment").
*
* @param group the group
* @param version the version
* @param kind the kind
* @return the api type class
*/
public static Class> getApiTypeClass(String group, String version, String kind) {
Class> clazz = classesByGVK.getByK(new GroupVersionKind(group, version, kind));
if (clazz != null) {
return clazz;
}
return preBuiltGetApiTypeClass(group, version, kind);
}
/**
* Gets the GVK by the given model class.
*
* @param clazz the clazz
* @return the group version kind by class
*/
public static GroupVersionKind getGroupVersionKindByClass(Class> clazz) {
return classesByGVK.getByV(clazz);
}
/**
* Gets the GVK by the given model class.
*
* @param clazz the clazz
* @return the group version kind by class
*/
public static GroupVersionResource getGroupVersionResourceByClass(Class> clazz) {
return classesByGVR.getByV(clazz);
}
/**
* Refreshes the model mapping by syncing up w/the api discovery info from the kubernetes
* apiserver. These mapping will be cached for {@link
* ModelMapper#DEFAULT_DISCOVERY_REFRESH_INTERVAL}.
*
* @param discovery the discovery
* @return the set
* @throws ApiException the api exception
*/
public static Set refresh(Discovery discovery) throws ApiException {
return refresh(discovery, DEFAULT_DISCOVERY_REFRESH_INTERVAL);
}
/**
* Refreshes the model mapping by syncing up w/the api discovery info from the kubernetes
* apiserver.
*
* Note: if model mappings can be incomplete if this method is never called.
*
* @param discovery the discovery
* @return the api-discovery set
* @throws ApiException the api exception
*/
public static Set refresh(Discovery discovery, Duration refreshInterval)
throws ApiException {
long nowMillis = System.currentTimeMillis();
if (nowMillis < nextDiscoveryRefreshTimeMillis) {
return lastAPIDiscovery;
}
Set apiResources = null;
try {
apiResources = discovery.findAll();
} catch (IncompleteDiscoveryException e) {
logger.warn("Error while getting all api resources, some api resources will not be refreshed", e);
apiResources = e.getDiscoveredResources();
}
for (Discovery.APIResource apiResource : apiResources) {
for (String version : apiResource.getVersions()) {
Class extends KubernetesObject> objClass =
(Class extends KubernetesObject>)
getApiTypeClass(apiResource.getGroup(), version, apiResource.getKind());
// no such classes registered in the ModelMapper, ignoring
if (objClass == null) {
continue;
}
Class extends KubernetesListObject> objListClass =
(Class extends KubernetesListObject>)
getApiTypeClass(apiResource.getGroup(), version, apiResource.getKind() + "List");
// sync up w/ the latest api discovery
addModelMap(
apiResource.getGroup(),
version,
apiResource.getKind(),
apiResource.getResourcePlural(),
apiResource.getNamespaced(),
objClass,
objListClass);
}
}
lastAPIDiscovery = apiResources;
nextDiscoveryRefreshTimeMillis = refreshInterval.toMillis() + nowMillis;
return lastAPIDiscovery;
}
public static boolean isApiDiscoveryRefreshed() {
return nextDiscoveryRefreshTimeMillis != 0;
}
static void preBuiltAddModelMap(String group, String version, String kind, Class> clazz) {
preBuiltClassesByGVK.put(new GroupVersionKind(group, version, kind), clazz);
}
public static Class> preBuiltGetApiTypeClass(String group, String version, String kind) {
Class> clazz = preBuiltClassesByGVK.get(new GroupVersionKind(group, version, kind));
if (clazz != null) {
return clazz;
}
return preBuiltClassesByGVK.get(new GroupVersionKind("", version, kind));
}
public static Optional preBuiltGetGroupVersionKindByClass(Class> clazz) {
return preBuiltClassesByGVK.entrySet().stream()
.filter(e -> clazz.equals(e.getValue()))
.map(Map.Entry::getKey)
.findFirst();
}
/**
* Checks whether the class is connected with a namespaced kubernetes resource.
*
* @param clazz the clazz
* @return the boolean
*/
public static Boolean isNamespaced(Class> clazz) {
return isNamespacedByClasses.get(clazz);
}
/**
* Find GroupVersionKind from fields in a Kubernetes Resource
*
* @param apiVersion The apiVersion field from the object
* @param kind The kind for the object
* @return the GroupVersionKind
*/
public static GroupVersionKind groupVersionKindFromApiVersionAndKind(
String apiVersion, String kind) {
int ix = apiVersion.indexOf("/");
String group = ix == -1 ? "" : apiVersion.substring(0, ix);
String version = apiVersion.substring(ix + 1);
return new GroupVersionKind(group, version, kind);
}
public static Discovery.APIResource findApiResourceByGroupVersionKind(GroupVersionKind gvk) {
Map apiResourcesByKey = new HashMap<>();
for (Discovery.APIResource apiResource : lastAPIDiscovery) {
for (String version : apiResource.getVersions()) {
String key = apiResource.getKind() + "/" + apiResource.getGroup() + "/" + version;
apiResourcesByKey.put(key, apiResource);
}
}
String key = gvk.getKind() + "/" + gvk.getGroup() + "/" + gvk.getVersion();
return apiResourcesByKey.get(key);
}
private static void initApiGroupMap() {
preBuiltApiGroups.put("Admissionregistration", "admissionregistration.k8s.io");
preBuiltApiGroups.put("Apiextensions", "apiextensions.k8s.io");
preBuiltApiGroups.put("Apiregistration", "apiregistration.k8s.io");
preBuiltApiGroups.put("Apps", "apps");
preBuiltApiGroups.put("Authentication", "authentication.k8s.io");
preBuiltApiGroups.put("Authorization", "authorization.k8s.io");
preBuiltApiGroups.put("Autoscaling", "autoscaling");
preBuiltApiGroups.put("Batch", "batch");
preBuiltApiGroups.put("Certificates", "certificates.k8s.io");
preBuiltApiGroups.put("Core", "");
preBuiltApiGroups.put("Extensions", "extensions");
preBuiltApiGroups.put("Events", "events.k8s.io");
preBuiltApiGroups.put("FlowControl", "flowcontrol.apiserver.k8s.io");
preBuiltApiGroups.put("Networking", "networking.k8s.io");
preBuiltApiGroups.put("Policy", "policy");
preBuiltApiGroups.put("RbacAuthorization", "rbac.authorization.k8s.io");
preBuiltApiGroups.put("Scheduling", "scheduling.k8s.io");
preBuiltApiGroups.put("Settings", "settings.k8s.io");
preBuiltApiGroups.put("Storage", "storage.k8s.io");
}
private static void initApiVersionList() {
// Order important
preBuiltApiVersions.add("V2beta1");
preBuiltApiVersions.add("V2beta2");
preBuiltApiVersions.add("V2alpha1");
preBuiltApiVersions.add("V1beta2");
preBuiltApiVersions.add("V1beta1");
preBuiltApiVersions.add("V1alpha1");
preBuiltApiVersions.add("V1");
preBuiltApiVersions.add("V2");
}
private static void initModelMap() throws IOException {
initApiGroupMap();
initApiVersionList();
for (String classInfo :
getClassNamesFromPackage(
Yaml.class.getClassLoader(), "io.kubernetes.client.openapi.models")) {
Class> clazz = loadClass(classInfo, Yaml.class.getClassLoader());
if (!KubernetesObject.class.isAssignableFrom(clazz)
&& !KubernetesListObject.class.isAssignableFrom(clazz)) {
continue;
}
Pair groupAndOther = getApiGroup(clazz.getSimpleName());
Pair versionAndOther = getApiVersion(groupAndOther.getRight());
String group = Strings.nullToEmpty(groupAndOther.getLeft());
String version = versionAndOther.getLeft();
String kind = versionAndOther.getRight();
preBuiltClassesByGVK.put(new GroupVersionKind(group, version, kind), clazz);
}
if (preBuiltClassesByGVK.size() == 0) {
logger.warn("No kubernetes api model classes found from classloader, " + "this may break automatic api discovery");
}
}
private static Class> loadClass(String name, ClassLoader classLoader) {
try {
return Class.forName(name, false, classLoader);
} catch (ClassNotFoundException e) {
return null;
}
}
private static Pair getApiGroup(String name) {
return preBuiltApiGroups.entrySet().stream()
.filter(e -> name.startsWith(e.getKey()))
.map(e -> new ImmutablePair<>(e.getValue(), name.substring(e.getKey().length())))
.findFirst()
.orElse(new ImmutablePair<>(null, name));
}
private static Pair getApiVersion(String name) {
return preBuiltApiVersions.stream()
.filter(v -> name.startsWith(v))
.map(v -> new ImmutablePair<>(v.toLowerCase(), name.substring(v.length())))
.findFirst()
.orElse(new ImmutablePair<>(null, name));
}
static class BiDirectionalMap {
private Map kvMap = new ConcurrentHashMap<>();
private Map vkMap = new ConcurrentHashMap<>();
void add(K k, V v) {
kvMap.put(k, v);
vkMap.put(v, k);
}
V getByK(K k) {
return kvMap.get(k);
}
K getByV(V v) {
return vkMap.get(v);
}
}
private static List getClassNamesFromPackage(ClassLoader classLoader, String pkg) throws IOException {
ArrayList names = new ArrayList<>();
String packageName = pkg.replace(".", "/");
URL packageURL = classLoader.getResource(packageName);
if (packageURL.getProtocol().equals("jar")) {
processJarPackage(packageURL, packageName, pkg, names);
} else {
processFilePackage(packageURL, pkg, names);
}
return names;
}
private static void processJarPackage(URL packageURL, String packageName, String pkg, ArrayList names) throws IOException {
String jarFileName = URLDecoder.decode(packageURL.getFile(), "UTF-8");
JarFile jf = null;
// jar: client in repository; nested: client in a fat jar
if (jarFileName.startsWith("jar:") || jarFileName.startsWith("nested:")) {
jf = ((JarURLConnection) packageURL.openConnection()).getJarFile();
}
// file: client is a file in target (unit test)
if (jarFileName.startsWith("file:") ) {
jarFileName = jarFileName.substring(5, jarFileName.indexOf("!"));
jf = new JarFile(jarFileName);
}
if (jf == null) {
logger.error("Loading classes from jar with error packageURL: {}", jarFileName);
return;
}
logger.info("Loading classes from jar {}", jarFileName);
Enumeration jarEntries = jf.entries();
while (jarEntries.hasMoreElements()) {
processJarEntry(jarEntries.nextElement(), packageName, pkg, names);
}
jf.close();
}
private static void processJarEntry(JarEntry jarEntry, String packageName, String pkg, ArrayList names) {
String entryName = jarEntry.getName();
if (entryName.startsWith(packageName) && entryName.length() > packageName.length() + 5) {
entryName = entryName.substring(packageName.length() + 1, entryName.lastIndexOf('.'));
names.add(pkg + "." + entryName);
}
}
private static void processFilePackage(URL packageURL, String pkg, ArrayList names) {
URI uri = URI.create(packageURL.toString());
File folder = new File(uri.getPath());
File[] contenuti = folder.listFiles();
if (contenuti == null) {
logger.warn("No files to load found in {}", folder.getPath());
return;
}
for (File actual : contenuti) {
processFileEntry(actual, pkg, names);
}
}
private static void processFileEntry(File actual, String pkg, ArrayList names) {
String entryName = actual.getName();
entryName = entryName.substring(0, entryName.lastIndexOf('.'));
names.add(pkg + "." + entryName);
}
}