Commit c65119d5 authored by Victor Stinner's avatar Victor Stinner Committed by GitHub

bpo-36670: Enhance regrtest WindowsLoadTracker (GH-16553)

The last line is now passed to the parser even if it does not end
with a newline, but only if it's a valid value.
parent 3e04cd26
...@@ -14,7 +14,7 @@ BUFSIZE = 8192 ...@@ -14,7 +14,7 @@ BUFSIZE = 8192
LOAD_FACTOR_1 = 0.9200444146293232478931553241 LOAD_FACTOR_1 = 0.9200444146293232478931553241
# Seconds per measurement # Seconds per measurement
SAMPLING_INTERVAL = 5 SAMPLING_INTERVAL = 1
# Windows registry subkey of HKEY_LOCAL_MACHINE where the counter names # Windows registry subkey of HKEY_LOCAL_MACHINE where the counter names
# of typeperf are registered # of typeperf are registered
COUNTER_REGISTRY_KEY = (r"SOFTWARE\Microsoft\Windows NT\CurrentVersion" COUNTER_REGISTRY_KEY = (r"SOFTWARE\Microsoft\Windows NT\CurrentVersion"
...@@ -32,7 +32,7 @@ class WindowsLoadTracker(): ...@@ -32,7 +32,7 @@ class WindowsLoadTracker():
def __init__(self): def __init__(self):
self.load = 0.0 self.load = 0.0
self.counter_name = '' self.counter_name = ''
self._buffer = b'' self._buffer = ''
self.popen = None self.popen = None
self.start() self.start()
...@@ -95,44 +95,60 @@ class WindowsLoadTracker(): ...@@ -95,44 +95,60 @@ class WindowsLoadTracker():
def __del__(self): def __del__(self):
self.close() self.close()
def read_output(self): def _parse_line(self, line):
# typeperf outputs in a CSV format like this:
# "07/19/2018 01:32:26.605","3.000000"
# (date, process queue length)
tokens = line.split(',')
if len(tokens) != 2:
raise ValueError
value = tokens[1]
if not value.startswith('"') or not value.endswith('"'):
raise ValueError
value = value[1:-1]
return float(value)
def read_lines(self):
overlapped, _ = _winapi.ReadFile(self.pipe, BUFSIZE, True) overlapped, _ = _winapi.ReadFile(self.pipe, BUFSIZE, True)
bytes_read, res = overlapped.GetOverlappedResult(False) bytes_read, res = overlapped.GetOverlappedResult(False)
if res != 0: if res != 0:
return return ()
# self._buffer stores an incomplete line output = overlapped.getbuffer()
output = self._buffer + overlapped.getbuffer() output = output.decode('oem', 'replace')
output, _, self._buffer = output.rpartition(b'\n') output = self._buffer + output
return output.decode('oem', 'replace') lines = output.splitlines(True)
# bpo-36670: typeperf only writes a newline *before* writing a value,
# not after. Sometimes, the written line in incomplete (ex: only
# timestamp, without the process queue length). Only pass the last line
# to the parser if it's a valid value, otherwise store it in
# self._buffer.
try:
self._parse_line(lines[-1])
except ValueError:
self._buffer = lines.pop(-1)
else:
self._buffer = ''
return lines
def getloadavg(self): def getloadavg(self):
typeperf_output = self.read_output() for line in self.read_lines():
# Nothing to update, just return the current load line = line.rstrip()
if not typeperf_output:
return self.load
# Process the backlog of load values
for line in typeperf_output.splitlines():
# Ignore the initial header: # Ignore the initial header:
# "(PDH-CSV 4.0)","\\\\WIN\\System\\Processor Queue Length" # "(PDH-CSV 4.0)","\\\\WIN\\System\\Processor Queue Length"
if '\\\\' in line: if 'PDH-CSV' in line:
continue continue
# Ignore blank lines # Ignore blank lines
if not line.strip(): if not line:
continue continue
# typeperf outputs in a CSV format like this:
# "07/19/2018 01:32:26.605","3.000000"
# (date, process queue length)
try: try:
tokens = line.split(',') load = self._parse_line(line)
if len(tokens) != 2:
raise ValueError
value = tokens[1].replace('"', '')
load = float(value)
except ValueError: except ValueError:
print_warning("Failed to parse typeperf output: %a" % line) print_warning("Failed to parse typeperf output: %a" % line)
continue continue
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment