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

graphql.schema.diffing.SchemaGraphFactory Maven / Gradle / Ivy

package graphql.schema.diffing;

import graphql.GraphQLContext;
import graphql.Internal;
import graphql.execution.ValuesResolver;
import graphql.introspection.Introspection;
import graphql.language.AstPrinter;
import graphql.schema.*;
import graphql.schema.idl.DirectiveInfo;
import graphql.schema.idl.ScalarInfo;
import graphql.util.TraversalControl;
import graphql.util.Traverser;
import graphql.util.TraverserContext;
import graphql.util.TraverserVisitor;

import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Set;

import static graphql.Assert.assertNotNull;

@Internal
public class SchemaGraphFactory {

    private int counter = 1;
    private final String debugPrefix;

    public SchemaGraphFactory(String debugPrefix) {
        this.debugPrefix = debugPrefix;
    }

    public SchemaGraphFactory() {
        this.debugPrefix = "";
    }

    public SchemaGraph createGraph(GraphQLSchema schema) {
        Set roots = new LinkedHashSet<>();
        roots.add(schema.getQueryType());
        if (schema.isSupportingMutations()) {
            roots.add(schema.getMutationType());
        }
        if (schema.isSupportingSubscriptions()) {
            roots.add(schema.getSubscriptionType());
        }
        roots.addAll(schema.getAdditionalTypes());
        roots.addAll(schema.getDirectives());
        roots.addAll(schema.getSchemaDirectives());
        roots.add(schema.getIntrospectionSchemaType());
        Traverser traverser = Traverser.depthFirst(GraphQLSchemaElement::getChildren);
        SchemaGraph schemaGraph = new SchemaGraph();
        class IntrospectionNode {

        }
        traverser.traverse(roots, new TraverserVisitor() {
            @Override
            public TraversalControl enter(TraverserContext context) {
                boolean isIntrospectionNode = false;
                if (context.thisNode() instanceof GraphQLNamedType) {
                    if (Introspection.isIntrospectionTypes((GraphQLNamedType) context.thisNode())) {
                        isIntrospectionNode = true;
                        context.setVar(IntrospectionNode.class, new IntrospectionNode());
                    }
                } else {
                    isIntrospectionNode = context.getVarFromParents(IntrospectionNode.class) != null;
                }
                if (context.thisNode() instanceof GraphQLObjectType) {
                    newObject((GraphQLObjectType) context.thisNode(), schemaGraph, isIntrospectionNode);
                }
                if (context.thisNode() instanceof GraphQLInterfaceType) {
                    newInterface((GraphQLInterfaceType) context.thisNode(), schemaGraph, isIntrospectionNode);
                }
                if (context.thisNode() instanceof GraphQLUnionType) {
                    newUnion((GraphQLUnionType) context.thisNode(), schemaGraph, isIntrospectionNode);
                }
                if (context.thisNode() instanceof GraphQLScalarType) {
                    newScalar((GraphQLScalarType) context.thisNode(), schemaGraph, isIntrospectionNode);
                }
                if (context.thisNode() instanceof GraphQLInputObjectType) {
                    newInputObject((GraphQLInputObjectType) context.thisNode(), schemaGraph, isIntrospectionNode);
                }
                if (context.thisNode() instanceof GraphQLEnumType) {
                    newEnum((GraphQLEnumType) context.thisNode(), schemaGraph, isIntrospectionNode);
                }
                if (context.thisNode() instanceof GraphQLDirective) {
                    // only continue if not applied directive
                    if (context.getParentNode() == null) {
                        newDirective((GraphQLDirective) context.thisNode(), schemaGraph);
                    }
                }
                return TraversalControl.CONTINUE;
            }

            @Override
            public TraversalControl leave(TraverserContext context) {
                return TraversalControl.CONTINUE;
            }
        });
        addSchemaVertex(schemaGraph, schema);

        ArrayList copyOfVertices = new ArrayList<>(schemaGraph.getVertices());
        for (Vertex vertex : copyOfVertices) {
            if (SchemaGraph.OBJECT.equals(vertex.getType())) {
                handleObjectVertex(vertex, schemaGraph, schema);
            }
            if (SchemaGraph.INTERFACE.equals(vertex.getType())) {
                handleInterfaceVertex(vertex, schemaGraph, schema);
            }
            if (SchemaGraph.UNION.equals(vertex.getType())) {
                handleUnion(vertex, schemaGraph, schema);
            }
            if (SchemaGraph.INPUT_OBJECT.equals(vertex.getType())) {
                handleInputObject(vertex, schemaGraph, schema);
            }
            if (SchemaGraph.DIRECTIVE.equals(vertex.getType())) {
                handleDirective(vertex, schemaGraph, schema);
            }
        }
        return schemaGraph;
    }

    private void addSchemaVertex(SchemaGraph schemaGraph, GraphQLSchema graphQLSchema) {
        GraphQLObjectType queryType = graphQLSchema.getQueryType();
        GraphQLObjectType mutationType = graphQLSchema.getMutationType();
        GraphQLObjectType subscriptionType = graphQLSchema.getSubscriptionType();
        Vertex schemaVertex = new Vertex(SchemaGraph.SCHEMA, "schema");
        schemaVertex.add("name", SchemaGraph.SCHEMA);
        schemaGraph.addVertex(schemaVertex);

        Vertex queryVertex = schemaGraph.getType(queryType.getName());
        schemaGraph.addEdge(new Edge(schemaVertex, queryVertex, "query"));
        if (mutationType != null) {
            Vertex mutationVertex = schemaGraph.getType(mutationType.getName());
            schemaGraph.addEdge(new Edge(schemaVertex, mutationVertex, "mutation"));
        }
        if (subscriptionType != null) {
            Vertex subscriptionVertex = schemaGraph.getType(subscriptionType.getName());
            schemaGraph.addEdge(new Edge(schemaVertex, subscriptionVertex, "subscription"));
        }
        createAppliedDirectives(schemaVertex, graphQLSchema.getSchemaDirectives(), schemaGraph);
    }

    private void handleInputObject(Vertex inputObject, SchemaGraph schemaGraph, GraphQLSchema graphQLSchema) {
        GraphQLInputObjectType inputObjectType = (GraphQLInputObjectType) graphQLSchema.getType(inputObject.get("name"));
        List inputFields = inputObjectType.getFields();
        for (GraphQLInputObjectField inputField : inputFields) {
            Vertex inputFieldVertex = schemaGraph.findTargetVertex(inputObject, vertex -> vertex.getType().equals("InputField") &&
                    vertex.get("name").equals(inputField.getName())).get();
            handleInputField(inputFieldVertex, inputField, schemaGraph, graphQLSchema);
        }
    }

    private void handleInputField(Vertex inputFieldVertex, GraphQLInputObjectField inputField, SchemaGraph
            schemaGraph, GraphQLSchema graphQLSchema) {
        GraphQLInputType type = inputField.getType();
        GraphQLUnmodifiedType graphQLUnmodifiedType = GraphQLTypeUtil.unwrapAll(type);
        Vertex typeVertex = assertNotNull(schemaGraph.getType(graphQLUnmodifiedType.getName()));
        Edge typeEdge = new Edge(inputFieldVertex, typeVertex);
        String typeEdgeLabel = "type=" + GraphQLTypeUtil.simplePrint(type) + ";defaultValue=";
        if (inputField.hasSetDefaultValue()) {
            typeEdgeLabel += AstPrinter.printAst(ValuesResolver.valueToLiteral(inputField.getInputFieldDefaultValue(), inputField.getType(), GraphQLContext.getDefault(), Locale.getDefault()));
        }

        typeEdge.setLabel(typeEdgeLabel);
        schemaGraph.addEdge(typeEdge);
    }

    private void handleUnion(Vertex unionVertex, SchemaGraph schemaGraph, GraphQLSchema graphQLSchema) {
        GraphQLUnionType unionType = (GraphQLUnionType) graphQLSchema.getType(unionVertex.get("name"));
        List types = unionType.getTypes();
        for (GraphQLNamedOutputType unionMemberType : types) {
            Vertex unionMemberVertex = assertNotNull(schemaGraph.getType(unionMemberType.getName()));
            schemaGraph.addEdge(new Edge(unionVertex, unionMemberVertex));
        }
    }

    private void handleInterfaceVertex(Vertex interfaceVertex, SchemaGraph schemaGraph, GraphQLSchema graphQLSchema) {
        GraphQLInterfaceType interfaceType = (GraphQLInterfaceType) graphQLSchema.getType(interfaceVertex.get("name"));

        for (GraphQLNamedOutputType implementsInterface : interfaceType.getInterfaces()) {
            Vertex implementsInterfaceVertex = assertNotNull(schemaGraph.getType(implementsInterface.getName()));
            schemaGraph.addEdge(new Edge(interfaceVertex, implementsInterfaceVertex, "implements " + implementsInterface.getName()));
        }

        List fieldDefinitions = interfaceType.getFieldDefinitions();
        for (GraphQLFieldDefinition fieldDefinition : fieldDefinitions) {
            Vertex fieldVertex = schemaGraph.findTargetVertex(interfaceVertex, vertex -> vertex.getType().equals("Field") &&
                    vertex.get("name").equals(fieldDefinition.getName())).get();
            handleField(fieldVertex, fieldDefinition, schemaGraph, graphQLSchema);
        }

    }

    private void handleObjectVertex(Vertex objectVertex, SchemaGraph schemaGraph, GraphQLSchema graphQLSchema) {
        GraphQLObjectType objectType = graphQLSchema.getObjectType(objectVertex.get("name"));

        for (GraphQLNamedOutputType implementsInterface : objectType.getInterfaces()) {
            Vertex implementsInterfaceVertex = assertNotNull(schemaGraph.getType(implementsInterface.getName()));
            schemaGraph.addEdge(new Edge(objectVertex, implementsInterfaceVertex, "implements " + implementsInterface.getName()));
        }

        List fieldDefinitions = objectType.getFieldDefinitions();
        for (GraphQLFieldDefinition fieldDefinition : fieldDefinitions) {
            Vertex fieldVertex = schemaGraph.findTargetVertex(objectVertex, vertex -> vertex.getType().equals("Field") &&
                    vertex.get("name").equals(fieldDefinition.getName())).get();
            handleField(fieldVertex, fieldDefinition, schemaGraph, graphQLSchema);
        }
    }

    private void handleField(Vertex fieldVertex, GraphQLFieldDefinition fieldDefinition, SchemaGraph
            schemaGraph, GraphQLSchema graphQLSchema) {
        GraphQLOutputType type = fieldDefinition.getType();
        GraphQLUnmodifiedType graphQLUnmodifiedType = GraphQLTypeUtil.unwrapAll(type);
        Vertex typeVertex = assertNotNull(schemaGraph.getType(graphQLUnmodifiedType.getName()));
        Edge typeEdge = new Edge(fieldVertex, typeVertex);
        typeEdge.setLabel("type=" + GraphQLTypeUtil.simplePrint(type) + ";");
        schemaGraph.addEdge(typeEdge);

        for (GraphQLArgument graphQLArgument : fieldDefinition.getArguments()) {
            Vertex argumentVertex = schemaGraph.findTargetVertex(fieldVertex, vertex -> vertex.getType().equals("Argument") &&
                    vertex.get("name").equals(graphQLArgument.getName())).get();
            handleArgument(argumentVertex, graphQLArgument, schemaGraph);
        }
    }

    private void handleDirective(Vertex directive, SchemaGraph schemaGraph, GraphQLSchema graphQLSchema) {
        GraphQLDirective graphQLDirective = graphQLSchema.getDirective(directive.getName());
        for (GraphQLArgument graphQLArgument : graphQLDirective.getArguments()) {
            Vertex argumentVertex = schemaGraph.findTargetVertex(directive, vertex -> vertex.isOfType(SchemaGraph.ARGUMENT) &&
                    vertex.getName().equals(graphQLArgument.getName())).get();
            handleArgument(argumentVertex, graphQLArgument, schemaGraph);
        }

    }

    private void handleArgument(Vertex argumentVertex, GraphQLArgument graphQLArgument, SchemaGraph schemaGraph) {
        GraphQLInputType type = graphQLArgument.getType();
        GraphQLUnmodifiedType graphQLUnmodifiedType = GraphQLTypeUtil.unwrapAll(type);
        Vertex typeVertex = assertNotNull(schemaGraph.getType(graphQLUnmodifiedType.getName()));
        Edge typeEdge = new Edge(argumentVertex, typeVertex);
        String typeEdgeLabel = "type=" + GraphQLTypeUtil.simplePrint(type) + ";defaultValue=";
        if (graphQLArgument.hasSetDefaultValue()) {
            typeEdgeLabel += AstPrinter.printAst(ValuesResolver.valueToLiteral(graphQLArgument.getArgumentDefaultValue(), graphQLArgument.getType(), GraphQLContext.getDefault(), Locale.getDefault()));
        }
        typeEdge.setLabel(typeEdgeLabel);
        schemaGraph.addEdge(typeEdge);
    }

    private void newObject(GraphQLObjectType graphQLObjectType, SchemaGraph schemaGraph, boolean isIntrospectionNode) {
        Vertex objectVertex = new Vertex(SchemaGraph.OBJECT, debugPrefix + String.valueOf(counter++));
        objectVertex.setBuiltInType(isIntrospectionNode);
        objectVertex.add("name", graphQLObjectType.getName());
        objectVertex.add("description", desc(graphQLObjectType.getDescription()));
        for (GraphQLFieldDefinition fieldDefinition : graphQLObjectType.getFieldDefinitions()) {
            Vertex newFieldVertex = newField(fieldDefinition, schemaGraph, isIntrospectionNode);
            schemaGraph.addVertex(newFieldVertex);
            schemaGraph.addEdge(new Edge(objectVertex, newFieldVertex));
        }
        schemaGraph.addVertex(objectVertex);
        schemaGraph.addType(graphQLObjectType.getName(), objectVertex);
        createAppliedDirectives(objectVertex, graphQLObjectType.getDirectives(), schemaGraph);
    }

    private Vertex newField(GraphQLFieldDefinition graphQLFieldDefinition, SchemaGraph schemaGraph, boolean isIntrospectionNode) {
        Vertex fieldVertex = new Vertex(SchemaGraph.FIELD, debugPrefix + String.valueOf(counter++));
        fieldVertex.setBuiltInType(isIntrospectionNode);
        fieldVertex.add("name", graphQLFieldDefinition.getName());
        fieldVertex.add("description", desc(graphQLFieldDefinition.getDescription()));
        for (GraphQLArgument argument : graphQLFieldDefinition.getArguments()) {
            Vertex argumentVertex = newArgument(argument, schemaGraph, isIntrospectionNode);
            schemaGraph.addVertex(argumentVertex);
            schemaGraph.addEdge(new Edge(fieldVertex, argumentVertex));
        }
        createAppliedDirectives(fieldVertex, graphQLFieldDefinition.getDirectives(), schemaGraph);
        return fieldVertex;
    }

    private Vertex newArgument(GraphQLArgument graphQLArgument, SchemaGraph schemaGraph, boolean isIntrospectionNode) {
        Vertex vertex = new Vertex(SchemaGraph.ARGUMENT, debugPrefix + String.valueOf(counter++));
        vertex.setBuiltInType(isIntrospectionNode);
        vertex.add("name", graphQLArgument.getName());
        vertex.add("description", desc(graphQLArgument.getDescription()));
        createAppliedDirectives(vertex, graphQLArgument.getDirectives(), schemaGraph);
        return vertex;
    }

    private void newScalar(GraphQLScalarType scalarType, SchemaGraph schemaGraph, boolean isIntrospectionNode) {
        Vertex scalarVertex = new Vertex(SchemaGraph.SCALAR, debugPrefix + String.valueOf(counter++));
        scalarVertex.setBuiltInType(isIntrospectionNode);
        if (ScalarInfo.isGraphqlSpecifiedScalar(scalarType.getName())) {
            scalarVertex.setBuiltInType(true);
        }
        scalarVertex.add("name", scalarType.getName());
        scalarVertex.add("description", desc(scalarType.getDescription()));
        scalarVertex.add("specifiedByUrl", scalarType.getSpecifiedByUrl());
        schemaGraph.addVertex(scalarVertex);
        schemaGraph.addType(scalarType.getName(), scalarVertex);
        createAppliedDirectives(scalarVertex, scalarType.getDirectives(), schemaGraph);
    }

    private void newInterface(GraphQLInterfaceType interfaceType, SchemaGraph schemaGraph, boolean isIntrospectionNode) {
        Vertex interfaceVertex = new Vertex(SchemaGraph.INTERFACE, debugPrefix + String.valueOf(counter++));
        interfaceVertex.setBuiltInType(isIntrospectionNode);
        interfaceVertex.add("name", interfaceType.getName());
        interfaceVertex.add("description", desc(interfaceType.getDescription()));
        for (GraphQLFieldDefinition fieldDefinition : interfaceType.getFieldDefinitions()) {
            Vertex newFieldVertex = newField(fieldDefinition, schemaGraph, isIntrospectionNode);
            schemaGraph.addVertex(newFieldVertex);
            schemaGraph.addEdge(new Edge(interfaceVertex, newFieldVertex));
        }
        schemaGraph.addVertex(interfaceVertex);
        schemaGraph.addType(interfaceType.getName(), interfaceVertex);
        createAppliedDirectives(interfaceVertex, interfaceType.getDirectives(), schemaGraph);
    }

    private void newEnum(GraphQLEnumType enumType, SchemaGraph schemaGraph, boolean isIntrospectionNode) {
        Vertex enumVertex = new Vertex(SchemaGraph.ENUM, debugPrefix + String.valueOf(counter++));
        enumVertex.setBuiltInType(isIntrospectionNode);
        enumVertex.add("name", enumType.getName());
        enumVertex.add("description", desc(enumType.getDescription()));
        for (GraphQLEnumValueDefinition enumValue : enumType.getValues()) {
            Vertex enumValueVertex = new Vertex(SchemaGraph.ENUM_VALUE, debugPrefix + String.valueOf(counter++));
            enumValueVertex.setBuiltInType(isIntrospectionNode);
            enumValueVertex.add("name", enumValue.getName());
            schemaGraph.addVertex(enumValueVertex);
            schemaGraph.addEdge(new Edge(enumVertex, enumValueVertex));
            createAppliedDirectives(enumValueVertex, enumValue.getDirectives(), schemaGraph);
        }
        schemaGraph.addVertex(enumVertex);
        schemaGraph.addType(enumType.getName(), enumVertex);
        createAppliedDirectives(enumVertex, enumType.getDirectives(), schemaGraph);
    }

    private void newUnion(GraphQLUnionType unionType, SchemaGraph schemaGraph, boolean isIntrospectionNode) {
        Vertex unionVertex = new Vertex(SchemaGraph.UNION, debugPrefix + String.valueOf(counter++));
        unionVertex.setBuiltInType(isIntrospectionNode);
        unionVertex.add("name", unionType.getName());
        unionVertex.add("description", desc(unionType.getDescription()));
        schemaGraph.addVertex(unionVertex);
        schemaGraph.addType(unionType.getName(), unionVertex);
        createAppliedDirectives(unionVertex, unionType.getDirectives(), schemaGraph);
    }

    private void newInputObject(GraphQLInputObjectType inputObject, SchemaGraph schemaGraph, boolean isIntrospectionNode) {
        Vertex inputObjectVertex = new Vertex(SchemaGraph.INPUT_OBJECT, debugPrefix + String.valueOf(counter++));
        inputObjectVertex.setBuiltInType(isIntrospectionNode);
        inputObjectVertex.add("name", inputObject.getName());
        inputObjectVertex.add("description", desc(inputObject.getDescription()));
        for (GraphQLInputObjectField inputObjectField : inputObject.getFieldDefinitions()) {
            Vertex newInputField = newInputField(inputObjectField, schemaGraph, isIntrospectionNode);
            Edge newEdge = new Edge(inputObjectVertex, newInputField);
            schemaGraph.addEdge(newEdge);
        }
        schemaGraph.addVertex(inputObjectVertex);
        schemaGraph.addType(inputObject.getName(), inputObjectVertex);
        createAppliedDirectives(inputObjectVertex, inputObject.getDirectives(), schemaGraph);
    }

    private void createAppliedDirectives(Vertex from,
                                         List appliedDirectives,
                                         SchemaGraph schemaGraph) {
        Map countByName = new LinkedHashMap<>();
        for (GraphQLDirective appliedDirective : appliedDirectives) {
            Vertex appliedDirectiveVertex = new Vertex(SchemaGraph.APPLIED_DIRECTIVE, debugPrefix + String.valueOf(counter++));
            appliedDirectiveVertex.add("name", appliedDirective.getName());
            for (GraphQLArgument appliedArgument : appliedDirective.getArguments()) {
                if (appliedArgument.hasSetValue()) {
                    Vertex appliedArgumentVertex = new Vertex(SchemaGraph.APPLIED_ARGUMENT, debugPrefix + String.valueOf(counter++));
                    appliedArgumentVertex.add("name", appliedArgument.getName());
                    appliedArgumentVertex.add("value", AstPrinter.printAst(ValuesResolver.valueToLiteral(appliedArgument.getArgumentValue(), appliedArgument.getType(), GraphQLContext.getDefault(), Locale.getDefault())));
                    schemaGraph.addVertex(appliedArgumentVertex);
                    schemaGraph.addEdge(new Edge(appliedDirectiveVertex, appliedArgumentVertex));
                }
            }
            schemaGraph.addVertex(appliedDirectiveVertex);

            // repeatable directives means we can have multiple directives with the same name
            // the edge label indicates the applied directive index
            Integer count = countByName.getOrDefault(appliedDirective.getName(), 0);
            schemaGraph.addEdge(new Edge(from, appliedDirectiveVertex, String.valueOf(count)));
            countByName.put(appliedDirective.getName(), count + 1);
        }
    }

    private void newDirective(GraphQLDirective directive, SchemaGraph schemaGraph) {
        Vertex directiveVertex = new Vertex(SchemaGraph.DIRECTIVE, debugPrefix + String.valueOf(counter++));
        directiveVertex.add("name", directive.getName());
        directiveVertex.add("repeatable", directive.isRepeatable());
        directiveVertex.add("locations", directive.validLocations());
        boolean graphqlSpecified = DirectiveInfo.isGraphqlSpecifiedDirective(directive.getName());
        directiveVertex.setBuiltInType(graphqlSpecified);
        directiveVertex.add("description", desc(directive.getDescription()));
        for (GraphQLArgument argument : directive.getArguments()) {
            Vertex argumentVertex = newArgument(argument, schemaGraph, graphqlSpecified);
            schemaGraph.addVertex(argumentVertex);
            schemaGraph.addEdge(new Edge(directiveVertex, argumentVertex));
        }
        schemaGraph.addDirective(directive.getName(), directiveVertex);
        schemaGraph.addVertex(directiveVertex);
    }

    private Vertex newInputField(GraphQLInputObjectField inputField, SchemaGraph schemaGraph, boolean isIntrospectionNode) {
        Vertex vertex = new Vertex(SchemaGraph.INPUT_FIELD, debugPrefix + String.valueOf(counter++));
        schemaGraph.addVertex(vertex);
        vertex.setBuiltInType(isIntrospectionNode);
        vertex.add("name", inputField.getName());
        vertex.add("description", desc(inputField.getDescription()));
        createAppliedDirectives(vertex, inputField.getDirectives(), schemaGraph);
        return vertex;
    }

    private String desc(String desc) {
        return desc;
//        return desc != null ? desc.replace("\n", "\\n") : null;
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy