ca.gc.aafc.dina.mapper.DinaMappingRegistry Maven / Gradle / Ivy
package ca.gc.aafc.dina.mapper;
import ca.gc.aafc.dina.dto.RelatedEntity;
import ca.gc.aafc.dina.repository.meta.JsonApiExternalRelation;
import io.crnk.core.resource.annotations.JsonApiId;
import io.crnk.core.resource.annotations.JsonApiRelation;
import lombok.Builder;
import lombok.Getter;
import lombok.NonNull;
import lombok.SneakyThrows;
import org.apache.commons.lang3.reflect.FieldUtils;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Collectors;
/**
* Registry to track information regarding a given resource class. Useful to obtain certain meta information
* regarding the domain of resource.
*/
public class DinaMappingRegistry {
@Getter
private final Map, Set> attributesPerClass;
// Set of entries tracked by class for faster lookup.
private final Map, DinaResourceEntry> resourceGraph;
/**
* The given resource class will have its graph traversed and registered into the registry. All
* relations will be considered a node of the graph and traversed accordingly.
*
* Parsing a given resource graph requires the use of reflection. A DinaMappingRegistry should not be
* constructed in a repetitive manner where performance is needed.
*
*
* Concepts
*
*
* A relation is a field that is marked as a {@link JsonApiRelation}
* A relation is considered internal unless marked with {@link JsonApiExternalRelation}
* An attribute is a field that is not {@link IgnoreDinaMapping} or marked as a relation and will be
* mapped directly as a value.
* An attribute must have the same data type on the DTO class and its related entity
* A field that is considered an attribute but with a different data type will throw an {@link
* IllegalStateException}, unless the data type of the field is a valid DTO/RelatedEntity mapping between
* the classes
*
*
* @param resourceClass - resource traverse and register
*/
public DinaMappingRegistry(@NonNull Class> resourceClass) {
resourceGraph = initGraph(resourceClass, new HashSet<>());
this.attributesPerClass = this.resourceGraph.entrySet().stream()
.collect(Collectors.toMap(Map.Entry::getKey, entry -> entry.getValue().getAttributeNames()));
}
private static Map, DinaResourceEntry> initGraph(Class> resourceClass, Set> visited) {
HashMap, DinaResourceEntry> graph = new HashMap<>();
if (visited.contains(resourceClass)) {
return graph;
}
visited.add(resourceClass);
if (resourceClass.getAnnotation(RelatedEntity.class) != null) {
Class> entityClass = resourceClass.getAnnotation(RelatedEntity.class).value();
DinaResourceEntry entry = parseRegistryEntry(entityClass, resourceClass, visited, graph);
graph.put(resourceClass, entry);
graph.put(entityClass, entry);
}
return graph;
}
private static DinaResourceEntry parseRegistryEntry(
Class> entityClass, Class> resourceClass,
Set> visited, Map, DinaResourceEntry> graph
) {
Set attributes = new HashSet<>();
Set internalRelations = new HashSet<>();
for (Field dtoField : getAllFields(resourceClass)) {
if (isMappableRelation(resourceClass, entityClass, dtoField)) {
// If relation register and traverse graph
internalRelations.add(mapToInternalRelation(dtoField));
graph.putAll(initGraph(parseGenericTypeForField(dtoField), visited));
} else if (isFieldConsideredAnAttribute(resourceClass, entityClass, dtoField)) {
if (fieldHasSameDataType(entityClass, dtoField)) {
// Un marked dtoField with same data type considered attribute
attributes.add(dtoField.getName());
} else {
// Un marked dtoField without the same data type but have a related entity are considered a hidden relation
if (!parseGenericTypeForField(dtoField).isAnnotationPresent(RelatedEntity.class)) {
throwDataTypeMismatchException(resourceClass, entityClass, dtoField.getName());
} else {
internalRelations.add(mapToInternalRelation(dtoField));
graph.putAll(initGraph(parseGenericTypeForField(dtoField), visited));
}
}
}
}
return buildResourceEntry(resourceClass, entityClass, attributes, internalRelations);
}
/**
* Returns a set of the mappable relations for a given class.
*
* @param cls - class with relations
* @return Returns a set of the mappable relations for a given class.
* @throws IllegalArgumentException if the class is not tracked by the registry
*/
public Set findMappableRelationsForClass(Class> cls) {
checkClassTracked(cls);
return this.resourceGraph.get(cls).getInternalRelations();
}
/**
* Returns the set of external relation field names tracked by the registry.
*
* @return set of external relation field names.
*/
public Set getExternalRelations(Class> cls) {
checkClassTracked(cls);
return this.resourceGraph.get(cls).getExternalNameToTypeMap().keySet();
}
/**
* Returns the {@link JsonApiExternalRelation} type of the given external relation field name if tracked by
* the registry.
*
* @param relationFieldName - field name of the external relation.
* @return type of the given external relation.
* @throws IllegalArgumentException if the relationFieldName is not tracked by the registry
*/
public String findExternalType(Class> cls, String relationFieldName) {
checkClassTracked(cls);
if (!this.resourceGraph.get(cls).getExternalNameToTypeMap().containsKey(relationFieldName)) {
throw new IllegalArgumentException(
"external relation with name: " + relationFieldName + " is not tracked by the registry");
}
return this.resourceGraph.get(cls).getExternalNameToTypeMap().get(relationFieldName);
}
/**
* Returns true if the relation with the given field name is external.
*
* @param relationFieldName - field name of the external relation.
* @return Returns true if the relation with the given field name is external.
*/
public boolean isRelationExternal(Class> cls, String relationFieldName) {
checkClassTracked(cls);
return this.resourceGraph.get(cls).getExternalNameToTypeMap().keySet().stream()
.anyMatch(relationFieldName::equalsIgnoreCase);
}
/**
* Returns the json id field name of a given class.
*
* @param cls - cls with json id field
* @return the json id field name of a given class.
* @throws IllegalArgumentException if the class is not tracked by the registry
*/
public String findJsonIdFieldName(Class> cls) {
checkClassTracked(cls);
return this.resourceGraph.get(cls).getJsonIdFieldName();
}
/**
* Returns the Dina field adapter for a given class or optional empty if it does not exist.
*
* @param cls - class of the {@link DinaFieldAdapterHandler}
* @return the {@link DinaFieldAdapterHandler} for a given class
*/
public Optional> findFieldAdapterForClass(Class> cls) {
if (!this.resourceGraph.containsKey(cls)) {
return Optional.empty();
}
return Optional.ofNullable(this.resourceGraph.get(cls).getFieldAdapterHandler());
}
/**
* Returns true if the registry is tracking a registered field adapter.
*
* @return true if the registry is tracking a registered field adapter.
*/
public boolean hasFieldAdapters() {
return this.resourceGraph.values().stream()
.map(DinaResourceEntry::getFieldAdapterHandler).findFirst().isPresent();
}
/**
* Returns the nested resource type from a given base resource type and attribute path. The deepest nested
* resource that can be resolved will be returned. The original resource is returned if a nested resource is
* not present in the attribute path, or the resources are not tracked by the registry.
*
* @param resource - base resource to traverse
* @param attributePath - attribute path to follow
* @return - the nested resource type from a given path.
*/
public Class> resolveNestedResourceFromPath(
@NonNull Class> resource,
@NonNull List attributePath
) {
Class> nested = resource;
for (String attribute : attributePath) {
Optional relation = this.findMappableRelationsForClass(nested).stream()
.filter(internalRelation -> internalRelation.getName().equalsIgnoreCase(attribute))
.findAny();
if (relation.isPresent()) {
nested = relation.get().getDtoType();
} else {
break;
}
}
return nested;
}
private static InternalRelation mapToInternalRelation(Field field) {
Class> fieldType = field.getType();
boolean isCollection = false;
if (isCollection(fieldType)) {
fieldType = parseGenericTypeForField(field);
isCollection = true;
}
return InternalRelation.builder()
.name(field.getName())
.isCollection(isCollection)
.dtoType(fieldType)
.entityType(fieldType.getAnnotation(RelatedEntity.class).value())
.build();
}
/**
* Returns a map of external relation field names to their JsonApiExternalRelation.type for a given class.
*
* @param resourceClass - a given class with external relations.
* @return a map of external relation field names to their JsonApiExternalRelation.type
*/
private static Map parseExternalRelationNamesToType(Class> resourceClass) {
return Map.copyOf(
FieldUtils.getFieldsListWithAnnotation(resourceClass, JsonApiExternalRelation.class)
.stream().collect(Collectors.toMap(
Field::getName,
field -> field.getAnnotation(JsonApiExternalRelation.class).type())));
}
/**
* Returns true if the dina repo should map the given field as an attribute. An attribute in this context is
* a field that has its value mapped directly.
*
* @param dtoClass dto class to validate against
* @param entityClass entity class to validate against
* @param field field to evaluate
* @return - true if the dina repo should not map the given field
*/
private static boolean isFieldConsideredAnAttribute(Class> dtoClass, Class> entityClass, Field field) {
return !field.isAnnotationPresent(IgnoreDinaMapping.class)
&& !field.isAnnotationPresent(JsonApiRelation.class)
&& fieldExistsInBothClasses(dtoClass, entityClass, field.getName())
&& !Modifier.isFinal(field.getModifiers())
&& !field.isSynthetic();
}
/**
* Returns true if the dina repo should map the given relation. A relation should be mapped if it is not
* external, and present in the given dto and entity classes.
*
* @param dto - resource class of the relation
* @param entity - entity class of the relation
* @param dtoRelationField - relation field to map
* @return true if the dina repo should map the given relation.
*/
private static boolean isMappableRelation(Class> dto, Class> entity, Field dtoRelationField) {
return dtoRelationField.isAnnotationPresent(JsonApiRelation.class)
&& !dtoRelationField.isAnnotationPresent(IgnoreDinaMapping.class)
&& !dtoRelationField.isAnnotationPresent(JsonApiExternalRelation.class)
&& fieldExistsInBothClasses(dto, entity, dtoRelationField.getName());
}
private static boolean fieldExistsInBothClasses(Class> dtoClass, Class> entityClass, String fieldName) {
return getAllFields(dtoClass).stream().anyMatch(field -> fieldName.equals(field.getName()))
&& getAllFields(entityClass).stream().anyMatch(field -> fieldName.equals(field.getName()));
}
@SneakyThrows
private static boolean fieldHasSameDataType(Class> entityClass, Field dtoField) {
Field entityClassDeclaredField = getAllFields(entityClass).stream()
.filter(f -> f.getName().equals(dtoField.getName()))
.findFirst()
.orElse(null);
if (entityClassDeclaredField == null) {
return false;
}
Type typeOnDto = dtoField.getGenericType();
if (!entityClassDeclaredField.getGenericType().equals(typeOnDto)) { // Data types must match!
return false;
}
if (typeOnDto instanceof ParameterizedType) { // If parameterized generic type must match
return parseGenericTypeForField(dtoField)
.equals(parseGenericTypeForField(entityClassDeclaredField));
}
return true;
}
private static Class> parseGenericTypeForField(Field field) {
if (field.getGenericType() instanceof ParameterizedType) {
return (Class>) ((ParameterizedType) field.getGenericType()).getActualTypeArguments()[0];
} else {
return field.getType();
}
}
private static String parseJsonIdFieldName(Class> resourceClass) {
for (Field field : getAllFields(resourceClass)) {
if (field.isAnnotationPresent(JsonApiId.class)) {
return field.getName();
}
}
return null;
}
private static DinaResourceEntry buildResourceEntry(
Class> resourceClass,
Class> entityClass,
Set attributes,
Set internalRelations
) {
return DinaResourceEntry.builder()
.dtoClass(resourceClass)
.entityClass(entityClass)
.externalNameToTypeMap(parseExternalRelationNamesToType(resourceClass))
.attributeNames(attributes)
.internalRelations(internalRelations)
.jsonIdFieldName(parseJsonIdFieldName(resourceClass))
.fieldAdapterHandler(new DinaFieldAdapterHandler<>(resourceClass))
.build();
}
private static List getAllFields(Class> type) {
List fields = new ArrayList<>(Arrays.asList(type.getDeclaredFields()));
if (type.getSuperclass() != null) {
fields.addAll(getAllFields(type.getSuperclass()));
}
return fields;
}
private void checkClassTracked(Class> cls) {
if (!this.resourceGraph.containsKey(cls)) {
throw new IllegalArgumentException(cls.getSimpleName() + " is not tracked by the registry");
}
}
private static void throwDataTypeMismatchException(Class> dto, Class> entity, String attrib) {
throw new IllegalStateException("data type for Field:{" + attrib + "} on DTO:{" + dto.getSimpleName()
+ "} does not match the field from Entity:{" + entity.getSimpleName() + "}");
}
/**
* Returns true if the given class is a collection
*
* @param clazz - class to check
* @return true if the given class is a collection
*/
private static boolean isCollection(Class> clazz) {
return Collection.class.isAssignableFrom(clazz);
}
/**
* Internal Relation Representing a field of class to be mapped
*/
@Builder
@Getter
public static class InternalRelation {
private final String name;
private final Class> dtoType;
private final Class> entityType;
private final boolean isCollection;
}
@Builder
@Getter
public static class DinaResourceEntry {
private Class> dtoClass;
private Class> entityClass;
private String jsonIdFieldName;
private Set attributeNames;
private Set internalRelations;
private Map externalNameToTypeMap;
private DinaFieldAdapterHandler> fieldAdapterHandler;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy