koder aka kdanilov | 4643fd6 | 2015-02-10 16:20:13 -0800 | [diff] [blame] | 1 | import abc |
| 2 | import json |
koder aka kdanilov | 4643fd6 | 2015-02-10 16:20:13 -0800 | [diff] [blame] | 3 | import os.path |
koder aka kdanilov | e21d747 | 2015-02-14 19:02:04 -0800 | [diff] [blame] | 4 | import logging |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 5 | from StringIO import StringIO |
| 6 | from ConfigParser import RawConfigParser |
koder aka kdanilov | e21d747 | 2015-02-14 19:02:04 -0800 | [diff] [blame] | 7 | |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 8 | from tests import io |
koder aka kdanilov | 3a6633e | 2015-03-26 18:20:00 +0200 | [diff] [blame] | 9 | from ssh_utils import copy_paths |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 10 | from utils import run_over_ssh, ssize_to_b |
koder aka kdanilov | 4643fd6 | 2015-02-10 16:20:13 -0800 | [diff] [blame] | 11 | |
| 12 | |
koder aka kdanilov | e21d747 | 2015-02-14 19:02:04 -0800 | [diff] [blame] | 13 | logger = logging.getLogger("io-perf-tool") |
| 14 | |
| 15 | |
koder aka kdanilov | 4643fd6 | 2015-02-10 16:20:13 -0800 | [diff] [blame] | 16 | class IPerfTest(object): |
| 17 | def __init__(self, on_result_cb): |
| 18 | self.on_result_cb = on_result_cb |
| 19 | |
koder aka kdanilov | 4643fd6 | 2015-02-10 16:20:13 -0800 | [diff] [blame] | 20 | def pre_run(self, conn): |
| 21 | pass |
| 22 | |
| 23 | @abc.abstractmethod |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 24 | def run(self, conn, barrier): |
koder aka kdanilov | 4643fd6 | 2015-02-10 16:20:13 -0800 | [diff] [blame] | 25 | pass |
| 26 | |
| 27 | |
Yulia Portnova | 7ddfa73 | 2015-02-24 17:32:58 +0200 | [diff] [blame] | 28 | class TwoScriptTest(IPerfTest): |
| 29 | def __init__(self, opts, testtool, on_result_cb, keep_tmp_files): |
| 30 | super(TwoScriptTest, self).__init__(on_result_cb) |
| 31 | self.opts = opts |
| 32 | self.pre_run_script = None |
| 33 | self.run_script = None |
| 34 | self.tmp_dir = "/tmp/" |
| 35 | self.set_run_script() |
| 36 | self.set_pre_run_script() |
| 37 | |
| 38 | def set_run_script(self): |
| 39 | self.pre_run_script = self.opts.pre_run_script |
| 40 | |
| 41 | def set_pre_run_script(self): |
| 42 | self.run_script = self.opts.run_script |
| 43 | |
| 44 | def get_remote_for_script(self, script): |
| 45 | return os.path.join(self.tmp_dir, script.rpartition('/')[2]) |
| 46 | |
| 47 | def copy_script(self, conn, src): |
| 48 | remote_path = self.get_remote_for_script(src) |
| 49 | copy_paths(conn, {src: remote_path}) |
| 50 | return remote_path |
| 51 | |
| 52 | def pre_run(self, conn): |
| 53 | remote_script = self.copy_script(conn, self.pre_run_script) |
| 54 | cmd = remote_script |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 55 | code, out_err = run_over_ssh(conn, cmd) |
Yulia Portnova | 7ddfa73 | 2015-02-24 17:32:58 +0200 | [diff] [blame] | 56 | if code != 0: |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 57 | raise Exception("Pre run failed. %s" % out_err) |
Yulia Portnova | 7ddfa73 | 2015-02-24 17:32:58 +0200 | [diff] [blame] | 58 | |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 59 | def run(self, conn, barrier): |
Yulia Portnova | 7ddfa73 | 2015-02-24 17:32:58 +0200 | [diff] [blame] | 60 | remote_script = self.copy_script(conn, self.run_script) |
Yulia Portnova | 3f65e7c | 2015-02-25 17:19:35 +0200 | [diff] [blame] | 61 | cmd = remote_script + ' ' + ' '.join(self.opts) |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 62 | code, out_err = run_over_ssh(conn, cmd) |
| 63 | self.on_result(code, out_err, cmd) |
Yulia Portnova | 7ddfa73 | 2015-02-24 17:32:58 +0200 | [diff] [blame] | 64 | |
| 65 | def parse_results(self, out): |
| 66 | for line in out.split("\n"): |
| 67 | key, separator, value = line.partition(":") |
| 68 | if key and value: |
| 69 | self.on_result_cb((key, float(value))) |
| 70 | |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 71 | def on_result(self, code, out_err, cmd): |
Yulia Portnova | 7ddfa73 | 2015-02-24 17:32:58 +0200 | [diff] [blame] | 72 | if 0 == code: |
| 73 | try: |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 74 | self.parse_results(out_err) |
| 75 | except Exception as exc: |
Yulia Portnova | 7ddfa73 | 2015-02-24 17:32:58 +0200 | [diff] [blame] | 76 | msg_templ = "Error during postprocessing results: {0!r}" |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 77 | raise RuntimeError(msg_templ.format(exc.message)) |
Yulia Portnova | 7ddfa73 | 2015-02-24 17:32:58 +0200 | [diff] [blame] | 78 | else: |
| 79 | templ = "Command {0!r} failed with code {1}. Error output is:\n{2}" |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 80 | logger.error(templ.format(cmd, code, out_err)) |
Yulia Portnova | 7ddfa73 | 2015-02-24 17:32:58 +0200 | [diff] [blame] | 81 | |
| 82 | |
| 83 | class PgBenchTest(TwoScriptTest): |
| 84 | |
| 85 | def set_run_script(self): |
| 86 | self.pre_run_script = "hl_tests/postgres/prepare.sh" |
| 87 | |
| 88 | def set_pre_run_script(self): |
| 89 | self.run_script = "hl_tests/postgres/run.sh" |
| 90 | |
| 91 | |
koder aka kdanilov | 4643fd6 | 2015-02-10 16:20:13 -0800 | [diff] [blame] | 92 | class IOPerfTest(IPerfTest): |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 93 | io_py_remote = "/tmp/io.py" |
| 94 | |
koder aka kdanilov | 4643fd6 | 2015-02-10 16:20:13 -0800 | [diff] [blame] | 95 | def __init__(self, |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 96 | test_options, |
| 97 | on_result_cb): |
koder aka kdanilov | 4643fd6 | 2015-02-10 16:20:13 -0800 | [diff] [blame] | 98 | IPerfTest.__init__(self, on_result_cb) |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 99 | self.options = test_options |
| 100 | self.config_fname = test_options['config_file'] |
| 101 | self.tool = test_options['tool'] |
| 102 | self.configs = [] |
koder aka kdanilov | 50f1864 | 2015-02-11 08:54:44 -0800 | [diff] [blame] | 103 | |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 104 | cp = RawConfigParser() |
| 105 | cp.readfp(open(self.config_fname)) |
koder aka kdanilov | 50f1864 | 2015-02-11 08:54:44 -0800 | [diff] [blame] | 106 | |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 107 | for secname in cp.sections(): |
| 108 | params = dict(cp.items(secname)) |
| 109 | self.configs.append((secname, params)) |
koder aka kdanilov | 4643fd6 | 2015-02-10 16:20:13 -0800 | [diff] [blame] | 110 | |
| 111 | def pre_run(self, conn): |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 112 | local_fname = io.__file__.rsplit('.')[0] + ".py" |
| 113 | self.files_to_copy = {local_fname: self.io_py_remote} |
koder aka kdanilov | 50f1864 | 2015-02-11 08:54:44 -0800 | [diff] [blame] | 114 | copy_paths(conn, self.files_to_copy) |
koder aka kdanilov | 4643fd6 | 2015-02-10 16:20:13 -0800 | [diff] [blame] | 115 | |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 116 | cmd_templ = "dd if=/dev/zero of={0} bs={1} count={2}" |
| 117 | for secname, params in self.configs: |
| 118 | sz = ssize_to_b(params['size']) |
| 119 | msz = msz = sz / (1024 ** 2) |
| 120 | if sz % (1024 ** 2) != 0: |
| 121 | msz += 1 |
koder aka kdanilov | 6e2ae79 | 2015-03-04 18:02:24 -0800 | [diff] [blame] | 122 | |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 123 | cmd = cmd_templ.format(params['filename'], 1024 ** 2, msz) |
| 124 | code, out_err = run_over_ssh(conn, cmd) |
| 125 | |
koder aka kdanilov | 6e2ae79 | 2015-03-04 18:02:24 -0800 | [diff] [blame] | 126 | if code != 0: |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 127 | raise RuntimeError("Preparation failed " + out_err) |
koder aka kdanilov | 6e2ae79 | 2015-03-04 18:02:24 -0800 | [diff] [blame] | 128 | |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 129 | def run(self, conn, barrier): |
| 130 | cmd_templ = "env python2 {0} --type {1} --json -" |
| 131 | cmd = cmd_templ.format(self.io_py_remote, self.tool) |
| 132 | try: |
| 133 | for secname, _params in self.configs: |
| 134 | params = _params.copy() |
| 135 | count = params.pop('count', 1) |
koder aka kdanilov | 4643fd6 | 2015-02-10 16:20:13 -0800 | [diff] [blame] | 136 | |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 137 | config = RawConfigParser() |
| 138 | config.add_section(secname) |
| 139 | |
| 140 | for k, v in params.items(): |
| 141 | config.set(secname, k, v) |
| 142 | |
| 143 | cfg = StringIO() |
| 144 | config.write(cfg) |
| 145 | |
| 146 | # FIX python config parser-fio incompatibility |
| 147 | # remove spaces around '=' |
| 148 | new_cfg = [] |
| 149 | config_data = cfg.getvalue() |
| 150 | for line in config_data.split("\n"): |
| 151 | if '=' in line: |
| 152 | name, val = line.split('=', 1) |
| 153 | name = name.strip() |
| 154 | val = val.strip() |
| 155 | line = "{0}={1}".format(name, val) |
| 156 | new_cfg.append(line) |
| 157 | |
| 158 | for _ in range(count): |
| 159 | barrier.wait() |
| 160 | code, out_err = run_over_ssh(conn, cmd, |
| 161 | stdin_data="\n".join(new_cfg)) |
| 162 | self.on_result(code, out_err, cmd) |
| 163 | finally: |
| 164 | barrier.exit() |
| 165 | |
| 166 | def on_result(self, code, out_err, cmd): |
koder aka kdanilov | 4643fd6 | 2015-02-10 16:20:13 -0800 | [diff] [blame] | 167 | if 0 == code: |
| 168 | try: |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 169 | for line in out_err.split("\n"): |
koder aka kdanilov | 4643fd6 | 2015-02-10 16:20:13 -0800 | [diff] [blame] | 170 | if line.strip() != "": |
| 171 | self.on_result_cb(json.loads(line)) |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 172 | except Exception as exc: |
koder aka kdanilov | 0f0546c | 2015-02-17 20:42:05 -0800 | [diff] [blame] | 173 | msg_templ = "Error during postprocessing results: {0!r}" |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 174 | raise RuntimeError(msg_templ.format(exc.message)) |
koder aka kdanilov | e21d747 | 2015-02-14 19:02:04 -0800 | [diff] [blame] | 175 | else: |
koder aka kdanilov | 2c47309 | 2015-03-29 17:12:13 +0300 | [diff] [blame] | 176 | templ = "Command {0!r} failed with code {1}. Output is:\n{2}" |
| 177 | logger.error(templ.format(cmd, code, out_err)) |