| import re |
| import time |
| import json |
| import stat |
| import random |
| import hashlib |
| import os.path |
| import logging |
| import datetime |
| import functools |
| import collections |
| from typing import Dict, List, Callable, Any, Tuple, Optional |
| |
| import yaml |
| import texttable |
| from paramiko.ssh_exception import SSHException |
| from concurrent.futures import ThreadPoolExecutor, wait |
| |
| import wally |
| |
| from ...pretty_yaml import dumps |
| from ...statistic import round_3_digit, data_property, average |
| from ...utils import ssize2b, sec_to_str, StopTestError, Barrier, get_os |
| from ...inode import INode |
| |
| from ..itest import (TimeSeriesValue, PerfTest, TestResults, TestConfig) |
| |
| from .fio_task_parser import (execution_time, fio_cfg_compile, |
| get_test_summary, get_test_summary_tuple, |
| get_test_sync_mode, FioJobSection) |
| from .rpc_plugin import parse_fio_result |
| |
| |
| logger = logging.getLogger("wally") |
| |
| |
| class NoData: |
| pass |
| |
| |
| def cached_prop(func: Callable[..., Any]) -> Callable[..., Any]: |
| @property |
| @functools.wraps(func) |
| def closure(self) -> Any: |
| val = getattr(self, "_" + func.__name__) |
| if val is NoData: |
| val = func(self) |
| setattr(self, "_" + func.__name__, val) |
| return val |
| return closure |
| |
| |
| def load_fio_log_file(fname: str) -> TimeSeriesValue: |
| with open(fname) as fd: |
| it = [ln.split(',')[:2] for ln in fd] |
| |
| vals = [(float(off) / 1000, # convert us to ms |
| float(val.strip()) + 0.5) # add 0.5 to compemsate average value |
| # as fio trimm all values in log to integer |
| for off, val in it] |
| |
| return TimeSeriesValue(vals) |
| |
| |
| READ_IOPS_DISCSTAT_POS = 3 |
| WRITE_IOPS_DISCSTAT_POS = 7 |
| |
| |
| def load_sys_log_file(ftype: str, fname: str) -> TimeSeriesValue: |
| assert ftype == 'iops' |
| pval = None |
| with open(fname) as fd: |
| iops = [] |
| for ln in fd: |
| params = ln.split() |
| cval = int(params[WRITE_IOPS_DISCSTAT_POS]) + \ |
| int(params[READ_IOPS_DISCSTAT_POS]) |
| if pval is not None: |
| iops.append(cval - pval) |
| pval = cval |
| |
| vals = [(idx * 1000, val) for idx, val in enumerate(iops)] |
| return TimeSeriesValue(vals) |
| |
| |
| def load_test_results(folder: str, run_num: int) -> 'FioRunResult': |
| res = {} |
| params = None |
| |
| fn = os.path.join(folder, str(run_num) + '_params.yaml') |
| params = yaml.load(open(fn).read()) |
| |
| conn_ids_set = set() |
| rr = r"{}_(?P<conn_id>.*?)_(?P<type>[^_.]*)\.\d+\.log$".format(run_num) |
| for fname in os.listdir(folder): |
| rm = re.match(rr, fname) |
| if rm is None: |
| continue |
| |
| conn_id_s = rm.group('conn_id') |
| conn_id = conn_id_s.replace('_', ':') |
| ftype = rm.group('type') |
| |
| if ftype not in ('iops', 'bw', 'lat'): |
| continue |
| |
| ts = load_fio_log_file(os.path.join(folder, fname)) |
| res.setdefault(ftype, {}).setdefault(conn_id, []).append(ts) |
| |
| conn_ids_set.add(conn_id) |
| |
| rr = r"{}_(?P<conn_id>.*?)_(?P<type>[^_.]*)\.sys\.log$".format(run_num) |
| for fname in os.listdir(folder): |
| rm = re.match(rr, fname) |
| if rm is None: |
| continue |
| |
| conn_id_s = rm.group('conn_id') |
| conn_id = conn_id_s.replace('_', ':') |
| ftype = rm.group('type') |
| |
| if ftype not in ('iops', 'bw', 'lat'): |
| continue |
| |
| ts = load_sys_log_file(ftype, os.path.join(folder, fname)) |
| res.setdefault(ftype + ":sys", {}).setdefault(conn_id, []).append(ts) |
| |
| conn_ids_set.add(conn_id) |
| |
| mm_res = {} |
| |
| if len(res) == 0: |
| raise ValueError("No data was found") |
| |
| for key, data in res.items(): |
| conn_ids = sorted(conn_ids_set) |
| awail_ids = [conn_id for conn_id in conn_ids if conn_id in data] |
| matr = [data[conn_id] for conn_id in awail_ids] |
| mm_res[key] = MeasurementMatrix(matr, awail_ids) |
| |
| raw_res = {} |
| for conn_id in conn_ids: |
| fn = os.path.join(folder, "{0}_{1}_rawres.json".format(run_num, conn_id_s)) |
| |
| # remove message hack |
| fc = "{" + open(fn).read().split('{', 1)[1] |
| raw_res[conn_id] = json.loads(fc) |
| |
| fio_task = FioJobSection(params['name']) |
| fio_task.vals.update(params['vals']) |
| |
| config = TestConfig('io', params, None, params['nodes'], folder, None) |
| return FioRunResult(config, fio_task, mm_res, raw_res, params['intervals'], run_num) |
| |
| |
| class Attrmapper: |
| def __init__(self, dct: Dict[str, Any]): |
| self.__dct = dct |
| |
| def __getattr__(self, name): |
| try: |
| return self.__dct[name] |
| except KeyError: |
| raise AttributeError(name) |
| |
| |
| class DiskPerfInfo: |
| def __init__(self, name: str, summary: str, params: Dict[str, Any], testnodes_count: int): |
| self.name = name |
| self.bw = None |
| self.iops = None |
| self.lat = None |
| self.lat_50 = None |
| self.lat_95 = None |
| self.lat_avg = None |
| |
| self.raw_bw = [] |
| self.raw_iops = [] |
| self.raw_lat = [] |
| |
| self.params = params |
| self.testnodes_count = testnodes_count |
| self.summary = summary |
| self.p = Attrmapper(self.params['vals']) |
| |
| self.sync_mode = get_test_sync_mode(self.params['vals']) |
| self.concurence = self.params['vals'].get('numjobs', 1) |
| |
| |
| def get_lat_perc_50_95(lat_mks: List[float]) -> Tuple[float, float]: |
| curr_perc = 0 |
| perc_50 = None |
| perc_95 = None |
| pkey = None |
| for key, val in sorted(lat_mks.items()): |
| if curr_perc + val >= 50 and perc_50 is None: |
| if pkey is None or val < 1.: |
| perc_50 = key |
| else: |
| perc_50 = (50. - curr_perc) / val * (key - pkey) + pkey |
| |
| if curr_perc + val >= 95: |
| if pkey is None or val < 1.: |
| perc_95 = key |
| else: |
| perc_95 = (95. - curr_perc) / val * (key - pkey) + pkey |
| break |
| |
| pkey = key |
| curr_perc += val |
| |
| # for k, v in sorted(lat_mks.items()): |
| # if k / 1000 > 0: |
| # print "{0:>4}".format(k / 1000), v |
| |
| # print perc_50 / 1000., perc_95 / 1000. |
| # exit(1) |
| return perc_50 / 1000., perc_95 / 1000. |
| |
| |
| class IOTestResults: |
| def __init__(self, suite_name: str, fio_results: 'FioRunResult', log_directory: str): |
| self.suite_name = suite_name |
| self.fio_results = fio_results |
| self.log_directory = log_directory |
| |
| def __iter__(self): |
| return iter(self.fio_results) |
| |
| def __len__(self): |
| return len(self.fio_results) |
| |
| def get_yamable(self) -> Dict[str, List[str]]: |
| items = [(fio_res.summary(), fio_res.idx) for fio_res in self] |
| return {self.suite_name: [self.log_directory] + items} |
| |
| |
| class FioRunResult(TestResults): |
| """ |
| Fio run results |
| config: TestConfig |
| fio_task: FioJobSection |
| ts_results: {str: MeasurementMatrix[TimeSeriesValue]} |
| raw_result: ???? |
| run_interval:(float, float) - test tun time, used for sensors |
| """ |
| def __init__(self, config, fio_task, ts_results, raw_result, run_interval, idx): |
| |
| self.name = fio_task.name.rsplit("_", 1)[0] |
| self.fio_task = fio_task |
| self.idx = idx |
| |
| self.bw = ts_results['bw'] |
| self.lat = ts_results['lat'] |
| self.iops = ts_results['iops'] |
| |
| if 'iops:sys' in ts_results: |
| self.iops_sys = ts_results['iops:sys'] |
| else: |
| self.iops_sys = None |
| |
| res = {"bw": self.bw, |
| "lat": self.lat, |
| "iops": self.iops, |
| "iops:sys": self.iops_sys} |
| |
| self.sensors_data = None |
| self._pinfo = None |
| TestResults.__init__(self, config, res, raw_result, run_interval) |
| |
| def get_params_from_fio_report(self): |
| nodes = self.bw.connections_ids |
| |
| iops = [self.raw_result[node]['jobs'][0]['mixed']['iops'] for node in nodes] |
| total_ios = [self.raw_result[node]['jobs'][0]['mixed']['total_ios'] for node in nodes] |
| runtime = [self.raw_result[node]['jobs'][0]['mixed']['runtime'] / 1000 for node in nodes] |
| flt_iops = [float(ios) / rtime for ios, rtime in zip(total_ios, runtime)] |
| |
| bw = [self.raw_result[node]['jobs'][0]['mixed']['bw'] for node in nodes] |
| total_bytes = [self.raw_result[node]['jobs'][0]['mixed']['io_bytes'] for node in nodes] |
| flt_bw = [float(tbytes) / rtime for tbytes, rtime in zip(total_bytes, runtime)] |
| |
| return {'iops': iops, |
| 'flt_iops': flt_iops, |
| 'bw': bw, |
| 'flt_bw': flt_bw} |
| |
| def summary(self): |
| return get_test_summary(self.fio_task, len(self.config.nodes)) |
| |
| def summary_tpl(self): |
| return get_test_summary_tuple(self.fio_task, len(self.config.nodes)) |
| |
| def get_lat_perc_50_95_multy(self): |
| lat_mks = collections.defaultdict(lambda: 0) |
| num_res = 0 |
| |
| for result in self.raw_result.values(): |
| num_res += len(result['jobs']) |
| for job_info in result['jobs']: |
| for k, v in job_info['latency_ms'].items(): |
| if isinstance(k, basestring) and k.startswith('>='): |
| lat_mks[int(k[2:]) * 1000] += v |
| else: |
| lat_mks[int(k) * 1000] += v |
| |
| for k, v in job_info['latency_us'].items(): |
| lat_mks[int(k)] += v |
| |
| for k, v in lat_mks.items(): |
| lat_mks[k] = float(v) / num_res |
| return get_lat_perc_50_95(lat_mks) |
| |
| def disk_perf_info(self, avg_interval=2.0): |
| |
| if self._pinfo is not None: |
| return self._pinfo |
| |
| testnodes_count = len(self.config.nodes) |
| |
| pinfo = DiskPerfInfo(self.name, |
| self.summary(), |
| self.params, |
| testnodes_count) |
| |
| def prepare(data, drop=1): |
| if data is None: |
| return data |
| |
| res = [] |
| for ts_data in data: |
| if ts_data.average_interval() < avg_interval: |
| ts_data = ts_data.derived(avg_interval) |
| |
| # drop last value on bounds |
| # as they may contains ranges without activities |
| assert len(ts_data.values) >= drop + 1, str(drop) + " " + str(ts_data.values) |
| |
| if drop > 0: |
| res.append(ts_data.values[:-drop]) |
| else: |
| res.append(ts_data.values) |
| |
| return res |
| |
| def agg_data(matr): |
| arr = sum(matr, []) |
| min_len = min(map(len, arr)) |
| res = [] |
| for idx in range(min_len): |
| res.append(sum(dt[idx] for dt in arr)) |
| return res |
| |
| pinfo.raw_lat = map(prepare, self.lat.per_vm()) |
| num_th = sum(map(len, pinfo.raw_lat)) |
| lat_avg = [val / num_th for val in agg_data(pinfo.raw_lat)] |
| pinfo.lat_avg = data_property(lat_avg).average / 1000 # us to ms |
| |
| pinfo.lat_50, pinfo.lat_95 = self.get_lat_perc_50_95_multy() |
| pinfo.lat = pinfo.lat_50 |
| |
| pinfo.raw_bw = map(prepare, self.bw.per_vm()) |
| pinfo.raw_iops = map(prepare, self.iops.per_vm()) |
| |
| if self.iops_sys is not None: |
| pinfo.raw_iops_sys = map(prepare, self.iops_sys.per_vm()) |
| pinfo.iops_sys = data_property(agg_data(pinfo.raw_iops_sys)) |
| else: |
| pinfo.raw_iops_sys = None |
| pinfo.iops_sys = None |
| |
| fparams = self.get_params_from_fio_report() |
| fio_report_bw = sum(fparams['flt_bw']) |
| fio_report_iops = sum(fparams['flt_iops']) |
| |
| agg_bw = agg_data(pinfo.raw_bw) |
| agg_iops = agg_data(pinfo.raw_iops) |
| |
| log_bw_avg = average(agg_bw) |
| log_iops_avg = average(agg_iops) |
| |
| # update values to match average from fio report |
| coef_iops = fio_report_iops / float(log_iops_avg) |
| coef_bw = fio_report_bw / float(log_bw_avg) |
| |
| bw_log = data_property([val * coef_bw for val in agg_bw]) |
| iops_log = data_property([val * coef_iops for val in agg_iops]) |
| |
| bw_report = data_property([fio_report_bw]) |
| iops_report = data_property([fio_report_iops]) |
| |
| # When IOPS/BW per thread is too low |
| # data from logs is rounded to match |
| iops_per_th = sum(sum(pinfo.raw_iops, []), []) |
| if average(iops_per_th) > 10: |
| pinfo.iops = iops_log |
| pinfo.iops2 = iops_report |
| else: |
| pinfo.iops = iops_report |
| pinfo.iops2 = iops_log |
| |
| bw_per_th = sum(sum(pinfo.raw_bw, []), []) |
| if average(bw_per_th) > 10: |
| pinfo.bw = bw_log |
| pinfo.bw2 = bw_report |
| else: |
| pinfo.bw = bw_report |
| pinfo.bw2 = bw_log |
| |
| self._pinfo = pinfo |
| |
| return pinfo |
| |
| |
| class IOPerfTest(PerfTest): |
| tcp_conn_timeout = 30 |
| max_pig_timeout = 5 |
| soft_runcycle = 5 * 60 |
| retry_time = 30 |
| |
| zero_md5_hash = hashlib.md5() |
| zero_md5_hash.update(b"\x00" * 1024) |
| zero_md5 = zero_md5_hash.hexdigest() |
| |
| def __init__(self, config): |
| PerfTest.__init__(self, config) |
| |
| get = self.config.params.get |
| do_get = self.config.params.__getitem__ |
| |
| self.config_fname = do_get('cfg') |
| |
| if '/' not in self.config_fname and '.' not in self.config_fname: |
| cfgs_dir = os.path.dirname(__file__) |
| self.config_fname = os.path.join(cfgs_dir, |
| self.config_fname + '.cfg') |
| |
| self.alive_check_interval = get('alive_check_interval') |
| self.use_system_fio = get('use_system_fio', False) |
| |
| if get('prefill_files') is not None: |
| logger.warning("prefill_files option is depricated. Use force_prefill instead") |
| |
| self.force_prefill = get('force_prefill', False) |
| self.config_params = get('params', {}).copy() |
| |
| self.io_py_remote = self.join_remote("agent.py") |
| self.results_file = self.join_remote("results.json") |
| self.pid_file = self.join_remote("pid") |
| self.task_file = self.join_remote("task.cfg") |
| self.sh_file = self.join_remote("cmd.sh") |
| self.err_out_file = self.join_remote("fio_err_out") |
| self.io_log_file = self.join_remote("io_log.txt") |
| self.exit_code_file = self.join_remote("exit_code") |
| |
| self.max_latency = get("max_lat", None) |
| self.min_bw_per_thread = get("min_bw", None) |
| |
| self.use_sudo = get("use_sudo", True) |
| |
| self.raw_cfg = open(self.config_fname).read() |
| self.fio_configs = None |
| |
| @classmethod |
| def load(cls, suite_name: str, folder: str) -> IOTestResults: |
| res = [] |
| for fname in os.listdir(folder): |
| if re.match("\d+_params.yaml$", fname): |
| num = int(fname.split('_')[0]) |
| res.append(load_test_results(folder, num)) |
| return IOTestResults(suite_name, res, folder) |
| |
| def cleanup(self): |
| # delete_file(conn, self.io_py_remote) |
| # Need to remove tempo files, used for testing |
| pass |
| |
| # size is megabytes |
| def check_prefill_required(self, node: INode, fname: str, size: int, num_blocks: Optional[int]=16) -> bool: |
| try: |
| fstats = node.stat_file(fname) |
| if stat.S_ISREG(fstats.st_mode) and fstats.st_size < size * 1024 ** 2: |
| return True |
| except EnvironmentError: |
| return True |
| |
| cmd = 'python -c "' + \ |
| "import sys;" + \ |
| "fd = open('{0}', 'rb');" + \ |
| "fd.seek({1});" + \ |
| "data = fd.read(1024); " + \ |
| "sys.stdout.write(data + ' ' * ( 1024 - len(data)))\" | md5sum" |
| |
| if self.use_sudo: |
| cmd = "sudo " + cmd |
| |
| bsize = size * (1024 ** 2) |
| offsets = [random.randrange(bsize - 1024) for _ in range(num_blocks)] |
| offsets.append(bsize - 1024) |
| offsets.append(0) |
| |
| for offset in offsets: |
| data = node.run(cmd.format(fname, offset), nolog=True) |
| |
| md = "" |
| for line in data.split("\n"): |
| if "unable to resolve" not in line: |
| md = line.split()[0].strip() |
| break |
| |
| if len(md) != 32: |
| logger.error("File data check is failed - " + data) |
| return True |
| |
| if self.zero_md5 == md: |
| return True |
| |
| return False |
| |
| def prefill_test_files(self, node: INode, files: List[str], force:bool=False) -> None: |
| if self.use_system_fio: |
| cmd_templ = "fio " |
| else: |
| cmd_templ = "{0}/fio ".format(self.config.remote_dir) |
| |
| if self.use_sudo: |
| cmd_templ = "sudo " + cmd_templ |
| |
| cmd_templ += "--name=xxx --filename={0} --direct=1" + \ |
| " --bs=4m --size={1}m --rw=write" |
| |
| ssize = 0 |
| |
| if force: |
| logger.info("File prefilling is forced") |
| |
| ddtime = 0 |
| for fname, curr_sz in files.items(): |
| if not force: |
| if not self.check_prefill_required(node, fname, curr_sz): |
| logger.debug("prefill is skipped") |
| continue |
| |
| logger.info("Prefilling file {0}".format(fname)) |
| cmd = cmd_templ.format(fname, curr_sz) |
| ssize += curr_sz |
| |
| stime = time.time() |
| node.run(cmd, timeout=curr_sz) |
| ddtime += time.time() - stime |
| |
| if ddtime > 1.0: |
| fill_bw = int(ssize / ddtime) |
| mess = "Initiall fio fill bw is {0} MiBps for this vm" |
| logger.info(mess.format(fill_bw)) |
| |
| def install_utils(self, node: INode) -> None: |
| need_install = [] |
| packs = [('screen', 'screen')] |
| os_info = get_os(node) |
| |
| if self.use_system_fio: |
| packs.append(('fio', 'fio')) |
| else: |
| packs.append(('bzip2', 'bzip2')) |
| |
| for bin_name, package in packs: |
| if bin_name is None: |
| need_install.append(package) |
| continue |
| |
| try: |
| node.run('which ' + bin_name, nolog=True) |
| except OSError: |
| need_install.append(package) |
| |
| if len(need_install) != 0: |
| if 'redhat' == os_info.distro: |
| cmd = "sudo yum -y install " + " ".join(need_install) |
| else: |
| cmd = "sudo apt-get -y install " + " ".join(need_install) |
| |
| try: |
| node.run(cmd) |
| except OSError as err: |
| raise OSError("Can't install - {}".format(" ".join(need_install))) from err |
| |
| if not self.use_system_fio: |
| fio_dir = os.path.dirname(os.path.dirname(wally.__file__)) |
| fio_dir = os.path.join(os.getcwd(), fio_dir) |
| fio_dir = os.path.join(fio_dir, 'fio_binaries') |
| fname = 'fio_{0.release}_{0.arch}.bz2'.format(os_info) |
| fio_path = os.path.join(fio_dir, fname) |
| |
| if not os.path.exists(fio_path): |
| raise RuntimeError("No prebuild fio binary available for {0}".format(os_info)) |
| |
| bz_dest = self.join_remote('fio.bz2') |
| node.copy_file(fio_path, bz_dest) |
| node.run("bzip2 --decompress {}" + bz_dest, nolog=True) |
| node.run("chmod a+x " + self.join_remote("fio"), nolog=True) |
| |
| def pre_run(self) -> None: |
| if 'FILESIZE' not in self.config_params: |
| raise NotImplementedError("File size detection is not implemented") |
| |
| self.fio_configs = fio_cfg_compile(self.raw_cfg, |
| self.config_fname, |
| self.config_params) |
| self.fio_configs = list(self.fio_configs) |
| |
| files = {} |
| for section in self.fio_configs: |
| sz = ssize2b(section.vals['size']) |
| msz = sz / (1024 ** 2) |
| |
| if sz % (1024 ** 2) != 0: |
| msz += 1 |
| |
| fname = section.vals['filename'] |
| |
| # if already has other test with the same file name |
| # take largest size |
| files[fname] = max(files.get(fname, 0), msz) |
| |
| with ThreadPoolExecutor(len(self.config.nodes)) as pool: |
| fc = functools.partial(self.pre_run_th, |
| files=files, |
| force=self.force_prefill) |
| list(pool.map(fc, self.config.nodes)) |
| |
| def pre_run_th(self, node: INode, files: List[str], force_prefil: Optional[bool]=False) -> None: |
| try: |
| cmd = 'mkdir -p "{0}"'.format(self.config.remote_dir) |
| if self.use_sudo: |
| cmd = "sudo " + cmd |
| cmd += " ; sudo chown {0} {1}".format(node.get_user(), |
| self.config.remote_dir) |
| node.run(cmd, nolog=True) |
| |
| assert self.config.remote_dir != "" and self.config.remote_dir != "/" |
| node.run("rm -rf {}/*".format(self.config.remote_dir), nolog=True) |
| |
| except Exception as exc: |
| msg = "Failed to create folder {} on remote {}." |
| msg = msg.format(self.config.remote_dir, node, exc) |
| logger.exception(msg) |
| raise StopTestError(msg) from exc |
| |
| self.install_utils(node) |
| self.prefill_test_files(node, files, force_prefil) |
| |
| def show_expected_execution_time(self) -> None: |
| if len(self.fio_configs) > 1: |
| # +10% - is a rough estimation for additional operations |
| # like sftp, etc |
| exec_time = int(sum(map(execution_time, self.fio_configs)) * 1.1) |
| exec_time_s = sec_to_str(exec_time) |
| now_dt = datetime.datetime.now() |
| end_dt = now_dt + datetime.timedelta(0, exec_time) |
| msg = "Entire test should takes aroud: {0} and finished at {1}" |
| logger.info(msg.format(exec_time_s, |
| end_dt.strftime("%H:%M:%S"))) |
| |
| def run(self) -> IOTestResults: |
| logger.debug("Run preparation") |
| self.pre_run() |
| self.show_expected_execution_time() |
| num_nodes = len(self.config.nodes) |
| |
| tname = os.path.basename(self.config_fname) |
| if tname.endswith('.cfg'): |
| tname = tname[:-4] |
| |
| barrier = Barrier(num_nodes) |
| results = [] |
| |
| # set of Operation_Mode_BlockSize str's |
| # which should not be tested anymore, as |
| # they already too slow with previous thread count |
| lat_bw_limit_reached = set() |
| |
| with ThreadPoolExecutor(num_nodes) as pool: |
| for pos, fio_cfg in enumerate(self.fio_configs): |
| test_descr = get_test_summary(fio_cfg.vals, noqd=True) |
| if test_descr in lat_bw_limit_reached: |
| continue |
| |
| logger.info("Will run {} test".format(fio_cfg.name)) |
| templ = "Test should takes about {}. Should finish at {}, will wait at most till {}" |
| exec_time = execution_time(fio_cfg) |
| exec_time_str = sec_to_str(exec_time) |
| timeout = int(exec_time + max(300, exec_time)) |
| |
| now_dt = datetime.datetime.now() |
| end_dt = now_dt + datetime.timedelta(0, exec_time) |
| wait_till = now_dt + datetime.timedelta(0, timeout) |
| |
| logger.info(templ.format(exec_time_str, |
| end_dt.strftime("%H:%M:%S"), |
| wait_till.strftime("%H:%M:%S"))) |
| |
| run_test_func = functools.partial(self.do_run, |
| barrier=barrier, |
| fio_cfg=fio_cfg, |
| pos=pos) |
| |
| max_retr = 3 |
| for idx in range(max_retr): |
| if 0 != idx: |
| logger.info("Sleeping %ss and retrying", self.retry_time) |
| time.sleep(self.retry_time) |
| |
| try: |
| intervals = list(pool.map(run_test_func, self.config.nodes)) |
| if None not in intervals: |
| break |
| except (EnvironmentError, SSHException) as exc: |
| if max_retr - 1 == idx: |
| raise StopTestError("Fio failed") from exc |
| logger.exception("During fio run") |
| |
| fname = "{}_task.fio".format(pos) |
| with open(os.path.join(self.config.log_directory, fname), "w") as fd: |
| fd.write(str(fio_cfg)) |
| |
| params = {'vm_count': num_nodes} |
| params['name'] = fio_cfg.name |
| params['vals'] = dict(fio_cfg.vals.items()) |
| params['intervals'] = intervals |
| params['nodes'] = [node.get_conn_id() for node in self.config.nodes] |
| |
| fname = "{}_params.yaml".format(pos) |
| with open(os.path.join(self.config.log_directory, fname), "w") as fd: |
| fd.write(dumps(params)) |
| |
| res = load_test_results(self.config.log_directory, pos) |
| results.append(res) |
| |
| if self.max_latency is not None: |
| lat_50, _ = res.get_lat_perc_50_95_multy() |
| |
| # conver us to ms |
| if self.max_latency < lat_50: |
| logger.info(("Will skip all subsequent tests of {} " + |
| "due to lat/bw limits").format(fio_cfg.name)) |
| lat_bw_limit_reached.add(test_descr) |
| |
| test_res = res.get_params_from_fio_report() |
| if self.min_bw_per_thread is not None: |
| if self.min_bw_per_thread > average(test_res['bw']): |
| lat_bw_limit_reached.add(test_descr) |
| |
| return IOTestResults(self.config.params['cfg'], |
| results, self.config.log_directory) |
| |
| def do_run(self, node: INode, barrier: Barrier, fio_cfg, pos: int, nolog: bool=False): |
| exec_folder = self.config.remote_dir |
| |
| if self.use_system_fio: |
| fio_path = "" |
| else: |
| if not exec_folder.endswith("/"): |
| fio_path = exec_folder + "/" |
| else: |
| fio_path = exec_folder |
| |
| exec_time = execution_time(fio_cfg) |
| barrier.wait() |
| run_data = node.rpc.fio.run_fio(self.use_sudo, |
| fio_path, |
| exec_folder, |
| str(fio_cfg), |
| exec_time + max(300, exec_time)) |
| return parse_fio_result(run_data) |
| |
| @classmethod |
| def prepare_data(cls, results) -> List[Dict[str, Any]]: |
| """create a table with io performance report for console""" |
| |
| def key_func(data: FioRunResult) -> Tuple[str, str, str, str, int]: |
| tpl = data.summary_tpl() |
| return (data.name, |
| tpl.oper, |
| tpl.mode, |
| ssize2b(tpl.bsize), |
| int(tpl.th_count) * int(tpl.vm_count)) |
| res = [] |
| |
| for item in sorted(results, key=key_func): |
| test_dinfo = item.disk_perf_info() |
| testnodes_count = len(item.config.nodes) |
| |
| iops, _ = test_dinfo.iops.rounded_average_conf() |
| |
| if test_dinfo.iops_sys is not None: |
| iops_sys, iops_sys_conf = test_dinfo.iops_sys.rounded_average_conf() |
| _, iops_sys_dev = test_dinfo.iops_sys.rounded_average_dev() |
| iops_sys_per_vm = round_3_digit(iops_sys / testnodes_count) |
| iops_sys = round_3_digit(iops_sys) |
| else: |
| iops_sys = None |
| iops_sys_per_vm = None |
| iops_sys_dev = None |
| iops_sys_conf = None |
| |
| bw, bw_conf = test_dinfo.bw.rounded_average_conf() |
| _, bw_dev = test_dinfo.bw.rounded_average_dev() |
| conf_perc = int(round(bw_conf * 100 / bw)) |
| dev_perc = int(round(bw_dev * 100 / bw)) |
| |
| lat_50 = round_3_digit(int(test_dinfo.lat_50)) |
| lat_95 = round_3_digit(int(test_dinfo.lat_95)) |
| lat_avg = round_3_digit(int(test_dinfo.lat_avg)) |
| |
| iops_per_vm = round_3_digit(iops / testnodes_count) |
| bw_per_vm = round_3_digit(bw / testnodes_count) |
| |
| iops = round_3_digit(iops) |
| bw = round_3_digit(bw) |
| |
| summ = "{0.oper}{0.mode} {0.bsize:>4} {0.th_count:>3}th {0.vm_count:>2}vm".format(item.summary_tpl()) |
| |
| res.append({"name": key_func(item)[0], |
| "key": key_func(item)[:4], |
| "summ": summ, |
| "iops": int(iops), |
| "bw": int(bw), |
| "conf": str(conf_perc), |
| "dev": str(dev_perc), |
| "iops_per_vm": int(iops_per_vm), |
| "bw_per_vm": int(bw_per_vm), |
| "lat_50": lat_50, |
| "lat_95": lat_95, |
| "lat_avg": lat_avg, |
| |
| "iops_sys": iops_sys, |
| "iops_sys_per_vm": iops_sys_per_vm, |
| "sys_conf": iops_sys_conf, |
| "sys_dev": iops_sys_dev}) |
| |
| return res |
| |
| Field = collections.namedtuple("Field", ("header", "attr", "allign", "size")) |
| fiels_and_header = [ |
| Field("Name", "name", "l", 7), |
| Field("Description", "summ", "l", 19), |
| Field("IOPS\ncum", "iops", "r", 3), |
| # Field("IOPS_sys\ncum", "iops_sys", "r", 3), |
| Field("KiBps\ncum", "bw", "r", 6), |
| Field("Cnf %\n95%", "conf", "r", 3), |
| Field("Dev%", "dev", "r", 3), |
| Field("iops\n/vm", "iops_per_vm", "r", 3), |
| Field("KiBps\n/vm", "bw_per_vm", "r", 6), |
| Field("lat ms\nmedian", "lat_50", "r", 3), |
| Field("lat ms\n95%", "lat_95", "r", 3), |
| Field("lat\navg", "lat_avg", "r", 3), |
| ] |
| |
| fiels_and_header_dct = dict((item.attr, item) for item in fiels_and_header) |
| |
| @classmethod |
| def format_for_console(cls, results) -> str: |
| """create a table with io performance report for console""" |
| |
| tab = texttable.Texttable(max_width=120) |
| tab.set_deco(tab.HEADER | tab.VLINES | tab.BORDER) |
| tab.set_cols_align([f.allign for f in cls.fiels_and_header]) |
| sep = ["-" * f.size for f in cls.fiels_and_header] |
| tab.header([f.header for f in cls.fiels_and_header]) |
| prev_k = None |
| for item in cls.prepare_data(results): |
| if prev_k is not None: |
| if prev_k != item["key"]: |
| tab.add_row(sep) |
| |
| prev_k = item["key"] |
| tab.add_row([item[f.attr] for f in cls.fiels_and_header]) |
| |
| return tab.draw() |
| |
| @classmethod |
| def format_diff_for_console(cls, list_of_results: List[Any]) -> str: |
| """create a table with io performance report for console""" |
| |
| tab = texttable.Texttable(max_width=200) |
| tab.set_deco(tab.HEADER | tab.VLINES | tab.BORDER) |
| |
| header = [ |
| cls.fiels_and_header_dct["name"].header, |
| cls.fiels_and_header_dct["summ"].header, |
| ] |
| allign = ["l", "l"] |
| |
| header.append("IOPS ~ Cnf% ~ Dev%") |
| allign.extend(["r"] * len(list_of_results)) |
| header.extend( |
| "IOPS_{0} %".format(i + 2) for i in range(len(list_of_results[1:])) |
| ) |
| |
| header.append("BW") |
| allign.extend(["r"] * len(list_of_results)) |
| header.extend( |
| "BW_{0} %".format(i + 2) for i in range(len(list_of_results[1:])) |
| ) |
| |
| header.append("LAT") |
| allign.extend(["r"] * len(list_of_results)) |
| header.extend( |
| "LAT_{0}".format(i + 2) for i in range(len(list_of_results[1:])) |
| ) |
| |
| tab.header(header) |
| sep = ["-" * 3] * len(header) |
| processed_results = map(cls.prepare_data, list_of_results) |
| |
| key2results = [] |
| for res in processed_results: |
| key2results.append(dict( |
| ((item["name"], item["summ"]), item) for item in res |
| )) |
| |
| prev_k = None |
| iops_frmt = "{0[iops]} ~ {0[conf]:>2} ~ {0[dev]:>2}" |
| for item in processed_results[0]: |
| if prev_k is not None: |
| if prev_k != item["key"]: |
| tab.add_row(sep) |
| |
| prev_k = item["key"] |
| |
| key = (item['name'], item['summ']) |
| line = list(key) |
| base = key2results[0][key] |
| |
| line.append(iops_frmt.format(base)) |
| |
| for test_results in key2results[1:]: |
| val = test_results.get(key) |
| if val is None: |
| line.append("-") |
| elif base['iops'] == 0: |
| line.append("Nan") |
| else: |
| prc_val = {'dev': val['dev'], 'conf': val['conf']} |
| prc_val['iops'] = int(100 * val['iops'] / base['iops']) |
| line.append(iops_frmt.format(prc_val)) |
| |
| line.append(base['bw']) |
| |
| for test_results in key2results[1:]: |
| val = test_results.get(key) |
| if val is None: |
| line.append("-") |
| elif base['bw'] == 0: |
| line.append("Nan") |
| else: |
| line.append(int(100 * val['bw'] / base['bw'])) |
| |
| for test_results in key2results: |
| val = test_results.get(key) |
| if val is None: |
| line.append("-") |
| else: |
| line.append("{0[lat_50]} - {0[lat_95]}".format(val)) |
| |
| tab.add_row(line) |
| |
| tab.set_cols_align(allign) |
| return tab.draw() |