blob: 35918042af25fb4e3112c5194c094aab54d485a8 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 org.apache.tinkerpop.gremlin.neo4j.structure;
import org.apache.tinkerpop.gremlin.FeatureRequirement;
import org.apache.tinkerpop.gremlin.neo4j.AbstractNeo4jGremlinTest;
import org.apache.tinkerpop.gremlin.neo4j.structure.trait.MultiMetaNeo4jTrait;
import org.apache.tinkerpop.gremlin.process.traversal.P;
import org.apache.tinkerpop.gremlin.process.traversal.Traversal;
import org.apache.tinkerpop.gremlin.structure.Graph;
import org.apache.tinkerpop.gremlin.structure.T;
import org.apache.tinkerpop.gremlin.structure.Vertex;
import org.apache.tinkerpop.gremlin.structure.VertexProperty;
import org.apache.tinkerpop.gremlin.util.TimeUtil;
import org.apache.tinkerpop.gremlin.util.iterator.IteratorUtils;
import org.junit.Test;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertNotEquals;
import static org.junit.Assert.assertTrue;
/**
* @author Marko A. Rodriguez (http://markorodriguez.com)
*/
public class NativeNeo4jIndexCheck extends AbstractNeo4jGremlinTest {
private static final Logger logger = LoggerFactory.getLogger(NativeNeo4jIndexCheck.class);
@Test
public void shouldHaveFasterRuntimeWithLabelKeyValueIndex() throws Exception {
final Neo4jGraph neo4j = (Neo4jGraph) this.graph;
int maxVertices = 10000;
for (int i = 0; i < maxVertices; i++) {
if (i % 2 == 0)
this.graph.addVertex(T.label, "something", "myId", i);
else
this.graph.addVertex(T.label, "nothing", "myId", i);
}
this.graph.tx().commit();
// traversal
final Runnable traversal = () -> {
final Traversal<Vertex, Vertex> t = g.V().hasLabel("something").has("myId", 2000);
final Vertex vertex = t.tryNext().get();
assertFalse(t.hasNext());
assertEquals(1, IteratorUtils.count(vertex.properties("myId")));
assertEquals("something", vertex.label());
};
// no index
this.graph.tx().readWrite();
assertFalse(this.getBaseGraph().hasSchemaIndex("something", "myId"));
TimeUtil.clock(20, traversal);
final double noIndexTime = TimeUtil.clock(20, traversal);
// index time
neo4j.cypher("CREATE INDEX ON :something(myId)").iterate();
this.graph.tx().commit();
this.graph.tx().readWrite();
Thread.sleep(5000); // wait for indices to be built
assertTrue(this.getBaseGraph().hasSchemaIndex("something", "myId"));
TimeUtil.clock(20, traversal);
final double indexTime = TimeUtil.clock(20, traversal);
logger.info("Query time (no-index vs. index): {}", noIndexTime + " vs. {}", indexTime);
assertTrue((noIndexTime / 10) > indexTime); // should be at least 10x faster
}
@Test
@FeatureRequirement(featureClass = Graph.Features.VertexFeatures.class, feature = Graph.Features.VertexFeatures.FEATURE_MULTI_PROPERTIES)
@FeatureRequirement(featureClass = Graph.Features.VertexFeatures.class, feature = Graph.Features.VertexFeatures.FEATURE_META_PROPERTIES)
public void shouldHaveFasterRuntimeWithLabelKeyValueIndexOnMultiProperties() throws Exception {
final Neo4jGraph neo4j = (Neo4jGraph) this.graph;
int maxVertices = 10000;
for (int i = 0; i < maxVertices; i++) {
if (i % 2 == 0)
this.graph.addVertex(T.label, "something", "myId", i, "myId", i + maxVertices + 1);
else
this.graph.addVertex(T.label, "nothing", "myId", i, "myId", i + maxVertices + 1);
}
this.graph.tx().commit();
// traversal
final Runnable traversal = () -> {
final Traversal<Vertex, Vertex> t = g.V().hasLabel("something").has("myId", 2000 + maxVertices + 1);
final Vertex vertex = t.tryNext().get();
assertFalse(t.hasNext());
assertEquals(2, IteratorUtils.count(vertex.properties("myId")));
assertEquals("something", vertex.label());
};
// no index
this.graph.tx().readWrite();
assertFalse(this.getBaseGraph().hasSchemaIndex("something", "myId"));
TimeUtil.clock(20, traversal);
final double noIndexTime = TimeUtil.clock(20, traversal);
// index time
neo4j.cypher("CREATE INDEX ON :something(myId)").iterate();
neo4j.cypher("CREATE INDEX ON :vertexProperty(myId)").iterate();
this.graph.tx().commit();
this.graph.tx().readWrite();
Thread.sleep(5000); // wait for indices to be built
assertTrue(this.getBaseGraph().hasSchemaIndex("something", "myId"));
TimeUtil.clock(20, traversal);
final double indexTime = TimeUtil.clock(20, traversal);
logger.info("Query time (no-index vs. index): {}", noIndexTime + " vs. {}", indexTime);
assertTrue((noIndexTime / 10) > indexTime); // should be at least 10x faster
}
@Test
public void shouldReturnResultsLabeledIndexOnVertexWithHasHas() {
this.graph.tx().readWrite();
this.getBaseGraph().execute("CREATE INDEX ON :Person(name)", null);
this.graph.tx().commit();
this.graph.addVertex(T.label, "Person", "name", "marko");
this.graph.addVertex(T.label, "Person", "name", "marko");
this.graph.tx().commit();
assertEquals(2, this.g.V().has(T.label, "Person").has("name", "marko").count().next(), 0);
assertEquals(2, this.g.V().has("name", "marko").count().next(), 0);
}
@Test
public void shouldEnsureColonedKeyIsTreatedAsNormalKey() {
this.graph.tx().readWrite();
this.getBaseGraph().execute("CREATE INDEX ON :Person(name)", null);
this.graph.tx().commit();
this.graph.addVertex(T.label, "Person", "name", "marko");
this.graph.addVertex(T.label, "Person", "name", "marko");
this.graph.tx().commit();
assertEquals(2, this.g.V().has(T.label, "Person").has("name", "marko").count().next(), 0);
assertEquals(0, this.g.V().has("Person:name", "marko").count().next(), 0);
}
@Test
public void shouldReturnResultsUsingLabeledIndexOnVertexWithHasHasHas() {
this.graph.tx().readWrite();
this.getBaseGraph().execute("CREATE INDEX ON :Person(name)", null);
this.graph.tx().commit();
this.graph.addVertex(T.label, "Person", "name", "marko", "color", "blue");
this.graph.addVertex(T.label, "Person", "name", "marko", "color", "green");
this.graph.tx().commit();
assertEquals(1, this.g.V().has(T.label, "Person").has("name", "marko").has("color", "blue").count().next(), 0);
assertEquals(2, this.g.V().has("name", "marko").count().next(), 0);
}
@Test
public void shouldReturnResultsOnVertexWithHasHasHasNoIndex() {
this.graph.tx().commit();
this.graph.addVertex(T.label, "Person", "name", "marko", "color", "blue");
this.graph.addVertex(T.label, "Person", "name", "marko", "color", "green");
this.graph.tx().commit();
assertEquals(1, this.g.V().has(T.label, "Person").has("name", "marko").has("color", "blue").count().next(), 0);
assertEquals(2, this.g.V().has("name", "marko").count().next(), 0);
}
@Test
public void shouldReturnResultsUsingLabeledIndexOnVertexWithColonFails() {
this.graph.tx().readWrite();
this.getBaseGraph().execute("CREATE INDEX ON :Person(name)", null);
this.graph.tx().commit();
this.graph.addVertex(T.label, "Person", "name", "marko");
this.graph.addVertex(T.label, "Person", "name", "marko");
this.graph.tx().commit();
assertNotEquals(2l, this.g.V().has("Person:name", "marko").count().next().longValue());
assertEquals(2, this.g.V().has("name", "marko").count().next(), 0);
}
@Test
public void shouldEnforceUniqueConstraint() {
this.graph.tx().readWrite();
this.getBaseGraph().execute("CREATE CONSTRAINT ON (p:Person) assert p.name is unique", null);
this.graph.tx().commit();
this.graph.tx().commit();
this.graph.addVertex(T.label, "Person", "name", "marko");
this.graph.tx().commit();
assertEquals("marko", g.V().has(T.label, "Person").has("name", "marko").next().value("name"));
}
@Test
public void shouldEnforceMultipleUniqueConstraint() {
this.graph.tx().readWrite();
this.getBaseGraph().execute("CREATE CONSTRAINT ON (p:Person) assert p.name is unique", null);
this.getBaseGraph().execute("CREATE CONSTRAINT ON (p:Person) assert p.surname is unique", null);
this.graph.tx().commit();
this.graph.addVertex(T.label, "Person", "name", "marko");
this.graph.addVertex(T.label, "Person", "surname", "aaaa");
this.graph.tx().commit();
boolean failSurname = false;
try {
this.graph.addVertex(T.label, "Person", "surname", "aaaa");
} catch (RuntimeException e) {
if (isConstraintViolation(e)) failSurname = true;
}
assertTrue(failSurname);
boolean failName = false;
try {
this.graph.addVertex(T.label, "Person", "name", "marko");
} catch (RuntimeException e) {
if (isConstraintViolation(e)) failName = true;
}
assertTrue(failName);
this.graph.tx().commit();
}
private boolean isConstraintViolation(RuntimeException e) {
return e.getClass().getSimpleName().equals("ConstraintViolationException");
}
@Test
public void shouldDropMultipleUniqueConstraint() {
this.graph.tx().readWrite();
this.getBaseGraph().execute("CREATE CONSTRAINT ON (p:Person) assert p.name is unique", null);
this.getBaseGraph().execute("CREATE CONSTRAINT ON (p:Person) assert p.surname is unique", null);
this.graph.tx().commit();
this.graph.addVertex(T.label, "Person", "name", "marko");
this.graph.addVertex(T.label, "Person", "surname", "aaaa");
this.graph.tx().commit();
boolean failSurname = false;
try {
this.graph.addVertex(T.label, "Person", "surname", "aaaa");
} catch (RuntimeException e) {
if (isConstraintViolation(e)) failSurname = true;
}
assertTrue(failSurname);
boolean failName = false;
try {
this.graph.addVertex(T.label, "Person", "name", "marko");
} catch (RuntimeException e) {
if (isConstraintViolation(e)) failName = true;
}
assertTrue(failName);
this.graph.tx().commit();
this.graph.tx().readWrite();
this.getBaseGraph().execute("DROP CONSTRAINT ON (p:Person) assert p.name is unique", null);
this.getBaseGraph().execute("DROP CONSTRAINT ON (p:Person) assert p.surname is unique", null);
this.graph.tx().commit();
assertEquals(1, this.g.V().has(T.label, "Person").has("name", "marko").count().next(), 0);
assertEquals(1, this.g.V().has(T.label, "Person").has("surname", "aaaa").count().next(), 0);
this.graph.addVertex(T.label, "Person", "surname", "aaaa");
this.graph.addVertex(T.label, "Person", "name", "marko");
this.graph.tx().commit();
assertEquals(2, this.g.V().has(T.label, "Person").has("name", "marko").count().next(), 0);
assertEquals(2, this.g.V().has(T.label, "Person").has("surname", "aaaa").count().next(), 0);
}
@Test(expected = RuntimeException.class)
public void shouldFailUniqueConstraint() {
this.graph.tx().readWrite();
this.getBaseGraph().execute("CREATE CONSTRAINT ON (p:Person) assert p.name is unique", null);
this.graph.tx().commit();
this.graph.addVertex(T.label, "Person", "name", "marko");
this.graph.tx().commit();
assertEquals("marko", g.V().has(T.label, "Person").has("name", "marko").next().value("name"));
this.graph.addVertex(T.label, "Person", "name", "marko");
}
@Test
public void shouldDoLabelAndIndexSearch() {
graph.tx().readWrite();
this.getBaseGraph().execute("CREATE INDEX ON :Person(name)", null);
this.graph.tx().commit();
this.graph.addVertex(T.label, "Person", "name", "marko");
this.graph.addVertex(T.label, "Person", "name", "john");
this.graph.addVertex(T.label, "Person", "name", "pete");
this.graph.tx().commit();
assertEquals(1, this.g.V().has("Person", "name", "marko").count().next(), 0);
assertEquals(3, this.g.V().has(T.label, "Person").count().next(), 0);
assertEquals(1, this.g.V().has(T.label, "Person").has("name", "marko").count().next(), 0);
}
@Test
@FeatureRequirement(featureClass = Graph.Features.VertexFeatures.class, feature = Graph.Features.VertexFeatures.FEATURE_MULTI_PROPERTIES)
public void shouldSupportVertexPropertyToVertexMappingOnIndexCalls() {
graph.tx().readWrite();
this.getBaseGraph().execute("CREATE INDEX ON :person(name)", null);
// this.graph.getBaseGraph().execute("CREATE INDEX ON :name(" + T.value.getAccessor() + ")", null);
this.graph.tx().commit();
final Vertex a = graph.addVertex(T.label, "person", "name", "marko", "age", 34);
a.property(VertexProperty.Cardinality.list, "name", "okram");
a.property(VertexProperty.Cardinality.list, "name", "marko a. rodriguez");
final Vertex b = graph.addVertex(T.label, "person", "name", "stephen");
final Vertex c = graph.addVertex("name", "matthias", "name", "mbroecheler");
tryCommit(graph, graph -> {
assertEquals(a.id(), g.V().has("person", "name", "okram").id().next());
assertEquals(1, g.V().has("person", "name", "okram").count().next().intValue());
assertEquals(34, ((Neo4jVertex) g.V().has("person", "name", "okram").next()).getBaseVertex().getProperty("age"));
assertEquals(MultiMetaNeo4jTrait.VERTEX_PROPERTY_TOKEN, ((Neo4jVertex) g.V().has("person", "name", "okram").next()).getBaseVertex().getProperty("name"));
///
assertEquals(b.id(), g.V().has("person", "name", "stephen").id().next());
assertEquals(1, g.V().has("person", "name", "stephen").count().next().intValue());
assertEquals("stephen", ((Neo4jVertex) g.V().has("person", "name", "stephen").next()).getBaseVertex().getProperty("name"));
///
assertEquals(c.id(), g.V().has("name", "matthias").id().next());
assertEquals(c.id(), g.V().has("name", "mbroecheler").id().next());
assertEquals(1, g.V().has("name", "matthias").count().next().intValue());
assertEquals(1, g.V().has("name", "mbroecheler").count().next().intValue());
assertEquals(0, g.V().has("person", "name", "matthias").count().next().intValue());
assertEquals(0, g.V().has("person", "name", "mbroecheler").count().next().intValue());
});
final Vertex d = graph.addVertex(T.label, "person", "name", "kuppitz");
tryCommit(graph, graph -> {
assertEquals(d.id(), g.V().has("person", "name", "kuppitz").id().next());
assertEquals("kuppitz", ((Neo4jVertex) g.V().has("person", "name", "kuppitz").next()).getBaseVertex().getProperty("name"));
});
d.property(VertexProperty.Cardinality.list, "name", "daniel", "acl", "private");
tryCommit(graph, graph -> {
assertEquals(d.id(), g.V().has("person", "name", P.within("daniel", "kuppitz")).id().next());
assertEquals(d.id(), g.V().has("person", "name", "kuppitz").id().next());
assertEquals(d.id(), g.V().has("person", "name", "daniel").id().next());
assertEquals(MultiMetaNeo4jTrait.VERTEX_PROPERTY_TOKEN, ((Neo4jVertex) g.V().has("person", "name", "kuppitz").next()).getBaseVertex().getProperty("name"));
});
d.property(VertexProperty.Cardinality.list, "name", "marko", "acl", "private");
tryCommit(graph, graph -> {
assertEquals(2, g.V().has("person", "name", "marko").count().next().intValue());
assertEquals(1, g.V().has("person", "name", "marko").properties("name").has(T.value, "marko").has("acl", "private").count().next().intValue());
g.V().has("person", "name", "marko").forEachRemaining(v -> {
assertEquals(MultiMetaNeo4jTrait.VERTEX_PROPERTY_TOKEN, ((Neo4jVertex) v).getBaseVertex().getProperty("name"));
});
});
}
@Test
public void shouldDoLabelsNamespaceBehavior() {
graph.tx().readWrite();
this.getBaseGraph().execute("CREATE INDEX ON :Person(name)", null);
this.getBaseGraph().execute("CREATE INDEX ON :Product(name)", null);
this.getBaseGraph().execute("CREATE INDEX ON :Corporate(name)", null);
this.graph.tx().commit();
this.graph.addVertex(T.label, "Person", "name", "marko");
this.graph.addVertex(T.label, "Person", "name", "john");
this.graph.addVertex(T.label, "Person", "name", "pete");
this.graph.addVertex(T.label, "Product", "name", "marko");
this.graph.addVertex(T.label, "Product", "name", "john");
this.graph.addVertex(T.label, "Product", "name", "pete");
this.graph.addVertex(T.label, "Corporate", "name", "marko");
this.graph.addVertex(T.label, "Corporate", "name", "john");
this.graph.addVertex(T.label, "Corporate", "name", "pete");
this.graph.tx().commit();
assertEquals(1, this.g.V().has(T.label, "Person").has("name", "marko").has(T.label, "Person").count().next(), 0);
assertEquals(1, this.g.V().has(T.label, "Product").has("name", "marko").has(T.label, "Product").count().next(), 0);
assertEquals(1, this.g.V().has(T.label, "Corporate").has("name", "marko").has(T.label, "Corporate").count().next(), 0);
assertEquals(0, this.g.V().has(T.label, "Person").has("name", "marko").has(T.label, "Product").count().next(), 0);
assertEquals(0, this.g.V().has(T.label, "Product").has("name", "marko").has(T.label, "Person").count().next(), 0);
assertEquals(0, this.g.V().has(T.label, "Corporate").has("name", "marko").has(T.label, "Person").count().next(), 0);
}
}