
io.fabric8.kubernetes.client.utils.KubernetesSerialization Maven / Gradle / Ivy
/*
* Copyright (C) 2015 Red Hat, Inc.
*
* 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.fabric8.kubernetes.client.utils;
import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.annotation.JsonInclude.Include;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.DeserializationConfig;
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.JsonDeserializer;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.JsonSerializer;
import com.fasterxml.jackson.databind.KeyDeserializer;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationConfig;
import com.fasterxml.jackson.databind.SerializationFeature;
import com.fasterxml.jackson.databind.cfg.HandlerInstantiator;
import com.fasterxml.jackson.databind.cfg.MapperConfig;
import com.fasterxml.jackson.databind.introspect.Annotated;
import com.fasterxml.jackson.databind.jsontype.TypeIdResolver;
import com.fasterxml.jackson.databind.jsontype.TypeResolverBuilder;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
import io.fabric8.kubernetes.api.model.KubernetesResource;
import io.fabric8.kubernetes.api.model.KubernetesResourceList;
import io.fabric8.kubernetes.api.model.runtime.RawExtension;
import io.fabric8.kubernetes.client.KubernetesClientException;
import io.fabric8.kubernetes.internal.KubernetesDeserializer;
import io.fabric8.kubernetes.model.jackson.GoCompatibilityModule;
import io.fabric8.kubernetes.model.jackson.UnmatchedFieldTypeModule;
import org.snakeyaml.engine.v2.api.Dump;
import org.snakeyaml.engine.v2.api.DumpSettings;
import org.snakeyaml.engine.v2.api.Load;
import org.snakeyaml.engine.v2.api.LoadSettings;
import org.snakeyaml.engine.v2.common.FlowStyle;
import org.snakeyaml.engine.v2.common.ScalarStyle;
import org.snakeyaml.engine.v2.nodes.NodeTuple;
import org.snakeyaml.engine.v2.nodes.ScalarNode;
import org.snakeyaml.engine.v2.nodes.Tag;
import org.snakeyaml.engine.v2.representer.StandardRepresenter;
import java.io.BufferedInputStream;
import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Type;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.ServiceLoader;
public class KubernetesSerialization {
private final ObjectMapper mapper;
private final UnmatchedFieldTypeModule unmatchedFieldTypeModule = new UnmatchedFieldTypeModule();
private KubernetesDeserializer kubernetesDeserializer;
private final boolean searchClassloaders;
/**
* Creates a new instance with a fresh ObjectMapper
*/
public KubernetesSerialization() {
this(new ObjectMapper(), true);
}
/**
* Creates a new instance with the given ObjectMapper, which will be configured for use for
* kubernetes resource serialization / deserialization.
*
* @param searchClassloaders if {@link KubernetesResource} should be automatically discovered via {@link ServiceLoader}
*/
public KubernetesSerialization(ObjectMapper mapper, boolean searchClassloaders) {
this.mapper = mapper;
this.searchClassloaders = searchClassloaders;
configureMapper(mapper);
}
protected void configureMapper(ObjectMapper mapper) {
mapper.registerModules(new JavaTimeModule(), new GoCompatibilityModule(), unmatchedFieldTypeModule);
mapper.disable(DeserializationFeature.FAIL_ON_INVALID_SUBTYPE);
mapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);
mapper.disable(SerializationFeature.WRITE_DURATIONS_AS_TIMESTAMPS);
// omit null fields, but keep null map values
mapper.setDefaultPropertyInclusion(JsonInclude.Value.construct(Include.NON_NULL, Include.ALWAYS));
HandlerInstantiator instanciator = mapper.getDeserializationConfig().getHandlerInstantiator();
mapper.setConfig(mapper.getDeserializationConfig().with(new HandlerInstantiator() {
@Override
public JsonDeserializer> deserializerInstance(DeserializationConfig config, Annotated annotated, Class> deserClass) {
if (deserClass == KubernetesDeserializer.class) {
return getKubernetesDeserializer();
}
if (instanciator == null) {
return null;
}
return instanciator.deserializerInstance(config, annotated, deserClass);
}
@Override
public KeyDeserializer keyDeserializerInstance(DeserializationConfig config, Annotated annotated,
Class> keyDeserClass) {
if (instanciator == null) {
return null;
}
return instanciator.keyDeserializerInstance(config, annotated, keyDeserClass);
}
@Override
public JsonSerializer> serializerInstance(SerializationConfig config, Annotated annotated, Class> serClass) {
if (instanciator == null) {
return null;
}
return instanciator.serializerInstance(config, annotated, serClass);
}
@Override
public TypeResolverBuilder> typeResolverBuilderInstance(MapperConfig> config, Annotated annotated,
Class> builderClass) {
if (instanciator == null) {
return null;
}
return instanciator.typeResolverBuilderInstance(config, annotated, builderClass);
}
@Override
public TypeIdResolver typeIdResolverInstance(MapperConfig> config, Annotated annotated, Class> resolverClass) {
if (instanciator == null) {
return null;
}
return instanciator.typeIdResolverInstance(config, annotated, resolverClass);
}
}));
}
private synchronized KubernetesDeserializer getKubernetesDeserializer() {
// created lazily to avoid holding all model classes in memory by default
if (this.kubernetesDeserializer == null) {
this.kubernetesDeserializer = new KubernetesDeserializer(searchClassloaders);
}
return kubernetesDeserializer;
}
/**
* Returns a JSON representation of the given object.
*
*
* If the provided object contains a JsonAnyGetter annotated method with a Map that contains an entry that
* overrides a field of the provided object, the Map entry will take precedence upon serialization. Properties won't
* be duplicated.
*
* @param object the object to serialize.
* @param the type of the object being serialized.
* @return a String containing a JSON representation of the provided object.
*/
public String asJson(T object) {
try {
return mapper.writeValueAsString(object);
} catch (JsonProcessingException e) {
throw KubernetesClientException.launderThrowable(e);
}
}
/**
* Returns a YAML representation of the given object.
*
*
* If the provided object contains a JsonAnyGetter annotated method with a Map that contains an entry that
* overrides a field of the provided object, the Map entry will take precedence upon serialization. Properties won't
* be duplicated.
*
* @param object the object to serialize.
* @param the type of the object being serialized.
* @return a String containing a JSON representation of the provided object.
*/
public String asYaml(T object) {
DumpSettings settings = DumpSettings.builder()
.setExplicitStart(true).setDefaultFlowStyle(FlowStyle.BLOCK).build();
final Dump yaml = new Dump(settings, new StandardRepresenter(settings) {
private boolean quote = true;
@Override
protected NodeTuple representMappingEntry(java.util.Map.Entry, ?> entry) {
Object key = entry.getKey();
if (key instanceof String) {
// to match the previous format, don't quote keys
quote = false;
String str = (String) key;
// the abbreviations y/n are not part of the snakeyaml core schema
if (str.length() == 1) {
char start = str.charAt(0);
quote = (start == 'y' || start == 'Y' || start == 'n' || start == 'N');
}
}
org.snakeyaml.engine.v2.nodes.Node nodeKey = representData(key);
quote = true;
return new NodeTuple(nodeKey, representData(entry.getValue()));
}
@Override
protected org.snakeyaml.engine.v2.nodes.Node representScalar(Tag tag, String value, ScalarStyle style) {
if (style == ScalarStyle.PLAIN) {
style = quote && tag == Tag.STR ? ScalarStyle.DOUBLE_QUOTED : this.defaultScalarStyle;
}
return new ScalarNode(tag, value, style);
}
});
return yaml.dumpToString(mapper.convertValue(object, Object.class));
}
/**
* Unmarshals a stream.
*
* The type is assumed to be {@link KubernetesResource}
*
* @param is The {@link InputStream}.
* @param The target type.
*
* @return returns de-serialized object
*/
public T unmarshal(InputStream is) {
return unmarshal(is, new TypeReference() {
@Override
public Type getType() {
return KubernetesResource.class;
}
});
}
public T unmarshal(InputStream is, TypeReference type) {
try (BufferedInputStream bis = new BufferedInputStream(is)) {
bis.mark(-1);
int intch;
do {
intch = bis.read();
} while (intch > -1 && Character.isWhitespace(intch));
bis.reset();
final T result;
if (intch != '{' && intch != '[') {
result = parseYaml(bis, type);
} else {
result = mapper.readerFor(type).readValue(bis);
}
return result;
} catch (IOException e) {
throw KubernetesClientException.launderThrowable(e);
}
}
/**
* If multiple docs exist, only non-null resources will be kept. Results spanning multiple docs
* will be returned as a List of KubernetesResource
*/
private T parseYaml(BufferedInputStream bis, TypeReference type) {
T result = null;
List listResult = null;
final Load yaml = new Load(LoadSettings.builder().build());
final Iterable