Skip to content

Commit d0619c7

Browse files
pspaceknicki-krizek
authored andcommitted
Enable live logging for non-parallel pytest runs
This provides incremental output when test is running _without xdist_, just like the old runner did. With xdist the live output is not available, I believe because of pytest-dev/pytest-xdist#402 pytest-dev/pytest-xdist#883 might help with that, but I'm not going to hold my breath until it is available on distros we use.
1 parent 8f57bce commit d0619c7

File tree

1 file changed

+33
-20
lines changed

1 file changed

+33
-20
lines changed

bin/tests/system/conftest.py

+33-20
Original file line numberDiff line numberDiff line change
@@ -90,6 +90,21 @@ def control_port():
9090

9191
# ---------------------- Module initialization ---------------------------
9292

93+
def avoid_duplicated_logs():
94+
"""
95+
Remove direct root logger output to file descriptors.
96+
This default is causing duplicates because all our messages go through
97+
regular logging as well and are thus displayed twice.
98+
"""
99+
todel = []
100+
for handler in logging.root.handlers:
101+
if handler.__class__ == logging.StreamHandler:
102+
# Beware: As for pytest 7.2.2, LiveLogging and LogCapture
103+
# handlers inherit from logging.StreamHandler
104+
todel.append(handler)
105+
for handler in todel:
106+
logging.root.handlers.remove(handler)
107+
93108
def parse_env(env_text):
94109
"""Parse the POSIX env format into Python dictionary."""
95110
out = {}
@@ -283,6 +298,7 @@ def system_test_name(request):
283298
@pytest.fixture(scope="module")
284299
def logger(system_test_name):
285300
"""Logging facility specific to this test."""
301+
avoid_duplicated_logs()
286302
return logging.getLogger(system_test_name)
287303

288304
@pytest.fixture(scope="module")
@@ -378,29 +394,26 @@ def _run_script( # pylint: disable=too-many-arguments
378394
raise FileNotFoundError(f"script {script} not found in {cwd}")
379395
logger.debug("running script: %s %s %s", interpreter, script, " ".join(args))
380396
logger.debug(" workdir: %s", cwd)
381-
stdout = b""
382397
returncode = 1
383-
try:
384-
proc = subprocess.run(
385-
[interpreter, script] + args,
386-
env=env,
387-
check=True,
388-
stdout=subprocess.PIPE,
389-
stderr=subprocess.STDOUT,
390-
)
391-
except subprocess.CalledProcessError as exc:
392-
stdout = exc.stdout
393-
returncode = exc.returncode
394-
raise exc
395-
else:
396-
stdout = proc.stdout
398+
399+
cmd = [interpreter, script] + args
400+
with subprocess.Popen(
401+
cmd,
402+
env=env,
403+
stdout=subprocess.PIPE,
404+
stderr=subprocess.STDOUT,
405+
bufsize=1,
406+
universal_newlines=True,
407+
errors="backslashreplace",
408+
) as proc:
409+
if proc.stdout:
410+
for line in proc.stdout:
411+
logger.info(" %s", line.rstrip("\n"))
412+
proc.communicate()
397413
returncode = proc.returncode
398-
finally:
399-
if stdout:
400-
for line in stdout.decode().splitlines():
401-
logger.debug(" %s", line)
414+
if returncode:
415+
raise subprocess.CalledProcessError(returncode, cmd)
402416
logger.debug(" exited with %d", returncode)
403-
return proc
404417

405418
@pytest.fixture(scope="module")
406419
def shell(env, system_test_dir, logger):

0 commit comments

Comments
 (0)