diff --git a/Tools/autotest/autotest.py b/Tools/autotest/autotest.py index 8a2d82e2c8..2ca2cf531e 100755 --- a/Tools/autotest/autotest.py +++ b/Tools/autotest/autotest.py @@ -382,7 +382,7 @@ def run_specific_test(step, *args, **kwargs): a = Test(a) print("Got %s" % (a.name)) if a.name == test: - return (tester.autotest(tests=[a], allow_skips=False), tester) + return tester.autotest(tests=[a], allow_skips=False, step_name=step), tester print("Failed to find test %s on %s" % (test, testname)) sys.exit(1) @@ -506,6 +506,7 @@ def run_step(step): "sup_binaries": supplementary_binaries, "reset_after_every_test": opts.reset_after_every_test, "build_opts": copy.copy(build_opts), + "generate_junit": opts.junit, } if opts.speedup is not None: fly_opts["speedup"] = opts.speedup @@ -516,7 +517,7 @@ def run_step(step): global tester tester = tester_class_map[step](binary, **fly_opts) # run the test and return its result and the tester itself - return (tester.autotest(), tester) + return tester.autotest(None, step_name=step), tester # handle "test.Copter.CPUFailsafe" etc: specific_test_to_run = find_specific_test_to_run(step) @@ -884,6 +885,10 @@ if __name__ == "__main__": action='store_true', default=False, help='configure with --Werror') + parser.add_option("--junit", + default=False, + action='store_true', + help='Generate Junit XML tests report') group_build = optparse.OptionGroup(parser, "Build options") group_build.add_option("--no-configure", diff --git a/Tools/autotest/vehicle_test_suite.py b/Tools/autotest/vehicle_test_suite.py index effe660496..fece6a02ea 100644 --- a/Tools/autotest/vehicle_test_suite.py +++ b/Tools/autotest/vehicle_test_suite.py @@ -18,6 +18,9 @@ import signal import sys import time import traceback +from datetime import datetime +from typing import List + import pexpect import fnmatch import operator @@ -1466,6 +1469,7 @@ class Result(object): self.reason = None self.exception = None self.debug_filename = None + self.time_elapsed = 0.0 # self.passed = False def __str__(self): @@ -1478,7 +1482,8 @@ class Result(object): ret += " (" + str(self.exception) + ")" if self.debug_filename is not None: ret += " (see " + self.debug_filename + ")" - + if self.time_elapsed is not None: + ret += " (duration " + self.time_elapsed + "s)" return ret @@ -1512,6 +1517,7 @@ class TestSuite(ABC): ubsan_abort=False, num_aux_imus=0, dronecan_tests=False, + generate_junit=False, build_opts={}): self.start_time = time.time() @@ -1540,6 +1546,12 @@ class TestSuite(ABC): self.ubsan_abort = ubsan_abort self.build_opts = build_opts self.num_aux_imus = num_aux_imus + self.generate_junit = generate_junit + if generate_junit: + try: + import junitparser + except ImportError as e: + raise ImportError(f"Junit export need junitparser package.\n {e} \nTry: python -m pip install junitparser") self.mavproxy = None self._mavproxy = None # for auto-cleanup on failed tests @@ -8070,6 +8082,7 @@ Also, ignores heartbeats not from our target system''' passed = False result = Result(test) + result.time_elapsed = self.test_timings[desc] ardupilot_alive = False try: @@ -11351,7 +11364,7 @@ switch value''' if not self.current_onboard_log_contains_message(messagetype): raise NotAchievedException("Current onboard log does not contain message %s" % messagetype) - def run_tests(self, tests): + def run_tests(self, tests) -> List[Result]: """Autotest vehicle in SITL.""" if self.run_tests_called: raise ValueError("run_tests called twice") @@ -13577,7 +13590,7 @@ SERIAL5_BAUD 128 print("Had to force-reset SITL %u times" % (self.forced_post_test_sitl_reboots,)) - def autotest(self, tests=None, allow_skips=True): + def autotest(self, tests=None, allow_skips=True, step_name=None): """Autotest used by ArduPilot autotest CI.""" if tests is None: tests = self.tests() @@ -13618,9 +13631,41 @@ SERIAL5_BAUD 128 print(str(failure)) self.post_tests_announcements() + if self.generate_junit: + if step_name is None: + step_name = "Unknown_step_name" + step_name.replace(".", "_") + self.create_junit_report(step_name, results, skip_list) return len(self.fail_list) == 0 + def create_junit_report(self, test_name: str, results: List[Result], skip_list: list[tuple[Test, dict[str, str]]]) -> None: + """Generate Junit report from the autotest results""" + from junitparser import TestCase, TestSuite, JUnitXml, Skipped, Failure + frame = self.vehicleinfo_key() + xml_filename = f"autotest_result_{frame}_{test_name}_junit.xml" + self.progress(f"Writing test result in jUnit format to {xml_filename}\n") + + suites = TestSuite("ArduPilot Autotest") + suites.timestamp = datetime.now().replace(microsecond=0).isoformat() + suite = TestSuite(f"Autotest {frame} {test_name}") + for result in results: + case = TestCase(f"{result.test.name}", f"{frame}", result.time_elapsed) + # f"{result.test.description}" + # case.file ## TODO : add file properties to match test location + if not result.passed: + case.result = [Failure(f"see {result.debug_filename}", f"{result.exception}")] + suite.add_testcase(case) + for skipped in skip_list: + (test, reason) = skipped + case = TestCase(f"{test.name}", f"{test.function}") + case.result = [Skipped(f"Skipped : {reason}")] + # Add suite to JunitXml + xml = JUnitXml() + xml.add_testsuite(suites) + xml.add_testsuite(suite) + xml.write(xml_filename, pretty=True) + def mavfft_fttd(self, sensor_type, sensor_instance, since, until): '''display fft for raw ACC data in current logfile'''