From 7d9757343257121a3f69dcd5b16d4eeb5685818c Mon Sep 17 00:00:00 2001 From: Boxuan Li Date: Sun, 6 Dec 2020 17:50:10 +0800 Subject: [PATCH] Fix neq and null condition query behavior Fixes #2205 Signed-off-by: Boxuan Li --- .../graphdb/JanusGraphIndexTest.java | 30 ++++++ .../janusgraph/graphdb/JanusGraphTest.java | 96 +++++++++++++++++-- .../janusgraph/graphdb/query/QueryUtil.java | 3 +- .../org/janusgraph/core/attribute/Cmp.java | 9 +- 4 files changed, 125 insertions(+), 13 deletions(-) diff --git a/janusgraph-backend-testutils/src/main/java/org/janusgraph/graphdb/JanusGraphIndexTest.java b/janusgraph-backend-testutils/src/main/java/org/janusgraph/graphdb/JanusGraphIndexTest.java index 71c3d72099..767b955158 100644 --- a/janusgraph-backend-testutils/src/main/java/org/janusgraph/graphdb/JanusGraphIndexTest.java +++ b/janusgraph-backend-testutils/src/main/java/org/janusgraph/graphdb/JanusGraphIndexTest.java @@ -188,6 +188,36 @@ public static void assertGraphOfTheGods(JanusGraph graphOfTheGods) { graphOfTheGods.tx().commit(); } + @Test + public void testNullQueries() { + makeKey("p2", String.class); + PropertyKey p3 = makeKey("p3", String.class); + PropertyKey p4 = makeKey("p4", String.class); + mgmt.buildIndex("composite", Vertex.class).addKey(p3).buildCompositeIndex(); + mgmt.buildIndex("mixed", Vertex.class).addKey(p4, Mapping.STRING.asParameter()).buildMixedIndex(INDEX); + finishSchema(); + + tx.addVertex(); + tx.commit(); + newTx(); + + // test neq query + assertFalse(tx.traversal().V().has("p4", P.neq("v")).hasNext()); + assertFalse(tx.traversal().V().has("p4", P.neq(null)).hasNext()); + + // test has null query + assertTrue(tx.traversal().V().has("p4", (Object) null).hasNext()); + + // test has not query + assertTrue(tx.traversal().V().hasNot("p4").hasNext()); + assertTrue(tx.query().hasNot("p4").vertices().iterator().hasNext()); + assertFalse(tx.query().hasNot("p4", null).vertices().iterator().hasNext()); + assertFalse(tx.query().hasNot("p4", "value").vertices().iterator().hasNext()); + + // test not has query + assertTrue(tx.traversal().V().not(__.has("p4")).hasNext()); + } + /** * Ensure clearing storage actually removes underlying graph and index databases. * @throws Exception diff --git a/janusgraph-backend-testutils/src/main/java/org/janusgraph/graphdb/JanusGraphTest.java b/janusgraph-backend-testutils/src/main/java/org/janusgraph/graphdb/JanusGraphTest.java index bcc3414ae5..58ac1be5e8 100644 --- a/janusgraph-backend-testutils/src/main/java/org/janusgraph/graphdb/JanusGraphTest.java +++ b/janusgraph-backend-testutils/src/main/java/org/janusgraph/graphdb/JanusGraphTest.java @@ -5792,16 +5792,100 @@ private static void assertElementOrderForWithoutIndex(StandardJanusGraph graph) //................................................ + @Test + public void testNeqQuery() { + makeKey("p2", String.class); + PropertyKey p3 = makeKey("p3", String.class); + mgmt.buildIndex("composite", Vertex.class).addKey(p3).buildCompositeIndex(); + finishSchema(); + + tx.addVertex(); + tx.commit(); + newTx(); + + // property not registered in schema + assertFalse(tx.traversal().V().has("p1", P.neq("v")).hasNext()); + assertFalse(tx.traversal().V().has("p1", P.neq(null)).hasNext()); + // property registered in schema + assertFalse(tx.traversal().V().has("p2", P.neq("v")).hasNext()); + assertFalse(tx.traversal().V().has("p2", P.neq(null)).hasNext()); + // property registered in schema and has composite index + assertFalse(tx.traversal().V().has("p3", P.neq("v")).hasNext()); + assertFalse(tx.traversal().V().has("p3", P.neq(null)).hasNext()); + } + + /** + * The behaviour of has(p, null) deviates from TinkerGraph. Since JanusGraph does not support null values, + * has(p, null) indicates hasNot(p). In other words, an absent property implicitly implies null value for that property. + */ + @Test + public void testHasNullQuery() { + makeKey("p2", String.class); + PropertyKey p3 = makeKey("p3", String.class); + mgmt.buildIndex("composite", Vertex.class).addKey(p3).buildCompositeIndex(); + finishSchema(); + + tx.addVertex(); + tx.commit(); + newTx(); + + // property not registered in schema + assertTrue(tx.traversal().V().has("p1", (Object) null).hasNext()); + // property registered in schema + assertTrue(tx.traversal().V().has("p2", (Object) null).hasNext()); + // property registered in schema and has composite index + assertTrue(tx.traversal().V().has("p3", (Object) null).hasNext()); + } + /** + * The behaviour of hasNot(p) is straight-forward: hasNot(p) means it does not have such property p. + * Note that hasNot(p, value) (which is a JanusGraph API rather than gremlin API) is a bit tricky and it is equivalent + * to has(p, neq(value)). Therefore, hasNot(p, null) means has(p, neq(null)) which is equivalent to has(p). + */ @Test public void testHasNot() { - JanusGraphVertex v1, v2; - v1 = graph.addVertex(); + makeKey("p2", String.class); + PropertyKey p3 = makeKey("p3", String.class); + mgmt.buildIndex("composite", Vertex.class).addKey(p3).buildCompositeIndex(); + + tx.addVertex(); + tx.commit(); + newTx(); + + // property not registered in schema + assertTrue(tx.traversal().V().hasNot("p1").hasNext()); + assertTrue(tx.query().hasNot("p1").vertices().iterator().hasNext()); + assertFalse(tx.query().hasNot("p1", null).vertices().iterator().hasNext()); + assertFalse(tx.query().hasNot("p1", "value").vertices().iterator().hasNext()); + // property registered in schema + assertTrue(tx.traversal().V().hasNot("p2").hasNext()); + assertTrue(tx.query().hasNot("p2").vertices().iterator().hasNext()); + assertFalse(tx.query().hasNot("p2", null).vertices().iterator().hasNext()); + assertFalse(tx.query().hasNot("p2", "value").vertices().iterator().hasNext()); + // property registered in schema and has composite index + assertTrue(tx.traversal().V().hasNot("p3").hasNext()); + assertTrue(tx.query().hasNot("p3").vertices().iterator().hasNext()); + assertFalse(tx.query().hasNot("p3", null).vertices().iterator().hasNext()); + assertFalse(tx.query().hasNot("p3", "value").vertices().iterator().hasNext()); + } + + @Test + public void testNotHas() { + makeKey("p2", String.class); + PropertyKey p3 = makeKey("p3", String.class); + mgmt.buildIndex("composite", Vertex.class).addKey(p3).buildCompositeIndex(); + finishSchema(); + + tx.addVertex(); + tx.commit(); + newTx(); - v2 = graph.query().hasNot("abcd").vertices().iterator().next(); - assertEquals(v1, v2); - v2 = graph.query().hasNot("abcd", true).vertices().iterator().next(); - assertEquals(v1, v2); + // property not registered in schema + assertTrue(tx.traversal().V().not(__.has("p1")).hasNext()); + // property registered in schema + assertTrue(tx.traversal().V().not(__.has("p2")).hasNext()); + // property registered in schema and has composite index + assertTrue(tx.traversal().V().not(__.has("p3")).hasNext()); } @Test diff --git a/janusgraph-core/src/main/java/org/janusgraph/graphdb/query/QueryUtil.java b/janusgraph-core/src/main/java/org/janusgraph/graphdb/query/QueryUtil.java index 2c455db170..370eb11017 100644 --- a/janusgraph-core/src/main/java/org/janusgraph/graphdb/query/QueryUtil.java +++ b/janusgraph-core/src/main/java/org/janusgraph/graphdb/query/QueryUtil.java @@ -160,8 +160,7 @@ public static And constraints2QNF(StandardJanus final RelationType type = getType(tx, atom.getKey()); if (type == null) { - if (atom.getPredicate() == Cmp.EQUAL && atom.getValue() == null || - (atom.getPredicate() == Cmp.NOT_EQUAL && atom.getValue() != null)) + if (atom.getPredicate() == Cmp.EQUAL && atom.getValue() == null) continue; //Ignore condition, its trivially satisfied return null; diff --git a/janusgraph-driver/src/main/java/org/janusgraph/core/attribute/Cmp.java b/janusgraph-driver/src/main/java/org/janusgraph/core/attribute/Cmp.java index 07fb4c8bff..39de74779b 100644 --- a/janusgraph-driver/src/main/java/org/janusgraph/core/attribute/Cmp.java +++ b/janusgraph-driver/src/main/java/org/janusgraph/core/attribute/Cmp.java @@ -73,11 +73,10 @@ public boolean isValidCondition(Object condition) { @Override public boolean test(Object value, Object condition) { - if (condition==null) { - return value!=null; - } else { - return !condition.equals(value); - } + // To align with TinkerPop behaviour, if an element does not have property p, then has(p, neq(anything)) + // should always evaluate to false. Note that JanusGraph does not support null value, so the "value == null" + // here implies the element does not have such property. + return value != null && !value.equals(condition); } @Override