Page MenuHomeSoftware Heritage

D1700.diff
No OneTemporary

D1700.diff

diff --git a/api/server/src/main/java/org/softwareheritage/graph/App.java b/api/server/src/main/java/org/softwareheritage/graph/App.java
--- a/api/server/src/main/java/org/softwareheritage/graph/App.java
+++ b/api/server/src/main/java/org/softwareheritage/graph/App.java
@@ -12,11 +12,8 @@
import org.softwareheritage.graph.Graph;
import org.softwareheritage.graph.SwhId;
-import org.softwareheritage.graph.algo.Leaves;
-import org.softwareheritage.graph.algo.Neighbors;
import org.softwareheritage.graph.algo.Stats;
-import org.softwareheritage.graph.algo.Visit;
-import org.softwareheritage.graph.algo.Walk;
+import org.softwareheritage.graph.algo.Traversal;
public class App {
public static void main(String[] args) throws IOException {
@@ -58,8 +55,8 @@
String direction = ctx.queryParam("direction", "forward");
String edgesFmt = ctx.queryParam("edges", "*");
- Leaves leaves = new Leaves(graph, src, edgesFmt, direction);
- ctx.json(leaves.getLeaves());
+ Traversal traversal = new Traversal(graph, direction, edgesFmt);
+ ctx.json(traversal.leavesEndpoint(src));
});
app.get("/neighbors/:src", ctx -> {
@@ -67,26 +64,27 @@
String direction = ctx.queryParam("direction", "forward");
String edgesFmt = ctx.queryParam("edges", "*");
- Neighbors neighbors = new Neighbors(graph, src, edgesFmt, direction);
- ctx.json(neighbors.getNeighbors());
+ Traversal traversal = new Traversal(graph, direction, edgesFmt);
+ ctx.json(traversal.neighborsEndpoint(src));
});
- app.get("/visit/:src", ctx -> {
+ // TODO: anonymous class to return both nodes/paths? (waiting on node types map merged/refactor)
+ /*app.get("/visit/:src", ctx -> {
SwhId src = new SwhId(ctx.pathParam("src"));
String direction = ctx.queryParam("direction", "forward");
String edgesFmt = ctx.queryParam("edges", "*");
Visit visit = new Visit(graph, src, edgesFmt, direction, Visit.OutputFmt.NODES_AND_PATHS);
ctx.json(visit);
- });
+ });*/
app.get("/visit/nodes/:src", ctx -> {
SwhId src = new SwhId(ctx.pathParam("src"));
String direction = ctx.queryParam("direction", "forward");
String edgesFmt = ctx.queryParam("edges", "*");
- Visit visit = new Visit(graph, src, edgesFmt, direction, Visit.OutputFmt.ONLY_NODES);
- ctx.json(visit.getNodes());
+ Traversal traversal = new Traversal(graph, direction, edgesFmt);
+ ctx.json(traversal.visitNodesEndpoint(src));
});
app.get("/visit/paths/:src", ctx -> {
@@ -94,8 +92,8 @@
String direction = ctx.queryParam("direction", "forward");
String edgesFmt = ctx.queryParam("edges", "*");
- Visit visit = new Visit(graph, src, edgesFmt, direction, Visit.OutputFmt.ONLY_PATHS);
- ctx.json(visit.getPaths());
+ Traversal traversal = new Traversal(graph, direction, edgesFmt);
+ ctx.json(traversal.visitPathsEndpoint(src));
});
app.get("/walk/:src/:dst", ctx -> {
@@ -103,10 +101,10 @@
String dstFmt = ctx.pathParam("dst");
String direction = ctx.queryParam("direction", "forward");
String edgesFmt = ctx.queryParam("edges", "*");
- String traversal = ctx.queryParam("traversal", "dfs");
+ String algorithm = ctx.queryParam("traversal", "dfs");
- Walk walk = new Walk(graph, src, dstFmt, edgesFmt, direction, traversal);
- ctx.json(walk.getPath());
+ Traversal traversal = new Traversal(graph, direction, edgesFmt);
+ ctx.json(traversal.walkEndpoint(src, dstFmt, algorithm));
});
app.exception(IllegalArgumentException.class, (e, ctx) -> {
diff --git a/api/server/src/main/java/org/softwareheritage/graph/Graph.java b/api/server/src/main/java/org/softwareheritage/graph/Graph.java
--- a/api/server/src/main/java/org/softwareheritage/graph/Graph.java
+++ b/api/server/src/main/java/org/softwareheritage/graph/Graph.java
@@ -3,7 +3,6 @@
import java.io.IOException;
import it.unimi.dsi.big.webgraph.BVGraph;
-import it.unimi.dsi.big.webgraph.LazyLongIterator;
import org.softwareheritage.graph.SwhId;
import org.softwareheritage.graph.backend.NodeIdMap;
@@ -45,16 +44,16 @@
return graph.numArcs();
}
- public LazyLongIterator successors(long nodeId) {
- return graph.successors(nodeId);
+ public long[][] successors(long nodeId) {
+ return graph.successorBigArray(nodeId);
}
public long outdegree(long nodeId) {
return graph.outdegree(nodeId);
}
- public LazyLongIterator predecessors(long nodeId) {
- return graphTransposed.successors(nodeId);
+ public long[][] predecessors(long nodeId) {
+ return graphTransposed.successorBigArray(nodeId);
}
public long indegree(long nodeId) {
@@ -65,7 +64,7 @@
return (useTransposed) ? indegree(nodeId) : outdegree(nodeId);
}
- public LazyLongIterator neighbors(long nodeId, boolean useTransposed) {
+ public long[][] neighbors(long nodeId, boolean useTransposed) {
return (useTransposed) ? predecessors(nodeId) : successors(nodeId);
}
}
diff --git a/api/server/src/main/java/org/softwareheritage/graph/Neighbors.java b/api/server/src/main/java/org/softwareheritage/graph/Neighbors.java
new file mode 100644
--- /dev/null
+++ b/api/server/src/main/java/org/softwareheritage/graph/Neighbors.java
@@ -0,0 +1,60 @@
+package org.softwareheritage.graph;
+
+import java.util.Iterator;
+
+import it.unimi.dsi.fastutil.longs.LongBigArrays;
+
+import org.softwareheritage.graph.AllowedEdges;
+import org.softwareheritage.graph.Graph;
+import org.softwareheritage.graph.SwhId;
+
+public class Neighbors implements Iterable<Long> {
+ Graph graph;
+ boolean useTransposed;
+ AllowedEdges edges;
+ SwhId src;
+
+ public Neighbors(Graph graph, boolean useTransposed, AllowedEdges edges, SwhId src) {
+ this.graph = graph;
+ this.useTransposed = useTransposed;
+ this.edges = edges;
+ this.src = src;
+ }
+
+ @Override
+ public Iterator<Long> iterator() {
+ return new NeighborsIterator();
+ }
+
+ public class NeighborsIterator implements Iterator<Long> {
+ long nextNeighborIdx;
+ long nbNeighbors;
+ long[][] neighbors;
+
+ public NeighborsIterator() {
+ long srcNodeId = graph.getNodeId(src);
+
+ this.nextNeighborIdx = -1;
+ this.nbNeighbors = graph.degree(srcNodeId, useTransposed);
+ this.neighbors = graph.neighbors(srcNodeId, useTransposed);
+ }
+
+ // Look ahead because with edge restriction not all neighbors are considered
+ public boolean hasNext() {
+ for (long lookAheadIdx = nextNeighborIdx + 1; lookAheadIdx < nbNeighbors; lookAheadIdx++) {
+ long nextNodeId = LongBigArrays.get(neighbors, lookAheadIdx);
+ SwhId nextSwhId = graph.getSwhId(nextNodeId);
+ if (edges.isAllowed(src.getType(), nextSwhId.getType())) {
+ nextNeighborIdx = lookAheadIdx;
+ return true;
+ }
+ }
+ return false;
+ }
+
+ public Long next() {
+ long nextNodeId = LongBigArrays.get(neighbors, nextNeighborIdx);
+ return nextNodeId;
+ }
+ }
+}
diff --git a/api/server/src/main/java/org/softwareheritage/graph/algo/Leaves.java b/api/server/src/main/java/org/softwareheritage/graph/algo/Leaves.java
deleted file mode 100644
--- a/api/server/src/main/java/org/softwareheritage/graph/algo/Leaves.java
+++ /dev/null
@@ -1,64 +0,0 @@
-package org.softwareheritage.graph.algo;
-
-import java.util.ArrayList;
-
-import it.unimi.dsi.big.webgraph.LazyLongIterator;
-import it.unimi.dsi.bits.LongArrayBitVector;
-
-import org.softwareheritage.graph.AllowedEdges;
-import org.softwareheritage.graph.Graph;
-import org.softwareheritage.graph.Node;
-import org.softwareheritage.graph.SwhId;
-
-public class Leaves {
- Graph graph;
- boolean useTransposed;
- AllowedEdges edges;
-
- ArrayList<SwhId> leaves;
- LongArrayBitVector visited;
-
- public Leaves(Graph graph, SwhId src, String edgesFmt, String direction) {
- if (!direction.matches("forward|backward")) {
- throw new IllegalArgumentException("Unknown traversal direction: " + direction);
- }
-
- this.graph = graph;
- this.useTransposed = (direction.equals("backward"));
- this.edges = new AllowedEdges(edgesFmt);
- this.leaves = new ArrayList<SwhId>();
- this.visited = LongArrayBitVector.ofLength(graph.getNbNodes());
-
- long nodeId = graph.getNodeId(src);
- dfs(nodeId);
- }
-
- public ArrayList<SwhId> getLeaves() {
- return leaves;
- }
-
- private void dfs(long currentNodeId) {
- visited.set(currentNodeId);
- SwhId currentSwhId = graph.getSwhId(currentNodeId);
-
- long degree = graph.degree(currentNodeId, useTransposed);
- LazyLongIterator neighbors = graph.neighbors(currentNodeId, useTransposed);
- long neighborsCnt = 0;
-
- while (degree-- > 0) {
- long neighborNodeId = neighbors.nextLong();
- Node.Type currentNodeType = currentSwhId.getType();
- Node.Type neighborNodeType = graph.getSwhId(neighborNodeId).getType();
- if (edges.isAllowed(currentNodeType, neighborNodeType)) {
- neighborsCnt++;
- if (!visited.getBoolean(neighborNodeId)) {
- dfs(neighborNodeId);
- }
- }
- }
-
- if (neighborsCnt == 0) {
- leaves.add(currentSwhId);
- }
- }
-}
diff --git a/api/server/src/main/java/org/softwareheritage/graph/algo/Neighbors.java b/api/server/src/main/java/org/softwareheritage/graph/algo/Neighbors.java
deleted file mode 100644
--- a/api/server/src/main/java/org/softwareheritage/graph/algo/Neighbors.java
+++ /dev/null
@@ -1,49 +0,0 @@
-package org.softwareheritage.graph.algo;
-
-import java.util.ArrayList;
-
-import it.unimi.dsi.big.webgraph.LazyLongIterator;
-
-import org.softwareheritage.graph.AllowedEdges;
-import org.softwareheritage.graph.Graph;
-import org.softwareheritage.graph.SwhId;
-
-public class Neighbors {
- Graph graph;
- boolean useTransposed;
- AllowedEdges edges;
-
- ArrayList<SwhId> neighbors;
-
- public Neighbors(Graph graph, SwhId src, String edgesFmt, String direction) {
- if (!direction.matches("forward|backward")) {
- throw new IllegalArgumentException("Unknown traversal direction: " + direction);
- }
-
- this.graph = graph;
- this.useTransposed = (direction.equals("backward"));
- this.edges = new AllowedEdges(edgesFmt);
- this.neighbors = new ArrayList<SwhId>();
-
- iterateNeighbors(src);
- }
-
- public ArrayList<SwhId> getNeighbors() {
- return neighbors;
- }
-
- private void iterateNeighbors(SwhId swhId) {
- long nodeId = graph.getNodeId(swhId);
- long degree = graph.degree(nodeId, useTransposed);
- LazyLongIterator neighborsNodeIds = graph.neighbors(nodeId, useTransposed);
-
- while (degree-- > 0) {
- long neighborNodeId = neighborsNodeIds.nextLong();
- SwhId neighborSwhId = graph.getSwhId(neighborNodeId);
- if (edges.isAllowed(swhId.getType(), neighborSwhId.getType())) {
- neighbors.add(neighborSwhId);
- }
- }
- }
-}
-
diff --git a/api/server/src/main/java/org/softwareheritage/graph/algo/Traversal.java b/api/server/src/main/java/org/softwareheritage/graph/algo/Traversal.java
new file mode 100644
--- /dev/null
+++ b/api/server/src/main/java/org/softwareheritage/graph/algo/Traversal.java
@@ -0,0 +1,254 @@
+package org.softwareheritage.graph.algo;
+
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.LinkedList;
+import java.util.Queue;
+import java.util.Stack;
+
+import it.unimi.dsi.bits.LongArrayBitVector;
+import it.unimi.dsi.fastutil.longs.LongBigArrays;
+
+import org.softwareheritage.graph.AllowedEdges;
+import org.softwareheritage.graph.Graph;
+import org.softwareheritage.graph.Neighbors;
+import org.softwareheritage.graph.Node;
+import org.softwareheritage.graph.SwhId;
+
+public class Traversal {
+ Graph graph;
+ boolean useTransposed;
+ AllowedEdges edges;
+
+ // Big array storing if we have visited a node
+ LongArrayBitVector visited;
+ // Big array storing parents to retrieve the path when backtracking
+ long[][] nodeParent;
+
+ public Traversal(Graph graph, String direction, String edgesFmt) {
+ if (!direction.matches("forward|backward")) {
+ throw new IllegalArgumentException("Unknown traversal direction: " + direction);
+ }
+
+ this.graph = graph;
+ this.useTransposed = (direction.equals("backward"));
+ this.edges = new AllowedEdges(edgesFmt);
+
+ long nbNodes = graph.getNbNodes();
+ this.visited = LongArrayBitVector.ofLength(nbNodes);
+ this.nodeParent = LongBigArrays.newBigArray(nbNodes);
+ }
+
+ // TODO: better separation between internal/external ids (waiting on node types map to be merged)
+ private ArrayList<SwhId> convertToSwhIds(ArrayList<Long> nodeIds) {
+ ArrayList<SwhId> swhIds = new ArrayList<SwhId>();
+ for (long nodeId : nodeIds) {
+ swhIds.add(graph.getSwhId(nodeId));
+ }
+ return swhIds;
+ }
+
+ public ArrayList<SwhId> leavesEndpoint(SwhId src) {
+ long nodeId = graph.getNodeId(src);
+ ArrayList<Long> leavesNodeIds = leavesInternalEndpoint(nodeId);
+ return convertToSwhIds(leavesNodeIds);
+ }
+
+ public ArrayList<SwhId> neighborsEndpoint(SwhId src) {
+ long nodeId = graph.getNodeId(src);
+ ArrayList<Long> neighborsNodeIds = neighborsInternalEndpoint(nodeId);
+ return convertToSwhIds(neighborsNodeIds);
+ }
+
+ public ArrayList<SwhId> walkEndpoint(SwhId src, String dstFmt, String traversal) {
+ if (!traversal.matches("dfs|bfs")) {
+ throw new IllegalArgumentException("Unknown traversal algorithm: " + traversal);
+ }
+
+ long srcNodeId = graph.getNodeId(src);
+ long dstNodeId = (traversal.equals("dfs")) ? dfs(srcNodeId, dstFmt) : bfs(srcNodeId, dstFmt);
+ if (dstNodeId == -1) {
+ throw new IllegalArgumentException("Unable to find destination point: " + dstFmt);
+ }
+
+ ArrayList<Long> path = backtracking(srcNodeId, dstNodeId);
+ return convertToSwhIds(path);
+ }
+
+ public ArrayList<SwhId> visitNodesEndpoint(SwhId src) {
+ long nodeId = graph.getNodeId(src);
+ ArrayList<Long> nodes = visitNodesInternalEndpoint(nodeId);
+ return convertToSwhIds(nodes);
+ }
+
+ public ArrayList<ArrayList<SwhId>> visitPathsEndpoint(SwhId src) {
+ long nodeId = graph.getNodeId(src);
+ ArrayList<ArrayList<Long>> pathNodeIds = new ArrayList<>();
+ Stack<Long> currentPath = new Stack<Long>();
+ visitPathsInternalEndpoint(nodeId, pathNodeIds, currentPath);
+
+ ArrayList<ArrayList<SwhId>> paths = new ArrayList<>();
+ for (ArrayList<Long> nodeIds : pathNodeIds) {
+ paths.add(convertToSwhIds(nodeIds));
+ }
+ return paths;
+ }
+
+ private ArrayList<Long> leavesInternalEndpoint(long srcNodeId) {
+ ArrayList<Long> leaves = new ArrayList<Long>();
+ Stack<Long> stack = new Stack<Long>();
+ this.visited.clear();
+
+ stack.push(srcNodeId);
+ visited.set(srcNodeId);
+
+ while (!stack.isEmpty()) {
+ long currentNodeId = stack.pop();
+ SwhId currentSwhId = graph.getSwhId(currentNodeId);
+
+ long neighborsCnt = 0;
+ for (long neighborNodeId : new Neighbors(graph, useTransposed, edges, currentSwhId)) {
+ neighborsCnt++;
+ if (!visited.getBoolean(neighborNodeId)) {
+ stack.push(neighborNodeId);
+ visited.set(neighborNodeId);
+ }
+ }
+
+ if (neighborsCnt == 0) {
+ leaves.add(currentNodeId);
+ }
+ }
+
+ return leaves;
+ }
+
+ private ArrayList<Long> neighborsInternalEndpoint(long srcNodeId) {
+ SwhId srcSwhId = graph.getSwhId(srcNodeId);
+ ArrayList<Long> neighbors = new ArrayList<Long>();
+ for (long neighborNodeId : new Neighbors(graph, useTransposed, edges, srcSwhId)) {
+ neighbors.add(neighborNodeId);
+ }
+ return neighbors;
+ }
+
+ private ArrayList<Long> visitNodesInternalEndpoint(long srcNodeId) {
+ // No specific destination point
+ String dstFmt = null;
+ dfs(srcNodeId, dstFmt);
+
+ ArrayList<Long> nodes = new ArrayList<Long>();
+ for (long nodeId = 0; nodeId < graph.getNbNodes(); nodeId++) {
+ if (this.visited.getBoolean(nodeId)) {
+ nodes.add(nodeId);
+ }
+ }
+ return nodes;
+ }
+
+ private void visitPathsInternalEndpoint(
+ long currentNodeId, ArrayList<ArrayList<Long>> paths, Stack<Long> currentPath) {
+ SwhId currentSwhId = graph.getSwhId(currentNodeId);
+ currentPath.push(currentNodeId);
+
+ long visitedNeighbors = 0;
+ for (long neighborNodeId : new Neighbors(graph, useTransposed, edges, currentSwhId)) {
+ visitPathsInternalEndpoint(neighborNodeId, paths, currentPath);
+ visitedNeighbors++;
+ }
+
+ if (visitedNeighbors == 0) {
+ ArrayList<Long> path = new ArrayList<Long>();
+ for (long nodeId : currentPath) {
+ path.add(nodeId);
+ }
+ paths.add(path);
+ }
+
+ currentPath.pop();
+ }
+
+ private ArrayList<Long> backtracking(long srcNodeId, long dstNodeId) {
+ ArrayList<Long> path = new ArrayList<Long>();
+ long currentNodeId = dstNodeId;
+ while (currentNodeId != srcNodeId) {
+ path.add(currentNodeId);
+ currentNodeId = LongBigArrays.get(nodeParent, currentNodeId);
+ }
+ path.add(srcNodeId);
+ Collections.reverse(path);
+ return path;
+ }
+
+ private long dfs(long srcNodeId, String dstFmt) {
+ Stack<Long> stack = new Stack<Long>();
+ this.visited.clear();
+
+ stack.push(srcNodeId);
+ visited.set(srcNodeId);
+
+ while (!stack.isEmpty()) {
+ long currentNodeId = stack.pop();
+ SwhId currentSwhId = graph.getSwhId(currentNodeId);
+ if (isDestinationNode(currentSwhId, dstFmt)) {
+ return currentNodeId;
+ }
+
+ for (long neighborNodeId : new Neighbors(graph, useTransposed, edges, currentSwhId)) {
+ if (!visited.getBoolean(neighborNodeId)) {
+ stack.push(neighborNodeId);
+ visited.set(neighborNodeId);
+ LongBigArrays.set(nodeParent, neighborNodeId, currentNodeId);
+ }
+ }
+ }
+
+ return -1;
+ }
+
+ private long bfs(long srcNodeId, String dstFmt) {
+ Queue<Long> queue = new LinkedList<Long>();
+ this.visited.clear();
+
+ queue.add(srcNodeId);
+ visited.set(srcNodeId);
+
+ while (!queue.isEmpty()) {
+ long currentNodeId = queue.poll();
+ SwhId currentSwhId = graph.getSwhId(currentNodeId);
+ if (isDestinationNode(currentSwhId, dstFmt)) {
+ return currentNodeId;
+ }
+
+ for (long neighborNodeId : new Neighbors(graph, useTransposed, edges, currentSwhId)) {
+ if (!visited.getBoolean(neighborNodeId)) {
+ queue.add(neighborNodeId);
+ visited.set(neighborNodeId);
+ LongBigArrays.set(nodeParent, neighborNodeId, currentNodeId);
+ }
+ }
+ }
+
+ return -1;
+ }
+
+ private boolean isDestinationNode(SwhId swhId, String dstFmt) {
+ // No destination node, early exit
+ if (dstFmt == null) {
+ return false;
+ }
+
+ // SwhId as destination node
+ if (swhId.toString().equals(dstFmt)) {
+ return true;
+ }
+
+ // Node.Type as destination node
+ try {
+ Node.Type dstType = Node.Type.fromStr(dstFmt);
+ return (swhId.getType().equals(dstType));
+ } catch (IllegalArgumentException e) {
+ return false;
+ }
+ }
+}
diff --git a/api/server/src/main/java/org/softwareheritage/graph/algo/Visit.java b/api/server/src/main/java/org/softwareheritage/graph/algo/Visit.java
deleted file mode 100644
--- a/api/server/src/main/java/org/softwareheritage/graph/algo/Visit.java
+++ /dev/null
@@ -1,104 +0,0 @@
-package org.softwareheritage.graph.algo;
-
-import java.util.ArrayList;
-import java.util.LinkedHashSet;
-import java.util.Stack;
-
-import it.unimi.dsi.big.webgraph.LazyLongIterator;
-import it.unimi.dsi.bits.LongArrayBitVector;
-
-import org.softwareheritage.graph.AllowedEdges;
-import org.softwareheritage.graph.Graph;
-import org.softwareheritage.graph.Node;
-import org.softwareheritage.graph.SwhId;
-import org.softwareheritage.graph.SwhPath;
-
-public class Visit {
- public enum OutputFmt { ONLY_NODES, ONLY_PATHS, NODES_AND_PATHS }
-
- Graph graph;
- boolean useTransposed;
- AllowedEdges edges;
-
- // LinkedHashSet is necessary to preserve insertion order
- LinkedHashSet<SwhId> nodes;
- ArrayList<SwhPath> paths;
- Stack<Long> currentPath;
- LongArrayBitVector visited;
-
- public Visit(Graph graph, SwhId src, String edgesFmt, String direction, OutputFmt output) {
- if (!direction.matches("forward|backward")) {
- throw new IllegalArgumentException("Unknown traversal direction: " + direction);
- }
-
- this.graph = graph;
- this.useTransposed = (direction.equals("backward"));
- this.edges = new AllowedEdges(edgesFmt);
- this.nodes = new LinkedHashSet<SwhId>();
- this.paths = new ArrayList<SwhPath>();
- this.currentPath = new Stack<Long>();
- this.visited = LongArrayBitVector.ofLength(graph.getNbNodes());
-
- long nodeId = graph.getNodeId(src);
- if (output == OutputFmt.ONLY_NODES) {
- dfsOutputOnlyNodes(nodeId);
- } else {
- dfs(nodeId);
- }
- }
-
- public LinkedHashSet<SwhId> getNodes() {
- return nodes;
- }
-
- public ArrayList<SwhPath> getPaths() {
- return paths;
- }
-
- private void dfs(long currentNodeId) {
- nodes.add(graph.getSwhId(currentNodeId));
- currentPath.push(currentNodeId);
-
- long degree = graph.degree(currentNodeId, useTransposed);
- LazyLongIterator neighbors = graph.neighbors(currentNodeId, useTransposed);
- long visitedNeighbors = 0;
-
- while (degree-- > 0) {
- long neighborNodeId = neighbors.nextLong();
- Node.Type currentNodeType = graph.getSwhId(currentNodeId).getType();
- Node.Type neighborNodeType = graph.getSwhId(neighborNodeId).getType();
- if (edges.isAllowed(currentNodeType, neighborNodeType)) {
- dfs(neighborNodeId);
- visitedNeighbors++;
- }
- }
-
- if (visitedNeighbors == 0) {
- SwhPath path = new SwhPath();
- for (long nodeId : currentPath) {
- path.add(graph.getSwhId(nodeId));
- }
- paths.add(path);
- }
-
- currentPath.pop();
- }
-
- private void dfsOutputOnlyNodes(long currentNodeId) {
- nodes.add(graph.getSwhId(currentNodeId));
- visited.set(currentNodeId);
-
- long degree = graph.degree(currentNodeId, useTransposed);
- LazyLongIterator neighbors = graph.neighbors(currentNodeId, useTransposed);
-
- while (degree-- > 0) {
- long neighborNodeId = neighbors.nextLong();
- Node.Type currentNodeType = graph.getSwhId(currentNodeId).getType();
- Node.Type neighborNodeType = graph.getSwhId(neighborNodeId).getType();
- if (!visited.getBoolean(neighborNodeId)
- && edges.isAllowed(currentNodeType, neighborNodeType)) {
- dfsOutputOnlyNodes(neighborNodeId);
- }
- }
- }
-}
diff --git a/api/server/src/main/java/org/softwareheritage/graph/algo/Walk.java b/api/server/src/main/java/org/softwareheritage/graph/algo/Walk.java
deleted file mode 100644
--- a/api/server/src/main/java/org/softwareheritage/graph/algo/Walk.java
+++ /dev/null
@@ -1,148 +0,0 @@
-package org.softwareheritage.graph.algo;
-
-import java.util.Stack;
-import java.util.Queue;
-import java.util.LinkedList;
-
-import it.unimi.dsi.big.webgraph.LazyLongIterator;
-import it.unimi.dsi.bits.LongArrayBitVector;
-import it.unimi.dsi.fastutil.longs.LongBigArrays;
-
-import org.softwareheritage.graph.AllowedEdges;
-import org.softwareheritage.graph.Graph;
-import org.softwareheritage.graph.Node;
-import org.softwareheritage.graph.SwhId;
-import org.softwareheritage.graph.SwhPath;
-
-public class Walk {
- Graph graph;
- boolean useTransposed;
- AllowedEdges edges;
-
- SwhPath path;
- long[][] nodeParent;
-
- public Walk(
- Graph graph, SwhId src, String dstFmt, String edgesFmt, String direction, String traversal) {
- if (!traversal.matches("dfs|bfs")) {
- throw new IllegalArgumentException("Unknown traversal algorithm: " + traversal);
- }
- if (!direction.matches("forward|backward")) {
- throw new IllegalArgumentException("Unknown traversal direction: " + direction);
- }
-
- this.graph = graph;
- this.useTransposed = (direction.equals("backward"));
- this.edges = new AllowedEdges(edgesFmt);
- this.path = new SwhPath();
- this.nodeParent = LongBigArrays.newBigArray(graph.getNbNodes());
- LongBigArrays.fill(nodeParent, -1);
-
- long srcNodeId = graph.getNodeId(src);
- long dstNodeId;
- if (traversal.equals("dfs")) {
- dstNodeId = dfs(srcNodeId, dstFmt);
- } else {
- dstNodeId = bfs(srcNodeId, dstFmt);
- }
-
- if (dstNodeId == -1) {
- throw new IllegalArgumentException("Unable to find destination point: " + dstFmt);
- } else {
- backtracking(srcNodeId, dstNodeId);
- }
- }
-
- public SwhPath getPath() {
- return path;
- }
-
- private void backtracking(long srcNodeId, long dstNodeId) {
- long currentNodeId = dstNodeId;
- while (currentNodeId != srcNodeId) {
- path.add(graph.getSwhId(currentNodeId));
- currentNodeId = LongBigArrays.get(nodeParent, currentNodeId);
- }
- path.add(graph.getSwhId(srcNodeId));
- path.reverse();
- }
-
- private long dfs(long srcNodeId, String dstFmt) {
- Stack<Long> stack = new Stack<Long>();
- LongArrayBitVector visited = LongArrayBitVector.ofLength(graph.getNbNodes());
-
- stack.push(srcNodeId);
- visited.set(srcNodeId);
-
- while (!stack.isEmpty()) {
- long currentNodeId = stack.pop();
- SwhId currentSwhId = graph.getSwhId(currentNodeId);
- if (isDestinationNode(currentSwhId, dstFmt)) {
- return currentNodeId;
- }
-
- long degree = graph.degree(currentNodeId, useTransposed);
- LazyLongIterator neighbors = graph.neighbors(currentNodeId, useTransposed);
-
- while (degree-- > 0) {
- long neighborNodeId = neighbors.nextLong();
- SwhId neighborSwhId = graph.getSwhId(neighborNodeId);
- if (!visited.getBoolean(neighborNodeId)
- && edges.isAllowed(currentSwhId.getType(), neighborSwhId.getType())) {
- stack.push(neighborNodeId);
- visited.set(neighborNodeId);
- LongBigArrays.set(nodeParent, neighborNodeId, currentNodeId);
- }
- }
- }
-
- return -1;
- }
-
- private long bfs(long srcNodeId, String dstFmt) {
- Queue<Long> queue = new LinkedList<Long>();
- LongArrayBitVector visited = LongArrayBitVector.ofLength(graph.getNbNodes());
-
- queue.add(srcNodeId);
- visited.set(srcNodeId);
-
- while (!queue.isEmpty()) {
- long currentNodeId = queue.poll();
- SwhId currentSwhId = graph.getSwhId(currentNodeId);
- if (isDestinationNode(currentSwhId, dstFmt)) {
- return currentNodeId;
- }
-
- long degree = graph.degree(currentNodeId, useTransposed);
- LazyLongIterator neighbors = graph.neighbors(currentNodeId, useTransposed);
-
- while (degree-- > 0) {
- long neighborNodeId = neighbors.nextLong();
- SwhId neighborSwhId = graph.getSwhId(neighborNodeId);
- if (!visited.getBoolean(neighborNodeId)
- && edges.isAllowed(currentSwhId.getType(), neighborSwhId.getType())) {
- queue.add(neighborNodeId);
- visited.set(neighborNodeId);
- LongBigArrays.set(nodeParent, neighborNodeId, currentNodeId);
- }
- }
- }
-
- return -1;
- }
-
- private boolean isDestinationNode(SwhId swhId, String dstFmt) {
- // dstFmt is either a SwhId...
- if (swhId.toString().equals(dstFmt)) {
- return true;
- }
-
- // ...or a Node.Type
- try {
- Node.Type dstType = Node.Type.fromStr(dstFmt);
- return (swhId.getType().equals(dstType));
- } catch (IllegalArgumentException e) {
- return false;
- }
- }
-}
diff --git a/api/server/src/main/java/org/softwareheritage/graph/benchmark/LinuxLog.java b/api/server/src/main/java/org/softwareheritage/graph/benchmark/LinuxLog.java
--- a/api/server/src/main/java/org/softwareheritage/graph/benchmark/LinuxLog.java
+++ b/api/server/src/main/java/org/softwareheritage/graph/benchmark/LinuxLog.java
@@ -4,7 +4,7 @@
import org.softwareheritage.graph.Graph;
import org.softwareheritage.graph.SwhId;
-import org.softwareheritage.graph.algo.Visit;
+import org.softwareheritage.graph.algo.Traversal;
public class LinuxLog {
public static void main(String[] args) throws IOException {
@@ -18,14 +18,14 @@
System.out.println("Expecting " + expectedCount + " commits");
long startTime = System.nanoTime();
- Visit visit = new Visit(graph, commit, "rev:rev", "forward", Visit.OutputFmt.ONLY_NODES);
+ Traversal traversal = new Traversal(graph, "forward", "rev:rev");
+ long count = traversal.visitNodesEndpoint(commit).size();
+ if (count != expectedCount) {
+ throw new IllegalArgumentException("Counted " + count + " commits");
+ }
long endTime = System.nanoTime();
double duration = (double) (endTime - startTime) / 1_000_000_000;
System.out.println("Visit operation done in: " + duration + " seconds");
- long count = visit.getNodes().size();
- if (count != expectedCount) {
- throw new IllegalArgumentException("Counted " + count + " commits");
- }
}
}
diff --git a/api/server/src/test/java/org/softwareheritage/graph/LeavesTest.java b/api/server/src/test/java/org/softwareheritage/graph/LeavesTest.java
--- a/api/server/src/test/java/org/softwareheritage/graph/LeavesTest.java
+++ b/api/server/src/test/java/org/softwareheritage/graph/LeavesTest.java
@@ -7,14 +7,14 @@
import org.softwareheritage.graph.Graph;
import org.softwareheritage.graph.GraphTest;
import org.softwareheritage.graph.SwhId;
-import org.softwareheritage.graph.algo.Leaves;
+import org.softwareheritage.graph.algo.Traversal;
public class LeavesTest extends GraphTest {
@Test
public void forwardFromSnp() {
Graph graph = getGraph();
SwhId src = new SwhId("swh:1:snp:0000000000000000000000000000000000000020");
- Leaves leaves = new Leaves(graph, src, "*", "forward");
+ Traversal traversal = new Traversal(graph, "forward", "*");
ArrayList<SwhId> expectedLeaves = new ArrayList<>();
expectedLeaves.add(new SwhId("swh:1:cnt:0000000000000000000000000000000000000001"));
@@ -22,14 +22,14 @@
expectedLeaves.add(new SwhId("swh:1:cnt:0000000000000000000000000000000000000005"));
expectedLeaves.add(new SwhId("swh:1:cnt:0000000000000000000000000000000000000007"));
- GraphTest.assertEqualsAnyOrder(expectedLeaves, leaves.getLeaves());
+ assertEquals(expectedLeaves, traversal.leavesEndpoint(src));
}
@Test
public void forwardFromRel() {
Graph graph = getGraph();
SwhId src = new SwhId("swh:1:rel:0000000000000000000000000000000000000019");
- Leaves leaves = new Leaves(graph, src, "*", "forward");
+ Traversal traversal = new Traversal(graph, "forward", "*");
ArrayList<SwhId> expectedLeaves = new ArrayList<>();
expectedLeaves.add(new SwhId("swh:1:cnt:0000000000000000000000000000000000000015"));
@@ -40,44 +40,43 @@
expectedLeaves.add(new SwhId("swh:1:cnt:0000000000000000000000000000000000000007"));
expectedLeaves.add(new SwhId("swh:1:cnt:0000000000000000000000000000000000000011"));
- GraphTest.assertEqualsAnyOrder(expectedLeaves, leaves.getLeaves());
+ assertEquals(expectedLeaves, traversal.leavesEndpoint(src));
}
@Test
public void backwardFromLeaf() {
Graph graph = getGraph();
+ Traversal traversal = new Traversal(graph, "backward", "*");
SwhId src1 = new SwhId("swh:1:cnt:0000000000000000000000000000000000000015");
- Leaves leaves1 = new Leaves(graph, src1, "*", "backward");
ArrayList<SwhId> expectedLeaves1 = new ArrayList<>();
expectedLeaves1.add(new SwhId("swh:1:rel:0000000000000000000000000000000000000019"));
- GraphTest.assertEqualsAnyOrder(expectedLeaves1, leaves1.getLeaves());
+ assertEquals(expectedLeaves1, traversal.leavesEndpoint(src1));
SwhId src2 = new SwhId("swh:1:cnt:0000000000000000000000000000000000000004");
- Leaves leaves2 = new Leaves(graph, src2, "*", "backward");
ArrayList<SwhId> expectedLeaves2 = new ArrayList<>();
expectedLeaves2.add(new SwhId("swh:1:snp:0000000000000000000000000000000000000020"));
expectedLeaves2.add(new SwhId("swh:1:rel:0000000000000000000000000000000000000019"));
- GraphTest.assertEqualsAnyOrder(expectedLeaves2, leaves2.getLeaves());
+ assertEquals(expectedLeaves2, traversal.leavesEndpoint(src2));
}
@Test
public void forwardRevToRevOnly() {
Graph graph = getGraph();
SwhId src = new SwhId("swh:1:rev:0000000000000000000000000000000000000018");
- Leaves leaves = new Leaves(graph, src, "rev:rev", "forward");
+ Traversal traversal = new Traversal(graph, "forward", "rev:rev");
ArrayList<SwhId> expectedLeaves = new ArrayList<>();
expectedLeaves.add(new SwhId("swh:1:rev:0000000000000000000000000000000000000003"));
- GraphTest.assertEqualsAnyOrder(expectedLeaves, leaves.getLeaves());
+ assertEquals(expectedLeaves, traversal.leavesEndpoint(src));
}
@Test
public void forwardDirToAll() {
Graph graph = getGraph();
SwhId src = new SwhId("swh:1:dir:0000000000000000000000000000000000000008");
- Leaves leaves = new Leaves(graph, src, "dir:*", "forward");
+ Traversal traversal = new Traversal(graph, "forward", "dir:*");
ArrayList<SwhId> expectedLeaves = new ArrayList<>();
expectedLeaves.add(new SwhId("swh:1:cnt:0000000000000000000000000000000000000004"));
@@ -85,18 +84,18 @@
expectedLeaves.add(new SwhId("swh:1:cnt:0000000000000000000000000000000000000001"));
expectedLeaves.add(new SwhId("swh:1:cnt:0000000000000000000000000000000000000007"));
- GraphTest.assertEqualsAnyOrder(expectedLeaves, leaves.getLeaves());
+ assertEquals(expectedLeaves, traversal.leavesEndpoint(src));
}
@Test
public void backwardCntToDirDirToDir() {
Graph graph = getGraph();
SwhId src = new SwhId("swh:1:cnt:0000000000000000000000000000000000000005");
- Leaves leaves = new Leaves(graph, src, "cnt:dir,dir:dir", "backward");
+ Traversal traversal = new Traversal(graph, "backward", "cnt:dir,dir:dir");
ArrayList<SwhId> expectedLeaves = new ArrayList<>();
expectedLeaves.add(new SwhId("swh:1:dir:0000000000000000000000000000000000000012"));
- GraphTest.assertEqualsAnyOrder(expectedLeaves, leaves.getLeaves());
+ assertEquals(expectedLeaves, traversal.leavesEndpoint(src));
}
}
diff --git a/api/server/src/test/java/org/softwareheritage/graph/NeighborsTest.java b/api/server/src/test/java/org/softwareheritage/graph/NeighborsTest.java
--- a/api/server/src/test/java/org/softwareheritage/graph/NeighborsTest.java
+++ b/api/server/src/test/java/org/softwareheritage/graph/NeighborsTest.java
@@ -7,7 +7,7 @@
import org.softwareheritage.graph.Graph;
import org.softwareheritage.graph.GraphTest;
import org.softwareheritage.graph.SwhId;
-import org.softwareheritage.graph.algo.Neighbors;
+import org.softwareheritage.graph.algo.Traversal;
public class NeighborsTest extends GraphTest {
@Test
@@ -16,24 +16,24 @@
ArrayList<SwhId> expectedNodes = new ArrayList<>();
SwhId src1 = new SwhId("swh:1:snp:0000000000000000000000000000000000000020");
- Neighbors neighbors1 = new Neighbors(graph, src1, "*", "backward");
- GraphTest.assertEqualsAnyOrder(expectedNodes, neighbors1.getNeighbors());
+ Traversal traversal1 = new Traversal(graph, "backward", "*");
+ GraphTest.assertEqualsAnyOrder(expectedNodes, traversal1.neighborsEndpoint(src1));
SwhId src2 = new SwhId("swh:1:cnt:0000000000000000000000000000000000000004");
- Neighbors neighbors2 = new Neighbors(graph, src2, "*", "forward");
- GraphTest.assertEqualsAnyOrder(expectedNodes, neighbors2.getNeighbors());
+ Traversal traversal2 = new Traversal(graph, "forward", "*");
+ GraphTest.assertEqualsAnyOrder(expectedNodes, traversal2.neighborsEndpoint(src2));
SwhId src3 = new SwhId("swh:1:cnt:0000000000000000000000000000000000000015");
- Neighbors neighbors3 = new Neighbors(graph, src3, "*", "forward");
- GraphTest.assertEqualsAnyOrder(expectedNodes, neighbors3.getNeighbors());
+ Traversal traversal3 = new Traversal(graph, "forward", "*");
+ GraphTest.assertEqualsAnyOrder(expectedNodes, traversal3.neighborsEndpoint(src3));
SwhId src4 = new SwhId("swh:1:rel:0000000000000000000000000000000000000019");
- Neighbors neighbors4 = new Neighbors(graph, src4, "*", "backward");
- GraphTest.assertEqualsAnyOrder(expectedNodes, neighbors4.getNeighbors());
+ Traversal traversal4 = new Traversal(graph, "backward", "*");
+ GraphTest.assertEqualsAnyOrder(expectedNodes, traversal4.neighborsEndpoint(src4));
SwhId src5 = new SwhId("swh:1:dir:0000000000000000000000000000000000000008");
- Neighbors neighbors5 = new Neighbors(graph, src5, "snp:*,rev:*,rel:*", "forward");
- GraphTest.assertEqualsAnyOrder(expectedNodes, neighbors5.getNeighbors());
+ Traversal traversal5 = new Traversal(graph, "forward", "snp:*,rev:*,rel:*");
+ GraphTest.assertEqualsAnyOrder(expectedNodes, traversal5.neighborsEndpoint(src5));
}
@Test
@@ -41,28 +41,28 @@
Graph graph = getGraph();
SwhId src1 = new SwhId("swh:1:rev:0000000000000000000000000000000000000003");
- Neighbors neighbors1 = new Neighbors(graph, src1, "*", "forward");
+ Traversal traversal1 = new Traversal(graph, "forward", "*");
ArrayList<SwhId> expectedNodes1 = new ArrayList<>();
expectedNodes1.add(new SwhId("swh:1:dir:0000000000000000000000000000000000000002"));
- GraphTest.assertEqualsAnyOrder(expectedNodes1, neighbors1.getNeighbors());
+ GraphTest.assertEqualsAnyOrder(expectedNodes1, traversal1.neighborsEndpoint(src1));
SwhId src2 = new SwhId("swh:1:dir:0000000000000000000000000000000000000017");
- Neighbors neighbors2 = new Neighbors(graph, src2, "dir:cnt", "forward");
+ Traversal traversal2 = new Traversal(graph, "forward", "dir:cnt");
ArrayList<SwhId> expectedNodes2 = new ArrayList<>();
expectedNodes2.add(new SwhId("swh:1:cnt:0000000000000000000000000000000000000014"));
- GraphTest.assertEqualsAnyOrder(expectedNodes2, neighbors2.getNeighbors());
+ GraphTest.assertEqualsAnyOrder(expectedNodes2, traversal2.neighborsEndpoint(src2));
SwhId src3 = new SwhId("swh:1:dir:0000000000000000000000000000000000000012");
- Neighbors neighbors3 = new Neighbors(graph, src3, "*", "backward");
+ Traversal traversal3 = new Traversal(graph, "backward", "*");
ArrayList<SwhId> expectedNodes3 = new ArrayList<>();
expectedNodes3.add(new SwhId("swh:1:rev:0000000000000000000000000000000000000013"));
- GraphTest.assertEqualsAnyOrder(expectedNodes3, neighbors3.getNeighbors());
+ GraphTest.assertEqualsAnyOrder(expectedNodes3, traversal3.neighborsEndpoint(src3));
SwhId src4 = new SwhId("swh:1:rev:0000000000000000000000000000000000000009");
- Neighbors neighbors4 = new Neighbors(graph, src4, "rev:rev", "backward");
+ Traversal traversal4 = new Traversal(graph, "backward", "rev:rev");
ArrayList<SwhId> expectedNodes4 = new ArrayList<>();
expectedNodes4.add(new SwhId("swh:1:rev:0000000000000000000000000000000000000013"));
- GraphTest.assertEqualsAnyOrder(expectedNodes4, neighbors4.getNeighbors());
+ GraphTest.assertEqualsAnyOrder(expectedNodes4, traversal4.neighborsEndpoint(src4));
}
@Test
@@ -70,32 +70,32 @@
Graph graph = getGraph();
SwhId src1 = new SwhId("swh:1:snp:0000000000000000000000000000000000000020");
- Neighbors neighbors1 = new Neighbors(graph, src1, "*", "forward");
+ Traversal traversal1 = new Traversal(graph, "forward", "*");
ArrayList<SwhId> expectedNodes1 = new ArrayList<>();
expectedNodes1.add(new SwhId("swh:1:rel:0000000000000000000000000000000000000010"));
expectedNodes1.add(new SwhId("swh:1:rev:0000000000000000000000000000000000000009"));
- GraphTest.assertEqualsAnyOrder(expectedNodes1, neighbors1.getNeighbors());
+ GraphTest.assertEqualsAnyOrder(expectedNodes1, traversal1.neighborsEndpoint(src1));
SwhId src2 = new SwhId("swh:1:dir:0000000000000000000000000000000000000008");
- Neighbors neighbors2 = new Neighbors(graph, src2, "dir:cnt", "forward");
+ Traversal traversal2 = new Traversal(graph, "forward", "dir:cnt");
ArrayList<SwhId> expectedNodes2 = new ArrayList<>();
expectedNodes2.add(new SwhId("swh:1:cnt:0000000000000000000000000000000000000001"));
expectedNodes2.add(new SwhId("swh:1:cnt:0000000000000000000000000000000000000007"));
- GraphTest.assertEqualsAnyOrder(expectedNodes2, neighbors2.getNeighbors());
+ GraphTest.assertEqualsAnyOrder(expectedNodes2, traversal2.neighborsEndpoint(src2));
SwhId src3 = new SwhId("swh:1:cnt:0000000000000000000000000000000000000001");
- Neighbors neighbors3 = new Neighbors(graph, src3, "*", "backward");
+ Traversal traversal3 = new Traversal(graph, "backward", "*");
ArrayList<SwhId> expectedNodes3 = new ArrayList<>();
expectedNodes3.add(new SwhId("swh:1:dir:0000000000000000000000000000000000000008"));
expectedNodes3.add(new SwhId("swh:1:dir:0000000000000000000000000000000000000002"));
- GraphTest.assertEqualsAnyOrder(expectedNodes3, neighbors3.getNeighbors());
+ GraphTest.assertEqualsAnyOrder(expectedNodes3, traversal3.neighborsEndpoint(src3));
SwhId src4 = new SwhId("swh:1:rev:0000000000000000000000000000000000000009");
- Neighbors neighbors4 = new Neighbors(graph, src4, "rev:snp,rev:rel", "backward");
+ Traversal traversal4 = new Traversal(graph, "backward", "rev:snp,rev:rel");
ArrayList<SwhId> expectedNodes4 = new ArrayList<>();
expectedNodes4.add(new SwhId("swh:1:snp:0000000000000000000000000000000000000020"));
expectedNodes4.add(new SwhId("swh:1:rel:0000000000000000000000000000000000000010"));
- GraphTest.assertEqualsAnyOrder(expectedNodes4, neighbors4.getNeighbors());
+ GraphTest.assertEqualsAnyOrder(expectedNodes4, traversal4.neighborsEndpoint(src4));
}
@Test
@@ -103,19 +103,19 @@
Graph graph = getGraph();
SwhId src1 = new SwhId("swh:1:dir:0000000000000000000000000000000000000008");
- Neighbors neighbors1 = new Neighbors(graph, src1, "*", "forward");
+ Traversal traversal1 = new Traversal(graph, "forward", "*");
ArrayList<SwhId> expectedNodes1 = new ArrayList<>();
expectedNodes1.add(new SwhId("swh:1:dir:0000000000000000000000000000000000000006"));
expectedNodes1.add(new SwhId("swh:1:cnt:0000000000000000000000000000000000000001"));
expectedNodes1.add(new SwhId("swh:1:cnt:0000000000000000000000000000000000000007"));
- GraphTest.assertEqualsAnyOrder(expectedNodes1, neighbors1.getNeighbors());
+ GraphTest.assertEqualsAnyOrder(expectedNodes1, traversal1.neighborsEndpoint(src1));
SwhId src2 = new SwhId("swh:1:rev:0000000000000000000000000000000000000009");
- Neighbors neighbors2 = new Neighbors(graph, src2, "*", "backward");
+ Traversal traversal2 = new Traversal(graph, "backward", "*");
ArrayList<SwhId> expectedNodes2 = new ArrayList<>();
expectedNodes2.add(new SwhId("swh:1:snp:0000000000000000000000000000000000000020"));
expectedNodes2.add(new SwhId("swh:1:rel:0000000000000000000000000000000000000010"));
expectedNodes2.add(new SwhId("swh:1:rev:0000000000000000000000000000000000000013"));
- GraphTest.assertEqualsAnyOrder(expectedNodes2, neighbors2.getNeighbors());
+ GraphTest.assertEqualsAnyOrder(expectedNodes2, traversal2.neighborsEndpoint(src2));
}
}
diff --git a/api/server/src/test/java/org/softwareheritage/graph/VisitTest.java b/api/server/src/test/java/org/softwareheritage/graph/VisitTest.java
--- a/api/server/src/test/java/org/softwareheritage/graph/VisitTest.java
+++ b/api/server/src/test/java/org/softwareheritage/graph/VisitTest.java
@@ -9,11 +9,9 @@
import org.softwareheritage.graph.GraphTest;
import org.softwareheritage.graph.SwhId;
import org.softwareheritage.graph.SwhPath;
-import org.softwareheritage.graph.algo.Visit;
+import org.softwareheritage.graph.algo.Traversal;
public class VisitTest extends GraphTest {
- Visit.OutputFmt outputFmt = Visit.OutputFmt.NODES_AND_PATHS;
-
private void assertSameNodesFromPaths(ArrayList<SwhPath> paths, LinkedHashSet<SwhId> nodes) {
LinkedHashSet<SwhId> expectedNodes = new LinkedHashSet<SwhId>();
for (SwhPath path : paths) {
@@ -28,9 +26,9 @@
public void forwardFromRoot() {
Graph graph = getGraph();
SwhId swhId = new SwhId("swh:1:snp:0000000000000000000000000000000000000020");
- Visit visit = new Visit(graph, swhId, "*", "forward", outputFmt);
- ArrayList<SwhPath> paths = visit.getPaths();
- LinkedHashSet<SwhId> nodes = visit.getNodes();
+ Traversal traversal = new Traversal(graph, "forward", "*");
+ ArrayList<SwhPath> paths = traversal.visitPathsEndpoint(swhId);
+ ArrayList<SwhId> nodes = traversal.visitNodesEndpoint(swhId);
ArrayList<SwhPath> expectedPaths = new ArrayList<SwhPath>();
expectedPaths.add(
@@ -123,9 +121,9 @@
public void forwardFromMiddle() {
Graph graph = getGraph();
SwhId swhId = new SwhId("swh:1:dir:0000000000000000000000000000000000000012");
- Visit visit = new Visit(graph, swhId, "*", "forward", outputFmt);
- ArrayList<SwhPath> paths = visit.getPaths();
- LinkedHashSet<SwhId> nodes = visit.getNodes();
+ Traversal traversal = new Traversal(graph, "forward", "*");
+ ArrayList<SwhPath> paths = traversal.visitPathsEndpoint(swhId);
+ ArrayList<SwhId> nodes = traversal.visitNodesEndpoint(swhId);
ArrayList<SwhPath> expectedPaths = new ArrayList<SwhPath>();
expectedPaths.add(
@@ -168,9 +166,9 @@
public void forwardFromLeaf() {
Graph graph = getGraph();
SwhId swhId = new SwhId("swh:1:cnt:0000000000000000000000000000000000000004");
- Visit visit = new Visit(graph, swhId, "*", "forward", outputFmt);
- ArrayList<SwhPath> paths = visit.getPaths();
- LinkedHashSet<SwhId> nodes = visit.getNodes();
+ Traversal traversal = new Traversal(graph, "forward", "*");
+ ArrayList<SwhPath> paths = traversal.visitPathsEndpoint(swhId);
+ ArrayList<SwhId> nodes = traversal.visitNodesEndpoint(swhId);
ArrayList<SwhPath> expectedPaths = new ArrayList<SwhPath>();
expectedPaths.add(
@@ -186,9 +184,9 @@
public void backwardFromRoot() {
Graph graph = getGraph();
SwhId swhId = new SwhId("swh:1:snp:0000000000000000000000000000000000000020");
- Visit visit = new Visit(graph, swhId, "*", "backward", outputFmt);
- ArrayList<SwhPath> paths = visit.getPaths();
- LinkedHashSet<SwhId> nodes = visit.getNodes();
+ Traversal traversal = new Traversal(graph, "backward", "*");
+ ArrayList<SwhPath> paths = traversal.visitPathsEndpoint(swhId);
+ ArrayList<SwhId> nodes = traversal.visitNodesEndpoint(swhId);
ArrayList<SwhPath> expectedPaths = new ArrayList<SwhPath>();
expectedPaths.add(
@@ -204,9 +202,9 @@
public void backwardFromMiddle() {
Graph graph = getGraph();
SwhId swhId = new SwhId("swh:1:dir:0000000000000000000000000000000000000012");
- Visit visit = new Visit(graph, swhId, "*", "backward", outputFmt);
- ArrayList<SwhPath> paths = visit.getPaths();
- LinkedHashSet<SwhId> nodes = visit.getNodes();
+ Traversal traversal = new Traversal(graph, "backward", "*");
+ ArrayList<SwhPath> paths = traversal.visitPathsEndpoint(swhId);
+ ArrayList<SwhId> nodes = traversal.visitNodesEndpoint(swhId);
ArrayList<SwhPath> expectedPaths = new ArrayList<SwhPath>();
expectedPaths.add(
@@ -225,9 +223,9 @@
public void backwardFromLeaf() {
Graph graph = getGraph();
SwhId swhId = new SwhId("swh:1:cnt:0000000000000000000000000000000000000004");
- Visit visit = new Visit(graph, swhId, "*", "backward", outputFmt);
- ArrayList<SwhPath> paths = visit.getPaths();
- LinkedHashSet<SwhId> nodes = visit.getNodes();
+ Traversal traversal = new Traversal(graph, "backward", "*");
+ ArrayList<SwhPath> paths = traversal.visitPathsEndpoint(swhId);
+ ArrayList<SwhId> nodes = traversal.visitNodesEndpoint(swhId);
ArrayList<SwhPath> expectedPaths = new ArrayList<SwhPath>();
expectedPaths.add(
@@ -276,9 +274,9 @@
public void forwardSnpToRev() {
Graph graph = getGraph();
SwhId swhId = new SwhId("swh:1:snp:0000000000000000000000000000000000000020");
- Visit visit = new Visit(graph, swhId, "snp:rev", "forward", outputFmt);
- ArrayList<SwhPath> paths = visit.getPaths();
- LinkedHashSet<SwhId> nodes = visit.getNodes();
+ Traversal traversal = new Traversal(graph, "forward", "snp:rev");
+ ArrayList<SwhPath> paths = traversal.visitPathsEndpoint(swhId);
+ ArrayList<SwhId> nodes = traversal.visitNodesEndpoint(swhId);
ArrayList<SwhPath> expectedPaths = new ArrayList<SwhPath>();
expectedPaths.add(
@@ -295,9 +293,9 @@
public void forwardRelToRevRevToRev() {
Graph graph = getGraph();
SwhId swhId = new SwhId("swh:1:rel:0000000000000000000000000000000000000010");
- Visit visit = new Visit(graph, swhId, "rel:rev,rev:rev", "forward", outputFmt);
- ArrayList<SwhPath> paths = visit.getPaths();
- LinkedHashSet<SwhId> nodes = visit.getNodes();
+ Traversal traversal = new Traversal(graph, "forward", "rel:rev,rev:rev");
+ ArrayList<SwhPath> paths = traversal.visitPathsEndpoint(swhId);
+ ArrayList<SwhId> nodes = traversal.visitNodesEndpoint(swhId);
ArrayList<SwhPath> expectedPaths = new ArrayList<SwhPath>();
expectedPaths.add(
@@ -315,9 +313,9 @@
public void forwardRevToAllDirToAll() {
Graph graph = getGraph();
SwhId swhId = new SwhId("swh:1:rev:0000000000000000000000000000000000000013");
- Visit visit = new Visit(graph, swhId, "rev:*,dir:*", "forward", outputFmt);
- ArrayList<SwhPath> paths = visit.getPaths();
- LinkedHashSet<SwhId> nodes = visit.getNodes();
+ Traversal traversal = new Traversal(graph, "forward", "rev:*,dir:*");
+ ArrayList<SwhPath> paths = traversal.visitPathsEndpoint(swhId);
+ ArrayList<SwhId> nodes = traversal.visitNodesEndpoint(swhId);
ArrayList<SwhPath> expectedPaths = new ArrayList<SwhPath>();
expectedPaths.add(
@@ -403,9 +401,9 @@
public void forwardSnpToAllRevToAll() {
Graph graph = getGraph();
SwhId swhId = new SwhId("swh:1:snp:0000000000000000000000000000000000000020");
- Visit visit = new Visit(graph, swhId, "snp:*,rev:*", "forward", outputFmt);
- ArrayList<SwhPath> paths = visit.getPaths();
- LinkedHashSet<SwhId> nodes = visit.getNodes();
+ Traversal traversal = new Traversal(graph, "forward", "snp:*,rev:*");
+ ArrayList<SwhPath> paths = traversal.visitPathsEndpoint(swhId);
+ ArrayList<SwhId> nodes = traversal.visitNodesEndpoint(swhId);
ArrayList<SwhPath> expectedPaths = new ArrayList<SwhPath>();
expectedPaths.add(
@@ -435,9 +433,9 @@
public void forwardNoEdges() {
Graph graph = getGraph();
SwhId swhId = new SwhId("swh:1:snp:0000000000000000000000000000000000000020");
- Visit visit = new Visit(graph, swhId, "", "forward", outputFmt);
- ArrayList<SwhPath> paths = visit.getPaths();
- LinkedHashSet<SwhId> nodes = visit.getNodes();
+ Traversal traversal = new Traversal(graph, "forward", "");
+ ArrayList<SwhPath> paths = traversal.visitPathsEndpoint(swhId);
+ ArrayList<SwhId> nodes = traversal.visitNodesEndpoint(swhId);
ArrayList<SwhPath> expectedPaths = new ArrayList<SwhPath>();
expectedPaths.add(
@@ -453,9 +451,9 @@
public void backwardRevToRevRevToRel() {
Graph graph = getGraph();
SwhId swhId = new SwhId("swh:1:rev:0000000000000000000000000000000000000003");
- Visit visit = new Visit(graph, swhId, "rev:rev,rev:rel", "backward", outputFmt);
- ArrayList<SwhPath> paths = visit.getPaths();
- LinkedHashSet<SwhId> nodes = visit.getNodes();
+ Traversal traversal = new Traversal(graph, "backward", "rev:rev,rev:rel");
+ ArrayList<SwhPath> paths = traversal.visitPathsEndpoint(swhId);
+ ArrayList<SwhId> nodes = traversal.visitNodesEndpoint(swhId);
ArrayList<SwhPath> expectedPaths = new ArrayList<SwhPath>();
expectedPaths.add(
@@ -481,10 +479,10 @@
public void forwardFromRootNodesOnly() {
Graph graph = getGraph();
SwhId swhId = new SwhId("swh:1:snp:0000000000000000000000000000000000000020");
- Visit visit = new Visit(graph, swhId, "*", "forward", Visit.OutputFmt.ONLY_NODES);
- LinkedHashSet<SwhId> nodes = visit.getNodes();
+ Traversal traversal = new Traversal(graph, "forward", "*");
+ ArrayList<SwhId> nodes = traversal.visitNodesEndpoint(swhId);
- LinkedHashSet<SwhId> expectedNodes = new LinkedHashSet<SwhId>();
+ ArrayList<SwhId> expectedNodes = new ArrayList<SwhId>();
expectedNodes.add(new SwhId("swh:1:snp:0000000000000000000000000000000000000020"));
expectedNodes.add(new SwhId("swh:1:rel:0000000000000000000000000000000000000010"));
expectedNodes.add(new SwhId("swh:1:rev:0000000000000000000000000000000000000009"));
@@ -504,10 +502,10 @@
public void backwardRevToAllNodesOnly() {
Graph graph = getGraph();
SwhId swhId = new SwhId("swh:1:rev:0000000000000000000000000000000000000003");
- Visit visit = new Visit(graph, swhId, "rev:*", "backward", Visit.OutputFmt.ONLY_NODES);
- LinkedHashSet<SwhId> nodes = visit.getNodes();
+ Traversal traversal = new Traversal(graph, "backward", "rev:*");
+ ArrayList<SwhId> nodes = traversal.visitNodesEndpoint(swhId);
- LinkedHashSet<SwhId> expectedNodes = new LinkedHashSet<SwhId>();
+ ArrayList<SwhId> expectedNodes = new ArrayList<SwhId>();
expectedNodes.add(new SwhId("swh:1:rev:0000000000000000000000000000000000000003"));
expectedNodes.add(new SwhId("swh:1:rev:0000000000000000000000000000000000000009"));
expectedNodes.add(new SwhId("swh:1:snp:0000000000000000000000000000000000000020"));
diff --git a/api/server/src/test/java/org/softwareheritage/graph/WalkTest.java b/api/server/src/test/java/org/softwareheritage/graph/WalkTest.java
--- a/api/server/src/test/java/org/softwareheritage/graph/WalkTest.java
+++ b/api/server/src/test/java/org/softwareheritage/graph/WalkTest.java
@@ -11,19 +11,17 @@
import org.softwareheritage.graph.GraphTest;
import org.softwareheritage.graph.SwhId;
import org.softwareheritage.graph.SwhPath;
-import org.softwareheritage.graph.algo.Walk;
+import org.softwareheritage.graph.algo.Traversal;
public class WalkTest extends GraphTest {
@Test
public void forwardRootToLeaf() {
Graph graph = getGraph();
+ Traversal traversal = new Traversal(graph, "forward", "*");
SwhId src = new SwhId("swh:1:snp:0000000000000000000000000000000000000020");
String dstFmt = "swh:1:cnt:0000000000000000000000000000000000000005";
- Walk dfsWalk = new Walk(graph, src, dstFmt, "*", "forward", "dfs");
- Walk bfsWalk = new Walk(graph, src, dstFmt, "*", "forward", "bfs");
-
- SwhPath dfsPath = dfsWalk.getPath();
+ SwhPath dfsPath = traversal.walkEndpoint(src, dstFmt, "dfs");
SwhPath dfsExpectedPath =
new SwhPath(
"swh:1:snp:0000000000000000000000000000000000000020",
@@ -35,40 +33,36 @@
Assert.assertEquals(dfsExpectedPath, dfsPath);
// Most of DFS and BFS traversal outputs are the same because it is a small test graph
- SwhPath bfsPath = bfsWalk.getPath();
+ SwhPath bfsPath = traversal.walkEndpoint(src, dstFmt, "bfs");
Assert.assertEquals(dfsExpectedPath, bfsPath);
}
@Test
public void forwardLeafToLeaf() {
Graph graph = getGraph();
+ Traversal traversal = new Traversal(graph, "forward", "*");
SwhId src = new SwhId("swh:1:cnt:0000000000000000000000000000000000000007");
String dstFmt = "cnt";
- Walk dfsWalk = new Walk(graph, src, dstFmt, "*", "forward", "dfs");
- Walk bfsWalk = new Walk(graph, src, dstFmt, "*", "forward", "bfs");
-
- SwhPath dfsPath = dfsWalk.getPath();
+ SwhPath dfsPath = traversal.walkEndpoint(src, dstFmt, "dfs");
SwhPath dfsExpectedPath =
new SwhPath(
"swh:1:cnt:0000000000000000000000000000000000000007"
);
Assert.assertEquals(dfsExpectedPath, dfsPath);
- SwhPath bfsPath = bfsWalk.getPath();
+ SwhPath bfsPath = traversal.walkEndpoint(src, dstFmt, "bfs");
Assert.assertEquals(dfsExpectedPath, bfsPath);
}
@Test
public void forwardRevToRev() {
Graph graph = getGraph();
+ Traversal traversal = new Traversal(graph, "forward", "rev:rev");
SwhId src = new SwhId("swh:1:rev:0000000000000000000000000000000000000018");
String dstFmt = "swh:1:rev:0000000000000000000000000000000000000003";
- Walk dfsWalk = new Walk(graph, src, dstFmt, "rev:rev", "forward", "dfs");
- Walk bfsWalk = new Walk(graph, src, dstFmt, "rev:rev", "forward", "bfs");
-
- SwhPath dfsPath = dfsWalk.getPath();
+ SwhPath dfsPath = traversal.walkEndpoint(src, dstFmt, "dfs");
SwhPath dfsExpectedPath =
new SwhPath(
"swh:1:rev:0000000000000000000000000000000000000018",
@@ -78,20 +72,18 @@
);
Assert.assertEquals(dfsExpectedPath, dfsPath);
- SwhPath bfsPath = bfsWalk.getPath();
+ SwhPath bfsPath = traversal.walkEndpoint(src, dstFmt, "bfs");
Assert.assertEquals(dfsExpectedPath, bfsPath);
}
@Test
public void backwardRevToRev() {
Graph graph = getGraph();
+ Traversal traversal = new Traversal(graph, "backward", "rev:rev");
SwhId src = new SwhId("swh:1:rev:0000000000000000000000000000000000000003");
String dstFmt = "swh:1:rev:0000000000000000000000000000000000000018";
- Walk dfsWalk = new Walk(graph, src, dstFmt, "rev:rev", "backward", "dfs");
- Walk bfsWalk = new Walk(graph, src, dstFmt, "rev:rev", "backward", "bfs");
-
- SwhPath dfsPath = dfsWalk.getPath();
+ SwhPath dfsPath = traversal.walkEndpoint(src, dstFmt, "dfs");
SwhPath dfsExpectedPath =
new SwhPath(
"swh:1:rev:0000000000000000000000000000000000000003",
@@ -101,20 +93,18 @@
);
Assert.assertEquals(dfsExpectedPath, dfsPath);
- SwhPath bfsPath = bfsWalk.getPath();
+ SwhPath bfsPath = traversal.walkEndpoint(src, dstFmt, "bfs");
Assert.assertEquals(dfsExpectedPath, bfsPath);
}
@Test
public void backwardCntToFirstSnp() {
Graph graph = getGraph();
+ Traversal traversal = new Traversal(graph, "backward", "*");
SwhId src = new SwhId("swh:1:cnt:0000000000000000000000000000000000000001");
String dstFmt = "snp";
- Walk dfsWalk = new Walk(graph, src, dstFmt, "*", "backward", "dfs");
- Walk bfsWalk = new Walk(graph, src, dstFmt, "*", "backward", "bfs");
-
- SwhPath dfsPath = dfsWalk.getPath();
+ SwhPath dfsPath = traversal.walkEndpoint(src, dstFmt, "dfs");
SwhPath dfsExpectedPath =
new SwhPath(
"swh:1:cnt:0000000000000000000000000000000000000001",
@@ -125,7 +115,7 @@
);
Assert.assertEquals(dfsExpectedPath, dfsPath);
- SwhPath bfsPath = bfsWalk.getPath();
+ SwhPath bfsPath = traversal.walkEndpoint(src, dstFmt, "bfs");
SwhPath bfsExpectedPath =
new SwhPath(
"swh:1:cnt:0000000000000000000000000000000000000001",
@@ -139,13 +129,11 @@
@Test
public void forwardRevToFirstCnt() {
Graph graph = getGraph();
+ Traversal traversal = new Traversal(graph, "forward", "rev:*,dir:*");
SwhId src = new SwhId("swh:1:rev:0000000000000000000000000000000000000013");
String dstFmt = "cnt";
- Walk dfsWalk = new Walk(graph, src, dstFmt, "rev:*,dir:*", "forward", "dfs");
- Walk bfsWalk = new Walk(graph, src, dstFmt, "rev:*,dir:*", "forward", "bfs");
-
- SwhPath dfsPath = dfsWalk.getPath();
+ SwhPath dfsPath = traversal.walkEndpoint(src, dstFmt, "dfs");
SwhPath dfsExpectedPath =
new SwhPath(
"swh:1:rev:0000000000000000000000000000000000000013",
@@ -154,20 +142,18 @@
);
Assert.assertEquals(dfsExpectedPath, dfsPath);
- SwhPath bfsPath = bfsWalk.getPath();
+ SwhPath bfsPath = traversal.walkEndpoint(src, dstFmt, "bfs");
Assert.assertEquals(dfsExpectedPath, bfsPath);
}
@Test
public void backwardDirToFirstRel() {
Graph graph = getGraph();
+ Traversal traversal = new Traversal(graph, "backward", "dir:dir,dir:rev,rev:*");
SwhId src = new SwhId("swh:1:dir:0000000000000000000000000000000000000016");
String dstFmt = "rel";
- Walk dfsWalk = new Walk(graph, src, dstFmt, "dir:dir,dir:rev,rev:*", "backward", "dfs");
- Walk bfsWalk = new Walk(graph, src, dstFmt, "dir:dir,dir:rev,rev:*", "backward", "bfs");
-
- SwhPath dfsPath = dfsWalk.getPath();
+ SwhPath dfsPath = traversal.walkEndpoint(src, dstFmt, "dfs");
SwhPath dfsExpectedPath =
new SwhPath(
"swh:1:dir:0000000000000000000000000000000000000016",
@@ -177,7 +163,7 @@
);
Assert.assertEquals(dfsExpectedPath, dfsPath);
- SwhPath bfsPath = bfsWalk.getPath();
+ SwhPath bfsPath = traversal.walkEndpoint(src, dstFmt, "bfs");
Assert.assertEquals(dfsExpectedPath, bfsPath);
}
}

File Metadata

Mime Type
text/plain
Expires
Tue, Dec 17, 6:24 AM (2 w, 4 d ago)
Storage Engine
blob
Storage Format
Raw Data
Storage Handle
3220026

Event Timeline