2020-03-06 20:19:08 +08:00
|
|
|
#!/usr/bin/python3 -u
|
2021-10-26 14:53:56 +08:00
|
|
|
|
|
|
|
# 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.
|
|
|
|
|
2020-03-06 20:19:08 +08:00
|
|
|
import argparse
|
|
|
|
import atexit
|
|
|
|
import os
|
|
|
|
import shutil
|
|
|
|
import subprocess
|
|
|
|
import sys
|
|
|
|
import tempfile
|
|
|
|
import time
|
|
|
|
|
|
|
|
SCRIPT_DIR = os.path.dirname(os.path.realpath(__file__))
|
|
|
|
PROJECT_DIR = os.path.normpath(os.path.join(SCRIPT_DIR, "..", "..", ".."))
|
|
|
|
TESTS_DIR = os.path.join(SCRIPT_DIR, "tests")
|
|
|
|
|
|
|
|
SNAPSHOT_FILE_NAME = "snapshot.bin"
|
|
|
|
WAL_FILE_NAME = "wal.bin"
|
2020-12-01 23:51:25 +08:00
|
|
|
|
|
|
|
DUMP_SNAPSHOT_FILE_NAME = "expected_snapshot.cypher"
|
|
|
|
DUMP_WAL_FILE_NAME = "expected_wal.cypher"
|
2020-03-06 20:19:08 +08:00
|
|
|
|
2023-10-25 22:01:59 +08:00
|
|
|
SIGNAL_SIGTERM = 15
|
|
|
|
|
2020-03-06 20:19:08 +08:00
|
|
|
|
|
|
|
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 sorted_content(file_path):
|
2023-10-25 22:01:59 +08:00
|
|
|
with open(file_path, "r") as fin:
|
2020-03-06 20:19:08 +08:00
|
|
|
return sorted(list(map(lambda x: x.strip(), fin.readlines())))
|
|
|
|
|
|
|
|
|
|
|
|
def list_to_string(data):
|
|
|
|
ret = "[\n"
|
|
|
|
for row in data:
|
|
|
|
ret += " " + row + "\n"
|
|
|
|
ret += "]"
|
|
|
|
return ret
|
|
|
|
|
|
|
|
|
2023-10-25 22:01:59 +08:00
|
|
|
def execute_test(memgraph_binary, dump_binary, test_directory, test_type, write_expected):
|
|
|
|
assert test_type in ["SNAPSHOT", "WAL"], "Test type should be either 'SNAPSHOT' or 'WAL'."
|
|
|
|
print("\033[1;36m~~ Executing test {} ({}) ~~\033[0m".format(os.path.relpath(test_directory, TESTS_DIR), test_type))
|
2020-03-06 20:19:08 +08:00
|
|
|
|
|
|
|
working_data_directory = tempfile.TemporaryDirectory()
|
|
|
|
if test_type == "SNAPSHOT":
|
|
|
|
snapshots_dir = os.path.join(working_data_directory.name, "snapshots")
|
|
|
|
os.makedirs(snapshots_dir)
|
2023-10-25 22:01:59 +08:00
|
|
|
shutil.copy(os.path.join(test_directory, SNAPSHOT_FILE_NAME), snapshots_dir)
|
2020-03-06 20:19:08 +08:00
|
|
|
else:
|
|
|
|
wal_dir = os.path.join(working_data_directory.name, "wal")
|
|
|
|
os.makedirs(wal_dir)
|
|
|
|
shutil.copy(os.path.join(test_directory, WAL_FILE_NAME), wal_dir)
|
|
|
|
|
2023-10-25 22:01:59 +08:00
|
|
|
memgraph_args = [
|
|
|
|
memgraph_binary,
|
|
|
|
"--storage-recover-on-startup",
|
|
|
|
"--storage-properties-on-edges",
|
|
|
|
"--data-directory",
|
|
|
|
working_data_directory.name,
|
|
|
|
]
|
2020-03-06 20:19:08 +08:00
|
|
|
|
|
|
|
# Start the memgraph binary
|
|
|
|
memgraph = subprocess.Popen(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:
|
2023-10-25 22:01:59 +08:00
|
|
|
pid = memgraph.pid
|
|
|
|
try:
|
|
|
|
os.kill(pid, SIGNAL_SIGTERM)
|
|
|
|
except os.OSError:
|
|
|
|
assert False
|
|
|
|
time.sleep(1)
|
2020-03-06 20:19:08 +08:00
|
|
|
|
|
|
|
# Execute `database dump`
|
|
|
|
dump_output_file = tempfile.NamedTemporaryFile()
|
|
|
|
dump_args = [dump_binary, "--use-ssl=false"]
|
|
|
|
subprocess.run(dump_args, stdout=dump_output_file, check=True)
|
|
|
|
|
|
|
|
# Shutdown the memgraph binary
|
2023-10-25 22:01:59 +08:00
|
|
|
pid = memgraph.pid
|
|
|
|
try:
|
|
|
|
os.kill(pid, SIGNAL_SIGTERM)
|
|
|
|
except os.OSError:
|
|
|
|
assert False
|
|
|
|
time.sleep(1)
|
2020-03-06 20:19:08 +08:00
|
|
|
|
2020-12-01 23:51:25 +08:00
|
|
|
dump_file_name = DUMP_SNAPSHOT_FILE_NAME if test_type == "SNAPSHOT" else DUMP_WAL_FILE_NAME
|
|
|
|
|
|
|
|
if write_expected:
|
2023-10-25 22:01:59 +08:00
|
|
|
with open(dump_output_file.name, "r") as dump:
|
2020-12-01 23:51:25 +08:00
|
|
|
queries_got = dump.readlines()
|
|
|
|
# Write dump files
|
|
|
|
expected_dump_file = os.path.join(test_directory, dump_file_name)
|
2023-10-25 22:01:59 +08:00
|
|
|
with open(expected_dump_file, "w") as expected:
|
2020-12-01 23:51:25 +08:00
|
|
|
expected.writelines(queries_got)
|
|
|
|
else:
|
|
|
|
# Compare dump files
|
|
|
|
expected_dump_file = os.path.join(test_directory, dump_file_name)
|
2023-10-25 22:01:59 +08:00
|
|
|
assert os.path.exists(expected_dump_file), "Could not find expected dump path {}".format(expected_dump_file)
|
2020-12-01 23:51:25 +08:00
|
|
|
queries_got = sorted_content(dump_output_file.name)
|
|
|
|
queries_expected = sorted_content(expected_dump_file)
|
2023-10-25 22:01:59 +08:00
|
|
|
assert queries_got == queries_expected, "Expected\n{}\nto be equal to\n" "{}".format(
|
|
|
|
list_to_string(queries_got), list_to_string(queries_expected)
|
|
|
|
)
|
2020-03-06 20:19:08 +08:00
|
|
|
|
|
|
|
print("\033[1;32m~~ Test successful ~~\033[0m\n")
|
|
|
|
|
|
|
|
|
|
|
|
def find_test_directories(directory):
|
|
|
|
"""
|
|
|
|
Finds all test directories. Test directory is a directory two levels below
|
|
|
|
the given directory which contains files 'snapshot.bin', 'wal.bin' and
|
|
|
|
'expected.cypher'.
|
|
|
|
"""
|
|
|
|
test_dirs = []
|
|
|
|
for entry_version in os.listdir(directory):
|
|
|
|
entry_version_path = os.path.join(directory, entry_version)
|
|
|
|
if not os.path.isdir(entry_version_path):
|
|
|
|
continue
|
|
|
|
for test_dir in os.listdir(entry_version_path):
|
|
|
|
test_dir_path = os.path.join(entry_version_path, test_dir)
|
|
|
|
if not os.path.isdir(test_dir_path):
|
|
|
|
continue
|
|
|
|
snapshot_file = os.path.join(test_dir_path, SNAPSHOT_FILE_NAME)
|
|
|
|
wal_file = os.path.join(test_dir_path, WAL_FILE_NAME)
|
2023-10-25 22:01:59 +08:00
|
|
|
dump_snapshot_file = os.path.join(test_dir_path, DUMP_SNAPSHOT_FILE_NAME)
|
2020-12-01 23:51:25 +08:00
|
|
|
dump_wal_file = os.path.join(test_dir_path, DUMP_WAL_FILE_NAME)
|
2023-10-25 22:01:59 +08:00
|
|
|
if (
|
|
|
|
os.path.isfile(snapshot_file)
|
|
|
|
and os.path.isfile(dump_snapshot_file)
|
|
|
|
and os.path.isfile(wal_file)
|
|
|
|
and os.path.isfile(dump_wal_file)
|
|
|
|
):
|
2020-03-06 20:19:08 +08:00
|
|
|
test_dirs.append(test_dir_path)
|
|
|
|
else:
|
2023-10-25 22:01:59 +08:00
|
|
|
raise Exception("Missing data in test directory '{}'".format(test_dir_path))
|
2020-03-06 20:19:08 +08:00
|
|
|
return test_dirs
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
memgraph_binary = os.path.join(PROJECT_DIR, "build", "memgraph")
|
|
|
|
dump_binary = os.path.join(PROJECT_DIR, "build", "tools", "src", "mg_dump")
|
|
|
|
|
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
parser.add_argument("--memgraph", default=memgraph_binary)
|
|
|
|
parser.add_argument("--dump", default=dump_binary)
|
2020-12-01 23:51:25 +08:00
|
|
|
parser.add_argument(
|
2023-10-25 22:01:59 +08:00
|
|
|
"--write-expected", action="store_true", help="Overwrite the expected cypher with results from current run"
|
|
|
|
)
|
2020-03-06 20:19:08 +08:00
|
|
|
args = parser.parse_args()
|
|
|
|
|
|
|
|
test_directories = find_test_directories(TESTS_DIR)
|
|
|
|
assert len(test_directories) > 0, "No tests have been found!"
|
|
|
|
|
|
|
|
for test_directory in test_directories:
|
2023-10-25 22:01:59 +08:00
|
|
|
execute_test(args.memgraph, args.dump, test_directory, "SNAPSHOT", args.write_expected)
|
|
|
|
execute_test(args.memgraph, args.dump, test_directory, "WAL", args.write_expected)
|
2020-03-06 20:19:08 +08:00
|
|
|
|
|
|
|
sys.exit(0)
|