-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathpluginMeasureCpu.py
124 lines (97 loc) · 3.38 KB
/
pluginMeasureCpu.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
114
115
116
117
118
119
120
121
122
123
124
import logging
import jc
import typing
from typing import Any
from typing import Optional
from ktoolbox import common
import task
import pluginbase
import tftbase
from task import PluginTask
from task import TaskOperation
from testSettings import TestSettings
from tftbase import BaseOutput
from tftbase import PluginOutput
from tftbase import TaskRole
logger = logging.getLogger("tft." + __name__)
class PluginMeasureCpu(pluginbase.Plugin):
PLUGIN_NAME = "measure_cpu"
def _enable(
self,
*,
ts: TestSettings,
perf_server: task.ServerTask,
perf_client: task.ClientTask,
tenant: bool,
) -> list[PluginTask]:
return [
TaskMeasureCPU(ts, TaskRole.SERVER, tenant),
TaskMeasureCPU(ts, TaskRole.CLIENT, tenant),
]
plugin = pluginbase.register_plugin(PluginMeasureCpu())
class TaskMeasureCPU(PluginTask):
@property
def plugin(self) -> pluginbase.Plugin:
return plugin
def __init__(self, ts: TestSettings, task_role: TaskRole, tenant: bool):
super().__init__(
ts=ts,
index=0,
task_role=task_role,
tenant=tenant,
)
self.in_file_template = tftbase.get_manifest("tools-pod.yaml.j2")
self.out_file_yaml = tftbase.get_manifest_renderpath(
f"tools-pod-{self.node_name}-measure-cpu.yaml"
)
self.pod_name = f"tools-pod-{self.node_name}-measure-cpu"
def initialize(self) -> None:
super().initialize()
self.render_file("Server Pod Yaml")
def _create_task_operation(self) -> TaskOperation:
def _thread_action() -> BaseOutput:
self.ts.clmo_barrier.wait()
cmd = f"mpstat -P ALL {self.get_duration()} 1"
r = self.run_oc_exec(cmd)
success = True
msg: Optional[str] = None
result: dict[str, Any] = {}
if not r.success:
success = False
msg = r.debug_msg()
if success:
try:
lst = typing.cast(list[dict[str, Any]], jc.parse("mpstat", r.out))
rdict = lst[0]
except Exception:
success = False
msg = f'Output of "{cmd}" cannot be parsed: {r.debug_msg()}'
if success:
if (
isinstance(rdict, dict)
and all(isinstance(k, str) for k in rdict)
and all(required_key in rdict for required_key in ("percent_idle",))
):
result = rdict
else:
success = False
msg = 'Output of "{cmd}" contains unexpected data: {r.debug_msg()}'
result["cmd"] = common.dataclass_to_dict(r)
return PluginOutput(
success=success,
msg=msg,
plugin_metadata=self.get_plugin_metadata(),
command=cmd,
result=result,
)
return TaskOperation(
log_name=self.log_name,
thread_action=_thread_action,
)
def _aggregate_output_log_success(
self,
result: tftbase.AggregatableOutput,
) -> None:
assert isinstance(result, PluginOutput)
p_idle = result.result["percent_idle"]
logger.info(f"Idle on {self.node_name} = {p_idle}%")