Commit bddfee98 authored by Michal Majewski's avatar Michal Majewski Committed by Commit Bot

[test] Add special result for rerun and json indicator.

Bug: v8:6917
Change-Id: I5136f183bd1728a1ab90a9ebb2560d978e17ef28
Cq-Include-Trybots: luci.v8.try:v8_linux64_fyi_rel_ng
Reviewed-on: https://chromium-review.googlesource.com/863623
Commit-Queue: Michał Majewski <majeski@google.com>
Reviewed-by: 's avatarMichael Achenbach <machenbach@chromium.org>
Cr-Commit-Position: refs/heads/master@{#50576}
parent 43ac9d51
...@@ -295,35 +295,42 @@ class JsonTestProgressIndicator(ProgressIndicator): ...@@ -295,35 +295,42 @@ class JsonTestProgressIndicator(ProgressIndicator):
self.tests = [] self.tests = []
def _on_result_for(self, test, result): def _on_result_for(self, test, result):
# TODO(majeski): Support for dummy/grouped results if result.is_rerun:
output = result.output self.process_results(test, result.results)
# Buffer all tests for sorting the durations in the end. else:
self.tests.append((test, output.duration)) self.process_results(test, [result])
# TODO(majeski): Previously we included reruns here. If we still want this def process_results(self, test, results):
# json progress indicator should be placed just before execution. for run, result in enumerate(results):
if not result.has_unexpected_output: # TODO(majeski): Support for dummy/grouped results
# Omit tests that run as expected. output = result.output
return # Buffer all tests for sorting the durations in the end.
self.tests.append((test, output.duration))
self.results.append({
"name": str(test), # Omit tests that run as expected on the first try.
"flags": test.cmd.args, # Everything that happens after the first run is included in the output
"command": test.cmd.to_string(relative=True), # even if it flakily passes.
"run": -100, # TODO(majeski): do we need this? if not result.has_unexpected_output and run == 0:
"stdout": output.stdout, continue
"stderr": output.stderr,
"exit_code": output.exit_code, self.results.append({
"result": test.output_proc.get_outcome(output), "name": str(test),
"expected": test.expected_outcomes, "flags": test.cmd.args,
"duration": output.duration, "command": test.cmd.to_string(relative=True),
"run": run + 1,
# TODO(machenbach): This stores only the global random seed from the "stdout": output.stdout,
# context and not possible overrides when using random-seed stress. "stderr": output.stderr,
"random_seed": self.random_seed, "exit_code": output.exit_code,
"target_name": test.get_shell(), "result": test.output_proc.get_outcome(output),
"variant": test.variant, "expected": test.expected_outcomes,
}) "duration": output.duration,
# TODO(machenbach): This stores only the global random seed from the
# context and not possible overrides when using random-seed stress.
"random_seed": self.random_seed,
"target_name": test.get_shell(),
"variant": test.variant,
})
def finished(self): def finished(self):
complete_results = [] complete_results = []
......
...@@ -2,43 +2,56 @@ ...@@ -2,43 +2,56 @@
# Use of this source code is governed by a BSD-style license that can be # Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file. # found in the LICENSE file.
import collections
from . import base from . import base
from .result import RerunResult
class RerunProc(base.TestProcProducer): class RerunProc(base.TestProcProducer):
def __init__(self, rerun_max, rerun_max_total=None): def __init__(self, rerun_max, rerun_max_total=None):
super(RerunProc, self).__init__('Rerun') super(RerunProc, self).__init__('Rerun')
self._rerun = dict() self._rerun = {}
self._results = collections.defaultdict(list)
self._rerun_max = rerun_max self._rerun_max = rerun_max
self._rerun_total_left = rerun_max_total self._rerun_total_left = rerun_max_total
def _next_test(self, test): def _next_test(self, test):
self._init_test(test)
self._send_next_subtest(test) self._send_next_subtest(test)
def _result_for(self, test, subtest, result): def _result_for(self, test, subtest, result):
# First result
if subtest.procid[-2:] == '-1':
# Passed, no reruns
if not result.has_unexpected_output:
self._send_result(test, result)
return
self._rerun[test.procid] = 0
results = self._results[test.procid]
results.append(result)
if self._needs_rerun(test, result): if self._needs_rerun(test, result):
self._rerun[test.procid] += 1 self._rerun[test.procid] += 1
if self._rerun_total_left is not None: if self._rerun_total_left is not None:
self._rerun_total_left -= 1 self._rerun_total_left -= 1
self._send_next_subtest(test) self._send_next_subtest(test, self._rerun[test.procid])
else: else:
result = RerunResult.create(results)
self._finalize_test(test) self._finalize_test(test)
self._send_result(test, result) self._send_result(test, result)
def _init_test(self, test):
self._rerun[test.procid] = 0
def _needs_rerun(self, test, result): def _needs_rerun(self, test, result):
# TODO(majeski): Limit reruns count for slow tests. # TODO(majeski): Limit reruns count for slow tests.
return ((self._rerun_total_left is None or self._rerun_total_left > 0) and return ((self._rerun_total_left is None or self._rerun_total_left > 0) and
self._rerun[test.procid] < self._rerun_max and self._rerun[test.procid] < self._rerun_max and
result.has_unexpected_output) result.has_unexpected_output)
def _send_next_subtest(self, test): def _send_next_subtest(self, test, run=0):
run = self._rerun[test.procid]
subtest = self._create_subtest(test, str(run + 1)) subtest = self._create_subtest(test, str(run + 1))
self._send_test(subtest) self._send_test(subtest)
def _finalize_test(self, test): def _finalize_test(self, test):
del self._rerun[test.procid] del self._rerun[test.procid]
del self._results[test.procid]
...@@ -12,6 +12,10 @@ class ResultBase(object): ...@@ -12,6 +12,10 @@ class ResultBase(object):
def is_grouped(self): def is_grouped(self):
return False return False
@property
def is_rerun(self):
return False
class Result(ResultBase): class Result(ResultBase):
"""Result created by the output processor.""" """Result created by the output processor."""
...@@ -58,3 +62,34 @@ class SkippedResult(ResultBase): ...@@ -58,3 +62,34 @@ class SkippedResult(ResultBase):
SKIPPED = SkippedResult() SKIPPED = SkippedResult()
class RerunResult(Result):
"""Result generated from several reruns of the same test. It's a subclass of
Result since the result of rerun is result of the last run. In addition to
normal result it contains results of all reruns.
"""
@staticmethod
def create(results):
"""Create RerunResult based on list of results. List cannot be empty. If it
has only one element it's returned as a result.
"""
assert results
if len(results) == 1:
return results[0]
return RerunResult(results)
def __init__(self, results):
"""Has unexpected output and the output itself of the RerunResult equals to
the last result in the passed list.
"""
assert results
last = results[-1]
super(RerunResult, self).__init__(last.has_unexpected_output, last.output)
self.results = results
@property
def is_rerun(self):
return True
...@@ -242,7 +242,7 @@ class SystemTest(unittest.TestCase): ...@@ -242,7 +242,7 @@ class SystemTest(unittest.TestCase):
self.assertIn('3 tests failed', result.stdout, result) self.assertIn('3 tests failed', result.stdout, result)
else: else:
# With test processors we don't count reruns as separated failures. # With test processors we don't count reruns as separated failures.
# TODO(majeski): fix it. # TODO(majeski): fix it?
self.assertIn('1 tests failed', result.stdout, result) self.assertIn('1 tests failed', result.stdout, result)
self.assertEqual(0, result.returncode, result) self.assertEqual(0, result.returncode, result)
...@@ -264,10 +264,7 @@ class SystemTest(unittest.TestCase): ...@@ -264,10 +264,7 @@ class SystemTest(unittest.TestCase):
replace_variable_data(data) replace_variable_data(data)
json_output['duration_mean'] = 1 json_output['duration_mean'] = 1
suffix = '' expected_results_name = 'expected_test_results1.json'
if infra_staging:
suffix = '-proc'
expected_results_name = 'expected_test_results1%s.json' % suffix
with open(os.path.join(TEST_DATA_ROOT, expected_results_name)) as f: with open(os.path.join(TEST_DATA_ROOT, expected_results_name)) as f:
expected_test_results = json.load(f) expected_test_results = json.load(f)
......
{
"arch": "x64",
"duration_mean": 1,
"mode": "release",
"results": [
{
"command": "/usr/bin/python out/Release/d8_mocked.py --random-seed=123 strawberries --nohard-abort",
"duration": 1,
"exit_code": 1,
"expected": [
"PASS"
],
"flags": [
"--random-seed=123",
"strawberries",
"--nohard-abort"
],
"name": "sweet/strawberries",
"random_seed": 123,
"result": "FAIL",
"run": -100,
"stderr": "",
"stdout": "--random-seed=123 strawberries --nohard-abort\n",
"target_name": "d8_mocked.py",
"variant": "default"
}
],
"slowest_tests": [
{
"command": "/usr/bin/python out/Release/d8_mocked.py --random-seed=123 strawberries --nohard-abort",
"duration": 1,
"flags": [
"--random-seed=123",
"strawberries",
"--nohard-abort"
],
"marked_slow": true,
"name": "sweet/strawberries"
}
],
"test_total": 1
}
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