-
Notifications
You must be signed in to change notification settings - Fork 96
/
jepsen_test.py
113 lines (93 loc) · 4.58 KB
/
jepsen_test.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as published by
# the Free Software Foundation; either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
#
# See LICENSE for more details.
#
# Copyright (c) 2020 ScyllaDB
import os
from functools import partial
import requests
from sdcm.remote import shell_script_cmd
from sdcm.tester import ClusterTester, teardown_on_exception
from sdcm.utils.decorators import log_run_info
from sdcm.utils.git import clone_repo
JEPSEN_WEB_SERVER_START_DELAY = 15 # seconds
DB_SSH_KEY = "db_node_ssh_key"
class JepsenTest(ClusterTester):
@property
def jepsen_node(self):
return self.loaders.nodes[0]
@teardown_on_exception
@log_run_info
def setup_jepsen(self):
remoter = self.jepsen_node.remoter
remoter.sudo(
"apt-get install -o DPkg::Lock::Timeout=300 -y openjdk-11-jre openjdk-11-jre-headless libjna-java gnuplot graphviz git")
remoter.run(shell_script_cmd("""\
curl -O https://raw.githubusercontent.com/technomancy/leiningen/stable/bin/lein
chmod +x lein
./lein
"""))
clone_repo(remoter=remoter,
repo_url=self.params.get('jepsen_scylla_repo'),
destination_dir_name="jepsen-scylla",
clone_as_root=False)
for db_node in self.db_cluster.nodes:
remoter.run(f"ssh-keyscan -t ed25519 {db_node.ip_address} >> ~/.ssh/known_hosts")
# newer jepsen test is using tcpdump, and expect it in /usr/bin
db_node.install_package("tcpdump")
db_node.remoter.sudo("ln -s /usr/bin/tcpdump /usr/sbin/tcpdump", ignore_status=True)
remoter.send_files(os.path.expanduser(self.db_cluster.nodes[0].ssh_login_info["key_file"]), DB_SSH_KEY)
def setUp(self):
super().setUp()
self.setup_jepsen()
def iter_jepsen_cmd(self, jepsen_cmd, ntimes=1):
for ntry in range(1, ntimes+1):
self.log.info("Run #%s/%s of Jepsen test: `%s'", ntry, ntimes, jepsen_cmd)
yield self.jepsen_node.remoter.run(jepsen_cmd, ignore_status=True, verbose=True).ok
def run_jepsen_cmd_most(self, jepsen_cmd, ntimes):
return sum(self.iter_jepsen_cmd(jepsen_cmd, ntimes)) << 1 > ntimes
def run_jepsen_cmd_any(self, jepsen_cmd, ntimes):
return any(self.iter_jepsen_cmd(jepsen_cmd, ntimes))
def run_jepsen_cmd_all(self, jepsen_cmd, ntimes):
return all(self.iter_jepsen_cmd(jepsen_cmd, ntimes))
def test_jepsen(self):
nodes = " ".join(f"--node {node.ip_address}" for node in self.db_cluster.nodes)
creds = f"--username {self.db_cluster.nodes[0].ssh_login_info['user']} --ssh-private-key ~/{DB_SSH_KEY}"
run_jepsen_cmd = partial(
getattr(self, f"run_jepsen_cmd_{self.params.get('jepsen_test_run_policy')}"),
ntimes=self.params.get("jepsen_test_count")
)
passed = True
for test in self.params.get("jepsen_test_cmd"):
jepsen_cmd = f"cd ~/jepsen-scylla && ~/lein run {test} {nodes} {creds} --no-install-scylla"
passed &= run_jepsen_cmd(jepsen_cmd)
self.assertTrue(passed, "Some of Jepsen tests were failed")
def save_jepsen_report(self):
url = f"http://{self.jepsen_node.external_address}:8080/"
self.log.info("Start web server to serve Jepsen results (will listen on %s)...", url)
self.jepsen_node.remoter.run(shell_script_cmd(f"""\
cd ~/jepsen-scylla
setsid ~/lein run serve > save_jepsen_report.log 2>&1 < /dev/null &
sleep {JEPSEN_WEB_SERVER_START_DELAY}
"""), verbose=True)
with open(os.path.join(self.logdir, "jepsen_report.html"), "wt", encoding="utf-8") as jepsen_report:
jepsen_report.write(requests.get(url).text)
self.log.info("Report has been saved to %s", jepsen_report.name)
return jepsen_report.name
def get_email_data(self):
self.log.info("Prepare data for email")
email_data = self._get_common_email_data()
email_data.update({
"jepsen_report": self.save_jepsen_report(),
"jepsen_scylla_repo": self.params.get("jepsen_scylla_repo"),
"jepsen_test_cmd": self.params.get("jepsen_test_cmd"),
"scylla_repo": self.params.get("scylla_repo"),
})
return email_data