be9ed7e879
* Rename mgp_graph_remove to mgp_graph_delete * Add mgp_graph_detach_delete * Add PyGraph functions * Add _mgp exceptions * Use unified error handling in python wrapper * Ignore clang-tidy warnings * Add mgp.Graph, mgp.Vertex and mgp.Edge mutable functions * Add python write procedure registration * Add `is_write` result field to mg.procedures * Use storage::View::NEW for write procedures * Add simple tests for write procedures * Remove false information about IDs
183 lines
7.0 KiB
Python
183 lines
7.0 KiB
Python
import typing
|
|
import mgclient
|
|
import sys
|
|
import pytest
|
|
from common import (execute_and_fetch_all,
|
|
has_one_result_row, has_n_result_row)
|
|
|
|
|
|
def test_is_write(connection):
|
|
is_write = 2
|
|
result_order = "name, signature, is_write"
|
|
cursor = connection.cursor()
|
|
for proc in execute_and_fetch_all(
|
|
cursor, "CALL mg.procedures() YIELD * WITH name, signature, "
|
|
"is_write WHERE name STARTS WITH 'write' "
|
|
f"RETURN {result_order}"):
|
|
assert proc[is_write] is True
|
|
|
|
for proc in execute_and_fetch_all(
|
|
cursor, "CALL mg.procedures() YIELD * WITH name, signature, "
|
|
"is_write WHERE NOT name STARTS WITH 'write' "
|
|
f"RETURN {result_order}"):
|
|
assert proc[is_write] is False
|
|
|
|
assert cursor.description[0].name == "name"
|
|
assert cursor.description[1].name == "signature"
|
|
assert cursor.description[2].name == "is_write"
|
|
|
|
|
|
def test_single_vertex(connection):
|
|
cursor = connection.cursor()
|
|
assert has_n_result_row(cursor, "MATCH (n) RETURN n", 0)
|
|
result = execute_and_fetch_all(
|
|
cursor, "CALL write.create_vertex() YIELD v RETURN v")
|
|
vertex = result[0][0]
|
|
assert isinstance(vertex, mgclient.Node)
|
|
assert has_one_result_row(cursor, "MATCH (n) RETURN n")
|
|
assert vertex.labels == set()
|
|
assert vertex.properties == {}
|
|
|
|
def add_label(label: str):
|
|
execute_and_fetch_all(
|
|
cursor, f"MATCH (n) CALL write.add_label(n, '{label}') "
|
|
"YIELD * RETURN *")
|
|
|
|
def remove_label(label: str):
|
|
execute_and_fetch_all(
|
|
cursor, f"MATCH (n) CALL write.remove_label(n, '{label}') "
|
|
"YIELD * RETURN *")
|
|
|
|
def get_vertex() -> mgclient.Node:
|
|
return execute_and_fetch_all(cursor, "MATCH (n) RETURN n")[0][0]
|
|
|
|
def set_property(property_name: str, property: typing.Any):
|
|
nonlocal cursor
|
|
execute_and_fetch_all(
|
|
cursor, f"MATCH (n) CALL write.set_property(n, '{property_name}', "
|
|
"$property) YIELD * RETURN *", {"property": property})
|
|
|
|
label_1 = "LABEL1"
|
|
label_2 = "LABEL2"
|
|
|
|
add_label(label_1)
|
|
assert get_vertex().labels == {label_1}
|
|
add_label(label_1)
|
|
assert get_vertex().labels == {label_1}
|
|
add_label(label_2)
|
|
assert get_vertex().labels == {label_1, label_2}
|
|
remove_label(label_1)
|
|
assert get_vertex().labels == {label_2}
|
|
property_name = "prop"
|
|
property_value_1 = 1
|
|
property_value_2 = [42, 24]
|
|
set_property(property_name, property_value_1)
|
|
assert get_vertex().properties == {property_name: property_value_1}
|
|
set_property(property_name, property_value_2)
|
|
assert get_vertex().properties == {property_name: property_value_2}
|
|
set_property(property_name, None)
|
|
assert get_vertex().properties == {}
|
|
|
|
execute_and_fetch_all(
|
|
cursor, "MATCH (n) CALL write.delete_vertex(n) YIELD * RETURN 1")
|
|
assert has_n_result_row(cursor, "MATCH (n) RETURN n", 0)
|
|
|
|
|
|
def test_single_edge(connection):
|
|
cursor = connection.cursor()
|
|
assert has_n_result_row(cursor, "MATCH (n) RETURN n", 0)
|
|
v1_id = execute_and_fetch_all(
|
|
cursor, "CALL write.create_vertex() YIELD v RETURN v")[0][0].id
|
|
v2_id = execute_and_fetch_all(
|
|
cursor, "CALL write.create_vertex() YIELD v RETURN v")[0][0].id
|
|
edge_type = "EDGE"
|
|
edge = execute_and_fetch_all(
|
|
cursor, f"MATCH (n) WHERE id(n) = {v1_id} "
|
|
f"MATCH (m) WHERE id(m) = {v2_id} "
|
|
f"CALL write.create_edge(n, m, '{edge_type}') "
|
|
"YIELD e RETURN e")[0][0]
|
|
|
|
assert edge.type == edge_type
|
|
assert edge.properties == {}
|
|
property_name = "very_looong_prooooperty_naaame"
|
|
property_value_1 = {"a": 1, "b": 3.4, "c": [666]}
|
|
property_value_2 = 64
|
|
|
|
def get_edge() -> mgclient.Node:
|
|
return execute_and_fetch_all(cursor, "MATCH ()-[e]->() RETURN e")[0][0]
|
|
|
|
def set_property(property_name: str, property: typing.Any):
|
|
nonlocal cursor
|
|
execute_and_fetch_all(
|
|
cursor, "MATCH ()-[e]->() "
|
|
f"CALL write.set_property(e, '{property_name}', "
|
|
"$property) YIELD * RETURN *", {"property": property})
|
|
|
|
set_property(property_name, property_value_1)
|
|
assert get_edge().properties == {property_name: property_value_1}
|
|
set_property(property_name, property_value_2)
|
|
assert get_edge().properties == {property_name: property_value_2}
|
|
set_property(property_name, None)
|
|
assert get_edge().properties == {}
|
|
execute_and_fetch_all(
|
|
cursor, "MATCH ()-[e]->() CALL write.delete_edge(e) YIELD * RETURN 1")
|
|
assert has_n_result_row(cursor, "MATCH ()-[e]->() RETURN e", 0)
|
|
|
|
|
|
def test_detach_delete_vertex(connection):
|
|
cursor = connection.cursor()
|
|
assert has_n_result_row(cursor, "MATCH (n) RETURN n", 0)
|
|
v1_id = execute_and_fetch_all(
|
|
cursor, "CALL write.create_vertex() YIELD v RETURN v")[0][0].id
|
|
v2_id = execute_and_fetch_all(
|
|
cursor, "CALL write.create_vertex() YIELD v RETURN v")[0][0].id
|
|
execute_and_fetch_all(
|
|
cursor, f"MATCH (n) WHERE id(n) = {v1_id} "
|
|
f"MATCH (m) WHERE id(m) = {v2_id} "
|
|
f"CALL write.create_edge(n, m, 'EDGE') "
|
|
"YIELD e RETURN e")
|
|
|
|
assert has_one_result_row(cursor, "MATCH (n)-[e]->(m) RETURN n, e, m")
|
|
execute_and_fetch_all(
|
|
cursor, f"MATCH (n) WHERE id(n) = {v1_id} "
|
|
"CALL write.detach_delete_vertex(n) YIELD * RETURN 1")
|
|
assert has_n_result_row(cursor, "MATCH (n)-[e]->(m) RETURN n, e, m", 0)
|
|
assert has_n_result_row(cursor, "MATCH ()-[e]->() RETURN e", 0)
|
|
assert has_one_result_row(
|
|
cursor, f"MATCH (n) WHERE id(n) = {v2_id} RETURN n")
|
|
|
|
|
|
def test_graph_mutability(connection):
|
|
cursor = connection.cursor()
|
|
assert has_n_result_row(cursor, "MATCH (n) RETURN n", 0)
|
|
v1_id = execute_and_fetch_all(
|
|
cursor, "CALL write.create_vertex() YIELD v RETURN v")[0][0].id
|
|
v2_id = execute_and_fetch_all(
|
|
cursor, "CALL write.create_vertex() YIELD v RETURN v")[0][0].id
|
|
execute_and_fetch_all(
|
|
cursor, f"MATCH (n) WHERE id(n) = {v1_id} "
|
|
f"MATCH (m) WHERE id(m) = {v2_id} "
|
|
f"CALL write.create_edge(n, m, 'EDGE') "
|
|
"YIELD e RETURN e")
|
|
|
|
def test_mutability(is_write: bool):
|
|
module = "write" if is_write else "read"
|
|
assert execute_and_fetch_all(
|
|
cursor, f"CALL {module}.graph_is_mutable() "
|
|
"YIELD mutable RETURN mutable")[0][0] is is_write
|
|
assert execute_and_fetch_all(
|
|
cursor, "MATCH (n) "
|
|
f"CALL {module}.underlying_graph_is_mutable(n) "
|
|
"YIELD mutable RETURN mutable")[0][0] is is_write
|
|
assert execute_and_fetch_all(
|
|
cursor, "MATCH (n)-[e]->(m) "
|
|
f"CALL {module}.underlying_graph_is_mutable(e) "
|
|
"YIELD mutable RETURN mutable")[0][0] is is_write
|
|
|
|
test_mutability(True)
|
|
test_mutability(False)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
sys.exit(pytest.main([__file__, "-rA"]))
|