koder aka kdanilov | f90de85 | 2017-01-20 18:12:27 +0200 | [diff] [blame] | 1 | import abc |
kdanylov aka koder | 13e5845 | 2018-07-15 02:51:51 +0300 | [diff] [blame] | 2 | from typing import Dict, Any, Tuple, cast, Union, NamedTuple |
koder aka kdanilov | f90de85 | 2017-01-20 18:12:27 +0200 | [diff] [blame] | 3 | from collections import OrderedDict |
| 4 | |
kdanylov aka koder | 026e5f2 | 2017-05-15 01:04:39 +0300 | [diff] [blame] | 5 | from cephlib.istorage import Storable |
koder aka kdanilov | f90de85 | 2017-01-20 18:12:27 +0200 | [diff] [blame] | 6 | |
| 7 | |
kdanylov aka koder | 13e5845 | 2018-07-15 02:51:51 +0300 | [diff] [blame] | 8 | Var = NamedTuple('Var', [('name', str)]) |
| 9 | |
| 10 | |
koder aka kdanilov | f90de85 | 2017-01-20 18:12:27 +0200 | [diff] [blame] | 11 | class JobParams(metaclass=abc.ABCMeta): |
| 12 | """Class contains all job parameters, which significantly affects job results. |
| 13 | Like block size or operation type, but not file name or file size. |
| 14 | Can be used as key in dictionary |
| 15 | """ |
| 16 | |
| 17 | def __init__(self, **params: Dict[str, Any]) -> None: |
| 18 | self.params = params |
| 19 | |
kdanylov aka koder | 150b219 | 2017-04-01 16:53:01 +0300 | [diff] [blame] | 20 | @property |
| 21 | @abc.abstractmethod |
koder aka kdanilov | f90de85 | 2017-01-20 18:12:27 +0200 | [diff] [blame] | 22 | def summary(self) -> str: |
| 23 | """Test short summary, used mostly for file names and short image description""" |
| 24 | pass |
| 25 | |
kdanylov aka koder | 150b219 | 2017-04-01 16:53:01 +0300 | [diff] [blame] | 26 | @property |
| 27 | @abc.abstractmethod |
koder aka kdanilov | f90de85 | 2017-01-20 18:12:27 +0200 | [diff] [blame] | 28 | def long_summary(self) -> str: |
| 29 | """Readable long summary for management and deployment engineers""" |
| 30 | pass |
| 31 | |
koder aka kdanilov | a732a60 | 2017-02-01 20:29:56 +0200 | [diff] [blame] | 32 | @abc.abstractmethod |
| 33 | def copy(self, **updated) -> 'JobParams': |
| 34 | pass |
| 35 | |
koder aka kdanilov | f90de85 | 2017-01-20 18:12:27 +0200 | [diff] [blame] | 36 | def __getitem__(self, name: str) -> Any: |
| 37 | return self.params[name] |
| 38 | |
| 39 | def __setitem__(self, name: str, val: Any) -> None: |
| 40 | self.params[name] = val |
| 41 | |
| 42 | def __hash__(self) -> int: |
koder aka kdanilov | a732a60 | 2017-02-01 20:29:56 +0200 | [diff] [blame] | 43 | return hash(self.char_tpl) |
koder aka kdanilov | f90de85 | 2017-01-20 18:12:27 +0200 | [diff] [blame] | 44 | |
koder aka kdanilov | a732a60 | 2017-02-01 20:29:56 +0200 | [diff] [blame] | 45 | def __eq__(self, o: object) -> bool: |
| 46 | if not isinstance(o, self.__class__): |
kdanylov aka koder | 13e5845 | 2018-07-15 02:51:51 +0300 | [diff] [blame] | 47 | raise TypeError(f"Can't compare {self.__class__.__qualname__!r} to {type(o).__qualname__!r}") |
koder aka kdanilov | a732a60 | 2017-02-01 20:29:56 +0200 | [diff] [blame] | 48 | return sorted(self.params.items()) == sorted(cast(JobParams, o).params.items()) |
| 49 | |
| 50 | def __lt__(self, o: object) -> bool: |
| 51 | if not isinstance(o, self.__class__): |
kdanylov aka koder | 13e5845 | 2018-07-15 02:51:51 +0300 | [diff] [blame] | 52 | raise TypeError(f"Can't compare {self.__class__.__qualname__!r} to {type(o).__qualname__!r}") |
koder aka kdanilov | a732a60 | 2017-02-01 20:29:56 +0200 | [diff] [blame] | 53 | return self.char_tpl < cast(JobParams, o).char_tpl |
| 54 | |
kdanylov aka koder | 150b219 | 2017-04-01 16:53:01 +0300 | [diff] [blame] | 55 | @property |
| 56 | @abc.abstractmethod |
koder aka kdanilov | a732a60 | 2017-02-01 20:29:56 +0200 | [diff] [blame] | 57 | def char_tpl(self) -> Tuple[Union[str, int, float, bool], ...]: |
| 58 | pass |
koder aka kdanilov | f90de85 | 2017-01-20 18:12:27 +0200 | [diff] [blame] | 59 | |
| 60 | |
| 61 | class JobConfig(Storable, metaclass=abc.ABCMeta): |
| 62 | """Job config class""" |
| 63 | |
| 64 | def __init__(self, idx: int) -> None: |
| 65 | # job id, used in storage to distinct jobs with same summary |
| 66 | self.idx = idx |
| 67 | |
| 68 | # time interval, in seconds, when test was running on all nodes |
kdanylov aka koder | 13e5845 | 2018-07-15 02:51:51 +0300 | [diff] [blame] | 69 | self.reliable_info_range: Tuple[int, int] = None # type: ignore |
kdanylov aka koder | 4518318 | 2017-04-30 23:55:40 +0300 | [diff] [blame] | 70 | |
koder aka kdanilov | f90de85 | 2017-01-20 18:12:27 +0200 | [diff] [blame] | 71 | # all job parameters, both from suite file and config file |
kdanylov aka koder | 13e5845 | 2018-07-15 02:51:51 +0300 | [diff] [blame] | 72 | self.vals: Dict[str, Any] = OrderedDict() |
koder aka kdanilov | f90de85 | 2017-01-20 18:12:27 +0200 | [diff] [blame] | 73 | |
| 74 | @property |
kdanylov aka koder | 4518318 | 2017-04-30 23:55:40 +0300 | [diff] [blame] | 75 | def reliable_info_range_s(self) -> Tuple[int, int]: |
| 76 | return (self.reliable_info_range[0] // 1000, self.reliable_info_range[1] // 1000) |
| 77 | |
| 78 | @property |
koder aka kdanilov | f90de85 | 2017-01-20 18:12:27 +0200 | [diff] [blame] | 79 | def storage_id(self) -> str: |
| 80 | """unique string, used as key in storage""" |
kdanylov aka koder | 13e5845 | 2018-07-15 02:51:51 +0300 | [diff] [blame] | 81 | return f"{self.summary}_{self.idx}" |
koder aka kdanilov | f90de85 | 2017-01-20 18:12:27 +0200 | [diff] [blame] | 82 | |
kdanylov aka koder | 150b219 | 2017-04-01 16:53:01 +0300 | [diff] [blame] | 83 | @property |
| 84 | @abc.abstractmethod |
koder aka kdanilov | f90de85 | 2017-01-20 18:12:27 +0200 | [diff] [blame] | 85 | def params(self) -> JobParams: |
| 86 | """Should return a copy""" |
| 87 | pass |
koder aka kdanilov | a732a60 | 2017-02-01 20:29:56 +0200 | [diff] [blame] | 88 | |
| 89 | @property |
| 90 | def summary(self) -> str: |
| 91 | return self.params.summary |