net.amygdalum.testrecorder.serializers.CollectionsMapSerializer Maven / Gradle / Ivy
package net.amygdalum.testrecorder.serializers;
import static java.util.stream.Collectors.toList;
import static net.amygdalum.testrecorder.util.TypeFilters.startingWith;
import static net.amygdalum.testrecorder.util.Types.inferType;
import static net.amygdalum.testrecorder.util.Types.parameterized;
import static net.amygdalum.testrecorder.util.Types.typeArgument;
import java.lang.reflect.Type;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.stream.Stream;
import net.amygdalum.testrecorder.types.SerializerSession;
import net.amygdalum.testrecorder.util.Reflections;
import net.amygdalum.testrecorder.values.SerializedMap;
public class CollectionsMapSerializer extends HiddenInnerClassSerializer {
public CollectionsMapSerializer() {
super(Collections.class);
}
@Override
public List> getMatchingClasses() {
return innerClasses().stream()
.filter(startingWith("Unmodifiable", "Synchronized", "Checked", "Empty", "Singleton"))
.filter(clazz -> Map.class.isAssignableFrom(clazz))
.collect(toList());
}
@Override
public Stream> components(Object object, SerializerSession session) {
return ((Map, ?>) object).entrySet().stream()
.flatMap(entry -> Stream.of(entry.getKey(), entry.getValue()));
}
@Override
public SerializedMap generate(Class> type, SerializerSession session) {
return new SerializedMap(type);
}
@Override
public void populate(SerializedMap serializedObject, Object object, SerializerSession session) {
Type[] componentTypes = computeComponentType(serializedObject, object);
Type keyType = componentTypes[0];
Type valueType = componentTypes[1];
for (Map.Entry, ?> element : ((Map, ?>) object).entrySet()) {
Object key = element.getKey();
Object value = element.getValue();
serializedObject.put(resolvedValueOf(session, keyType, key), resolvedValueOf(session, valueType, value));
}
Type newType = parameterized(Map.class, null, componentTypes);
serializedObject.useAs(newType);
}
private Type[] computeComponentType(SerializedMap serializedObject, Object object) {
if (object.getClass().getSimpleName().contains("Checked")) {
return new Type[] { getKeyTypeField(object), getValueTypeField(object) };
}
Stream keyDefinedTypes = Arrays.stream(serializedObject.getUsedTypes())
.map(type -> typeArgument(type, 0).orElse(Object.class));
Stream keyElementTypes = ((Map, ?>) object).keySet().stream()
.filter(Objects::nonNull)
.map(element -> element.getClass());
Stream keyTypes = Stream.concat(keyDefinedTypes, keyElementTypes);
Stream valueDefinedTypes = Arrays.stream(serializedObject.getUsedTypes())
.map(type -> typeArgument(type, 1).orElse(Object.class));
Stream valueElementTypes = ((Map, ?>) object).values().stream()
.filter(Objects::nonNull)
.map(element -> element.getClass());
Stream valueTypes = Stream.concat(valueDefinedTypes, valueElementTypes);
return new Type[] { inferType(keyTypes.toArray(Type[]::new)), inferType(valueTypes.toArray(Type[]::new)) };
}
private Class> getKeyTypeField(Object object) {
try {
return (Class>) Reflections.getValue("keyType", object);
} catch (ReflectiveOperationException e) {
return Object.class;
}
}
private Class> getValueTypeField(Object object) {
try {
return (Class>) Reflections.getValue("valueType", object);
} catch (ReflectiveOperationException e) {
return Object.class;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy