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

overflowdb.tinkerpop.optimizations.OdbGraphStep Maven / Gradle / Ivy

package overflowdb.tinkerpop.optimizations;

import org.apache.tinkerpop.gremlin.process.traversal.Compare;
import org.apache.tinkerpop.gremlin.process.traversal.P;
import org.apache.tinkerpop.gremlin.process.traversal.step.HasContainerHolder;
import org.apache.tinkerpop.gremlin.process.traversal.step.map.GraphStep;
import org.apache.tinkerpop.gremlin.process.traversal.step.util.HasContainer;
import org.apache.tinkerpop.gremlin.process.traversal.util.AndP;
import org.apache.tinkerpop.gremlin.structure.Edge;
import org.apache.tinkerpop.gremlin.structure.Element;
import org.apache.tinkerpop.gremlin.structure.T;
import org.apache.tinkerpop.gremlin.structure.Vertex;
import org.apache.tinkerpop.gremlin.structure.util.StringFactory;
import org.apache.tinkerpop.gremlin.util.iterator.IteratorUtils;
import overflowdb.Node;
import overflowdb.NodeRef;
import overflowdb.tinkerpop.NodeTp3;
import overflowdb.tinkerpop.OdbGraphTp3;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Optional;
import java.util.Set;

public final class OdbGraphStep extends GraphStep implements HasContainerHolder {

  private final List hasContainers = new ArrayList<>();

  public OdbGraphStep(final GraphStep originalGraphStep) {
    super(originalGraphStep.getTraversal(), originalGraphStep.getReturnClass(), originalGraphStep.isStartStep(), originalGraphStep.getIds());
    originalGraphStep.getLabels().forEach(this::addLabel);

    // we used to only setIteratorSupplier() if there were no ids OR the first id was instanceof Element,
    // but that allowed the filter in g.V(v).has('k','v') to be ignored.  this created problems for
    // PartitionStrategy which wants to prevent someone from passing "v" from one TraversalSource to
    // another TraversalSource using a different partition
    this.setIteratorSupplier(() -> (Iterator) (Vertex.class.isAssignableFrom(this.returnClass) ? this.vertices() : this.edges()));
  }

  private Iterator edges() {
    final OdbGraphTp3 graph = (OdbGraphTp3) this.getTraversal().getGraph().get();
    final Optional hasLabelContainer = findHasLabelStep();
    // ids are present, filter on them first
    if (null == this.ids)
      return Collections.emptyIterator();
    else if (this.ids.length > 0)
      return this.iteratorList(graph.edges(this.ids));
    else
      return this.iteratorList(graph.edges());
  }

  private Iterator vertices() {
    final OdbGraphTp3 graph = (OdbGraphTp3) this.getTraversal().getGraph().get();
    final HasContainer indexedContainer = getIndexKey();
    final Optional hasLabelContainer = findHasLabelStep();
    // ids are present, filter on them first
    if (null == this.ids)
      return Collections.emptyIterator();
    else if (this.ids.length > 0)
      return this.iteratorList(graph.vertices(this.ids));
    else if (hasLabelContainer.isPresent()) {
      P hasLabelPredicate = (P) hasLabelContainer.get().getPredicate();
      // unfortunately TP3 api doesn't seem to find out if it's the `Compare.eq` bipredicate, so we can optimise single-label lookups
      final Iterator nodes = graph.graph.nodes(hasLabelPredicate);
      return IteratorUtils.map(nodes, node -> NodeTp3.wrap((NodeRef) node));
    } else {
      if (indexedContainer == null) return this.iteratorList(graph.vertices());
      else {
        final Iterator nodes = IteratorUtils.filter(
            graph.graph.indexManager.lookup(indexedContainer.getKey(), indexedContainer.getPredicate().getValue()).iterator(),
            nodeRef -> HasContainer.testAll(NodeTp3.wrap(nodeRef), this.hasContainers));
        return IteratorUtils.map(nodes, node -> NodeTp3.wrap(node));
      }
    }
  }

  // only optimize if hasLabel is the _only_ hasContainer, since that's the simplest case
  // TODO implement other cases as well, e.g. for `g.V.hasLabel(lbl).has(k,v)`
  private Optional findHasLabelStep() {
    if (hasContainers.size() == 1) {
      if (T.label.getAccessor().equals(hasContainers.get(0).getKey())) {
        return Optional.of(hasContainers.get(0));
      }
    }
    return Optional.empty();
  }

  private HasContainer getIndexKey() {
    final OdbGraphTp3 graph = (OdbGraphTp3) this.getTraversal().getGraph().get();
    final Set indexedKeys = graph.graph.indexManager.getIndexedNodeProperties();

    final Iterator itty = IteratorUtils.filter(hasContainers.iterator(),
        c -> c.getPredicate().getBiPredicate() == Compare.eq && indexedKeys.contains(c.getKey()));
    return itty.hasNext() ? itty.next() : null;
  }

  @Override
  public String toString() {
    if (this.hasContainers.isEmpty())
      return super.toString();
    else
      return 0 == this.ids.length ?
          StringFactory.stepString(this, this.returnClass.getSimpleName().toLowerCase(), this.hasContainers) :
          StringFactory.stepString(this, this.returnClass.getSimpleName().toLowerCase(), Arrays.toString(this.ids), this.hasContainers);
  }

  private  Iterator iteratorList(final Iterator iterator) {
    final List list = new ArrayList<>();
    while (iterator.hasNext()) {
      final E e = iterator.next();
      if (HasContainer.testAll(e, this.hasContainers))
        list.add(e);
    }
    return list.iterator();
  }

  @Override
  public List getHasContainers() {
    return Collections.unmodifiableList(this.hasContainers);
  }

  @Override
  public void addHasContainer(final HasContainer hasContainer) {
    if (hasContainer.getPredicate() instanceof AndP) {
      for (final P predicate : ((AndP) hasContainer.getPredicate()).getPredicates()) {
        this.addHasContainer(new HasContainer(hasContainer.getKey(), predicate));
      }
    } else
      this.hasContainers.add(hasContainer);
  }

  @Override
  public int hashCode() {
    return super.hashCode() ^ this.hasContainers.hashCode();
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy