All Downloads are FREE. Search and download functionalities are using the official Maven repository.

ioinformarics.oss.jackson.module.jsonld.JsonldContextFactory Maven / Gradle / Ivy

The newest version!
package ioinformarics.oss.jackson.module.jsonld;

import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.node.ArrayNode;
import com.fasterxml.jackson.databind.node.JsonNodeFactory;
import com.fasterxml.jackson.databind.node.ObjectNode;
import com.fasterxml.jackson.databind.node.TextNode;
import io.github.lukehutch.fastclasspathscanner.FastClasspathScanner;
import ioinformarics.oss.jackson.module.jsonld.annotation.*;
import ioinformarics.oss.jackson.module.jsonld.util.AnnotationsUtils;
import ioinformarics.oss.jackson.module.jsonld.util.JsonUtils;
import ioinformarics.oss.jackson.module.jsonld.util.JsonldResourceUtils;
import org.apache.commons.lang3.ClassUtils;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.*;
import java.util.stream.Stream;

/**
 * @author Alexander De Leon
 */
public class JsonldContextFactory {

    public static ObjectNode fromPackage(String packageName) {
        ObjectNode generatedContext = JsonNodeFactory.withExactBigDecimals(true).objectNode();
        FastClasspathScanner scanner = new FastClasspathScanner(packageName);
        scanner.matchAllStandardClasses((clazz) -> {
            if(!Modifier.isAbstract(clazz.getModifiers()) && AnnotationsUtils.isAnnotationPresent(clazz, JsonldTypeFromJavaClass.class)) {
                Optional type = JsonldResourceUtils.dynamicTypeLookup(clazz);
                type.ifPresent(t ->generatedContext.set(clazz.getSimpleName(), TextNode.valueOf(t)));
            }
            if(AnnotationsUtils.isAnnotationPresent(clazz, ioinformarics.oss.jackson.module.jsonld.annotation.JsonldResource.class)) {
                Optional resourceContext = fromAnnotations(clazz);
                resourceContext.ifPresent((context) -> JsonUtils.merge(generatedContext, context));
            }
        });
        scanner.scan();
        return (ObjectNode) JsonNodeFactory.withExactBigDecimals(true).objectNode().set("@context", generatedContext);
    }

    public static Optional fromAnnotations(Object instance) {
        return fromAnnotations(instance.getClass());
    }

    public static Optional fromAnnotations(Iterable instances) {
        ObjectNode mergedContext = JsonNodeFactory.withExactBigDecimals(true).objectNode();
        instances.forEach(e -> fromAnnotations(e).map(mergedContext::setAll));
        return mergedContext.size() != 0 ? Optional.of(mergedContext) : Optional.empty();
    }

    public static Optional fromAnnotations(Class objType) {
        ObjectNode generatedContext = JsonNodeFactory.withExactBigDecimals(true).objectNode();
        generateNamespaces(objType).forEach((name, uri) -> generatedContext.set(name, new TextNode(uri)));
        //TODO: This is bad...it does not consider other Jackson annotations. Need to use a AnnotationIntrospector?
        final Map fieldContexts = generateContextsForFields(objType);
        fieldContexts.forEach(generatedContext::set);
        //add links
        JsonldLink[] links = objType.getAnnotationsByType(JsonldLink.class);
        if (links != null) {
            for (int i = 0; i < links.length; i++) {
                com.fasterxml.jackson.databind.node.ObjectNode linkNode = JsonNodeFactory.withExactBigDecimals(true)
                                                                                         .objectNode();
                linkNode.set("@id", new TextNode(links[i].rel()));
                linkNode.set("@type", new TextNode("@id"));
                generatedContext.set(links[i].name(), linkNode);
            }
        }
        //Return absent optional if context is empty
        return generatedContext.size() != 0 ? Optional.of(generatedContext) : Optional.empty();
    }

    private static Map generateContextsForFields(Class objType) {
        return generateContextsForFields(objType, new ArrayList<>());
    }

    private static Map generateContextsForFields(Class objType, List> ignoreTypes) {
        final Map contexts = new HashMap<>();
        Class currentClass = objType;
        Optional namespace = Optional.ofNullable(currentClass.getAnnotation(JsonldNamespace.class));
        while (currentClass != null && !currentClass.equals(Object.class)) {
            final Field[] fields = currentClass.getDeclaredFields();
            for (Field f : fields) {
                if(f.isAnnotationPresent(JsonldId.class) || f.getName().equals("this$0")) {
                    continue;
                }
                final JsonldProperty jsonldProperty = f.getAnnotation(JsonldProperty.class);
                Optional propertyId = Optional.empty();
                // Most concrete field overrides any field with the same name defined higher up the hierarchy
                if (jsonldProperty != null && !contexts.containsKey(f.getName())) {
                    propertyId = Optional.of(jsonldProperty.value());
                }
                else if(jsonldProperty == null && namespace.map(JsonldNamespace::applyToProperties).orElse(false)) {
                    propertyId = Optional.of(namespace.get().name() + ":" + f.getName());
                }
                propertyId.ifPresent((id) -> {
                    if(isRelation(f)) {
                        ObjectNode node = JsonNodeFactory.withExactBigDecimals(true).objectNode();
                        node.set("@id", TextNode.valueOf(id));
                        node.set("@type", TextNode.valueOf("@id"));
                        contexts.put(f.getName(), node);
                    }
                    else {
                        contexts.put(f.getName(), TextNode.valueOf(id));
                    }
                });
            }
            currentClass = currentClass.getSuperclass();
            if(!namespace.isPresent()) {
                namespace = Optional.ofNullable(currentClass.getAnnotation(JsonldNamespace.class));
            }
        }
        return contexts;
    }

    private static Class relationType(Field field) {
        Class type = field.getType();
        if(Collection.class.isAssignableFrom(type)) {
            ParameterizedType parameterizedType = (ParameterizedType) field.getGenericType();
            Type t = parameterizedType.getActualTypeArguments()[0];
            if(Class.class.isAssignableFrom(t.getClass())) {
                type = (Class) t;
            }
            else if(ParameterizedType.class.isAssignableFrom(t.getClass())) {
                type = (Class)((ParameterizedType) t).getRawType();
            }
        }
        if(type.isArray()) {
            type = type.getComponentType();
        }
        return type;
    }

    private static boolean isRelation(Field field) {
        Class type = relationType(field);
        return Stream.concat(Stream.of(type),
                Stream.concat(ClassUtils.getAllSuperclasses(type).stream(), ClassUtils.getAllInterfaces(type).stream()))
                .flatMap(currentClass -> Stream.concat(
                        Stream.of(currentClass.getDeclaredFields()),
                        Stream.of(currentClass.getDeclaredMethods())
                ))
                .anyMatch(p -> p.getAnnotation(JsonldId.class) != null);
    }

    private static Map generateNamespaces(Class objType) {
        JsonldNamespace[] namespaceAnnotations = objType.getAnnotationsByType(JsonldNamespace.class);
        Map namespaces = new HashMap<>(namespaceAnnotations.length);
        Arrays.asList(namespaceAnnotations).forEach((ns) -> namespaces.put(ns.name(), ns.uri()));
        return namespaces;
    }


    public static Optional multiContext(Optional externalContext,
                                                  Optional internalContext) {
        if (internalContext.isPresent()) {
            return externalContext.isPresent() ?
                   Optional.of((JsonNode) buildMultiContext(externalContext.get(), internalContext.get())) :
                   internalContext.map(it -> (JsonNode) it);
        }
        return externalContext.map(TextNode::valueOf);
    }

    private static ArrayNode buildMultiContext(String context, JsonNode generatedContext) {
        return JsonNodeFactory.withExactBigDecimals(true).arrayNode().add(context).add(generatedContext);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy