AdditionalPropertiesSerializer.java
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.
package com.azure.core.implementation.serializer.jackson;
import com.azure.core.implementation.util.TypeUtil;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.databind.BeanDescription;
import com.fasterxml.jackson.databind.JsonMappingException;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.JsonSerializer;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationConfig;
import com.fasterxml.jackson.databind.SerializerProvider;
import com.fasterxml.jackson.databind.jsontype.TypeSerializer;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.node.ObjectNode;
import com.fasterxml.jackson.databind.ser.BeanSerializerModifier;
import com.fasterxml.jackson.databind.ser.ResolvableSerializer;
import com.fasterxml.jackson.databind.ser.std.StdSerializer;
import java.io.IOException;
import java.lang.reflect.Field;
import java.util.Iterator;
import java.util.Map.Entry;
/**
* Custom serializer for serializing complex types with additional properties.
* If a complex type has a property named "additionalProperties" with serialized
* name empty ("") of type Map<String, Object>, all items in this map will
* become top level properties for this complex type.
*/
final class AdditionalPropertiesSerializer extends StdSerializer<Object> implements ResolvableSerializer {
/**
* The default mapperAdapter for the current type.
*/
private final JsonSerializer<?> defaultSerializer;
/**
* The object mapper for default serializations.
*/
private final ObjectMapper mapper;
/**
* Creates an instance of FlatteningSerializer.
* @param vc handled type
* @param defaultSerializer the default JSON serializer
* @param mapper the object mapper for default serializations
*/
protected AdditionalPropertiesSerializer(Class<?> vc, JsonSerializer<?> defaultSerializer, ObjectMapper mapper) {
super(vc, false);
this.defaultSerializer = defaultSerializer;
this.mapper = mapper;
}
/**
* Gets a module wrapping this serializer as an adapter for the Jackson
* ObjectMapper.
*
* @param mapper the object mapper for default serializations
* @return a simple module to be plugged onto Jackson ObjectMapper.
*/
public static SimpleModule getModule(final ObjectMapper mapper) {
SimpleModule module = new SimpleModule();
module.setSerializerModifier(new BeanSerializerModifier() {
@Override
public JsonSerializer<?> modifySerializer(SerializationConfig config, BeanDescription beanDesc,
JsonSerializer<?> serializer) {
for (Class<?> c : TypeUtil.getAllClasses(beanDesc.getBeanClass())) {
if (c.isAssignableFrom(Object.class)) {
continue;
}
Field[] fields = c.getDeclaredFields();
for (Field field : fields) {
if ("additionalProperties".equalsIgnoreCase(field.getName())) {
JsonProperty property = field.getAnnotation(JsonProperty.class);
if (property != null && property.value().isEmpty()) {
return new AdditionalPropertiesSerializer(beanDesc.getBeanClass(), serializer, mapper);
}
}
}
}
return serializer;
}
});
return module;
}
@Override
public void serialize(Object value, JsonGenerator jgen, SerializerProvider provider) throws IOException {
// serialize the original object into JsonNode
ObjectNode root = mapper.valueToTree(value);
// take additional properties node out
Entry<String, JsonNode> additionalPropertiesField = null;
Iterator<Entry<String, JsonNode>> fields = root.fields();
while (fields.hasNext()) {
Entry<String, JsonNode> field = fields.next();
if ("additionalProperties".equalsIgnoreCase(field.getKey())) {
additionalPropertiesField = field;
break;
}
}
if (additionalPropertiesField != null) {
root.remove(additionalPropertiesField.getKey());
// put each item back in
ObjectNode extraProperties = (ObjectNode) additionalPropertiesField.getValue();
fields = extraProperties.fields();
while (fields.hasNext()) {
Entry<String, JsonNode> field = fields.next();
root.put(field.getKey(), field.getValue());
}
}
jgen.writeTree(root);
}
@Override
public void resolve(SerializerProvider provider) throws JsonMappingException {
((ResolvableSerializer) defaultSerializer).resolve(provider);
}
@Override
public void serializeWithType(Object value, JsonGenerator gen, SerializerProvider provider,
TypeSerializer typeSerializer) throws IOException {
serialize(value, gen, provider);
}
}