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

io.stargate.graphql.schema.graphqlfirst.util.DirectedGraph Maven / Gradle / Ivy

There is a newer version: 2.0.31
Show newest version
/*
 * Copyright The Stargate Authors
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package io.stargate.graphql.schema.graphqlfirst.util;

import com.datastax.oss.driver.shaded.guava.common.annotations.VisibleForTesting;
import com.datastax.oss.driver.shaded.guava.common.base.Preconditions;
import com.datastax.oss.driver.shaded.guava.common.collect.LinkedHashMultimap;
import com.datastax.oss.driver.shaded.guava.common.collect.Lists;
import com.datastax.oss.driver.shaded.guava.common.collect.Maps;
import com.datastax.oss.driver.shaded.guava.common.collect.Multimap;
import java.util.ArrayDeque;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Queue;
import net.jcip.annotations.NotThreadSafe;

/**
 * A basic directed graph implementation to perform topological sorts.
 *
 * 

This class is not a general-purpose graph implementation: it is only intended for * clients that will build a graph, sort it, and then immediately throw it away. In particular, once * {@link #topologicalSort()} has been invoked, it is not possible to add vertices or sort again. */ @NotThreadSafe public class DirectedGraph { /** How many predecessors each vertex has. */ private final Map predecessorCounts; /** The list successors of each vertex. */ private final Multimap adjacencyList; private boolean wasSorted; public DirectedGraph(Collection vertices) { this.predecessorCounts = Maps.newLinkedHashMapWithExpectedSize(vertices.size()); this.adjacencyList = LinkedHashMultimap.create(); for (VertexT vertex : vertices) { this.predecessorCounts.put(vertex, 0); } } @VisibleForTesting @SafeVarargs DirectedGraph(VertexT... vertices) { this(Arrays.asList(vertices)); } /** * @throws IllegalStateException if {@link #topologicalSort()} was invoked before * @throws IllegalArgumentException if {@code from} or {@code to} is not part of the vertices * passed to the constructor */ public void addEdge(VertexT from, VertexT to) { Preconditions.checkState(!wasSorted); Preconditions.checkArgument( predecessorCounts.containsKey(from) && predecessorCounts.containsKey(to)); adjacencyList.put(from, to); predecessorCounts.put(to, predecessorCounts.get(to) + 1); } /** * Performs a topological sort of the graph: for every directed edge {@code from->to}, {@code * from} will come before {@code to} in the resulting list. * *

Implementation note: this is based on Kahn's algorithm, * the running time is linear in the number of nodes plus the number of edges. * * @throws IllegalStateException if this method was already invoked on this graph before * @throws IllegalArgumentException if the graph has a cycle */ public List topologicalSort() { Preconditions.checkState(!wasSorted); wasSorted = true; Queue startNodes = new ArrayDeque<>(); List result = Lists.newArrayList(); // Start with all the nodes that have no predecessors for (Map.Entry entry : predecessorCounts.entrySet()) { if (entry.getValue() == 0) { startNodes.add(entry.getKey()); } } // For each start node: // - move it to the result // - for each of its successors, decrement the predecessor count, and possibly promote to a // start node while (!startNodes.isEmpty()) { VertexT vertex = startNodes.remove(); result.add(vertex); for (VertexT successor : adjacencyList.get(vertex)) { if (decrementAndGetCount(successor) == 0) { startNodes.add(successor); } } } // If we've moved every node to the result, we have a solution, otherwise there is a cycle if (result.size() != predecessorCounts.size()) { throw new IllegalArgumentException("failed to perform topological sort, graph has a cycle"); } return result; } @SuppressWarnings("") private int decrementAndGetCount(VertexT vertex) { return predecessorCounts.compute( vertex, (__, count) -> { assert count != null; return count - 1; }); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy