209 lines
11 KiB
Python
209 lines
11 KiB
Python
|
# Copyright 2022 Memgraph Ltd.
|
||
|
#
|
||
|
# Use of this software is governed by the Business Source License
|
||
|
# included in the file licenses/BSL.txt; by using this file, you agree to be bound by the terms of the Business Source
|
||
|
# License, and you may not use this file except in compliance with the Business Source License.
|
||
|
#
|
||
|
# As of the Change Date specified in that file, in accordance with
|
||
|
# the Business Source License, use of this software will be governed
|
||
|
# by the Apache License, Version 2.0, included in the file
|
||
|
# licenses/APL.txt.
|
||
|
|
||
|
import sys
|
||
|
|
||
|
import pytest
|
||
|
from common import connect, execute_and_fetch_all
|
||
|
|
||
|
|
||
|
def test_import_mode_disabled_for_in_memory_storages():
|
||
|
cursor = connect().cursor()
|
||
|
with pytest.raises(Exception):
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
|
||
|
|
||
|
def test_import_mode_on_off():
|
||
|
cursor = connect().cursor()
|
||
|
execute_and_fetch_all(cursor, "STORAGE MODE ON_DISK_TRANSACTIONAL")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE INACTIVE")
|
||
|
|
||
|
|
||
|
def test_creating_vertices():
|
||
|
cursor = connect().cursor()
|
||
|
execute_and_fetch_all(cursor, "STORAGE MODE ON_DISK_TRANSACTIONAL")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 1})")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 2})")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n) RETURN n"))) == 2
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n) RETURN n"))) == 2
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE INACTIVE")
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n) RETURN n"))) == 2
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n) RETURN n"))) == 2
|
||
|
execute_and_fetch_all(cursor, "MATCH (n) DETACH DELETE n")
|
||
|
|
||
|
|
||
|
def test_creating_edges():
|
||
|
cursor = connect().cursor()
|
||
|
execute_and_fetch_all(cursor, "STORAGE MODE ON_DISK_TRANSACTIONAL")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 1})")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 2})")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User {id: 1}), (m:User {id: 2}) CREATE (n)-[r:FRIENDS {id: 3}]->(m)")
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n)-[r]->(m) RETURN n, r, m"))) == 1
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE INACTIVE")
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n)-[r]->(m) RETURN n, r, m"))) == 1
|
||
|
execute_and_fetch_all(cursor, "MATCH (n) DETACH DELETE n")
|
||
|
|
||
|
|
||
|
def test_label_index_vertices_loading():
|
||
|
cursor = connect().cursor()
|
||
|
execute_and_fetch_all(cursor, "STORAGE MODE ON_DISK_TRANSACTIONAL")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 1})")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 2})")
|
||
|
execute_and_fetch_all(cursor, "CREATE INDEX ON :User")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n:User) RETURN n"))) == 2
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n:User) RETURN n"))) == 2
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE INACTIVE")
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n:User) RETURN n"))) == 2
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n:User) RETURN n"))) == 2
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User) DETACH DELETE n")
|
||
|
|
||
|
|
||
|
def test_label_index_edges_creation():
|
||
|
cursor = connect().cursor()
|
||
|
execute_and_fetch_all(cursor, "STORAGE MODE ON_DISK_TRANSACTIONAL")
|
||
|
execute_and_fetch_all(cursor, "CREATE INDEX ON :User")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 1})")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 2})")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User {id: 1}), (m:User {id: 2}) CREATE (n)-[r:FRIENDS {id: 3}]->(m)")
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n)-[r]->(m) RETURN n, r, m"))) == 1
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE INACTIVE")
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n:User) RETURN n"))) == 2
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n:User) RETURN n"))) == 2
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n)-[r]->(m) RETURN n, r, m"))) == 1
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User) DETACH DELETE n")
|
||
|
|
||
|
|
||
|
def test_label_property_index_vertices_loading():
|
||
|
cursor = connect().cursor()
|
||
|
execute_and_fetch_all(cursor, "STORAGE MODE ON_DISK_TRANSACTIONAL")
|
||
|
execute_and_fetch_all(cursor, "CREATE INDEX ON :User(id)")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 1})")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 2})")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n:User) WHERE n.id IS NOT NULL RETURN n"))) == 2
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n:User) WHERE n.id IS NOT NULL RETURN n"))) == 2
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE INACTIVE")
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n:User) WHERE n.id IS NOT NULL RETURN n"))) == 2
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n:User) WHERE n.id IS NOT NULL RETURN n"))) == 2
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User) DETACH DELETE n")
|
||
|
|
||
|
|
||
|
def test_label_property_index_edges_creation():
|
||
|
cursor = connect().cursor()
|
||
|
execute_and_fetch_all(cursor, "STORAGE MODE ON_DISK_TRANSACTIONAL")
|
||
|
execute_and_fetch_all(cursor, "CREATE INDEX ON :User(id)")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 1})")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 2})")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User {id: 1}), (m:User {id: 2}) CREATE (n)-[r:FRIENDS {id: 3}]->(m)")
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n)-[r]->(m) RETURN n, r, m"))) == 1
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE INACTIVE")
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n:User) RETURN n"))) == 2
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n:User) RETURN n"))) == 2
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n)-[r]->(m) RETURN n, r, m"))) == 1
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User) DETACH DELETE n")
|
||
|
|
||
|
|
||
|
def test_edge_deletion_in_edge_import_mode():
|
||
|
cursor = connect().cursor()
|
||
|
execute_and_fetch_all(cursor, "STORAGE MODE ON_DISK_TRANSACTIONAL")
|
||
|
execute_and_fetch_all(cursor, "CREATE INDEX ON :User(id)")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 1})")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 2})")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User {id: 1}), (m:User {id: 2}) CREATE (n)-[r:FRIENDS {id: 3}]->(m)")
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n)-[r]->(m) RETURN n, r, m"))) == 1
|
||
|
execute_and_fetch_all(cursor, "MATCH (n)-[r:FRIENDS {id: 3}]->(m) DELETE r")
|
||
|
assert len(list(execute_and_fetch_all(cursor, "MATCH (n)-[r]->(m) RETURN n, r, m"))) == 0
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE INACTIVE")
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User) DETACH DELETE n")
|
||
|
|
||
|
|
||
|
def test_modification_of_edge_properties_in_edge_import_mode():
|
||
|
cursor = connect().cursor()
|
||
|
execute_and_fetch_all(cursor, "STORAGE MODE ON_DISK_TRANSACTIONAL")
|
||
|
execute_and_fetch_all(cursor, "CREATE INDEX ON :User(id)")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 1})")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 2})")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
execute_and_fetch_all(
|
||
|
cursor, "MATCH (n:User {id: 1}), (m:User {id: 2}) CREATE (n)-[r:FRIENDS {id: 3, balance: 1000}]->(m)"
|
||
|
)
|
||
|
assert list(execute_and_fetch_all(cursor, "MATCH (n)-[r]->(m) RETURN r.balance"))[0][0] == 1000
|
||
|
execute_and_fetch_all(cursor, "MATCH (n)-[r:FRIENDS {id: 3}]->(m) SET r.balance = 2000")
|
||
|
assert list(execute_and_fetch_all(cursor, "MATCH (n)-[r]->(m) RETURN r.balance"))[0][0] == 2000
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE INACTIVE")
|
||
|
assert list(execute_and_fetch_all(cursor, "MATCH (n)-[r]->(m) RETURN r.balance"))[0][0] == 2000
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User) DETACH DELETE n")
|
||
|
|
||
|
|
||
|
def test_throw_on_vertex_add_label_during_edge_import_mode():
|
||
|
cursor = connect().cursor()
|
||
|
execute_and_fetch_all(cursor, "STORAGE MODE ON_DISK_TRANSACTIONAL")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 1})")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
with pytest.raises(Exception):
|
||
|
execute_and_fetch_all(cursor, "MATCH (u:User {id: 1}) SET u:User:Person RETURN u")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE INACTIVE")
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User) DETACH DELETE n")
|
||
|
|
||
|
|
||
|
def test_throw_on_vertex_remove_label_during_edge_import_mode():
|
||
|
cursor = connect().cursor()
|
||
|
execute_and_fetch_all(cursor, "STORAGE MODE ON_DISK_TRANSACTIONAL")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User:Person {id: 1})")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
with pytest.raises(Exception):
|
||
|
execute_and_fetch_all(cursor, "MATCH (u:User {id: 1}) REMOVE u:Person RETURN u")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE INACTIVE")
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User) DETACH DELETE n")
|
||
|
|
||
|
|
||
|
def test_throw_on_vertex_set_property_during_edge_import_mode():
|
||
|
cursor = connect().cursor()
|
||
|
execute_and_fetch_all(cursor, "STORAGE MODE ON_DISK_TRANSACTIONAL")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 1, balance: 1000})")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
with pytest.raises(Exception):
|
||
|
execute_and_fetch_all(cursor, "MATCH (u:User {id: 1}) SET u.balance = 2000")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE INACTIVE")
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User) DETACH DELETE n")
|
||
|
|
||
|
|
||
|
def test_throw_on_vertex_create_vertex_during_edge_import_mode():
|
||
|
cursor = connect().cursor()
|
||
|
execute_and_fetch_all(cursor, "STORAGE MODE ON_DISK_TRANSACTIONAL")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 1, balance: 1000})")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
with pytest.raises(Exception):
|
||
|
execute_and_fetch_all(cursor, "CREATE (m:Mother {id: 10})")
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE INACTIVE")
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User) DETACH DELETE n")
|
||
|
|
||
|
|
||
|
def test_throw_changing_import_mode_while_in_explicit_tx():
|
||
|
cursor = connect().cursor()
|
||
|
execute_and_fetch_all(cursor, "STORAGE MODE ON_DISK_TRANSACTIONAL")
|
||
|
execute_and_fetch_all(cursor, "CREATE (u:User {id: 1, balance: 1000})")
|
||
|
execute_and_fetch_all(cursor, "BEGIN")
|
||
|
with pytest.raises(Exception):
|
||
|
execute_and_fetch_all(cursor, "EDGE IMPORT MODE ACTIVE")
|
||
|
execute_and_fetch_all(cursor, "MATCH (n:User) DETACH DELETE n")
|
||
|
|
||
|
|
||
|
if __name__ == "__main__":
|
||
|
sys.exit(pytest.main([__file__, "-rA"]))
|