-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathexternal.py
127 lines (102 loc) · 4.03 KB
/
external.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
125
126
127
import os
import subprocess
from typing import Iterator
from ..settings import SETTINGS
from ._report import Report, ReportData
def _hwdata() -> Iterator[ReportData]:
def parse_output(exe: os.PathLike) -> ReportData:
try:
# check exe is executable
# => AssertionError
assert os.access(exe, os.X_OK)
try:
# run exe
# => TimeoutExpired: execution took too long
execution = subprocess.run(
args=[exe],
stdout=subprocess.PIPE,
timeout=SETTINGS.external.timeout,
)
stdout = execution.stdout
returncode = execution.returncode
except subprocess.TimeoutExpired as e:
# output might still be valid
assert (stdout := e.stdout) is not None
returncode = 0
# look at the first four output lines
# => UnicodeDecodeError: output is not decodable
output = stdout.decode().split("\n")[:4]
# extract and check name (fail if empty)
# => IndexError, AssertionError
assert (
name := "".join(char for char in output[0] if char.isprintable())[:100]
) != ""
# check exit status
# => AssertionError
assert returncode == 0
except (AssertionError, UnicodeDecodeError, IndexError):
return ReportData.from_settings(
name=os.path.basename(exe)[:100],
value=100,
settings=SETTINGS.external,
)
try:
# check output length
# => AssertionError
assert len(output) == 4
# extract threshold and value
# => ValueError
threshold = float(output[1])
value = float(output[3])
# extract and check inversion
# => AssertionError
assert (inverted := output[2].strip().lower()) in (
"normal",
"inverted",
)
except (AssertionError, ValueError):
return ReportData.from_settings(
name=name,
value=100,
settings=SETTINGS.external,
)
# success
return ReportData(
name=name,
value=value,
threshold=threshold,
inverted=inverted == "inverted",
format=SETTINGS.external.report,
)
yield from (parse_output(exe) for exe in SETTINGS.external.executables)
def external() -> Report | None:
"""
External Metric
=====
This metric's values are defined external executables (e.g. shell scripts).
Any executable with suitable output can be used as a value for this metric.
To comply, the executable's output must be UTF-8 decodable and start with
four consecutive lines holding the following information:
1. value name
2. percent threshold
3. the string "normal" or "inverted", without quotes
4. percent current value
The executable may produce additional output, which will be ignored.
Percentages may be floating point numbers and must use a decimal point "."
as a separator in that case.
The output is evaluated once execution finishes with exit status 0.
A report is generated for each executable. Its value name is stripped of
non-printable characters and limited to a length of 100.
Non-compliance will be reported as failed values, i.e. normal values with a
threshold of 0% and a value of 100%, in these cases:
- non-executable files and executables outputting non-UTF8:
reported as the files' basename
- executables with generally noncompliant outputs:
reported as the first line of output
- failure to parse any of the threshold, inversion or current value
- otherwise compliant executables with non-zero exit status
"""
return Report.aggregate(
settings=SETTINGS.external,
get_data=_hwdata,
)