org.neo4j.graphdb.impl.traversal.BidirectionalTraversalBranchPath Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of neo4j-graphdb-api Show documentation
Show all versions of neo4j-graphdb-api Show documentation
Graph Database API for Neo4j.
/*
* Copyright (c) "Neo4j"
* Neo4j Sweden AB [https://neo4j.com]
*
* This file is part of Neo4j.
*
* Neo4j is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
package org.neo4j.graphdb.impl.traversal;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import org.neo4j.graphdb.Entity;
import org.neo4j.graphdb.Node;
import org.neo4j.graphdb.Path;
import org.neo4j.graphdb.Relationship;
import org.neo4j.graphdb.traversal.Paths;
import org.neo4j.graphdb.traversal.TraversalBranch;
class BidirectionalTraversalBranchPath implements Path {
private final TraversalBranch start;
private final TraversalBranch end;
private final Node endNode;
private final Relationship lastRelationship;
private Node cachedStartNode;
private List cachedRelationships;
BidirectionalTraversalBranchPath(TraversalBranch start, TraversalBranch end) {
this.start = start;
this.end = end;
// Most used properties: endNode and lastRelationship, so cache them right away (semi-expensive though).
Iterator endPathEntities = end.iterator();
this.endNode = (Node) endPathEntities.next();
this.lastRelationship =
endPathEntities.hasNext() ? (Relationship) endPathEntities.next() : start.lastRelationship();
}
@Override
public Node startNode() {
// Getting the start node is expensive in some Path implementations, so cache it
if (cachedStartNode == null) {
cachedStartNode = start.startNode();
}
return cachedStartNode;
}
@Override
public Node endNode() {
return this.endNode;
}
@Override
public Relationship lastRelationship() {
return this.lastRelationship;
}
@Override
public Iterable relationships() {
// Cache the relationships since we use them in hashCode/equals too.
if (cachedRelationships == null) {
cachedRelationships = gatherRelationships(start, end);
}
return cachedRelationships;
}
@Override
public Iterable reverseRelationships() {
return gatherRelationships(end, start);
}
private List gatherRelationships(TraversalBranch first, TraversalBranch then) {
// TODO Don't loop through them all up front
LinkedList relationships = new LinkedList<>();
TraversalBranch branch = first;
while (branch.length() > 0) {
relationships.addFirst(branch.lastRelationship());
branch = branch.parent();
}
// We can might as well cache start node since we're right now there anyway
if (cachedStartNode == null && first == start && branch.length() >= 0) {
cachedStartNode = branch.endNode();
}
branch = then;
while (branch.length() > 0) {
relationships.add(branch.lastRelationship());
branch = branch.parent();
}
if (cachedStartNode == null && then == start && branch.length() >= 0) {
cachedStartNode = branch.endNode();
}
return relationships;
}
@Override
public Iterable nodes() {
return gatherNodes(start, end);
}
@Override
public Iterable reverseNodes() {
return gatherNodes(end, start);
}
private Iterable gatherNodes(TraversalBranch first, TraversalBranch then) {
// TODO Don't loop through them all up front
LinkedList nodes = new LinkedList<>();
TraversalBranch branch = first;
while (branch.length() > 0) {
nodes.addFirst(branch.endNode());
branch = branch.parent();
}
if (cachedStartNode == null && first == start && branch.length() >= 0) {
cachedStartNode = branch.endNode();
}
nodes.addFirst(branch.endNode());
branch = then.parent();
if (branch != null) {
while (branch.length() > 0) {
nodes.add(branch.endNode());
branch = branch.parent();
}
if (branch.length() >= 0) {
nodes.add(branch.endNode());
}
}
if (cachedStartNode == null && then == start && branch != null && branch.length() >= 0) {
cachedStartNode = branch.endNode();
}
return nodes;
}
@Override
public int length() {
return start.length() + end.length();
}
@Override
public Iterator iterator() {
// TODO Don't loop through them all up front
LinkedList entities = new LinkedList<>();
TraversalBranch branch = start;
while (branch.length() > 0) {
entities.addFirst(branch.endNode());
entities.addFirst(branch.lastRelationship());
branch = branch.parent();
}
entities.addFirst(branch.endNode());
if (cachedStartNode == null) {
cachedStartNode = branch.endNode();
}
if (end.length() > 0) {
entities.add(end.lastRelationship());
branch = end.parent();
while (branch.length() > 0) {
entities.add(branch.endNode());
entities.add(branch.lastRelationship());
branch = branch.parent();
}
entities.add(branch.endNode());
}
return entities.iterator();
}
@Override
public int hashCode() {
return relationships().hashCode();
}
@Override
public boolean equals(Object obj) {
if (obj == this) {
return true;
}
if (!(obj instanceof Path other)) {
return false;
}
return relationships().equals(other.relationships())
&& other.startNode().equals(cachedStartNode);
}
@Override
public String toString() {
return Paths.defaultPathToString(this);
}
}