226 lines
7.8 KiB
Python
Executable File
226 lines
7.8 KiB
Python
Executable File
#!/usr/bin/python3 -u
|
|
|
|
# Copyright 2021 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 argparse
|
|
import atexit
|
|
import os
|
|
import subprocess
|
|
import sys
|
|
import tempfile
|
|
import time
|
|
|
|
import yaml
|
|
|
|
SCRIPT_DIR = os.path.dirname(os.path.realpath(__file__))
|
|
BASE_DIR = os.path.normpath(os.path.join(SCRIPT_DIR, "..", "..", ".."))
|
|
BUILD_DIR = os.path.join(BASE_DIR, "build")
|
|
SIGNAL_SIGTERM = 15
|
|
|
|
|
|
def wait_for_server(port, delay=0.1):
|
|
cmd = ["nc", "-z", "-w", "1", "127.0.0.1", str(port)]
|
|
while subprocess.call(cmd) != 0:
|
|
time.sleep(0.01)
|
|
time.sleep(delay)
|
|
|
|
|
|
def extract_rows(data):
|
|
return list(map(lambda x: x.strip(), data.strip().split("\n")))
|
|
|
|
|
|
def list_to_string(data):
|
|
ret = "[\n"
|
|
for row in data:
|
|
ret += " " + row + "\n"
|
|
ret += "]"
|
|
return ret
|
|
|
|
|
|
def verify_lifetime(memgraph_binary, mg_import_csv_binary):
|
|
print("\033[1;36m~~ Verifying that mg_import_csv can't be started while " "memgraph is running ~~\033[0m")
|
|
storage_directory = tempfile.TemporaryDirectory()
|
|
|
|
# Generate common args
|
|
common_args = ["--data-directory", storage_directory.name, "--storage-properties-on-edges=false"]
|
|
|
|
# Start the memgraph binary
|
|
memgraph_args = [memgraph_binary, "--storage-recover-on-startup"] + common_args
|
|
memgraph = subprocess.Popen(list(map(str, memgraph_args)))
|
|
time.sleep(0.1)
|
|
assert memgraph.poll() is None, "Memgraph process died prematurely!"
|
|
wait_for_server(7687)
|
|
|
|
# Register cleanup function
|
|
@atexit.register
|
|
def cleanup():
|
|
if memgraph.poll() is None:
|
|
pid = memgraph.pid
|
|
try:
|
|
os.kill(pid, SIGNAL_SIGTERM)
|
|
except os.OSError:
|
|
assert False, "Memgraph process didn't exit cleanly!"
|
|
time.sleep(1)
|
|
|
|
# Execute mg_import_csv.
|
|
mg_import_csv_args = [mg_import_csv_binary, "--nodes", "/dev/null"] + common_args
|
|
ret = subprocess.run(mg_import_csv_args)
|
|
|
|
# Check the return code
|
|
if ret.returncode == 0:
|
|
raise Exception("The importer was able to run while memgraph was running!")
|
|
|
|
# Shutdown the memgraph binary
|
|
pid = memgraph.pid
|
|
try:
|
|
os.kill(pid, SIGNAL_SIGTERM)
|
|
except os.OSError:
|
|
assert False, "Memgraph process didn't exit cleanly!"
|
|
time.sleep(1)
|
|
|
|
print("\033[1;32m~~ Test successful ~~\033[0m\n")
|
|
|
|
|
|
def execute_test(name, test_path, test_config, memgraph_binary, mg_import_csv_binary, tester_binary, write_expected):
|
|
print("\033[1;36m~~ Executing test", name, "~~\033[0m")
|
|
storage_directory = tempfile.TemporaryDirectory()
|
|
|
|
# Verify test configuration
|
|
if ("import_should_fail" not in test_config and "expected" not in test_config) or (
|
|
"import_should_fail" in test_config and "expected" in test_config
|
|
):
|
|
raise Exception("The test should specify either 'import_should_fail' " "or 'expected'!")
|
|
|
|
expected_path = test_config.pop("expected", "")
|
|
import_should_fail = test_config.pop("import_should_fail", False)
|
|
|
|
# Generate common args
|
|
properties_on_edges = bool(test_config.pop("properties_on_edges", False))
|
|
common_args = [
|
|
"--data-directory",
|
|
storage_directory.name,
|
|
"--storage-properties-on-edges=" + str(properties_on_edges).lower(),
|
|
]
|
|
|
|
# Generate mg_import_csv args using flags specified in the test
|
|
mg_import_csv_args = [mg_import_csv_binary] + common_args
|
|
for key, value in test_config.items():
|
|
flag = "--" + key.replace("_", "-")
|
|
if isinstance(value, list):
|
|
for item in value:
|
|
mg_import_csv_args.extend([flag, str(item)])
|
|
elif isinstance(value, bool):
|
|
mg_import_csv_args.append(flag + "=" + str(value).lower())
|
|
else:
|
|
mg_import_csv_args.extend([flag, str(value)])
|
|
|
|
# Execute mg_import_csv
|
|
ret = subprocess.run(mg_import_csv_args, cwd=test_path)
|
|
|
|
if import_should_fail:
|
|
if ret.returncode == 0:
|
|
raise Exception("The import should have failed, but it " "succeeded instead!")
|
|
else:
|
|
print("\033[1;32m~~ Test successful ~~\033[0m\n")
|
|
return
|
|
else:
|
|
if ret.returncode != 0:
|
|
raise Exception("The import should have succeeded, but it " "failed instead!")
|
|
|
|
# Start the memgraph binary
|
|
memgraph_args = [memgraph_binary, "--storage-recover-on-startup"] + common_args
|
|
memgraph = subprocess.Popen(list(map(str, memgraph_args)))
|
|
time.sleep(0.1)
|
|
assert memgraph.poll() is None, "Memgraph process died prematurely!"
|
|
wait_for_server(7687)
|
|
|
|
# Register cleanup function
|
|
@atexit.register
|
|
def cleanup():
|
|
if memgraph.poll() is None:
|
|
pid = memgraph.pid
|
|
try:
|
|
os.kill(pid, SIGNAL_SIGTERM)
|
|
except os.OSError:
|
|
assert False, "Memgraph process didn't exit cleanly!"
|
|
time.sleep(1)
|
|
|
|
# Get the contents of the database
|
|
queries_got = extract_rows(
|
|
subprocess.run([tester_binary], stdout=subprocess.PIPE, check=True).stdout.decode("utf-8")
|
|
)
|
|
|
|
# Shutdown the memgraph binary
|
|
pid = memgraph.pid
|
|
try:
|
|
os.kill(pid, SIGNAL_SIGTERM)
|
|
except os.OSError:
|
|
assert False, "Memgraph process didn't exit cleanly!"
|
|
time.sleep(1)
|
|
|
|
if write_expected:
|
|
with open(os.path.join(test_path, expected_path), "w") as expected:
|
|
expected.write("\n".join(queries_got))
|
|
|
|
else:
|
|
if expected_path:
|
|
with open(os.path.join(test_path, expected_path)) as f:
|
|
queries_expected = extract_rows(f.read())
|
|
else:
|
|
queries_expected = ""
|
|
|
|
# Verify the queries
|
|
queries_expected.sort()
|
|
queries_got.sort()
|
|
assert queries_got == queries_expected, "Got:\n{}\nExpected:\n" "{}".format(
|
|
list_to_string(queries_got), list_to_string(queries_expected)
|
|
)
|
|
print("\033[1;32m~~ Test successful ~~\033[0m\n")
|
|
|
|
|
|
if __name__ == "__main__":
|
|
memgraph_binary = os.path.join(BUILD_DIR, "memgraph")
|
|
mg_import_csv_binary = os.path.join(BUILD_DIR, "src", "mg_import_csv")
|
|
tester_binary = os.path.join(BUILD_DIR, "tests", "integration", "mg_import_csv", "tester")
|
|
|
|
parser = argparse.ArgumentParser()
|
|
parser.add_argument("--memgraph", default=memgraph_binary)
|
|
parser.add_argument("--mg-import-csv", default=mg_import_csv_binary)
|
|
parser.add_argument("--tester", default=tester_binary)
|
|
parser.add_argument(
|
|
"--write-expected",
|
|
action="store_true",
|
|
help="Overwrite the expected values with the results of the current run",
|
|
)
|
|
args = parser.parse_args()
|
|
|
|
# First test whether the CSV importer can be started while the main
|
|
# Memgraph binary is running.
|
|
verify_lifetime(memgraph_binary, mg_import_csv_binary)
|
|
|
|
# Run all import scenarios.
|
|
test_dir = os.path.join(SCRIPT_DIR, "tests")
|
|
tests_list = sorted(os.listdir(test_dir))
|
|
assert len(tests_list) > 0, "No tests were found!"
|
|
for name in tests_list:
|
|
print("\033[1;34m~~ Processing tests from", name, "~~\033[0m\n")
|
|
test_path = os.path.join(test_dir, name)
|
|
with open(os.path.join(test_path, "test.yaml")) as f:
|
|
testcases = yaml.safe_load(f)
|
|
for test_config in testcases:
|
|
test_name = name + "/" + test_config.pop("name")
|
|
execute_test(
|
|
test_name, test_path, test_config, args.memgraph, args.mg_import_csv, args.tester, args.write_expected
|
|
)
|
|
|
|
sys.exit(0)
|