From: "Juraj Linkeš" <juraj.linkes@pantheon.tech>
To: thomas@monjalon.net, Honnappa.Nagarahalli@arm.com,
jspewock@iol.unh.edu, probb@iol.unh.edu, paul.szczepanek@arm.com,
Luca.Vizzarro@arm.com, npratte@iol.unh.edu
Cc: dev@dpdk.org, "Juraj Linkeš" <juraj.linkes@pantheon.tech>
Subject: [PATCH v4 5/7] dts: block all test cases when earlier setup fails
Date: Fri, 1 Mar 2024 11:55:20 +0100 [thread overview]
Message-ID: <20240301105522.79870-6-juraj.linkes@pantheon.tech> (raw)
In-Reply-To: <20240301105522.79870-1-juraj.linkes@pantheon.tech>
In case of a failure before a test suite, the child results will be
recursively recorded as blocked, giving us a full report which was
missing previously.
Signed-off-by: Juraj Linkeš <juraj.linkes@pantheon.tech>
---
dts/framework/runner.py | 21 ++--
dts/framework/test_result.py | 186 +++++++++++++++++++++++++----------
2 files changed, 148 insertions(+), 59 deletions(-)
diff --git a/dts/framework/runner.py b/dts/framework/runner.py
index 5f6bcbbb86..864015c350 100644
--- a/dts/framework/runner.py
+++ b/dts/framework/runner.py
@@ -60,13 +60,15 @@ class DTSRunner:
Each setup or teardown of each stage is recorded in a :class:`~framework.test_result.DTSResult`
or one of its subclasses. The test case results are also recorded.
- If an error occurs, the current stage is aborted, the error is recorded and the run continues in
- the next iteration of the same stage. The return code is the highest `severity` of all
+ If an error occurs, the current stage is aborted, the error is recorded, everything in
+ the inner stages is marked as blocked and the run continues in the next iteration
+ of the same stage. The return code is the highest `severity` of all
:class:`~.framework.exception.DTSError`\s.
Example:
- An error occurs in a build target setup. The current build target is aborted and the run
- continues with the next build target. If the errored build target was the last one in the
+ An error occurs in a build target setup. The current build target is aborted,
+ all test suites and their test cases are marked as blocked and the run continues
+ with the next build target. If the errored build target was the last one in the
given execution, the next execution begins.
"""
@@ -100,6 +102,10 @@ def run(self):
test case within the test suite is set up, executed and torn down. After all test cases
have been executed, the test suite is torn down and the next build target will be tested.
+ In order to properly mark test suites and test cases as blocked in case of a failure,
+ we need to have discovered which test suites and test cases to run before any failures
+ happen. The discovery happens at the earliest point at the start of each execution.
+
All the nested steps look like this:
#. Execution setup
@@ -134,7 +140,7 @@ def run(self):
self._logger.info(
f"Running execution with SUT '{execution.system_under_test_node.name}'."
)
- execution_result = self._result.add_execution(execution.system_under_test_node)
+ execution_result = self._result.add_execution(execution)
# we don't want to modify the original config, so create a copy
execution_test_suites = list(execution.test_suites)
if not execution.skip_smoke_tests:
@@ -143,6 +149,7 @@ def run(self):
test_suites_with_cases = self._get_test_suites_with_cases(
execution_test_suites, execution.func, execution.perf
)
+ execution_result.test_suites_with_cases = test_suites_with_cases
except Exception as e:
self._logger.exception(
f"Invalid test suite configuration found: " f"{execution_test_suites}."
@@ -493,9 +500,7 @@ def _run_test_suites(
"""
end_build_target = False
for test_suite_with_cases in test_suites_with_cases:
- test_suite_result = build_target_result.add_test_suite(
- test_suite_with_cases.test_suite_class.__name__
- )
+ test_suite_result = build_target_result.add_test_suite(test_suite_with_cases)
try:
self._run_test_suite(sut_node, tg_node, test_suite_result, test_suite_with_cases)
except BlockingTestSuiteError as e:
diff --git a/dts/framework/test_result.py b/dts/framework/test_result.py
index abdbafab10..eedb2d20ee 100644
--- a/dts/framework/test_result.py
+++ b/dts/framework/test_result.py
@@ -37,7 +37,7 @@
BuildTargetInfo,
Compiler,
CPUType,
- NodeConfiguration,
+ ExecutionConfiguration,
NodeInfo,
TestSuiteConfig,
)
@@ -88,6 +88,8 @@ class Result(Enum):
ERROR = auto()
#:
SKIP = auto()
+ #:
+ BLOCK = auto()
def __bool__(self) -> bool:
"""Only PASS is True."""
@@ -141,21 +143,26 @@ class BaseResult(object):
Attributes:
setup_result: The result of the setup of the particular stage.
teardown_result: The results of the teardown of the particular stage.
+ child_results: The results of the descendants in the results hierarchy.
"""
setup_result: FixtureResult
teardown_result: FixtureResult
- _inner_results: MutableSequence["BaseResult"]
+ child_results: MutableSequence["BaseResult"]
def __init__(self):
"""Initialize the constructor."""
self.setup_result = FixtureResult()
self.teardown_result = FixtureResult()
- self._inner_results = []
+ self.child_results = []
def update_setup(self, result: Result, error: Exception | None = None) -> None:
"""Store the setup result.
+ If the result is :attr:`~Result.BLOCK`, :attr:`~Result.ERROR` or :attr:`~Result.FAIL`,
+ then the corresponding child results in result hierarchy
+ are also marked with :attr:`~Result.BLOCK`.
+
Args:
result: The result of the setup.
error: The error that occurred in case of a failure.
@@ -163,6 +170,16 @@ def update_setup(self, result: Result, error: Exception | None = None) -> None:
self.setup_result.result = result
self.setup_result.error = error
+ if result in [Result.BLOCK, Result.ERROR, Result.FAIL]:
+ self.update_teardown(Result.BLOCK)
+ self._block_result()
+
+ def _block_result(self) -> None:
+ r"""Mark the result as :attr:`~Result.BLOCK`\ed.
+
+ The blocking of child results should be done in overloaded methods.
+ """
+
def update_teardown(self, result: Result, error: Exception | None = None) -> None:
"""Store the teardown result.
@@ -181,10 +198,8 @@ def _get_setup_teardown_errors(self) -> list[Exception]:
errors.append(self.teardown_result.error)
return errors
- def _get_inner_errors(self) -> list[Exception]:
- return [
- error for inner_result in self._inner_results for error in inner_result.get_errors()
- ]
+ def _get_child_errors(self) -> list[Exception]:
+ return [error for child_result in self.child_results for error in child_result.get_errors()]
def get_errors(self) -> list[Exception]:
"""Compile errors from the whole result hierarchy.
@@ -192,7 +207,7 @@ def get_errors(self) -> list[Exception]:
Returns:
The errors from setup, teardown and all errors found in the whole result hierarchy.
"""
- return self._get_setup_teardown_errors() + self._get_inner_errors()
+ return self._get_setup_teardown_errors() + self._get_child_errors()
def add_stats(self, statistics: "Statistics") -> None:
"""Collate stats from the whole result hierarchy.
@@ -200,8 +215,8 @@ def add_stats(self, statistics: "Statistics") -> None:
Args:
statistics: The :class:`Statistics` object where the stats will be collated.
"""
- for inner_result in self._inner_results:
- inner_result.add_stats(statistics)
+ for child_result in self.child_results:
+ child_result.add_stats(statistics)
class DTSResult(BaseResult):
@@ -242,18 +257,18 @@ def __init__(self, logger: DTSLOG):
self._stats_result = None
self._stats_filename = os.path.join(SETTINGS.output_dir, "statistics.txt")
- def add_execution(self, sut_node: NodeConfiguration) -> "ExecutionResult":
- """Add and return the inner result (execution).
+ def add_execution(self, execution: ExecutionConfiguration) -> "ExecutionResult":
+ """Add and return the child result (execution).
Args:
- sut_node: The SUT node's test run configuration.
+ execution: The execution's test run configuration.
Returns:
The execution's result.
"""
- execution_result = ExecutionResult(sut_node)
- self._inner_results.append(execution_result)
- return execution_result
+ result = ExecutionResult(execution)
+ self.child_results.append(result)
+ return result
def add_error(self, error: Exception) -> None:
"""Record an error that occurred outside any execution.
@@ -266,8 +281,8 @@ def add_error(self, error: Exception) -> None:
def process(self) -> None:
"""Process the data after a whole DTS run.
- The data is added to inner objects during runtime and this object is not updated
- at that time. This requires us to process the inner data after it's all been gathered.
+ The data is added to child objects during runtime and this object is not updated
+ at that time. This requires us to process the child data after it's all been gathered.
The processing gathers all errors and the statistics of test case results.
"""
@@ -305,28 +320,30 @@ class ExecutionResult(BaseResult):
The internal list stores the results of all build targets in a given execution.
Attributes:
- sut_node: The SUT node used in the execution.
sut_os_name: The operating system of the SUT node.
sut_os_version: The operating system version of the SUT node.
sut_kernel_version: The operating system kernel version of the SUT node.
"""
- sut_node: NodeConfiguration
sut_os_name: str
sut_os_version: str
sut_kernel_version: str
+ _config: ExecutionConfiguration
+ _parent_result: DTSResult
+ _test_suites_with_cases: list[TestSuiteWithCases]
- def __init__(self, sut_node: NodeConfiguration):
- """Extend the constructor with the `sut_node`'s config.
+ def __init__(self, execution: ExecutionConfiguration):
+ """Extend the constructor with the execution's config and DTSResult.
Args:
- sut_node: The SUT node's test run configuration used in the execution.
+ execution: The execution's test run configuration.
"""
super(ExecutionResult, self).__init__()
- self.sut_node = sut_node
+ self._config = execution
+ self._test_suites_with_cases = []
def add_build_target(self, build_target: BuildTargetConfiguration) -> "BuildTargetResult":
- """Add and return the inner result (build target).
+ """Add and return the child result (build target).
Args:
build_target: The build target's test run configuration.
@@ -334,9 +351,34 @@ def add_build_target(self, build_target: BuildTargetConfiguration) -> "BuildTarg
Returns:
The build target's result.
"""
- build_target_result = BuildTargetResult(build_target)
- self._inner_results.append(build_target_result)
- return build_target_result
+ result = BuildTargetResult(
+ self._test_suites_with_cases,
+ build_target,
+ )
+ self.child_results.append(result)
+ return result
+
+ @property
+ def test_suites_with_cases(self) -> list[TestSuiteWithCases]:
+ """The test suites with test cases to be executed in this execution.
+
+ The test suites can only be assigned once.
+
+ Returns:
+ The list of test suites with test cases. If an error occurs between
+ the initialization of :class:`ExecutionResult` and assigning test cases to the instance,
+ return an empty list, representing that we don't know what to execute.
+ """
+ return self._test_suites_with_cases
+
+ @test_suites_with_cases.setter
+ def test_suites_with_cases(self, test_suites_with_cases: list[TestSuiteWithCases]) -> None:
+ if self._test_suites_with_cases:
+ raise ValueError(
+ "Attempted to assign test suites to an execution result "
+ "which already has test suites."
+ )
+ self._test_suites_with_cases = test_suites_with_cases
def add_sut_info(self, sut_info: NodeInfo) -> None:
"""Add SUT information gathered at runtime.
@@ -348,6 +390,12 @@ def add_sut_info(self, sut_info: NodeInfo) -> None:
self.sut_os_version = sut_info.os_version
self.sut_kernel_version = sut_info.kernel_version
+ def _block_result(self) -> None:
+ r"""Mark the result as :attr:`~Result.BLOCK`\ed."""
+ for build_target in self._config.build_targets:
+ child_result = self.add_build_target(build_target)
+ child_result.update_setup(Result.BLOCK)
+
class BuildTargetResult(BaseResult):
"""The build target specific result.
@@ -369,11 +417,17 @@ class BuildTargetResult(BaseResult):
compiler: Compiler
compiler_version: str | None
dpdk_version: str | None
+ _test_suites_with_cases: list[TestSuiteWithCases]
- def __init__(self, build_target: BuildTargetConfiguration):
- """Extend the constructor with the `build_target`'s build target config.
+ def __init__(
+ self,
+ test_suites_with_cases: list[TestSuiteWithCases],
+ build_target: BuildTargetConfiguration,
+ ):
+ """Extend the constructor with the build target's config and ExecutionResult.
Args:
+ test_suites_with_cases: The test suites with test cases to be run in this build target.
build_target: The build target's test run configuration.
"""
super(BuildTargetResult, self).__init__()
@@ -383,6 +437,23 @@ def __init__(self, build_target: BuildTargetConfiguration):
self.compiler = build_target.compiler
self.compiler_version = None
self.dpdk_version = None
+ self._test_suites_with_cases = test_suites_with_cases
+
+ def add_test_suite(
+ self,
+ test_suite_with_cases: TestSuiteWithCases,
+ ) -> "TestSuiteResult":
+ """Add and return the child result (test suite).
+
+ Args:
+ test_suite_with_cases: The test suite with test cases.
+
+ Returns:
+ The test suite's result.
+ """
+ result = TestSuiteResult(test_suite_with_cases)
+ self.child_results.append(result)
+ return result
def add_build_target_info(self, versions: BuildTargetInfo) -> None:
"""Add information about the build target gathered at runtime.
@@ -393,15 +464,11 @@ def add_build_target_info(self, versions: BuildTargetInfo) -> None:
self.compiler_version = versions.compiler_version
self.dpdk_version = versions.dpdk_version
- def add_test_suite(self, test_suite_name: str) -> "TestSuiteResult":
- """Add and return the inner result (test suite).
-
- Returns:
- The test suite's result.
- """
- test_suite_result = TestSuiteResult(test_suite_name)
- self._inner_results.append(test_suite_result)
- return test_suite_result
+ def _block_result(self) -> None:
+ r"""Mark the result as :attr:`~Result.BLOCK`\ed."""
+ for test_suite_with_cases in self._test_suites_with_cases:
+ child_result = self.add_test_suite(test_suite_with_cases)
+ child_result.update_setup(Result.BLOCK)
class TestSuiteResult(BaseResult):
@@ -410,29 +477,42 @@ class TestSuiteResult(BaseResult):
The internal list stores the results of all test cases in a given test suite.
Attributes:
- suite_name: The test suite name.
+ test_suite_name: The test suite name.
"""
- suite_name: str
+ test_suite_name: str
+ _test_suite_with_cases: TestSuiteWithCases
+ _parent_result: BuildTargetResult
+ _child_configs: list[str]
- def __init__(self, suite_name: str):
- """Extend the constructor with `suite_name`.
+ def __init__(self, test_suite_with_cases: TestSuiteWithCases):
+ """Extend the constructor with test suite's config and BuildTargetResult.
Args:
- suite_name: The test suite's name.
+ test_suite_with_cases: The test suite with test cases.
"""
super(TestSuiteResult, self).__init__()
- self.suite_name = suite_name
+ self.test_suite_name = test_suite_with_cases.test_suite_class.__name__
+ self._test_suite_with_cases = test_suite_with_cases
def add_test_case(self, test_case_name: str) -> "TestCaseResult":
- """Add and return the inner result (test case).
+ """Add and return the child result (test case).
+
+ Args:
+ test_case_name: The name of the test case.
Returns:
The test case's result.
"""
- test_case_result = TestCaseResult(test_case_name)
- self._inner_results.append(test_case_result)
- return test_case_result
+ result = TestCaseResult(test_case_name)
+ self.child_results.append(result)
+ return result
+
+ def _block_result(self) -> None:
+ r"""Mark the result as :attr:`~Result.BLOCK`\ed."""
+ for test_case_method in self._test_suite_with_cases.test_cases:
+ child_result = self.add_test_case(test_case_method.__name__)
+ child_result.update_setup(Result.BLOCK)
class TestCaseResult(BaseResult, FixtureResult):
@@ -449,7 +529,7 @@ class TestCaseResult(BaseResult, FixtureResult):
test_case_name: str
def __init__(self, test_case_name: str):
- """Extend the constructor with `test_case_name`.
+ """Extend the constructor with test case's name and TestSuiteResult.
Args:
test_case_name: The test case's name.
@@ -470,7 +550,7 @@ def update(self, result: Result, error: Exception | None = None) -> None:
self.result = result
self.error = error
- def _get_inner_errors(self) -> list[Exception]:
+ def _get_child_errors(self) -> list[Exception]:
if self.error:
return [self.error]
return []
@@ -486,6 +566,10 @@ def add_stats(self, statistics: "Statistics") -> None:
"""
statistics += self.result
+ def _block_result(self) -> None:
+ r"""Mark the result as :attr:`~Result.BLOCK`\ed."""
+ self.update(Result.BLOCK)
+
def __bool__(self) -> bool:
"""The test case passed only if setup, teardown and the test case itself passed."""
return bool(self.setup_result) and bool(self.teardown_result) and bool(self.result)
--
2.34.1
next prev parent reply other threads:[~2024-03-01 10:56 UTC|newest]
Thread overview: 44+ messages / expand[flat|nested] mbox.gz Atom feed top
2023-12-20 10:33 [RFC PATCH v1 0/5] test case blocking and logging Juraj Linkeš
2023-12-20 10:33 ` [RFC PATCH v1 1/5] dts: convert dts.py methods to class Juraj Linkeš
2023-12-20 10:33 ` [RFC PATCH v1 2/5] dts: move test suite execution logic to DTSRunner Juraj Linkeš
2023-12-20 10:33 ` [RFC PATCH v1 3/5] dts: process test suites at the beginning of run Juraj Linkeš
2023-12-20 10:33 ` [RFC PATCH v1 4/5] dts: block all testcases when earlier setup fails Juraj Linkeš
2023-12-20 10:33 ` [RFC PATCH v1 5/5] dts: refactor logging configuration Juraj Linkeš
2024-01-08 18:47 ` [RFC PATCH v1 0/5] test case blocking and logging Jeremy Spewock
2024-02-06 14:57 ` [PATCH v2 0/7] " Juraj Linkeš
2024-02-06 14:57 ` [PATCH v2 1/7] dts: convert dts.py methods to class Juraj Linkeš
2024-02-06 14:57 ` [PATCH v2 2/7] dts: move test suite execution logic to DTSRunner Juraj Linkeš
2024-02-06 14:57 ` [PATCH v2 3/7] dts: filter test suites in executions Juraj Linkeš
2024-02-12 16:44 ` Jeremy Spewock
2024-02-14 9:55 ` Juraj Linkeš
2024-02-06 14:57 ` [PATCH v2 4/7] dts: reorganize test result Juraj Linkeš
2024-02-06 14:57 ` [PATCH v2 5/7] dts: block all test cases when earlier setup fails Juraj Linkeš
2024-02-06 14:57 ` [PATCH v2 6/7] dts: refactor logging configuration Juraj Linkeš
2024-02-12 16:45 ` Jeremy Spewock
2024-02-14 7:49 ` Juraj Linkeš
2024-02-14 16:51 ` Jeremy Spewock
2024-02-06 14:57 ` [PATCH v2 7/7] dts: improve test suite and case filtering Juraj Linkeš
2024-02-23 7:54 ` [PATCH v3 0/7] test case blocking and logging Juraj Linkeš
2024-02-23 7:54 ` [PATCH v3 1/7] dts: convert dts.py methods to class Juraj Linkeš
2024-02-23 7:54 ` [PATCH v3 2/7] dts: move test suite execution logic to DTSRunner Juraj Linkeš
2024-02-23 7:54 ` [PATCH v3 3/7] dts: filter test suites in executions Juraj Linkeš
2024-02-27 21:21 ` Jeremy Spewock
2024-02-28 9:16 ` Juraj Linkeš
2024-02-23 7:54 ` [PATCH v3 4/7] dts: reorganize test result Juraj Linkeš
2024-02-23 7:55 ` [PATCH v3 5/7] dts: block all test cases when earlier setup fails Juraj Linkeš
2024-02-23 7:55 ` [PATCH v3 6/7] dts: refactor logging configuration Juraj Linkeš
2024-02-23 7:55 ` [PATCH v3 7/7] dts: improve test suite and case filtering Juraj Linkeš
2024-02-27 21:24 ` [PATCH v3 0/7] test case blocking and logging Jeremy Spewock
2024-03-01 10:55 ` [PATCH v4 " Juraj Linkeš
2024-03-01 10:55 ` [PATCH v4 1/7] dts: convert dts.py methods to class Juraj Linkeš
2024-03-01 10:55 ` [PATCH v4 2/7] dts: move test suite execution logic to DTSRunner Juraj Linkeš
2024-03-01 10:55 ` [PATCH v4 3/7] dts: filter test suites in executions Juraj Linkeš
2024-03-01 10:55 ` [PATCH v4 4/7] dts: reorganize test result Juraj Linkeš
2024-03-01 10:55 ` Juraj Linkeš [this message]
2024-03-01 10:55 ` [PATCH v4 6/7] dts: refactor logging configuration Juraj Linkeš
2024-03-01 10:55 ` [PATCH v4 7/7] dts: improve test suite and case filtering Juraj Linkeš
2024-03-01 17:41 ` Jeremy Spewock
2024-03-01 17:43 ` Jeremy Spewock
2024-03-07 11:04 ` Thomas Monjalon
2024-03-01 16:11 ` [PATCH v4 0/7] test case blocking and logging Patrick Robb
2024-03-07 14:55 ` Thomas Monjalon
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20240301105522.79870-6-juraj.linkes@pantheon.tech \
--to=juraj.linkes@pantheon.tech \
--cc=Honnappa.Nagarahalli@arm.com \
--cc=Luca.Vizzarro@arm.com \
--cc=dev@dpdk.org \
--cc=jspewock@iol.unh.edu \
--cc=npratte@iol.unh.edu \
--cc=paul.szczepanek@arm.com \
--cc=probb@iol.unh.edu \
--cc=thomas@monjalon.net \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).