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

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

package graphql.schema.diffing;


import com.google.common.collect.HashBasedTable;
import com.google.common.collect.Iterables;
import com.google.common.collect.LinkedHashMultimap;
import com.google.common.collect.Multimap;
import com.google.common.collect.Table;
import graphql.ExperimentalApi;
import org.jetbrains.annotations.Nullable;

import java.util.ArrayList;
import java.util.Collection;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.function.Predicate;

import static graphql.Assert.assertTrue;
import static java.lang.String.format;

@ExperimentalApi
public class SchemaGraph {

    public static final String SCHEMA = "Schema";
    public static final String OBJECT = "Object";
    public static final String INTERFACE = "Interface";
    public static final String UNION = "Union";
    public static final String FIELD = "Field";
    public static final String ARGUMENT = "Argument";
    public static final String SCALAR = "Scalar";
    public static final String ENUM = "Enum";
    public static final String ENUM_VALUE = "EnumValue";
    public static final String INPUT_OBJECT = "InputObject";
    public static final String INPUT_FIELD = "InputField";
    public static final String DIRECTIVE = "Directive";
    public static final String APPLIED_DIRECTIVE = "AppliedDirective";
    public static final String APPLIED_ARGUMENT = "AppliedArgument";
    public static final String ISOLATED = "__ISOLATED";


    private List vertices = new ArrayList<>();
    private List edges = new ArrayList<>();

    private Map typesByName = new LinkedHashMap<>();
    private Map directivesByName = new LinkedHashMap<>();
    private Table edgesByDirection = HashBasedTable.create();
    private Table edgesByInverseDirection = HashBasedTable.create();
    private Multimap typeToVertices = LinkedHashMultimap.create();

    public SchemaGraph() {

    }

    public SchemaGraph(List vertices, List edges, Table edgeByVertexPair) {
        this.vertices = vertices;
        this.edges = edges;
        this.edgesByDirection = edgeByVertexPair;
    }

    public void addVertex(Vertex vertex) {
        vertices.add(vertex);
        typeToVertices.put(vertex.getType(), vertex);
    }

    public void addVertices(Collection vertices) {
        for (Vertex vertex : vertices) {
            this.vertices.add(vertex);
            typeToVertices.put(vertex.getType(), vertex);
        }
    }

    public Collection getVerticesByType(String type) {
        return typeToVertices.get(type);
    }

    public Multimap getVerticesByType() {
        return typeToVertices;
    }

    public void addEdge(Edge edge) {
        edges.add(edge);
        edgesByDirection.put(edge.getFrom(), edge.getTo(), edge);
        edgesByInverseDirection.put(edge.getTo(), edge.getFrom(), edge);
    }

    //
//    public List getAdjacentEdges(Vertex from) {
//        return new ArrayList<>(edgesByDirection.row(from).values());
//    }
    public Collection getAdjacentEdgesNonCopy(Vertex from) {
        return edgesByDirection.row(from).values();
    }

    public Iterable getAdjacentEdgesAndInverseNonCopy(Vertex fromAndTo) {
        Collection edges = edgesByInverseDirection.row(fromAndTo).values();
        Collection edgesInverse = edgesByDirection.row(fromAndTo).values();
        return Iterables.concat(edges, edgesInverse);
    }

    public int adjacentEdgesAndInverseCount(Vertex fromAndTo) {
        return edgesByInverseDirection.row(fromAndTo).size() + edgesByDirection.row(fromAndTo).size();
    }

    public List getAdjacentVertices(Vertex from) {
        return getAdjacentVertices(from, x -> true);
    }

    public List getAdjacentVertices(Vertex from, Predicate predicate) {
        List result = new ArrayList<>();
        for (Edge edge : edgesByDirection.row(from).values()) {
            Vertex v = edge.getTo();
            if (predicate.test(v)) {
                result.add(v);
            }
        }
        return result;
    }

    public List getAdjacentVerticesInverse(Vertex to) {
        return getAdjacentVerticesInverse(to, x -> true);
    }

    public List getAdjacentVerticesInverse(Vertex to, Predicate predicate) {
        List result = new ArrayList<>();
        for (Edge edge : edgesByInverseDirection.row(to).values()) {
            Vertex v = edge.getFrom();
            if (predicate.test(v)) {
                result.add(v);
            }
        }
        return result;
    }

    public List getAdjacentEdges(Vertex from, Predicate predicate) {
        List result = new ArrayList<>();
        for (Edge edge : edgesByDirection.row(from).values()) {
            Vertex v = edge.getTo();
            if (predicate.test(v)) {
                result.add(edge);
            }
        }
        return result;
    }

    public List getAdjacentEdgesInverseCopied(Vertex to) {
        return new ArrayList<>(edgesByInverseDirection.row(to).values());
    }

    public Collection getAdjacentEdgesInverseNonCopy(Vertex to) {
        return edgesByInverseDirection.row(to).values();
    }

    public List getAdjacentEdgesInverse(Vertex to, Predicate predicate) {
        List result = new ArrayList<>();
        for (Edge edge : edgesByInverseDirection.row(to).values()) {
            Vertex v = edge.getFrom();
            if (predicate.test(v)) {
                result.add(edge);
            }
        }
        return result;
    }

    public Edge getSingleAdjacentEdge(Vertex from, Predicate predicate) {
        for (Edge edge : edgesByDirection.row(from).values()) {
            if (predicate.test(edge)) {
                return edge;
            }
        }
        return null;
    }

    public List getEdges() {
        return edges;
    }

    // null if the edge doesn't exist
    public @Nullable Edge getEdge(Vertex from, Vertex to) {
        return edgesByDirection.get(from, to);
    }

    public @Nullable Edge getEdgeOrInverse(Vertex from, Vertex to) {
        Edge result = edgesByDirection.get(from, to);
        return result != null ? result : edgesByInverseDirection.get(from, to);
    }

    public List getVertices() {
        return vertices;
    }

    public void setVertices(List vertices) {
        this.vertices = vertices;
    }

    public void addType(String name, Vertex vertex) {
        typesByName.put(name, vertex);
    }

    public void addDirective(String name, Vertex vertex) {
        directivesByName.put(name, vertex);
    }

    public Vertex getType(String name) {
        return typesByName.get(name);
    }

    public Vertex getDirective(String name) {
        return directivesByName.get(name);
    }

    public Optional findTargetVertex(Vertex from, Predicate vertexPredicate) {
        return edgesByDirection.row(from).values().stream().map(Edge::getTo).filter(vertexPredicate).findFirst();
    }

    public int size() {
        return vertices.size();
    }

    public List addIsolatedVertices(int count, String debugPrefix) {
        List result = new ArrayList<>();
        for (int i = 0; i < count; i++) {
            Vertex isolatedVertex = Vertex.newIsolatedNode(debugPrefix + i);
            vertices.add(isolatedVertex);
            result.add(isolatedVertex);
        }
        return result;
    }

    public Vertex getFieldOrDirectiveForArgument(Vertex argument) {
        List adjacentVertices = getAdjacentVerticesInverse(argument);
        assertTrue(adjacentVertices.size() == 1, () -> format("No field or directive found for %s", argument));
        return adjacentVertices.get(0);
    }

    public Vertex getFieldsContainerForField(Vertex field) {
        List adjacentVertices = getAdjacentVerticesInverse(field);
        assertTrue(adjacentVertices.size() == 1, () -> format("No fields container found for %s", field));
        return adjacentVertices.get(0);
    }

    public Vertex getInputObjectForInputField(Vertex inputField) {
        List adjacentVertices = this.getAdjacentVerticesInverse(inputField);
        assertTrue(adjacentVertices.size() == 1, () -> format("No input object found for %s", inputField));
        return adjacentVertices.get(0);
    }

    public Vertex getAppliedDirectiveForAppliedArgument(Vertex appliedArgument) {
        List adjacentVertices = this.getAdjacentVerticesInverse(appliedArgument);
        assertTrue(adjacentVertices.size() == 1, () -> format("No applied directive found for %s", appliedArgument));
        return adjacentVertices.get(0);
    }

    public Vertex getAppliedDirectiveContainerForAppliedDirective(Vertex appliedDirective) {
        List adjacentVertices = this.getAdjacentVerticesInverse(appliedDirective);
        assertTrue(adjacentVertices.size() == 1, () -> format("No applied directive container found for %s", appliedDirective));
        return adjacentVertices.get(0);
    }

    /**
     * Gets the one inverse adjacent edge to the input and gets the other vertex.
     *
     * @param input  the vertex input
     * @return a vertex
     */
    public Vertex getSingleAdjacentInverseVertex(Vertex input) {
        Collection adjacentVertices = this.getAdjacentEdgesInverseNonCopy(input);
        assertTrue(adjacentVertices.size() == 1, () -> format("No parent found for %s", input));
        return adjacentVertices.iterator().next().getFrom();
    }

    public int getAppliedDirectiveIndex(Vertex appliedDirective) {
        List adjacentEdges = this.getAdjacentEdgesInverseCopied(appliedDirective);
        assertTrue(adjacentEdges.size() == 1, () -> format("No applied directive container found for %s", appliedDirective));
        return Integer.parseInt(adjacentEdges.get(0).getLabel());
    }

    public Vertex getEnumForEnumValue(Vertex enumValue) {
        List adjacentVertices = this.getAdjacentVerticesInverse(enumValue);
        assertTrue(adjacentVertices.size() == 1, () -> format("No enum found for %s", enumValue));
        return adjacentVertices.get(0);
    }


    public List getAllAdjacentEdges(List fromList, Vertex to) {
        List result = new ArrayList<>();
        for (Vertex from : fromList) {
            Edge edge = getEdge(from, to);
            if (edge == null) {
                continue;
            }
            result.add(edge);
        }
        return result;
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy