Commit 9ea588d5 authored by Matthias Liedtke's avatar Matthias Liedtke Committed by V8 LUCI CQ

[test] testrunner: Print diff if stdout does not match

This extends the ExpectedOutProc runner to print a diff between the
expected and actual output. The behavior of other runners is unchanged.

Change-Id: If2b89d39cf98b8d257b1a209b5471a79ec3868ff
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3771641Reviewed-by: 's avatarJakob Kummerow <jkummerow@chromium.org>
Reviewed-by: 's avatarLiviu Rau <liviurau@chromium.org>
Commit-Queue: Clemens Backes <clemensb@chromium.org>
Auto-Submit: Matthias Liedtke <mliedtke@google.com>
Reviewed-by: 's avatarClemens Backes <clemensb@chromium.org>
Cr-Commit-Position: refs/heads/main@{#81966}
parent 1380ffd6
...@@ -9,6 +9,7 @@ from ..testproc.base import ( ...@@ -9,6 +9,7 @@ from ..testproc.base import (
from ..local import statusfile from ..local import statusfile
from ..testproc.result import Result from ..testproc.result import Result
import difflib
OUTCOMES_PASS = [statusfile.PASS] OUTCOMES_PASS = [statusfile.PASS]
OUTCOMES_FAIL = [statusfile.FAIL] OUTCOMES_FAIL = [statusfile.FAIL]
...@@ -38,15 +39,17 @@ class BaseOutProc(object): ...@@ -38,15 +39,17 @@ class BaseOutProc(object):
""" """
if reduction == DROP_RESULT: if reduction == DROP_RESULT:
return None return None
error_details = \
self._get_error_details(output) if has_unexpected_output else None
if reduction == DROP_OUTPUT: if reduction == DROP_OUTPUT:
return Result(has_unexpected_output, None) return Result(has_unexpected_output, None, error_details=error_details)
if not has_unexpected_output: if not has_unexpected_output:
if reduction == DROP_PASS_OUTPUT: if reduction == DROP_PASS_OUTPUT:
return Result(has_unexpected_output, None) return Result(has_unexpected_output, None)
if reduction == DROP_PASS_STDOUT: if reduction == DROP_PASS_STDOUT:
return Result(has_unexpected_output, output.without_text()) return Result(has_unexpected_output, output.without_text())
return Result(has_unexpected_output, output) return Result(has_unexpected_output, output, error_details=error_details)
def get_outcome(self, output): def get_outcome(self, output):
if output.HasCrashed(): if output.HasCrashed():
...@@ -67,6 +70,9 @@ class BaseOutProc(object): ...@@ -67,6 +70,9 @@ class BaseOutProc(object):
def _is_failure_output(self, output): def _is_failure_output(self, output):
return output.exit_code != 0 return output.exit_code != 0
def _get_error_details(self, output):
return None
@property @property
def negative(self): def negative(self):
return False return False
...@@ -163,6 +169,16 @@ class ExpectedOutProc(OutProc): ...@@ -163,6 +169,16 @@ class ExpectedOutProc(OutProc):
for _, line in enumerate(lines): for _, line in enumerate(lines):
f.write(line+'\n') f.write(line+'\n')
def _get_error_details(self, output):
"""Return diff between expected and actual output."""
expected = open(self._expected_filename, 'r', encoding='utf-8').readlines()
actual = output.stdout.splitlines(True)
if expected == actual:
return None
lines = difflib.unified_diff(
expected, actual, fromfile=self._expected_filename, tofile='<actual>')
return 'Output does not match expectation:\n' + ''.join(lines)
def _act_block_iterator(self, output): def _act_block_iterator(self, output):
"""Iterates over blocks of actual output lines.""" """Iterates over blocks of actual output lines."""
lines = output.stdout.splitlines() lines = output.stdout.splitlines()
......
...@@ -420,6 +420,18 @@ class StandardRunnerTest(TestRunnerTest): ...@@ -420,6 +420,18 @@ class StandardRunnerTest(TestRunnerTest):
result.stdout_includes('sweet/bananas') result.stdout_includes('sweet/bananas')
result.has_returncode(1) result.has_returncode(1)
def testCompactErrorDetails(self):
result = self.run_tests(
'--progress=mono',
'sweet/cherries',
'sweet/strawberries',
infra_staging=False,
)
result.stdout_includes('sweet/cherries')
result.stdout_includes('sweet/strawberries')
result.stdout_includes('+Mock diff')
result.has_returncode(1)
def testExitAfterNFailures(self): def testExitAfterNFailures(self):
result = self.run_tests( result = self.run_tests(
'--progress=verbose', '--progress=verbose',
......
...@@ -4,6 +4,7 @@ ...@@ -4,6 +4,7 @@
{ {
"command": "/usr/bin/python out/build/d8_mocked.py --test strawberries --random-seed=123 --nohard-abort --testing-d8-test-runner", "command": "/usr/bin/python out/build/d8_mocked.py --test strawberries --random-seed=123 --nohard-abort --testing-d8-test-runner",
"duration": 1, "duration": 1,
"error_details": "+Mock diff",
"exit_code": 1, "exit_code": 1,
"expected": [ "expected": [
"PASS" "PASS"
...@@ -29,6 +30,7 @@ ...@@ -29,6 +30,7 @@
{ {
"command": "/usr/bin/python out/build/d8_mocked.py --test strawberries --random-seed=123 --nohard-abort --testing-d8-test-runner", "command": "/usr/bin/python out/build/d8_mocked.py --test strawberries --random-seed=123 --nohard-abort --testing-d8-test-runner",
"duration": 1, "duration": 1,
"error_details": "+Mock diff",
"exit_code": 1, "exit_code": 1,
"expected": [ "expected": [
"PASS" "PASS"
...@@ -54,6 +56,7 @@ ...@@ -54,6 +56,7 @@
{ {
"command": "/usr/bin/python out/build/d8_mocked.py --test strawberries --random-seed=123 --nohard-abort --testing-d8-test-runner", "command": "/usr/bin/python out/build/d8_mocked.py --test strawberries --random-seed=123 --nohard-abort --testing-d8-test-runner",
"duration": 1, "duration": 1,
"error_details": "+Mock diff",
"exit_code": 1, "exit_code": 1,
"expected": [ "expected": [
"PASS" "PASS"
......
...@@ -4,6 +4,7 @@ ...@@ -4,6 +4,7 @@
{ {
"command": "/usr/bin/python out/build/d8_mocked.py bananaflakes --random-seed=123 --nohard-abort --testing-d8-test-runner", "command": "/usr/bin/python out/build/d8_mocked.py bananaflakes --random-seed=123 --nohard-abort --testing-d8-test-runner",
"duration": 1, "duration": 1,
"error_details": null,
"exit_code": 1, "exit_code": 1,
"expected": [ "expected": [
"PASS" "PASS"
...@@ -28,6 +29,7 @@ ...@@ -28,6 +29,7 @@
{ {
"command": "/usr/bin/python out/build/d8_mocked.py bananaflakes --random-seed=123 --nohard-abort --testing-d8-test-runner", "command": "/usr/bin/python out/build/d8_mocked.py bananaflakes --random-seed=123 --nohard-abort --testing-d8-test-runner",
"duration": 1, "duration": 1,
"error_details": null,
"exit_code": 0, "exit_code": 0,
"expected": [ "expected": [
"PASS" "PASS"
......
...@@ -8,6 +8,8 @@ Dummy test suite extension with some fruity tests. ...@@ -8,6 +8,8 @@ Dummy test suite extension with some fruity tests.
from testrunner.local import testsuite from testrunner.local import testsuite
from testrunner.objects import testcase from testrunner.objects import testcase
from testrunner.outproc.base import OutProc
class TestLoader(testsuite.TestLoader): class TestLoader(testsuite.TestLoader):
def _list_test_filenames(self): def _list_test_filenames(self):
...@@ -25,9 +27,25 @@ class TestSuite(testsuite.TestSuite): ...@@ -25,9 +27,25 @@ class TestSuite(testsuite.TestSuite):
return TestCase return TestCase
class MockOutProc(OutProc):
def __init__(self, expected_outcomes):
OutProc.__init__(self, expected_outcomes)
def _get_error_details(self, output):
return "+Mock diff"
class TestCase(testcase.D8TestCase): class TestCase(testcase.D8TestCase):
def get_shell(self): def get_shell(self):
return 'd8_mocked.py' return 'd8_mocked.py'
def _get_files_params(self): def _get_files_params(self):
return [self.name] return [self.name]
@property
def output_proc(self):
if self.name == 'strawberries':
return MockOutProc([])
else:
return super(TestCase, self).output_proc
...@@ -239,6 +239,8 @@ class CompactProgressIndicator(ProgressIndicator): ...@@ -239,6 +239,8 @@ class CompactProgressIndicator(ProgressIndicator):
self.printFormatted('stdout', stdout) self.printFormatted('stdout', stdout)
if len(stderr): if len(stderr):
self.printFormatted('stderr', stderr) self.printFormatted('stderr', stderr)
if result.error_details:
self.printFormatted('failure', result.error_details)
self.printFormatted('command', self.printFormatted('command',
"Command: %s" % result.cmd.to_string(relative=True)) "Command: %s" % result.cmd.to_string(relative=True))
if output.HasCrashed(): if output.HasCrashed():
...@@ -374,6 +376,7 @@ class JsonTestProgressIndicator(ProgressIndicator): ...@@ -374,6 +376,7 @@ class JsonTestProgressIndicator(ProgressIndicator):
"result": test.output_proc.get_outcome(output), "result": test.output_proc.get_outcome(output),
"stdout": output.stdout, "stdout": output.stdout,
"stderr": output.stderr, "stderr": output.stderr,
"error_details": result.error_details,
}) })
self.results.append(record) self.results.append(record)
......
...@@ -20,10 +20,15 @@ class ResultBase(object): ...@@ -20,10 +20,15 @@ class ResultBase(object):
class Result(ResultBase): class Result(ResultBase):
"""Result created by the output processor.""" """Result created by the output processor."""
def __init__(self, has_unexpected_output, output, cmd=None): def __init__(self,
has_unexpected_output,
output,
cmd=None,
error_details=None):
self.has_unexpected_output = has_unexpected_output self.has_unexpected_output = has_unexpected_output
self.output = output self.output = output
self.cmd = cmd self.cmd = cmd
self.error_details = error_details
def status(self): def status(self):
if self.has_unexpected_output: if self.has_unexpected_output:
......
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