diff --git a/tests/qa/continuous_integration b/tests/qa/continuous_integration index 3922b03cc..13060011f 100755 --- a/tests/qa/continuous_integration +++ b/tests/qa/continuous_integration @@ -22,7 +22,7 @@ log = logging.getLogger(__name__) # constants memgraph_suite = "memgraph_V1" -extra_suites = ["openCypher_M06"] +extra_suites = ["openCypher_M09"] results_folder = os.path.join("tck_engine", "results") suite_suffix = "memgraph-{}.json" qa_status_path = ".quality_assurance_status" diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/Aggregation.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/Aggregation.feature new file mode 100644 index 000000000..4d1d5ffb7 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/Aggregation.feature @@ -0,0 +1,162 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: Aggregation + + Scenario: `max()` over strings + Given any graph + When executing query: + """ + UNWIND ['a', 'b', 'B', null, 'abc', 'abc1'] AS i + RETURN max(i) + """ + Then the result should be: + | max(i) | + | 'b' | + And no side effects + + Scenario: `min()` over strings + Given any graph + When executing query: + """ + UNWIND ['a', 'b', 'B', null, 'abc', 'abc1'] AS i + RETURN min(i) + """ + Then the result should be: + | min(i) | + | 'B' | + And no side effects + + Scenario: `max()` over integers + Given any graph + When executing query: + """ + UNWIND [1, 2, 0, null, -1] AS x + RETURN max(x) + """ + Then the result should be: + | max(x) | + | 2 | + And no side effects + + Scenario: `min()` over integers + Given any graph + When executing query: + """ + UNWIND [1, 2, 0, null, -1] AS x + RETURN min(x) + """ + Then the result should be: + | min(x) | + | -1 | + And no side effects + + Scenario: `max()` over floats + Given any graph + When executing query: + """ + UNWIND [1.0, 2.0, 0.5, null] AS x + RETURN max(x) + """ + Then the result should be: + | max(x) | + | 2.0 | + And no side effects + + Scenario: `min()` over floats + Given any graph + When executing query: + """ + UNWIND [1.0, 2.0, 0.5, null] AS x + RETURN min(x) + """ + Then the result should be: + | min(x) | + | 0.5 | + And no side effects + + Scenario: `max()` over mixed numeric values + Given any graph + When executing query: + """ + UNWIND [1, 2.0, 5, null, 3.2, 0.1] AS x + RETURN max(x) + """ + Then the result should be: + | max(x) | + | 5 | + And no side effects + + Scenario: `min()` over mixed numeric values + Given any graph + When executing query: + """ + UNWIND [1, 2.0, 5, null, 3.2, 0.1] AS x + RETURN min(x) + """ + Then the result should be: + | min(x) | + | 0.1 | + And no side effects + + Scenario: `max()` over mixed values + Given any graph + When executing query: + """ + UNWIND [1, 'a', null, [1, 2], 0.2, 'b'] AS x + RETURN max(x) + """ + Then the result should be: + | max(x) | + | 1 | + And no side effects + + Scenario: `min()` over mixed values + Given any graph + When executing query: + """ + UNWIND [1, 'a', null, [1, 2], 0.2, 'b'] AS x + RETURN min(x) + """ + Then the result should be: + | min(x) | + | [1, 2] | + And no side effects + + Scenario: `max()` over list values + Given any graph + When executing query: + """ + UNWIND [[1], [2], [2, 1]] AS x + RETURN max(x) + """ + Then the result should be: + | max(x) | + | [2, 1] | + And no side effects + + Scenario: `min()` over list values + Given any graph + When executing query: + """ + UNWIND [[1], [2], [2, 1]] AS x + RETURN min(x) + """ + Then the result should be: + | min(x) | + | [1] | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/AggregationAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/AggregationAcceptance.feature new file mode 100644 index 000000000..e7060609b --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/AggregationAcceptance.feature @@ -0,0 +1,473 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: AggregationAcceptance + + Scenario: Support multiple divisions in aggregate function + Given an empty graph + And having executed: + """ + UNWIND range(0, 7250) AS i + CREATE () + """ + When executing query: + """ + MATCH (n) + RETURN count(n) / 60 / 60 AS count + """ + Then the result should be: + | count | + | 2 | + And no side effects + + Scenario: Support column renaming for aggregates as well + Given an empty graph + And having executed: + """ + UNWIND range(0, 10) AS i + CREATE () + """ + When executing query: + """ + MATCH () + RETURN count(*) AS columnName + """ + Then the result should be: + | columnName | + | 11 | + And no side effects + + Scenario: Aggregates inside normal functions + Given an empty graph + And having executed: + """ + UNWIND range(0, 10) AS i + CREATE () + """ + When executing query: + """ + MATCH (a) + RETURN size(collect(a)) + """ + Then the result should be: + | size(collect(a)) | + | 11 | + And no side effects + + Scenario: Handle aggregates inside non-aggregate expressions + Given an empty graph + When executing query: + """ + MATCH (a {name: 'Andres'})<-[:FATHER]-(child) + RETURN {foo: a.name='Andres', kids: collect(child.name)} + """ + Then the result should be: + | {foo: a.name='Andres', kids: collect(child.name)} | + And no side effects + + Scenario: Count nodes + Given an empty graph + And having executed: + """ + CREATE (a:L), (b1), (b2) + CREATE (a)-[:A]->(b1), (a)-[:A]->(b2) + """ + When executing query: + """ + MATCH (a:L)-[rel]->(b) + RETURN a, count(*) + """ + Then the result should be: + | a | count(*) | + | (:L) | 2 | + And no side effects + + Scenario: Sort on aggregate function and normal property + Given an empty graph + And having executed: + """ + CREATE ({division: 'Sweden'}) + CREATE ({division: 'Germany'}) + CREATE ({division: 'England'}) + CREATE ({division: 'Sweden'}) + """ + When executing query: + """ + MATCH (n) + RETURN n.division, count(*) + ORDER BY count(*) DESC, n.division ASC + """ + Then the result should be, in order: + | n.division | count(*) | + | 'Sweden' | 2 | + | 'England' | 1 | + | 'Germany' | 1 | + And no side effects + + Scenario: Aggregate on property + Given an empty graph + And having executed: + """ + CREATE ({x: 33}) + CREATE ({x: 33}) + CREATE ({x: 42}) + """ + When executing query: + """ + MATCH (n) + RETURN n.x, count(*) + """ + Then the result should be: + | n.x | count(*) | + | 42 | 1 | + | 33 | 2 | + And no side effects + + Scenario: Count non-null values + Given an empty graph + And having executed: + """ + CREATE ({y: 'a', x: 33}) + CREATE ({y: 'a'}) + CREATE ({y: 'b', x: 42}) + """ + When executing query: + """ + MATCH (n) + RETURN n.y, count(n.x) + """ + Then the result should be: + | n.y | count(n.x) | + | 'a' | 1 | + | 'b' | 1 | + And no side effects + + Scenario: Sum non-null values + Given an empty graph + And having executed: + """ + CREATE ({y: 'a', x: 33}) + CREATE ({y: 'a'}) + CREATE ({y: 'a', x: 42}) + """ + When executing query: + """ + MATCH (n) + RETURN n.y, sum(n.x) + """ + Then the result should be: + | n.y | sum(n.x) | + | 'a' | 75 | + And no side effects + + Scenario: Handle aggregation on functions + Given an empty graph + And having executed: + """ + CREATE (a:L), (b1), (b2) + CREATE (a)-[:A]->(b1), (a)-[:A]->(b2) + """ + When executing query: + """ + MATCH p=(a:L)-[*]->(b) + RETURN b, avg(length(p)) + """ + Then the result should be: + | b | avg(length(p)) | + | () | 1.0 | + | () | 1.0 | + And no side effects + + Scenario: Distinct on unbound node + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (a) + RETURN count(DISTINCT a) + """ + Then the result should be: + | count(DISTINCT a) | + | 0 | + And no side effects + + Scenario: Distinct on null + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (a) + RETURN count(DISTINCT a.foo) + """ + Then the result should be: + | count(DISTINCT a.foo) | + | 0 | + And no side effects + + Scenario: Collect distinct nulls + Given any graph + When executing query: + """ + UNWIND [null, null] AS x + RETURN collect(DISTINCT x) AS c + """ + Then the result should be: + | c | + | [] | + And no side effects + + Scenario: Collect distinct values mixed with nulls + Given any graph + When executing query: + """ + UNWIND [null, 1, null] AS x + RETURN collect(DISTINCT x) AS c + """ + Then the result should be: + | c | + | [1] | + And no side effects + + Scenario: Aggregate on list values + Given an empty graph + And having executed: + """ + CREATE ({color: ['red']}) + CREATE ({color: ['blue']}) + CREATE ({color: ['red']}) + """ + When executing query: + """ + MATCH (a) + RETURN DISTINCT a.color, count(*) + """ + Then the result should be: + | a.color | count(*) | + | ['red'] | 2 | + | ['blue'] | 1 | + And no side effects + + Scenario: Aggregates in aggregates + Given any graph + When executing query: + """ + RETURN count(count(*)) + """ + Then a SyntaxError should be raised at compile time: NestedAggregation + + Scenario: Aggregates with arithmetics + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH () + RETURN count(*) * 10 AS c + """ + Then the result should be: + | c | + | 10 | + And no side effects + + Scenario: Aggregates ordered by arithmetics + Given an empty graph + And having executed: + """ + CREATE (:A), (:X), (:X) + """ + When executing query: + """ + MATCH (a:A), (b:X) + RETURN count(a) * 10 + count(b) * 5 AS x + ORDER BY x + """ + Then the result should be, in order: + | x | + | 30 | + And no side effects + + Scenario: Multiple aggregates on same variable + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + RETURN count(n), collect(n) + """ + Then the result should be: + | count(n) | collect(n) | + | 1 | [()] | + And no side effects + + Scenario: Simple counting of nodes + Given an empty graph + And having executed: + """ + UNWIND range(1, 100) AS i + CREATE () + """ + When executing query: + """ + MATCH () + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 100 | + And no side effects + + Scenario: Aggregation of named paths + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B), (c:C), (d:D), (e:E), (f:F) + CREATE (a)-[:R]->(b) + CREATE (c)-[:R]->(d) + CREATE (d)-[:R]->(e) + CREATE (e)-[:R]->(f) + """ + When executing query: + """ + MATCH p = (a)-[*]->(b) + RETURN collect(nodes(p)) AS paths, length(p) AS l + ORDER BY l + """ + Then the result should be, in order: + | paths | l | + | [[(:A), (:B)], [(:C), (:D)], [(:D), (:E)], [(:E), (:F)]] | 1 | + | [[(:C), (:D), (:E)], [(:D), (:E), (:F)]] | 2 | + | [[(:C), (:D), (:E), (:F)]] | 3 | + And no side effects + + Scenario: Aggregation with `min()` + Given an empty graph + And having executed: + """ + CREATE (a:T {name: 'a'}), (b:T {name: 'b'}), (c:T {name: 'c'}) + CREATE (a)-[:R]->(b) + CREATE (a)-[:R]->(c) + CREATE (c)-[:R]->(b) + """ + When executing query: + """ + MATCH p = (a:T {name: 'a'})-[:R*]->(other:T) + WHERE other <> a + WITH a, other, min(length(p)) AS len + RETURN a.name AS name, collect(other.name) AS others, len + """ + Then the result should be (ignoring element order for lists): + | name | others | len | + | 'a' | ['c', 'b'] | 1 | + And no side effects + + Scenario: Handle subexpression in aggregation also occurring as standalone expression with nested aggregation in a literal map + Given an empty graph + And having executed: + """ + CREATE (:A), (:B {prop: 42}) + """ + When executing query: + """ + MATCH (a:A), (b:B) + RETURN coalesce(a.prop, b.prop) AS foo, + b.prop AS bar, + {y: count(b)} AS baz + """ + Then the result should be: + | foo | bar | baz | + | 42 | 42 | {y: 1} | + And no side effects + + Scenario: Projection during aggregation in WITH before MERGE and after WITH with predicate + Given an empty graph + And having executed: + """ + CREATE (:A {prop: 42}) + """ + When executing query: + """ + UNWIND [42] AS props + WITH props WHERE props > 32 + WITH DISTINCT props AS p + MERGE (a:A {prop: p}) + RETURN a.prop AS prop + """ + Then the result should be: + | prop | + | 42 | + And no side effects + + Scenario: No overflow during summation + Given any graph + When executing query: + """ + UNWIND range(1000000, 2000000) AS i + WITH i + LIMIT 3000 + RETURN sum(i) + """ + Then the result should be: + | sum(i) | + | 3004498500 | + And no side effects + + Scenario: Counting with loops + Given an empty graph + And having executed: + """ + CREATE (a), (a)-[:R]->(a) + """ + When executing query: + """ + MATCH ()-[r]-() + RETURN count(r) + """ + Then the result should be: + | count(r) | + | 1 | + And no side effects + + Scenario: `max()` should aggregate strings + Given any graph + When executing query: + """ + UNWIND ['a', 'b', 'B', null, 'abc', 'abc1'] AS i + RETURN max(i) + """ + Then the result should be: + | max(i) | + | 'b' | + And no side effects + + Scenario: `min()` should aggregate strings + Given any graph + When executing query: + """ + UNWIND ['a', 'b', 'B', null, 'abc', 'abc1'] AS i + RETURN min(i) + """ + Then the result should be: + | min(i) | + | 'B' | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/ColumnNameAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/ColumnNameAcceptance.feature new file mode 100644 index 000000000..169a80cec --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/ColumnNameAcceptance.feature @@ -0,0 +1,68 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: ColumnNameAcceptance + + Background: + Given an empty graph + And having executed: + """ + CREATE () + """ + + Scenario: Keeping used expression 1 + When executing query: + """ + MATCH (n) + RETURN cOuNt( * ) + """ + Then the result should be: + | cOuNt( * ) | + | 1 | + And no side effects + + Scenario: Keeping used expression 2 + When executing query: + """ + MATCH p = (n)-->(b) + RETURN nOdEs( p ) + """ + Then the result should be: + | nOdEs( p ) | + And no side effects + + Scenario: Keeping used expression 3 + When executing query: + """ + MATCH p = (n)-->(b) + RETURN coUnt( dIstInct p ) + """ + Then the result should be: + | coUnt( dIstInct p ) | + | 0 | + And no side effects + + Scenario: Keeping used expression 4 + When executing query: + """ + MATCH p = (n)-->(b) + RETURN aVg( n.aGe ) + """ + Then the result should be: + | aVg( n.aGe ) | + | null | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/Comparability.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/Comparability.feature new file mode 100644 index 000000000..721ea485b --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/Comparability.feature @@ -0,0 +1,87 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: Comparability + + Scenario: Comparing strings and integers using > in an AND'd predicate + Given an empty graph + And having executed: + """ + CREATE (root:Root)-[:T]->(:Child {id: 0}), + (root)-[:T]->(:Child {id: 'xx'}), + (root)-[:T]->(:Child) + """ + When executing query: + """ + MATCH (:Root)-->(i:Child) + WHERE exists(i.id) AND i.id > 'x' + RETURN i.id + """ + Then the result should be: + | i.id | + | 'xx' | + And no side effects + + Scenario: Comparing strings and integers using > in a OR'd predicate + Given an empty graph + And having executed: + """ + CREATE (root:Root)-[:T]->(:Child {id: 0}), + (root)-[:T]->(:Child {id: 'xx'}), + (root)-[:T]->(:Child) + """ + When executing query: + """ + MATCH (:Root)-->(i:Child) + WHERE NOT exists(i.id) OR i.id > 'x' + RETURN i.id + """ + Then the result should be: + | i.id | + | 'xx' | + | null | + And no side effects + + Scenario Outline: Comparing across types yields null, except numbers + Given an empty graph + And having executed: + """ + CREATE ()-[:T]->() + """ + When executing query: + """ + MATCH p = (n)-[r]->() + WITH [n, r, p, '', 1, 3.14, true, null, [], {}] AS types + UNWIND range(0, size(types) - 1) AS i + UNWIND range(0, size(types) - 1) AS j + WITH types[i] AS lhs, types[j] AS rhs + WHERE i <> j + WITH lhs, rhs, lhs <operator> rhs AS result + WHERE result + RETURN lhs, rhs + """ + Then the result should be: + | lhs | rhs | + | <lhs> | <rhs> | + And no side effects + + Examples: + | operator | lhs | rhs | + | < | 1 | 3.14 | + | <= | 1 | 3.14 | + | >= | 3.14 | 1 | + | > | 3.14 | 1 | diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/ComparisonOperatorAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/ComparisonOperatorAcceptance.feature new file mode 100644 index 000000000..35b198b36 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/ComparisonOperatorAcceptance.feature @@ -0,0 +1,208 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: ComparisonOperatorAcceptance + + Scenario: Handling numerical ranges 1 + Given an empty graph + And having executed: + """ + UNWIND [1, 2, 3] AS i + CREATE ({value: i}) + """ + When executing query: + """ + MATCH (n) + WHERE 1 < n.value < 3 + RETURN n.value + """ + Then the result should be: + | n.value | + | 2 | + And no side effects + + Scenario: Handling numerical ranges 2 + Given an empty graph + And having executed: + """ + UNWIND [1, 2, 3] AS i + CREATE ({value: i}) + """ + When executing query: + """ + MATCH (n) + WHERE 1 < n.value <= 3 + RETURN n.value + """ + Then the result should be: + | n.value | + | 2 | + | 3 | + And no side effects + + Scenario: Handling numerical ranges 3 + Given an empty graph + And having executed: + """ + UNWIND [1, 2, 3] AS i + CREATE ({value: i}) + """ + When executing query: + """ + MATCH (n) + WHERE 1 <= n.value < 3 + RETURN n.value + """ + Then the result should be: + | n.value | + | 1 | + | 2 | + And no side effects + + Scenario: Handling numerical ranges 4 + Given an empty graph + And having executed: + """ + UNWIND [1, 2, 3] AS i + CREATE ({value: i}) + """ + When executing query: + """ + MATCH (n) + WHERE 1 <= n.value <= 3 + RETURN n.value + """ + Then the result should be: + | n.value | + | 1 | + | 2 | + | 3 | + And no side effects + + Scenario: Handling string ranges 1 + Given an empty graph + And having executed: + """ + UNWIND ['a', 'b', 'c'] AS c + CREATE ({value: c}) + """ + When executing query: + """ + MATCH (n) + WHERE 'a' < n.value < 'c' + RETURN n.value + """ + Then the result should be: + | n.value | + | 'b' | + And no side effects + + Scenario: Handling string ranges 2 + Given an empty graph + And having executed: + """ + UNWIND ['a', 'b', 'c'] AS c + CREATE ({value: c}) + """ + When executing query: + """ + MATCH (n) + WHERE 'a' < n.value <= 'c' + RETURN n.value + """ + Then the result should be: + | n.value | + | 'b' | + | 'c' | + And no side effects + + Scenario: Handling string ranges 3 + Given an empty graph + And having executed: + """ + UNWIND ['a', 'b', 'c'] AS c + CREATE ({value: c}) + """ + When executing query: + """ + MATCH (n) + WHERE 'a' <= n.value < 'c' + RETURN n.value + """ + Then the result should be: + | n.value | + | 'a' | + | 'b' | + And no side effects + + Scenario: Handling string ranges 4 + Given an empty graph + And having executed: + """ + UNWIND ['a', 'b', 'c'] AS c + CREATE ({value: c}) + """ + When executing query: + """ + MATCH (n) + WHERE 'a' <= n.value <= 'c' + RETURN n.value + """ + Then the result should be: + | n.value | + | 'a' | + | 'b' | + | 'c' | + And no side effects + + Scenario: Handling empty range + Given an empty graph + And having executed: + """ + CREATE ({value: 3}) + """ + When executing query: + """ + MATCH (n) + WHERE 10 < n.value <= 3 + RETURN n.value + """ + Then the result should be: + | n.value | + And no side effects + + Scenario: Handling long chains of operators + Given an empty graph + And having executed: + """ + CREATE (a:A {prop1: 3, prop2: 4}) + CREATE (b:B {prop1: 4, prop2: 5}) + CREATE (c:C {prop1: 4, prop2: 4}) + CREATE (a)-[:R]->(b) + CREATE (b)-[:R]->(c) + CREATE (c)-[:R]->(a) + """ + When executing query: + """ + MATCH (n)-->(m) + WHERE n.prop1 < m.prop1 = n.prop2 <> m.prop2 + RETURN labels(m) + """ + Then the result should be: + | labels(m) | + | ['B'] | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/Create.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/Create.feature new file mode 100644 index 000000000..d117f0459 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/Create.feature @@ -0,0 +1,71 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: Create + + Scenario: Creating a node + Given any graph + When executing query: + """ + CREATE () + """ + Then the result should be empty + And the side effects should be: + | +nodes | 1 | + + Scenario: Creating two nodes + Given any graph + When executing query: + """ + CREATE (), () + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + + Scenario: Creating two nodes and a relationship + Given any graph + When executing query: + """ + CREATE ()-[:TYPE]->() + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + + Scenario: Creating a node with a label + Given an empty graph + When executing query: + """ + CREATE (:Label) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 1 | + | +labels | 1 | + + Scenario: Creating a node with a property + Given any graph + When executing query: + """ + CREATE ({created: true}) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 1 | + | +properties | 1 | diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/CreateAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/CreateAcceptance.feature new file mode 100644 index 000000000..0cff64d36 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/CreateAcceptance.feature @@ -0,0 +1,522 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: CreateAcceptance + + Scenario: Create a single node with multiple labels + Given an empty graph + When executing query: + """ + CREATE (:A:B:C:D) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 1 | + | +labels | 4 | + + Scenario: Combine MATCH and CREATE + Given an empty graph + And having executed: + """ + CREATE (), () + """ + When executing query: + """ + MATCH () + CREATE () + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + + Scenario: Combine MATCH, WITH and CREATE + Given an empty graph + And having executed: + """ + CREATE (), () + """ + When executing query: + """ + MATCH () + CREATE () + WITH * + MATCH () + CREATE () + """ + Then the result should be empty + And the side effects should be: + | +nodes | 10 | + + Scenario: Newly-created nodes not visible to preceding MATCH + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH () + CREATE () + """ + Then the result should be empty + And the side effects should be: + | +nodes | 1 | + + Scenario: Create a single node with properties + Given any graph + When executing query: + """ + CREATE (n {prop: 'foo'}) + RETURN n.prop AS p + """ + Then the result should be: + | p | + | 'foo' | + And the side effects should be: + | +nodes | 1 | + | +properties | 1 | + + Scenario: Creating a node with null properties should not return those properties + Given any graph + When executing query: + """ + CREATE (n {id: 12, property: null}) + RETURN n.id AS id + """ + Then the result should be: + | id | + | 12 | + And the side effects should be: + | +nodes | 1 | + | +properties | 1 | + + Scenario: Creating a relationship with null properties should not return those properties + Given any graph + When executing query: + """ + CREATE ()-[r:X {id: 12, property: null}]->() + RETURN r.id + """ + Then the result should be: + | r.id | + | 12 | + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + | +properties | 1 | + + Scenario: Create a simple pattern + Given any graph + When executing query: + """ + CREATE ()-[:R]->() + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + + Scenario: Create a self loop + Given an empty graph + When executing query: + """ + CREATE (root:R)-[:LINK]->(root) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 1 | + | +relationships | 1 | + | +labels | 1 | + + Scenario: Create a self loop using MATCH + Given an empty graph + And having executed: + """ + CREATE (:R) + """ + When executing query: + """ + MATCH (root:R) + CREATE (root)-[:LINK]->(root) + """ + Then the result should be empty + And the side effects should be: + | +relationships | 1 | + + Scenario: Create nodes and relationships + Given any graph + When executing query: + """ + CREATE (a), (b), + (a)-[:R]->(b) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + + Scenario: Create a relationship with a property + Given any graph + When executing query: + """ + CREATE ()-[:R {prop: 42}]->() + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + | +properties | 1 | + + Scenario: Create a relationship with the correct direction + Given an empty graph + And having executed: + """ + CREATE (:X) + CREATE (:Y) + """ + When executing query: + """ + MATCH (x:X), (y:Y) + CREATE (x)<-[:TYPE]-(y) + """ + Then the result should be empty + And the side effects should be: + | +relationships | 1 | + When executing control query: + """ + MATCH (x:X)<-[:TYPE]-(y:Y) + RETURN x, y + """ + Then the result should be: + | x | y | + | (:X) | (:Y) | + + Scenario: Create a relationship and an end node from a matched starting node + Given an empty graph + And having executed: + """ + CREATE (:Begin) + """ + When executing query: + """ + MATCH (x:Begin) + CREATE (x)-[:TYPE]->(:End) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 1 | + | +relationships | 1 | + | +labels | 1 | + When executing control query: + """ + MATCH (x:Begin)-[:TYPE]->() + RETURN x + """ + Then the result should be: + | x | + | (:Begin) | + + Scenario: Create a single node after a WITH + Given an empty graph + And having executed: + """ + CREATE (), () + """ + When executing query: + """ + MATCH () + CREATE () + WITH * + CREATE () + """ + Then the result should be empty + And the side effects should be: + | +nodes | 4 | + + Scenario: Create a relationship with a reversed direction + Given an empty graph + When executing query: + """ + CREATE (:A)<-[:R]-(:B) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + | +labels | 2 | + When executing control query: + """ + MATCH (a:A)<-[:R]-(b:B) + RETURN a, b + """ + Then the result should be: + | a | b | + | (:A) | (:B) | + + Scenario: Create a pattern with multiple hops + Given an empty graph + When executing query: + """ + CREATE (:A)-[:R]->(:B)-[:R]->(:C) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 3 | + | +relationships | 2 | + | +labels | 3 | + When executing control query: + """ + MATCH (a:A)-[:R]->(b:B)-[:R]->(c:C) + RETURN a, b, c + """ + Then the result should be: + | a | b | c | + | (:A) | (:B) | (:C) | + + Scenario: Create a pattern with multiple hops in the reverse direction + Given an empty graph + When executing query: + """ + CREATE (:A)<-[:R]-(:B)<-[:R]-(:C) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 3 | + | +relationships | 2 | + | +labels | 3 | + When executing control query: + """ + MATCH (a)<-[:R]-(b)<-[:R]-(c) + RETURN a, b, c + """ + Then the result should be: + | a | b | c | + | (:A) | (:B) | (:C) | + + Scenario: Create a pattern with multiple hops in varying directions + Given an empty graph + When executing query: + """ + CREATE (:A)-[:R]->(:B)<-[:R]-(:C) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 3 | + | +relationships | 2 | + | +labels | 3 | + When executing control query: + """ + MATCH (a:A)-[r1:R]->(b:B)<-[r2:R]-(c:C) + RETURN a, b, c + """ + Then the result should be: + | a | b | c | + | (:A) | (:B) | (:C) | + + Scenario: Create a pattern with multiple hops with multiple types and varying directions + Given any graph + When executing query: + """ + CREATE ()-[:R1]->()<-[:R2]-()-[:R3]->() + """ + Then the result should be empty + And the side effects should be: + | +nodes | 4 | + | +relationships | 3 | + When executing query: + """ + MATCH ()-[r1:R1]->()<-[r2:R2]-()-[r3:R3]->() + RETURN r1, r2, r3 + """ + Then the result should be: + | r1 | r2 | r3 | + | [:R1] | [:R2] | [:R3] | + + Scenario: Nodes are not created when aliases are applied to variable names + Given an empty graph + And having executed: + """ + CREATE ({foo: 1}) + """ + When executing query: + """ + MATCH (n) + MATCH (m) + WITH n AS a, m AS b + CREATE (a)-[:T]->(b) + RETURN a, b + """ + Then the result should be: + | a | b | + | ({foo: 1}) | ({foo: 1}) | + And the side effects should be: + | +relationships | 1 | + + Scenario: Only a single node is created when an alias is applied to a variable name + Given an empty graph + And having executed: + """ + CREATE (:X) + """ + When executing query: + """ + MATCH (n) + WITH n AS a + CREATE (a)-[:T]->() + RETURN a + """ + Then the result should be: + | a | + | (:X) | + And the side effects should be: + | +nodes | 1 | + | +relationships | 1 | + + Scenario: Nodes are not created when aliases are applied to variable names multiple times + Given an empty graph + And having executed: + """ + CREATE ({foo: 'A'}) + """ + When executing query: + """ + MATCH (n) + MATCH (m) + WITH n AS a, m AS b + CREATE (a)-[:T]->(b) + WITH a AS x, b AS y + CREATE (x)-[:T]->(y) + RETURN x, y + """ + Then the result should be: + | x | y | + | ({foo: 'A'}) | ({foo: 'A'}) | + And the side effects should be: + | +relationships | 2 | + + Scenario: Only a single node is created when an alias is applied to a variable name multiple times + Given an empty graph + And having executed: + """ + CREATE ({foo: 5}) + """ + When executing query: + """ + MATCH (n) + WITH n AS a + CREATE (a)-[:T]->() + WITH a AS x + CREATE (x)-[:T]->() + RETURN x + """ + Then the result should be: + | x | + | ({foo: 5}) | + And the side effects should be: + | +nodes | 2 | + | +relationships | 2 | + + Scenario: A bound node should be recognized after projection with WITH + WITH + Given any graph + When executing query: + """ + CREATE (a) + WITH a + WITH * + CREATE (b) + CREATE (a)<-[:T]-(b) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + + Scenario: A bound node should be recognized after projection with WITH + UNWIND + Given any graph + When executing query: + """ + CREATE (a) + WITH a + UNWIND [0] AS i + CREATE (b) + CREATE (a)<-[:T]-(b) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + + Scenario: A bound node should be recognized after projection with WITH + MERGE node + Given an empty graph + When executing query: + """ + CREATE (a) + WITH a + MERGE () + CREATE (b) + CREATE (a)<-[:T]-(b) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + + Scenario: A bound node should be recognized after projection with WITH + MERGE pattern + Given an empty graph + When executing query: + """ + CREATE (a) + WITH a + MERGE (x) + MERGE (y) + MERGE (x)-[:T]->(y) + CREATE (b) + CREATE (a)<-[:T]-(b) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + | +relationships | 2 | + + Scenario: Fail when trying to create using an undirected relationship pattern + Given any graph + When executing query: + """ + CREATE ({id: 2})-[r:KNOWS]-({id: 1}) + RETURN r + """ + Then a SyntaxError should be raised at compile time: RequiresDirectedRelationship + + Scenario: Creating a pattern with multiple hops and changing directions + Given an empty graph + When executing query: + """ + CREATE (:A)<-[:R1]-(:B)-[:R2]->(:C) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 3 | + | +relationships | 2 | + | +labels | 3 | + When executing control query: + """ + MATCH (a:A)<-[r1:R1]-(b:B)-[r2:R2]->(c:C) + RETURN * + """ + Then the result should be: + | a | b | c | r1 | r2 | + | (:A) | (:B) | (:C) | [:R1] | [:R2] | diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/DeleteAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/DeleteAcceptance.feature new file mode 100644 index 000000000..b07f2ff49 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/DeleteAcceptance.feature @@ -0,0 +1,375 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: DeleteAcceptance + + Scenario: Delete nodes + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + DELETE n + """ + Then the result should be empty + And the side effects should be: + | -nodes | 1 | + + Scenario: Detach delete node + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + DETACH DELETE n + """ + Then the result should be empty + And the side effects should be: + | -nodes | 1 | + + Scenario: Delete relationships + Given an empty graph + And having executed: + """ + UNWIND range(0, 2) AS i + CREATE ()-[:R]->() + """ + When executing query: + """ + MATCH ()-[r]-() + DELETE r + """ + Then the result should be empty + And the side effects should be: + | -relationships | 3 | + + Scenario: Deleting connected nodes + Given an empty graph + And having executed: + """ + CREATE (x:X) + CREATE (x)-[:R]->() + CREATE (x)-[:R]->() + CREATE (x)-[:R]->() + """ + When executing query: + """ + MATCH (n:X) + DELETE n + """ + Then a ConstraintVerificationFailed should be raised at runtime: DeleteConnectedNode + + Scenario: Detach deleting connected nodes and relationships + Given an empty graph + And having executed: + """ + CREATE (x:X) + CREATE (x)-[:R]->() + CREATE (x)-[:R]->() + CREATE (x)-[:R]->() + """ + When executing query: + """ + MATCH (n:X) + DETACH DELETE n + """ + Then the result should be empty + And the side effects should be: + | -nodes | 1 | + | -relationships | 3 | + | -labels | 1 | + + Scenario: Detach deleting paths + Given an empty graph + And having executed: + """ + CREATE (x:X), (n1), (n2), (n3) + CREATE (x)-[:R]->(n1) + CREATE (n1)-[:R]->(n2) + CREATE (n2)-[:R]->(n3) + """ + When executing query: + """ + MATCH p = (:X)-->()-->()-->() + DETACH DELETE p + """ + Then the result should be empty + And the side effects should be: + | -nodes | 4 | + | -relationships | 3 | + | -labels | 1 | + + Scenario: Undirected expand followed by delete and count + Given an empty graph + And having executed: + """ + CREATE ()-[:R]->() + """ + When executing query: + """ + MATCH (a)-[r]-(b) + DELETE r, a, b + RETURN count(*) AS c + """ + Then the result should be: + | c | + | 2 | + And the side effects should be: + | -nodes | 2 | + | -relationships | 1 | + + Scenario: Undirected variable length expand followed by delete and count + Given an empty graph + And having executed: + """ + CREATE (n1), (n2), (n3) + CREATE (n1)-[:R]->(n2) + CREATE (n2)-[:R]->(n3) + """ + When executing query: + """ + MATCH (a)-[*]-(b) + DETACH DELETE a, b + RETURN count(*) AS c + """ + Then the result should be: + | c | + | 6 | + And the side effects should be: + | -nodes | 3 | + | -relationships | 2 | + + Scenario: Create and delete in same query + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH () + CREATE (n) + DELETE n + """ + Then the result should be empty + And no side effects + + Scenario: Delete optionally matched relationship + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + OPTIONAL MATCH (n)-[r]-() + DELETE n, r + """ + Then the result should be empty + And the side effects should be: + | -nodes | 1 | + + Scenario: Delete on null node + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (n) + DELETE n + """ + Then the result should be empty + And no side effects + + Scenario: Detach delete on null node + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (n) + DETACH DELETE n + """ + Then the result should be empty + And no side effects + + Scenario: Delete on null path + Given an empty graph + When executing query: + """ + OPTIONAL MATCH p = ()-->() + DETACH DELETE p + """ + Then the result should be empty + And no side effects + + Scenario: Delete node from a list + Given an empty graph + And having executed: + """ + CREATE (u:User) + CREATE (u)-[:FRIEND]->() + CREATE (u)-[:FRIEND]->() + CREATE (u)-[:FRIEND]->() + CREATE (u)-[:FRIEND]->() + """ + And parameters are: + | friendIndex | 1 | + When executing query: + """ + MATCH (:User)-[:FRIEND]->(n) + WITH collect(n) AS friends + DETACH DELETE friends[$friendIndex] + """ + Then the result should be empty + And the side effects should be: + | -nodes | 1 | + | -relationships | 1 | + + Scenario: Delete relationship from a list + Given an empty graph + And having executed: + """ + CREATE (u:User) + CREATE (u)-[:FRIEND]->() + CREATE (u)-[:FRIEND]->() + CREATE (u)-[:FRIEND]->() + CREATE (u)-[:FRIEND]->() + """ + And parameters are: + | friendIndex | 1 | + When executing query: + """ + MATCH (:User)-[r:FRIEND]->() + WITH collect(r) AS friendships + DETACH DELETE friendships[$friendIndex] + """ + Then the result should be empty + And the side effects should be: + | -relationships | 1 | + + Scenario: Delete nodes from a map + Given an empty graph + And having executed: + """ + CREATE (:User), (:User) + """ + When executing query: + """ + MATCH (u:User) + WITH {key: u} AS nodes + DELETE nodes.key + """ + Then the result should be empty + And the side effects should be: + | -nodes | 2 | + | -labels | 1 | + + Scenario: Delete relationships from a map + Given an empty graph + And having executed: + """ + CREATE (a:User), (b:User) + CREATE (a)-[:R]->(b) + CREATE (b)-[:R]->(a) + """ + When executing query: + """ + MATCH (:User)-[r]->(:User) + WITH {key: r} AS rels + DELETE rels.key + """ + Then the result should be empty + And the side effects should be: + | -relationships | 2 | + + Scenario: Detach delete nodes from nested map/list + Given an empty graph + And having executed: + """ + CREATE (a:User), (b:User) + CREATE (a)-[:R]->(b) + CREATE (b)-[:R]->(a) + """ + When executing query: + """ + MATCH (u:User) + WITH {key: collect(u)} AS nodeMap + DETACH DELETE nodeMap.key[0] + """ + Then the result should be empty + And the side effects should be: + | -nodes | 1 | + | -relationships | 2 | + + Scenario: Delete relationships from nested map/list + Given an empty graph + And having executed: + """ + CREATE (a:User), (b:User) + CREATE (a)-[:R]->(b) + CREATE (b)-[:R]->(a) + """ + When executing query: + """ + MATCH (:User)-[r]->(:User) + WITH {key: {key: collect(r)}} AS rels + DELETE rels.key.key[0] + """ + Then the result should be empty + And the side effects should be: + | -relationships | 1 | + + Scenario: Delete paths from nested map/list + Given an empty graph + And having executed: + """ + CREATE (a:User), (b:User) + CREATE (a)-[:R]->(b) + CREATE (b)-[:R]->(a) + """ + When executing query: + """ + MATCH p = (:User)-[r]->(:User) + WITH {key: collect(p)} AS pathColls + DELETE pathColls.key[0], pathColls.key[1] + """ + Then the result should be empty + And the side effects should be: + | -nodes | 2 | + | -relationships | 2 | + | -labels | 1 | + + Scenario: Delete relationship with bidirectional matching + Given an empty graph + And having executed: + """ + CREATE ()-[:T {id: 42}]->() + """ + When executing query: + """ + MATCH p = ()-[r:T]-() + WHERE r.id = 42 + DELETE r + """ + Then the result should be empty + And the side effects should be: + | -relationships | 1 | + | -properties | 1 | diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/EqualsAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/EqualsAcceptance.feature new file mode 100644 index 000000000..d8565fc6e --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/EqualsAcceptance.feature @@ -0,0 +1,111 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: EqualsAcceptance + + Scenario: Number-typed integer comparison + Given an empty graph + And having executed: + """ + CREATE ({id: 0}) + """ + When executing query: + """ + WITH collect([0, 0.0]) AS numbers + UNWIND numbers AS arr + WITH arr[0] AS expected + MATCH (n) WHERE toInteger(n.id) = expected + RETURN n + """ + Then the result should be: + | n | + | ({id: 0}) | + And no side effects + + Scenario: Number-typed float comparison + Given an empty graph + And having executed: + """ + CREATE ({id: 0}) + """ + When executing query: + """ + WITH collect([0.5, 0]) AS numbers + UNWIND numbers AS arr + WITH arr[0] AS expected + MATCH (n) WHERE toInteger(n.id) = expected + RETURN n + """ + Then the result should be: + | n | + And no side effects + + Scenario: Any-typed string comparison + Given an empty graph + And having executed: + """ + CREATE ({id: 0}) + """ + When executing query: + """ + WITH collect(['0', 0]) AS things + UNWIND things AS arr + WITH arr[0] AS expected + MATCH (n) WHERE toInteger(n.id) = expected + RETURN n + """ + Then the result should be: + | n | + And no side effects + + Scenario: Comparing nodes to nodes + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (a) + WITH a + MATCH (b) + WHERE a = b + RETURN count(b) + """ + Then the result should be: + | count(b) | + | 1 | + And no side effects + + Scenario: Comparing relationships to relationships + Given an empty graph + And having executed: + """ + CREATE ()-[:T]->() + """ + When executing query: + """ + MATCH ()-[a]->() + WITH a + MATCH ()-[b]->() + WHERE a = b + RETURN count(b) + """ + Then the result should be: + | count(b) | + | 1 | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/ExpressionAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/ExpressionAcceptance.feature new file mode 100644 index 000000000..9300d5c64 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/ExpressionAcceptance.feature @@ -0,0 +1,248 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: ExpressionAcceptance + + Background: + Given any graph + + Scenario: IN should work with nested list subscripting + When executing query: + """ + WITH [[1, 2, 3]] AS list + RETURN 3 IN list[0] AS r + """ + Then the result should be: + | r | + | true | + And no side effects + + Scenario: IN should work with nested literal list subscripting + When executing query: + """ + RETURN 3 IN [[1, 2, 3]][0] AS r + """ + Then the result should be: + | r | + | true | + And no side effects + + Scenario: IN should work with list slices + When executing query: + """ + WITH [1, 2, 3] AS list + RETURN 3 IN list[0..1] AS r + """ + Then the result should be: + | r | + | false | + And no side effects + + Scenario: IN should work with literal list slices + When executing query: + """ + RETURN 3 IN [1, 2, 3][0..1] AS r + """ + Then the result should be: + | r | + | false | + And no side effects + + Scenario: Execute n[0] + When executing query: + """ + RETURN [1, 2, 3][0] AS value + """ + Then the result should be: + | value | + | 1 | + And no side effects + + Scenario: Execute n['name'] in read queries + And having executed: + """ + CREATE ({name: 'Apa'}) + """ + When executing query: + """ + MATCH (n {name: 'Apa'}) + RETURN n['nam' + 'e'] AS value + """ + Then the result should be: + | value | + | 'Apa' | + And no side effects + + Scenario: Execute n['name'] in update queries + When executing query: + """ + CREATE (n {name: 'Apa'}) + RETURN n['nam' + 'e'] AS value + """ + Then the result should be: + | value | + | 'Apa' | + And the side effects should be: + | +nodes | 1 | + | +properties | 1 | + + Scenario: Use dynamic property lookup based on parameters when there is no type information + And parameters are: + | expr | {name: 'Apa'} | + | idx | 'name' | + When executing query: + """ + WITH $expr AS expr, $idx AS idx + RETURN expr[idx] AS value + """ + Then the result should be: + | value | + | 'Apa' | + And no side effects + + Scenario: Use dynamic property lookup based on parameters when there is lhs type information + And parameters are: + | idx | 'name' | + When executing query: + """ + CREATE (n {name: 'Apa'}) + RETURN n[$idx] AS value + """ + Then the result should be: + | value | + | 'Apa' | + And the side effects should be: + | +nodes | 1 | + | +properties | 1 | + + Scenario: Use dynamic property lookup based on parameters when there is rhs type information + And parameters are: + | expr | {name: 'Apa'} | + | idx | 'name' | + When executing query: + """ + WITH $expr AS expr, $idx AS idx + RETURN expr[toString(idx)] AS value + """ + Then the result should be: + | value | + | 'Apa' | + And no side effects + + Scenario: Use collection lookup based on parameters when there is no type information + And parameters are: + | expr | ['Apa'] | + | idx | 0 | + When executing query: + """ + WITH $expr AS expr, $idx AS idx + RETURN expr[idx] AS value + """ + Then the result should be: + | value | + | 'Apa' | + And no side effects + + Scenario: Use collection lookup based on parameters when there is lhs type information + And parameters are: + | idx | 0 | + When executing query: + """ + WITH ['Apa'] AS expr + RETURN expr[$idx] AS value + """ + Then the result should be: + | value | + | 'Apa' | + And no side effects + + Scenario: Use collection lookup based on parameters when there is rhs type information + And parameters are: + | expr | ['Apa'] | + | idx | 0 | + When executing query: + """ + WITH $expr AS expr, $idx AS idx + RETURN expr[toInteger(idx)] AS value + """ + Then the result should be: + | value | + | 'Apa' | + And no side effects + + Scenario: Fail at runtime when attempting to index with an Int into a Map + And parameters are: + | expr | {name: 'Apa'} | + | idx | 0 | + When executing query: + """ + WITH $expr AS expr, $idx AS idx + RETURN expr[idx] + """ + Then a TypeError should be raised at runtime: MapElementAccessByNonString + + Scenario: Fail at runtime when trying to index into a map with a non-string + And parameters are: + | expr | {name: 'Apa'} | + | idx | 12.3 | + When executing query: + """ + WITH $expr AS expr, $idx AS idx + RETURN expr[idx] + """ + Then a TypeError should be raised at runtime: MapElementAccessByNonString + + Scenario: Fail at runtime when attempting to index with a String into a Collection + And parameters are: + | expr | ['Apa'] | + | idx | 'name' | + When executing query: + """ + WITH $expr AS expr, $idx AS idx + RETURN expr[idx] + """ + Then a TypeError should be raised at runtime: ListElementAccessByNonInteger + + Scenario: Fail at runtime when trying to index into a list with a list + And parameters are: + | expr | ['Apa'] | + | idx | ['Apa'] | + When executing query: + """ + WITH $expr AS expr, $idx AS idx + RETURN expr[idx] + """ + Then a TypeError should be raised at runtime: ListElementAccessByNonInteger + + Scenario: Fail at compile time when attempting to index with a non-integer into a list + When executing query: + """ + WITH [1, 2, 3, 4, 5] AS list, 3.14 AS idx + RETURN list[idx] + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: Fail at runtime when trying to index something which is not a map or collection + And parameters are: + | expr | 100 | + | idx | 0 | + When executing query: + """ + WITH $expr AS expr, $idx AS idx + RETURN expr[idx] + """ + Then a TypeError should be raised at runtime: InvalidElementAccess diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/FunctionsAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/FunctionsAcceptance.feature new file mode 100644 index 000000000..720be853b --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/FunctionsAcceptance.feature @@ -0,0 +1,485 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: FunctionsAcceptance + + Scenario: Run coalesce + Given an empty graph + And having executed: + """ + CREATE ({name: 'Emil Eifrem', title: 'CEO'}), ({name: 'Nobody'}) + """ + When executing query: + """ + MATCH (a) + RETURN coalesce(a.title, a.name) + """ + Then the result should be: + | coalesce(a.title, a.name) | + | 'CEO' | + | 'Nobody' | + And no side effects + + Scenario: Functions should return null if they get path containing unbound + Given any graph + When executing query: + """ + WITH null AS a + OPTIONAL MATCH p = (a)-[r]->() + RETURN length(nodes(p)), type(r), nodes(p), relationships(p) + """ + Then the result should be: + | length(nodes(p)) | type(r) | nodes(p) | relationships(p) | + | null | null | null | null | + And no side effects + + Scenario: `split()` + Given any graph + When executing query: + """ + UNWIND split('one1two', '1') AS item + RETURN count(item) AS item + """ + Then the result should be: + | item | + | 2 | + And no side effects + + Scenario: `properties()` on a node + Given an empty graph + And having executed: + """ + CREATE (n:Person {name: 'Popeye', level: 9001}) + """ + When executing query: + """ + MATCH (p:Person) + RETURN properties(p) AS m + """ + Then the result should be: + | m | + | {name: 'Popeye', level: 9001} | + And no side effects + + Scenario: `properties()` on a relationship + Given an empty graph + And having executed: + """ + CREATE (n)-[:R {name: 'Popeye', level: 9001}]->(n) + """ + When executing query: + """ + MATCH ()-[r:R]->() + RETURN properties(r) AS m + """ + Then the result should be: + | m | + | {name: 'Popeye', level: 9001} | + And no side effects + + Scenario: `properties()` on a map + Given any graph + When executing query: + """ + RETURN properties({name: 'Popeye', level: 9001}) AS m + """ + Then the result should be: + | m | + | {name: 'Popeye', level: 9001} | + And no side effects + + Scenario: `properties()` failing on an integer literal + Given any graph + When executing query: + """ + RETURN properties(1) + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: `properties()` failing on a string literal + Given any graph + When executing query: + """ + RETURN properties('Cypher') + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: `properties()` failing on a list of booleans + Given any graph + When executing query: + """ + RETURN properties([true, false]) + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: `properties()` on null + Given any graph + When executing query: + """ + RETURN properties(null) + """ + Then the result should be: + | properties(null) | + | null | + And no side effects + + Scenario: `reverse()` + Given any graph + When executing query: + """ + RETURN reverse('raksO') + """ + Then the result should be: + | reverse('raksO') | + | 'Oskar' | + And no side effects + + Scenario: `exists()` with dynamic property lookup + Given an empty graph + And having executed: + """ + CREATE (:Person {prop: 'foo'}), + (:Person) + """ + When executing query: + """ + MATCH (n:Person) + WHERE exists(n['prop']) + RETURN n + """ + Then the result should be: + | n | + | (:Person {prop: 'foo'}) | + And no side effects + + Scenario Outline: `exists()` with literal maps + Given any graph + When executing query: + """ + WITH <map> AS map + RETURN exists(map.name) AS result + """ + Then the result should be: + | result | + | <result> | + And no side effects + + Examples: + | map | result | + | {name: 'Mats', name2: 'Pontus'} | true | + | {name: null} | false | + | {notName: 0, notName2: null} | false | + + Scenario Outline: IS NOT NULL with literal maps + Given any graph + When executing query: + """ + WITH <map> AS map + RETURN map.name IS NOT NULL + """ + Then the result should be: + | map.name IS NOT NULL | + | <result> | + And no side effects + + Examples: + | map | result | + | {name: 'Mats', name2: 'Pontus'} | true | + | {name: null} | false | + | {notName: 0, notName2: null} | false | + + Scenario Outline: `percentileDisc()` + Given an empty graph + And having executed: + """ + CREATE ({prop: 10.0}), + ({prop: 20.0}), + ({prop: 30.0}) + """ + And parameters are: + | percentile | <p> | + When executing query: + """ + MATCH (n) + RETURN percentileDisc(n.prop, $percentile) AS p + """ + Then the result should be: + | p | + | <result> | + And no side effects + + Examples: + | p | result | + | 0.0 | 10.0 | + | 0.5 | 20.0 | + | 1.0 | 30.0 | + + Scenario Outline: `percentileCont()` + Given an empty graph + And having executed: + """ + CREATE ({prop: 10.0}), + ({prop: 20.0}), + ({prop: 30.0}) + """ + And parameters are: + | percentile | <p> | + When executing query: + """ + MATCH (n) + RETURN percentileCont(n.prop, $percentile) AS p + """ + Then the result should be: + | p | + | <result> | + And no side effects + + Examples: + | p | result | + | 0.0 | 10.0 | + | 0.5 | 20.0 | + | 1.0 | 30.0 | + + Scenario Outline: `percentileCont()` failing on bad arguments + Given an empty graph + And having executed: + """ + CREATE ({prop: 10.0}) + """ + And parameters are: + | param | <percentile> | + When executing query: + """ + MATCH (n) + RETURN percentileCont(n.prop, $param) + """ + Then a ArgumentError should be raised at runtime: NumberOutOfRange + + Examples: + | percentile | + | 1000 | + | -1 | + | 1.1 | + + Scenario Outline: `percentileDisc()` failing on bad arguments + Given an empty graph + And having executed: + """ + CREATE ({prop: 10.0}) + """ + And parameters are: + | param | <percentile> | + When executing query: + """ + MATCH (n) + RETURN percentileDisc(n.prop, $param) + """ + Then a ArgumentError should be raised at runtime: NumberOutOfRange + + Examples: + | percentile | + | 1000 | + | -1 | + | 1.1 | + + Scenario: `percentileDisc()` failing in more involved query + Given an empty graph + And having executed: + """ + UNWIND range(0, 10) AS i + CREATE (s:S) + WITH s, i + UNWIND range(0, i) AS j + CREATE (s)-[:REL]->() + """ + When executing query: + """ + MATCH (n:S) + WITH n, size([(n)-->() | 1]) AS deg + WHERE deg > 2 + WITH deg + LIMIT 100 + RETURN percentileDisc(0.90, deg), deg + """ + Then a ArgumentError should be raised at runtime: NumberOutOfRange + + Scenario: `type()` + Given an empty graph + And having executed: + """ + CREATE ()-[:T]->() + """ + When executing query: + """ + MATCH ()-[r]->() + RETURN type(r) + """ + Then the result should be: + | type(r) | + | 'T' | + And no side effects + + Scenario: `type()` on two relationships + Given an empty graph + And having executed: + """ + CREATE ()-[:T1]->()-[:T2]->() + """ + When executing query: + """ + MATCH ()-[r1]->()-[r2]->() + RETURN type(r1), type(r2) + """ + Then the result should be: + | type(r1) | type(r2) | + | 'T1' | 'T2' | + And no side effects + + Scenario: `type()` on null relationship + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (a) + OPTIONAL MATCH (a)-[r:NOT_THERE]->() + RETURN type(r) + """ + Then the result should be: + | type(r) | + | null | + And no side effects + + Scenario: `type()` on mixed null and non-null relationships + Given an empty graph + And having executed: + """ + CREATE ()-[:T]->() + """ + When executing query: + """ + MATCH (a) + OPTIONAL MATCH (a)-[r:T]->() + RETURN type(r) + """ + Then the result should be: + | type(r) | + | 'T' | + | null | + And no side effects + + Scenario: `type()` handling Any type + Given an empty graph + And having executed: + """ + CREATE ()-[:T]->() + """ + When executing query: + """ + MATCH (a)-[r]->() + WITH [r, 1] AS list + RETURN type(list[0]) + """ + Then the result should be: + | type(list[0]) | + | 'T' | + And no side effects + + Scenario Outline: `type()` failing on invalid arguments + Given an empty graph + And having executed: + """ + CREATE ()-[:T]->() + """ + When executing query: + """ + MATCH p = (n)-[r:T]->() + RETURN [x IN [r, <invalid>] | type(x) ] AS list + """ + Then a TypeError should be raised at runtime: InvalidArgumentValue + + Examples: + | invalid | + | 0 | + | 1.0 | + | true | + | '' | + | [] | + + Scenario: `labels()` should accept type Any + Given an empty graph + And having executed: + """ + CREATE (:Foo), (:Foo:Bar) + """ + When executing query: + """ + MATCH (a) + WITH [a, 1] AS list + RETURN labels(list[0]) AS l + """ + Then the result should be (ignoring element order for lists): + | l | + | ['Foo'] | + | ['Foo', 'Bar'] | + And no side effects + + Scenario: `labels()` failing on a path + Given an empty graph + And having executed: + """ + CREATE (:Foo), (:Foo:Bar) + """ + When executing query: + """ + MATCH p = (a) + RETURN labels(p) AS l + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: `labels()` failing on invalid arguments + Given an empty graph + And having executed: + """ + CREATE (:Foo), (:Foo:Bar) + """ + When executing query: + """ + MATCH (a) + WITH [a, 1] AS list + RETURN labels(list[1]) AS l + """ + Then a TypeError should be raised at runtime: InvalidArgumentValue + + Scenario: `exists()` is case insensitive + Given an empty graph + And having executed: + """ + CREATE (a:X {prop: 42}), (:X) + """ + When executing query: + """ + MATCH (n:X) + RETURN n, EXIsTS(n.prop) AS b + """ + Then the result should be: + | n | b | + | (:X {prop: 42}) | true | + | (:X) | false | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/JoinAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/JoinAcceptance.feature new file mode 100644 index 000000000..a3989714e --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/JoinAcceptance.feature @@ -0,0 +1,66 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: ValueHashJoinAcceptance + + Scenario: Find friends of others + Given an empty graph + And having executed: + """ + CREATE (:A {id: 1}), + (:A {id: 2}), + (:B {id: 2}), + (:B {id: 3}) + """ + When executing query: + """ + MATCH (a:A), (b:B) + WHERE a.id = b.id + RETURN a, b + """ + Then the result should be: + | a | b | + | (:A {id: 2}) | (:B {id: 2}) | + And no side effects + + Scenario: Should only join when matching + Given an empty graph + And having executed: + """ + UNWIND range(0, 1000) AS i + CREATE (:A {id: i}) + MERGE (:B {id: i % 10}) + """ + When executing query: + """ + MATCH (a:A), (b:B) + WHERE a.id = b.id + RETURN a, b + """ + Then the result should be: + | a | b | + | (:A {id: 0}) | (:B {id: 0}) | + | (:A {id: 1}) | (:B {id: 1}) | + | (:A {id: 2}) | (:B {id: 2}) | + | (:A {id: 3}) | (:B {id: 3}) | + | (:A {id: 4}) | (:B {id: 4}) | + | (:A {id: 5}) | (:B {id: 5}) | + | (:A {id: 6}) | (:B {id: 6}) | + | (:A {id: 7}) | (:B {id: 7}) | + | (:A {id: 8}) | (:B {id: 8}) | + | (:A {id: 9}) | (:B {id: 9}) | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/KeysAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/KeysAcceptance.feature new file mode 100644 index 000000000..c4e26fb56 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/KeysAcceptance.feature @@ -0,0 +1,163 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: KeysAcceptance + + Scenario: Using `keys()` on a single node, non-empty result + Given an empty graph + And having executed: + """ + CREATE ({name: 'Andres', surname: 'Lopez'}) + """ + When executing query: + """ + MATCH (n) + UNWIND keys(n) AS x + RETURN DISTINCT x AS theProps + """ + Then the result should be: + | theProps | + | 'name' | + | 'surname' | + And no side effects + + Scenario: Using `keys()` on multiple nodes, non-empty result + Given an empty graph + And having executed: + """ + CREATE ({name: 'Andres', surname: 'Lopez'}), + ({otherName: 'Andres', otherSurname: 'Lopez'}) + """ + When executing query: + """ + MATCH (n) + UNWIND keys(n) AS x + RETURN DISTINCT x AS theProps + """ + Then the result should be: + | theProps | + | 'name' | + | 'surname' | + | 'otherName' | + | 'otherSurname' | + And no side effects + + Scenario: Using `keys()` on a single node, empty result + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + UNWIND keys(n) AS x + RETURN DISTINCT x AS theProps + """ + Then the result should be: + | theProps | + And no side effects + + Scenario: Using `keys()` on an optionally matched node + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + OPTIONAL MATCH (n) + UNWIND keys(n) AS x + RETURN DISTINCT x AS theProps + """ + Then the result should be: + | theProps | + And no side effects + + Scenario: Using `keys()` on a relationship, non-empty result + Given an empty graph + And having executed: + """ + CREATE ()-[:KNOWS {level: 'bad', year: '2015'}]->() + """ + When executing query: + """ + MATCH ()-[r:KNOWS]-() + UNWIND keys(r) AS x + RETURN DISTINCT x AS theProps + """ + Then the result should be: + | theProps | + | 'level' | + | 'year' | + And no side effects + + Scenario: Using `keys()` on a relationship, empty result + Given an empty graph + And having executed: + """ + CREATE ()-[:KNOWS]->() + """ + When executing query: + """ + MATCH ()-[r:KNOWS]-() + UNWIND keys(r) AS x + RETURN DISTINCT x AS theProps + """ + Then the result should be: + | theProps | + And no side effects + + Scenario: Using `keys()` on an optionally matched relationship + Given an empty graph + And having executed: + """ + CREATE ()-[:KNOWS]->() + """ + When executing query: + """ + OPTIONAL MATCH ()-[r:KNOWS]-() + UNWIND keys(r) AS x + RETURN DISTINCT x AS theProps + """ + Then the result should be: + | theProps | + And no side effects + + Scenario: Using `keys()` on a literal map + Given any graph + When executing query: + """ + RETURN keys({name: 'Alice', age: 38, address: {city: 'London', residential: true}}) AS k + """ + Then the result should be: + | k | + | ['name', 'age', 'address'] | + And no side effects + + Scenario: Using `keys()` on a parameter map + Given any graph + And parameters are: + | param | {name: 'Alice', age: 38, address: {city: 'London', residential: true}} | + When executing query: + """ + RETURN keys($param) AS k + """ + Then the result should be (ignoring element order for lists): + | k | + | ['address', 'name', 'age'] | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/LabelsAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/LabelsAcceptance.feature new file mode 100644 index 000000000..2133e614a --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/LabelsAcceptance.feature @@ -0,0 +1,247 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: LabelsAcceptance + + Background: + Given an empty graph + + Scenario: Adding a single label + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + SET n:Foo + RETURN labels(n) + """ + Then the result should be: + | labels(n) | + | ['Foo'] | + And the side effects should be: + | +labels | 1 | + + Scenario: Ignore space before colon + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + SET n :Foo + RETURN labels(n) + """ + Then the result should be: + | labels(n) | + | ['Foo'] | + And the side effects should be: + | +labels | 1 | + + Scenario: Adding multiple labels + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + SET n:Foo:Bar + RETURN labels(n) + """ + Then the result should be: + | labels(n) | + | ['Foo', 'Bar'] | + And the side effects should be: + | +labels | 2 | + + Scenario: Ignoring intermediate whitespace 1 + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + SET n :Foo :Bar + RETURN labels(n) + """ + Then the result should be: + | labels(n) | + | ['Foo', 'Bar'] | + And the side effects should be: + | +labels | 2 | + + Scenario: Ignoring intermediate whitespace 2 + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + SET n :Foo:Bar + RETURN labels(n) + """ + Then the result should be: + | labels(n) | + | ['Foo', 'Bar'] | + And the side effects should be: + | +labels | 2 | + + Scenario: Creating node without label + When executing query: + """ + CREATE (node) + RETURN labels(node) + """ + Then the result should be: + | labels(node) | + | [] | + And the side effects should be: + | +nodes | 1 | + + Scenario: Creating node with two labels + When executing query: + """ + CREATE (node:Foo:Bar {name: 'Mattias'}) + RETURN labels(node) + """ + Then the result should be: + | labels(node) | + | ['Foo', 'Bar'] | + And the side effects should be: + | +nodes | 1 | + | +labels | 2 | + | +properties | 1 | + + Scenario: Ignore space when creating node with labels + When executing query: + """ + CREATE (node :Foo:Bar) + RETURN labels(node) + """ + Then the result should be: + | labels(node) | + | ['Foo', 'Bar'] | + And the side effects should be: + | +nodes | 1 | + | +labels | 2 | + + Scenario: Create node with label in pattern + When executing query: + """ + CREATE (n:Person)-[:OWNS]->(:Dog) + RETURN labels(n) + """ + Then the result should be: + | labels(n) | + | ['Person'] | + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + | +labels | 2 | + + Scenario: Fail when adding a new label predicate on a node that is already bound 1 + When executing query: + """ + CREATE (n:Foo)-[:T1]->(), + (n:Bar)-[:T2]->() + """ + Then a SyntaxError should be raised at compile time: VariableAlreadyBound + + Scenario: Fail when adding new label predicate on a node that is already bound 2 + When executing query: + """ + CREATE ()<-[:T2]-(n:Foo), + (n:Bar)<-[:T1]-() + """ + Then a SyntaxError should be raised at compile time: VariableAlreadyBound + + Scenario: Fail when adding new label predicate on a node that is already bound 3 + When executing query: + """ + CREATE (n:Foo) + CREATE (n:Bar)-[:OWNS]->(:Dog) + """ + Then a SyntaxError should be raised at compile time: VariableAlreadyBound + + Scenario: Fail when adding new label predicate on a node that is already bound 4 + When executing query: + """ + CREATE (n {}) + CREATE (n:Bar)-[:OWNS]->(:Dog) + """ + Then a SyntaxError should be raised at compile time: VariableAlreadyBound + + Scenario: Fail when adding new label predicate on a node that is already bound 5 + When executing query: + """ + CREATE (n:Foo) + CREATE (n {})-[:OWNS]->(:Dog) + """ + Then a SyntaxError should be raised at compile time: VariableAlreadyBound + + Scenario: Using `labels()` in return clauses + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + RETURN labels(n) + """ + Then the result should be: + | labels(n) | + | [] | + And no side effects + + Scenario: Removing a label + And having executed: + """ + CREATE (:Foo:Bar) + """ + When executing query: + """ + MATCH (n) + REMOVE n:Foo + RETURN labels(n) + """ + Then the result should be: + | labels(n) | + | ['Bar'] | + And the side effects should be: + | -labels | 1 | + + Scenario: Removing a non-existent label + And having executed: + """ + CREATE (:Foo) + """ + When executing query: + """ + MATCH (n) + REMOVE n:Bar + RETURN labels(n) + """ + Then the result should be: + | labels(n) | + | ['Foo'] | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/LargeCreateQuery.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/LargeCreateQuery.feature new file mode 100644 index 000000000..951d8f95b --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/LargeCreateQuery.feature @@ -0,0 +1,1361 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: LargeCreateQuery + + Scenario: Generate the movie graph correctly + Given an empty graph + When executing query: + """ + CREATE (theMatrix:Movie {title: 'The Matrix', released: 1999, tagline: 'Welcome to the Real World'}) + CREATE (keanu:Person {name: 'Keanu Reeves', born: 1964}) + CREATE (carrie:Person {name: 'Carrie-Anne Moss', born: 1967}) + CREATE (laurence:Person {name: 'Laurence Fishburne', born: 1961}) + CREATE (hugo:Person {name: 'Hugo Weaving', born: 1960}) + CREATE (andyW:Person {name: 'Andy Wachowski', born: 1967}) + CREATE (lanaW:Person {name: 'Lana Wachowski', born: 1965}) + CREATE (joelS:Person {name: 'Joel Silver', born: 1952}) + CREATE + (keanu)-[:ACTED_IN {roles: ['Neo']}]->(theMatrix), + (carrie)-[:ACTED_IN {roles: ['Trinity']}]->(theMatrix), + (laurence)-[:ACTED_IN {roles: ['Morpheus']}]->(theMatrix), + (hugo)-[:ACTED_IN {roles: ['Agent Smith']}]->(theMatrix), + (andyW)-[:DIRECTED]->(theMatrix), + (lanaW)-[:DIRECTED]->(theMatrix), + (joelS)-[:PRODUCED]->(theMatrix) + + CREATE (emil:Person {name: 'Emil Eifrem', born: 1978}) + CREATE (emil)-[:ACTED_IN {roles: ['Emil']}]->(theMatrix) + + CREATE (theMatrixReloaded:Movie {title: 'The Matrix Reloaded', released: 2003, + tagline: 'Free your mind'}) + CREATE + (keanu)-[:ACTED_IN {roles: ['Neo'] }]->(theMatrixReloaded), + (carrie)-[:ACTED_IN {roles: ['Trinity']}]->(theMatrixReloaded), + (laurence)-[:ACTED_IN {roles: ['Morpheus']}]->(theMatrixReloaded), + (hugo)-[:ACTED_IN {roles: ['Agent Smith']}]->(theMatrixReloaded), + (andyW)-[:DIRECTED]->(theMatrixReloaded), + (lanaW)-[:DIRECTED]->(theMatrixReloaded), + (joelS)-[:PRODUCED]->(theMatrixReloaded) + + CREATE (theMatrixRevolutions:Movie {title: 'The Matrix Revolutions', released: 2003, + tagline: 'Everything that has a beginning has an end'}) + CREATE + (keanu)-[:ACTED_IN {roles: ['Neo']}]->(theMatrixRevolutions), + (carrie)-[:ACTED_IN {roles: ['Trinity']}]->(theMatrixRevolutions), + (laurence)-[:ACTED_IN {roles: ['Morpheus']}]->(theMatrixRevolutions), + (hugo)-[:ACTED_IN {roles: ['Agent Smith']}]->(theMatrixRevolutions), + (andyW)-[:DIRECTED]->(theMatrixRevolutions), + (lanaW)-[:DIRECTED]->(theMatrixRevolutions), + (joelS)-[:PRODUCED]->(theMatrixRevolutions) + + CREATE (theDevilsAdvocate:Movie {title: 'The Devil\'s Advocate', released: 1997, + tagline: 'Evil has its winning ways'}) + CREATE (charlize:Person {name: 'Charlize Theron', born: 1975}) + CREATE (al:Person {name: 'Al Pacino', born: 1940}) + CREATE (taylor:Person {name: 'Taylor Hackford', born: 1944}) + CREATE + (keanu)-[:ACTED_IN {roles: ['Kevin Lomax']}]->(theDevilsAdvocate), + (charlize)-[:ACTED_IN {roles: ['Mary Ann Lomax']}]->(theDevilsAdvocate), + (al)-[:ACTED_IN {roles: ['John Milton']}]->(theDevilsAdvocate), + (taylor)-[:DIRECTED]->(theDevilsAdvocate) + + CREATE (aFewGoodMen:Movie {title: 'A Few Good Men', released: 1992, + tagline: 'Deep within the heart of the nation\'s capital, one man will stop at nothing to keep his honor, ...'}) + CREATE (tomC:Person {name: 'Tom Cruise', born: 1962}) + CREATE (jackN:Person {name: 'Jack Nicholson', born: 1937}) + CREATE (demiM:Person {name: 'Demi Moore', born: 1962}) + CREATE (kevinB:Person {name: 'Kevin Bacon', born: 1958}) + CREATE (kieferS:Person {name: 'Kiefer Sutherland', born: 1966}) + CREATE (noahW:Person {name: 'Noah Wyle', born: 1971}) + CREATE (cubaG:Person {name: 'Cuba Gooding Jr.', born: 1968}) + CREATE (kevinP:Person {name: 'Kevin Pollak', born: 1957}) + CREATE (jTW:Person {name: 'J.T. Walsh', born: 1943}) + CREATE (jamesM:Person {name: 'James Marshall', born: 1967}) + CREATE (christopherG:Person {name: 'Christopher Guest', born: 1948}) + CREATE (robR:Person {name: 'Rob Reiner', born: 1947}) + CREATE (aaronS:Person {name: 'Aaron Sorkin', born: 1961}) + CREATE + (tomC)-[:ACTED_IN {roles: ['Lt. Daniel Kaffee']}]->(aFewGoodMen), + (jackN)-[:ACTED_IN {roles: ['Col. Nathan R. Jessup']}]->(aFewGoodMen), + (demiM)-[:ACTED_IN {roles: ['Lt. Cdr. JoAnne Galloway']}]->(aFewGoodMen), + (kevinB)-[:ACTED_IN {roles: ['Capt. Jack Ross']}]->(aFewGoodMen), + (kieferS)-[:ACTED_IN {roles: ['Lt. Jonathan Kendrick']}]->(aFewGoodMen), + (noahW)-[:ACTED_IN {roles: ['Cpl. Jeffrey Barnes']}]->(aFewGoodMen), + (cubaG)-[:ACTED_IN {roles: ['Cpl. Carl Hammaker']}]->(aFewGoodMen), + (kevinP)-[:ACTED_IN {roles: ['Lt. Sam Weinberg']}]->(aFewGoodMen), + (jTW)-[:ACTED_IN {roles: ['Lt. Col. Matthew Andrew Markinson']}]->(aFewGoodMen), + (jamesM)-[:ACTED_IN {roles: ['Pfc. Louden Downey']}]->(aFewGoodMen), + (christopherG)-[:ACTED_IN {roles: ['Dr. Stone']}]->(aFewGoodMen), + (aaronS)-[:ACTED_IN {roles: ['Bar patron']}]->(aFewGoodMen), + (robR)-[:DIRECTED]->(aFewGoodMen), + (aaronS)-[:WROTE]->(aFewGoodMen) + + CREATE (topGun:Movie {title: 'Top Gun', released: 1986, + tagline: 'I feel the need, the need for speed.'}) + CREATE (kellyM:Person {name: 'Kelly McGillis', born: 1957}) + CREATE (valK:Person {name: 'Val Kilmer', born: 1959}) + CREATE (anthonyE:Person {name: 'Anthony Edwards', born: 1962}) + CREATE (tomS:Person {name: 'Tom Skerritt', born: 1933}) + CREATE (megR:Person {name: 'Meg Ryan', born: 1961}) + CREATE (tonyS:Person {name: 'Tony Scott', born: 1944}) + CREATE (jimC:Person {name: 'Jim Cash', born: 1941}) + CREATE + (tomC)-[:ACTED_IN {roles: ['Maverick']}]->(topGun), + (kellyM)-[:ACTED_IN {roles: ['Charlie']}]->(topGun), + (valK)-[:ACTED_IN {roles: ['Iceman']}]->(topGun), + (anthonyE)-[:ACTED_IN {roles: ['Goose']}]->(topGun), + (tomS)-[:ACTED_IN {roles: ['Viper']}]->(topGun), + (megR)-[:ACTED_IN {roles: ['Carole']}]->(topGun), + (tonyS)-[:DIRECTED]->(topGun), + (jimC)-[:WROTE]->(topGun) + + CREATE (jerryMaguire:Movie {title: 'Jerry Maguire', released: 2000, + tagline: 'The rest of his life begins now.'}) + CREATE (reneeZ:Person {name: 'Renee Zellweger', born: 1969}) + CREATE (kellyP:Person {name: 'Kelly Preston', born: 1962}) + CREATE (jerryO:Person {name: 'Jerry O\'Connell', born: 1974}) + CREATE (jayM:Person {name: 'Jay Mohr', born: 1970}) + CREATE (bonnieH:Person {name: 'Bonnie Hunt', born: 1961}) + CREATE (reginaK:Person {name: 'Regina King', born: 1971}) + CREATE (jonathanL:Person {name: 'Jonathan Lipnicki', born: 1996}) + CREATE (cameronC:Person {name: 'Cameron Crowe', born: 1957}) + CREATE + (tomC)-[:ACTED_IN {roles: ['Jerry Maguire']}]->(jerryMaguire), + (cubaG)-[:ACTED_IN {roles: ['Rod Tidwell']}]->(jerryMaguire), + (reneeZ)-[:ACTED_IN {roles: ['Dorothy Boyd']}]->(jerryMaguire), + (kellyP)-[:ACTED_IN {roles: ['Avery Bishop']}]->(jerryMaguire), + (jerryO)-[:ACTED_IN {roles: ['Frank Cushman']}]->(jerryMaguire), + (jayM)-[:ACTED_IN {roles: ['Bob Sugar']}]->(jerryMaguire), + (bonnieH)-[:ACTED_IN {roles: ['Laurel Boyd']}]->(jerryMaguire), + (reginaK)-[:ACTED_IN {roles: ['Marcee Tidwell']}]->(jerryMaguire), + (jonathanL)-[:ACTED_IN {roles: ['Ray Boyd']}]->(jerryMaguire), + (cameronC)-[:DIRECTED]->(jerryMaguire), + (cameronC)-[:PRODUCED]->(jerryMaguire), + (cameronC)-[:WROTE]->(jerryMaguire) + + CREATE (standByMe:Movie {title: 'Stand-By-Me', released: 1986, + tagline: 'The last real taste of innocence'}) + CREATE (riverP:Person {name: 'River Phoenix', born: 1970}) + CREATE (coreyF:Person {name: 'Corey Feldman', born: 1971}) + CREATE (wilW:Person {name: 'Wil Wheaton', born: 1972}) + CREATE (johnC:Person {name: 'John Cusack', born: 1966}) + CREATE (marshallB:Person {name: 'Marshall Bell', born: 1942}) + CREATE + (wilW)-[:ACTED_IN {roles: ['Gordie Lachance']}]->(standByMe), + (riverP)-[:ACTED_IN {roles: ['Chris Chambers']}]->(standByMe), + (jerryO)-[:ACTED_IN {roles: ['Vern Tessio']}]->(standByMe), + (coreyF)-[:ACTED_IN {roles: ['Teddy Duchamp']}]->(standByMe), + (johnC)-[:ACTED_IN {roles: ['Denny Lachance']}]->(standByMe), + (kieferS)-[:ACTED_IN {roles: ['Ace Merrill']}]->(standByMe), + (marshallB)-[:ACTED_IN {roles: ['Mr. Lachance']}]->(standByMe), + (robR)-[:DIRECTED]->(standByMe) + + CREATE (asGoodAsItGets:Movie {title: 'As-good-as-it-gets', released: 1997, + tagline: 'A comedy from the heart that goes for the throat'}) + CREATE (helenH:Person {name: 'Helen Hunt', born: 1963}) + CREATE (gregK:Person {name: 'Greg Kinnear', born: 1963}) + CREATE (jamesB:Person {name: 'James L. Brooks', born: 1940}) + CREATE + (jackN)-[:ACTED_IN {roles: ['Melvin Udall']}]->(asGoodAsItGets), + (helenH)-[:ACTED_IN {roles: ['Carol Connelly']}]->(asGoodAsItGets), + (gregK)-[:ACTED_IN {roles: ['Simon Bishop']}]->(asGoodAsItGets), + (cubaG)-[:ACTED_IN {roles: ['Frank Sachs']}]->(asGoodAsItGets), + (jamesB)-[:DIRECTED]->(asGoodAsItGets) + + CREATE (whatDreamsMayCome:Movie {title: 'What Dreams May Come', released: 1998, + tagline: 'After life there is more. The end is just the beginning.'}) + CREATE (annabellaS:Person {name: 'Annabella Sciorra', born: 1960}) + CREATE (maxS:Person {name: 'Max von Sydow', born: 1929}) + CREATE (wernerH:Person {name: 'Werner Herzog', born: 1942}) + CREATE (robin:Person {name: 'Robin Williams', born: 1951}) + CREATE (vincentW:Person {name: 'Vincent Ward', born: 1956}) + CREATE + (robin)-[:ACTED_IN {roles: ['Chris Nielsen']}]->(whatDreamsMayCome), + (cubaG)-[:ACTED_IN {roles: ['Albert Lewis']}]->(whatDreamsMayCome), + (annabellaS)-[:ACTED_IN {roles: ['Annie Collins-Nielsen']}]->(whatDreamsMayCome), + (maxS)-[:ACTED_IN {roles: ['The Tracker']}]->(whatDreamsMayCome), + (wernerH)-[:ACTED_IN {roles: ['The Face']}]->(whatDreamsMayCome), + (vincentW)-[:DIRECTED]->(whatDreamsMayCome) + + CREATE (snowFallingonCedars:Movie {title: 'Snow-Falling-on-Cedars', released: 1999, + tagline: 'First loves last. Forever.'}) + CREATE (ethanH:Person {name: 'Ethan Hawke', born: 1970}) + CREATE (rickY:Person {name: 'Rick Yune', born: 1971}) + CREATE (jamesC:Person {name: 'James Cromwell', born: 1940}) + CREATE (scottH:Person {name: 'Scott Hicks', born: 1953}) + CREATE + (ethanH)-[:ACTED_IN {roles: ['Ishmael Chambers']}]->(snowFallingonCedars), + (rickY)-[:ACTED_IN {roles: ['Kazuo Miyamoto']}]->(snowFallingonCedars), + (maxS)-[:ACTED_IN {roles: ['Nels Gudmundsson']}]->(snowFallingonCedars), + (jamesC)-[:ACTED_IN {roles: ['Judge Fielding']}]->(snowFallingonCedars), + (scottH)-[:DIRECTED]->(snowFallingonCedars) + + CREATE (youveGotMail:Movie {title: 'You\'ve Got Mail', released: 1998, + tagline: 'At-odds-in-life, in-love-on-line'}) + CREATE (parkerP:Person {name: 'Parker Posey', born: 1968}) + CREATE (daveC:Person {name: 'Dave Chappelle', born: 1973}) + CREATE (steveZ:Person {name: 'Steve Zahn', born: 1967}) + CREATE (tomH:Person {name: 'Tom Hanks', born: 1956}) + CREATE (noraE:Person {name: 'Nora Ephron', born: 1941}) + CREATE + (tomH)-[:ACTED_IN {roles: ['Joe Fox']}]->(youveGotMail), + (megR)-[:ACTED_IN {roles: ['Kathleen Kelly']}]->(youveGotMail), + (gregK)-[:ACTED_IN {roles: ['Frank Navasky']}]->(youveGotMail), + (parkerP)-[:ACTED_IN {roles: ['Patricia Eden']}]->(youveGotMail), + (daveC)-[:ACTED_IN {roles: ['Kevin Jackson']}]->(youveGotMail), + (steveZ)-[:ACTED_IN {roles: ['George Pappas']}]->(youveGotMail), + (noraE)-[:DIRECTED]->(youveGotMail) + + CREATE (sleeplessInSeattle:Movie {title: 'Sleepless-in-Seattle', released: 1993, + tagline: 'What if someone you never met, someone you never saw, someone you never knew was the only someone for you?'}) + CREATE (ritaW:Person {name: 'Rita Wilson', born: 1956}) + CREATE (billPull:Person {name: 'Bill Pullman', born: 1953}) + CREATE (victorG:Person {name: 'Victor Garber', born: 1949}) + CREATE (rosieO:Person {name: 'Rosie O\'Donnell', born: 1962}) + CREATE + (tomH)-[:ACTED_IN {roles: ['Sam Baldwin']}]->(sleeplessInSeattle), + (megR)-[:ACTED_IN {roles: ['Annie Reed']}]->(sleeplessInSeattle), + (ritaW)-[:ACTED_IN {roles: ['Suzy']}]->(sleeplessInSeattle), + (billPull)-[:ACTED_IN {roles: ['Walter']}]->(sleeplessInSeattle), + (victorG)-[:ACTED_IN {roles: ['Greg']}]->(sleeplessInSeattle), + (rosieO)-[:ACTED_IN {roles: ['Becky']}]->(sleeplessInSeattle), + (noraE)-[:DIRECTED]->(sleeplessInSeattle) + + CREATE (joeVersustheVolcano:Movie {title: 'Joe-Versus-the-Volcano', released: 1990, + tagline: 'A story of love'}) + CREATE (johnS:Person {name: 'John Patrick Stanley', born: 1950}) + CREATE (nathan:Person {name: 'Nathan Lane', born: 1956}) + CREATE + (tomH)-[:ACTED_IN {roles: ['Joe Banks']}]->(joeVersustheVolcano), + (megR)-[:ACTED_IN {roles: ['DeDe', 'Angelica Graynamore', 'Patricia Graynamore']}]->(joeVersustheVolcano), + (nathan)-[:ACTED_IN {roles: ['Baw']}]->(joeVersustheVolcano), + (johnS)-[:DIRECTED]->(joeVersustheVolcano) + + CREATE (whenHarryMetSally:Movie {title: 'When-Harry-Met-Sally', released: 1998, + tagline: 'When-Harry-Met-Sally'}) + CREATE (billyC:Person {name: 'Billy Crystal', born: 1948}) + CREATE (carrieF:Person {name: 'Carrie Fisher', born: 1956}) + CREATE (brunoK:Person {name: 'Bruno Kirby', born: 1949}) + CREATE + (billyC)-[:ACTED_IN {roles: ['Harry Burns']}]->(whenHarryMetSally), + (megR)-[:ACTED_IN {roles: ['Sally Albright']}]->(whenHarryMetSally), + (carrieF)-[:ACTED_IN {roles: ['Marie']}]->(whenHarryMetSally), + (brunoK)-[:ACTED_IN {roles: ['Jess']}]->(whenHarryMetSally), + (robR)-[:DIRECTED]->(whenHarryMetSally), + (robR)-[:PRODUCED]->(whenHarryMetSally), + (noraE)-[:PRODUCED]->(whenHarryMetSally), + (noraE)-[:WROTE]->(whenHarryMetSally) + + CREATE (thatThingYouDo:Movie {title: 'That-Thing-You-Do', released: 1996, + tagline: 'There comes a time...'}) + CREATE (livT:Person {name: 'Liv Tyler', born: 1977}) + CREATE + (tomH)-[:ACTED_IN {roles: ['Mr. White']}]->(thatThingYouDo), + (livT)-[:ACTED_IN {roles: ['Faye Dolan']}]->(thatThingYouDo), + (charlize)-[:ACTED_IN {roles: ['Tina']}]->(thatThingYouDo), + (tomH)-[:DIRECTED]->(thatThingYouDo) + + CREATE (theReplacements:Movie {title: 'The Replacements', released: 2000, + tagline: 'Pain heals, Chicks dig scars... Glory lasts forever'}) + CREATE (brooke:Person {name: 'Brooke Langton', born: 1970}) + CREATE (gene:Person {name: 'Gene Hackman', born: 1930}) + CREATE (orlando:Person {name: 'Orlando Jones', born: 1968}) + CREATE (howard:Person {name: 'Howard Deutch', born: 1950}) + CREATE + (keanu)-[:ACTED_IN {roles: ['Shane Falco']}]->(theReplacements), + (brooke)-[:ACTED_IN {roles: ['Annabelle Farrell']}]->(theReplacements), + (gene)-[:ACTED_IN {roles: ['Jimmy McGinty']}]->(theReplacements), + (orlando)-[:ACTED_IN {roles: ['Clifford Franklin']}]->(theReplacements), + (howard)-[:DIRECTED]->(theReplacements) + + CREATE (rescueDawn:Movie {title: 'RescueDawn', released: 2006, + tagline: 'The extraordinary true story'}) + CREATE (christianB:Person {name: 'Christian Bale', born: 1974}) + CREATE (zachG:Person {name: 'Zach Grenier', born: 1954}) + CREATE + (marshallB)-[:ACTED_IN {roles: ['Admiral']}]->(rescueDawn), + (christianB)-[:ACTED_IN {roles: ['Dieter Dengler']}]->(rescueDawn), + (zachG)-[:ACTED_IN {roles: ['Squad Leader']}]->(rescueDawn), + (steveZ)-[:ACTED_IN {roles: ['Duane']}]->(rescueDawn), + (wernerH)-[:DIRECTED]->(rescueDawn) + + CREATE (theBirdcage:Movie {title: 'The-Birdcage', released: 1996, tagline: 'Come-as-you-are'}) + CREATE (mikeN:Person {name: 'Mike Nichols', born: 1931}) + CREATE + (robin)-[:ACTED_IN {roles: ['Armand Goldman']}]->(theBirdcage), + (nathan)-[:ACTED_IN {roles: ['Albert Goldman']}]->(theBirdcage), + (gene)-[:ACTED_IN {roles: ['Sen. Kevin Keeley']}]->(theBirdcage), + (mikeN)-[:DIRECTED]->(theBirdcage) + + CREATE (unforgiven:Movie {title: 'Unforgiven', released: 1992, + tagline: 'It\'s a hell of a thing, killing a man'}) + CREATE (richardH:Person {name: 'Richard Harris', born: 1930}) + CREATE (clintE:Person {name: 'Clint Eastwood', born: 1930}) + CREATE + (richardH)-[:ACTED_IN {roles: ['English Bob']}]->(unforgiven), + (clintE)-[:ACTED_IN {roles: ['Bill Munny']}]->(unforgiven), + (gene)-[:ACTED_IN {roles: ['Little Bill Daggett']}]->(unforgiven), + (clintE)-[:DIRECTED]->(unforgiven) + + CREATE (johnnyMnemonic:Movie {title: 'Johnny-Mnemonic', released: 1995, + tagline: 'The-hottest-data-in-the-coolest-head'}) + CREATE (takeshi:Person {name: 'Takeshi Kitano', born: 1947}) + CREATE (dina:Person {name: 'Dina Meyer', born: 1968}) + CREATE (iceT:Person {name: 'Ice-T', born: 1958}) + CREATE (robertL:Person {name: 'Robert Longo', born: 1953}) + CREATE + (keanu)-[:ACTED_IN {roles: ['Johnny Mnemonic']}]->(johnnyMnemonic), + (takeshi)-[:ACTED_IN {roles: ['Takahashi']}]->(johnnyMnemonic), + (dina)-[:ACTED_IN {roles: ['Jane']}]->(johnnyMnemonic), + (iceT)-[:ACTED_IN {roles: ['J-Bone']}]->(johnnyMnemonic), + (robertL)-[:DIRECTED]->(johnnyMnemonic) + + CREATE (cloudAtlas:Movie {title: 'Cloud Atlas', released: 2012, tagline: 'Everything is connected'}) + CREATE (halleB:Person {name: 'Halle Berry', born: 1966}) + CREATE (jimB:Person {name: 'Jim Broadbent', born: 1949}) + CREATE (tomT:Person {name: 'Tom Tykwer', born: 1965}) + CREATE (davidMitchell:Person {name: 'David Mitchell', born: 1969}) + CREATE (stefanArndt:Person {name: 'Stefan Arndt', born: 1961}) + CREATE + (tomH)-[:ACTED_IN {roles: ['Zachry', 'Dr. Henry Goose', 'Isaac Sachs', 'Dermot Hoggins']}]->(cloudAtlas), + (hugo)-[:ACTED_IN {roles: ['Bill Smoke', 'Haskell Moore', 'Tadeusz Kesselring', 'Nurse Noakes', 'Boardman Mephi', 'Old Georgie']}]->(cloudAtlas), + (halleB)-[:ACTED_IN {roles: ['Luisa Rey', 'Jocasta Ayrs', 'Ovid', 'Meronym']}]->(cloudAtlas), + (jimB)-[:ACTED_IN {roles: ['Vyvyan Ayrs', 'Captain Molyneux', 'Timothy Cavendish']}]->(cloudAtlas), + (tomT)-[:DIRECTED]->(cloudAtlas), + (andyW)-[:DIRECTED]->(cloudAtlas), + (lanaW)-[:DIRECTED]->(cloudAtlas), + (davidMitchell)-[:WROTE]->(cloudAtlas), + (stefanArndt)-[:PRODUCED]->(cloudAtlas) + + CREATE (theDaVinciCode:Movie {title: 'The Da Vinci Code', released: 2006, tagline: 'Break The Codes'}) + CREATE (ianM:Person {name: 'Ian McKellen', born: 1939}) + CREATE (audreyT:Person {name: 'Audrey Tautou', born: 1976}) + CREATE (paulB:Person {name: 'Paul Bettany', born: 1971}) + CREATE (ronH:Person {name: 'Ron Howard', born: 1954}) + CREATE + (tomH)-[:ACTED_IN {roles: ['Dr. Robert Langdon']}]->(theDaVinciCode), + (ianM)-[:ACTED_IN {roles: ['Sir Leight Teabing']}]->(theDaVinciCode), + (audreyT)-[:ACTED_IN {roles: ['Sophie Neveu']}]->(theDaVinciCode), + (paulB)-[:ACTED_IN {roles: ['Silas']}]->(theDaVinciCode), + (ronH)-[:DIRECTED]->(theDaVinciCode) + + CREATE (vforVendetta:Movie {title: 'V for Vendetta', released: 2006, tagline: 'Freedom! Forever!'}) + CREATE (natalieP:Person {name: 'Natalie Portman', born: 1981}) + CREATE (stephenR:Person {name: 'Stephen Rea', born: 1946}) + CREATE (johnH:Person {name: 'John Hurt', born: 1940}) + CREATE (benM:Person {name: 'Ben Miles', born: 1967}) + CREATE + (hugo)-[:ACTED_IN {roles: ['V']}]->(vforVendetta), + (natalieP)-[:ACTED_IN {roles: ['Evey Hammond']}]->(vforVendetta), + (stephenR)-[:ACTED_IN {roles: ['Eric Finch']}]->(vforVendetta), + (johnH)-[:ACTED_IN {roles: ['High Chancellor Adam Sutler']}]->(vforVendetta), + (benM)-[:ACTED_IN {roles: ['Dascomb']}]->(vforVendetta), + (jamesM)-[:DIRECTED]->(vforVendetta), + (andyW)-[:PRODUCED]->(vforVendetta), + (lanaW)-[:PRODUCED]->(vforVendetta), + (joelS)-[:PRODUCED]->(vforVendetta), + (andyW)-[:WROTE]->(vforVendetta), + (lanaW)-[:WROTE]->(vforVendetta) + + CREATE (speedRacer:Movie {title: 'Speed Racer', released: 2008, tagline: 'Speed has no limits'}) + CREATE (emileH:Person {name: 'Emile Hirsch', born: 1985}) + CREATE (johnG:Person {name: 'John Goodman', born: 1960}) + CREATE (susanS:Person {name: 'Susan Sarandon', born: 1946}) + CREATE (matthewF:Person {name: 'Matthew Fox', born: 1966}) + CREATE (christinaR:Person {name: 'Christina Ricci', born: 1980}) + CREATE (rain:Person {name: 'Rain', born: 1982}) + CREATE + (emileH)-[:ACTED_IN {roles: ['Speed Racer']}]->(speedRacer), + (johnG)-[:ACTED_IN {roles: ['Pops']}]->(speedRacer), + (susanS)-[:ACTED_IN {roles: ['Mom']}]->(speedRacer), + (matthewF)-[:ACTED_IN {roles: ['Racer X']}]->(speedRacer), + (christinaR)-[:ACTED_IN {roles: ['Trixie']}]->(speedRacer), + (rain)-[:ACTED_IN {roles: ['Taejo Togokahn']}]->(speedRacer), + (benM)-[:ACTED_IN {roles: ['Cass Jones']}]->(speedRacer), + (andyW)-[:DIRECTED]->(speedRacer), + (lanaW)-[:DIRECTED]->(speedRacer), + (andyW)-[:WROTE]->(speedRacer), + (lanaW)-[:WROTE]->(speedRacer), + (joelS)-[:PRODUCED]->(speedRacer) + + CREATE (ninjaAssassin:Movie {title: 'Ninja Assassin', released: 2009, + tagline: 'Prepare to enter a secret world of assassins'}) + CREATE (naomieH:Person {name: 'Naomie Harris'}) + CREATE + (rain)-[:ACTED_IN {roles: ['Raizo']}]->(ninjaAssassin), + (naomieH)-[:ACTED_IN {roles: ['Mika Coretti']}]->(ninjaAssassin), + (rickY)-[:ACTED_IN {roles: ['Takeshi']}]->(ninjaAssassin), + (benM)-[:ACTED_IN {roles: ['Ryan Maslow']}]->(ninjaAssassin), + (jamesM)-[:DIRECTED]->(ninjaAssassin), + (andyW)-[:PRODUCED]->(ninjaAssassin), + (lanaW)-[:PRODUCED]->(ninjaAssassin), + (joelS)-[:PRODUCED]->(ninjaAssassin) + + CREATE (theGreenMile:Movie {title: 'The Green Mile', released: 1999, + tagline: 'Walk a mile you\'ll never forget.'}) + CREATE (michaelD:Person {name: 'Michael Clarke Duncan', born: 1957}) + CREATE (davidM:Person {name: 'David Morse', born: 1953}) + CREATE (samR:Person {name: 'Sam Rockwell', born: 1968}) + CREATE (garyS:Person {name: 'Gary Sinise', born: 1955}) + CREATE (patriciaC:Person {name: 'Patricia Clarkson', born: 1959}) + CREATE (frankD:Person {name: 'Frank Darabont', born: 1959}) + CREATE + (tomH)-[:ACTED_IN {roles: ['Paul Edgecomb']}]->(theGreenMile), + (michaelD)-[:ACTED_IN {roles: ['John Coffey']}]->(theGreenMile), + (davidM)-[:ACTED_IN {roles: ['Brutus Brutal Howell']}]->(theGreenMile), + (bonnieH)-[:ACTED_IN {roles: ['Jan Edgecomb']}]->(theGreenMile), + (jamesC)-[:ACTED_IN {roles: ['Warden Hal Moores']}]->(theGreenMile), + (samR)-[:ACTED_IN {roles: ['Wild Bill Wharton']}]->(theGreenMile), + (garyS)-[:ACTED_IN {roles: ['Burt Hammersmith']}]->(theGreenMile), + (patriciaC)-[:ACTED_IN {roles: ['Melinda Moores']}]->(theGreenMile), + (frankD)-[:DIRECTED]->(theGreenMile) + + CREATE (frostNixon:Movie {title: 'Frost/Nixon', released: 2008, + tagline: '400 million people were waiting for the truth.'}) + CREATE (frankL:Person {name: 'Frank Langella', born: 1938}) + CREATE (michaelS:Person {name: 'Michael Sheen', born: 1969}) + CREATE (oliverP:Person {name: 'Oliver Platt', born: 1960}) + CREATE + (frankL)-[:ACTED_IN {roles: ['Richard Nixon']}]->(frostNixon), + (michaelS)-[:ACTED_IN {roles: ['David Frost']}]->(frostNixon), + (kevinB)-[:ACTED_IN {roles: ['Jack Brennan']}]->(frostNixon), + (oliverP)-[:ACTED_IN {roles: ['Bob Zelnick']}]->(frostNixon), + (samR)-[:ACTED_IN {roles: ['James Reston, Jr.']}]->(frostNixon), + (ronH)-[:DIRECTED]->(frostNixon) + + CREATE (hoffa:Movie {title: 'Hoffa', released: 1992, tagline: "He didn't want law. He wanted justice."}) + CREATE (dannyD:Person {name: 'Danny DeVito', born: 1944}) + CREATE (johnR:Person {name: 'John C. Reilly', born: 1965}) + CREATE + (jackN)-[:ACTED_IN {roles: ['Hoffa']}]->(hoffa), + (dannyD)-[:ACTED_IN {roles: ['Robert Bobby Ciaro']}]->(hoffa), + (jTW)-[:ACTED_IN {roles: ['Frank Fitzsimmons']}]->(hoffa), + (johnR)-[:ACTED_IN {roles: ['Peter Connelly']}]->(hoffa), + (dannyD)-[:DIRECTED]->(hoffa) + + CREATE (apollo13:Movie {title: 'Apollo 13', released: 1995, tagline: 'Houston, we have a problem.'}) + CREATE (edH:Person {name: 'Ed Harris', born: 1950}) + CREATE (billPax:Person {name: 'Bill Paxton', born: 1955}) + CREATE + (tomH)-[:ACTED_IN {roles: ['Jim Lovell']}]->(apollo13), + (kevinB)-[:ACTED_IN {roles: ['Jack Swigert']}]->(apollo13), + (edH)-[:ACTED_IN {roles: ['Gene Kranz']}]->(apollo13), + (billPax)-[:ACTED_IN {roles: ['Fred Haise']}]->(apollo13), + (garyS)-[:ACTED_IN {roles: ['Ken Mattingly']}]->(apollo13), + (ronH)-[:DIRECTED]->(apollo13) + + CREATE (twister:Movie {title: 'Twister', released: 1996, tagline: 'Don\'t Breathe. Don\'t Look Back.'}) + CREATE (philipH:Person {name: 'Philip Seymour Hoffman', born: 1967}) + CREATE (janB:Person {name: 'Jan de Bont', born: 1943}) + CREATE + (billPax)-[:ACTED_IN {roles: ['Bill Harding']}]->(twister), + (helenH)-[:ACTED_IN {roles: ['Dr. Jo Harding']}]->(twister), + (zachG)-[:ACTED_IN {roles: ['Eddie']}]->(twister), + (philipH)-[:ACTED_IN {roles: ['Dustin Davis']}]->(twister), + (janB)-[:DIRECTED]->(twister) + + CREATE (castAway:Movie {title: 'Cast Away', released: 2000, + tagline: 'At the edge of the world, his journey begins.'}) + CREATE (robertZ:Person {name: 'Robert Zemeckis', born: 1951}) + CREATE + (tomH)-[:ACTED_IN {roles: ['Chuck Noland']}]->(castAway), + (helenH)-[:ACTED_IN {roles: ['Kelly Frears']}]->(castAway), + (robertZ)-[:DIRECTED]->(castAway) + + CREATE (oneFlewOvertheCuckoosNest:Movie {title: 'One Flew Over the Cuckoo\'s Nest', released: 1975, + tagline: 'If he is crazy, what does that make you?'}) + CREATE (milosF:Person {name: 'Milos Forman', born: 1932}) + CREATE + (jackN)-[:ACTED_IN {roles: ['Randle McMurphy']}]->(oneFlewOvertheCuckoosNest), + (dannyD)-[:ACTED_IN {roles: ['Martini']}]->(oneFlewOvertheCuckoosNest), + (milosF)-[:DIRECTED]->(oneFlewOvertheCuckoosNest) + + CREATE (somethingsGottaGive:Movie {title: 'Something\'s Gotta Give', released: 2003}) + CREATE (dianeK:Person {name: 'Diane Keaton', born: 1946}) + CREATE (nancyM:Person {name: 'Nancy Meyers', born: 1949}) + CREATE + (jackN)-[:ACTED_IN {roles: ['Harry Sanborn']}]->(somethingsGottaGive), + (dianeK)-[:ACTED_IN {roles: ['Erica Barry']}]->(somethingsGottaGive), + (keanu)-[:ACTED_IN {roles: ['Julian Mercer']}]->(somethingsGottaGive), + (nancyM)-[:DIRECTED]->(somethingsGottaGive), + (nancyM)-[:PRODUCED]->(somethingsGottaGive), + (nancyM)-[:WROTE]->(somethingsGottaGive) + + CREATE (bicentennialMan:Movie {title: 'Bicentennial Man', released: 1999, + tagline: 'One robot\'s 200 year journey to become an ordinary man.'}) + CREATE (chrisC:Person {name: 'Chris Columbus', born: 1958}) + CREATE + (robin)-[:ACTED_IN {roles: ['Andrew Marin']}]->(bicentennialMan), + (oliverP)-[:ACTED_IN {roles: ['Rupert Burns']}]->(bicentennialMan), + (chrisC)-[:DIRECTED]->(bicentennialMan) + + CREATE (charlieWilsonsWar:Movie {title: 'Charlie Wilson\'s War', released: 2007, + tagline: 'A stiff drink. A little mascara. A lot of nerve. Who said they could not bring down the Soviet empire.'}) + CREATE (juliaR:Person {name: 'Julia Roberts', born: 1967}) + CREATE + (tomH)-[:ACTED_IN {roles: ['Rep. Charlie Wilson']}]->(charlieWilsonsWar), + (juliaR)-[:ACTED_IN {roles: ['Joanne Herring']}]->(charlieWilsonsWar), + (philipH)-[:ACTED_IN {roles: ['Gust Avrakotos']}]->(charlieWilsonsWar), + (mikeN)-[:DIRECTED]->(charlieWilsonsWar) + + CREATE (thePolarExpress:Movie {title: 'The Polar Express', released: 2004, + tagline: 'This Holiday Season... Believe'}) + CREATE + (tomH)-[:ACTED_IN {roles: ['Hero Boy', 'Father', 'Conductor', 'Hobo', 'Scrooge', 'Santa Claus']}]->(thePolarExpress), + (robertZ)-[:DIRECTED]->(thePolarExpress) + + CREATE (aLeagueofTheirOwn:Movie {title: 'A League of Their Own', released: 1992, + tagline: 'A league of their own'}) + CREATE (madonna:Person {name: 'Madonna', born: 1954}) + CREATE (geenaD:Person {name: 'Geena Davis', born: 1956}) + CREATE (loriP:Person {name: 'Lori Petty', born: 1963}) + CREATE (pennyM:Person {name: 'Penny Marshall', born: 1943}) + CREATE + (tomH)-[:ACTED_IN {roles: ['Jimmy Dugan']}]->(aLeagueofTheirOwn), + (geenaD)-[:ACTED_IN {roles: ['Dottie Hinson']}]->(aLeagueofTheirOwn), + (loriP)-[:ACTED_IN {roles: ['Kit Keller']}]->(aLeagueofTheirOwn), + (rosieO)-[:ACTED_IN {roles: ['Doris Murphy']}]->(aLeagueofTheirOwn), + (madonna)-[:ACTED_IN {roles: ['Mae Mordabito']}]->(aLeagueofTheirOwn), + (billPax)-[:ACTED_IN {roles: ['Bob Hinson']}]->(aLeagueofTheirOwn), + (pennyM)-[:DIRECTED]->(aLeagueofTheirOwn) + + CREATE (paulBlythe:Person {name: 'Paul Blythe'}) + CREATE (angelaScope:Person {name: 'Angela Scope'}) + CREATE (jessicaThompson:Person {name: 'Jessica Thompson'}) + CREATE (jamesThompson:Person {name: 'James Thompson'}) + + CREATE + (jamesThompson)-[:FOLLOWS]->(jessicaThompson), + (angelaScope)-[:FOLLOWS]->(jessicaThompson), + (paulBlythe)-[:FOLLOWS]->(angelaScope) + + CREATE + (jessicaThompson)-[:REVIEWED {summary: 'An amazing journey', rating: 95}]->(cloudAtlas), + (jessicaThompson)-[:REVIEWED {summary: 'Silly, but fun', rating: 65}]->(theReplacements), + (jamesThompson)-[:REVIEWED {summary: 'The coolest football movie ever', rating: 100}]->(theReplacements), + (angelaScope)-[:REVIEWED {summary: 'Pretty funny at times', rating: 62}]->(theReplacements), + (jessicaThompson)-[:REVIEWED {summary: 'Dark, but compelling', rating: 85}]->(unforgiven), + (jessicaThompson)-[:REVIEWED {summary: 'Slapstick', rating: 45}]->(theBirdcage), + (jessicaThompson)-[:REVIEWED {summary: 'A solid romp', rating: 68}]->(theDaVinciCode), + (jamesThompson)-[:REVIEWED {summary: 'Fun, but a little far fetched', rating: 65}]->(theDaVinciCode), + (jessicaThompson)-[:REVIEWED {summary: 'You had me at Jerry', rating: 92}]->(jerryMaguire) + + """ + Then the result should be empty + And the side effects should be: + | +nodes | 171 | + | +relationships | 253 | + | +properties | 564 | + | +labels | 2 | + + Scenario: Many CREATE clauses + Given an empty graph + When executing query: + """ + CREATE (hf:School {name: 'Hilly Fields Technical College'}) + CREATE (hf)-[:STAFF]->(mrb:Teacher {name: 'Mr Balls'}) + CREATE (hf)-[:STAFF]->(mrspb:Teacher {name: 'Ms Packard-Bell'}) + CREATE (hf)-[:STAFF]->(mrs:Teacher {name: 'Mr Smith'}) + CREATE (hf)-[:STAFF]->(mrsa:Teacher {name: 'Mrs Adenough'}) + CREATE (hf)-[:STAFF]->(mrvdg:Teacher {name: 'Mr Van der Graaf'}) + CREATE (hf)-[:STAFF]->(msn:Teacher {name: 'Ms Noethe'}) + CREATE (hf)-[:STAFF]->(mrsn:Teacher {name: 'Mrs Noakes'}) + CREATE (hf)-[:STAFF]->(mrm:Teacher {name: 'Mr Marker'}) + CREATE (hf)-[:STAFF]->(msd:Teacher {name: 'Ms Delgado'}) + CREATE (hf)-[:STAFF]->(mrsg:Teacher {name: 'Mrs Glass'}) + CREATE (hf)-[:STAFF]->(mrf:Teacher {name: 'Mr Flint'}) + CREATE (hf)-[:STAFF]->(mrk:Teacher {name: 'Mr Kearney'}) + CREATE (hf)-[:STAFF]->(msf:Teacher {name: 'Mrs Forrester'}) + CREATE (hf)-[:STAFF]->(mrsf:Teacher {name: 'Mrs Fischer'}) + CREATE (hf)-[:STAFF]->(mrj:Teacher {name: 'Mr Jameson'}) + + CREATE (hf)-[:STUDENT]->(_001:Student {name: 'Portia Vasquez'}) + CREATE (hf)-[:STUDENT]->(_002:Student {name: 'Andrew Parks'}) + CREATE (hf)-[:STUDENT]->(_003:Student {name: 'Germane Frye'}) + CREATE (hf)-[:STUDENT]->(_004:Student {name: 'Yuli Gutierrez'}) + CREATE (hf)-[:STUDENT]->(_005:Student {name: 'Kamal Solomon'}) + CREATE (hf)-[:STUDENT]->(_006:Student {name: 'Lysandra Porter'}) + CREATE (hf)-[:STUDENT]->(_007:Student {name: 'Stella Santiago'}) + CREATE (hf)-[:STUDENT]->(_008:Student {name: 'Brenda Torres'}) + CREATE (hf)-[:STUDENT]->(_009:Student {name: 'Heidi Dunlap'}) + + CREATE (hf)-[:STUDENT]->(_010:Student {name: 'Halee Taylor'}) + CREATE (hf)-[:STUDENT]->(_011:Student {name: 'Brennan Crosby'}) + CREATE (hf)-[:STUDENT]->(_012:Student {name: 'Rooney Cook'}) + CREATE (hf)-[:STUDENT]->(_013:Student {name: 'Xavier Morrison'}) + CREATE (hf)-[:STUDENT]->(_014:Student {name: 'Zelenia Santana'}) + CREATE (hf)-[:STUDENT]->(_015:Student {name: 'Eaton Bonner'}) + CREATE (hf)-[:STUDENT]->(_016:Student {name: 'Leilani Bishop'}) + CREATE (hf)-[:STUDENT]->(_017:Student {name: 'Jamalia Pickett'}) + CREATE (hf)-[:STUDENT]->(_018:Student {name: 'Wynter Russell'}) + CREATE (hf)-[:STUDENT]->(_019:Student {name: 'Liberty Melton'}) + + CREATE (hf)-[:STUDENT]->(_020:Student {name: 'MacKensie Obrien'}) + CREATE (hf)-[:STUDENT]->(_021:Student {name: 'Oprah Maynard'}) + CREATE (hf)-[:STUDENT]->(_022:Student {name: 'Lyle Parks'}) + CREATE (hf)-[:STUDENT]->(_023:Student {name: 'Madonna Justice'}) + CREATE (hf)-[:STUDENT]->(_024:Student {name: 'Herman Frederick'}) + CREATE (hf)-[:STUDENT]->(_025:Student {name: 'Preston Stevenson'}) + CREATE (hf)-[:STUDENT]->(_026:Student {name: 'Drew Carrillo'}) + CREATE (hf)-[:STUDENT]->(_027:Student {name: 'Hamilton Woodward'}) + CREATE (hf)-[:STUDENT]->(_028:Student {name: 'Buckminster Bradley'}) + CREATE (hf)-[:STUDENT]->(_029:Student {name: 'Shea Cote'}) + + CREATE (hf)-[:STUDENT]->(_030:Student {name: 'Raymond Leonard'}) + CREATE (hf)-[:STUDENT]->(_031:Student {name: 'Gavin Branch'}) + CREATE (hf)-[:STUDENT]->(_032:Student {name: 'Kylan Powers'}) + CREATE (hf)-[:STUDENT]->(_033:Student {name: 'Hedy Bowers'}) + CREATE (hf)-[:STUDENT]->(_034:Student {name: 'Derek Church'}) + CREATE (hf)-[:STUDENT]->(_035:Student {name: 'Silas Santiago'}) + CREATE (hf)-[:STUDENT]->(_036:Student {name: 'Elton Bright'}) + CREATE (hf)-[:STUDENT]->(_037:Student {name: 'Dora Schmidt'}) + CREATE (hf)-[:STUDENT]->(_038:Student {name: 'Julian Sullivan'}) + CREATE (hf)-[:STUDENT]->(_039:Student {name: 'Willow Morton'}) + + CREATE (hf)-[:STUDENT]->(_040:Student {name: 'Blaze Hines'}) + CREATE (hf)-[:STUDENT]->(_041:Student {name: 'Felicia Tillman'}) + CREATE (hf)-[:STUDENT]->(_042:Student {name: 'Ralph Webb'}) + CREATE (hf)-[:STUDENT]->(_043:Student {name: 'Roth Gilmore'}) + CREATE (hf)-[:STUDENT]->(_044:Student {name: 'Dorothy Burgess'}) + CREATE (hf)-[:STUDENT]->(_045:Student {name: 'Lana Sandoval'}) + CREATE (hf)-[:STUDENT]->(_046:Student {name: 'Nevada Strickland'}) + CREATE (hf)-[:STUDENT]->(_047:Student {name: 'Lucian Franco'}) + CREATE (hf)-[:STUDENT]->(_048:Student {name: 'Jasper Talley'}) + CREATE (hf)-[:STUDENT]->(_049:Student {name: 'Madaline Spears'}) + + CREATE (hf)-[:STUDENT]->(_050:Student {name: 'Upton Browning'}) + CREATE (hf)-[:STUDENT]->(_051:Student {name: 'Cooper Leon'}) + CREATE (hf)-[:STUDENT]->(_052:Student {name: 'Celeste Ortega'}) + CREATE (hf)-[:STUDENT]->(_053:Student {name: 'Willa Hewitt'}) + CREATE (hf)-[:STUDENT]->(_054:Student {name: 'Rooney Bryan'}) + CREATE (hf)-[:STUDENT]->(_055:Student {name: 'Nayda Hays'}) + CREATE (hf)-[:STUDENT]->(_056:Student {name: 'Kadeem Salazar'}) + CREATE (hf)-[:STUDENT]->(_057:Student {name: 'Halee Allen'}) + CREATE (hf)-[:STUDENT]->(_058:Student {name: 'Odysseus Mayo'}) + CREATE (hf)-[:STUDENT]->(_059:Student {name: 'Kato Merrill'}) + + CREATE (hf)-[:STUDENT]->(_060:Student {name: 'Halee Juarez'}) + CREATE (hf)-[:STUDENT]->(_061:Student {name: 'Chloe Charles'}) + CREATE (hf)-[:STUDENT]->(_062:Student {name: 'Abel Montoya'}) + CREATE (hf)-[:STUDENT]->(_063:Student {name: 'Hilda Welch'}) + CREATE (hf)-[:STUDENT]->(_064:Student {name: 'Britanni Bean'}) + CREATE (hf)-[:STUDENT]->(_065:Student {name: 'Joelle Beach'}) + CREATE (hf)-[:STUDENT]->(_066:Student {name: 'Ciara Odom'}) + CREATE (hf)-[:STUDENT]->(_067:Student {name: 'Zia Williams'}) + CREATE (hf)-[:STUDENT]->(_068:Student {name: 'Darrel Bailey'}) + CREATE (hf)-[:STUDENT]->(_069:Student {name: 'Lance Mcdowell'}) + + CREATE (hf)-[:STUDENT]->(_070:Student {name: 'Clayton Bullock'}) + CREATE (hf)-[:STUDENT]->(_071:Student {name: 'Roanna Mosley'}) + CREATE (hf)-[:STUDENT]->(_072:Student {name: 'Amethyst Mcclure'}) + CREATE (hf)-[:STUDENT]->(_073:Student {name: 'Hanae Mann'}) + CREATE (hf)-[:STUDENT]->(_074:Student {name: 'Graiden Haynes'}) + CREATE (hf)-[:STUDENT]->(_075:Student {name: 'Marcia Byrd'}) + CREATE (hf)-[:STUDENT]->(_076:Student {name: 'Yoshi Joyce'}) + CREATE (hf)-[:STUDENT]->(_077:Student {name: 'Gregory Sexton'}) + CREATE (hf)-[:STUDENT]->(_078:Student {name: 'Nash Carey'}) + CREATE (hf)-[:STUDENT]->(_079:Student {name: 'Rae Stevens'}) + + CREATE (hf)-[:STUDENT]->(_080:Student {name: 'Blossom Fulton'}) + CREATE (hf)-[:STUDENT]->(_081:Student {name: 'Lev Curry'}) + CREATE (hf)-[:STUDENT]->(_082:Student {name: 'Margaret Gamble'}) + CREATE (hf)-[:STUDENT]->(_083:Student {name: 'Rylee Patterson'}) + CREATE (hf)-[:STUDENT]->(_084:Student {name: 'Harper Perkins'}) + CREATE (hf)-[:STUDENT]->(_085:Student {name: 'Kennan Murphy'}) + CREATE (hf)-[:STUDENT]->(_086:Student {name: 'Hilda Coffey'}) + CREATE (hf)-[:STUDENT]->(_087:Student {name: 'Marah Reed'}) + CREATE (hf)-[:STUDENT]->(_088:Student {name: 'Blaine Wade'}) + CREATE (hf)-[:STUDENT]->(_089:Student {name: 'Geraldine Sanders'}) + + CREATE (hf)-[:STUDENT]->(_090:Student {name: 'Kerry Rollins'}) + CREATE (hf)-[:STUDENT]->(_091:Student {name: 'Virginia Sweet'}) + CREATE (hf)-[:STUDENT]->(_092:Student {name: 'Sophia Merrill'}) + CREATE (hf)-[:STUDENT]->(_093:Student {name: 'Hedda Carson'}) + CREATE (hf)-[:STUDENT]->(_094:Student {name: 'Tamekah Charles'}) + CREATE (hf)-[:STUDENT]->(_095:Student {name: 'Knox Barton'}) + CREATE (hf)-[:STUDENT]->(_096:Student {name: 'Ariel Porter'}) + CREATE (hf)-[:STUDENT]->(_097:Student {name: 'Berk Wooten'}) + CREATE (hf)-[:STUDENT]->(_098:Student {name: 'Galena Glenn'}) + CREATE (hf)-[:STUDENT]->(_099:Student {name: 'Jolene Anderson'}) + + CREATE (hf)-[:STUDENT]->(_100:Student {name: 'Leonard Hewitt'}) + CREATE (hf)-[:STUDENT]->(_101:Student {name: 'Maris Salazar'}) + CREATE (hf)-[:STUDENT]->(_102:Student {name: 'Brian Frost'}) + CREATE (hf)-[:STUDENT]->(_103:Student {name: 'Zane Moses'}) + CREATE (hf)-[:STUDENT]->(_104:Student {name: 'Serina Finch'}) + CREATE (hf)-[:STUDENT]->(_105:Student {name: 'Anastasia Fletcher'}) + CREATE (hf)-[:STUDENT]->(_106:Student {name: 'Glenna Chapman'}) + CREATE (hf)-[:STUDENT]->(_107:Student {name: 'Mufutau Gillespie'}) + CREATE (hf)-[:STUDENT]->(_108:Student {name: 'Basil Guthrie'}) + CREATE (hf)-[:STUDENT]->(_109:Student {name: 'Theodore Marsh'}) + + CREATE (hf)-[:STUDENT]->(_110:Student {name: 'Jaime Contreras'}) + CREATE (hf)-[:STUDENT]->(_111:Student {name: 'Irma Poole'}) + CREATE (hf)-[:STUDENT]->(_112:Student {name: 'Buckminster Bender'}) + CREATE (hf)-[:STUDENT]->(_113:Student {name: 'Elton Morris'}) + CREATE (hf)-[:STUDENT]->(_114:Student {name: 'Barbara Nguyen'}) + CREATE (hf)-[:STUDENT]->(_115:Student {name: 'Tanya Kidd'}) + CREATE (hf)-[:STUDENT]->(_116:Student {name: 'Kaden Hoover'}) + CREATE (hf)-[:STUDENT]->(_117:Student {name: 'Christopher Bean'}) + CREATE (hf)-[:STUDENT]->(_118:Student {name: 'Trevor Daugherty'}) + CREATE (hf)-[:STUDENT]->(_119:Student {name: 'Rudyard Bates'}) + + CREATE (hf)-[:STUDENT]->(_120:Student {name: 'Stacy Monroe'}) + CREATE (hf)-[:STUDENT]->(_121:Student {name: 'Kieran Keller'}) + CREATE (hf)-[:STUDENT]->(_122:Student {name: 'Ivy Garrison'}) + CREATE (hf)-[:STUDENT]->(_123:Student {name: 'Miranda Haynes'}) + CREATE (hf)-[:STUDENT]->(_124:Student {name: 'Abigail Heath'}) + CREATE (hf)-[:STUDENT]->(_125:Student {name: 'Margaret Santiago'}) + CREATE (hf)-[:STUDENT]->(_126:Student {name: 'Cade Floyd'}) + CREATE (hf)-[:STUDENT]->(_127:Student {name: 'Allen Crane'}) + CREATE (hf)-[:STUDENT]->(_128:Student {name: 'Stella Gilliam'}) + CREATE (hf)-[:STUDENT]->(_129:Student {name: 'Rashad Miller'}) + + CREATE (hf)-[:STUDENT]->(_130:Student {name: 'Francis Cox'}) + CREATE (hf)-[:STUDENT]->(_131:Student {name: 'Darryl Rosario'}) + CREATE (hf)-[:STUDENT]->(_132:Student {name: 'Michael Daniels'}) + CREATE (hf)-[:STUDENT]->(_133:Student {name: 'Aretha Henderson'}) + CREATE (hf)-[:STUDENT]->(_134:Student {name: 'Roth Barrera'}) + CREATE (hf)-[:STUDENT]->(_135:Student {name: 'Yael Day'}) + CREATE (hf)-[:STUDENT]->(_136:Student {name: 'Wynter Richmond'}) + CREATE (hf)-[:STUDENT]->(_137:Student {name: 'Quyn Flowers'}) + CREATE (hf)-[:STUDENT]->(_138:Student {name: 'Yvette Marquez'}) + CREATE (hf)-[:STUDENT]->(_139:Student {name: 'Teagan Curry'}) + + CREATE (hf)-[:STUDENT]->(_140:Student {name: 'Brenden Bishop'}) + CREATE (hf)-[:STUDENT]->(_141:Student {name: 'Montana Black'}) + CREATE (hf)-[:STUDENT]->(_142:Student {name: 'Ramona Parker'}) + CREATE (hf)-[:STUDENT]->(_143:Student {name: 'Merritt Hansen'}) + CREATE (hf)-[:STUDENT]->(_144:Student {name: 'Melvin Vang'}) + CREATE (hf)-[:STUDENT]->(_145:Student {name: 'Samantha Perez'}) + CREATE (hf)-[:STUDENT]->(_146:Student {name: 'Thane Porter'}) + CREATE (hf)-[:STUDENT]->(_147:Student {name: 'Vaughan Haynes'}) + CREATE (hf)-[:STUDENT]->(_148:Student {name: 'Irma Miles'}) + CREATE (hf)-[:STUDENT]->(_149:Student {name: 'Amery Jensen'}) + + CREATE (hf)-[:STUDENT]->(_150:Student {name: 'Montana Holman'}) + CREATE (hf)-[:STUDENT]->(_151:Student {name: 'Kimberly Langley'}) + CREATE (hf)-[:STUDENT]->(_152:Student {name: 'Ebony Bray'}) + CREATE (hf)-[:STUDENT]->(_153:Student {name: 'Ishmael Pollard'}) + CREATE (hf)-[:STUDENT]->(_154:Student {name: 'Illana Thompson'}) + CREATE (hf)-[:STUDENT]->(_155:Student {name: 'Rhona Bowers'}) + CREATE (hf)-[:STUDENT]->(_156:Student {name: 'Lilah Dotson'}) + CREATE (hf)-[:STUDENT]->(_157:Student {name: 'Shelly Roach'}) + CREATE (hf)-[:STUDENT]->(_158:Student {name: 'Celeste Woodward'}) + CREATE (hf)-[:STUDENT]->(_159:Student {name: 'Christen Lynn'}) + + CREATE (hf)-[:STUDENT]->(_160:Student {name: 'Miranda Slater'}) + CREATE (hf)-[:STUDENT]->(_161:Student {name: 'Lunea Clements'}) + CREATE (hf)-[:STUDENT]->(_162:Student {name: 'Lester Francis'}) + CREATE (hf)-[:STUDENT]->(_163:Student {name: 'David Fischer'}) + CREATE (hf)-[:STUDENT]->(_164:Student {name: 'Kyra Bean'}) + CREATE (hf)-[:STUDENT]->(_165:Student {name: 'Imelda Alston'}) + CREATE (hf)-[:STUDENT]->(_166:Student {name: 'Finn Farrell'}) + CREATE (hf)-[:STUDENT]->(_167:Student {name: 'Kirby House'}) + CREATE (hf)-[:STUDENT]->(_168:Student {name: 'Amanda Zamora'}) + CREATE (hf)-[:STUDENT]->(_169:Student {name: 'Rina Franco'}) + + CREATE (hf)-[:STUDENT]->(_170:Student {name: 'Sonia Lane'}) + CREATE (hf)-[:STUDENT]->(_171:Student {name: 'Nora Jefferson'}) + CREATE (hf)-[:STUDENT]->(_172:Student {name: 'Colton Ortiz'}) + CREATE (hf)-[:STUDENT]->(_173:Student {name: 'Alden Munoz'}) + CREATE (hf)-[:STUDENT]->(_174:Student {name: 'Ferdinand Cline'}) + CREATE (hf)-[:STUDENT]->(_175:Student {name: 'Cynthia Prince'}) + CREATE (hf)-[:STUDENT]->(_176:Student {name: 'Asher Hurst'}) + CREATE (hf)-[:STUDENT]->(_177:Student {name: 'MacKensie Stevenson'}) + CREATE (hf)-[:STUDENT]->(_178:Student {name: 'Sydnee Sosa'}) + CREATE (hf)-[:STUDENT]->(_179:Student {name: 'Dante Callahan'}) + + CREATE (hf)-[:STUDENT]->(_180:Student {name: 'Isabella Santana'}) + CREATE (hf)-[:STUDENT]->(_181:Student {name: 'Raven Bowman'}) + CREATE (hf)-[:STUDENT]->(_182:Student {name: 'Kirby Bolton'}) + CREATE (hf)-[:STUDENT]->(_183:Student {name: 'Peter Shaffer'}) + CREATE (hf)-[:STUDENT]->(_184:Student {name: 'Fletcher Beard'}) + CREATE (hf)-[:STUDENT]->(_185:Student {name: 'Irene Lowe'}) + CREATE (hf)-[:STUDENT]->(_186:Student {name: 'Ella Talley'}) + CREATE (hf)-[:STUDENT]->(_187:Student {name: 'Jorden Kerr'}) + CREATE (hf)-[:STUDENT]->(_188:Student {name: 'Macey Delgado'}) + CREATE (hf)-[:STUDENT]->(_189:Student {name: 'Ulysses Graves'}) + + CREATE (hf)-[:STUDENT]->(_190:Student {name: 'Declan Blake'}) + CREATE (hf)-[:STUDENT]->(_191:Student {name: 'Lila Hurst'}) + CREATE (hf)-[:STUDENT]->(_192:Student {name: 'David Rasmussen'}) + CREATE (hf)-[:STUDENT]->(_193:Student {name: 'Desiree Cortez'}) + CREATE (hf)-[:STUDENT]->(_194:Student {name: 'Myles Horton'}) + CREATE (hf)-[:STUDENT]->(_195:Student {name: 'Rylee Willis'}) + CREATE (hf)-[:STUDENT]->(_196:Student {name: 'Kelsey Yates'}) + CREATE (hf)-[:STUDENT]->(_197:Student {name: 'Alika Stanton'}) + CREATE (hf)-[:STUDENT]->(_198:Student {name: 'Ria Campos'}) + CREATE (hf)-[:STUDENT]->(_199:Student {name: 'Elijah Hendricks'}) + + CREATE (hf)-[:STUDENT]->(_200:Student {name: 'Hayes House'}) + + CREATE (hf)-[:DEPARTMENT]->(md:Department {name: 'Mathematics'}) + CREATE (hf)-[:DEPARTMENT]->(sd:Department {name: 'Science'}) + CREATE (hf)-[:DEPARTMENT]->(ed:Department {name: 'Engineering'}) + + CREATE (pm:Subject {name: 'Pure Mathematics'}) + CREATE (am:Subject {name: 'Applied Mathematics'}) + CREATE (ph:Subject {name: 'Physics'}) + CREATE (ch:Subject {name: 'Chemistry'}) + CREATE (bi:Subject {name: 'Biology'}) + CREATE (es:Subject {name: 'Earth Science'}) + CREATE (me:Subject {name: 'Mechanical Engineering'}) + CREATE (ce:Subject {name: 'Chemical Engineering'}) + CREATE (se:Subject {name: 'Systems Engineering'}) + CREATE (ve:Subject {name: 'Civil Engineering'}) + CREATE (ee:Subject {name: 'Electrical Engineering'}) + + CREATE (sd)-[:CURRICULUM]->(ph) + CREATE (sd)-[:CURRICULUM]->(ch) + CREATE (sd)-[:CURRICULUM]->(bi) + CREATE (sd)-[:CURRICULUM]->(es) + CREATE (md)-[:CURRICULUM]->(pm) + CREATE (md)-[:CURRICULUM]->(am) + CREATE (ed)-[:CURRICULUM]->(me) + CREATE (ed)-[:CURRICULUM]->(se) + CREATE (ed)-[:CURRICULUM]->(ce) + CREATE (ed)-[:CURRICULUM]->(ee) + CREATE (ed)-[:CURRICULUM]->(ve) + + CREATE (ph)-[:TAUGHT_BY]->(mrb) + CREATE (ph)-[:TAUGHT_BY]->(mrk) + CREATE (ch)-[:TAUGHT_BY]->(mrk) + CREATE (ch)-[:TAUGHT_BY]->(mrsn) + CREATE (bi)-[:TAUGHT_BY]->(mrsn) + CREATE (bi)-[:TAUGHT_BY]->(mrsf) + CREATE (es)-[:TAUGHT_BY]->(msn) + CREATE (pm)-[:TAUGHT_BY]->(mrf) + CREATE (pm)-[:TAUGHT_BY]->(mrm) + CREATE (pm)-[:TAUGHT_BY]->(mrvdg) + CREATE (am)-[:TAUGHT_BY]->(mrsg) + CREATE (am)-[:TAUGHT_BY]->(mrspb) + CREATE (am)-[:TAUGHT_BY]->(mrvdg) + CREATE (me)-[:TAUGHT_BY]->(mrj) + CREATE (ce)-[:TAUGHT_BY]->(mrsa) + CREATE (se)-[:TAUGHT_BY]->(mrs) + CREATE (ve)-[:TAUGHT_BY]->(msd) + CREATE (ee)-[:TAUGHT_BY]->(mrsf) + + CREATE(_001)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_188) + CREATE(_002)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_198) + CREATE(_003)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_106) + CREATE(_004)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_029) + CREATE(_005)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_153) + CREATE(_006)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_061) + CREATE(_007)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_177) + CREATE(_008)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_115) + CREATE(_009)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_131) + CREATE(_010)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_142) + CREATE(_011)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_043) + CREATE(_012)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_065) + CREATE(_013)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_074) + CREATE(_014)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_165) + CREATE(_015)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_117) + CREATE(_016)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_086) + CREATE(_017)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_062) + CREATE(_018)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_033) + CREATE(_019)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_171) + CREATE(_020)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_117) + CREATE(_021)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_086) + CREATE(_022)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_121) + CREATE(_023)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_049) + CREATE(_024)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_152) + CREATE(_025)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_152) + CREATE(_026)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_085) + CREATE(_027)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_084) + CREATE(_028)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_143) + CREATE(_029)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_099) + CREATE(_030)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_094) + CREATE(_031)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_125) + CREATE(_032)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_024) + CREATE(_033)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_075) + CREATE(_034)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_161) + CREATE(_035)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_197) + CREATE(_036)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_067) + CREATE(_037)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_049) + CREATE(_038)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_038) + CREATE(_039)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_116) + CREATE(_040)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_149) + CREATE(_041)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_044) + CREATE(_042)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_150) + CREATE(_043)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_095) + CREATE(_044)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_016) + CREATE(_045)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_021) + CREATE(_046)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_123) + CREATE(_047)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_189) + CREATE(_048)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_094) + CREATE(_049)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_161) + CREATE(_050)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_098) + CREATE(_051)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_145) + CREATE(_052)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_148) + CREATE(_053)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_123) + CREATE(_054)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_196) + CREATE(_055)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_175) + CREATE(_056)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_010) + CREATE(_057)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_042) + CREATE(_058)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_196) + CREATE(_059)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_067) + CREATE(_060)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_034) + CREATE(_061)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_002) + CREATE(_062)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_088) + CREATE(_063)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_142) + CREATE(_064)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_88) + CREATE(_065)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_099) + CREATE(_066)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_178) + CREATE(_067)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_041) + CREATE(_068)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_022) + CREATE(_069)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_109) + CREATE(_070)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_045) + CREATE(_071)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_182) + CREATE(_072)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_144) + CREATE(_073)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_140) + CREATE(_074)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_128) + CREATE(_075)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_149) + CREATE(_076)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_038) + CREATE(_077)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_104) + CREATE(_078)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_032) + CREATE(_079)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_123) + CREATE(_080)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_117) + CREATE(_081)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_174) + CREATE(_082)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_162) + CREATE(_083)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_011) + CREATE(_084)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_145) + CREATE(_085)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_003) + CREATE(_086)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_067) + CREATE(_087)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_173) + CREATE(_088)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_128) + CREATE(_089)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_177) + CREATE(_090)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_076) + CREATE(_091)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_137) + CREATE(_092)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_024) + CREATE(_093)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_156) + CREATE(_094)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_020) + CREATE(_095)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_112) + CREATE(_096)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_193) + CREATE(_097)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_006) + CREATE(_098)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_117) + CREATE(_099)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_141) + CREATE(_100)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_001) + CREATE(_101)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_169) + CREATE(_102)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_161) + CREATE(_103)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_136) + CREATE(_104)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_125) + CREATE(_105)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_127) + CREATE(_106)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_095) + CREATE(_107)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_036) + CREATE(_108)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_074) + CREATE(_109)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_150) + CREATE(_110)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_191) + CREATE(_111)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_068) + CREATE(_112)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_019) + CREATE(_113)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_035) + CREATE(_114)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_061) + CREATE(_115)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_070) + CREATE(_116)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_069) + CREATE(_117)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_096) + CREATE(_118)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_107) + CREATE(_119)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_140) + CREATE(_120)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_167) + CREATE(_121)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_120) + CREATE(_122)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_090) + CREATE(_123)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_004) + CREATE(_124)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_083) + CREATE(_125)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_094) + CREATE(_126)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_174) + CREATE(_127)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_168) + CREATE(_128)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_084) + CREATE(_129)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_186) + CREATE(_130)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_090) + CREATE(_131)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_010) + CREATE(_132)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_031) + CREATE(_133)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_059) + CREATE(_134)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_037) + CREATE(_135)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_012) + CREATE(_136)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_197) + CREATE(_137)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_059) + CREATE(_138)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_065) + CREATE(_139)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_175) + CREATE(_140)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_170) + CREATE(_141)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_191) + CREATE(_142)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_139) + CREATE(_143)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_054) + CREATE(_144)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_176) + CREATE(_145)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_188) + CREATE(_146)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_072) + CREATE(_147)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_096) + CREATE(_148)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_108) + CREATE(_149)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_155) + CREATE(_150)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_002) + CREATE(_151)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_076) + CREATE(_152)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_169) + CREATE(_153)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_179) + CREATE(_154)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_186) + CREATE(_155)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_058) + CREATE(_156)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_071) + CREATE(_157)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_073) + CREATE(_158)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_003) + CREATE(_159)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_182) + CREATE(_160)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_199) + CREATE(_161)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_072) + CREATE(_162)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_014) + CREATE(_163)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_163) + CREATE(_164)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_038) + CREATE(_165)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_044) + CREATE(_166)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_136) + CREATE(_167)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_038) + CREATE(_168)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_110) + CREATE(_169)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_198) + CREATE(_170)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_178) + CREATE(_171)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_022) + CREATE(_172)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_020) + CREATE(_173)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_164) + CREATE(_174)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_075) + CREATE(_175)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_175) + CREATE(_176)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_003) + CREATE(_177)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_120) + CREATE(_178)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_006) + CREATE(_179)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_057) + CREATE(_180)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_185) + CREATE(_181)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_074) + CREATE(_182)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_120) + CREATE(_183)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_131) + CREATE(_184)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_045) + CREATE(_185)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_200) + CREATE(_186)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_140) + CREATE(_187)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_150) + CREATE(_188)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_014) + CREATE(_189)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_096) + CREATE(_190)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_063) + CREATE(_191)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_079) + CREATE(_192)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_121) + CREATE(_193)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_196) + CREATE(_194)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_029) + CREATE(_195)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_164) + CREATE(_196)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_083) + CREATE(_197)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_101) + CREATE(_198)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_039) + CREATE(_199)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_011) + CREATE(_200)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_073) + CREATE(_001)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_129) + CREATE(_002)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_078) + CREATE(_003)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_181) + CREATE(_004)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_162) + CREATE(_005)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_057) + CREATE(_006)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_111) + CREATE(_007)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_027) + CREATE(_008)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_123) + CREATE(_009)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_132) + CREATE(_010)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_147) + CREATE(_011)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_083) + CREATE(_012)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_118) + CREATE(_013)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_099) + CREATE(_014)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_140) + CREATE(_015)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_107) + CREATE(_016)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_116) + CREATE(_017)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_002) + CREATE(_018)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_069) + CREATE(_019)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_024) + CREATE(_020)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_022) + CREATE(_021)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_184) + CREATE(_022)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_200) + CREATE(_023)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_200) + CREATE(_024)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_075) + CREATE(_025)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_087) + CREATE(_026)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_163) + CREATE(_027)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_115) + CREATE(_028)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_042) + CREATE(_029)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_058) + CREATE(_030)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_188) + CREATE(_031)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_123) + CREATE(_032)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_015) + CREATE(_033)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_130) + CREATE(_034)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_141) + CREATE(_035)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_158) + CREATE(_036)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_020) + CREATE(_037)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_102) + CREATE(_038)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_184) + CREATE(_039)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_196) + CREATE(_040)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_003) + CREATE(_041)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_171) + CREATE(_042)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_050) + CREATE(_043)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_085) + CREATE(_044)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_025) + CREATE(_045)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_084) + CREATE(_046)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_118) + CREATE(_047)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_002) + CREATE(_048)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_099) + CREATE(_049)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_071) + CREATE(_050)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_178) + CREATE(_051)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_200) + CREATE(_052)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_059) + CREATE(_053)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_095) + CREATE(_054)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_185) + CREATE(_055)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_108) + CREATE(_056)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_083) + CREATE(_057)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_031) + CREATE(_058)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_054) + CREATE(_059)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_198) + CREATE(_060)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_138) + CREATE(_061)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_176) + CREATE(_062)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_086) + CREATE(_063)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_032) + CREATE(_064)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_101) + CREATE(_065)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_181) + CREATE(_066)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_153) + CREATE(_067)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_166) + CREATE(_068)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_003) + CREATE(_069)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_027) + CREATE(_070)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_021) + CREATE(_071)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_193) + CREATE(_072)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_022) + CREATE(_073)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_108) + CREATE(_074)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_174) + CREATE(_075)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_019) + CREATE(_076)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_179) + CREATE(_077)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_005) + CREATE(_078)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_014) + CREATE(_079)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_017) + CREATE(_080)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_146) + CREATE(_081)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_098) + CREATE(_082)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_171) + CREATE(_083)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_099) + CREATE(_084)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_161) + CREATE(_085)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_098) + CREATE(_086)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_199) + CREATE(_087)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_057) + CREATE(_088)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_164) + CREATE(_089)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_064) + CREATE(_090)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_109) + CREATE(_091)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_077) + CREATE(_092)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_124) + CREATE(_093)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_181) + CREATE(_094)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_142) + CREATE(_095)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_191) + CREATE(_096)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_093) + CREATE(_097)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_031) + CREATE(_098)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_045) + CREATE(_099)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_182) + CREATE(_100)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_043) + CREATE(_101)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_146) + CREATE(_102)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_141) + CREATE(_103)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_040) + CREATE(_104)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_199) + CREATE(_105)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_063) + CREATE(_106)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_180) + CREATE(_107)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_010) + CREATE(_108)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_122) + CREATE(_109)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_111) + CREATE(_110)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_065) + CREATE(_111)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_199) + CREATE(_112)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_135) + CREATE(_113)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_172) + CREATE(_114)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_096) + CREATE(_115)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_028) + CREATE(_116)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_109) + CREATE(_117)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_191) + CREATE(_118)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_169) + CREATE(_119)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_101) + CREATE(_120)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_184) + CREATE(_121)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_032) + CREATE(_122)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_127) + CREATE(_123)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_129) + CREATE(_124)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_116) + CREATE(_125)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_150) + CREATE(_126)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_175) + CREATE(_127)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_018) + CREATE(_128)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_165) + CREATE(_129)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_117) + CREATE(_130)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_066) + CREATE(_131)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_050) + CREATE(_132)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_197) + CREATE(_133)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_111) + CREATE(_134)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_125) + CREATE(_135)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_112) + CREATE(_136)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_173) + CREATE(_137)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_181) + CREATE(_138)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_072) + CREATE(_139)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_115) + CREATE(_140)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_013) + CREATE(_141)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_140) + CREATE(_142)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_003) + CREATE(_143)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_144) + CREATE(_144)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_002) + CREATE(_145)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_015) + CREATE(_146)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_061) + CREATE(_147)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_009) + CREATE(_148)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_145) + CREATE(_149)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_176) + CREATE(_150)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_152) + CREATE(_151)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_055) + CREATE(_152)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_157) + CREATE(_153)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_090) + CREATE(_154)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_162) + CREATE(_155)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_146) + CREATE(_156)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_073) + CREATE(_157)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_044) + CREATE(_158)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_154) + CREATE(_159)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_123) + CREATE(_160)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_168) + CREATE(_161)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_122) + CREATE(_162)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_015) + CREATE(_163)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_041) + CREATE(_164)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_087) + CREATE(_165)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_104) + CREATE(_166)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_116) + CREATE(_167)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_019) + CREATE(_168)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_021) + CREATE(_169)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_065) + CREATE(_170)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_183) + CREATE(_171)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_147) + CREATE(_172)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_045) + CREATE(_173)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_172) + CREATE(_174)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_137) + CREATE(_175)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_145) + CREATE(_176)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_138) + CREATE(_177)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_078) + CREATE(_178)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_176) + CREATE(_179)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_062) + CREATE(_180)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_145) + CREATE(_181)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_178) + CREATE(_182)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_173) + CREATE(_183)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_107) + CREATE(_184)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_198) + CREATE(_185)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_057) + CREATE(_186)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_041) + CREATE(_187)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_076) + CREATE(_188)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_132) + CREATE(_189)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_093) + CREATE(_190)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_002) + CREATE(_191)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_183) + CREATE(_192)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_140) + CREATE(_193)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_196) + CREATE(_194)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_117) + CREATE(_195)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_054) + CREATE(_196)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_197) + CREATE(_197)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_086) + CREATE(_198)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_190) + CREATE(_199)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_143) + CREATE(_200)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_144) + CREATE(_001)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_050) + CREATE(_002)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_024) + CREATE(_003)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_135) + CREATE(_004)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_094) + CREATE(_005)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_143) + CREATE(_006)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_066) + CREATE(_007)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_193) + CREATE(_008)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_022) + CREATE(_009)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_074) + CREATE(_010)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_166) + CREATE(_011)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_131) + CREATE(_012)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_036) + CREATE(_013)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_016) + CREATE(_014)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_108) + CREATE(_015)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_083) + CREATE(_016)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_120) + CREATE(_017)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_016) + CREATE(_018)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_130) + CREATE(_019)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_013) + CREATE(_020)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_186) + CREATE(_021)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_026) + CREATE(_022)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_040) + CREATE(_023)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_064) + CREATE(_024)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_072) + CREATE(_025)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_017) + CREATE(_026)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_159) + CREATE(_027)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_076) + CREATE(_028)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_014) + CREATE(_029)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_089) + CREATE(_030)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_157) + CREATE(_031)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_029) + CREATE(_032)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_184) + CREATE(_033)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_131) + CREATE(_034)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_171) + CREATE(_035)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_051) + CREATE(_036)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_031) + CREATE(_037)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_200) + CREATE(_038)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_057) + CREATE(_039)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_023) + CREATE(_040)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_109) + CREATE(_041)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_177) + CREATE(_042)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_020) + CREATE(_043)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_069) + CREATE(_044)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_068) + CREATE(_045)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_027) + CREATE(_046)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_018) + CREATE(_047)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_154) + CREATE(_048)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_090) + CREATE(_049)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_166) + CREATE(_050)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_150) + CREATE(_051)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_045) + CREATE(_052)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_123) + CREATE(_053)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_160) + CREATE(_054)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_088) + CREATE(_055)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_196) + CREATE(_056)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_120) + CREATE(_057)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_110) + CREATE(_058)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_060) + CREATE(_059)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_084) + CREATE(_060)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_030) + CREATE(_061)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_170) + CREATE(_062)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_027) + CREATE(_063)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_018) + CREATE(_064)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_004) + CREATE(_065)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_138) + CREATE(_066)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_009) + CREATE(_067)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_172) + CREATE(_068)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_077) + CREATE(_069)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_112) + CREATE(_070)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_069) + CREATE(_071)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_018) + CREATE(_072)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_172) + CREATE(_073)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_053) + CREATE(_074)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_098) + CREATE(_075)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_068) + CREATE(_076)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_132) + CREATE(_077)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_134) + CREATE(_078)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_138) + CREATE(_079)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_002) + CREATE(_080)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_125) + CREATE(_081)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_129) + CREATE(_082)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_048) + CREATE(_083)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_145) + CREATE(_084)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_101) + CREATE(_085)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_131) + CREATE(_086)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_011) + CREATE(_087)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_200) + CREATE(_088)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_070) + CREATE(_089)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_008) + CREATE(_090)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_107) + CREATE(_091)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_002) + CREATE(_092)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_180) + CREATE(_093)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_001) + CREATE(_094)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_120) + CREATE(_095)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_135) + CREATE(_096)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_116) + CREATE(_097)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_171) + CREATE(_098)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_122) + CREATE(_099)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_100) + CREATE(_100)-[:BUDDY]->(:StudyBuddy)<-[:BUDDY]-(_130) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 731 | + | +relationships | 1247 | + | +labels | 6 | + | +properties | 230 | diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/LargeIntegerEquality.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/LargeIntegerEquality.feature new file mode 100644 index 000000000..54becd78c --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/LargeIntegerEquality.feature @@ -0,0 +1,80 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: LargeIntegerEquality + + Background: + Given an empty graph + And having executed: + """ + CREATE (:Label {id: 4611686018427387905}) + """ + + Scenario: Does not lose precision + When executing query: + """ + MATCH (p:Label) + RETURN p.id + """ + Then the result should be: + | p.id | + | 4611686018427387905 | + And no side effects + + Scenario: Handling inlined equality of large integer + When executing query: + """ + MATCH (p:Label {id: 4611686018427387905}) + RETURN p.id + """ + Then the result should be: + | p.id | + | 4611686018427387905 | + And no side effects + + Scenario: Handling explicit equality of large integer + When executing query: + """ + MATCH (p:Label) + WHERE p.id = 4611686018427387905 + RETURN p.id + """ + Then the result should be: + | p.id | + | 4611686018427387905 | + And no side effects + + Scenario: Handling inlined equality of large integer, non-equal values + When executing query: + """ + MATCH (p:Label {id : 4611686018427387900}) + RETURN p.id + """ + Then the result should be: + | p.id | + And no side effects + + Scenario: Handling explicit equality of large integer, non-equal values + When executing query: + """ + MATCH (p:Label) + WHERE p.id = 4611686018427387900 + RETURN p.id + """ + Then the result should be: + | p.id | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/ListComprehension.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/ListComprehension.feature new file mode 100644 index 000000000..048339d05 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/ListComprehension.feature @@ -0,0 +1,75 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: ListComprehension + + Scenario: Returning a list comprehension + Given an empty graph + And having executed: + """ + CREATE (a:A) + CREATE (a)-[:T]->(:B), + (a)-[:T]->(:C) + """ + When executing query: + """ + MATCH p = (n)-->() + RETURN [x IN collect(p) | head(nodes(x))] AS p + """ + Then the result should be: + | p | + | [(:A), (:A)] | + And no side effects + + Scenario: Using a list comprehension in a WITH + Given an empty graph + And having executed: + """ + CREATE (a:A) + CREATE (a)-[:T]->(:B), + (a)-[:T]->(:C) + """ + When executing query: + """ + MATCH p = (n:A)-->() + WITH [x IN collect(p) | head(nodes(x))] AS p, count(n) AS c + RETURN p, c + """ + Then the result should be: + | p | c | + | [(:A), (:A)] | 2 | + And no side effects + + Scenario: Using a list comprehension in a WHERE + Given an empty graph + And having executed: + """ + CREATE (a:A {prop: 'c'}) + CREATE (a)-[:T]->(:B), + (a)-[:T]->(:C) + """ + When executing query: + """ + MATCH (n)-->(b) + WHERE n.prop IN [x IN labels(b) | lower(x)] + RETURN b + """ + Then the result should be: + | b | + | (:C) | + And no side effects + diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/Literals.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/Literals.feature new file mode 100644 index 000000000..5bb1a2db7 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/Literals.feature @@ -0,0 +1,131 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: Literals + + Background: + Given any graph + + Scenario: Return an integer + When executing query: + """ + RETURN 1 AS literal + """ + Then the result should be: + | literal | + | 1 | + And no side effects + + Scenario: Return a float + When executing query: + """ + RETURN 1.0 AS literal + """ + Then the result should be: + | literal | + | 1.0 | + And no side effects + + Scenario: Return a float in exponent form + When executing query: + """ + RETURN -1e-9 AS literal + """ + Then the result should be: + | literal | + | -.000000001 | + And no side effects + + Scenario: Return a boolean + When executing query: + """ + RETURN true AS literal + """ + Then the result should be: + | literal | + | true | + And no side effects + + Scenario: Return a single-quoted string + When executing query: + """ + RETURN '' AS literal + """ + Then the result should be: + | literal | + | '' | + And no side effects + + Scenario: Return a double-quoted string + When executing query: + """ + RETURN "" AS literal + """ + Then the result should be: + | literal | + | '' | + And no side effects + + Scenario: Return null + When executing query: + """ + RETURN null AS literal + """ + Then the result should be: + | literal | + | null | + And no side effects + + Scenario: Return an empty list + When executing query: + """ + RETURN [] AS literal + """ + Then the result should be: + | literal | + | [] | + And no side effects + + Scenario: Return a nonempty list + When executing query: + """ + RETURN [0, 1, 2] AS literal + """ + Then the result should be: + | literal | + | [0, 1, 2] | + And no side effects + + Scenario: Return an empty map + When executing query: + """ + RETURN {} AS literal + """ + Then the result should be: + | literal | + | {} | + And no side effects + + Scenario: Return a nonempty map + When executing query: + """ + RETURN {k1: 0, k2: 'string'} AS literal + """ + Then the result should be: + | literal | + | {k1: 0, k2: 'string'} | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/MatchAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/MatchAcceptance.feature new file mode 100644 index 000000000..15deadb08 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/MatchAcceptance.feature @@ -0,0 +1,552 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: MatchAcceptance + + Scenario: Path query should return results in written order + Given an empty graph + And having executed: + """ + CREATE (:Label1)<-[:TYPE]-(:Label2) + """ + When executing query: + """ + MATCH p = (a:Label1)<--(:Label2) + RETURN p + """ + Then the result should be: + | p | + | <(:Label1)<-[:TYPE]-(:Label2)> | + And no side effects + + Scenario: Longer path query should return results in written order + Given an empty graph + And having executed: + """ + CREATE (:Label1)<-[:T1]-(:Label2)-[:T2]->(:Label3) + """ + When executing query: + """ + MATCH p = (a:Label1)<--(:Label2)--() + RETURN p + """ + Then the result should be: + | p | + | <(:Label1)<-[:T1]-(:Label2)-[:T2]->(:Label3)> | + And no side effects + + Scenario: Use multiple MATCH clauses to do a Cartesian product + Given an empty graph + And having executed: + """ + CREATE ({value: 1}), + ({value: 2}), + ({value: 3}) + """ + When executing query: + """ + MATCH (n), (m) + RETURN n.value AS n, m.value AS m + """ + Then the result should be: + | n | m | + | 1 | 1 | + | 1 | 2 | + | 1 | 3 | + | 2 | 1 | + | 2 | 2 | + | 2 | 3 | + | 3 | 3 | + | 3 | 1 | + | 3 | 2 | + And no side effects + + Scenario: Use params in pattern matching predicates + Given an empty graph + And having executed: + """ + CREATE (:A)-[:T {foo: 'bar'}]->(:B {name: 'me'}) + """ + And parameters are: + | param | 'bar' | + When executing query: + """ + MATCH (a)-[r]->(b) + WHERE r.foo = $param + RETURN b + """ + Then the result should be: + | b | + | (:B {name: 'me'}) | + And no side effects + + Scenario: Filter out based on node prop name + Given an empty graph + And having executed: + """ + CREATE ({name: 'Someone'})<-[:X]-()-[:X]->({name: 'Andres'}) + """ + When executing query: + """ + MATCH ()-[rel:X]-(a) + WHERE a.name = 'Andres' + RETURN a + """ + Then the result should be: + | a | + | ({name: 'Andres'}) | + And no side effects + + Scenario: Honour the column name for RETURN items + Given an empty graph + And having executed: + """ + CREATE ({name: 'Someone'}) + """ + When executing query: + """ + MATCH (a) + WITH a.name AS a + RETURN a + """ + Then the result should be: + | a | + | 'Someone' | + And no side effects + + Scenario: Filter based on rel prop name + Given an empty graph + And having executed: + """ + CREATE (:A)<-[:KNOWS {name: 'monkey'}]-()-[:KNOWS {name: 'woot'}]->(:B) + """ + When executing query: + """ + MATCH (node)-[r:KNOWS]->(a) + WHERE r.name = 'monkey' + RETURN a + """ + Then the result should be: + | a | + | (:A) | + And no side effects + + Scenario: Cope with shadowed variables + Given an empty graph + And having executed: + """ + CREATE ({value: 1, name: 'King Kong'}), + ({value: 2, name: 'Ann Darrow'}) + """ + When executing query: + """ + MATCH (n) + WITH n.name AS n + RETURN n + """ + Then the result should be: + | n | + | 'Ann Darrow' | + | 'King Kong' | + And no side effects + + Scenario: Get neighbours + Given an empty graph + And having executed: + """ + CREATE (a:A {value: 1})-[:KNOWS]->(b:B {value: 2}) + """ + When executing query: + """ + MATCH (n1)-[rel:KNOWS]->(n2) + RETURN n1, n2 + """ + Then the result should be: + | n1 | n2 | + | (:A {value: 1}) | (:B {value: 2}) | + And no side effects + + Scenario: Get two related nodes + Given an empty graph + And having executed: + """ + CREATE (a:A {value: 1}), + (a)-[:KNOWS]->(b:B {value: 2}), + (a)-[:KNOWS]->(c:C {value: 3}) + """ + When executing query: + """ + MATCH ()-[rel:KNOWS]->(x) + RETURN x + """ + Then the result should be: + | x | + | (:B {value: 2}) | + | (:C {value: 3}) | + And no side effects + + Scenario: Get related to related to + Given an empty graph + And having executed: + """ + CREATE (a:A {value: 1})-[:KNOWS]->(b:B {value: 2})-[:FRIEND]->(c:C {value: 3}) + """ + When executing query: + """ + MATCH (n)-->(a)-->(b) + RETURN b + """ + Then the result should be: + | b | + | (:C {value: 3}) | + And no side effects + + Scenario: Handle comparison between node properties + Given an empty graph + And having executed: + """ + CREATE (a:A {animal: 'monkey'}), + (b:B {animal: 'cow'}), + (c:C {animal: 'monkey'}), + (d:D {animal: 'cow'}), + (a)-[:KNOWS]->(b), + (a)-[:KNOWS]->(c), + (d)-[:KNOWS]->(b), + (d)-[:KNOWS]->(c) + """ + When executing query: + """ + MATCH (n)-[rel]->(x) + WHERE n.animal = x.animal + RETURN n, x + """ + Then the result should be: + | n | x | + | (:A {animal: 'monkey'}) | (:C {animal: 'monkey'}) | + | (:D {animal: 'cow'}) | (:B {animal: 'cow'}) | + And no side effects + + Scenario: Return two subgraphs with bound undirected relationship + Given an empty graph + And having executed: + """ + CREATE (a:A {value: 1})-[:REL {name: 'r'}]->(b:B {value: 2}) + """ + When executing query: + """ + MATCH (a)-[r {name: 'r'}]-(b) + RETURN a, b + """ + Then the result should be: + | a | b | + | (:B {value: 2}) | (:A {value: 1}) | + | (:A {value: 1}) | (:B {value: 2}) | + And no side effects + + Scenario: Return two subgraphs with bound undirected relationship and optional relationship + Given an empty graph + And having executed: + """ + CREATE (a:A {value: 1})-[:REL {name: 'r1'}]->(b:B {value: 2})-[:REL {name: 'r2'}]->(c:C {value: 3}) + """ + When executing query: + """ + MATCH (a)-[r {name: 'r1'}]-(b) + OPTIONAL MATCH (b)-[r2]-(c) + WHERE r <> r2 + RETURN a, b, c + """ + Then the result should be: + | a | b | c | + | (:A {value: 1}) | (:B {value: 2}) | (:C {value: 3}) | + | (:B {value: 2}) | (:A {value: 1}) | null | + And no side effects + + Scenario: Rel type function works as expected + Given an empty graph + And having executed: + """ + CREATE (a:A {name: 'A'}), + (b:B {name: 'B'}), + (c:C {name: 'C'}), + (a)-[:KNOWS]->(b), + (a)-[:HATES]->(c) + """ + When executing query: + """ + MATCH (n {name: 'A'})-[r]->(x) + WHERE type(r) = 'KNOWS' + RETURN x + """ + Then the result should be: + | x | + | (:B {name: 'B'}) | + And no side effects + + Scenario: Walk alternative relationships + Given an empty graph + And having executed: + """ + CREATE (a {name: 'A'}), + (b {name: 'B'}), + (c {name: 'C'}), + (a)-[:KNOWS]->(b), + (a)-[:HATES]->(c), + (a)-[:WONDERS]->(c) + """ + When executing query: + """ + MATCH (n)-[r]->(x) + WHERE type(r) = 'KNOWS' OR type(r) = 'HATES' + RETURN r + """ + Then the result should be: + | r | + | [:KNOWS] | + | [:HATES] | + And no side effects + + Scenario: Handle OR in the WHERE clause + Given an empty graph + And having executed: + """ + CREATE (a:A {p1: 12}), + (b:B {p2: 13}), + (c:C) + """ + When executing query: + """ + MATCH (n) + WHERE n.p1 = 12 OR n.p2 = 13 + RETURN n + """ + Then the result should be: + | n | + | (:A {p1: 12}) | + | (:B {p2: 13}) | + And no side effects + + Scenario: Return a simple path + Given an empty graph + And having executed: + """ + CREATE (a:A {name: 'A'})-[:KNOWS]->(b:B {name: 'B'}) + """ + When executing query: + """ + MATCH p = (a {name: 'A'})-->(b) + RETURN p + """ + Then the result should be: + | p | + | <(:A {name: 'A'})-[:KNOWS]->(:B {name: 'B'})> | + And no side effects + + Scenario: Return a three node path + Given an empty graph + And having executed: + """ + CREATE (a:A {name: 'A'})-[:KNOWS]->(b:B {name: 'B'})-[:KNOWS]->(c:C {name: 'C'}) + """ + When executing query: + """ + MATCH p = (a {name: 'A'})-[rel1]->(b)-[rel2]->(c) + RETURN p + """ + Then the result should be: + | p | + | <(:A {name: 'A'})-[:KNOWS]->(:B {name: 'B'})-[:KNOWS]->(:C {name: 'C'})> | + And no side effects + + Scenario: Do not return anything because path length does not match + Given an empty graph + And having executed: + """ + CREATE (a:A {name: 'A'})-[:KNOWS]->(b:B {name: 'B'}) + """ + When executing query: + """ + MATCH p = (n)-->(x) + WHERE length(p) = 10 + RETURN x + """ + Then the result should be: + | x | + And no side effects + + Scenario: Pass the path length test + Given an empty graph + And having executed: + """ + CREATE (a:A {name: 'A'})-[:KNOWS]->(b:B {name: 'B'}) + """ + When executing query: + """ + MATCH p = (n)-->(x) + WHERE length(p) = 1 + RETURN x + """ + Then the result should be: + | x | + | (:B {name: 'B'}) | + And no side effects + + Scenario: Return relationships by fetching them from the path - starting from the end + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:REL {value: 1}]->(b:B)-[:REL {value: 2}]->(e:End) + """ + When executing query: + """ + MATCH p = (a)-[:REL*2..2]->(b:End) + RETURN relationships(p) + """ + Then the result should be: + | relationships(p) | + | [[:REL {value: 1}], [:REL {value: 2}]] | + And no side effects + + Scenario: Return relationships by fetching them from the path + Given an empty graph + And having executed: + """ + CREATE (s:Start)-[:REL {value: 1}]->(b:B)-[:REL {value: 2}]->(c:C) + """ + When executing query: + """ + MATCH p = (a:Start)-[:REL*2..2]->(b) + RETURN relationships(p) + """ + Then the result should be: + | relationships(p) | + | [[:REL {value: 1}], [:REL {value: 2}]] | + And no side effects + + Scenario: Return relationships by collecting them as a list - directed, one way + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:REL {value: 1}]->(b:B)-[:REL {value: 2}]->(e:End) + """ + When executing query: + """ + MATCH (a)-[r:REL*2..2]->(b:End) + RETURN r + """ + Then the result should be: + | r | + | [[:REL {value: 1}], [:REL {value: 2}]] | + And no side effects + + Scenario: Return relationships by collecting them as a list - undirected, starting from two extremes + Given an empty graph + And having executed: + """ + CREATE (a:End)-[:REL {value: 1}]->(b:B)-[:REL {value: 2}]->(c:End) + """ + When executing query: + """ + MATCH (a)-[r:REL*2..2]-(b:End) + RETURN r + """ + Then the result should be: + | r | + | [[:REL {value:1}], [:REL {value:2}]] | + | [[:REL {value:2}], [:REL {value:1}]] | + And no side effects + + Scenario: Return relationships by collecting them as a list - undirected, starting from one extreme + Given an empty graph + And having executed: + """ + CREATE (s:Start)-[:REL {value: 1}]->(b:B)-[:REL {value: 2}]->(c:C) + """ + When executing query: + """ + MATCH (a:Start)-[r:REL*2..2]-(b) + RETURN r + """ + Then the result should be: + | r | + | [[:REL {value: 1}], [:REL {value: 2}]] | + And no side effects + + Scenario: Return a var length path + Given an empty graph + And having executed: + """ + CREATE (a:A {name: 'A'})-[:KNOWS {value: 1}]->(b:B {name: 'B'})-[:KNOWS {value: 2}]->(c:C {name: 'C'}) + """ + When executing query: + """ + MATCH p = (n {name: 'A'})-[:KNOWS*1..2]->(x) + RETURN p + """ + Then the result should be: + | p | + | <(:A {name: 'A'})-[:KNOWS {value: 1}]->(:B {name: 'B'})> | + | <(:A {name: 'A'})-[:KNOWS {value: 1}]->(:B {name: 'B'})-[:KNOWS {value: 2}]->(:C {name: 'C'})> | + And no side effects + + Scenario: Return a var length path of length zero + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:REL]->(b:B) + """ + When executing query: + """ + MATCH p = (a)-[*0..1]->(b) + RETURN a, b, length(p) AS l + """ + Then the result should be: + | a | b | l | + | (:A) | (:A) | 0 | + | (:B) | (:B) | 0 | + | (:A) | (:B) | 1 | + And no side effects + + Scenario: Return a named var length path of length zero + Given an empty graph + And having executed: + """ + CREATE (a:A {name: 'A'})-[:KNOWS]->(b:B {name: 'B'})-[:FRIEND]->(c:C {name: 'C'}) + """ + When executing query: + """ + MATCH p = (a {name: 'A'})-[:KNOWS*0..1]->(b)-[:FRIEND*0..1]->(c) + RETURN p + """ + Then the result should be: + | p | + | <(:A {name: 'A'})> | + | <(:A {name: 'A'})-[:KNOWS]->(:B {name: 'B'})> | + | <(:A {name: 'A'})-[:KNOWS]->(:B {name: 'B'})-[:FRIEND]->(:C {name: 'C'})> | + And no side effects + + Scenario: Accept skip zero + Given any graph + When executing query: + """ + MATCH (n) + WHERE 1 = 0 + RETURN n SKIP 0 + """ + Then the result should be: + | n | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/MatchAcceptance2.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/MatchAcceptance2.feature new file mode 100644 index 000000000..029b4272c --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/MatchAcceptance2.feature @@ -0,0 +1,1845 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: MatchAcceptance2 + + Scenario: Do not return non-existent nodes + Given an empty graph + When executing query: + """ + MATCH (n) + RETURN n + """ + Then the result should be: + | n | + And no side effects + + Scenario: Do not return non-existent relationships + Given an empty graph + When executing query: + """ + MATCH ()-[r]->() + RETURN r + """ + Then the result should be: + | r | + And no side effects + + Scenario: Do not fail when evaluating predicates with illegal operations if the AND'ed predicate evaluates to false + Given an empty graph + And having executed: + """ + CREATE (root:Root {name: 'x'}), + (child1:TextNode {id: 'text'}), + (child2:IntNode {id: 0}) + CREATE (root)-[:T]->(child1), + (root)-[:T]->(child2) + """ + When executing query: + """ + MATCH (:Root {name: 'x'})-->(i:TextNode) + WHERE i.id > 'te' + RETURN i + """ + Then the result should be: + | i | + | (:TextNode {id: 'text'}) | + And no side effects + + Scenario: Do not fail when evaluating predicates with illegal operations if the OR'd predicate evaluates to true + Given an empty graph + And having executed: + """ + CREATE (root:Root {name: 'x'}), + (child1:TextNode {id: 'text'}), + (child2:IntNode {id: 0}) + CREATE (root)-[:T]->(child1), + (root)-[:T]->(child2) + """ + When executing query: + """ + MATCH (:Root {name: 'x'})-->(i) + WHERE exists(i.id) OR i.id > 'te' + RETURN i + """ + Then the result should be: + | i | + | (:TextNode {id: 'text'}) | + | (:IntNode {id: 0}) | + And no side effects + + Scenario: Aggregation with named paths + Given an empty graph + And having executed: + """ + CREATE (n1 {num: 1}), (n2 {num: 2}), + (n3 {num: 3}), (n4 {num: 4}) + CREATE (n1)-[:T]->(n2), + (n3)-[:T]->(n4) + """ + When executing query: + """ + MATCH p = ()-[*]->() + WITH count(*) AS count, p AS p + WITH nodes(p) AS nodes + RETURN * + """ + Then the result should be: + | nodes | + | [({num: 1}), ({num: 2})] | + | [({num: 3}), ({num: 4})] | + And no side effects + + Scenario: Zero-length variable length pattern in the middle of the pattern + Given an empty graph + And having executed: + """ + CREATE (a {name: 'A'}), (b {name: 'B'}), + (c {name: 'C'}), ({name: 'D'}), + ({name: 'E'}) + CREATE (a)-[:CONTAINS]->(b), + (b)-[:FRIEND]->(c) + """ + When executing query: + """ + MATCH (a {name: 'A'})-[:CONTAINS*0..1]->(b)-[:FRIEND*0..1]->(c) + RETURN a, b, c + """ + Then the result should be: + | a | b | c | + | ({name: 'A'}) | ({name: 'A'}) | ({name: 'A'}) | + | ({name: 'A'}) | ({name: 'B'}) | ({name: 'B'}) | + | ({name: 'A'}) | ({name: 'B'}) | ({name: 'C'}) | + And no side effects + + Scenario: Simple variable length pattern + Given an empty graph + And having executed: + """ + CREATE (a {name: 'A'}), (b {name: 'B'}), + (c {name: 'C'}), (d {name: 'D'}) + CREATE (a)-[:CONTAINS]->(b), + (b)-[:CONTAINS]->(c), + (c)-[:CONTAINS]->(d) + """ + When executing query: + """ + MATCH (a {name: 'A'})-[*]->(x) + RETURN x + """ + Then the result should be: + | x | + | ({name: 'B'}) | + | ({name: 'C'}) | + | ({name: 'D'}) | + And no side effects + + Scenario: Variable length relationship without lower bound + Given an empty graph + And having executed: + """ + CREATE (a {name: 'A'}), (b {name: 'B'}), + (c {name: 'C'}) + CREATE (a)-[:KNOWS]->(b), + (b)-[:KNOWS]->(c) + """ + When executing query: + """ + MATCH p = ({name: 'A'})-[:KNOWS*..2]->() + RETURN p + """ + Then the result should be: + | p | + | <({name: 'A'})-[:KNOWS]->({name: 'B'})> | + | <({name: 'A'})-[:KNOWS]->({name: 'B'})-[:KNOWS]->({name: 'C'})> | + And no side effects + + Scenario: Variable length relationship without bounds + Given an empty graph + And having executed: + """ + CREATE (a {name: 'A'}), (b {name: 'B'}), + (c {name: 'C'}) + CREATE (a)-[:KNOWS]->(b), + (b)-[:KNOWS]->(c) + """ + When executing query: + """ + MATCH p = ({name: 'A'})-[:KNOWS*..]->() + RETURN p + """ + Then the result should be: + | p | + | <({name: 'A'})-[:KNOWS]->({name: 'B'})> | + | <({name: 'A'})-[:KNOWS]->({name: 'B'})-[:KNOWS]->({name: 'C'})> | + And no side effects + + Scenario: Returning bound nodes that are not part of the pattern + Given an empty graph + And having executed: + """ + CREATE (a {name: 'A'}), (b {name: 'B'}), + (c {name: 'C'}) + CREATE (a)-[:KNOWS]->(b) + """ + When executing query: + """ + MATCH (a {name: 'A'}), (c {name: 'C'}) + MATCH (a)-->(b) + RETURN a, b, c + """ + Then the result should be: + | a | b | c | + | ({name: 'A'}) | ({name: 'B'}) | ({name: 'C'}) | + And no side effects + + Scenario: Two bound nodes pointing to the same node + Given an empty graph + And having executed: + """ + CREATE (a {name: 'A'}), (b {name: 'B'}), + (x1 {name: 'x1'}), (x2 {name: 'x2'}) + CREATE (a)-[:KNOWS]->(x1), + (a)-[:KNOWS]->(x2), + (b)-[:KNOWS]->(x1), + (b)-[:KNOWS]->(x2) + """ + When executing query: + """ + MATCH (a {name: 'A'}), (b {name: 'B'}) + MATCH (a)-->(x)<-->(b) + RETURN x + """ + Then the result should be: + | x | + | ({name: 'x1'}) | + | ({name: 'x2'}) | + And no side effects + + Scenario: Three bound nodes pointing to the same node + Given an empty graph + And having executed: + """ + CREATE (a {name: 'A'}), (b {name: 'B'}), (c {name: 'C'}), + (x1 {name: 'x1'}), (x2 {name: 'x2'}) + CREATE (a)-[:KNOWS]->(x1), + (a)-[:KNOWS]->(x2), + (b)-[:KNOWS]->(x1), + (b)-[:KNOWS]->(x2), + (c)-[:KNOWS]->(x1), + (c)-[:KNOWS]->(x2) + """ + When executing query: + """ + MATCH (a {name: 'A'}), (b {name: 'B'}), (c {name: 'C'}) + MATCH (a)-->(x), (b)-->(x), (c)-->(x) + RETURN x + """ + Then the result should be: + | x | + | ({name: 'x1'}) | + | ({name: 'x2'}) | + And no side effects + + Scenario: Three bound nodes pointing to the same node with extra connections + Given an empty graph + And having executed: + """ + CREATE (a {name: 'a'}), (b {name: 'b'}), (c {name: 'c'}), + (d {name: 'd'}), (e {name: 'e'}), (f {name: 'f'}), + (g {name: 'g'}), (h {name: 'h'}), (i {name: 'i'}), + (j {name: 'j'}), (k {name: 'k'}) + CREATE (a)-[:KNOWS]->(d), + (a)-[:KNOWS]->(e), + (a)-[:KNOWS]->(f), + (a)-[:KNOWS]->(g), + (a)-[:KNOWS]->(i), + (b)-[:KNOWS]->(d), + (b)-[:KNOWS]->(e), + (b)-[:KNOWS]->(f), + (b)-[:KNOWS]->(h), + (b)-[:KNOWS]->(k), + (c)-[:KNOWS]->(d), + (c)-[:KNOWS]->(e), + (c)-[:KNOWS]->(h), + (c)-[:KNOWS]->(g), + (c)-[:KNOWS]->(j) + """ + When executing query: + """ + MATCH (a {name: 'a'}), (b {name: 'b'}), (c {name: 'c'}) + MATCH (a)-->(x), (b)-->(x), (c)-->(x) + RETURN x + """ + Then the result should be: + | x | + | ({name: 'd'}) | + | ({name: 'e'}) | + And no side effects + + Scenario: MATCH with OPTIONAL MATCH in longer pattern + Given an empty graph + And having executed: + """ + CREATE (a {name: 'A'}), (b {name: 'B'}), (c {name: 'C'}) + CREATE (a)-[:KNOWS]->(b), + (b)-[:KNOWS]->(c) + """ + When executing query: + """ + MATCH (a {name: 'A'}) + OPTIONAL MATCH (a)-[:KNOWS]->()-[:KNOWS]->(foo) + RETURN foo + """ + Then the result should be: + | foo | + | ({name: 'C'}) | + And no side effects + + Scenario: Optionally matching named paths + Given an empty graph + And having executed: + """ + CREATE (a {name: 'A'}), (b {name: 'B'}), (c {name: 'C'}) + CREATE (a)-[:X]->(b) + """ + When executing query: + """ + MATCH (a {name: 'A'}), (x) + WHERE x.name IN ['B', 'C'] + OPTIONAL MATCH p = (a)-->(x) + RETURN x, p + """ + Then the result should be: + | x | p | + | ({name: 'B'}) | <({name: 'A'})-[:X]->({name: 'B'})> | + | ({name: 'C'}) | null | + And no side effects + + Scenario: Optionally matching named paths with single and variable length patterns + Given an empty graph + And having executed: + """ + CREATE (a {name: 'A'}), (b {name: 'B'}) + CREATE (a)-[:X]->(b) + """ + When executing query: + """ + MATCH (a {name: 'A'}) + OPTIONAL MATCH p = (a)-->(b)-[*]->(c) + RETURN p + """ + Then the result should be: + | p | + | null | + And no side effects + + Scenario: Optionally matching named paths with variable length patterns + Given an empty graph + And having executed: + """ + CREATE (a {name: 'A'}), (b {name: 'B'}), (c {name: 'C'}) + CREATE (a)-[:X]->(b) + """ + When executing query: + """ + MATCH (a {name: 'A'}), (x) + WHERE x.name IN ['B', 'C'] + OPTIONAL MATCH p = (a)-[r*]->(x) + RETURN r, x, p + """ + Then the result should be: + | r | x | p | + | [[:X]] | ({name: 'B'}) | <({name: 'A'})-[:X]->({name: 'B'})> | + | null | ({name: 'C'}) | null | + And no side effects + + Scenario: Matching variable length patterns from a bound node + Given an empty graph + And having executed: + """ + CREATE (a:A), (b), (c) + CREATE (a)-[:X]->(b), + (b)-[:Y]->(c) + """ + When executing query: + """ + MATCH (a:A) + MATCH (a)-[r*2]->() + RETURN r + """ + Then the result should be (ignoring element order for lists): + | r | + | [[:X], [:Y]] | + And no side effects + + Scenario: Excluding connected nodes + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B {id: 1}), (:B {id: 2}) + CREATE (a)-[:T]->(b) + """ + When executing query: + """ + MATCH (a:A), (other:B) + OPTIONAL MATCH (a)-[r]->(other) + WITH other WHERE r IS NULL + RETURN other + """ + Then the result should be: + | other | + | (:B {id: 2}) | + And no side effects + + Scenario: Do not fail when predicates on optionally matched and missed nodes are invalid + Given an empty graph + And having executed: + """ + CREATE (a), (b {name: 'Mark'}) + CREATE (a)-[:T]->(b) + """ + When executing query: + """ + MATCH (n)-->(x0) + OPTIONAL MATCH (x0)-->(x1) + WHERE x1.foo = 'bar' + RETURN x0.name + """ + Then the result should be: + | x0.name | + | 'Mark' | + And no side effects + + Scenario: MATCH and OPTIONAL MATCH on same pattern + Given an empty graph + And having executed: + """ + CREATE (a {name: 'A'}), (b:B {name: 'B'}), (c:C {name: 'C'}) + CREATE (a)-[:T]->(b), + (a)-[:T]->(c) + """ + When executing query: + """ + MATCH (a)-->(b) + WHERE b:B + OPTIONAL MATCH (a)-->(c) + WHERE c:C + RETURN a.name + """ + Then the result should be: + | a.name | + | 'A' | + And no side effects + + Scenario: Matching using an undirected pattern + Given an empty graph + And having executed: + """ + CREATE (:A {id: 0})-[:ADMIN]->(:B {id: 1}) + """ + When executing query: + """ + MATCH (a)-[:ADMIN]-(b) + WHERE a:A + RETURN a.id, b.id + """ + Then the result should be: + | a.id | b.id | + | 0 | 1 | + And no side effects + + Scenario: Matching all nodes + Given an empty graph + And having executed: + """ + CREATE (:A), (:B) + """ + When executing query: + """ + MATCH (n) + RETURN n + """ + Then the result should be: + | n | + | (:A) | + | (:B) | + And no side effects + + Scenario: Comparing nodes for equality + Given an empty graph + And having executed: + """ + CREATE (:A), (:B) + """ + When executing query: + """ + MATCH (a), (b) + WHERE a <> b + RETURN a, b + """ + Then the result should be: + | a | b | + | (:A) | (:B) | + | (:B) | (:A) | + And no side effects + + Scenario: Matching using self-referencing pattern returns no result + Given an empty graph + And having executed: + """ + CREATE (a), (b), (c) + CREATE (a)-[:T]->(b), + (b)-[:T]->(c) + """ + When executing query: + """ + MATCH (a)-->(b), (b)-->(b) + RETURN b + """ + Then the result should be: + | b | + And no side effects + + Scenario: Variable length relationship in OPTIONAL MATCH + Given an empty graph + And having executed: + """ + CREATE (:A), (:B) + """ + When executing query: + """ + MATCH (a:A), (b:B) + OPTIONAL MATCH (a)-[r*]-(b) + WHERE r IS NULL + AND a <> b + RETURN b + """ + Then the result should be: + | b | + | (:B) | + And no side effects + + Scenario: Matching using relationship predicate with multiples of the same type + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:T]->(b) + """ + When executing query: + """ + MATCH (a)-[:T|:T]->(b) + RETURN b + """ + Then the result should be: + | b | + | (:B) | + And no side effects + + Scenario: ORDER BY with LIMIT + Given an empty graph + And having executed: + """ + CREATE (a:A), (n1 {x: 1}), (n2 {x: 2}), + (m1), (m2) + CREATE (a)-[:T]->(n1), + (n1)-[:T]->(m1), + (a)-[:T]->(n2), + (n2)-[:T]->(m2) + """ + When executing query: + """ + MATCH (a:A)-->(n)-->(m) + RETURN n.x, count(*) + ORDER BY n.x + LIMIT 1000 + """ + Then the result should be, in order: + | n.x | count(*) | + | 1 | 1 | + | 2 | 1 | + And no side effects + + Scenario: Simple node property predicate + Given an empty graph + And having executed: + """ + CREATE ({foo: 'bar'}) + """ + When executing query: + """ + MATCH (n) + WHERE n.foo = 'bar' + RETURN n + """ + Then the result should be: + | n | + | ({foo: 'bar'}) | + And no side effects + + Scenario: Handling direction of named paths + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:T]->(b:B) + """ + When executing query: + """ + MATCH p = (b)<--(a) + RETURN p + """ + Then the result should be: + | p | + | <(:B)<-[:T]-(:A)> | + And no side effects + + Scenario: Simple OPTIONAL MATCH on empty graph + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (n) + RETURN n + """ + Then the result should be: + | n | + | null | + And no side effects + + Scenario: OPTIONAL MATCH with previously bound nodes + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + OPTIONAL MATCH (n)-[:NOT_EXIST]->(x) + RETURN n, x + """ + Then the result should be: + | n | x | + | () | null | + And no side effects + + Scenario: `collect()` filtering nulls + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + OPTIONAL MATCH (n)-[:NOT_EXIST]->(x) + RETURN n, collect(x) + """ + Then the result should be: + | n | collect(x) | + | () | [] | + And no side effects + + Scenario: Multiple anonymous nodes in a pattern + Given an empty graph + And having executed: + """ + CREATE (:A) + """ + When executing query: + """ + MATCH (a)<--()<--(b)-->()-->(c) + WHERE a:A + RETURN c + """ + Then the result should be: + | c | + And no side effects + + Scenario: Matching a relationship pattern using a label predicate + Given an empty graph + And having executed: + """ + CREATE (a), (b1:Foo), (b2) + CREATE (a)-[:T]->(b1), + (a)-[:T]->(b2) + """ + When executing query: + """ + MATCH (a)-->(b:Foo) + RETURN b + """ + Then the result should be: + | b | + | (:Foo) | + And no side effects + + Scenario: Matching a relationship pattern using a label predicate on both sides + Given an empty graph + And having executed: + """ + CREATE (:A)-[:T1]->(:B), + (:B)-[:T2]->(:A), + (:B)-[:T3]->(:B), + (:A)-[:T4]->(:A) + """ + When executing query: + """ + MATCH (:A)-[r]->(:B) + RETURN r + """ + Then the result should be: + | r | + | [:T1] | + And no side effects + + Scenario: Matching nodes using multiple labels + Given an empty graph + And having executed: + """ + CREATE (:A:B:C), (:A:B), (:A:C), (:B:C), + (:A), (:B), (:C) + """ + When executing query: + """ + MATCH (a:A:B:C) + RETURN a + """ + Then the result should be: + | a | + | (:A:B:C) | + And no side effects + + Scenario: Returning label predicate expression + Given an empty graph + And having executed: + """ + CREATE (), (:Foo) + """ + When executing query: + """ + MATCH (n) + RETURN (n:Foo) + """ + Then the result should be: + | (n:Foo) | + | true | + | false | + And no side effects + + Scenario: Matching with many predicates and larger pattern + Given an empty graph + And having executed: + """ + CREATE (advertiser {name: 'advertiser1', id: 0}), + (thing {name: 'Color', id: 1}), + (red {name: 'red'}), + (p1 {name: 'product1'}), + (p2 {name: 'product4'}) + CREATE (advertiser)-[:ADV_HAS_PRODUCT]->(p1), + (advertiser)-[:ADV_HAS_PRODUCT]->(p2), + (thing)-[:AA_HAS_VALUE]->(red), + (p1)-[:AP_HAS_VALUE]->(red), + (p2)-[:AP_HAS_VALUE]->(red) + """ + And parameters are: + | 1 | 0 | + | 2 | 1 | + When executing query: + """ + MATCH (advertiser)-[:ADV_HAS_PRODUCT]->(out)-[:AP_HAS_VALUE]->(red)<-[:AA_HAS_VALUE]-(a) + WHERE advertiser.id = $1 + AND a.id = $2 + AND red.name = 'red' + AND out.name = 'product1' + RETURN out.name + """ + Then the result should be: + | out.name | + | 'product1' | + And no side effects + + Scenario: Matching using a simple pattern with label predicate + Given an empty graph + And having executed: + """ + CREATE (a:Person {name: 'Alice'}), (b:Person {name: 'Bob'}), + (c), (d) + CREATE (a)-[:T]->(c), + (b)-[:T]->(d) + """ + When executing query: + """ + MATCH (n:Person)-->() + WHERE n.name = 'Bob' + RETURN n + """ + Then the result should be: + | n | + | (:Person {name: 'Bob'}) | + And no side effects + + Scenario: Matching disconnected patterns + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B), (c:C) + CREATE (a)-[:T]->(b), + (a)-[:T]->(c) + """ + When executing query: + """ + MATCH (a)-->(b) + MATCH (c)-->(d) + RETURN a, b, c, d + """ + Then the result should be: + | a | b | c | d | + | (:A) | (:B) | (:A) | (:B) | + | (:A) | (:B) | (:A) | (:C) | + | (:A) | (:C) | (:A) | (:B) | + | (:A) | (:C) | (:A) | (:C) | + And no side effects + + Scenario: Non-optional matches should not return nulls + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B {id: 1}), (c:C {id: 2}), (d:D) + CREATE (a)-[:T]->(b), + (a)-[:T]->(c), + (a)-[:T]->(d), + (b)-[:T]->(c), + (b)-[:T]->(d), + (c)-[:T]->(d) + """ + When executing query: + """ + MATCH (a)--(b)--(c)--(d)--(a), (b)--(d) + WHERE a.id = 1 + AND c.id = 2 + RETURN d + """ + Then the result should be: + | d | + | (:A) | + | (:D) | + And no side effects + + Scenario: Handling cyclic patterns + Given an empty graph + And having executed: + """ + CREATE (a {name: 'a'}), (b {name: 'b'}), (c {name: 'c'}) + CREATE (a)-[:A]->(b), + (b)-[:B]->(a), + (b)-[:B]->(c) + """ + When executing query: + """ + MATCH (a)-[:A]->()-[:B]->(a) + RETURN a.name + """ + Then the result should be: + | a.name | + | 'a' | + And no side effects + + Scenario: Handling cyclic patterns when separated into two parts + Given an empty graph + And having executed: + """ + CREATE (a {name: 'a'}), (b {name: 'b'}), (c {name: 'c'}) + CREATE (a)-[:A]->(b), + (b)-[:B]->(a), + (b)-[:B]->(c) + """ + When executing query: + """ + MATCH (a)-[:A]->(b), (b)-[:B]->(a) + RETURN a.name + """ + Then the result should be: + | a.name | + | 'a' | + And no side effects + + Scenario: Handling fixed-length variable length pattern + Given an empty graph + And having executed: + """ + CREATE ()-[:T]->() + """ + When executing query: + """ + MATCH (a)-[r*1..1]->(b) + RETURN r + """ + Then the result should be: + | r | + | [[:T]] | + And no side effects + + Scenario: Matching from null nodes should return no results owing to finding no matches + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (a) + WITH a + MATCH (a)-->(b) + RETURN b + """ + Then the result should be: + | b | + And no side effects + + Scenario: Matching from null nodes should return no results owing to matches being filtered out + Given an empty graph + And having executed: + """ + CREATE ()-[:T]->() + """ + When executing query: + """ + OPTIONAL MATCH (a:Label) + WITH a + MATCH (a)-->(b) + RETURN b + """ + Then the result should be: + | b | + And no side effects + + Scenario: Optionally matching from null nodes should return null + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (a) + WITH a + OPTIONAL MATCH (a)-->(b) + RETURN b + """ + Then the result should be: + | b | + | null | + And no side effects + + Scenario: OPTIONAL MATCH returns null + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (a) + RETURN a + """ + Then the result should be: + | a | + | null | + And no side effects + + Scenario: Zero-length named path + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH p = (a) + RETURN p + """ + Then the result should be: + | p | + | <()> | + And no side effects + + Scenario: Variable-length named path + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH p = ()-[*0..]->() + RETURN p + """ + Then the result should be: + | p | + | <()> | + And no side effects + + Scenario: Matching with aggregation + Given an empty graph + And having executed: + """ + CREATE ({prop: 42}) + """ + When executing query: + """ + MATCH (n) + RETURN n.prop AS n, count(n) AS count + """ + Then the result should be: + | n | count | + | 42 | 1 | + And no side effects + + Scenario: Matching using a relationship that is already bound + Given an empty graph + And having executed: + """ + CREATE ()-[:T1]->(), + ()-[:T2]->() + """ + When executing query: + """ + MATCH ()-[r1]->() + WITH r1 AS r2 + MATCH ()-[r2]->() + RETURN r2 AS rel + """ + Then the result should be: + | rel | + | [:T1] | + | [:T2] | + And no side effects + + Scenario: Matching using a relationship that is already bound, in conjunction with aggregation + Given an empty graph + And having executed: + """ + CREATE ()-[:T1]->(), + ()-[:T2]->() + """ + When executing query: + """ + MATCH ()-[r1]->() + WITH r1 AS r2, count(*) AS c + ORDER BY c + MATCH ()-[r2]->() + RETURN r2 AS rel + """ + Then the result should be: + | rel | + | [:T1] | + | [:T2] | + And no side effects + + Scenario: Matching using a relationship that is already bound, in conjunction with aggregation and ORDER BY + Given an empty graph + And having executed: + """ + CREATE ()-[:T1 {id: 0}]->(), + ()-[:T2 {id: 1}]->() + """ + When executing query: + """ + MATCH (a)-[r]->(b) + WITH a, r, b, count(*) AS c + ORDER BY c + MATCH (a)-[r]->(b) + RETURN r AS rel + ORDER BY rel.id + """ + Then the result should be, in order: + | rel | + | [:T1 {id: 0}] | + | [:T2 {id: 1}] | + And no side effects + + Scenario: Matching with LIMIT and optionally matching using a relationship that is already bound + Given an empty graph + And having executed: + """ + CREATE (:A)-[:T]->(:B) + """ + When executing query: + """ + MATCH ()-[r]->() + WITH r + LIMIT 1 + OPTIONAL MATCH (a2)-[r]->(b2) + RETURN a2, r, b2 + """ + Then the result should be: + | a2 | r | b2 | + | (:A) | [:T] | (:B) | + And no side effects + + Scenario: Matching with LIMIT and optionally matching using a relationship and node that are both already bound + Given an empty graph + And having executed: + """ + CREATE (:A)-[:T]->(:B) + """ + When executing query: + """ + MATCH (a1)-[r]->() + WITH r, a1 + LIMIT 1 + OPTIONAL MATCH (a1)-[r]->(b2) + RETURN a1, r, b2 + """ + Then the result should be: + | a1 | r | b2 | + | (:A) | [:T] | (:B) | + And no side effects + + Scenario: Matching with LIMIT, then matching again using a relationship and node that are both already bound along with an additional predicate + Given an empty graph + And having executed: + """ + CREATE ()-[:T]->() + """ + When executing query: + """ + MATCH (a1)-[r]->() + WITH r, a1 + LIMIT 1 + MATCH (a1:X)-[r]->(b2) + RETURN a1, r, b2 + """ + Then the result should be: + | a1 | r | b2 | + And no side effects + + Scenario: Matching with LIMIT and predicates, then matching again using a relationship and node that are both already bound along with a duplicate predicate + Given an empty graph + And having executed: + """ + CREATE (:X:Y)-[:T]->() + """ + When executing query: + """ + MATCH (a1:X:Y)-[r]->() + WITH r, a1 + LIMIT 1 + MATCH (a1:Y)-[r]->(b2) + RETURN a1, r, b2 + """ + Then the result should be: + | a1 | r | b2 | + | (:X:Y) | [:T] | () | + And no side effects + + Scenario: Matching twice with conflicting relationship types on same relationship + Given an empty graph + And having executed: + """ + CREATE ()-[:T]->() + """ + When executing query: + """ + MATCH (a1)-[r:T]->() + WITH r, a1 + LIMIT 1 + MATCH (a1)-[r:Y]->(b2) + RETURN a1, r, b2 + """ + Then the result should be: + | a1 | r | b2 | + And no side effects + + Scenario: Matching twice with duplicate relationship types on same relationship + Given an empty graph + And having executed: + """ + CREATE (:A)-[:T]->(:B) + """ + When executing query: + """ + MATCH (a1)-[r:T]->() WITH r, a1 + LIMIT 1 + MATCH (a1)-[r:T]->(b2) + RETURN a1, r, b2 + """ + Then the result should be: + | a1 | r | b2 | + | (:A) | [:T] | (:B) | + And no side effects + + Scenario: Matching relationships into a list and matching variable length using the list + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B), (c:C) + CREATE (a)-[:Y]->(b), + (b)-[:Y]->(c) + """ + When executing query: + """ + MATCH ()-[r1]->()-[r2]->() + WITH [r1, r2] AS rs + LIMIT 1 + MATCH (first)-[rs*]->(second) + RETURN first, second + """ + Then the result should be: + | first | second | + | (:A) | (:C) | + And no side effects + + Scenario: Matching relationships into a list and matching variable length using the list, with bound nodes + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B), (c:C) + CREATE (a)-[:Y]->(b), + (b)-[:Y]->(c) + """ + When executing query: + """ + MATCH (a)-[r1]->()-[r2]->(b) + WITH [r1, r2] AS rs, a AS first, b AS second + LIMIT 1 + MATCH (first)-[rs*]->(second) + RETURN first, second + """ + Then the result should be: + | first | second | + | (:A) | (:C) | + And no side effects + + Scenario: Matching relationships into a list and matching variable length using the list, with bound nodes, wrong direction + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B), (c:C) + CREATE (a)-[:Y]->(b), + (b)-[:Y]->(c) + """ + When executing query: + """ + MATCH (a)-[r1]->()-[r2]->(b) + WITH [r1, r2] AS rs, a AS second, b AS first + LIMIT 1 + MATCH (first)-[rs*]->(second) + RETURN first, second + """ + Then the result should be: + | first | second | + And no side effects + + Scenario: Matching and optionally matching with bound nodes in reverse direction + Given an empty graph + And having executed: + """ + CREATE (:A)-[:T]->(:B) + """ + When executing query: + """ + MATCH (a1)-[r]->() + WITH r, a1 + LIMIT 1 + OPTIONAL MATCH (a1)<-[r]-(b2) + RETURN a1, r, b2 + """ + Then the result should be: + | a1 | r | b2 | + | (:A) | [:T] | null | + And no side effects + + Scenario: Matching and optionally matching with unbound nodes and equality predicate in reverse direction + Given an empty graph + And having executed: + """ + CREATE (:A)-[:T]->(:B) + """ + When executing query: + """ + MATCH (a1)-[r]->() + WITH r, a1 + LIMIT 1 + OPTIONAL MATCH (a2)<-[r]-(b2) + WHERE a1 = a2 + RETURN a1, r, b2, a2 + """ + Then the result should be: + | a1 | r | b2 | a2 | + | (:A) | [:T] | null | null | + And no side effects + + Scenario: Fail when using property access on primitive type + Given an empty graph + And having executed: + """ + CREATE ({prop: 42}) + """ + When executing query: + """ + MATCH (n) + WITH n.prop AS n2 + RETURN n2.prop + """ + Then a TypeError should be raised at runtime: PropertyAccessOnNonMap + + Scenario: Matching and returning ordered results, with LIMIT + Given an empty graph + And having executed: + """ + CREATE ({bar: 1}), ({bar: 3}), ({bar: 2}) + """ + When executing query: + """ + MATCH (foo) + RETURN foo.bar AS x + ORDER BY x DESC + LIMIT 4 + """ + Then the result should be, in order: + | x | + | 3 | + | 2 | + | 1 | + And no side effects + + Scenario: Counting an empty graph + Given an empty graph + When executing query: + """ + MATCH (a) + RETURN count(a) > 0 + """ + Then the result should be: + | count(a) > 0 | + | false | + And no side effects + + Scenario: Matching variable length pattern with property predicate + Given an empty graph + And having executed: + """ + CREATE (a:Artist:A), (b:Artist:B), (c:Artist:C) + CREATE (a)-[:WORKED_WITH {year: 1987}]->(b), + (b)-[:WORKED_WITH {year: 1988}]->(c) + """ + When executing query: + """ + MATCH (a:Artist)-[:WORKED_WITH* {year: 1988}]->(b:Artist) + RETURN * + """ + Then the result should be: + | a | b | + | (:Artist:B) | (:Artist:C) | + And no side effects + + Scenario: Variable length pattern checking labels on endnodes + Given an empty graph + And having executed: + """ + CREATE (a:Label {id: 0}), (b:Label {id: 1}), (c:Label {id: 2}) + CREATE (a)-[:T]->(b), + (b)-[:T]->(c) + """ + When executing query: + """ + MATCH (a), (b) + WHERE a.id = 0 + AND (a)-[:T]->(b:Label) + OR (a)-[:T*]->(b:MissingLabel) + RETURN DISTINCT b + """ + Then the result should be: + | b | + | (:Label {id: 1}) | + And no side effects + + Scenario: Variable length pattern with label predicate on both sides + Given an empty graph + And having executed: + """ + CREATE (a:Blue), (b:Red), (c:Green), (d:Yellow) + CREATE (a)-[:T]->(b), + (b)-[:T]->(c), + (b)-[:T]->(d) + """ + When executing query: + """ + MATCH (a:Blue)-[r*]->(b:Green) + RETURN count(r) + """ + Then the result should be: + | count(r) | + | 1 | + And no side effects + + Scenario: Undirected named path + Given an empty graph + And having executed: + """ + CREATE (a:Movie), (b) + CREATE (b)-[:T]->(a) + """ + When executing query: + """ + MATCH p = (n:Movie)--(m) + RETURN p + LIMIT 1 + """ + Then the result should be: + | p | + | <(:Movie)<-[:T]-()> | + And no side effects + + Scenario: Named path with WITH + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH p = (a) + WITH p + RETURN p + """ + Then the result should be: + | p | + | <()> | + And no side effects + + Scenario: Named path with alternating directed/undirected relationships + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B), (c:C) + CREATE (b)-[:T]->(a), + (c)-[:T]->(b) + """ + When executing query: + """ + MATCH p = (n)-->(m)--(o) + RETURN p + """ + Then the result should be: + | p | + | <(:C)-[:T]->(:B)-[:T]->(:A)> | + And no side effects + + Scenario: Named path with multiple alternating directed/undirected relationships + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B), (c:C), (d:D) + CREATE (b)-[:T]->(a), + (c)-[:T]->(b), + (d)-[:T]->(c) + """ + When executing query: + """ + MATCH path = (n)-->(m)--(o)--(p) + RETURN path + """ + Then the result should be: + | path | + | <(:D)-[:T]->(:C)-[:T]->(:B)-[:T]->(:A)> | + And no side effects + + Scenario: Named path with undirected fixed variable length pattern + Given an empty graph + And having executed: + """ + CREATE (db1:Start), (db2:End), (mid), (other) + CREATE (mid)-[:CONNECTED_TO]->(db1), + (mid)-[:CONNECTED_TO]->(db2), + (mid)-[:CONNECTED_TO]->(db2), + (mid)-[:CONNECTED_TO]->(other), + (mid)-[:CONNECTED_TO]->(other) + """ + When executing query: + """ + MATCH topRoute = (:Start)<-[:CONNECTED_TO]-()-[:CONNECTED_TO*3..3]-(:End) + RETURN topRoute + """ + Then the result should be: + | topRoute | + | <(:Start)<-[:CONNECTED_TO]-()-[:CONNECTED_TO]->()<-[:CONNECTED_TO]-()-[:CONNECTED_TO]->(:End)> | + | <(:Start)<-[:CONNECTED_TO]-()-[:CONNECTED_TO]->()<-[:CONNECTED_TO]-()-[:CONNECTED_TO]->(:End)> | + | <(:Start)<-[:CONNECTED_TO]-()-[:CONNECTED_TO]->()<-[:CONNECTED_TO]-()-[:CONNECTED_TO]->(:End)> | + | <(:Start)<-[:CONNECTED_TO]-()-[:CONNECTED_TO]->()<-[:CONNECTED_TO]-()-[:CONNECTED_TO]->(:End)> | + And no side effects + + Scenario: Returning a node property value + Given an empty graph + And having executed: + """ + CREATE ({prop: 1}) + """ + When executing query: + """ + MATCH (a) + RETURN a.prop + """ + Then the result should be: + | a.prop | + | 1 | + And no side effects + + Scenario: Returning a relationship property value + Given an empty graph + And having executed: + """ + CREATE ()-[:T {prop: 1}]->() + """ + When executing query: + """ + MATCH ()-[r]->() + RETURN r.prop + """ + Then the result should be: + | r.prop | + | 1 | + And no side effects + + Scenario: Projecting nodes and relationships + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:T]->(b) + """ + When executing query: + """ + MATCH (a)-[r]->() + RETURN a AS foo, r AS bar + """ + Then the result should be: + | foo | bar | + | (:A) | [:T] | + And no side effects + + Scenario: Missing node property should become null + Given an empty graph + And having executed: + """ + CREATE ({foo: 1}) + """ + When executing query: + """ + MATCH (a) + RETURN a.bar + """ + Then the result should be: + | a.bar | + | null | + And no side effects + + Scenario: Missing relationship property should become null + Given an empty graph + And having executed: + """ + CREATE ()-[:T {foo: 1}]->() + """ + When executing query: + """ + MATCH ()-[r]->() + RETURN r.bar + """ + Then the result should be: + | r.bar | + | null | + And no side effects + + Scenario: Returning multiple node property values + Given an empty graph + And having executed: + """ + CREATE ({name: 'Philip J. Fry', age: 2046, seasons: [1, 2, 3, 4, 5, 6, 7]}) + """ + When executing query: + """ + MATCH (a) + RETURN a.name, a.age, a.seasons + """ + Then the result should be: + | a.name | a.age | a.seasons | + | 'Philip J. Fry' | 2046 | [1, 2, 3, 4, 5, 6, 7] | + And no side effects + + Scenario: Adding a property and a literal in projection + Given an empty graph + And having executed: + """ + CREATE ({prop: 1}) + """ + When executing query: + """ + MATCH (a) + RETURN a.prop + 1 AS foo + """ + Then the result should be: + | foo | + | 2 | + And no side effects + + Scenario: Adding list properties in projection + Given an empty graph + And having executed: + """ + CREATE ({prop1: [1, 2, 3], prop2: [4, 5]}) + """ + When executing query: + """ + MATCH (a) + RETURN a.prop2 + a.prop1 AS foo + """ + Then the result should be: + | foo | + | [4, 5, 1, 2, 3] | + And no side effects + + Scenario: Variable length relationship variables are lists of relationships + Given an empty graph + And having executed: + """ + CREATE (a), (b), (c) + CREATE (a)-[:T]->(b) + """ + When executing query: + """ + MATCH ()-[r*0..1]-() + RETURN last(r) AS l + """ + Then the result should be: + | l | + | [:T] | + | [:T] | + | null | + | null | + | null | + And no side effects + + Scenario: Variable length patterns and nulls + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + """ + When executing query: + """ + MATCH (a:A) + OPTIONAL MATCH (a)-[:FOO]->(b:B) + OPTIONAL MATCH (b)<-[:BAR*]-(c:B) + RETURN a, b, c + """ + Then the result should be: + | a | b | c | + | (:A) | null | null | + And no side effects + + Scenario: Projecting a list of nodes and relationships + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:T]->(b) + """ + When executing query: + """ + MATCH (n)-[r]->(m) + RETURN [n, r, m] AS r + """ + Then the result should be: + | r | + | [(:A), [:T], (:B)] | + And no side effects + + Scenario: Projecting a map of nodes and relationships + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:T]->(b) + """ + When executing query: + """ + MATCH (n)-[r]->(m) + RETURN {node1: n, rel: r, node2: m} AS m + """ + Then the result should be: + | m | + | {node1: (:A), rel: [:T], node2: (:B)} | + And no side effects + + Scenario: Respecting direction when matching existing path + Given an empty graph + And having executed: + """ + CREATE (a {prop: 'a'}), (b {prop: 'b'}) + CREATE (a)-[:T]->(b) + """ + When executing query: + """ + MATCH p = ({prop: 'a'})-->({prop: 'b'}) + RETURN p + """ + Then the result should be: + | p | + | <({prop: 'a'})-[:T]->({prop: 'b'})> | + And no side effects + + Scenario: Respecting direction when matching non-existent path + Given an empty graph + And having executed: + """ + CREATE (a {prop: 'a'}), (b {prop: 'b'}) + CREATE (a)-[:T]->(b) + """ + When executing query: + """ + MATCH p = ({prop: 'a'})<--({prop: 'b'}) + RETURN p + """ + Then the result should be: + | p | + And no side effects + + Scenario: Respecting direction when matching non-existent path with multiple directions + Given an empty graph + And having executed: + """ + CREATE (a), (b) + CREATE (a)-[:T]->(b), + (b)-[:T]->(a) + """ + When executing query: + """ + MATCH p = (n)-->(k)<--(n) + RETURN p + """ + Then the result should be: + | p | + And no side effects + + Scenario: Matching path with both directions should respect other directions + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:T1]->(b), + (b)-[:T2]->(a) + """ + When executing query: + """ + MATCH p = (n)<-->(k)<--(n) + RETURN p + """ + Then the result should be: + | p | + | <(:A)<-[:T2]-(:B)<-[:T1]-(:A)> | + | <(:B)<-[:T1]-(:A)<-[:T2]-(:B)> | + And no side effects + + Scenario: Matching path with multiple bidirectional relationships + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:T1]->(b), + (b)-[:T2]->(a) + """ + When executing query: + """ + MATCH p=(n)<-->(k)<-->(n) + RETURN p + """ + Then the result should be: + | p | + | <(:A)<-[:T2]-(:B)<-[:T1]-(:A)> | + | <(:A)-[:T1]->(:B)-[:T2]->(:A)> | + | <(:B)<-[:T1]-(:A)<-[:T2]-(:B)> | + | <(:B)-[:T2]->(:A)-[:T1]->(:B)> | + And no side effects + + Scenario: Matching nodes with many labels + Given an empty graph + And having executed: + """ + CREATE (a:A:B:C:D:E:F:G:H:I:J:K:L:M), + (b:U:V:W:X:Y:Z) + CREATE (a)-[:T]->(b) + """ + When executing query: + """ + MATCH (n:A:B:C:D:E:F:G:H:I:J:K:L:M)-[:T]->(m:Z:Y:X:W:V:U) + RETURN n, m + """ + Then the result should be: + | n | m | + | (:A:B:C:D:E:F:G:H:I:J:K:L:M) | (:Z:Y:X:W:V:U) | + And no side effects + + Scenario: Matching longer variable length paths + Given an empty graph + And having executed: + """ + CREATE (a {prop: 'start'}), (b {prop: 'end'}) + WITH * + UNWIND range(1, 20) AS i + CREATE (n {prop: i}) + WITH [a] + collect(n) + [b] AS nodeList + UNWIND range(0, size(nodeList) - 2, 1) AS i + WITH nodeList[i] AS n1, nodeList[i+1] AS n2 + CREATE (n1)-[:T]->(n2) + """ + When executing query: + """ + MATCH (n {prop: 'start'})-[:T*]->(m {prop: 'end'}) + RETURN m + """ + Then the result should be: + | m | + | ({prop: 'end'}) | + And no side effects + + Scenario: Counting rows after MATCH, MERGE, OPTIONAL MATCH + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:T1]->(b), + (b)-[:T2]->(a) + """ + When executing query: + """ + MATCH (a) + MERGE (b) + WITH * + OPTIONAL MATCH (a)--(b) + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 6 | + And no side effects + + Scenario: Matching a self-loop + Given an empty graph + And having executed: + """ + CREATE (a) + CREATE (a)-[:T]->(a) + """ + When executing query: + """ + MATCH ()-[r]-() + RETURN type(r) AS r + """ + Then the result should be: + | r | + | 'T' | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/MatchingSelfRelationships.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/MatchingSelfRelationships.feature new file mode 100644 index 000000000..2d5d90caf --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/MatchingSelfRelationships.feature @@ -0,0 +1,338 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: MatchingSelfRelationships + + Scenario: Undirected match in self-relationship graph + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:LOOP]->(a) + """ + When executing query: + """ + MATCH (a)-[r]-(b) + RETURN a, r, b + """ + Then the result should be: + | a | r | b | + | (:A) | [:LOOP] | (:A) | + And no side effects + + Scenario: Undirected match in self-relationship graph, count + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:LOOP]->(a) + """ + When executing query: + """ + MATCH ()--() + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 1 | + And no side effects + + Scenario: Undirected match of self-relationship in self-relationship graph + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:LOOP]->(a) + """ + When executing query: + """ + MATCH (n)-[r]-(n) + RETURN n, r + """ + Then the result should be: + | n | r | + | (:A) | [:LOOP] | + And no side effects + + Scenario: Undirected match of self-relationship in self-relationship graph, count + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:LOOP]->(a) + """ + When executing query: + """ + MATCH (n)--(n) + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 1 | + And no side effects + + Scenario: Undirected match on simple relationship graph + Given an empty graph + And having executed: + """ + CREATE (:A)-[:LOOP]->(:B) + """ + When executing query: + """ + MATCH (a)-[r]-(b) + RETURN a, r, b + """ + Then the result should be: + | a | r | b | + | (:A) | [:LOOP] | (:B) | + | (:B) | [:LOOP] | (:A) | + And no side effects + + Scenario: Undirected match on simple relationship graph, count + Given an empty graph + And having executed: + """ + CREATE (:A)-[:LOOP]->(:B) + """ + When executing query: + """ + MATCH ()--() + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 2 | + And no side effects + + Scenario: Directed match on self-relationship graph + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:LOOP]->(a) + """ + When executing query: + """ + MATCH (a)-[r]->(b) + RETURN a, r, b + """ + Then the result should be: + | a | r | b | + | (:A) | [:LOOP] | (:A) | + And no side effects + + Scenario: Directed match on self-relationship graph, count + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:LOOP]->(a) + """ + When executing query: + """ + MATCH ()-->() + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 1 | + And no side effects + + Scenario: Directed match of self-relationship on self-relationship graph + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:LOOP]->(a) + """ + When executing query: + """ + MATCH (n)-[r]->(n) + RETURN n, r + """ + Then the result should be: + | n | r | + | (:A) | [:LOOP] | + And no side effects + + Scenario: Directed match of self-relationship on self-relationship graph, count + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:LOOP]->(a) + """ + When executing query: + """ + MATCH (n)-->(n) + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 1 | + And no side effects + + Scenario: Counting undirected self-relationships in self-relationship graph + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:LOOP]->(a) + """ + When executing query: + """ + MATCH (n)-[r]-(n) + RETURN count(r) + """ + Then the result should be: + | count(r) | + | 1 | + And no side effects + + Scenario: Counting distinct undirected self-relationships in self-relationship graph + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:LOOP]->(a) + """ + When executing query: + """ + MATCH (n)-[r]-(n) + RETURN count(DISTINCT r) + """ + Then the result should be: + | count(DISTINCT r) | + | 1 | + And no side effects + + Scenario: Directed match of a simple relationship + Given an empty graph + And having executed: + """ + CREATE (:A)-[:LOOP]->(:B) + """ + When executing query: + """ + MATCH (a)-[r]->(b) + RETURN a, r, b + """ + Then the result should be: + | a | r | b | + | (:A) | [:LOOP] | (:B) | + And no side effects + + Scenario: Directed match of a simple relationship, count + Given an empty graph + And having executed: + """ + CREATE (:A)-[:LOOP]->(:B) + """ + When executing query: + """ + MATCH ()-->() + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 1 | + And no side effects + + Scenario: Counting directed self-relationships + Given an empty graph + And having executed: + """ + CREATE (a:A)-[:LOOP]->(a), + ()-[:T]->() + """ + When executing query: + """ + MATCH (n)-[r]->(n) + RETURN count(r) + """ + Then the result should be: + | count(r) | + | 1 | + And no side effects + + Scenario: Mixing directed and undirected pattern parts with self-relationship, simple + Given an empty graph + And having executed: + """ + CREATE (:A)-[:T1]->(l:Looper), + (l)-[:LOOP]->(l), + (l)-[:T2]->(:B) + """ + When executing query: + """ + MATCH (x:A)-[r1]->(y)-[r2]-(z) + RETURN x, r1, y, r2, z + """ + Then the result should be: + | x | r1 | y | r2 | z | + | (:A) | [:T1] | (:Looper) | [:LOOP] | (:Looper) | + | (:A) | [:T1] | (:Looper) | [:T2] | (:B) | + And no side effects + + Scenario: Mixing directed and undirected pattern parts with self-relationship, count + Given an empty graph + And having executed: + """ + CREATE (:A)-[:T1]->(l:Looper), + (l)-[:LOOP]->(l), + (l)-[:T2]->(:B) + """ + When executing query: + """ + MATCH (:A)-->()--() + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 2 | + And no side effects + + Scenario: Mixing directed and undirected pattern parts with self-relationship, undirected + Given an empty graph + And having executed: + """ + CREATE (:A)-[:T1]->(l:Looper), + (l)-[:LOOP]->(l), + (l)-[:T2]->(:B) + """ + When executing query: + """ + MATCH (x)-[r1]-(y)-[r2]-(z) + RETURN x, r1, y, r2, z + """ + Then the result should be: + | x | r1 | y | r2 | z | + | (:A) | [:T1] | (:Looper) | [:LOOP] | (:Looper) | + | (:A) | [:T1] | (:Looper) | [:T2] | (:B) | + | (:Looper) | [:LOOP] | (:Looper) | [:T1] | (:A) | + | (:Looper) | [:LOOP] | (:Looper) | [:T2] | (:B) | + | (:B) | [:T2] | (:Looper) | [:LOOP] | (:Looper) | + | (:B) | [:T2] | (:Looper) | [:T1] | (:A) | + And no side effects + + Scenario: Mixing directed and undirected pattern parts with self-relationship, undirected count + Given an empty graph + And having executed: + """ + CREATE (:A)-[:T1]->(l:Looper), + (l)-[:LOOP]->(l), + (l)-[:T2]->(:B) + """ + When executing query: + """ + MATCH ()-[]-()-[]-() + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 6 | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/MergeIntoAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/MergeIntoAcceptance.feature new file mode 100644 index 000000000..f25dc496a --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/MergeIntoAcceptance.feature @@ -0,0 +1,154 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: MergeIntoAcceptance + + Background: + Given an empty graph + And having executed: + """ + CREATE (:A {name: 'A'}), (:B {name: 'B'}) + """ + + Scenario: Updating one property with ON CREATE + When executing query: + """ + MATCH (a {name: 'A'}), (b {name: 'B'}) + MERGE (a)-[r:TYPE]->(b) + ON CREATE SET r.name = 'foo' + """ + Then the result should be empty + And the side effects should be: + | +relationships | 1 | + | +properties | 1 | + When executing control query: + """ + MATCH ()-[r:TYPE]->() + RETURN [key IN keys(r) | key + '->' + r[key]] AS keyValue + """ + Then the result should be: + | keyValue | + | ['name->foo'] | + + Scenario: Null-setting one property with ON CREATE + When executing query: + """ + MATCH (a {name: 'A'}), (b {name: 'B'}) + MERGE (a)-[r:TYPE]->(b) + ON CREATE SET r.name = null + """ + Then the result should be empty + And the side effects should be: + | +relationships | 1 | + When executing control query: + """ + MATCH ()-[r:TYPE]->() + RETURN [key IN keys(r) | key + '->' + r[key]] AS keyValue + """ + Then the result should be: + | keyValue | + | [] | + + Scenario: Copying properties from node with ON CREATE + When executing query: + """ + MATCH (a {name: 'A'}), (b {name: 'B'}) + MERGE (a)-[r:TYPE]->(b) + ON CREATE SET r = a + """ + Then the result should be empty + And the side effects should be: + | +relationships | 1 | + | +properties | 1 | + When executing control query: + """ + MATCH ()-[r:TYPE]->() + RETURN [key IN keys(r) | key + '->' + r[key]] AS keyValue + """ + Then the result should be: + | keyValue | + | ['name->A'] | + + Scenario: Copying properties from node with ON MATCH + And having executed: + """ + MATCH (a:A), (b:B) + CREATE (a)-[:TYPE {foo: 'bar'}]->(b) + """ + When executing query: + """ + MATCH (a {name: 'A'}), (b {name: 'B'}) + MERGE (a)-[r:TYPE]->(b) + ON MATCH SET r = a + """ + Then the result should be empty + And the side effects should be: + | +properties | 1 | + | -properties | 1 | + When executing control query: + """ + MATCH ()-[r:TYPE]->() + RETURN [key IN keys(r) | key + '->' + r[key]] AS keyValue + """ + Then the result should be: + | keyValue | + | ['name->A'] | + + Scenario: Copying properties from literal map with ON CREATE + When executing query: + """ + MATCH (a {name: 'A'}), (b {name: 'B'}) + MERGE (a)-[r:TYPE]->(b) + ON CREATE SET r += {foo: 'bar', bar: 'baz'} + """ + Then the result should be empty + And the side effects should be: + | +relationships | 1 | + | +properties | 2 | + When executing control query: + """ + MATCH ()-[r:TYPE]->() + RETURN [key IN keys(r) | key + '->' + r[key]] AS keyValue + """ + Then the result should be (ignoring element order for lists): + | keyValue | + | ['foo->bar', 'bar->baz'] | + + Scenario: Copying properties from literal map with ON MATCH + And having executed: + """ + MATCH (a:A), (b:B) + CREATE (a)-[:TYPE {foo: 'bar'}]->(b) + """ + When executing query: + """ + MATCH (a {name: 'A'}), (b {name: 'B'}) + MERGE (a)-[r:TYPE]->(b) + ON MATCH SET r += {foo: 'baz', bar: 'baz'} + """ + Then the result should be empty + And the side effects should be: + | +properties | 2 | + | -properties | 1 | + When executing control query: + """ + MATCH ()-[r:TYPE]->() + RETURN [key IN keys(r) | key + '->' + r[key]] AS keyValue + """ + Then the result should be (ignoring element order for lists): + | keyValue | + | ['foo->baz', 'bar->baz'] | diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/MergeNodeAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/MergeNodeAcceptance.feature new file mode 100644 index 000000000..60b38bcbc --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/MergeNodeAcceptance.feature @@ -0,0 +1,483 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: MergeNodeAcceptance + + Scenario: Merge node when no nodes exist + Given an empty graph + When executing query: + """ + MERGE (a) + RETURN count(*) AS n + """ + Then the result should be: + | n | + | 1 | + And the side effects should be: + | +nodes | 1 | + + Scenario: Merge node with label + Given an empty graph + When executing query: + """ + MERGE (a:Label) + RETURN labels(a) + """ + Then the result should be: + | labels(a) | + | ['Label'] | + And the side effects should be: + | +nodes | 1 | + | +labels | 1 | + + Scenario: Merge node with label add label on create + Given an empty graph + When executing query: + """ + MERGE (a:Label) + ON CREATE SET a:Foo + RETURN labels(a) + """ + Then the result should be: + | labels(a) | + | ['Label', 'Foo'] | + And the side effects should be: + | +nodes | 1 | + | +labels | 2 | + + Scenario: Merge node with label add property on create + Given an empty graph + When executing query: + """ + MERGE (a:Label) + ON CREATE SET a.prop = 42 + RETURN a.prop + """ + Then the result should be: + | a.prop | + | 42 | + And the side effects should be: + | +nodes | 1 | + | +labels | 1 | + | +properties | 1 | + + Scenario: Merge node with label when it exists + Given an empty graph + And having executed: + """ + CREATE (:Label {id: 1}) + """ + When executing query: + """ + MERGE (a:Label) + RETURN a.id + """ + Then the result should be: + | a.id | + | 1 | + And no side effects + + Scenario: Merge node should create when it doesn't match, properties + Given an empty graph + And having executed: + """ + CREATE ({prop: 42}) + """ + When executing query: + """ + MERGE (a {prop: 43}) + RETURN a.prop + """ + Then the result should be: + | a.prop | + | 43 | + And the side effects should be: + | +nodes | 1 | + | +properties | 1 | + + Scenario: Merge node should create when it doesn't match, properties and label + Given an empty graph + And having executed: + """ + CREATE (:Label {prop: 42}) + """ + When executing query: + """ + MERGE (a:Label {prop: 43}) + RETURN a.prop + """ + Then the result should be: + | a.prop | + | 43 | + And the side effects should be: + | +nodes | 1 | + | +properties | 1 | + + Scenario: Merge node with prop and label + Given an empty graph + And having executed: + """ + CREATE (:Label {prop: 42}) + """ + When executing query: + """ + MERGE (a:Label {prop: 42}) + RETURN a.prop + """ + Then the result should be: + | a.prop | + | 42 | + And no side effects + + Scenario: Merge node with label add label on match when it exists + Given an empty graph + And having executed: + """ + CREATE (:Label) + """ + When executing query: + """ + MERGE (a:Label) + ON MATCH SET a:Foo + RETURN labels(a) + """ + Then the result should be: + | labels(a) | + | ['Label', 'Foo'] | + And the side effects should be: + | +labels | 1 | + + Scenario: Merge node with label add property on update when it exists + Given an empty graph + And having executed: + """ + CREATE (:Label) + """ + When executing query: + """ + MERGE (a:Label) + ON CREATE SET a.prop = 42 + RETURN a.prop + """ + Then the result should be: + | a.prop | + | null | + And no side effects + + Scenario: Merge node and set property on match + Given an empty graph + And having executed: + """ + CREATE (:Label) + """ + When executing query: + """ + MERGE (a:Label) + ON MATCH SET a.prop = 42 + RETURN a.prop + """ + Then the result should be: + | a.prop | + | 42 | + And the side effects should be: + | +properties | 1 | + + Scenario: Should work when finding multiple elements + Given an empty graph + When executing query: + """ + CREATE (:X) + CREATE (:X) + MERGE (:X) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + | +labels | 1 | + + Scenario: Should handle argument properly + Given an empty graph + And having executed: + """ + CREATE ({x: 42}), + ({x: 'not42'}) + """ + When executing query: + """ + WITH 42 AS x + MERGE (c:N {x: x}) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 1 | + | +labels | 1 | + | +properties | 1 | + + Scenario: Should handle arguments properly with only write clauses + Given an empty graph + When executing query: + """ + CREATE (a {p: 1}) + MERGE ({v: a.p}) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + | +properties | 2 | + + Scenario: Should be able to merge using property from match + Given an empty graph + And having executed: + """ + CREATE (:Person {name: 'A', bornIn: 'New York'}) + CREATE (:Person {name: 'B', bornIn: 'Ohio'}) + CREATE (:Person {name: 'C', bornIn: 'New Jersey'}) + CREATE (:Person {name: 'D', bornIn: 'New York'}) + CREATE (:Person {name: 'E', bornIn: 'Ohio'}) + CREATE (:Person {name: 'F', bornIn: 'New Jersey'}) + """ + When executing query: + """ + MATCH (person:Person) + MERGE (city:City {name: person.bornIn}) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 3 | + | +labels | 1 | + | +properties | 3 | + + Scenario: Should be able to use properties from match in ON CREATE + Given an empty graph + And having executed: + """ + CREATE (:Person {bornIn: 'New York'}), + (:Person {bornIn: 'Ohio'}) + """ + When executing query: + """ + MATCH (person:Person) + MERGE (city:City) + ON CREATE SET city.name = person.bornIn + RETURN person.bornIn + """ + Then the result should be: + | person.bornIn | + | 'New York' | + | 'Ohio' | + And the side effects should be: + | +nodes | 1 | + | +labels | 1 | + | +properties | 1 | + + Scenario: Should be able to use properties from match in ON MATCH + Given an empty graph + And having executed: + """ + CREATE (:Person {bornIn: 'New York'}), + (:Person {bornIn: 'Ohio'}) + """ + When executing query: + """ + MATCH (person:Person) + MERGE (city:City) + ON MATCH SET city.name = person.bornIn + RETURN person.bornIn + """ + Then the result should be: + | person.bornIn | + | 'New York' | + | 'Ohio' | + And the side effects should be: + | +nodes | 1 | + | +labels | 1 | + | +properties | 1 | + + Scenario: Should be able to use properties from match in ON MATCH and ON CREATE + Given an empty graph + And having executed: + """ + CREATE (:Person {bornIn: 'New York'}), + (:Person {bornIn: 'Ohio'}) + """ + When executing query: + """ + MATCH (person:Person) + MERGE (city:City) + ON MATCH SET city.name = person.bornIn + ON CREATE SET city.name = person.bornIn + RETURN person.bornIn + """ + Then the result should be: + | person.bornIn | + | 'New York' | + | 'Ohio' | + And the side effects should be: + | +nodes | 1 | + | +labels | 1 | + | +properties | 1 | + + Scenario: Should be able to set labels on match + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MERGE (a) + ON MATCH SET a:L + """ + Then the result should be empty + And the side effects should be: + | +labels | 1 | + + Scenario: Should be able to set labels on match and on create + Given an empty graph + And having executed: + """ + CREATE (), () + """ + When executing query: + """ + MATCH () + MERGE (a:L) + ON MATCH SET a:M1 + ON CREATE SET a:M2 + """ + Then the result should be empty + And the side effects should be: + | +nodes | 1 | + | +labels | 3 | + + Scenario: Should support updates while merging + Given an empty graph + And having executed: + """ + UNWIND [0, 1, 2] AS x + UNWIND [0, 1, 2] AS y + CREATE ({x: x, y: y}) + """ + When executing query: + """ + MATCH (foo) + WITH foo.x AS x, foo.y AS y + MERGE (:N {x: x, y: y + 1}) + MERGE (:N {x: x, y: y}) + MERGE (:N {x: x + 1, y: y}) + RETURN x, y + """ + Then the result should be: + | x | y | + | 0 | 0 | + | 0 | 1 | + | 0 | 2 | + | 1 | 0 | + | 1 | 1 | + | 1 | 2 | + | 2 | 0 | + | 2 | 1 | + | 2 | 2 | + And the side effects should be: + | +nodes | 15 | + | +labels | 1 | + | +properties | 30 | + + Scenario: Merge must properly handle multiple labels + Given an empty graph + And having executed: + """ + CREATE (:L:A {prop: 42}) + """ + When executing query: + """ + MERGE (test:L:B {prop: 42}) + RETURN labels(test) AS labels + """ + Then the result should be: + | labels | + | ['L', 'B'] | + And the side effects should be: + | +nodes | 1 | + | +labels | 1 | + | +properties | 1 | + + Scenario: Merge followed by multiple creates + Given an empty graph + When executing query: + """ + MERGE (t:T {id: 42}) + CREATE (f:R) + CREATE (t)-[:REL]->(f) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + | +labels | 2 | + | +properties | 1 | + + Scenario: Unwind combined with merge + Given an empty graph + When executing query: + """ + UNWIND [1, 2, 3, 4] AS int + MERGE (n {id: int}) + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 4 | + And the side effects should be: + | +nodes | 4 | + | +properties | 4 | + + Scenario: Merges should not be able to match on deleted nodes + Given an empty graph + And having executed: + """ + CREATE (:A {value: 1}), + (:A {value: 2}) + """ + When executing query: + """ + MATCH (a:A) + DELETE a + MERGE (a2:A) + RETURN a2.value + """ + Then the result should be: + | a2.value | + | null | + | null | + And the side effects should be: + | +nodes | 1 | + | -nodes | 2 | + | -properties | 2 | + + Scenario: ON CREATE on created nodes + Given an empty graph + When executing query: + """ + MERGE (b) + ON CREATE SET b.created = 1 + """ + Then the result should be empty + And the side effects should be: + | +nodes | 1 | + | +properties | 1 | + diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/MergeRelationshipAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/MergeRelationshipAcceptance.feature new file mode 100644 index 000000000..87bba6143 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/MergeRelationshipAcceptance.feature @@ -0,0 +1,599 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: MergeRelationshipAcceptance + + Scenario: Creating a relationship + Given an empty graph + And having executed: + """ + CREATE (:A), (:B) + """ + When executing query: + """ + MATCH (a:A), (b:B) + MERGE (a)-[r:TYPE]->(b) + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 1 | + And the side effects should be: + | +relationships | 1 | + + Scenario: Matching a relationship + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:TYPE]->(b) + """ + When executing query: + """ + MATCH (a:A), (b:B) + MERGE (a)-[r:TYPE]->(b) + RETURN count(r) + """ + Then the result should be: + | count(r) | + | 1 | + And no side effects + + Scenario: Matching two relationships + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:TYPE]->(b) + CREATE (a)-[:TYPE]->(b) + """ + When executing query: + """ + MATCH (a:A), (b:B) + MERGE (a)-[r:TYPE]->(b) + RETURN count(r) + """ + Then the result should be: + | count(r) | + | 2 | + And no side effects + + Scenario: Filtering relationships + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:TYPE {name: 'r1'}]->(b) + CREATE (a)-[:TYPE {name: 'r2'}]->(b) + """ + When executing query: + """ + MATCH (a:A), (b:B) + MERGE (a)-[r:TYPE {name: 'r2'}]->(b) + RETURN count(r) + """ + Then the result should be: + | count(r) | + | 1 | + And no side effects + + Scenario: Creating relationship when all matches filtered out + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:TYPE {name: 'r1'}]->(b) + """ + When executing query: + """ + MATCH (a:A), (b:B) + MERGE (a)-[r:TYPE {name: 'r2'}]->(b) + RETURN count(r) + """ + Then the result should be: + | count(r) | + | 1 | + And the side effects should be: + | +relationships | 1 | + | +properties | 1 | + + Scenario: Matching incoming relationship + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (b)-[:TYPE]->(a) + CREATE (a)-[:TYPE]->(b) + """ + When executing query: + """ + MATCH (a:A), (b:B) + MERGE (a)<-[r:TYPE]-(b) + RETURN count(r) + """ + Then the result should be: + | count(r) | + | 1 | + And no side effects + + Scenario: Creating relationship with property + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + """ + When executing query: + """ + MATCH (a:A), (b:B) + MERGE (a)-[r:TYPE {name: 'Lola'}]->(b) + RETURN count(r) + """ + Then the result should be: + | count(r) | + | 1 | + And the side effects should be: + | +relationships | 1 | + | +properties | 1 | + + Scenario: Using ON CREATE on a node + Given an empty graph + And having executed: + """ + CREATE (:A), (:B) + """ + When executing query: + """ + MATCH (a:A), (b:B) + MERGE (a)-[:KNOWS]->(b) + ON CREATE SET b.created = 1 + """ + Then the result should be empty + And the side effects should be: + | +relationships | 1 | + | +properties | 1 | + + Scenario: Using ON CREATE on a relationship + Given an empty graph + And having executed: + """ + CREATE (:A), (:B) + """ + When executing query: + """ + MATCH (a:A), (b:B) + MERGE (a)-[r:TYPE]->(b) + ON CREATE SET r.name = 'Lola' + RETURN count(r) + """ + Then the result should be: + | count(r) | + | 1 | + And the side effects should be: + | +relationships | 1 | + | +properties | 1 | + + Scenario: Using ON MATCH on created node + Given an empty graph + And having executed: + """ + CREATE (:A), (:B) + """ + When executing query: + """ + MATCH (a:A), (b:B) + MERGE (a)-[:KNOWS]->(b) + ON MATCH SET b.created = 1 + """ + Then the result should be empty + And the side effects should be: + | +relationships | 1 | + + Scenario: Using ON MATCH on created relationship + Given an empty graph + And having executed: + """ + CREATE (:A), (:B) + """ + When executing query: + """ + MATCH (a:A), (b:B) + MERGE (a)-[r:KNOWS]->(b) + ON MATCH SET r.created = 1 + """ + Then the result should be empty + And the side effects should be: + | +relationships | 1 | + + Scenario: Using ON MATCH on a relationship + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:TYPE]->(b) + """ + When executing query: + """ + MATCH (a:A), (b:B) + MERGE (a)-[r:TYPE]->(b) + ON MATCH SET r.name = 'Lola' + RETURN count(r) + """ + Then the result should be: + | count(r) | + | 1 | + And the side effects should be: + | +properties | 1 | + + Scenario: Using ON CREATE and ON MATCH + Given an empty graph + And having executed: + """ + CREATE (a:A {id: 1}), (b:B {id: 2}) + CREATE (a)-[:TYPE]->(b) + CREATE (:A {id: 3}), (:B {id: 4}) + """ + When executing query: + """ + MATCH (a:A), (b:B) + MERGE (a)-[r:TYPE]->(b) + ON CREATE SET r.name = 'Lola' + ON MATCH SET r.name = 'RUN' + RETURN count(r) + """ + Then the result should be: + | count(r) | + | 4 | + And the side effects should be: + | +relationships | 3 | + | +properties | 4 | + + Scenario: Creating relationship using merged nodes + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + """ + When executing query: + """ + MERGE (a:A) + MERGE (b:B) + MERGE (a)-[:FOO]->(b) + """ + Then the result should be empty + And the side effects should be: + | +relationships | 1 | + + Scenario: Mixing MERGE with CREATE + Given an empty graph + When executing query: + """ + CREATE (a:A), (b:B) + MERGE (a)-[:KNOWS]->(b) + CREATE (b)-[:KNOWS]->(c:C) + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 1 | + And the side effects should be: + | +nodes | 3 | + | +relationships | 2 | + | +labels | 3 | + + Scenario: Introduce named paths 1 + Given an empty graph + When executing query: + """ + MERGE (a {x: 1}) + MERGE (b {x: 2}) + MERGE p = (a)-[:R]->(b) + RETURN p + """ + Then the result should be: + | p | + | <({x: 1})-[:R]->({x: 2})> | + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + | +properties | 2 | + + Scenario: Introduce named paths 2 + Given an empty graph + When executing query: + """ + MERGE p = (a {x: 1}) + RETURN p + """ + Then the result should be: + | p | + | <({x: 1})> | + And the side effects should be: + | +nodes | 1 | + | +properties | 1 | + + Scenario: Use outgoing direction when unspecified + Given an empty graph + When executing query: + """ + CREATE (a {id: 2}), (b {id: 1}) + MERGE (a)-[r:KNOWS]-(b) + RETURN startNode(r).id AS s, endNode(r).id AS e + """ + Then the result should be: + | s | e | + | 2 | 1 | + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + | +properties | 2 | + + Scenario: Match outgoing relationship when direction unspecified + Given an empty graph + And having executed: + """ + CREATE (a {id: 1}), (b {id: 2}) + CREATE (a)-[:KNOWS]->(b) + """ + When executing query: + """ + MATCH (a {id: 2}), (b {id: 1}) + MERGE (a)-[r:KNOWS]-(b) + RETURN r + """ + Then the result should be: + | r | + | [:KNOWS] | + And no side effects + + Scenario: Match both incoming and outgoing relationships when direction unspecified + Given an empty graph + And having executed: + """ + CREATE (a {id: 2}), (b {id: 1}), (c {id: 1}), (d {id: 2}) + CREATE (a)-[:KNOWS {name: 'ab'}]->(b) + CREATE (c)-[:KNOWS {name: 'cd'}]->(d) + """ + When executing query: + """ + MATCH (a {id: 2})--(b {id: 1}) + MERGE (a)-[r:KNOWS]-(b) + RETURN r + """ + Then the result should be: + | r | + | [:KNOWS {name: 'ab'}] | + | [:KNOWS {name: 'cd'}] | + And no side effects + + Scenario: Fail when imposing new predicates on a variable that is already bound + Given any graph + When executing query: + """ + CREATE (a:Foo) + MERGE (a)-[r:KNOWS]->(a:Bar) + """ + Then a SyntaxError should be raised at compile time: VariableAlreadyBound + + Scenario: Using list properties via variable + Given an empty graph + When executing query: + """ + CREATE (a:Foo), (b:Bar) + WITH a, b + UNWIND ['a,b', 'a,b'] AS str + WITH a, b, split(str, ',') AS roles + MERGE (a)-[r:FB {foobar: roles}]->(b) + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 2 | + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + | +labels | 2 | + | +properties | 1 | + + Scenario: Matching using list property + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:T {prop: [42, 43]}]->(b) + """ + When executing query: + """ + MATCH (a:A), (b:B) + MERGE (a)-[r:T {prop: [42, 43]}]->(b) + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 1 | + And no side effects + + Scenario: Using bound variables from other updating clause + Given an empty graph + When executing query: + """ + CREATE (a), (b) + MERGE (a)-[:X]->(b) + RETURN count(a) + """ + Then the result should be: + | count(a) | + | 1 | + And the side effects should be: + | +nodes | 2 | + | +relationships | 1 | + + Scenario: UNWIND with multiple merges + Given an empty graph + When executing query: + """ + UNWIND ['Keanu Reeves', 'Hugo Weaving', 'Carrie-Anne Moss', 'Laurence Fishburne'] AS actor + MERGE (m:Movie {name: 'The Matrix'}) + MERGE (p:Person {name: actor}) + MERGE (p)-[:ACTED_IN]->(m) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 5 | + | +relationships | 4 | + | +labels | 2 | + | +properties | 5 | + + Scenario: Do not match on deleted entities + Given an empty graph + And having executed: + """ + CREATE (a:A) + CREATE (b1:B {value: 0}), (b2:B {value: 1}) + CREATE (c1:C), (c2:C) + CREATE (a)-[:REL]->(b1), + (a)-[:REL]->(b2), + (b1)-[:REL]->(c1), + (b2)-[:REL]->(c2) + """ + When executing query: + """ + MATCH (a:A)-[ab]->(b:B)-[bc]->(c:C) + DELETE ab, bc, b, c + MERGE (newB:B {value: 1}) + MERGE (a)-[:REL]->(newB) + MERGE (newC:C) + MERGE (newB)-[:REL]->(newC) + """ + Then the result should be empty + And the side effects should be: + | +nodes | 2 | + | -nodes | 4 | + | +relationships | 2 | + | -relationships | 4 | + | +properties | 1 | + | -properties | 2 | + + Scenario: Do not match on deleted relationships + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:T {name: 'rel1'}]->(b), + (a)-[:T {name: 'rel2'}]->(b) + """ + When executing query: + """ + MATCH (a)-[t:T]->(b) + DELETE t + MERGE (a)-[t2:T {name: 'rel3'}]->(b) + RETURN t2.name + """ + Then the result should be: + | t2.name | + | 'rel3' | + | 'rel3' | + And the side effects should be: + | +relationships | 1 | + | -relationships | 2 | + | +properties | 1 | + | -properties | 2 | + + Scenario: Aliasing of existing nodes 1 + Given an empty graph + And having executed: + """ + CREATE ({id: 0}) + """ + When executing query: + """ + MATCH (n) + MATCH (m) + WITH n AS a, m AS b + MERGE (a)-[r:T]->(b) + RETURN a.id AS a, b.id AS b + """ + Then the result should be: + | a | b | + | 0 | 0 | + And the side effects should be: + | +relationships | 1 | + + Scenario: Aliasing of existing nodes 2 + Given an empty graph + And having executed: + """ + CREATE ({id: 0}) + """ + When executing query: + """ + MATCH (n) + WITH n AS a, n AS b + MERGE (a)-[r:T]->(b) + RETURN a.id AS a + """ + Then the result should be: + | a | + | 0 | + And the side effects should be: + | +relationships | 1 | + + Scenario: Double aliasing of existing nodes 1 + Given an empty graph + And having executed: + """ + CREATE ({id: 0}) + """ + When executing query: + """ + MATCH (n) + MATCH (m) + WITH n AS a, m AS b + MERGE (a)-[:T]->(b) + WITH a AS x, b AS y + MERGE (a) + MERGE (b) + MERGE (a)-[:T]->(b) + RETURN x.id AS x, y.id AS y + """ + Then the result should be: + | x | y | + | 0 | 0 | + And the side effects should be: + | +relationships | 1 | + + Scenario: Double aliasing of existing nodes 2 + Given an empty graph + And having executed: + """ + CREATE ({id: 0}) + """ + When executing query: + """ + MATCH (n) + WITH n AS a + MERGE (c) + MERGE (a)-[:T]->(c) + WITH a AS x + MERGE (c) + MERGE (x)-[:T]->(c) + RETURN x.id AS x + """ + Then the result should be: + | x | + | 0 | + And the side effects should be: + | +relationships | 1 | diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/MiscellaneousErrorAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/MiscellaneousErrorAcceptance.feature new file mode 100644 index 000000000..3a63b6b67 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/MiscellaneousErrorAcceptance.feature @@ -0,0 +1,210 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: MiscellaneousErrorAcceptance + + Background: + Given any graph + + Scenario: Failing on incorrect unicode literal + When executing query: + """ + RETURN '\uH' + """ + Then a SyntaxError should be raised at compile time: InvalidUnicodeLiteral + + Scenario: Failing on merging relationship with null property + When executing query: + """ + CREATE (a), (b) + MERGE (a)-[r:X {p: null}]->(b) + """ + Then a SemanticError should be raised at compile time: MergeReadOwnWrites + + Scenario: Failing on merging node with null property + When executing query: + """ + MERGE ({p: null}) + """ + Then a SemanticError should be raised at compile time: MergeReadOwnWrites + + Scenario: Failing on aggregation in WHERE + When executing query: + """ + MATCH (a) + WHERE count(a) > 10 + RETURN a + """ + Then a SyntaxError should be raised at compile time: InvalidAggregation + + Scenario: Failing on aggregation in ORDER BY after RETURN + When executing query: + """ + MATCH (n) + RETURN n.prop1 + ORDER BY max(n.prop2) + """ + Then a SyntaxError should be raised at compile time: InvalidAggregation + + Scenario: Failing on aggregation in ORDER BY after WITH + When executing query: + """ + MATCH (n) + WITH n.prop1 AS foo + ORDER BY max(n.prop2) + RETURN foo AS foo + """ + Then a SyntaxError should be raised at compile time: InvalidAggregation + + Scenario: Failing when not aliasing expressions in WITH + When executing query: + """ + MATCH (a) + WITH a, count(*) + RETURN a + """ + Then a SyntaxError should be raised at compile time: NoExpressionAlias + + Scenario: Failing when using undefined variable in pattern + When executing query: + """ + MATCH (a) + CREATE (a)-[:KNOWS]->(b {name: missing}) + RETURN b + """ + Then a SyntaxError should be raised at compile time: UndefinedVariable + + Scenario: Failing when using undefined variable in SET + When executing query: + """ + MATCH (a) + SET a.name = missing + RETURN a + """ + Then a SyntaxError should be raised at compile time: UndefinedVariable + + Scenario: Failing when using undefined variable in DELETE + When executing query: + """ + MATCH (a) + DELETE x + """ + Then a SyntaxError should be raised at compile time: UndefinedVariable + + Scenario: Failing when using a variable that is already bound in CREATE + When executing query: + """ + MATCH (a) + CREATE (a {name: 'foo'}) + RETURN a + """ + Then a SyntaxError should be raised at compile time: VariableAlreadyBound + + Scenario: Failing when using a path variable that is already bound + When executing query: + """ + MATCH p = (a) + WITH p, a + MATCH p = (a)-->(b) + RETURN a + """ + Then a SyntaxError should be raised at compile time: VariableAlreadyBound + + Scenario: Failing when using a list as a node + When executing query: + """ + MATCH (n) + WITH [n] AS users + MATCH (users)-->(messages) + RETURN messages + """ + Then a SyntaxError should be raised at compile time: VariableTypeConflict + + Scenario: Failing when using a variable length relationship as a single relationship + When executing query: + """ + MATCH (n) + MATCH (n)-[r*]->() + WHERE r.foo = 'apa' + RETURN r + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: Failing when UNION has different columns + When executing query: + """ + RETURN 1 AS a + UNION + RETURN 2 AS b + """ + Then a SyntaxError should be raised at compile time: DifferentColumnsInUnion + + Scenario: Failing when mixing UNION and UNION ALL + When executing query: + """ + RETURN 1 AS a + UNION + RETURN 2 AS a + UNION ALL + RETURN 3 AS a + """ + Then a SyntaxError should be raised at compile time: InvalidClauseComposition + + Scenario: Failing when creating without direction + When executing query: + """ + CREATE (a)-[:FOO]-(b) + """ + Then a SyntaxError should be raised at compile time: RequiresDirectedRelationship + + Scenario: Failing when creating with two directions + When executing query: + """ + CREATE (a)<-[:FOO]->(b) + """ + Then a SyntaxError should be raised at compile time: RequiresDirectedRelationship + + Scenario: Failing when deleting a label + When executing query: + """ + MATCH (n) + DELETE n:Person + """ + Then a SyntaxError should be raised at compile time: InvalidDelete + + Scenario: Failing when setting a list of maps as a property + When executing query: + """ + CREATE (a) + SET a.foo = [{x: 1}] + """ + Then a TypeError should be raised at compile time: InvalidPropertyType + + Scenario: Failing when multiple columns have the same name + When executing query: + """ + RETURN 1 AS a, 2 AS a + """ + Then a SyntaxError should be raised at compile time: ColumnNameConflict + + Scenario: Failing when using RETURN * without variables in scope + When executing query: + """ + MATCH () + RETURN * + """ + Then a SyntaxError should be raised at compile time: NoVariablesInScope diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/NullAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/NullAcceptance.feature new file mode 100644 index 000000000..61c1d61a9 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/NullAcceptance.feature @@ -0,0 +1,122 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: NullAcceptance + + Scenario: Ignore null when setting property + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (a:DoesNotExist) + SET a.prop = 42 + RETURN a + """ + Then the result should be: + | a | + | null | + And no side effects + + Scenario: Ignore null when removing property + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (a:DoesNotExist) + REMOVE a.prop + RETURN a + """ + Then the result should be: + | a | + | null | + And no side effects + + Scenario: Ignore null when setting properties using an appending map + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (a:DoesNotExist) + SET a += {prop: 42} + RETURN a + """ + Then the result should be: + | a | + | null | + And no side effects + + Scenario: Ignore null when setting properties using an overriding map + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (a:DoesNotExist) + SET a = {prop: 42} + RETURN a + """ + Then the result should be: + | a | + | null | + And no side effects + + Scenario: Ignore null when setting label + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (a:DoesNotExist) + SET a:L + RETURN a + """ + Then the result should be: + | a | + | null | + And no side effects + + Scenario: Ignore null when removing label + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (a:DoesNotExist) + REMOVE a:L + RETURN a + """ + Then the result should be: + | a | + | null | + And no side effects + + Scenario: Ignore null when deleting node + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (a:DoesNotExist) + DELETE a + RETURN a + """ + Then the result should be: + | a | + | null | + And no side effects + + Scenario: Ignore null when deleting relationship + Given an empty graph + When executing query: + """ + OPTIONAL MATCH ()-[r:DoesNotExist]-() + DELETE r + RETURN r + """ + Then the result should be: + | r | + | null | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/OptionalMatch.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/OptionalMatch.feature new file mode 100644 index 000000000..eee1c6130 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/OptionalMatch.feature @@ -0,0 +1,74 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: OptionalMatch + + Scenario: Satisfies the open world assumption, relationships between same nodes + Given an empty graph + And having executed: + """ + CREATE (a:Player), (b:Team) + CREATE (a)-[:PLAYS_FOR]->(b), + (a)-[:SUPPORTS]->(b) + """ + When executing query: + """ + MATCH (p:Player)-[:PLAYS_FOR]->(team:Team) + OPTIONAL MATCH (p)-[s:SUPPORTS]->(team) + RETURN count(*) AS matches, s IS NULL AS optMatch + """ + Then the result should be: + | matches | optMatch | + | 1 | false | + And no side effects + + Scenario: Satisfies the open world assumption, single relationship + Given an empty graph + And having executed: + """ + CREATE (a:Player), (b:Team) + CREATE (a)-[:PLAYS_FOR]->(b) + """ + When executing query: + """ + MATCH (p:Player)-[:PLAYS_FOR]->(team:Team) + OPTIONAL MATCH (p)-[s:SUPPORTS]->(team) + RETURN count(*) AS matches, s IS NULL AS optMatch + """ + Then the result should be: + | matches | optMatch | + | 1 | true | + And no side effects + + Scenario: Satisfies the open world assumption, relationships between different nodes + Given an empty graph + And having executed: + """ + CREATE (a:Player), (b:Team), (c:Team) + CREATE (a)-[:PLAYS_FOR]->(b), + (a)-[:SUPPORTS]->(c) + """ + When executing query: + """ + MATCH (p:Player)-[:PLAYS_FOR]->(team:Team) + OPTIONAL MATCH (p)-[s:SUPPORTS]->(team) + RETURN count(*) AS matches, s IS NULL AS optMatch + """ + Then the result should be: + | matches | optMatch | + | 1 | true | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/OptionalMatchAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/OptionalMatchAcceptance.feature new file mode 100644 index 000000000..391afe337 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/OptionalMatchAcceptance.feature @@ -0,0 +1,325 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: OptionalMatchAcceptance + + Background: + Given an empty graph + And having executed: + """ + CREATE (s:Single), (a:A {prop: 42}), + (b:B {prop: 46}), (c:C) + CREATE (s)-[:REL]->(a), + (s)-[:REL]->(b), + (a)-[:REL]->(c), + (b)-[:LOOP]->(b) + """ + + Scenario: Return null when no matches due to inline label predicate + When executing query: + """ + MATCH (n:Single) + OPTIONAL MATCH (n)-[r]-(m:NonExistent) + RETURN r + """ + Then the result should be: + | r | + | null | + And no side effects + + Scenario: Return null when no matches due to label predicate in WHERE + When executing query: + """ + MATCH (n:Single) + OPTIONAL MATCH (n)-[r]-(m) + WHERE m:NonExistent + RETURN r + """ + Then the result should be: + | r | + | null | + And no side effects + + Scenario: Respect predicates on the OPTIONAL MATCH + When executing query: + """ + MATCH (n:Single) + OPTIONAL MATCH (n)-[r]-(m) + WHERE m.prop = 42 + RETURN m + """ + Then the result should be: + | m | + | (:A {prop: 42}) | + And no side effects + + Scenario: Returning label predicate on null node + When executing query: + """ + MATCH (n:Single) + OPTIONAL MATCH (n)-[r:TYPE]-(m) + RETURN m:TYPE + """ + Then the result should be: + | m:TYPE | + | null | + And no side effects + + Scenario: MATCH after OPTIONAL MATCH + When executing query: + """ + MATCH (a:Single) + OPTIONAL MATCH (a)-->(b:NonExistent) + OPTIONAL MATCH (a)-->(c:NonExistent) + WITH coalesce(b, c) AS x + MATCH (x)-->(d) + RETURN d + """ + Then the result should be: + | d | + And no side effects + + Scenario: WITH after OPTIONAL MATCH + When executing query: + """ + OPTIONAL MATCH (a:A) + WITH a AS a + MATCH (b:B) + RETURN a, b + """ + Then the result should be: + | a | b | + | (:A {prop: 42}) | (:B {prop: 46}) | + And no side effects + + Scenario: Named paths in optional matches + When executing query: + """ + MATCH (a:A) + OPTIONAL MATCH p = (a)-[:X]->(b) + RETURN p + """ + Then the result should be: + | p | + | null | + And no side effects + + Scenario: OPTIONAL MATCH and bound nodes + When executing query: + """ + MATCH (a:A), (b:C) + OPTIONAL MATCH (x)-->(b) + RETURN x + """ + Then the result should be: + | x | + | (:A {prop: 42}) | + And no side effects + + Scenario: OPTIONAL MATCH with labels on the optional end node + And having executed: + """ + CREATE (:X), (x:X), (y1:Y), (y2:Y:Z) + CREATE (x)-[:REL]->(y1), + (x)-[:REL]->(y2) + """ + When executing query: + """ + MATCH (a:X) + OPTIONAL MATCH (a)-->(b:Y) + RETURN b + """ + Then the result should be: + | b | + | null | + | (:Y) | + | (:Y:Z) | + And no side effects + + Scenario: Named paths inside optional matches with node predicates + When executing query: + """ + MATCH (a:A), (b:B) + OPTIONAL MATCH p = (a)-[:X]->(b) + RETURN p + """ + Then the result should be: + | p | + | null | + And no side effects + + Scenario: Variable length optional relationships + When executing query: + """ + MATCH (a:Single) + OPTIONAL MATCH (a)-[*]->(b) + RETURN b + """ + Then the result should be: + | b | + | (:A {prop: 42}) | + | (:B {prop: 46}) | + | (:B {prop: 46}) | + | (:C) | + And no side effects + + Scenario: Variable length optional relationships with length predicates + When executing query: + """ + MATCH (a:Single) + OPTIONAL MATCH (a)-[*3..]-(b) + RETURN b + """ + Then the result should be: + | b | + | null | + And no side effects + + Scenario: Optionally matching self-loops + When executing query: + """ + MATCH (a:B) + OPTIONAL MATCH (a)-[r]-(a) + RETURN r + """ + Then the result should be: + | r | + | [:LOOP] | + And no side effects + + Scenario: Optionally matching self-loops without matches + When executing query: + """ + MATCH (a) + WHERE NOT (a:B) + OPTIONAL MATCH (a)-[r]->(a) + RETURN r + """ + Then the result should be: + | r | + | null | + | null | + | null | + And no side effects + + Scenario: Variable length optional relationships with bound nodes + When executing query: + """ + MATCH (a:Single), (x:C) + OPTIONAL MATCH (a)-[*]->(x) + RETURN x + """ + Then the result should be: + | x | + | (:C) | + And no side effects + + Scenario: Variable length optional relationships with bound nodes, no matches + When executing query: + """ + MATCH (a:A), (b:B) + OPTIONAL MATCH p = (a)-[*]->(b) + RETURN p + """ + Then the result should be: + | p | + | null | + And no side effects + + Scenario: Longer pattern with bound nodes + When executing query: + """ + MATCH (a:Single), (c:C) + OPTIONAL MATCH (a)-->(b)-->(c) + RETURN b + """ + Then the result should be: + | b | + | (:A {prop: 42}) | + And no side effects + + Scenario: Longer pattern with bound nodes without matches + When executing query: + """ + MATCH (a:A), (c:C) + OPTIONAL MATCH (a)-->(b)-->(c) + RETURN b + """ + Then the result should be: + | b | + | null | + And no side effects + + Scenario: Handling correlated optional matches; first does not match implies second does not match + When executing query: + """ + MATCH (a:A), (b:B) + OPTIONAL MATCH (a)-->(x) + OPTIONAL MATCH (x)-[r]->(b) + RETURN x, r + """ + Then the result should be: + | x | r | + | (:C) | null | + And no side effects + + Scenario: Handling optional matches between optionally matched entities + When executing query: + """ + OPTIONAL MATCH (a:NotThere) + WITH a + MATCH (b:B) + WITH a, b + OPTIONAL MATCH (b)-[r:NOR_THIS]->(a) + RETURN a, b, r + """ + Then the result should be: + | a | b | r | + | null | (:B {prop: 46}) | null | + And no side effects + + Scenario: Handling optional matches between nulls + When executing query: + """ + OPTIONAL MATCH (a:NotThere) + OPTIONAL MATCH (b:NotThere) + WITH a, b + OPTIONAL MATCH (b)-[r:NOR_THIS]->(a) + RETURN a, b, r + """ + Then the result should be: + | a | b | r | + | null | null | null | + And no side effects + + Scenario: OPTIONAL MATCH and `collect()` + And having executed: + """ + CREATE (:DoesExist {property: 42}) + CREATE (:DoesExist {property: 43}) + CREATE (:DoesExist {property: 44}) + """ + When executing query: + """ + OPTIONAL MATCH (f:DoesExist) + OPTIONAL MATCH (n:DoesNotExist) + RETURN collect(DISTINCT n.property) AS a, collect(DISTINCT f.property) AS b + """ + Then the result should be: + | a | b | + | [] | [42, 43, 44] | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/OrderByAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/OrderByAcceptance.feature new file mode 100644 index 000000000..91c224e0e --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/OrderByAcceptance.feature @@ -0,0 +1,293 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: OrderByAcceptance + + Background: + Given an empty graph + + Scenario: ORDER BY should return results in ascending order + And having executed: + """ + CREATE (n1 {prop: 1}), + (n2 {prop: 3}), + (n3 {prop: -5}) + """ + When executing query: + """ + MATCH (n) + RETURN n.prop AS prop + ORDER BY n.prop + """ + Then the result should be, in order: + | prop | + | -5 | + | 1 | + | 3 | + And no side effects + + Scenario: ORDER BY DESC should return results in descending order + And having executed: + """ + CREATE (n1 {prop: 1}), + (n2 {prop: 3}), + (n3 {prop: -5}) + """ + When executing query: + """ + MATCH (n) + RETURN n.prop AS prop + ORDER BY n.prop DESC + """ + Then the result should be, in order: + | prop | + | 3 | + | 1 | + | -5 | + And no side effects + + Scenario: ORDER BY of a column introduced in RETURN should return salient results in ascending order + When executing query: + """ + WITH [0, 1] AS prows, [[2], [3, 4]] AS qrows + UNWIND prows AS p + UNWIND qrows[p] AS q + WITH p, count(q) AS rng + RETURN p + ORDER BY rng + """ + Then the result should be, in order: + | p | + | 0 | + | 1 | + And no side effects + + Scenario: Renaming columns before ORDER BY should return results in ascending order + And having executed: + """ + CREATE (n1 {prop: 1}), + (n2 {prop: 3}), + (n3 {prop: -5}) + """ + When executing query: + """ + MATCH (n) + RETURN n.prop AS n + ORDER BY n + 2 + """ + Then the result should be, in order: + | n | + | -5 | + | 1 | + | 3 | + And no side effects + + Scenario: Handle projections with ORDER BY - GH#4937 + And having executed: + """ + CREATE (c1:Crew {name: 'Neo', rank: 1}), + (c2:Crew {name: 'Neo', rank: 2}), + (c3:Crew {name: 'Neo', rank: 3}), + (c4:Crew {name: 'Neo', rank: 4}), + (c5:Crew {name: 'Neo', rank: 5}) + """ + When executing query: + """ + MATCH (c:Crew {name: 'Neo'}) + WITH c, 0 AS relevance + RETURN c.rank AS rank + ORDER BY relevance, c.rank + """ + Then the result should be, in order: + | rank | + | 1 | + | 2 | + | 3 | + | 4 | + | 5 | + And no side effects + + Scenario: ORDER BY should order booleans in the expected order + When executing query: + """ + UNWIND [true, false] AS bools + RETURN bools + ORDER BY bools + """ + Then the result should be, in order: + | bools | + | false | + | true | + And no side effects + + Scenario: ORDER BY DESC should order booleans in the expected order + When executing query: + """ + UNWIND [true, false] AS bools + RETURN bools + ORDER BY bools DESC + """ + Then the result should be, in order: + | bools | + | true | + | false | + And no side effects + + Scenario: ORDER BY should order strings in the expected order + When executing query: + """ + UNWIND ['.*', '', ' ', 'one'] AS strings + RETURN strings + ORDER BY strings + """ + Then the result should be, in order: + | strings | + | '' | + | ' ' | + | '.*' | + | 'one' | + And no side effects + + Scenario: ORDER BY DESC should order strings in the expected order + When executing query: + """ + UNWIND ['.*', '', ' ', 'one'] AS strings + RETURN strings + ORDER BY strings DESC + """ + Then the result should be, in order: + | strings | + | 'one' | + | '.*' | + | ' ' | + | '' | + And no side effects + + Scenario: ORDER BY should order ints in the expected order + When executing query: + """ + UNWIND [1, 3, 2] AS ints + RETURN ints + ORDER BY ints + """ + Then the result should be, in order: + | ints | + | 1 | + | 2 | + | 3 | + And no side effects + + Scenario: ORDER BY DESC should order ints in the expected order + When executing query: + """ + UNWIND [1, 3, 2] AS ints + RETURN ints + ORDER BY ints DESC + """ + Then the result should be, in order: + | ints | + | 3 | + | 2 | + | 1 | + And no side effects + + Scenario: ORDER BY should order floats in the expected order + When executing query: + """ + UNWIND [1.5, 1.3, 999.99] AS floats + RETURN floats + ORDER BY floats + """ + Then the result should be, in order: + | floats | + | 1.3 | + | 1.5 | + | 999.99 | + And no side effects + + Scenario: ORDER BY DESC should order floats in the expected order + When executing query: + """ + UNWIND [1.5, 1.3, 999.99] AS floats + RETURN floats + ORDER BY floats DESC + """ + Then the result should be, in order: + | floats | + | 999.99 | + | 1.5 | + | 1.3 | + And no side effects + + Scenario: Handle ORDER BY with LIMIT 1 + And having executed: + """ + CREATE (s:Person {name: 'Steven'}), + (c:Person {name: 'Craig'}) + """ + When executing query: + """ + MATCH (p:Person) + RETURN p.name AS name + ORDER BY p.name + LIMIT 1 + """ + Then the result should be, in order: + | name | + | 'Craig' | + And no side effects + + Scenario: ORDER BY with LIMIT 0 should not generate errors + When executing query: + """ + MATCH (p:Person) + RETURN p.name AS name + ORDER BY p.name + LIMIT 0 + """ + Then the result should be, in order: + | name | + And no side effects + + Scenario: ORDER BY with negative parameter for LIMIT should not generate errors + And parameters are: + | limit | -1 | + When executing query: + """ + MATCH (p:Person) + RETURN p.name AS name + ORDER BY p.name + LIMIT $`limit` + """ + Then the result should be, in order: + | name | + And no side effects + + Scenario: ORDER BY with a negative LIMIT should fail with a syntax exception + And having executed: + """ + CREATE (s:Person {name: 'Steven'}), + (c:Person {name: 'Craig'}) + """ + When executing query: + """ + MATCH (p:Person) + RETURN p.name AS name + ORDER BY p.name + LIMIT -1 + """ + Then a SyntaxError should be raised at compile time: NegativeIntegerArgument diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/PathEquality.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/PathEquality.feature new file mode 100644 index 000000000..a15513732 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/PathEquality.feature @@ -0,0 +1,35 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: PathEquality + + Scenario: Direction of traversed relationship is not significant for path equality, simple + Given an empty graph + And having executed: + """ + CREATE (n:A)-[:LOOP]->(n) + """ + When executing query: + """ + MATCH p1 = (:A)-->() + MATCH p2 = (:A)<--() + RETURN p1 = p2 + """ + Then the result should be: + | p1 = p2 | + | true | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/PatternComprehension.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/PatternComprehension.feature new file mode 100644 index 000000000..f10eee88b --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/PatternComprehension.feature @@ -0,0 +1,302 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: PatternComprehension + + Scenario: Pattern comprehension and ORDER BY + Given an empty graph + And having executed: + """ + CREATE (a {time: 10}), (b {time: 20}) + CREATE (a)-[:T]->(b) + """ + When executing query: + """ + MATCH (liker) + RETURN [p = (liker)--() | p] AS isNew + ORDER BY liker.time + """ + Then the result should be: + | isNew | + | [<({time: 10})-[:T]->({time: 20})>] | + | [<({time: 20})<-[:T]-({time: 10})>] | + And no side effects + + Scenario: Returning a pattern comprehension + Given an empty graph + And having executed: + """ + CREATE (a:A) + CREATE (a)-[:T]->(:B), + (a)-[:T]->(:C) + """ + When executing query: + """ + MATCH (n) + RETURN [p = (n)-->() | p] AS ps + """ + Then the result should be: + | ps | + | [<(:A)-[:T]->(:C)>, <(:A)-[:T]->(:B)>] | + | [] | + | [] | + And no side effects + + Scenario: Returning a pattern comprehension with label predicate + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B), (c:C), (d:D) + CREATE (a)-[:T]->(b), + (a)-[:T]->(c), + (a)-[:T]->(d) + """ + When executing query: + """ + MATCH (n:A) + RETURN [p = (n)-->(:B) | p] AS x + """ + Then the result should be: + | x | + | [<(:A)-[:T]->(:B)>] | + And no side effects + + Scenario: Returning a pattern comprehension with bound nodes + Given an empty graph + And having executed: + """ + CREATE (a:A), (b:B) + CREATE (a)-[:T]->(b) + """ + When executing query: + """ + MATCH (a:A), (b:B) + RETURN [p = (a)-[*]->(b) | p] AS paths + """ + Then the result should be: + | paths | + | [<(:A)-[:T]->(:B)>] | + And no side effects + + Scenario: Using a pattern comprehension in a WITH + Given an empty graph + And having executed: + """ + CREATE (a:A) + CREATE (a)-[:T]->(:B), + (a)-[:T]->(:C) + """ + When executing query: + """ + MATCH (n)-->(b) + WITH [p = (n)-->() | p] AS ps, count(b) AS c + RETURN ps, c + """ + Then the result should be: + | ps | c | + | [<(:A)-[:T]->(:C)>, <(:A)-[:T]->(:B)>] | 2 | + And no side effects + + Scenario: Using a variable-length pattern comprehension in a WITH + Given an empty graph + And having executed: + """ + CREATE (:A)-[:T]->(:B) + """ + When executing query: + """ + MATCH (a:A), (b:B) + WITH [p = (a)-[*]->(b) | p] AS paths, count(a) AS c + RETURN paths, c + """ + Then the result should be: + | paths | c | + | [<(:A)-[:T]->(:B)>] | 1 | + And no side effects + + Scenario: Using pattern comprehension in RETURN + Given an empty graph + And having executed: + """ + CREATE (a:A), (:A), (:A) + CREATE (a)-[:HAS]->() + """ + When executing query: + """ + MATCH (n:A) + RETURN [p = (n)-[:HAS]->() | p] AS ps + """ + Then the result should be: + | ps | + | [<(:A)-[:HAS]->()>] | + | [] | + | [] | + And no side effects + + Scenario: Aggregating on pattern comprehension + Given an empty graph + And having executed: + """ + CREATE (a:A), (:A), (:A) + CREATE (a)-[:HAS]->() + """ + When executing query: + """ + MATCH (n:A) + RETURN count([p = (n)-[:HAS]->() | p]) AS c + """ + Then the result should be: + | c | + | 3 | + And no side effects + + Scenario: Using pattern comprehension to test existence + Given an empty graph + And having executed: + """ + CREATE (a:X {prop: 42}), (:X {prop: 43}) + CREATE (a)-[:T]->() + """ + When executing query: + """ + MATCH (n:X) + RETURN n, size([(n)--() | 1]) > 0 AS b + """ + Then the result should be: + | n | b | + | (:X {prop: 42}) | true | + | (:X {prop: 43}) | false | + And no side effects + + Scenario: Pattern comprehension inside list comprehension + Given an empty graph + And having executed: + """ + CREATE (n1:X {n: 1}), (m1:Y), (i1:Y), (i2:Y) + CREATE (n1)-[:T]->(m1), + (m1)-[:T]->(i1), + (m1)-[:T]->(i2) + CREATE (n2:X {n: 2}), (m2), (i3:L), (i4:Y) + CREATE (n2)-[:T]->(m2), + (m2)-[:T]->(i3), + (m2)-[:T]->(i4) + """ + When executing query: + """ + MATCH p = (n:X)-->(b) + RETURN n, [x IN nodes(p) | size([(x)-->(:Y) | 1])] AS list + """ + Then the result should be: + | n | list | + | (:X {n: 1}) | [1, 2] | + | (:X {n: 2}) | [0, 1] | + And no side effects + + Scenario: Get node degree via size of pattern comprehension + Given an empty graph + And having executed: + """ + CREATE (x:X), + (x)-[:T]->(), + (x)-[:T]->(), + (x)-[:T]->() + """ + When executing query: + """ + MATCH (a:X) + RETURN size([(a)-->() | 1]) AS length + """ + Then the result should be: + | length | + | 3 | + And no side effects + + Scenario: Get node degree via size of pattern comprehension that specifies a relationship type + Given an empty graph + And having executed: + """ + CREATE (x:X), + (x)-[:T]->(), + (x)-[:T]->(), + (x)-[:T]->(), + (x)-[:OTHER]->() + """ + When executing query: + """ + MATCH (a:X) + RETURN size([(a)-[:T]->() | 1]) AS length + """ + Then the result should be: + | length | + | 3 | + And no side effects + + Scenario: Get node degree via size of pattern comprehension that specifies multiple relationship types + Given an empty graph + And having executed: + """ + CREATE (x:X), + (x)-[:T]->(), + (x)-[:T]->(), + (x)-[:T]->(), + (x)-[:OTHER]->() + """ + When executing query: + """ + MATCH (a:X) + RETURN size([(a)-[:T|OTHER]->() | 1]) AS length + """ + Then the result should be: + | length | + | 4 | + And no side effects + + Scenario: Introducing new node variable in pattern comprehension + Given an empty graph + And having executed: + """ + CREATE (a), (b {prop: 'val'}) + CREATE (a)-[:T]->(b) + """ + When executing query: + """ + MATCH (n) + RETURN [(n)-[:T]->(b) | b.prop] AS list + """ + Then the result should be: + | list | + | ['val'] | + | [] | + And no side effects + + Scenario: Introducing new relationship variable in pattern comprehension + Given an empty graph + And having executed: + """ + CREATE (a), (b) + CREATE (a)-[:T {prop: 'val'}]->(b) + """ + When executing query: + """ + MATCH (n) + RETURN [(n)-[r:T]->() | r.prop] AS list + """ + Then the result should be: + | list | + | ['val'] | + | [] | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/ProcedureCallAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/ProcedureCallAcceptance.feature new file mode 100644 index 000000000..d3e600ff0 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/ProcedureCallAcceptance.feature @@ -0,0 +1,517 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: ProcedureCallAcceptance + + Background: + Given an empty graph + + Scenario: In-query call to procedure that takes arguments fails when trying to pass them implicitly + And there exists a procedure test.my.proc(in :: INTEGER?) :: (out :: INTEGER?): + | in | out | + When executing query: + """ + CALL test.my.proc YIELD out + RETURN out + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentPassingMode + + Scenario: Standalone call to procedure that takes no arguments + And there exists a procedure test.labels() :: (label :: STRING?): + | label | + | 'A' | + | 'B' | + | 'C' | + When executing query: + """ + CALL test.labels() + """ + Then the result should be, in order: + | label | + | 'A' | + | 'B' | + | 'C' | + And no side effects + + Scenario: In-query call to procedure that takes no arguments + And there exists a procedure test.labels() :: (label :: STRING?): + | label | + | 'A' | + | 'B' | + | 'C' | + When executing query: + """ + CALL test.labels() YIELD label + RETURN label + """ + Then the result should be, in order: + | label | + | 'A' | + | 'B' | + | 'C' | + And no side effects + + Scenario: Calling the same procedure twice using the same outputs in each call + And there exists a procedure test.labels() :: (label :: STRING?): + | label | + | 'A' | + | 'B' | + | 'C' | + When executing query: + """ + CALL test.labels() YIELD label + WITH count(*) AS c + CALL test.labels() YIELD label + RETURN * + """ + Then the result should be, in order: + | c | label | + | 3 | 'A' | + | 3 | 'B' | + | 3 | 'C' | + And no side effects + + Scenario: Standalone call to VOID procedure that takes no arguments + And there exists a procedure test.doNothing() :: VOID: + | + When executing query: + """ + CALL test.doNothing() + """ + Then the result should be empty + And no side effects + + Scenario: In-query call to VOID procedure that takes no arguments + And there exists a procedure test.doNothing() :: VOID: + | + When executing query: + """ + MATCH (n) + CALL test.doNothing() + RETURN n + """ + Then the result should be: + | n | + And no side effects + + Scenario: In-query call to VOID procedure does not consume rows + And there exists a procedure test.doNothing() :: VOID: + | + And having executed: + """ + CREATE (:A {name: 'a'}) + CREATE (:B {name: 'b'}) + CREATE (:C {name: 'c'}) + """ + When executing query: + """ + MATCH (n) + CALL test.doNothing() + RETURN n.name AS `name` + """ + Then the result should be: + | name | + | 'a' | + | 'b' | + | 'c' | + And no side effects + + Scenario: Standalone call to VOID procedure that takes no arguments, called with implicit arguments + And there exists a procedure test.doNothing() :: VOID: + | + When executing query: + """ + CALL test.doNothing + """ + Then the result should be empty + And no side effects + + Scenario: In-query call to procedure that takes no arguments and yields no results + And there exists a procedure test.doNothing() :: (): + | + When executing query: + """ + CALL test.doNothing() YIELD - RETURN 1 + """ + Then the result should be: + | 1 | + And no side effects + + Scenario: Standalone call to procedure that takes no arguments and yields no results + And there exists a procedure test.doNothing() :: (): + | + When executing query: + """ + CALL test.doNothing() + """ + Then the result should be empty + And no side effects + + Scenario: Standalone call to procedure that takes no arguments and yields no results, called with implicit arguments + And there exists a procedure test.doNothing() :: (): + | + When executing query: + """ + CALL test.doNothing + """ + Then the result should be empty + And no side effects + + Scenario: In-query call to procedure with explicit arguments + And there exists a procedure test.my.proc(name :: STRING?, id :: INTEGER?) :: (city :: STRING?, country_code :: INTEGER?): + | name | id | city | country_code | + | 'Andres' | 1 | 'Malmö' | 46 | + | 'Tobias' | 1 | 'Malmö' | 46 | + | 'Mats' | 1 | 'Malmö' | 46 | + | 'Stefan' | 1 | 'Berlin' | 49 | + | 'Stefan' | 2 | 'München' | 49 | + | 'Petra' | 1 | 'London' | 44 | + When executing query: + """ + CALL test.my.proc('Stefan', 1) YIELD city, country_code + RETURN city, country_code + """ + Then the result should be, in order: + | city | country_code | + | 'Berlin' | 49 | + And no side effects + + Scenario: In-query call to procedure with explicit arguments that drops all result fields + And there exists a procedure test.my.proc(name :: STRING?, id :: INTEGER?) :: (city :: STRING?, country_code :: INTEGER?): + | name | id | city | country_code | + | 'Andres' | 1 | 'Malmö' | 46 | + | 'Tobias' | 1 | 'Malmö' | 46 | + | 'Mats' | 1 | 'Malmö' | 46 | + | 'Stefan' | 1 | 'Berlin' | 49 | + | 'Stefan' | 2 | 'München' | 49 | + | 'Petra' | 1 | 'London' | 44 | + When executing query: + """ + WITH 'Stefan' AS name, 1 AS id + CALL test.my.proc(name, id) YIELD - + RETURN name, id, count(*) AS count + """ + Then the result should be, in order: + | name | id | count | + | 'Stefan' | 1 | 1 | + And no side effects + + Scenario: Standalone call to procedure with explicit arguments + And there exists a procedure test.my.proc(name :: STRING?, id :: INTEGER?) :: (city :: STRING?, country_code :: INTEGER?): + | name | id | city | country_code | + | 'Andres' | 1 | 'Malmö' | 46 | + | 'Tobias' | 1 | 'Malmö' | 46 | + | 'Mats' | 1 | 'Malmö' | 46 | + | 'Stefan' | 1 | 'Berlin' | 49 | + | 'Stefan' | 2 | 'München' | 49 | + | 'Petra' | 1 | 'London' | 44 | + When executing query: + """ + CALL test.my.proc('Stefan', 1) + """ + Then the result should be, in order: + | city | country_code | + | 'Berlin' | 49 | + And no side effects + + Scenario: Standalone call to procedure with implicit arguments + And there exists a procedure test.my.proc(name :: STRING?, id :: INTEGER?) :: (city :: STRING?, country_code :: INTEGER?): + | name | id | city | country_code | + | 'Andres' | 1 | 'Malmö' | 46 | + | 'Tobias' | 1 | 'Malmö' | 46 | + | 'Mats' | 1 | 'Malmö' | 46 | + | 'Stefan' | 1 | 'Berlin' | 49 | + | 'Stefan' | 2 | 'München' | 49 | + | 'Petra' | 1 | 'London' | 44 | + And parameters are: + | name | 'Stefan' | + | id | 1 | + When executing query: + """ + CALL test.my.proc + """ + Then the result should be, in order: + | city | country_code | + | 'Berlin' | 49 | + And no side effects + + Scenario: Standalone call to procedure with argument of type NUMBER accepts value of type INTEGER + And there exists a procedure test.my.proc(in :: NUMBER?) :: (out :: STRING?): + | in | out | + | 42 | 'wisdom' | + | 42.3 | 'about right' | + When executing query: + """ + CALL test.my.proc(42) + """ + Then the result should be, in order: + | out | + | 'wisdom' | + And no side effects + + Scenario: In-query call to procedure with argument of type NUMBER accepts value of type INTEGER + And there exists a procedure test.my.proc(in :: NUMBER?) :: (out :: STRING?): + | in | out | + | 42 | 'wisdom' | + | 42.3 | 'about right' | + When executing query: + """ + CALL test.my.proc(42) YIELD out + RETURN out + """ + Then the result should be, in order: + | out | + | 'wisdom' | + And no side effects + + Scenario: Standalone call to procedure with argument of type NUMBER accepts value of type FLOAT + And there exists a procedure test.my.proc(in :: NUMBER?) :: (out :: STRING?): + | in | out | + | 42 | 'wisdom' | + | 42.3 | 'about right' | + When executing query: + """ + CALL test.my.proc(42.3) + """ + Then the result should be, in order: + | out | + | 'about right' | + And no side effects + + Scenario: In-query call to procedure with argument of type NUMBER accepts value of type FLOAT + And there exists a procedure test.my.proc(in :: NUMBER?) :: (out :: STRING?): + | in | out | + | 42 | 'wisdom' | + | 42.3 | 'about right' | + When executing query: + """ + CALL test.my.proc(42.3) YIELD out + RETURN out + """ + Then the result should be, in order: + | out | + | 'about right' | + And no side effects + + Scenario: Standalone call to procedure with argument of type FLOAT accepts value of type INTEGER + And there exists a procedure test.my.proc(in :: FLOAT?) :: (out :: STRING?): + | in | out | + | 42.0 | 'close enough' | + When executing query: + """ + CALL test.my.proc(42) + """ + Then the result should be, in order: + | out | + | 'close enough' | + And no side effects + + Scenario: In-query call to procedure with argument of type FLOAT accepts value of type INTEGER + And there exists a procedure test.my.proc(in :: FLOAT?) :: (out :: STRING?): + | in | out | + | 42.0 | 'close enough' | + When executing query: + """ + CALL test.my.proc(42) YIELD out + RETURN out + """ + Then the result should be, in order: + | out | + | 'close enough' | + And no side effects + + Scenario: Standalone call to procedure with argument of type INTEGER accepts value of type FLOAT + And there exists a procedure test.my.proc(in :: INTEGER?) :: (out :: STRING?): + | in | out | + | 42 | 'close enough' | + When executing query: + """ + CALL test.my.proc(42.0) + """ + Then the result should be, in order: + | out | + | 'close enough' | + And no side effects + + Scenario: In-query call to procedure with argument of type INTEGER accepts value of type FLOAT + And there exists a procedure test.my.proc(in :: INTEGER?) :: (out :: STRING?): + | in | out | + | 42 | 'close enough' | + When executing query: + """ + CALL test.my.proc(42.0) YIELD out + RETURN out + """ + Then the result should be, in order: + | out | + | 'close enough' | + And no side effects + + Scenario: Standalone call to procedure with null argument + And there exists a procedure test.my.proc(in :: INTEGER?) :: (out :: STRING?): + | in | out | + | null | 'nix' | + When executing query: + """ + CALL test.my.proc(null) + """ + Then the result should be, in order: + | out | + | 'nix' | + And no side effects + + Scenario: In-query call to procedure with null argument + And there exists a procedure test.my.proc(in :: INTEGER?) :: (out :: STRING?): + | in | out | + | null | 'nix' | + When executing query: + """ + CALL test.my.proc(null) YIELD out + RETURN out + """ + Then the result should be, in order: + | out | + | 'nix' | + And no side effects + + Scenario: Standalone call to procedure should fail if input type is wrong + And there exists a procedure test.my.proc(in :: INTEGER?) :: (out :: INTEGER?): + | in | out | + When executing query: + """ + CALL test.my.proc(true) + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: In-query call to procedure should fail if input type is wrong + And there exists a procedure test.my.proc(in :: INTEGER?) :: (out :: INTEGER?): + | in | out | + When executing query: + """ + CALL test.my.proc(true) YIELD out + RETURN out + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: Standalone call to procedure should fail if explicit argument is missing + And there exists a procedure test.my.proc(name :: STRING?, in :: INTEGER?) :: (out :: INTEGER?): + | name | in | out | + When executing query: + """ + CALL test.my.proc('Dobby') + """ + Then a SyntaxError should be raised at compile time: InvalidNumberOfArguments + + Scenario: In-query call to procedure should fail if explicit argument is missing + And there exists a procedure test.my.proc(name :: STRING?, in :: INTEGER?) :: (out :: INTEGER?): + | name | in | out | + When executing query: + """ + CALL test.my.proc('Dobby') YIELD out + RETURN out + """ + Then a SyntaxError should be raised at compile time: InvalidNumberOfArguments + + Scenario: Standalone call to procedure should fail if too many explicit argument are given + And there exists a procedure test.my.proc(in :: INTEGER?) :: (out :: INTEGER?): + | in | out | + When executing query: + """ + CALL test.my.proc(1, 2, 3, 4) + """ + Then a SyntaxError should be raised at compile time: InvalidNumberOfArguments + + Scenario: In-query call to procedure should fail if too many explicit argument are given + And there exists a procedure test.my.proc(in :: INTEGER?) :: (out :: INTEGER?): + | in | out | + When executing query: + """ + CALL test.my.proc(1, 2, 3, 4) YIELD out + RETURN out + """ + Then a SyntaxError should be raised at compile time: InvalidNumberOfArguments + + Scenario: Standalone call to procedure should fail if implicit argument is missing + And there exists a procedure test.my.proc(name :: STRING?, in :: INTEGER?) :: (out :: INTEGER?): + | name | in | out | + And parameters are: + | name | 'Stefan' | + When executing query: + """ + CALL test.my.proc + """ + Then a ParameterMissing should be raised at compile time: MissingParameter + + Scenario: In-query call to procedure that has outputs fails if no outputs are yielded + And there exists a procedure test.my.proc(in :: INTEGER?) :: (out :: INTEGER?): + | in | out | + When executing query: + """ + CALL test.my.proc(1) + RETURN out + """ + Then a SyntaxError should be raised at compile time: UndefinedVariable + + Scenario: In-query call to procedure that both takes arguments and has outputs fails if the arguments are passed implicitly and no outputs are yielded + And there exists a procedure test.my.proc(in :: INTEGER?) :: (out :: INTEGER?): + | in | out | + When executing query: + """ + CALL test.my.proc + RETURN out + """ + Then a SyntaxError should be raised at compile time: UndefinedVariable + + Scenario: Standalone call to unknown procedure should fail + When executing query: + """ + CALL test.my.proc + """ + Then a ProcedureError should be raised at compile time: ProcedureNotFound + + Scenario: In-query call to unknown procedure should fail + When executing query: + """ + CALL test.my.proc() YIELD out + RETURN out + """ + Then a ProcedureError should be raised at compile time: ProcedureNotFound + + Scenario: In-query procedure call should fail if shadowing an already bound variable + And there exists a procedure test.labels() :: (label :: STRING?): + | label | + | 'A' | + | 'B' | + | 'C' | + When executing query: + """ + WITH 'Hi' AS label + CALL test.labels() YIELD label + RETURN * + """ + Then a SyntaxError should be raised at compile time: VariableAlreadyBound + + Scenario: In-query procedure call should fail if one of the argument expressions uses an aggregation function + And there exists a procedure test.labels(in :: INTEGER?) :: (label :: STRING?): + | in | label | + When executing query: + """ + MATCH (n) + CALL test.labels(count(n)) YIELD label + RETURN label + """ + Then a SyntaxError should be raised at compile time: InvalidAggregation diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/RemoveAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/RemoveAcceptance.feature new file mode 100644 index 000000000..c1b2857ce --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/RemoveAcceptance.feature @@ -0,0 +1,161 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: RemoveAcceptance + + Scenario: Should ignore nulls + Given an empty graph + And having executed: + """ + CREATE ({prop: 42}) + """ + When executing query: + """ + MATCH (n) + OPTIONAL MATCH (n)-[r]->() + REMOVE r.prop + RETURN n + """ + Then the result should be: + | n | + | ({prop: 42}) | + And no side effects + + Scenario: Remove a single label + Given an empty graph + And having executed: + """ + CREATE (:L {prop: 42}) + """ + When executing query: + """ + MATCH (n) + REMOVE n:L + RETURN n.prop + """ + Then the result should be: + | n.prop | + | 42 | + And the side effects should be: + | -labels | 1 | + + Scenario: Remove multiple labels + Given an empty graph + And having executed: + """ + CREATE (:L1:L2:L3 {prop: 42}) + """ + When executing query: + """ + MATCH (n) + REMOVE n:L1:L3 + RETURN labels(n) + """ + Then the result should be: + | labels(n) | + | ['L2'] | + And the side effects should be: + | -labels | 2 | + + Scenario: Remove a single node property + Given an empty graph + And having executed: + """ + CREATE (:L {prop: 42}) + """ + When executing query: + """ + MATCH (n) + REMOVE n.prop + RETURN exists(n.prop) AS still_there + """ + Then the result should be: + | still_there | + | false | + And the side effects should be: + | -properties | 1 | + + Scenario: Remove multiple node properties + Given an empty graph + And having executed: + """ + CREATE (:L {prop: 42, a: 'a', b: 'B'}) + """ + When executing query: + """ + MATCH (n) + REMOVE n.prop, n.a + RETURN size(keys(n)) AS props + """ + Then the result should be: + | props | + | 1 | + And the side effects should be: + | -properties | 2 | + + Scenario: Remove a single relationship property + Given an empty graph + And having executed: + """ + CREATE (a), (b), (a)-[:X {prop: 42}]->(b) + """ + When executing query: + """ + MATCH ()-[r]->() + REMOVE r.prop + RETURN exists(r.prop) AS still_there + """ + Then the result should be: + | still_there | + | false | + And the side effects should be: + | -properties | 1 | + + Scenario: Remove multiple relationship properties + Given an empty graph + And having executed: + """ + CREATE (a), (b), (a)-[:X {prop: 42, a: 'a', b: 'B'}]->(b) + """ + When executing query: + """ + MATCH ()-[r]->() + REMOVE r.prop, r.a + RETURN size(keys(r)) AS props + """ + Then the result should be: + | props | + | 1 | + And the side effects should be: + | -properties | 2 | + + Scenario: Remove a missing property should be a valid operation + Given an empty graph + And having executed: + """ + CREATE (), (), () + """ + When executing query: + """ + MATCH (n) + REMOVE n.prop + RETURN sum(size(keys(n))) AS totalNumberOfProps + """ + Then the result should be: + | totalNumberOfProps | + | 0 | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/ReturnAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/ReturnAcceptance.feature new file mode 100644 index 000000000..b01a988f5 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/ReturnAcceptance.feature @@ -0,0 +1,312 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: ReturnAcceptanceTest + + Scenario: Allow addition + Given an empty graph + And having executed: + """ + CREATE ({id: 1337, version: 99}) + """ + When executing query: + """ + MATCH (a) + WHERE a.id = 1337 + RETURN a.version + 5 + """ + Then the result should be: + | a.version + 5 | + | 104 | + And no side effects + + Scenario: Limit to two hits + Given an empty graph + When executing query: + """ + UNWIND [1, 1, 1, 1, 1] AS i + RETURN i + LIMIT 2 + """ + Then the result should be: + | i | + | 1 | + | 1 | + And no side effects + + Scenario: Limit to two hits with explicit order + Given an empty graph + And having executed: + """ + CREATE ({name: 'A'}), + ({name: 'B'}), + ({name: 'C'}), + ({name: 'D'}), + ({name: 'E'}) + """ + When executing query: + """ + MATCH (n) + RETURN n + ORDER BY n.name ASC + LIMIT 2 + """ + Then the result should be: + | n | + | ({name: 'A'}) | + | ({name: 'B'}) | + And no side effects + + Scenario: Start the result from the second row + Given an empty graph + And having executed: + """ + CREATE ({name: 'A'}), + ({name: 'B'}), + ({name: 'C'}), + ({name: 'D'}), + ({name: 'E'}) + """ + When executing query: + """ + MATCH (n) + RETURN n + ORDER BY n.name ASC + SKIP 2 + """ + Then the result should be, in order: + | n | + | ({name: 'C'}) | + | ({name: 'D'}) | + | ({name: 'E'}) | + And no side effects + + Scenario: Start the result from the second row by param + Given an empty graph + And having executed: + """ + CREATE ({name: 'A'}), + ({name: 'B'}), + ({name: 'C'}), + ({name: 'D'}), + ({name: 'E'}) + """ + And parameters are: + | skipAmount | 2 | + When executing query: + """ + MATCH (n) + RETURN n + ORDER BY n.name ASC + SKIP $skipAmount + """ + Then the result should be, in order: + | n | + | ({name: 'C'}) | + | ({name: 'D'}) | + | ({name: 'E'}) | + And no side effects + + Scenario: Get rows in the middle + Given an empty graph + And having executed: + """ + CREATE ({name: 'A'}), + ({name: 'B'}), + ({name: 'C'}), + ({name: 'D'}), + ({name: 'E'}) + """ + When executing query: + """ + MATCH (n) + RETURN n + ORDER BY n.name ASC + SKIP 2 + LIMIT 2 + """ + Then the result should be, in order: + | n | + | ({name: 'C'}) | + | ({name: 'D'}) | + And no side effects + + Scenario: Get rows in the middle by param + Given an empty graph + And having executed: + """ + CREATE ({name: 'A'}), + ({name: 'B'}), + ({name: 'C'}), + ({name: 'D'}), + ({name: 'E'}) + """ + And parameters are: + | s | 2 | + | l | 2 | + When executing query: + """ + MATCH (n) + RETURN n + ORDER BY n.name ASC + SKIP $s + LIMIT $l + """ + Then the result should be, in order: + | n | + | ({name: 'C'}) | + | ({name: 'D'}) | + And no side effects + + Scenario: Sort on aggregated function + Given an empty graph + And having executed: + """ + CREATE ({division: 'A', age: 22}), + ({division: 'B', age: 33}), + ({division: 'B', age: 44}), + ({division: 'C', age: 55}) + """ + When executing query: + """ + MATCH (n) + RETURN n.division, max(n.age) + ORDER BY max(n.age) + """ + Then the result should be, in order: + | n.division | max(n.age) | + | 'A' | 22 | + | 'B' | 44 | + | 'C' | 55 | + And no side effects + + Scenario: Support sort and distinct + Given an empty graph + And having executed: + """ + CREATE ({name: 'A'}), + ({name: 'B'}), + ({name: 'C'}) + """ + When executing query: + """ + MATCH (a) + RETURN DISTINCT a + ORDER BY a.name + """ + Then the result should be, in order: + | a | + | ({name: 'A'}) | + | ({name: 'B'}) | + | ({name: 'C'}) | + And no side effects + + Scenario: Support column renaming + Given an empty graph + And having executed: + """ + CREATE (:Singleton) + """ + When executing query: + """ + MATCH (a) + RETURN a AS ColumnName + """ + Then the result should be: + | ColumnName | + | (:Singleton) | + And no side effects + + Scenario: Support ordering by a property after being distinct-ified + Given an empty graph + And having executed: + """ + CREATE (:A)-[:T]->(:B) + """ + When executing query: + """ + MATCH (a)-->(b) + RETURN DISTINCT b + ORDER BY b.name + """ + Then the result should be, in order: + | b | + | (:B) | + And no side effects + + Scenario: Arithmetic precedence test + Given any graph + When executing query: + """ + RETURN 12 / 4 * 3 - 2 * 4 + """ + Then the result should be: + | 12 / 4 * 3 - 2 * 4 | + | 1 | + And no side effects + + Scenario: Arithmetic precedence with parenthesis test + Given any graph + When executing query: + """ + RETURN 12 / 4 * (3 - 2 * 4) + """ + Then the result should be: + | 12 / 4 * (3 - 2 * 4) | + | -15 | + And no side effects + + Scenario: Count star should count everything in scope + Given an empty graph + And having executed: + """ + CREATE (:L1), (:L2), (:L3) + """ + When executing query: + """ + MATCH (a) + RETURN a, count(*) + ORDER BY count(*) + """ + Then the result should be: + | a | count(*) | + | (:L1) | 1 | + | (:L2) | 1 | + | (:L3) | 1 | + And no side effects + + Scenario: Absolute function + Given any graph + When executing query: + """ + RETURN abs(-1) + """ + Then the result should be: + | abs(-1) | + | 1 | + And no side effects + + Scenario: Return collection size + Given any graph + When executing query: + """ + RETURN size([1, 2, 3]) AS n + """ + Then the result should be: + | n | + | 3 | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/ReturnAcceptance2.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/ReturnAcceptance2.feature new file mode 100644 index 000000000..f804ca7f4 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/ReturnAcceptance2.feature @@ -0,0 +1,623 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: ReturnAcceptance2 + + Scenario: Fail when returning properties of deleted nodes + Given an empty graph + And having executed: + """ + CREATE ({p: 0}) + """ + When executing query: + """ + MATCH (n) + DELETE n + RETURN n.p + """ + Then a EntityNotFound should be raised at runtime: DeletedEntityAccess + + Scenario: Fail when returning labels of deleted nodes + Given an empty graph + And having executed: + """ + CREATE (:A) + """ + When executing query: + """ + MATCH (n) + DELETE n + RETURN labels(n) + """ + Then a EntityNotFound should be raised at runtime: DeletedEntityAccess + + Scenario: Fail when returning properties of deleted relationships + Given an empty graph + And having executed: + """ + CREATE ()-[:T {p: 0}]->() + """ + When executing query: + """ + MATCH ()-[r]->() + DELETE r + RETURN r.p + """ + Then a EntityNotFound should be raised at runtime: DeletedEntityAccess + + Scenario: Do not fail when returning type of deleted relationships + Given an empty graph + And having executed: + """ + CREATE ()-[:T]->() + """ + When executing query: + """ + MATCH ()-[r]->() + DELETE r + RETURN type(r) + """ + Then the result should be: + | type(r) | + | 'T' | + And the side effects should be: + | -relationships | 1 | + + Scenario: Accept valid Unicode literal + Given any graph + When executing query: + """ + RETURN '\u01FF' AS a + """ + Then the result should be: + | a | + | 'ǿ' | + And no side effects + + Scenario: LIMIT 0 should return an empty result + Given an empty graph + And having executed: + """ + CREATE (), (), () + """ + When executing query: + """ + MATCH (n) + RETURN n + LIMIT 0 + """ + Then the result should be: + | n | + And no side effects + + Scenario: Fail when sorting on variable removed by DISTINCT + Given an empty graph + And having executed: + """ + CREATE ({name: 'A', age: 13}), ({name: 'B', age: 12}), ({name: 'C', age: 11}) + """ + When executing query: + """ + MATCH (a) + RETURN DISTINCT a.name + ORDER BY a.age + """ + Then a SyntaxError should be raised at compile time: UndefinedVariable + + Scenario: Ordering with aggregation + Given an empty graph + And having executed: + """ + CREATE ({name: 'nisse'}) + """ + When executing query: + """ + MATCH (n) + RETURN n.name, count(*) AS foo + ORDER BY n.name + """ + Then the result should be: + | n.name | foo | + | 'nisse' | 1 | + And no side effects + + Scenario: DISTINCT on nullable values + Given an empty graph + And having executed: + """ + CREATE ({name: 'Florescu'}), (), () + """ + When executing query: + """ + MATCH (n) + RETURN DISTINCT n.name + """ + Then the result should be: + | n.name | + | 'Florescu' | + | null | + And no side effects + + Scenario: Return all variables + Given an empty graph + And having executed: + """ + CREATE (:Start)-[:T]->() + """ + When executing query: + """ + MATCH p = (a:Start)-->(b) + RETURN * + """ + Then the result should be: + | a | b | p | + | (:Start) | () | <(:Start)-[:T]->()> | + And no side effects + + Scenario: Setting and returning the size of a list property + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + SET n.x = [1, 2, 3] + RETURN size(n.x) + """ + Then the result should be: + | size(n.x) | + | 3 | + And the side effects should be: + | +properties | 1 | + + Scenario: `sqrt()` returning float values + Given any graph + When executing query: + """ + RETURN sqrt(12.96) + """ + Then the result should be: + | sqrt(12.96) | + | 3.6 | + And no side effects + + Scenario: Arithmetic expressions inside aggregation + Given an empty graph + And having executed: + """ + CREATE (andres {name: 'Andres'}), + (michael {name: 'Michael'}), + (peter {name: 'Peter'}), + (bread {type: 'Bread'}), + (veggies {type: 'Veggies'}), + (meat {type: 'Meat'}) + CREATE (andres)-[:ATE {times: 10}]->(bread), + (andres)-[:ATE {times: 8}]->(veggies), + (michael)-[:ATE {times: 4}]->(veggies), + (michael)-[:ATE {times: 6}]->(bread), + (michael)-[:ATE {times: 9}]->(meat), + (peter)-[:ATE {times: 7}]->(veggies), + (peter)-[:ATE {times: 7}]->(bread), + (peter)-[:ATE {times: 4}]->(meat) + """ + When executing query: + """ + MATCH (me)-[r1:ATE]->()<-[r2:ATE]-(you) + WHERE me.name = 'Michael' + WITH me, count(DISTINCT r1) AS H1, count(DISTINCT r2) AS H2, you + MATCH (me)-[r1:ATE]->()<-[r2:ATE]-(you) + RETURN me, you, sum((1 - abs(r1.times / H1 - r2.times / H2)) * (r1.times + r2.times) / (H1 + H2)) AS sum + """ + Then the result should be: + | me | you | sum | + | ({name: 'Michael'}) | ({name: 'Andres'}) | -7 | + | ({name: 'Michael'}) | ({name: 'Peter'}) | 0 | + And no side effects + + Scenario: Matching and disregarding output, then matching again + Given an empty graph + And having executed: + """ + CREATE (andres {name: 'Andres'}), + (michael {name: 'Michael'}), + (peter {name: 'Peter'}), + (bread {type: 'Bread'}), + (veggies {type: 'Veggies'}), + (meat {type: 'Meat'}) + CREATE (andres)-[:ATE {times: 10}]->(bread), + (andres)-[:ATE {times: 8}]->(veggies), + (michael)-[:ATE {times: 4}]->(veggies), + (michael)-[:ATE {times: 6}]->(bread), + (michael)-[:ATE {times: 9}]->(meat), + (peter)-[:ATE {times: 7}]->(veggies), + (peter)-[:ATE {times: 7}]->(bread), + (peter)-[:ATE {times: 4}]->(meat) + """ + When executing query: + """ + MATCH ()-->() + WITH 1 AS x + MATCH ()-[r1]->()<--() + RETURN sum(r1.times) + """ + Then the result should be: + | sum(r1.times) | + | 776 | + And no side effects + + Scenario: Returning a list property + Given an empty graph + And having executed: + """ + CREATE ({foo: [1, 2, 3]}) + """ + When executing query: + """ + MATCH (n) + RETURN n + """ + Then the result should be: + | n | + | ({foo: [1, 2, 3]}) | + And no side effects + + Scenario: Returning a projected map + Given an empty graph + And having executed: + """ + CREATE ({foo: [1, 2, 3]}) + """ + When executing query: + """ + RETURN {a: 1, b: 'foo'} + """ + Then the result should be: + | {a: 1, b: 'foo'} | + | {a: 1, b: 'foo'} | + And no side effects + + Scenario: Returning an expression + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (a) + RETURN exists(a.id), a IS NOT NULL + """ + Then the result should be: + | exists(a.id) | a IS NOT NULL | + | false | true | + And no side effects + + Scenario: Concatenating and returning the size of literal lists + Given any graph + When executing query: + """ + RETURN size([[], []] + [[]]) AS l + """ + Then the result should be: + | l | + | 3 | + And no side effects + + Scenario: Returning nested expressions based on list property + Given an empty graph + And having executed: + """ + CREATE () + """ + When executing query: + """ + MATCH (n) + SET n.array = [1, 2, 3, 4, 5] + RETURN tail(tail(n.array)) + """ + Then the result should be: + | tail(tail(n.array)) | + | [3, 4, 5] | + And the side effects should be: + | +properties | 1 | + + Scenario: Limiting amount of rows when there are fewer left than the LIMIT argument + Given an empty graph + And having executed: + """ + UNWIND range(0, 15) AS i + CREATE ({count: i}) + """ + When executing query: + """ + MATCH (a) + RETURN a.count + ORDER BY a.count + SKIP 10 + LIMIT 10 + """ + Then the result should be, in order: + | a.count | + | 10 | + | 11 | + | 12 | + | 13 | + | 14 | + | 15 | + And no side effects + + Scenario: `substring()` with default second argument + Given any graph + When executing query: + """ + RETURN substring('0123456789', 1) AS s + """ + Then the result should be: + | s | + | '123456789' | + And no side effects + + Scenario: Returning all variables with ordering + Given an empty graph + And having executed: + """ + CREATE ({id: 1}), ({id: 10}) + """ + When executing query: + """ + MATCH (n) + RETURN * + ORDER BY n.id + """ + Then the result should be, in order: + | n | + | ({id: 1}) | + | ({id: 10}) | + And no side effects + + Scenario: Using aliased DISTINCT expression in ORDER BY + Given an empty graph + And having executed: + """ + CREATE ({id: 1}), ({id: 10}) + """ + When executing query: + """ + MATCH (n) + RETURN DISTINCT n.id AS id + ORDER BY id DESC + """ + Then the result should be, in order: + | id | + | 10 | + | 1 | + And no side effects + + Scenario: Returned columns do not change from using ORDER BY + Given an empty graph + And having executed: + """ + CREATE ({id: 1}), ({id: 10}) + """ + When executing query: + """ + MATCH (n) + RETURN DISTINCT n + ORDER BY n.id + """ + Then the result should be, in order: + | n | + | ({id: 1}) | + | ({id: 10}) | + And no side effects + + Scenario: Arithmetic expressions should propagate null values + Given any graph + When executing query: + """ + RETURN 1 + (2 - (3 * (4 / (5 ^ (6 % null))))) AS a + """ + Then the result should be: + | a | + | null | + And no side effects + + Scenario: Indexing into nested literal lists + Given any graph + When executing query: + """ + RETURN [[1]][0][0] + """ + Then the result should be: + | [[1]][0][0] | + | 1 | + And no side effects + + Scenario: Aliasing expressions + Given an empty graph + And having executed: + """ + CREATE ({id: 42}) + """ + When executing query: + """ + MATCH (a) + RETURN a.id AS a, a.id + """ + Then the result should be: + | a | a.id | + | 42 | 42 | + And no side effects + + Scenario: Projecting an arithmetic expression with aggregation + Given an empty graph + And having executed: + """ + CREATE ({id: 42}) + """ + When executing query: + """ + MATCH (a) + RETURN a, count(a) + 3 + """ + Then the result should be: + | a | count(a) + 3 | + | ({id: 42}) | 4 | + And no side effects + + Scenario: Multiple aliasing and backreferencing + Given any graph + When executing query: + """ + CREATE (m {id: 0}) + WITH {first: m.id} AS m + WITH {second: m.first} AS m + RETURN m.second + """ + Then the result should be: + | m.second | + | 0 | + And the side effects should be: + | +nodes | 1 | + | +properties | 1 | + + Scenario: Aggregating by a list property has a correct definition of equality + Given an empty graph + And having executed: + """ + CREATE ({a: [1, 2, 3]}), ({a: [1, 2, 3]}) + """ + When executing query: + """ + MATCH (a) + WITH a.a AS a, count(*) AS count + RETURN count + """ + Then the result should be: + | count | + | 2 | + And no side effects + + Scenario: Reusing variable names + Given an empty graph + And having executed: + """ + CREATE (a:Person), (b:Person), (m:Message {id: 10}) + CREATE (a)-[:LIKE {creationDate: 20160614}]->(m)-[:POSTED_BY]->(b) + """ + When executing query: + """ + MATCH (person:Person)<--(message)<-[like]-(:Person) + WITH like.creationDate AS likeTime, person AS person + ORDER BY likeTime, message.id + WITH head(collect({likeTime: likeTime})) AS latestLike, person AS person + RETURN latestLike.likeTime AS likeTime + ORDER BY likeTime + """ + Then the result should be, in order: + | likeTime | + | 20160614 | + And no side effects + + Scenario: Concatenating lists of same type + Given any graph + When executing query: + """ + RETURN [1, 10, 100] + [4, 5] AS foo + """ + Then the result should be: + | foo | + | [1, 10, 100, 4, 5] | + And no side effects + + Scenario: Appending lists of same type + Given any graph + When executing query: + """ + RETURN [false, true] + false AS foo + """ + Then the result should be: + | foo | + | [false, true, false] | + And no side effects + + Scenario: DISTINCT inside aggregation should work with lists in maps + Given an empty graph + And having executed: + """ + CREATE ({list: ['A', 'B']}), ({list: ['A', 'B']}) + """ + When executing query: + """ + MATCH (n) + RETURN count(DISTINCT {foo: n.list}) AS count + """ + Then the result should be: + | count | + | 1 | + And no side effects + + Scenario: Handling DISTINCT with lists in maps + Given an empty graph + And having executed: + """ + CREATE ({list: ['A', 'B']}), ({list: ['A', 'B']}) + """ + When executing query: + """ + MATCH (n) + WITH DISTINCT {foo: n.list} AS map + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 1 | + And no side effects + + Scenario: DISTINCT inside aggregation should work with nested lists in maps + Given an empty graph + And having executed: + """ + CREATE ({list: ['A', 'B']}), ({list: ['A', 'B']}) + """ + When executing query: + """ + MATCH (n) + RETURN count(DISTINCT {foo: [[n.list, n.list], [n.list, n.list]]}) AS count + """ + Then the result should be: + | count | + | 1 | + And no side effects + + Scenario: DISTINCT inside aggregation should work with nested lists of maps in maps + Given an empty graph + And having executed: + """ + CREATE ({list: ['A', 'B']}), ({list: ['A', 'B']}) + """ + When executing query: + """ + MATCH (n) + RETURN count(DISTINCT {foo: [{bar: n.list}, {baz: {apa: n.list}}]}) AS count + """ + Then the result should be: + | count | + | 1 | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/SemanticErrorAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/SemanticErrorAcceptance.feature new file mode 100644 index 000000000..7ede72817 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/SemanticErrorAcceptance.feature @@ -0,0 +1,394 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: SemanticErrorAcceptance + + Background: + Given any graph + + Scenario: Failing when returning an undefined variable + When executing query: + """ + MATCH () + RETURN foo + """ + Then a SyntaxError should be raised at compile time: UndefinedVariable + + Scenario: Failing when comparing to an undefined variable + When executing query: + """ + MATCH (s) + WHERE s.name = undefinedVariable + AND s.age = 10 + RETURN s + """ + Then a SyntaxError should be raised at compile time: UndefinedVariable + + Scenario: Failing when using IN on a string literal + When executing query: + """ + MATCH (n) + WHERE n.id IN '' + RETURN 1 + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: Failing when using IN on an integer literal + When executing query: + """ + MATCH (n) + WHERE n.id IN 1 + RETURN 1 + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: Failing when using IN on a float literal + When executing query: + """ + MATCH (n) + WHERE n.id IN 1.0 + RETURN 1 + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: Failing when using IN on a boolean literal + When executing query: + """ + MATCH (n) + WHERE n.id IN true + RETURN 1 + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: Failing when a node is used as a relationship + When executing query: + """ + MATCH (r) + MATCH ()-[r]-() + RETURN r + """ + Then a SyntaxError should be raised at compile time: VariableTypeConflict + + Scenario: Failing when a relationship is used as a node + When executing query: + """ + MATCH ()-[r]-(r) + RETURN r + """ + Then a SyntaxError should be raised at compile time: VariableTypeConflict + + Scenario: Failing when using `type()` on a node + When executing query: + """ + MATCH (r) + RETURN type(r) + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: Failing when using `length()` on a node + When executing query: + """ + MATCH (r) + RETURN length(r) + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: Failing when re-using a relationship in the same pattern + When executing query: + """ + MATCH (a)-[r]->()-[r]->(a) + RETURN r + """ + Then a SyntaxError should be raised at compile time: RelationshipUniquenessViolation + + Scenario: Failing when using NOT on string literal + When executing query: + """ + RETURN NOT 'foo' + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: Failing when using variable length relationship in CREATE + When executing query: + """ + CREATE ()-[:FOO*2]->() + """ + Then a SyntaxError should be raised at compile time: CreatingVarLength + + Scenario: Failing when using variable length relationship in MERGE + When executing query: + """ + MERGE (a) + MERGE (b) + MERGE (a)-[:FOO*2]->(b) + """ + Then a SyntaxError should be raised at compile time: CreatingVarLength + + Scenario: Failing when using parameter as node predicate in MATCH + When executing query: + """ + MATCH (n $param) + RETURN n + """ + Then a SyntaxError should be raised at compile time: InvalidParameterUse + + Scenario: Failing when using parameter as relationship predicate in MATCH + When executing query: + """ + MATCH ()-[r:FOO $param]->() + RETURN r + """ + Then a SyntaxError should be raised at compile time: InvalidParameterUse + + Scenario: Failing when using parameter as node predicate in MERGE + When executing query: + """ + MERGE (n $param) + RETURN n + """ + Then a SyntaxError should be raised at compile time: InvalidParameterUse + + Scenario: Failing when using parameter as relationship predicate in MERGE + When executing query: + """ + MERGE (a) + MERGE (b) + MERGE (a)-[r:FOO $param]->(b) + RETURN r + """ + Then a SyntaxError should be raised at compile time: InvalidParameterUse + + Scenario: Failing when deleting an integer expression + When executing query: + """ + MATCH () + DELETE 1 + 1 + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: Failing when using CREATE on a node that is already bound + When executing query: + """ + MATCH (a) + CREATE (a) + """ + Then a SyntaxError should be raised at compile time: VariableAlreadyBound + + Scenario: Failing when using MERGE on a node that is already bound + When executing query: + """ + MATCH (a) + CREATE (a) + """ + Then a SyntaxError should be raised at compile time: VariableAlreadyBound + + Scenario: Failing when using CREATE on a relationship that is already bound + When executing query: + """ + MATCH ()-[r]->() + CREATE ()-[r]->() + """ + Then a SyntaxError should be raised at compile time: VariableAlreadyBound + + Scenario: Failing when using MERGE on a relationship that is already bound + When executing query: + """ + MATCH (a)-[r]->(b) + MERGE (a)-[r]->(b) + """ + Then a SyntaxError should be raised at compile time: VariableAlreadyBound + + Scenario: Failing when using undefined variable in ON CREATE + When executing query: + """ + MERGE (n) + ON CREATE SET x.foo = 1 + """ + Then a SyntaxError should be raised at compile time: UndefinedVariable + + Scenario: Failing when using undefined variable in ON MATCH + When executing query: + """ + MERGE (n) + ON MATCH SET x.foo = 1 + """ + Then a SyntaxError should be raised at compile time: UndefinedVariable + + Scenario: Failing when using MATCH after OPTIONAL MATCH + When executing query: + """ + OPTIONAL MATCH ()-->() + MATCH ()-->(d) + RETURN d + """ + Then a SyntaxError should be raised at compile time: InvalidClauseComposition + + Scenario: Failing when float value is too large + When executing query: + """ + RETURN 1.34E999 + """ + Then a SyntaxError should be raised at compile time: FloatingPointOverflow + + Scenario: Handling property access on the Any type + When executing query: + """ + WITH [{prop: 0}, 1] AS list + RETURN (list[0]).prop + """ + Then the result should be: + | (list[0]).prop | + | 0 | + And no side effects + + Scenario: Failing when performing property access on a non-map 1 + When executing query: + """ + WITH [{prop: 0}, 1] AS list + RETURN (list[1]).prop + """ + Then a TypeError should be raised at runtime: PropertyAccessOnNonMap + + Scenario: Failing when performing property access on a non-map 2 + When executing query: + """ + CREATE (n {prop: 'foo'}) + WITH n.prop AS n2 + RETURN n2.prop + """ + Then a TypeError should be raised at runtime: PropertyAccessOnNonMap + + Scenario: Failing when checking existence of a non-property and non-pattern + When executing query: + """ + MATCH (n) + RETURN exists(n.prop + 1) + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentExpression + + Scenario: Bad arguments for `range()` + When executing query: + """ + RETURN range(2, 8, 0) + """ + Then a ArgumentError should be raised at runtime: NumberOutOfRange + + Scenario: Fail for invalid Unicode hyphen in subtraction + When executing query: + """ + RETURN 42 — 41 + """ + Then a SyntaxError should be raised at compile time: InvalidUnicodeCharacter + + Scenario: Failing for `size()` on paths + When executing query: + """ + MATCH p = (a)-[*]->(b) + RETURN size(p) + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: Failing when using aggregation in list comprehension + When executing query: + """ + MATCH (n) + RETURN [x IN [1, 2, 3, 4, 5] | count(*)] + """ + Then a SyntaxError should be raised at compile time: InvalidAggregation + + Scenario: Failing when using non-constants in SKIP + When executing query: + """ + MATCH (n) + RETURN n + SKIP n.count + """ + Then a SyntaxError should be raised at compile time: NonConstantExpression + + Scenario: Failing when using negative value in SKIP + When executing query: + """ + MATCH (n) + RETURN n + SKIP -1 + """ + Then a SyntaxError should be raised at compile time: NegativeIntegerArgument + + Scenario: Failing when using non-constants in LIMIT + When executing query: + """ + MATCH (n) + RETURN n + LIMIT n.count + """ + Then a SyntaxError should be raised at compile time: NonConstantExpression + + Scenario: Failing when using negative value in LIMIT + When executing query: + """ + MATCH (n) + RETURN n + LIMIT -1 + """ + Then a SyntaxError should be raised at compile time: NegativeIntegerArgument + + Scenario: Failing when using floating point in LIMIT + When executing query: + """ + MATCH (n) + RETURN n + LIMIT 1.7 + """ + Then a SyntaxError should be raised at compile time: InvalidArgumentType + + Scenario: Failing when creating relationship without type + When executing query: + """ + CREATE ()-->() + """ + Then a SyntaxError should be raised at compile time: NoSingleRelationshipType + + Scenario: Failing when merging relationship without type + When executing query: + """ + CREATE (a), (b) + MERGE (a)-->(b) + """ + Then a SyntaxError should be raised at compile time: NoSingleRelationshipType + + Scenario: Failing when merging relationship without type, no colon + When executing query: + """ + MATCH (a), (b) + MERGE (a)-[NO_COLON]->(b) + """ + Then a SyntaxError should be raised at compile time: NoSingleRelationshipType + + Scenario: Failing when creating relationship with more than one type + When executing query: + """ + CREATE ()-[:A|:B]->() + """ + Then a SyntaxError should be raised at compile time: NoSingleRelationshipType + + Scenario: Failing when merging relationship with more than one type + When executing query: + """ + CREATE (a), (b) + MERGE (a)-[:A|:B]->(b) + """ + Then a SyntaxError should be raised at compile time: NoSingleRelationshipType diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/SetAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/SetAcceptance.feature new file mode 100644 index 000000000..cf0835899 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/SetAcceptance.feature @@ -0,0 +1,289 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: SetAcceptance + + Scenario: Setting a node property to null removes the existing property + Given an empty graph + And having executed: + """ + CREATE (:A {property1: 23, property2: 46}) + """ + When executing query: + """ + MATCH (n:A) + SET n.property1 = null + RETURN n + """ + Then the result should be: + | n | + | (:A {property2: 46}) | + And the side effects should be: + | -properties | 1 | + + Scenario: Setting a relationship property to null removes the existing property + Given an empty graph + And having executed: + """ + CREATE ()-[:REL {property1: 12, property2: 24}]->() + """ + When executing query: + """ + MATCH ()-[r]->() + SET r.property1 = null + RETURN r + """ + Then the result should be: + | r | + | [:REL {property2: 24}] | + And the side effects should be: + | -properties | 1 | + + Scenario: Set a property + Given any graph + And having executed: + """ + CREATE (:A {name: 'Andres'}) + """ + When executing query: + """ + MATCH (n:A) + WHERE n.name = 'Andres' + SET n.name = 'Michael' + RETURN n + """ + Then the result should be: + | n | + | (:A {name: 'Michael'}) | + And the side effects should be: + | +properties | 1 | + | -properties | 1 | + + Scenario: Set a property to an expression + Given an empty graph + And having executed: + """ + CREATE (:A {name: 'Andres'}) + """ + When executing query: + """ + MATCH (n:A) + WHERE n.name = 'Andres' + SET n.name = n.name + ' was here' + RETURN n + """ + Then the result should be: + | n | + | (:A {name: 'Andres was here'}) | + And the side effects should be: + | +properties | 1 | + | -properties | 1 | + + Scenario: Set a property by selecting the node using a simple expression + Given an empty graph + And having executed: + """ + CREATE (:A) + """ + When executing query: + """ + MATCH (n:A) + SET (n).name = 'neo4j' + RETURN n + """ + Then the result should be: + | n | + | (:A {name: 'neo4j'}) | + And the side effects should be: + | +properties | 1 | + + Scenario: Set a property by selecting the relationship using a simple expression + Given an empty graph + And having executed: + """ + CREATE ()-[:REL]->() + """ + When executing query: + """ + MATCH ()-[r:REL]->() + SET (r).name = 'neo4j' + RETURN r + """ + Then the result should be: + | r | + | [:REL {name: 'neo4j'}] | + And the side effects should be: + | +properties | 1 | + + Scenario: Setting a property to null removes the property + Given an empty graph + And having executed: + """ + CREATE (:A {name: 'Michael', age: 35}) + """ + When executing query: + """ + MATCH (n) + WHERE n.name = 'Michael' + SET n.name = null + RETURN n + """ + Then the result should be: + | n | + | (:A {age: 35}) | + And the side effects should be: + | -properties | 1 | + + Scenario: Add a label to a node + Given an empty graph + And having executed: + """ + CREATE (:A) + """ + When executing query: + """ + MATCH (n:A) + SET n:Foo + RETURN n + """ + Then the result should be: + | n | + | (:A:Foo) | + And the side effects should be: + | +labels | 1 | + + Scenario: Adding a list property + Given an empty graph + And having executed: + """ + CREATE (:A) + """ + When executing query: + """ + MATCH (n:A) + SET n.x = [1, 2, 3] + RETURN [i IN n.x | i / 2.0] AS x + """ + Then the result should be: + | x | + | [0.5, 1.0, 1.5] | + And the side effects should be: + | +properties | 1 | + + Scenario: Concatenate elements onto a list property + Given any graph + When executing query: + """ + CREATE (a {foo: [1, 2, 3]}) + SET a.foo = a.foo + [4, 5] + RETURN a.foo + """ + Then the result should be: + | a.foo | + | [1, 2, 3, 4, 5] | + And the side effects should be: + | +nodes | 1 | + | +properties | 1 | + + Scenario: Concatenate elements in reverse onto a list property + Given any graph + When executing query: + """ + CREATE (a {foo: [3, 4, 5]}) + SET a.foo = [1, 2] + a.foo + RETURN a.foo + """ + Then the result should be: + | a.foo | + | [1, 2, 3, 4, 5] | + And the side effects should be: + | +nodes | 1 | + | +properties | 1 | + + Scenario: Overwrite values when using += + Given an empty graph + And having executed: + """ + CREATE (:X {foo: 'A', bar: 'B'}) + """ + When executing query: + """ + MATCH (n:X {foo: 'A'}) + SET n += {bar: 'C'} + RETURN n + """ + Then the result should be: + | n | + | (:X {foo: 'A', bar: 'C'}) | + And the side effects should be: + | +properties | 1 | + | -properties | 1 | + + Scenario: Retain old values when using += + Given an empty graph + And having executed: + """ + CREATE (:X {foo: 'A'}) + """ + When executing query: + """ + MATCH (n:X {foo: 'A'}) + SET n += {bar: 'B'} + RETURN n + """ + Then the result should be: + | n | + | (:X {foo: 'A', bar: 'B'}) | + And the side effects should be: + | +properties | 1 | + + Scenario: Explicit null values in a map remove old values + Given an empty graph + And having executed: + """ + CREATE (:X {foo: 'A', bar: 'B'}) + """ + When executing query: + """ + MATCH (n:X {foo: 'A'}) + SET n += {foo: null} + RETURN n + """ + Then the result should be: + | n | + | (:X {bar: 'B'}) | + And the side effects should be: + | -properties | 1 | + + Scenario: Non-existent values in a property map are removed with SET = + Given an empty graph + And having executed: + """ + CREATE (:X {foo: 'A', bar: 'B'}) + """ + When executing query: + """ + MATCH (n:X {foo: 'A'}) + SET n = {foo: 'B', baz: 'C'} + RETURN n + """ + Then the result should be: + | n | + | (:X {foo: 'B', baz: 'C'}) | + And the side effects should be: + | +properties | 2 | + | -properties | 2 | diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/SkipLimitAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/SkipLimitAcceptance.feature new file mode 100644 index 000000000..226d4483b --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/SkipLimitAcceptance.feature @@ -0,0 +1,71 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: SkipLimitAcceptanceTest + + Background: + Given any graph + + Scenario: SKIP with an expression that depends on variables should fail + When executing query: + """ + MATCH (n) RETURN n SKIP n.count + """ + Then a SyntaxError should be raised at compile time: NonConstantExpression + + Scenario: LIMIT with an expression that depends on variables should fail + When executing query: + """ + MATCH (n) RETURN n LIMIT n.count + """ + Then a SyntaxError should be raised at compile time: NonConstantExpression + + Scenario: SKIP with an expression that does not depend on variables + And having executed: + """ + UNWIND range(1, 10) AS i + CREATE ({nr: i}) + """ + When executing query: + """ + MATCH (n) + WITH n SKIP toInteger(rand()*9) + WITH count(*) AS count + RETURN count > 0 AS nonEmpty + """ + Then the result should be: + | nonEmpty | + | true | + And no side effects + + + Scenario: LIMIT with an expression that does not depend on variables + And having executed: + """ + UNWIND range(1, 3) AS i + CREATE ({nr: i}) + """ + When executing query: + """ + MATCH (n) + WITH n LIMIT toInteger(ceil(1.7)) + RETURN count(*) AS count + """ + Then the result should be: + | count | + | 2 | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/StartingPointAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/StartingPointAcceptance.feature new file mode 100644 index 000000000..4b0b5c3f1 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/StartingPointAcceptance.feature @@ -0,0 +1,76 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: StartingPointAcceptance + + Scenario: Find all nodes + Given an empty graph + And having executed: + """ + CREATE ({name: 'a'}), + ({name: 'b'}), + ({name: 'c'}) + """ + When executing query: + """ + MATCH (n) + RETURN n + """ + Then the result should be: + | n | + | ({name: 'a'}) | + | ({name: 'b'}) | + | ({name: 'c'}) | + And no side effects + + Scenario: Find labelled nodes + Given an empty graph + And having executed: + """ + CREATE ({name: 'a'}), + (:Person), + (:Animal), + (:Animal) + """ + When executing query: + """ + MATCH (n:Animal) + RETURN n + """ + Then the result should be: + | n | + | (:Animal) | + | (:Animal) | + And no side effects + + Scenario: Find nodes by property + Given an empty graph + And having executed: + """ + CREATE ({prop: 1}), + ({prop: 2}) + """ + When executing query: + """ + MATCH (n) + WHERE n.prop = 2 + RETURN n + """ + Then the result should be: + | n | + | ({prop: 2}) | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/StartsWithAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/StartsWithAcceptance.feature new file mode 100644 index 000000000..0141d1dd6 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/StartsWithAcceptance.feature @@ -0,0 +1,360 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: StartsWithAcceptance + + Background: + Given an empty graph + And having executed: + """ + CREATE (:Label {name: 'ABCDEF'}), (:Label {name: 'AB'}), + (:Label {name: 'abcdef'}), (:Label {name: 'ab'}), + (:Label {name: ''}), (:Label) + """ + + Scenario: Finding exact matches + When executing query: + """ + MATCH (a) + WHERE a.name STARTS WITH 'ABCDEF' + RETURN a + """ + Then the result should be: + | a | + | (:Label {name: 'ABCDEF'}) | + And no side effects + + Scenario: Finding beginning of string + When executing query: + """ + MATCH (a) + WHERE a.name STARTS WITH 'ABC' + RETURN a + """ + Then the result should be: + | a | + | (:Label {name: 'ABCDEF'}) | + And no side effects + + Scenario: Finding end of string 1 + When executing query: + """ + MATCH (a) + WHERE a.name ENDS WITH 'DEF' + RETURN a + """ + Then the result should be: + | a | + | (:Label {name: 'ABCDEF'}) | + And no side effects + + Scenario: Finding end of string 2 + When executing query: + """ + MATCH (a) + WHERE a.name ENDS WITH 'AB' + RETURN a + """ + Then the result should be: + | a | + | (:Label {name: 'AB'}) | + And no side effects + + Scenario: Finding middle of string + When executing query: + """ + MATCH (a) + WHERE a.name STARTS WITH 'a' + AND a.name ENDS WITH 'f' + RETURN a + """ + Then the result should be: + | a | + | (:Label {name: 'abcdef'}) | + And no side effects + + Scenario: Finding the empty string + When executing query: + """ + MATCH (a) + WHERE a.name STARTS WITH '' + RETURN a + """ + Then the result should be: + | a | + | (:Label {name: 'ABCDEF'}) | + | (:Label {name: 'AB'}) | + | (:Label {name: 'abcdef'}) | + | (:Label {name: 'ab'}) | + | (:Label {name: ''}) | + And no side effects + + Scenario: Finding when the middle is known + When executing query: + """ + MATCH (a) + WHERE a.name CONTAINS 'CD' + RETURN a + """ + Then the result should be: + | a | + | (:Label {name: 'ABCDEF'}) | + And no side effects + + Scenario: Finding strings starting with whitespace + And having executed: + """ + CREATE (:Label {name: ' Foo '}), + (:Label {name: '\nFoo\n'}), + (:Label {name: '\tFoo\t'}) + """ + When executing query: + """ + MATCH (a) + WHERE a.name STARTS WITH ' ' + RETURN a.name AS name + """ + Then the result should be: + | name | + | ' Foo ' | + And no side effects + + Scenario: Finding strings starting with newline + And having executed: + """ + CREATE (:Label {name: ' Foo '}), + (:Label {name: '\nFoo\n'}), + (:Label {name: '\tFoo\t'}) + """ + When executing query: + """ + MATCH (a) + WHERE a.name STARTS WITH '\n' + RETURN a.name AS name + """ + Then the result should be: + | name | + | '\nFoo\n' | + And no side effects + + Scenario: Finding strings ending with newline + And having executed: + """ + CREATE (:Label {name: ' Foo '}), + (:Label {name: '\nFoo\n'}), + (:Label {name: '\tFoo\t'}) + """ + When executing query: + """ + MATCH (a) + WHERE a.name ENDS WITH '\n' + RETURN a.name AS name + """ + Then the result should be: + | name | + | '\nFoo\n' | + And no side effects + + Scenario: Finding strings ending with whitespace + And having executed: + """ + CREATE (:Label {name: ' Foo '}), + (:Label {name: '\nFoo\n'}), + (:Label {name: '\tFoo\t'}) + """ + When executing query: + """ + MATCH (a) + WHERE a.name ENDS WITH ' ' + RETURN a.name AS name + """ + Then the result should be: + | name | + | ' Foo ' | + And no side effects + + Scenario: Finding strings containing whitespace + And having executed: + """ + CREATE (:Label {name: ' Foo '}), + (:Label {name: '\nFoo\n'}), + (:Label {name: '\tFoo\t'}) + """ + When executing query: + """ + MATCH (a) + WHERE a.name CONTAINS ' ' + RETURN a.name AS name + """ + Then the result should be: + | name | + | ' Foo ' | + And no side effects + + Scenario: Finding strings containing newline + And having executed: + """ + CREATE (:Label {name: ' Foo '}), + (:Label {name: '\nFoo\n'}), + (:Label {name: '\tFoo\t'}) + """ + When executing query: + """ + MATCH (a) + WHERE a.name CONTAINS '\n' + RETURN a.name AS name + """ + Then the result should be: + | name | + | '\nFoo\n' | + And no side effects + + Scenario: No string starts with null + When executing query: + """ + MATCH (a) + WHERE a.name STARTS WITH null + RETURN a + """ + Then the result should be: + | a | + And no side effects + + Scenario: No string does not start with null + When executing query: + """ + MATCH (a) + WHERE NOT a.name STARTS WITH null + RETURN a + """ + Then the result should be: + | a | + And no side effects + + Scenario: No string ends with null + When executing query: + """ + MATCH (a) + WHERE a.name ENDS WITH null + RETURN a + """ + Then the result should be: + | a | + And no side effects + + Scenario: No string does not end with null + When executing query: + """ + MATCH (a) + WHERE NOT a.name ENDS WITH null + RETURN a + """ + Then the result should be: + | a | + And no side effects + + Scenario: No string contains null + When executing query: + """ + MATCH (a) + WHERE a.name CONTAINS null + RETURN a + """ + Then the result should be: + | a | + And no side effects + + Scenario: No string does not contain null + When executing query: + """ + MATCH (a) + WHERE NOT a.name CONTAINS null + RETURN a + """ + Then the result should be: + | a | + And no side effects + + Scenario: Combining string operators + When executing query: + """ + MATCH (a) + WHERE a.name STARTS WITH 'A' + AND a.name CONTAINS 'C' + AND a.name ENDS WITH 'EF' + RETURN a + """ + Then the result should be: + | a | + | (:Label {name: 'ABCDEF'}) | + And no side effects + + Scenario: NOT with CONTAINS + When executing query: + """ + MATCH (a) + WHERE NOT a.name CONTAINS 'b' + RETURN a + """ + Then the result should be: + | a | + | (:Label {name: 'ABCDEF'}) | + | (:Label {name: 'AB'}) | + | (:Label {name: ''}) | + And no side effects + + Scenario: Handling non-string operands for STARTS WITH + When executing query: + """ + WITH [1, 3.14, true, [], {}, null] AS operands + UNWIND operands AS op1 + UNWIND operands AS op2 + WITH op1 STARTS WITH op2 AS v + RETURN v, count(*) + """ + Then the result should be: + | v | count(*) | + | null | 36 | + And no side effects + + Scenario: Handling non-string operands for CONTAINS + When executing query: + """ + WITH [1, 3.14, true, [], {}, null] AS operands + UNWIND operands AS op1 + UNWIND operands AS op2 + WITH op1 STARTS WITH op2 AS v + RETURN v, count(*) + """ + Then the result should be: + | v | count(*) | + | null | 36 | + And no side effects + + Scenario: Handling non-string operands for ENDS WITH + When executing query: + """ + WITH [1, 3.14, true, [], {}, null] AS operands + UNWIND operands AS op1 + UNWIND operands AS op2 + WITH op1 STARTS WITH op2 AS v + RETURN v, count(*) + """ + Then the result should be: + | v | count(*) | + | null | 36 | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/SyntaxErrorAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/SyntaxErrorAcceptance.feature new file mode 100644 index 000000000..fdd57fa18 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/SyntaxErrorAcceptance.feature @@ -0,0 +1,50 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: SyntaxErrorAcceptance + + Background: + Given any graph + + Scenario: Using a non-existent function + When executing query: + """ + MATCH (a) + RETURN foo(a) + """ + Then a SyntaxError should be raised at compile time: UnknownFunction + + Scenario: Using `rand()` in aggregations + When executing query: + """ + RETURN count(rand()) + """ + Then a SyntaxError should be raised at compile time: NonConstantExpression + + Scenario: Supplying invalid hexadecimal literal 1 + When executing query: + """ + RETURN 0x23G34 + """ + Then a SyntaxError should be raised at compile time: InvalidNumberLiteral + + Scenario: Supplying invalid hexadecimal literal 2 + When executing query: + """ + RETURN 0x23j + """ + Then a SyntaxError should be raised at compile time: InvalidNumberLiteral diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/TernaryLogicAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/TernaryLogicAcceptance.feature new file mode 100644 index 000000000..a384bf66c --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/TernaryLogicAcceptance.feature @@ -0,0 +1,161 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: TernaryLogicAcceptanceTest + + Background: + Given any graph + + Scenario: The inverse of a null is a null + When executing query: + """ + RETURN NOT null AS value + """ + Then the result should be: + | value | + | null | + And no side effects + + Scenario: A literal null IS null + When executing query: + """ + RETURN null IS NULL AS value + """ + Then the result should be: + | value | + | true | + And no side effects + + Scenario: A literal null is not IS NOT null + When executing query: + """ + RETURN null IS NOT NULL AS value + """ + Then the result should be: + | value | + | false | + And no side effects + + Scenario: It is unknown - i.e. null - if a null is equal to a null + When executing query: + """ + RETURN null = null AS value + """ + Then the result should be: + | value | + | null | + And no side effects + + Scenario: It is unknown - i.e. null - if a null is not equal to a null + When executing query: + """ + RETURN null <> null AS value + """ + Then the result should be: + | value | + | null | + And no side effects + + Scenario Outline: Using null in AND + And parameters are: + | par | val | + | lhs | <lhs> | + | rhs | <rhs> | + When executing query: + """ + RETURN $lhs AND $rhs AS result + """ + Then the result should be: + | result | + | <result> | + And no side effects + + Examples: + | lhs | rhs | result | + | null | null | null | + | null | true | null | + | true | null | null | + | null | false | false | + | false | null | false | + + Scenario Outline: Using null in OR + And parameters are: + | par | val | + | lhs | <lhs> | + | rhs | <rhs> | + When executing query: + """ + RETURN $lhs OR $rhs AS result + """ + Then the result should be: + | result | + | <result> | + And no side effects + + Examples: + | lhs | rhs | result | + | null | null | null | + | null | true | true | + | true | null | true | + | null | false | null | + | false | null | null | + + Scenario Outline: Using null in XOR + And parameters are: + | par | val | + | lhs | <lhs> | + | rhs | <rhs> | + When executing query: + """ + RETURN $lhs XOR $rhs AS result + """ + Then the result should be: + | result | + | <result> | + And no side effects + + Examples: + | lhs | rhs | result | + | null | null | null | + | null | true | null | + | true | null | null | + | null | false | null | + | false | null | null | + + Scenario Outline: Using null in IN + And parameters are: + | par | val | + | elt | <elt> | + | coll | <coll> | + When executing query: + """ + RETURN $elt IN $coll AS result + """ + Then the result should be: + | result | + | <result> | + And no side effects + + Examples: + | elt | coll | result | + | null | null | null | + | null | [1, 2, 3] | null | + | null | [1, 2, 3, null] | null | + | null | [] | false | + | 1 | [1, 2, 3, null] | true | + | 1 | [null, 1] | true | + | 5 | [1, 2, 3, null] | null | diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/TriadicSelection.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/TriadicSelection.feature new file mode 100644 index 000000000..acdca96f1 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/TriadicSelection.feature @@ -0,0 +1,327 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: TriadicSelection + + Scenario: Handling triadic friend of a friend + Given the binary-tree-1 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b)-->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b2' | + | 'b3' | + | 'c11' | + | 'c12' | + | 'c21' | + | 'c22' | + And no side effects + + Scenario: Handling triadic friend of a friend that is not a friend + Given the binary-tree-1 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b)-->(c) + OPTIONAL MATCH (a)-[r:KNOWS]->(c) + WITH c WHERE r IS NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b3' | + | 'c11' | + | 'c12' | + | 'c21' | + | 'c22' | + And no side effects + + Scenario: Handling triadic friend of a friend that is not a friend with different relationship type + Given the binary-tree-1 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b)-->(c) + OPTIONAL MATCH (a)-[r:FOLLOWS]->(c) + WITH c WHERE r IS NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b2' | + | 'c11' | + | 'c12' | + | 'c21' | + | 'c22' | + And no side effects + + Scenario: Handling triadic friend of a friend that is not a friend with superset of relationship type + Given the binary-tree-1 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b)-->(c) + OPTIONAL MATCH (a)-[r]->(c) + WITH c WHERE r IS NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'c11' | + | 'c12' | + | 'c21' | + | 'c22' | + And no side effects + + Scenario: Handling triadic friend of a friend that is not a friend with implicit subset of relationship type + Given the binary-tree-1 graph + When executing query: + """ + MATCH (a:A)-->(b)-->(c) + OPTIONAL MATCH (a)-[r:KNOWS]->(c) + WITH c WHERE r IS NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b3' | + | 'b4' | + | 'c11' | + | 'c12' | + | 'c21' | + | 'c22' | + | 'c31' | + | 'c32' | + | 'c41' | + | 'c42' | + And no side effects + + Scenario: Handling triadic friend of a friend that is not a friend with explicit subset of relationship type + Given the binary-tree-1 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS|FOLLOWS]->(b)-->(c) + OPTIONAL MATCH (a)-[r:KNOWS]->(c) + WITH c WHERE r IS NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b3' | + | 'b4' | + | 'c11' | + | 'c12' | + | 'c21' | + | 'c22' | + | 'c31' | + | 'c32' | + | 'c41' | + | 'c42' | + And no side effects + + Scenario: Handling triadic friend of a friend that is not a friend with same labels + Given the binary-tree-2 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b:X)-->(c:X) + OPTIONAL MATCH (a)-[r:KNOWS]->(c) + WITH c WHERE r IS NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b3' | + | 'c11' | + | 'c21' | + And no side effects + + Scenario: Handling triadic friend of a friend that is not a friend with different labels + Given the binary-tree-2 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b:X)-->(c:Y) + OPTIONAL MATCH (a)-[r:KNOWS]->(c) + WITH c WHERE r IS NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'c12' | + | 'c22' | + And no side effects + + Scenario: Handling triadic friend of a friend that is not a friend with implicit subset of labels + Given the binary-tree-2 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b)-->(c:X) + OPTIONAL MATCH (a)-[r:KNOWS]->(c) + WITH c WHERE r IS NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b3' | + | 'c11' | + | 'c21' | + And no side effects + + Scenario: Handling triadic friend of a friend that is not a friend with implicit superset of labels + Given the binary-tree-2 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b:X)-->(c) + OPTIONAL MATCH (a)-[r:KNOWS]->(c) + WITH c WHERE r IS NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b3' | + | 'c11' | + | 'c12' | + | 'c21' | + | 'c22' | + And no side effects + + Scenario: Handling triadic friend of a friend that is a friend + Given the binary-tree-2 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b)-->(c) + OPTIONAL MATCH (a)-[r:KNOWS]->(c) + WITH c WHERE r IS NOT NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b2' | + And no side effects + + Scenario: Handling triadic friend of a friend that is a friend with different relationship type + Given the binary-tree-1 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b)-->(c) + OPTIONAL MATCH (a)-[r:FOLLOWS]->(c) + WITH c WHERE r IS NOT NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b3' | + And no side effects + + Scenario: Handling triadic friend of a friend that is a friend with superset of relationship type + Given the binary-tree-1 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b)-->(c) + OPTIONAL MATCH (a)-[r]->(c) + WITH c WHERE r IS NOT NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b2' | + | 'b3' | + And no side effects + + Scenario: Handling triadic friend of a friend that is a friend with implicit subset of relationship type + Given the binary-tree-1 graph + When executing query: + """ + MATCH (a:A)-->(b)-->(c) + OPTIONAL MATCH (a)-[r:KNOWS]->(c) + WITH c WHERE r IS NOT NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b1' | + | 'b2' | + And no side effects + + Scenario: Handling triadic friend of a friend that is a friend with explicit subset of relationship type + Given the binary-tree-1 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS|FOLLOWS]->(b)-->(c) + OPTIONAL MATCH (a)-[r:KNOWS]->(c) + WITH c WHERE r IS NOT NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b1' | + | 'b2' | + And no side effects + + Scenario: Handling triadic friend of a friend that is a friend with same labels + Given the binary-tree-2 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b:X)-->(c:X) + OPTIONAL MATCH (a)-[r:KNOWS]->(c) + WITH c WHERE r IS NOT NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b2' | + And no side effects + + Scenario: Handling triadic friend of a friend that is a friend with different labels + Given the binary-tree-2 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b:X)-->(c:Y) + OPTIONAL MATCH (a)-[r:KNOWS]->(c) + WITH c WHERE r IS NOT NULL + RETURN c.name + """ + Then the result should be: + | c.name | + And no side effects + + Scenario: Handling triadic friend of a friend that is a friend with implicit subset of labels + Given the binary-tree-2 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b)-->(c:X) + OPTIONAL MATCH (a)-[r:KNOWS]->(c) + WITH c WHERE r IS NOT NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b2' | + And no side effects + + Scenario: Handling triadic friend of a friend that is a friend with implicit superset of labels + Given the binary-tree-2 graph + When executing query: + """ + MATCH (a:A)-[:KNOWS]->(b:X)-->(c) + OPTIONAL MATCH (a)-[r:KNOWS]->(c) + WITH c WHERE r IS NOT NULL + RETURN c.name + """ + Then the result should be: + | c.name | + | 'b2' | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/TypeConversionFunctions.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/TypeConversionFunctions.feature new file mode 100644 index 000000000..303ccd288 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/TypeConversionFunctions.feature @@ -0,0 +1,416 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: TypeConversionFunctions + + Scenario: `toBoolean()` on valid literal string + Given any graph + When executing query: + """ + RETURN toBoolean('true') AS b + """ + Then the result should be: + | b | + | true | + And no side effects + + Scenario: `toBoolean()` on booleans + Given any graph + When executing query: + """ + UNWIND [true, false] AS b + RETURN toBoolean(b) AS b + """ + Then the result should be: + | b | + | true | + | false | + And no side effects + + Scenario: `toBoolean()` on variables with valid string values + Given any graph + When executing query: + """ + UNWIND ['true', 'false'] AS s + RETURN toBoolean(s) AS b + """ + Then the result should be: + | b | + | true | + | false | + And no side effects + + Scenario: `toBoolean()` on invalid strings + Given any graph + When executing query: + """ + UNWIND [null, '', ' tru ', 'f alse'] AS things + RETURN toBoolean(things) AS b + """ + Then the result should be: + | b | + | null | + | null | + | null | + | null | + And no side effects + + Scenario Outline: `toBoolean()` on invalid types + Given any graph + When executing query: + """ + WITH [true, <invalid>] AS list + RETURN toBoolean(list[1]) AS b + """ + Then a TypeError should be raised at runtime: InvalidArgumentValue + + Examples: + | invalid | + | [] | + | {} | + | 1 | + | 1.0 | + + + Scenario: `toInteger()` + Given an empty graph + And having executed: + """ + CREATE (:Person {age: '42'}) + """ + When executing query: + """ + MATCH (p:Person { age: '42' }) + WITH * + MATCH (n) + RETURN toInteger(n.age) AS age + """ + Then the result should be: + | age | + | 42 | + And no side effects + + Scenario: `toInteger()` on float + Given any graph + When executing query: + """ + WITH 82.9 AS weight + RETURN toInteger(weight) + """ + Then the result should be: + | toInteger(weight) | + | 82 | + And no side effects + + Scenario: `toInteger()` returning null on non-numerical string + Given any graph + When executing query: + """ + WITH 'foo' AS foo_string, '' AS empty_string + RETURN toInteger(foo_string) AS foo, toInteger(empty_string) AS empty + """ + Then the result should be: + | foo | empty | + | null | null | + And no side effects + + Scenario: `toInteger()` handling mixed number types + Given any graph + When executing query: + """ + WITH [2, 2.9] AS numbers + RETURN [n IN numbers | toInteger(n)] AS int_numbers + """ + Then the result should be: + | int_numbers | + | [2, 2] | + And no side effects + + Scenario: `toInteger()` handling Any type + Given any graph + When executing query: + """ + WITH [2, 2.9, '1.7'] AS things + RETURN [n IN things | toInteger(n)] AS int_numbers + """ + Then the result should be: + | int_numbers | + | [2, 2, 1] | + And no side effects + + Scenario: `toInteger()` on a list of strings + Given any graph + When executing query: + """ + WITH ['2', '2.9', 'foo'] AS numbers + RETURN [n IN numbers | toInteger(n)] AS int_numbers + """ + Then the result should be: + | int_numbers | + | [2, 2, null] | + And no side effects + + Scenario: `toInteger()` on a complex-typed expression + Given any graph + And parameters are: + | param | 1 | + When executing query: + """ + RETURN toInteger(1 - $param) AS result + """ + Then the result should be: + | result | + | 0 | + And no side effects + + Scenario Outline: `toInteger()` failing on invalid arguments + Given an empty graph + And having executed: + """ + CREATE ()-[:T]->() + """ + When executing query: + """ + MATCH p = (n)-[r:T]->() + RETURN [x IN [1, <invalid>] | toInteger(x) ] AS list + """ + Then a TypeError should be raised at runtime: InvalidArgumentValue + + Examples: + | invalid | + | true | + | [] | + | {} | + | n | + | r | + | p | + + Scenario: `toFloat()` + Given an empty graph + And having executed: + """ + CREATE (:Movie {rating: 4}) + """ + When executing query: + """ + MATCH (m:Movie { rating: 4 }) + WITH * + MATCH (n) + RETURN toFloat(n.rating) AS float + """ + Then the result should be: + | float | + | 4.0 | + And no side effects + + Scenario: `toFloat()` on mixed number types + Given any graph + When executing query: + """ + WITH [3.4, 3] AS numbers + RETURN [n IN numbers | toFloat(n)] AS float_numbers + """ + Then the result should be: + | float_numbers | + | [3.4, 3.0] | + And no side effects + + Scenario: `toFloat()` returning null on non-numerical string + Given any graph + When executing query: + """ + WITH 'foo' AS foo_string, '' AS empty_string + RETURN toFloat(foo_string) AS foo, toFloat(empty_string) AS empty + """ + Then the result should be: + | foo | empty | + | null | null | + And no side effects + + Scenario: `toFloat()` handling Any type + Given any graph + When executing query: + """ + WITH [3.4, 3, '5'] AS numbers + RETURN [n IN numbers | toFloat(n)] AS float_numbers + """ + Then the result should be: + | float_numbers | + | [3.4, 3.0, 5.0] | + And no side effects + + Scenario: `toFloat()` on a list of strings + Given any graph + When executing query: + """ + WITH ['1', '2', 'foo'] AS numbers + RETURN [n IN numbers | toFloat(n)] AS float_numbers + """ + Then the result should be: + | float_numbers | + | [1.0, 2.0, null] | + And no side effects + + Scenario Outline: `toFloat()` failing on invalid arguments + Given an empty graph + And having executed: + """ + CREATE ()-[:T]->() + """ + When executing query: + """ + MATCH p = (n)-[r:T]->() + RETURN [x IN [1.0, <invalid>] | toFloat(x) ] AS list + """ + Then a TypeError should be raised at runtime: InvalidArgumentValue + + Examples: + | invalid | + | true | + | [] | + | {} | + | n | + | r | + | p | + + Scenario: `toString()` + Given an empty graph + And having executed: + """ + CREATE (:Movie {rating: 4}) + """ + When executing query: + """ + MATCH (m:Movie { rating: 4 }) + WITH * + MATCH (n) + RETURN toString(n.rating) + """ + Then the result should be: + | toString(n.rating) | + | '4' | + And no side effects + + Scenario: `toString()` handling boolean properties + Given an empty graph + And having executed: + """ + CREATE (:Movie {watched: true}) + """ + When executing query: + """ + MATCH (m:Movie) + RETURN toString(m.watched) + """ + Then the result should be: + | toString(m.watched) | + | 'true' | + And no side effects + + Scenario: `toString()` handling inlined boolean + Given any graph + When executing query: + """ + RETURN toString(1 < 0) AS bool + """ + Then the result should be: + | bool | + | 'false' | + And no side effects + + Scenario: `toString()` handling boolean literal + Given any graph + When executing query: + """ + RETURN toString(true) AS bool + """ + Then the result should be: + | bool | + | 'true' | + And no side effects + + Scenario: `toString()` should work on Any type + Given any graph + When executing query: + """ + RETURN [x IN [1, 2.3, true, 'apa'] | toString(x) ] AS list + """ + Then the result should be: + | list | + | ['1', '2.3', 'true', 'apa'] | + And no side effects + + Scenario: `toString()` on a list of integers + Given any graph + When executing query: + """ + WITH [1, 2, 3] AS numbers + RETURN [n IN numbers | toString(n)] AS string_numbers + """ + Then the result should be: + | string_numbers | + | ['1', '2', '3'] | + And no side effects + + Scenario Outline: `toString()` failing on invalid arguments + Given an empty graph + And having executed: + """ + CREATE ()-[:T]->() + """ + When executing query: + """ + MATCH p = (n)-[r:T]->() + RETURN [x IN [1, '', <invalid>] | toString(x) ] AS list + """ + Then a TypeError should be raised at runtime: InvalidArgumentValue + + Examples: + | invalid | + | [] | + | {} | + | n | + | r | + | p | + + Scenario: `toString()` should accept potentially correct types 1 + Given any graph + When executing query: + """ + UNWIND ['male', 'female', null] AS gen + RETURN coalesce(toString(gen), 'x') AS result + """ + Then the result should be: + | result | + | 'male' | + | 'female' | + | 'x' | + And no side effects + + Scenario: `toString()` should accept potentially correct types 2 + Given any graph + When executing query: + """ + UNWIND ['male', 'female', null] AS gen + RETURN toString(coalesce(gen, 'x')) AS result + """ + Then the result should be: + | result | + | 'male' | + | 'female' | + | 'x' | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/UnionAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/UnionAcceptance.feature new file mode 100644 index 000000000..505f3dc07 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/UnionAcceptance.feature @@ -0,0 +1,99 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: UnionAcceptance + + Scenario: Should be able to create text output from union queries + Given an empty graph + And having executed: + """ + CREATE (:A), (:B) + """ + When executing query: + """ + MATCH (a:A) + RETURN a AS a + UNION + MATCH (b:B) + RETURN b AS a + """ + Then the result should be: + | a | + | (:A) | + | (:B) | + And no side effects + + Scenario: Two elements, both unique, not distinct + Given an empty graph + When executing query: + """ + RETURN 1 AS x + UNION ALL + RETURN 2 AS x + """ + Then the result should be: + | x | + | 1 | + | 2 | + And no side effects + + Scenario: Two elements, both unique, distinct + Given an empty graph + When executing query: + """ + RETURN 1 AS x + UNION + RETURN 2 AS x + """ + Then the result should be: + | x | + | 1 | + | 2 | + And no side effects + + Scenario: Three elements, two unique, distinct + Given an empty graph + When executing query: + """ + RETURN 2 AS x + UNION + RETURN 1 AS x + UNION + RETURN 2 AS x + """ + Then the result should be: + | x | + | 2 | + | 1 | + And no side effects + + Scenario: Three elements, two unique, not distinct + Given an empty graph + When executing query: + """ + RETURN 2 AS x + UNION ALL + RETURN 1 AS x + UNION ALL + RETURN 2 AS x + """ + Then the result should be: + | x | + | 2 | + | 1 | + | 2 | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/UnwindAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/UnwindAcceptance.feature new file mode 100644 index 000000000..11747a163 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/UnwindAcceptance.feature @@ -0,0 +1,268 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: UnwindAcceptance + + Scenario: Unwinding a list + Given any graph + When executing query: + """ + UNWIND [1, 2, 3] AS x + RETURN x + """ + Then the result should be: + | x | + | 1 | + | 2 | + | 3 | + And no side effects + + Scenario: Unwinding a range + Given any graph + When executing query: + """ + UNWIND range(1, 3) AS x + RETURN x + """ + Then the result should be: + | x | + | 1 | + | 2 | + | 3 | + And no side effects + + Scenario: Unwinding a concatenation of lists + Given any graph + When executing query: + """ + WITH [1, 2, 3] AS first, [4, 5, 6] AS second + UNWIND (first + second) AS x + RETURN x + """ + Then the result should be: + | x | + | 1 | + | 2 | + | 3 | + | 4 | + | 5 | + | 6 | + And no side effects + + Scenario: Unwinding a collected unwound expression + Given any graph + When executing query: + """ + UNWIND RANGE(1, 2) AS row + WITH collect(row) AS rows + UNWIND rows AS x + RETURN x + """ + Then the result should be: + | x | + | 1 | + | 2 | + And no side effects + + Scenario: Unwinding a collected expression + Given an empty graph + And having executed: + """ + CREATE ({id: 1}), ({id: 2}) + """ + When executing query: + """ + MATCH (row) + WITH collect(row) AS rows + UNWIND rows AS node + RETURN node.id + """ + Then the result should be: + | node.id | + | 1 | + | 2 | + And no side effects + + Scenario: Creating nodes from an unwound parameter list + Given an empty graph + And having executed: + """ + CREATE (:Year {year: 2016}) + """ + And parameters are: + | events | [{year: 2016, id: 1}, {year: 2016, id: 2}] | + When executing query: + """ + UNWIND $events AS event + MATCH (y:Year {year: event.year}) + MERGE (e:Event {id: event.id}) + MERGE (y)<-[:IN]-(e) + RETURN e.id AS x + ORDER BY x + """ + Then the result should be, in order: + | x | + | 1 | + | 2 | + And the side effects should be: + | +nodes | 2 | + | +relationships | 2 | + | +labels | 1 | + | +properties | 2 | + + Scenario: Double unwinding a list of lists + Given any graph + When executing query: + """ + WITH [[1, 2, 3], [4, 5, 6]] AS lol + UNWIND lol AS x + UNWIND x AS y + RETURN y + """ + Then the result should be: + | y | + | 1 | + | 2 | + | 3 | + | 4 | + | 5 | + | 6 | + And no side effects + + Scenario: Unwinding the empty list + Given any graph + When executing query: + """ + UNWIND [] AS empty + RETURN empty + """ + Then the result should be: + | empty | + And no side effects + + Scenario: Unwinding null + Given any graph + When executing query: + """ + UNWIND null AS nil + RETURN nil + """ + Then the result should be: + | nil | + And no side effects + + Scenario: Unwinding list with duplicates + Given any graph + When executing query: + """ + UNWIND [1, 1, 2, 2, 3, 3, 4, 4, 5, 5] AS duplicate + RETURN duplicate + """ + Then the result should be: + | duplicate | + | 1 | + | 1 | + | 2 | + | 2 | + | 3 | + | 3 | + | 4 | + | 4 | + | 5 | + | 5 | + And no side effects + + Scenario: Unwind does not prune context + Given any graph + When executing query: + """ + WITH [1, 2, 3] AS list + UNWIND list AS x + RETURN * + """ + Then the result should be: + | list | x | + | [1, 2, 3] | 1 | + | [1, 2, 3] | 2 | + | [1, 2, 3] | 3 | + And no side effects + + Scenario: Unwind does not remove variables from scope + Given an empty graph + And having executed: + """ + CREATE (s:S), + (n), + (e:E), + (s)-[:X]->(e), + (s)-[:Y]->(e), + (n)-[:Y]->(e) + """ + When executing query: + """ + MATCH (a:S)-[:X]->(b1) + WITH a, collect(b1) AS bees + UNWIND bees AS b2 + MATCH (a)-[:Y]->(b2) + RETURN a, b2 + """ + Then the result should be: + | a | b2 | + | (:S) | (:E) | + And no side effects + + Scenario: Multiple unwinds after each other + Given any graph + When executing query: + """ + WITH [1, 2] AS xs, [3, 4] AS ys, [5, 6] AS zs + UNWIND xs AS x + UNWIND ys AS y + UNWIND zs AS z + RETURN * + """ + Then the result should be: + | x | xs | y | ys | z | zs | + | 1 | [1, 2] | 3 | [3, 4] | 5 | [5, 6] | + | 1 | [1, 2] | 3 | [3, 4] | 6 | [5, 6] | + | 1 | [1, 2] | 4 | [3, 4] | 5 | [5, 6] | + | 1 | [1, 2] | 4 | [3, 4] | 6 | [5, 6] | + | 2 | [1, 2] | 3 | [3, 4] | 5 | [5, 6] | + | 2 | [1, 2] | 3 | [3, 4] | 6 | [5, 6] | + | 2 | [1, 2] | 4 | [3, 4] | 5 | [5, 6] | + | 2 | [1, 2] | 4 | [3, 4] | 6 | [5, 6] | + And no side effects + + Scenario: Unwind with merge + Given an empty graph + And parameters are: + | props | [{login: 'login1', name: 'name1'}, {login: 'login2', name: 'name2'}] | + When executing query: + """ + UNWIND $props AS prop + MERGE (p:Person {login: prop.login}) + SET p.name = prop.name + RETURN p.name, p.login + """ + Then the result should be: + | p.name | p.login | + | 'name1' | 'login1' | + | 'name2' | 'login2' | + And the side effects should be: + | +nodes | 2 | + | +labels | 1 | + | +properties | 4 | diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/VarLengthAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/VarLengthAcceptance.feature new file mode 100644 index 000000000..e89b79543 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/VarLengthAcceptance.feature @@ -0,0 +1,657 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: VarLengthAcceptance + + # TODO: Replace this with a named graph (or two) + Background: + Given an empty graph + And having executed: + """ + CREATE (n0:A {name: 'n0'}), + (n00:B {name: 'n00'}), + (n01:B {name: 'n01'}), + (n000:C {name: 'n000'}), + (n001:C {name: 'n001'}), + (n010:C {name: 'n010'}), + (n011:C {name: 'n011'}), + (n0000:D {name: 'n0000'}), + (n0001:D {name: 'n0001'}), + (n0010:D {name: 'n0010'}), + (n0011:D {name: 'n0011'}), + (n0100:D {name: 'n0100'}), + (n0101:D {name: 'n0101'}), + (n0110:D {name: 'n0110'}), + (n0111:D {name: 'n0111'}) + CREATE (n0)-[:LIKES]->(n00), + (n0)-[:LIKES]->(n01), + (n00)-[:LIKES]->(n000), + (n00)-[:LIKES]->(n001), + (n01)-[:LIKES]->(n010), + (n01)-[:LIKES]->(n011), + (n000)-[:LIKES]->(n0000), + (n000)-[:LIKES]->(n0001), + (n001)-[:LIKES]->(n0010), + (n001)-[:LIKES]->(n0011), + (n010)-[:LIKES]->(n0100), + (n010)-[:LIKES]->(n0101), + (n011)-[:LIKES]->(n0110), + (n011)-[:LIKES]->(n0111) + """ + + Scenario: Handling unbounded variable length match + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00' | + | 'n01' | + | 'n000' | + | 'n001' | + | 'n010' | + | 'n011' | + | 'n0000' | + | 'n0001' | + | 'n0010' | + | 'n0011' | + | 'n0100' | + | 'n0101' | + | 'n0110' | + | 'n0111' | + And no side effects + + Scenario: Handling explicitly unbounded variable length match + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*..]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00' | + | 'n01' | + | 'n000' | + | 'n001' | + | 'n010' | + | 'n011' | + | 'n0000' | + | 'n0001' | + | 'n0010' | + | 'n0011' | + | 'n0100' | + | 'n0101' | + | 'n0110' | + | 'n0111' | + And no side effects + + Scenario: Fail when asterisk operator is missing + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES..]->(c) + RETURN c.name + """ + Then a SyntaxError should be raised at compile time: InvalidRelationshipPattern + + Scenario: Handling single bounded variable length match 1 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*0]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n0' | + And no side effects + + Scenario: Handling single bounded variable length match 2 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*1]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00' | + | 'n01' | + And no side effects + + Scenario: Handling single bounded variable length match 3 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*2]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n000' | + | 'n001' | + | 'n010' | + | 'n011' | + And no side effects + + Scenario: Handling upper and lower bounded variable length match 1 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*0..2]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n0' | + | 'n00' | + | 'n01' | + | 'n000' | + | 'n001' | + | 'n010' | + | 'n011' | + And no side effects + + Scenario: Handling upper and lower bounded variable length match 2 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*1..2]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00' | + | 'n01' | + | 'n000' | + | 'n001' | + | 'n010' | + | 'n011' | + And no side effects + + Scenario: Handling symmetrically bounded variable length match, bounds are zero + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*0..0]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n0' | + And no side effects + + Scenario: Handling symmetrically bounded variable length match, bounds are one + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*1..1]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00' | + | 'n01' | + And no side effects + + Scenario: Handling symmetrically bounded variable length match, bounds are two + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*2..2]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n000' | + | 'n001' | + | 'n010' | + | 'n011' | + And no side effects + + Scenario: Fail on negative bound + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*-2]->(c) + RETURN c.name + """ + Then a SyntaxError should be raised at compile time: InvalidRelationshipPattern + + Scenario: Handling upper and lower bounded variable length match, empty interval 1 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*2..1]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + And no side effects + + Scenario: Handling upper and lower bounded variable length match, empty interval 2 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*1..0]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + And no side effects + + Scenario: Handling upper bounded variable length match, empty interval + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*..0]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + And no side effects + + Scenario: Handling upper bounded variable length match 1 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*..1]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00' | + | 'n01' | + And no side effects + + Scenario: Handling upper bounded variable length match 2 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*..2]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00' | + | 'n01' | + | 'n000' | + | 'n001' | + | 'n010' | + | 'n011' | + And no side effects + + Scenario: Handling lower bounded variable length match 1 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*0..]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n0' | + | 'n00' | + | 'n01' | + | 'n000' | + | 'n001' | + | 'n010' | + | 'n011' | + | 'n0000' | + | 'n0001' | + | 'n0010' | + | 'n0011' | + | 'n0100' | + | 'n0101' | + | 'n0110' | + | 'n0111' | + And no side effects + + Scenario: Handling lower bounded variable length match 2 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*1..]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00' | + | 'n01' | + | 'n000' | + | 'n001' | + | 'n010' | + | 'n011' | + | 'n0000' | + | 'n0001' | + | 'n0010' | + | 'n0011' | + | 'n0100' | + | 'n0101' | + | 'n0110' | + | 'n0111' | + And no side effects + + Scenario: Handling lower bounded variable length match 3 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*2..]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n000' | + | 'n001' | + | 'n010' | + | 'n011' | + | 'n0000' | + | 'n0001' | + | 'n0010' | + | 'n0011' | + | 'n0100' | + | 'n0101' | + | 'n0110' | + | 'n0111' | + And no side effects + + Scenario: Handling a variable length relationship and a standard relationship in chain, zero length 1 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*0]->()-[:LIKES]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00' | + | 'n01' | + And no side effects + + Scenario: Handling a variable length relationship and a standard relationship in chain, zero length 2 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES]->()-[:LIKES*0]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00' | + | 'n01' | + And no side effects + + Scenario: Handling a variable length relationship and a standard relationship in chain, single length 1 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*1]->()-[:LIKES]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n000' | + | 'n001' | + | 'n010' | + | 'n011' | + And no side effects + + Scenario: Handling a variable length relationship and a standard relationship in chain, single length 2 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES]->()-[:LIKES*1]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n000' | + | 'n001' | + | 'n010' | + | 'n011' | + And no side effects + + Scenario: Handling a variable length relationship and a standard relationship in chain, longer 1 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES*2]->()-[:LIKES]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n0000' | + | 'n0001' | + | 'n0010' | + | 'n0011' | + | 'n0100' | + | 'n0101' | + | 'n0110' | + | 'n0111' | + And no side effects + + Scenario: Handling a variable length relationship and a standard relationship in chain, longer 2 + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES]->()-[:LIKES*2]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n0000' | + | 'n0001' | + | 'n0010' | + | 'n0011' | + | 'n0100' | + | 'n0101' | + | 'n0110' | + | 'n0111' | + And no side effects + + Scenario: Handling a variable length relationship and a standard relationship in chain, longer 3 + And having executed: + """ + MATCH (d:D) + CREATE (e1:E {name: d.name + '0'}), + (e2:E {name: d.name + '1'}) + CREATE (d)-[:LIKES]->(e1), + (d)-[:LIKES]->(e2) + """ + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES]->()-[:LIKES*3]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00000' | + | 'n00001' | + | 'n00010' | + | 'n00011' | + | 'n00100' | + | 'n00101' | + | 'n00110' | + | 'n00111' | + | 'n01000' | + | 'n01001' | + | 'n01010' | + | 'n01011' | + | 'n01100' | + | 'n01101' | + | 'n01110' | + | 'n01111' | + And no side effects + + Scenario: Handling mixed relationship patterns and directions 1 + And having executed: + """ + MATCH (a:A)-[r]->(b) + DELETE r + CREATE (b)-[:LIKES]->(a) + """ + And having executed: + """ + MATCH (d:D) + CREATE (e1:E {name: d.name + '0'}), + (e2:E {name: d.name + '1'}) + CREATE (d)-[:LIKES]->(e1), + (d)-[:LIKES]->(e2) + """ + When executing query: + """ + MATCH (a:A) + MATCH (a)<-[:LIKES]-()-[:LIKES*3]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00000' | + | 'n00001' | + | 'n00010' | + | 'n00011' | + | 'n00100' | + | 'n00101' | + | 'n00110' | + | 'n00111' | + | 'n01000' | + | 'n01001' | + | 'n01010' | + | 'n01011' | + | 'n01100' | + | 'n01101' | + | 'n01110' | + | 'n01111' | + And no side effects + + Scenario: Handling mixed relationship patterns and directions 2 + # This gets hard to follow for a human mind. The answer is named graphs, but it's not crucial to fix. + And having executed: + """ + MATCH (a)-[r]->(b) + WHERE NOT a:A + DELETE r + CREATE (b)-[:LIKES]->(a) + """ + And having executed: + """ + MATCH (d:D) + CREATE (e1:E {name: d.name + '0'}), + (e2:E {name: d.name + '1'}) + CREATE (d)-[:LIKES]->(e1), + (d)-[:LIKES]->(e2) + """ + When executing query: + """ + MATCH (a:A) + MATCH (a)-[:LIKES]->()<-[:LIKES*3]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00000' | + | 'n00001' | + | 'n00010' | + | 'n00011' | + | 'n00100' | + | 'n00101' | + | 'n00110' | + | 'n00111' | + | 'n01000' | + | 'n01001' | + | 'n01010' | + | 'n01011' | + | 'n01100' | + | 'n01101' | + | 'n01110' | + | 'n01111' | + And no side effects + + Scenario: Handling mixed relationship patterns 1 + And having executed: + """ + MATCH (d:D) + CREATE (e1:E {name: d.name + '0'}), + (e2:E {name: d.name + '1'}) + CREATE (d)-[:LIKES]->(e1), + (d)-[:LIKES]->(e2) + """ + When executing query: + """ + MATCH (a:A) + MATCH (p)-[:LIKES*1]->()-[:LIKES]->()-[r:LIKES*2]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00000' | + | 'n00001' | + | 'n00010' | + | 'n00011' | + | 'n00100' | + | 'n00101' | + | 'n00110' | + | 'n00111' | + | 'n01000' | + | 'n01001' | + | 'n01010' | + | 'n01011' | + | 'n01100' | + | 'n01101' | + | 'n01110' | + | 'n01111' | + And no side effects + + Scenario: Handling mixed relationship patterns 2 + And having executed: + """ + MATCH (d:D) + CREATE (e1:E {name: d.name + '0'}), + (e2:E {name: d.name + '1'}) + CREATE (d)-[:LIKES]->(e1), + (d)-[:LIKES]->(e2) + """ + When executing query: + """ + MATCH (a:A) + MATCH (p)-[:LIKES]->()-[:LIKES*2]->()-[r:LIKES]->(c) + RETURN c.name + """ + Then the result should be: + | c.name | + | 'n00000' | + | 'n00001' | + | 'n00010' | + | 'n00011' | + | 'n00100' | + | 'n00101' | + | 'n00110' | + | 'n00111' | + | 'n01000' | + | 'n01001' | + | 'n01010' | + | 'n01011' | + | 'n01100' | + | 'n01101' | + | 'n01110' | + | 'n01111' | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/VarLengthAcceptance2.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/VarLengthAcceptance2.feature new file mode 100644 index 000000000..f03acf62c --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/VarLengthAcceptance2.feature @@ -0,0 +1,41 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: VarLengthAcceptance2 + + Scenario: Handling relationships that are already bound in variable length paths + Given an empty graph + And having executed: + """ + CREATE (n0:Node), + (n1:Node), + (n2:Node), + (n3:Node), + (n0)-[:EDGE]->(n1), + (n1)-[:EDGE]->(n2), + (n2)-[:EDGE]->(n3) + """ + When executing query: + """ + MATCH ()-[r:EDGE]-() + MATCH p = (n)-[*0..1]-()-[r]-()-[*0..1]-(m) + RETURN count(p) AS c + """ + Then the result should be: + | c | + | 32 | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/WhereAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/WhereAcceptance.feature new file mode 100644 index 000000000..d25651086 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/WhereAcceptance.feature @@ -0,0 +1,35 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: WhereAcceptance + + Scenario: NOT and false + Given an empty graph + And having executed: + """ + CREATE ({name: 'a'}) + """ + When executing query: + """ + MATCH (n) + WHERE NOT(n.name = 'apa' AND false) + RETURN n + """ + Then the result should be: + | n | + | ({name: 'a'}) | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/features/WithAcceptance.feature b/tests/qa/tck_engine/tests/openCypher_M09/features/WithAcceptance.feature new file mode 100644 index 000000000..f1d3f8c7f --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/features/WithAcceptance.feature @@ -0,0 +1,363 @@ +# +# Copyright (c) 2015-2018 "Neo Technology," +# Network Engine for Objects in Lund AB [http://neotechnology.com] +# +# 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. +# + +Feature: WithAcceptance + + Scenario: Passing on pattern nodes + Given an empty graph + And having executed: + """ + CREATE (:A)-[:REL]->(:B) + """ + When executing query: + """ + MATCH (a:A) + WITH a + MATCH (a)-->(b) + RETURN * + """ + Then the result should be: + | a | b | + | (:A) | (:B) | + And no side effects + + Scenario: ORDER BY and LIMIT can be used + Given an empty graph + And having executed: + """ + CREATE (a:A), (), (), (), + (a)-[:REL]->() + """ + When executing query: + """ + MATCH (a:A) + WITH a + ORDER BY a.name + LIMIT 1 + MATCH (a)-->(b) + RETURN a + """ + Then the result should be: + | a | + | (:A) | + And no side effects + + Scenario: No dependencies between the query parts + Given an empty graph + And having executed: + """ + CREATE (:A), (:B) + """ + When executing query: + """ + MATCH (a) + WITH a + MATCH (b) + RETURN a, b + """ + Then the result should be: + | a | b | + | (:A) | (:A) | + | (:A) | (:B) | + | (:B) | (:A) | + | (:B) | (:B) | + And no side effects + + Scenario: Aliasing + Given an empty graph + And having executed: + """ + CREATE (:Begin {prop: 42}), + (:End {prop: 42}), + (:End {prop: 3}) + """ + When executing query: + """ + MATCH (a:Begin) + WITH a.prop AS property + MATCH (b:End) + WHERE property = b.prop + RETURN b + """ + Then the result should be: + | b | + | (:End {prop: 42}) | + And no side effects + + Scenario: Handle dependencies across WITH + Given an empty graph + And having executed: + """ + CREATE (a:End {prop: 42, id: 0}), + (:End {prop: 3}), + (:Begin {prop: a.id}) + """ + When executing query: + """ + MATCH (a:Begin) + WITH a.prop AS property + LIMIT 1 + MATCH (b) + WHERE b.id = property + RETURN b + """ + Then the result should be: + | b | + | (:End {prop: 42, id: 0}) | + And no side effects + + Scenario: Handle dependencies across WITH with SKIP + Given an empty graph + And having executed: + """ + CREATE (a {prop: 'A', key: 0, id: 0}), + ({prop: 'B', key: a.id, id: 1}), + ({prop: 'C', key: 0, id: 2}) + """ + When executing query: + """ + MATCH (a) + WITH a.prop AS property, a.key AS idToUse + ORDER BY property + SKIP 1 + MATCH (b) + WHERE b.id = idToUse + RETURN DISTINCT b + """ + Then the result should be: + | b | + | ({prop: 'A', key: 0, id: 0}) | + And no side effects + + Scenario: WHERE after WITH should filter results + Given an empty graph + And having executed: + """ + CREATE ({name: 'A'}), + ({name: 'B'}), + ({name: 'C'}) + """ + When executing query: + """ + MATCH (a) + WITH a + WHERE a.name = 'B' + RETURN a + """ + Then the result should be: + | a | + | ({name: 'B'}) | + And no side effects + + Scenario: WHERE after WITH can filter on top of an aggregation + Given an empty graph + And having executed: + """ + CREATE (a {name: 'A'}), + (b {name: 'B'}) + CREATE (a)-[:REL]->(), + (a)-[:REL]->(), + (a)-[:REL]->(), + (b)-[:REL]->() + """ + When executing query: + """ + MATCH (a)-->() + WITH a, count(*) AS relCount + WHERE relCount > 1 + RETURN a + """ + Then the result should be: + | a | + | ({name: 'A'}) | + And no side effects + + Scenario: ORDER BY on an aggregating key + Given an empty graph + And having executed: + """ + CREATE ({bar: 'A'}), + ({bar: 'A'}), + ({bar: 'B'}) + """ + When executing query: + """ + MATCH (a) + WITH a.bar AS bars, count(*) AS relCount + ORDER BY a.bar + RETURN * + """ + Then the result should be: + | bars | relCount | + | 'A' | 2 | + | 'B' | 1 | + And no side effects + + Scenario: ORDER BY a DISTINCT column + Given an empty graph + And having executed: + """ + CREATE ({bar: 'A'}), + ({bar: 'A'}), + ({bar: 'B'}) + """ + When executing query: + """ + MATCH (a) + WITH DISTINCT a.bar AS bars + ORDER BY a.bar + RETURN * + """ + Then the result should be: + | bars | + | 'A' | + | 'B' | + And no side effects + + Scenario: WHERE on a DISTINCT column + Given an empty graph + And having executed: + """ + CREATE ({bar: 'A'}), + ({bar: 'A'}), + ({bar: 'B'}) + """ + When executing query: + """ + MATCH (a) + WITH DISTINCT a.bar AS bars + WHERE a.bar = 'B' + RETURN * + """ + Then the result should be: + | bars | + | 'B' | + And no side effects + + Scenario: A simple pattern with one bound endpoint + Given an empty graph + And having executed: + """ + CREATE (:A)-[:REL]->(:B) + """ + When executing query: + """ + MATCH (a:A)-[r:REL]->(b:B) + WITH a AS b, b AS tmp, r AS r + WITH b AS a, r + LIMIT 1 + MATCH (a)-[r]->(b) + RETURN a, r, b + """ + Then the result should be: + | a | r | b | + | (:A) | [:REL] | (:B) | + And no side effects + + Scenario: Null handling + Given an empty graph + When executing query: + """ + OPTIONAL MATCH (a:Start) + WITH a + MATCH (a)-->(b) + RETURN * + """ + Then the result should be: + | a | b | + And no side effects + + Scenario: Nested maps + Given an empty graph + When executing query: + """ + WITH {foo: {bar: 'baz'}} AS nestedMap + RETURN nestedMap.foo.bar + """ + Then the result should be: + | nestedMap.foo.bar | + | 'baz' | + And no side effects + + Scenario: Connected components succeeding WITH + Given an empty graph + And having executed: + """ + CREATE (:A)-[:REL]->(:X) + CREATE (:B) + """ + When executing query: + """ + MATCH (n:A) + WITH n + LIMIT 1 + MATCH (m:B), (n)-->(x:X) + RETURN * + """ + Then the result should be: + | m | n | x | + | (:B) | (:A) | (:X) | + And no side effects + + Scenario: Single WITH using a predicate and aggregation + Given an empty graph + And having executed: + """ + CREATE ({prop: 43}), ({prop: 42}) + """ + When executing query: + """ + MATCH (n) + WITH n + WHERE n.prop = 42 + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 1 | + And no side effects + + Scenario: Multiple WITHs using a predicate and aggregation + Given an empty graph + And having executed: + """ + CREATE (a {name: 'David'}), + (b {name: 'Other'}), + (c {name: 'NotOther'}), + (d {name: 'NotOther2'}), + (a)-[:REL]->(b), + (a)-[:REL]->(c), + (a)-[:REL]->(d), + (b)-[:REL]->(), + (b)-[:REL]->(), + (c)-[:REL]->(), + (c)-[:REL]->(), + (d)-[:REL]->() + """ + When executing query: + """ + MATCH (david {name: 'David'})--(otherPerson)-->() + WITH otherPerson, count(*) AS foaf + WHERE foaf > 1 + WITH otherPerson + WHERE otherPerson.name <> 'NotOther' + RETURN count(*) + """ + Then the result should be: + | count(*) | + | 1 | + And no side effects diff --git a/tests/qa/tck_engine/tests/openCypher_M09/graphs/binary-tree-1/binary-tree-1.cypher b/tests/qa/tck_engine/tests/openCypher_M09/graphs/binary-tree-1/binary-tree-1.cypher new file mode 100644 index 000000000..cd901b34c --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/graphs/binary-tree-1/binary-tree-1.cypher @@ -0,0 +1,29 @@ +CREATE (a:A {name: 'a'}), + (b1:X {name: 'b1'}), + (b2:X {name: 'b2'}), + (b3:X {name: 'b3'}), + (b4:X {name: 'b4'}), + (c11:X {name: 'c11'}), + (c12:X {name: 'c12'}), + (c21:X {name: 'c21'}), + (c22:X {name: 'c22'}), + (c31:X {name: 'c31'}), + (c32:X {name: 'c32'}), + (c41:X {name: 'c41'}), + (c42:X {name: 'c42'}) +CREATE (a)-[:KNOWS]->(b1), + (a)-[:KNOWS]->(b2), + (a)-[:FOLLOWS]->(b3), + (a)-[:FOLLOWS]->(b4) +CREATE (b1)-[:FRIEND]->(c11), + (b1)-[:FRIEND]->(c12), + (b2)-[:FRIEND]->(c21), + (b2)-[:FRIEND]->(c22), + (b3)-[:FRIEND]->(c31), + (b3)-[:FRIEND]->(c32), + (b4)-[:FRIEND]->(c41), + (b4)-[:FRIEND]->(c42) +CREATE (b1)-[:FRIEND]->(b2), + (b2)-[:FRIEND]->(b3), + (b3)-[:FRIEND]->(b4), + (b4)-[:FRIEND]->(b1); diff --git a/tests/qa/tck_engine/tests/openCypher_M09/graphs/binary-tree-2/binary-tree-2.cypher b/tests/qa/tck_engine/tests/openCypher_M09/graphs/binary-tree-2/binary-tree-2.cypher new file mode 100644 index 000000000..09462a35a --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/graphs/binary-tree-2/binary-tree-2.cypher @@ -0,0 +1,29 @@ +CREATE (a:A {name: 'a'}), + (b1:X {name: 'b1'}), + (b2:X {name: 'b2'}), + (b3:X {name: 'b3'}), + (b4:X {name: 'b4'}), + (c11:X {name: 'c11'}), + (c12:Y {name: 'c12'}), + (c21:X {name: 'c21'}), + (c22:Y {name: 'c22'}), + (c31:X {name: 'c31'}), + (c32:Y {name: 'c32'}), + (c41:X {name: 'c41'}), + (c42:Y {name: 'c42'}) +CREATE (a)-[:KNOWS]->(b1), + (a)-[:KNOWS]->(b2), + (a)-[:FOLLOWS]->(b3), + (a)-[:FOLLOWS]->(b4) +CREATE (b1)-[:FRIEND]->(c11), + (b1)-[:FRIEND]->(c12), + (b2)-[:FRIEND]->(c21), + (b2)-[:FRIEND]->(c22), + (b3)-[:FRIEND]->(c31), + (b3)-[:FRIEND]->(c32), + (b4)-[:FRIEND]->(c41), + (b4)-[:FRIEND]->(c42) +CREATE (b1)-[:FRIEND]->(b2), + (b2)-[:FRIEND]->(b3), + (b3)-[:FRIEND]->(b4), + (b4)-[:FRIEND]->(b1); diff --git a/tests/qa/tck_engine/tests/openCypher_M09/graphs/yago/openCypher-yago-graph.cypher b/tests/qa/tck_engine/tests/openCypher_M09/graphs/yago/openCypher-yago-graph.cypher new file mode 100644 index 000000000..30d409f18 --- /dev/null +++ b/tests/qa/tck_engine/tests/openCypher_M09/graphs/yago/openCypher-yago-graph.cypher @@ -0,0 +1,64 @@ +/* +This graph is based upon YAGO, which is derived from Wikipedia. +The idea is to enlarge it over time. +http://www.mpi-inf.mpg.de/departments/databases-and-information-systems/research/yago-naga/yago/ +*/ + +CREATE (rachel:Person:Actor {name: 'Rachel Kempson', birthyear: 1910}) +CREATE (michael:Person:Actor {name: 'Michael Redgrave', birthyear: 1908}) +CREATE (vanessa:Person:Actor {name: 'Vanessa Redgrave', birthyear: 1937}) +CREATE (corin:Person:Actor {name: 'Corin Redgrave', birthyear: 1939}) +CREATE (liam:Person:Actor {name: 'Liam Neeson', birthyear: 1952}) +CREATE (natasha:Person:Actor {name: 'Natasha Richardson', birthyear: 1963}) +CREATE (richard:Person:Actor {name: 'Richard Harris', birthyear: 1930}) +CREATE (dennis:Person:Actor {name: 'Dennis Quaid', birthyear: 1954}) +CREATE (lindsay:Person:Actor {name: 'Lindsay Lohan', birthyear: 1986}) +CREATE (jemma:Person:Actor {name: 'Jemma Redgrave', birthyear: 1965}) +CREATE (roy:Person:Actor {name: 'Roy Redgrave', birthyear: 1873}) + +CREATE (john:Person {name: 'John Williams', birthyear: 1932}) +CREATE (christopher:Person {name: 'Christopher Nolan', birthyear: 1970}) + +CREATE (newyork:City {name: 'New York'}) +CREATE (london:City {name: 'London'}) +CREATE (houston:City {name: 'Houston'}) + +CREATE (mrchips:Film {title: 'Goodbye, Mr. Chips'}) +CREATE (batmanbegins:Film {title: 'Batman Begins'}) +CREATE (harrypotter:Film {title: 'Harry Potter and the Sorcerer\'s Stone'}) +CREATE (parent:Film {title: 'The Parent Trap'}) +CREATE (camelot:Film {title: 'Camelot'}) + +CREATE (rachel)-[:HAS_CHILD]->(vanessa), + (rachel)-[:HAS_CHILD]->(corin), + (michael)-[:HAS_CHILD]->(vanessa), + (michael)-[:HAS_CHILD]->(corin), + (corin)-[:HAS_CHILD]->(jemma), + (vanessa)-[:HAS_CHILD]->(natasha), + (roy)-[:HAS_CHILD]->(michael), + + (rachel)-[:MARRIED]->(michael), + (michael)-[:MARRIED]->(rachel), + (natasha)-[:MARRIED]->(liam), + (liam)-[:MARRIED]->(natasha), + + (vanessa)-[:BORN_IN]->(london), + (natasha)-[:BORN_IN]->(london), + (christopher)-[:BORN_IN]->(london), + (dennis)-[:BORN_IN]->(houston), + (lindsay)-[:BORN_IN]->(newyork), + (john)-[:BORN_IN]->(newyork), + + (christopher)-[:DIRECTED]->(batmanbegins), + + (john)-[:WROTE_MUSIC_FOR]->(harrypotter), + (john)-[:WROTE_MUSIC_FOR]->(mrchips), + + (michael)-[:ACTED_IN {charactername: 'The Headmaster'}]->(mrchips), + (vanessa)-[:ACTED_IN {charactername: 'Guenevere'}]->(camelot), + (richard)-[:ACTED_IN {charactername: 'King Arthur'}]->(camelot), + (richard)-[:ACTED_IN {charactername: 'Albus Dumbledore'}]->(harrypotter), + (natasha)-[:ACTED_IN {charactername: 'Liz James'}]->(parent), + (dennis)-[:ACTED_IN {charactername: 'Nick Parker'}]->(parent), + (lindsay)-[:ACTED_IN {charactername: 'Halle/Annie'}]->(parent), + (liam)-[:ACTED_IN {charactername: 'Henri Ducard'}]->(batmanbegins)