apoc.path.RelationshipSequenceExpander Maven / Gradle / Ivy
/*
* Copyright (c) "Neo4j"
* Neo4j Sweden AB [http://neo4j.com]
*
* This file is part of Neo4j.
*
* 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 apoc.path;
import apoc.util.collection.Iterables;
import apoc.util.collection.Iterators;
import apoc.util.collection.NestingResourceIterator;
import apoc.util.collection.ResourceClosingIterator;
import java.util.ArrayList;
import java.util.List;
import org.apache.commons.lang3.tuple.Pair;
import org.neo4j.graphdb.Direction;
import org.neo4j.graphdb.Node;
import org.neo4j.graphdb.Path;
import org.neo4j.graphdb.PathExpander;
import org.neo4j.graphdb.Relationship;
import org.neo4j.graphdb.RelationshipType;
import org.neo4j.graphdb.ResourceIterable;
import org.neo4j.graphdb.ResourceIterator;
import org.neo4j.graphdb.traversal.BranchState;
/**
* An expander for repeating sequences of relationships. The sequence provided should be a string consisting of
* relationship type/direction patterns (exactly the same as the `relationshipFilter`), separated by commas.
* Each comma-separated pattern represents the relationships that will be expanded with each step of expansion, which
* repeats indefinitely (unless otherwise stopped by `maxLevel`, `limit`, or terminator filtering from the other expander config options).
* The exception is if `beginSequenceAtStart` is false. This indicates that the sequence should not begin from the start node,
* but from one node distant. In this case, we may still need a restriction on the relationship used to reach the start node
* of the sequence, so when `beginSequenceAtStart` is false, then the first relationship step in the sequence given will not
* actually be used as part of the sequence, but will only be used once to reach the starting node of the sequence.
* The remaining relationship steps will be used as the repeating relationship sequence.
*/
public class RelationshipSequenceExpander implements PathExpander {
private final List>> relSequences = new ArrayList<>();
private List> initialRels = null;
public RelationshipSequenceExpander(String relSequenceString, boolean beginSequenceAtStart) {
int index = 0;
for (String sequenceStep : relSequenceString.split(",")) {
sequenceStep = sequenceStep.trim();
Iterable> relDirIterable =
RelationshipTypeAndDirections.parse(sequenceStep);
List> stepRels = new ArrayList<>();
for (Pair pair : relDirIterable) {
stepRels.add(pair);
}
if (!beginSequenceAtStart && index == 0) {
initialRels = stepRels;
} else {
relSequences.add(stepRels);
}
index++;
}
}
public RelationshipSequenceExpander(List relSequenceList, boolean beginSequenceAtStart) {
int index = 0;
for (String sequenceStep : relSequenceList) {
sequenceStep = sequenceStep.trim();
Iterable> relDirIterable =
RelationshipTypeAndDirections.parse(sequenceStep);
List> stepRels = new ArrayList<>();
for (Pair pair : relDirIterable) {
stepRels.add(pair);
}
if (!beginSequenceAtStart && index == 0) {
initialRels = stepRels;
} else {
relSequences.add(stepRels);
}
index++;
}
}
@Override
public ResourceIterable expand(Path path, BranchState state) {
final Node node = path.endNode();
final int depth = path.length();
final List> stepRels;
if (depth == 0 && initialRels != null) {
stepRels = initialRels;
} else {
stepRels = relSequences.get((initialRels == null ? depth : depth - 1) % relSequences.size());
}
return Iterables.asResourceIterable(Iterators.asList(new NestingResourceIterator<>(stepRels.iterator()) {
@Override
protected ResourceIterator createNestedIterator(Pair entry) {
RelationshipType type = entry.getLeft();
Direction dir = entry.getRight();
ResourceIterable relationships1;
if (type == null) {
relationships1 = (dir == Direction.BOTH) ? node.getRelationships() : node.getRelationships(dir);
} else {
relationships1 =
(dir == Direction.BOTH) ? node.getRelationships(type) : node.getRelationships(dir, type);
}
return ResourceClosingIterator.fromResourceIterable(relationships1);
}
}));
}
@Override
public PathExpander reverse() {
throw new RuntimeException("Not implemented");
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy