2017-08-02 16:48:33 +08:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
import argparse
|
2017-08-18 20:48:21 +08:00
|
|
|
import atexit
|
2017-08-02 16:48:33 +08:00
|
|
|
import json
|
|
|
|
import multiprocessing
|
|
|
|
import os
|
|
|
|
import subprocess
|
|
|
|
import sys
|
2017-08-18 20:48:21 +08:00
|
|
|
import time
|
2017-08-02 16:48:33 +08:00
|
|
|
|
2017-08-18 20:48:21 +08:00
|
|
|
# dataset calibrated for running on Apollo (total 4min)
|
2017-11-16 21:29:19 +08:00
|
|
|
# bipartite.py runs for approx. 30s
|
|
|
|
# create_match.py runs for approx. 30s
|
2017-08-18 20:48:21 +08:00
|
|
|
# long_running runs for 1min
|
|
|
|
# long_running runs for 2min
|
2017-08-02 16:48:33 +08:00
|
|
|
SMALL_DATASET = [
|
|
|
|
{
|
|
|
|
"test": "bipartite.py",
|
|
|
|
"options": ["--u-count", "100", "--v-count", "100"],
|
2017-08-18 20:48:21 +08:00
|
|
|
"timeout": 5,
|
2017-08-02 16:48:33 +08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"test": "create_match.py",
|
|
|
|
"options": ["--vertex-count", "40000", "--create-pack-size", "100"],
|
2017-08-18 20:48:21 +08:00
|
|
|
"timeout": 5,
|
2017-08-02 16:48:33 +08:00
|
|
|
},
|
|
|
|
{
|
2017-09-18 20:30:27 +08:00
|
|
|
"test": "long_running.cpp",
|
2017-10-06 03:32:04 +08:00
|
|
|
"options": ["--vertex-count", "1000", "--edge-count", "5000", "--max-time", "1", "--verify", "20"],
|
2017-08-18 20:48:21 +08:00
|
|
|
"timeout": 5,
|
|
|
|
},
|
|
|
|
{
|
2017-09-18 20:30:27 +08:00
|
|
|
"test": "long_running.cpp",
|
2017-10-06 03:32:04 +08:00
|
|
|
"options": ["--vertex-count", "10000", "--edge-count", "50000", "--max-time", "2", "--verify", "30"],
|
2017-08-18 20:48:21 +08:00
|
|
|
"timeout": 5,
|
2017-08-02 16:48:33 +08:00
|
|
|
},
|
|
|
|
]
|
|
|
|
|
2017-08-18 20:48:21 +08:00
|
|
|
# dataset calibrated for running on daily stress instance (total 9h)
|
2017-11-16 21:29:19 +08:00
|
|
|
# bipartite.py and create_match.py run for approx. 15min
|
2017-08-18 20:48:21 +08:00
|
|
|
# long_running runs for 5min x 6 times = 30min
|
|
|
|
# long_running runs for 8h
|
2017-08-02 16:48:33 +08:00
|
|
|
LARGE_DATASET = [
|
|
|
|
{
|
|
|
|
"test": "bipartite.py",
|
|
|
|
"options": ["--u-count", "300", "--v-count", "300"],
|
2017-08-18 20:48:21 +08:00
|
|
|
"timeout": 30,
|
2017-08-02 16:48:33 +08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"test": "create_match.py",
|
|
|
|
"options": ["--vertex-count", "500000", "--create-pack-size", "500"],
|
2017-08-18 20:48:21 +08:00
|
|
|
"timeout": 30,
|
2017-08-02 16:48:33 +08:00
|
|
|
},
|
2017-08-18 20:48:21 +08:00
|
|
|
] + [
|
2017-08-02 16:48:33 +08:00
|
|
|
{
|
2017-09-18 20:30:27 +08:00
|
|
|
"test": "long_running.cpp",
|
2017-10-06 03:32:04 +08:00
|
|
|
"options": ["--vertex-count", "10000", "--edge-count", "40000", "--max-time", "5", "--verify", "60"],
|
2017-08-18 20:48:21 +08:00
|
|
|
"timeout": 8,
|
|
|
|
},
|
|
|
|
] * 6 + [
|
|
|
|
{
|
2017-09-18 20:30:27 +08:00
|
|
|
"test": "long_running.cpp",
|
2018-03-27 04:35:55 +08:00
|
|
|
"options": ["--vertex-count", "200000", "--edge-count", "1000000", "--max-time", "480", "--verify", "300"],
|
2017-08-18 20:48:21 +08:00
|
|
|
"timeout": 500,
|
2017-08-02 16:48:33 +08:00
|
|
|
},
|
|
|
|
]
|
|
|
|
|
|
|
|
# paths
|
|
|
|
SCRIPT_DIR = os.path.dirname(os.path.realpath(__file__))
|
|
|
|
BASE_DIR = os.path.normpath(os.path.join(SCRIPT_DIR, "..", ".."))
|
2017-12-28 23:27:30 +08:00
|
|
|
MEASUREMENTS_FILE = os.path.join(SCRIPT_DIR, ".apollo_measurements")
|
2018-06-20 23:44:47 +08:00
|
|
|
KEY_FILE = os.path.join(SCRIPT_DIR, ".key.pem")
|
|
|
|
CERT_FILE = os.path.join(SCRIPT_DIR, ".cert.pem")
|
2017-12-28 23:27:30 +08:00
|
|
|
|
|
|
|
# long running stats file
|
|
|
|
STATS_FILE = os.path.join(SCRIPT_DIR, ".long_running_stats")
|
|
|
|
SMALL_DATASET[-1]["options"].extend(["--stats-file", STATS_FILE])
|
|
|
|
LARGE_DATASET[-1]["options"].extend(["--stats-file", STATS_FILE])
|
2017-08-02 16:48:33 +08:00
|
|
|
|
2017-08-18 20:48:21 +08:00
|
|
|
# get number of threads
|
|
|
|
if "THREADS" in os.environ:
|
|
|
|
THREADS = os.environ["THREADS"]
|
|
|
|
else:
|
|
|
|
THREADS = multiprocessing.cpu_count()
|
2017-08-02 16:48:33 +08:00
|
|
|
|
|
|
|
|
2020-02-14 18:04:14 +08:00
|
|
|
def get_build_dir():
|
|
|
|
if os.path.exists(os.path.join(BASE_DIR, "build_release")):
|
|
|
|
return os.path.join(BASE_DIR, "build_release")
|
|
|
|
if os.path.exists(os.path.join(BASE_DIR, "build_community")):
|
|
|
|
return os.path.join(BASE_DIR, "build_community")
|
|
|
|
return os.path.join(BASE_DIR, "build")
|
|
|
|
|
|
|
|
|
2019-11-28 23:49:51 +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)
|
|
|
|
|
|
|
|
|
2017-08-18 20:48:21 +08:00
|
|
|
# run test helper function
|
|
|
|
def run_test(args, test, options, timeout):
|
|
|
|
print("Running test '{}'".format(test))
|
2017-08-02 16:48:33 +08:00
|
|
|
|
2017-09-18 20:30:27 +08:00
|
|
|
# find binary
|
|
|
|
if test.endswith(".py"):
|
|
|
|
logging = "DEBUG" if args.verbose else "WARNING"
|
|
|
|
binary = [args.python, "-u", os.path.join(SCRIPT_DIR, test),
|
|
|
|
"--logging", logging]
|
|
|
|
elif test.endswith(".cpp"):
|
2020-02-14 18:04:14 +08:00
|
|
|
exe = os.path.join(get_build_dir(), "tests", "stress", test[:-4])
|
2017-09-18 20:30:27 +08:00
|
|
|
binary = [exe]
|
|
|
|
else:
|
|
|
|
raise Exception("Test '{}' binary not supported!".format(test))
|
|
|
|
|
2017-08-02 16:48:33 +08:00
|
|
|
# start test
|
2017-09-18 20:30:27 +08:00
|
|
|
cmd = binary + ["--worker-count", str(THREADS)] + options
|
2017-08-18 20:48:21 +08:00
|
|
|
start = time.time()
|
|
|
|
ret_test = subprocess.run(cmd, cwd = SCRIPT_DIR, timeout = timeout * 60)
|
|
|
|
|
2017-08-02 16:48:33 +08:00
|
|
|
if ret_test.returncode != 0:
|
|
|
|
raise Exception("Test '{}' binary returned non-zero ({})!".format(
|
|
|
|
test, ret_test.returncode))
|
|
|
|
|
2017-08-18 20:48:21 +08:00
|
|
|
runtime = time.time() - start
|
|
|
|
print(" Done after {:.3f} seconds".format(runtime))
|
|
|
|
|
2017-12-28 23:27:30 +08:00
|
|
|
return runtime
|
|
|
|
|
2017-08-02 16:48:33 +08:00
|
|
|
|
|
|
|
# parse arguments
|
|
|
|
parser = argparse.ArgumentParser(description = "Run stress tests on Memgraph.")
|
2020-02-14 18:04:14 +08:00
|
|
|
parser.add_argument("--memgraph", default = os.path.join(get_build_dir(),
|
2017-08-02 16:48:33 +08:00
|
|
|
"memgraph"))
|
2017-08-18 20:48:21 +08:00
|
|
|
parser.add_argument("--log-file", default = "")
|
2019-12-05 20:24:30 +08:00
|
|
|
parser.add_argument("--data-directory", default = "")
|
2017-08-02 16:48:33 +08:00
|
|
|
parser.add_argument("--python", default = os.path.join(SCRIPT_DIR,
|
|
|
|
"ve3", "bin", "python3"), type = str)
|
|
|
|
parser.add_argument("--large-dataset", action = "store_const",
|
|
|
|
const = True, default = False)
|
2018-06-20 23:44:47 +08:00
|
|
|
parser.add_argument("--use-ssl", action = "store_const",
|
|
|
|
const = True, default = False)
|
2017-08-02 16:48:33 +08:00
|
|
|
parser.add_argument("--verbose", action = "store_const",
|
|
|
|
const = True, default = False)
|
|
|
|
args = parser.parse_args()
|
|
|
|
|
2018-06-20 23:44:47 +08:00
|
|
|
# generate temporary SSL certs
|
|
|
|
if args.use_ssl:
|
|
|
|
# https://unix.stackexchange.com/questions/104171/create-ssl-certificate-non-interactively
|
|
|
|
subj = "/C=HR/ST=Zagreb/L=Zagreb/O=Memgraph/CN=db.memgraph.com"
|
|
|
|
subprocess.run(["openssl", "req", "-new", "-newkey", "rsa:4096",
|
|
|
|
"-days", "365", "-nodes", "-x509", "-subj", subj,
|
|
|
|
"-keyout", KEY_FILE, "-out", CERT_FILE], check=True)
|
|
|
|
|
2017-08-18 20:48:21 +08:00
|
|
|
# start memgraph
|
|
|
|
cwd = os.path.dirname(args.memgraph)
|
2019-12-08 18:42:59 +08:00
|
|
|
cmd = [args.memgraph, "--bolt-num-workers=" + str(THREADS),
|
2019-12-05 20:24:30 +08:00
|
|
|
"--storage-properties-on-edges=true",
|
|
|
|
"--storage-snapshot-on-exit=true",
|
|
|
|
"--storage-snapshot-interval-sec=600",
|
|
|
|
"--storage-snapshot-retention-count=1",
|
|
|
|
"--storage-wal-enabled=true",
|
|
|
|
"--storage-recover-on-startup=false",
|
2020-02-05 17:46:18 +08:00
|
|
|
"--query-execution-timeout-sec=1200"]
|
2017-08-18 20:48:21 +08:00
|
|
|
if not args.verbose:
|
2017-10-04 22:20:38 +08:00
|
|
|
cmd += ["--min-log-level", "1"]
|
2017-08-18 20:48:21 +08:00
|
|
|
if args.log_file:
|
|
|
|
cmd += ["--log-file", args.log_file]
|
2019-12-05 20:24:30 +08:00
|
|
|
if args.data_directory:
|
|
|
|
cmd += ["--data-directory", args.data_directory]
|
2018-06-20 23:44:47 +08:00
|
|
|
if args.use_ssl:
|
2019-12-08 18:42:59 +08:00
|
|
|
cmd += ["--bolt-cert-file", CERT_FILE, "--bolt-key-file", KEY_FILE]
|
2019-12-05 20:24:30 +08:00
|
|
|
proc_mg = subprocess.Popen(cmd, cwd = cwd)
|
2019-11-28 23:49:51 +08:00
|
|
|
wait_for_server(7687)
|
|
|
|
assert proc_mg.poll() is None, "The database binary died prematurely!"
|
2017-08-18 20:48:21 +08:00
|
|
|
|
|
|
|
# at exit cleanup
|
|
|
|
@atexit.register
|
|
|
|
def cleanup():
|
|
|
|
global proc_mg
|
|
|
|
if proc_mg.poll() != None: return
|
|
|
|
proc_mg.kill()
|
|
|
|
proc_mg.wait()
|
|
|
|
|
2017-08-02 16:48:33 +08:00
|
|
|
# run tests
|
2017-12-28 23:27:30 +08:00
|
|
|
runtimes = {}
|
2017-08-02 16:48:33 +08:00
|
|
|
dataset = LARGE_DATASET if args.large_dataset else SMALL_DATASET
|
|
|
|
for test in dataset:
|
2018-06-20 23:44:47 +08:00
|
|
|
if args.use_ssl:
|
|
|
|
test["options"] += ["--use-ssl"]
|
2017-12-28 23:27:30 +08:00
|
|
|
runtime = run_test(args, **test)
|
|
|
|
runtimes[os.path.splitext(test["test"])[0]] = runtime
|
2017-08-18 20:48:21 +08:00
|
|
|
|
|
|
|
# stop memgraph
|
|
|
|
proc_mg.terminate()
|
|
|
|
ret_mg = proc_mg.wait()
|
|
|
|
if ret_mg != 0:
|
|
|
|
raise Exception("Memgraph binary returned non-zero ({})!".format(ret_mg))
|
|
|
|
|
2018-06-20 23:44:47 +08:00
|
|
|
# cleanup certificates
|
|
|
|
if args.use_ssl:
|
|
|
|
os.remove(KEY_FILE)
|
|
|
|
os.remove(CERT_FILE)
|
|
|
|
|
2017-12-28 23:27:30 +08:00
|
|
|
# measurements
|
|
|
|
measurements = ""
|
|
|
|
for key, value in runtimes.items():
|
|
|
|
measurements += "{}.runtime {}\n".format(key, value)
|
|
|
|
with open(STATS_FILE) as f:
|
|
|
|
stats = f.read().split("\n")
|
|
|
|
measurements += "long_running.queries.executed {}\n".format(stats[0])
|
|
|
|
measurements += "long_running.queries.failed {}\n".format(stats[1])
|
|
|
|
with open(MEASUREMENTS_FILE, "w") as f:
|
|
|
|
f.write(measurements)
|
|
|
|
|
2017-08-02 16:48:33 +08:00
|
|
|
print("Done!")
|