Source code for pyEDAA.Reports.Unittesting.JUnit.AntJUnit4

# ==================================================================================================================== #
#              _____ ____    _        _      ____                       _                                              #
#  _ __  _   _| ____|  _ \  / \      / \    |  _ \ ___ _ __   ___  _ __| |_ ___                                        #
# | '_ \| | | |  _| | | | |/ _ \    / _ \   | |_) / _ \ '_ \ / _ \| '__| __/ __|                                       #
# | |_) | |_| | |___| |_| / ___ \  / ___ \ _|  _ <  __/ |_) | (_) | |  | |_\__ \                                       #
# | .__/ \__, |_____|____/_/   \_\/_/   \_(_)_| \_\___| .__/ \___/|_|   \__|___/                                       #
# |_|    |___/                                        |_|                                                              #
# ==================================================================================================================== #
# Authors:                                                                                                             #
#   Patrick Lehmann                                                                                                    #
#                                                                                                                      #
# License:                                                                                                             #
# ==================================================================================================================== #
# Copyright 2024-2026 Electronic Design Automation Abstraction (EDA²)                                                  #
# Copyright 2023-2023 Patrick Lehmann - Bötzingen, Germany                                                             #
#                                                                                                                      #
# Licensed under the Apache License, Version 2.0 (the "License");                                                      #
# you may not use this file except in compliance with the License.                                                     #
# You may obtain a copy of the License at                                                                              #
#                                                                                                                      #
#   http://www.apache.org/licenses/LICENSE-2.0                                                                         #
#                                                                                                                      #
# Unless required by applicable law or agreed to in writing, software                                                  #
# distributed under the License is distributed on an "AS IS" BASIS,                                                    #
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.                                             #
# See the License for the specific language governing permissions and                                                  #
# limitations under the License.                                                                                       #
#                                                                                                                      #
# SPDX-License-Identifier: Apache-2.0                                                                                  #
# ==================================================================================================================== #
#
"""
Reader for JUnit unit testing summary files in XML format.
"""
from pathlib              import Path
from time                 import perf_counter_ns
from typing               import Optional as Nullable, Generator, Tuple, Union, TypeVar, Type, ClassVar

from lxml.etree           import ElementTree, Element, SubElement, tostring, _Element
from pyTooling.Common     import firstValue
from pyTooling.Decorators import export, InheritDocString

from pyEDAA.Reports.Unittesting       import UnittestException, TestsuiteKind
from pyEDAA.Reports.Unittesting       import TestcaseStatus, TestsuiteStatus, IterationScheme
from pyEDAA.Reports.Unittesting       import TestsuiteSummary as ut_TestsuiteSummary, Testsuite as ut_Testsuite
from pyEDAA.Reports.Unittesting.JUnit import Testcase as ju_Testcase, Testclass as ju_Testclass, Testsuite as ju_Testsuite
from pyEDAA.Reports.Unittesting.JUnit import TestsuiteSummary as ju_TestsuiteSummary, Document as ju_Document

TestsuiteType = TypeVar("TestsuiteType", bound="Testsuite")
TestcaseAggregateReturnType = Tuple[int, int, int]
TestsuiteAggregateReturnType = Tuple[int, int, int, int, int]


[docs] @export @InheritDocString(ju_Testcase, merge=True) class Testcase(ju_Testcase): """ This is a derived implementation for the Ant + JUnit4 dialect. """
[docs] @export @InheritDocString(ju_Testclass, merge=True) class Testclass(ju_Testclass): """ This is a derived implementation for the Ant + JUnit4 dialect. """
[docs] @export @InheritDocString(ju_Testsuite, merge=True) class Testsuite(ju_Testsuite): """ This is a derived implementation for the Ant + JUnit4 dialect. """
[docs] @classmethod def FromTestsuite(cls, testsuite: ut_Testsuite) -> "Testsuite": """ Convert a test suite of the unified test entity data model to the JUnit specific data model's test suite object adhering to the Ant + JUnit4 dialect. :param testsuite: Test suite from unified data model. :returns: Test suite from JUnit specific data model (Ant + JUnit4 dialect). """ juTestsuite = cls( testsuite._name, startTime=testsuite._startTime, duration=testsuite._totalDuration, status= testsuite._status, ) juTestsuite._tests = testsuite._tests juTestsuite._skipped = testsuite._skipped juTestsuite._errored = testsuite._errored juTestsuite._failed = testsuite._failed juTestsuite._passed = testsuite._passed for tc in testsuite.IterateTestcases(): ts = tc._parent if ts is None: raise UnittestException(f"Testcase '{tc._name}' is not part of a hierarchy.") classname = ts._name ts = ts._parent while ts is not None and ts._kind > TestsuiteKind.Logical: classname = f"{ts._name}.{classname}" ts = ts._parent if classname in juTestsuite._testclasses: juClass = juTestsuite._testclasses[classname] else: juClass = Testclass(classname, parent=juTestsuite) juClass.AddTestcase(Testcase.FromTestcase(tc)) return juTestsuite
[docs] @export @InheritDocString(ju_TestsuiteSummary, merge=True) class TestsuiteSummary(ju_TestsuiteSummary): """ This is a derived implementation for the Ant + JUnit4 dialect. """
[docs] @classmethod def FromTestsuiteSummary(cls, testsuiteSummary: ut_TestsuiteSummary) -> "TestsuiteSummary": """ Convert a test suite summary of the unified test entity data model to the JUnit specific data model's test suite summary object adhering to the Ant + JUnit4 dialect. :param testsuiteSummary: Test suite summary from unified data model. :returns: Test suite summary from JUnit specific data model (Ant + JUnit4 dialect). """ return cls( testsuiteSummary._name, startTime=testsuiteSummary._startTime, duration=testsuiteSummary._totalDuration, status=testsuiteSummary._status, testsuites=(ut_Testsuite.FromTestsuite(testsuite) for testsuite in testsuiteSummary._testsuites.values()) )
[docs] @export class Document(ju_Document): """ A document reader and writer for the Ant + JUnit4 XML file format. This class reads, validates and transforms an XML file in the Ant + JUnit4 format into a JUnit data model. It can then be converted into a unified test entity data model. In reverse, a JUnit data model instance with the specific Ant + JUnit4 file format can be created from a unified test entity data model. This data model can be written as XML into a file. """ _TESTCASE: ClassVar[Type[Testcase]] = Testcase _TESTCLASS: ClassVar[Type[Testclass]] = Testclass _TESTSUITE: ClassVar[Type[Testsuite]] = Testsuite
[docs] @classmethod def FromTestsuiteSummary(cls, xmlReportFile: Path, testsuiteSummary: ut_TestsuiteSummary): doc = cls(xmlReportFile) doc._name = testsuiteSummary._name doc._startTime = testsuiteSummary._startTime doc._duration = testsuiteSummary._totalDuration doc._status = testsuiteSummary._status doc._tests = testsuiteSummary._tests doc._skipped = testsuiteSummary._skipped doc._errored = testsuiteSummary._errored doc._failed = testsuiteSummary._failed doc._passed = testsuiteSummary._passed doc.AddTestsuites(Testsuite.FromTestsuite(testsuite) for testsuite in testsuiteSummary._testsuites.values()) return doc
[docs] def Analyze(self) -> None: """ Analyze the XML file, parse the content into an XML data structure and validate the data structure using an XML schema. .. hint:: The time spend for analysis will be made available via property :data:`AnalysisDuration`.. The used XML schema definition is specific to the Ant JUnit4 dialect. """ xmlSchemaFile = "Ant-JUnit4.xsd" self._Analyze(xmlSchemaFile)
[docs] def Write(self, path: Nullable[Path] = None, overwrite: bool = False, regenerate: bool = False) -> None: """ Write the data model as XML into a file adhering to the Ant + JUnit4 dialect. :param path: Optional path to the XMl file, if internal path shouldn't be used. :param overwrite: If true, overwrite an existing file. :param regenerate: If true, regenerate the XML structure from data model. :raises UnittestException: If the file cannot be overwritten. :raises UnittestException: If the internal XML data structure wasn't generated. :raises UnittestException: If the file cannot be opened or written. """ if path is None: path = self._path if not overwrite and path.exists(): raise UnittestException(f"JUnit XML file '{path}' can not be overwritten.") \ from FileExistsError(f"File '{path}' already exists.") if regenerate: self.Generate(overwrite=True) if self._xmlDocument is None: ex = UnittestException(f"Internal XML document tree is empty and needs to be generated before write is possible.") ex.add_note(f"Call 'JUnitDocument.Generate()' or 'JUnitDocument.Write(..., regenerate=True)'.") raise ex try: with path.open("wb") as file: file.write(tostring(self._xmlDocument, encoding="utf-8", xml_declaration=True, pretty_print=True)) except Exception as ex: raise UnittestException(f"JUnit XML file '{path}' can not be written.") from ex
[docs] def Convert(self) -> None: """ Convert the parsed and validated XML data structure into a JUnit test entity hierarchy. This method converts the root element. .. hint:: The time spend for model conversion will be made available via property :data:`ModelConversionDuration`. :raises UnittestException: If XML was not read and parsed before. """ if self._xmlDocument is None: ex = UnittestException(f"JUnit XML file '{self._path}' needs to be read and analyzed by an XML parser.") ex.add_note(f"Call 'JUnitDocument.Analyze()' or create the document using 'JUnitDocument(path, parse=True)'.") raise ex startConversion = perf_counter_ns() rootElement: _Element = self._xmlDocument.getroot() self._name = self._ConvertName(rootElement, optional=True) self._startTime =self._ConvertTimestamp(rootElement, optional=True) self._duration = self._ConvertTime(rootElement, optional=True) # tests = rootElement.getAttribute("tests") # skipped = rootElement.getAttribute("skipped") # errors = rootElement.getAttribute("errors") # failures = rootElement.getAttribute("failures") # assertions = rootElement.getAttribute("assertions") ts = Testsuite(self._name, startTime=self._startTime, duration=self._duration, parent=self) self._ConvertTestsuiteChildren(rootElement, ts) self.Aggregate() endConversation = perf_counter_ns() self._modelConversion = (endConversation - startConversion) / 1e9
[docs] def _ConvertTestsuite(self, parent: TestsuiteSummary, testsuitesNode: _Element) -> None: """ Convert the XML data structure of a ``<testsuite>`` to a test suite. This method uses private helper methods provided by the base-class. :param parent: The test suite summary as a parent element in the test entity hierarchy. :param testsuitesNode: The current XML element node representing a test suite. """ newTestsuite = Testsuite( self._ConvertName(testsuitesNode, optional=False), self._ConvertHostname(testsuitesNode, optional=False), self._ConvertTimestamp(testsuitesNode, optional=False), self._ConvertTime(testsuitesNode, optional=False), parent=parent ) self._ConvertTestsuiteChildren(testsuitesNode, newTestsuite)
[docs] def Generate(self, overwrite: bool = False) -> None: """ Generate the internal XML data structure from test suites and test cases. This method generates the XML root element (``<testsuite>``) and recursively calls other generated methods. :param overwrite: Overwrite the internal XML data structure. :raises UnittestException: If overwrite is false and the internal XML data structure is not empty. """ if not overwrite and self._xmlDocument is not None: raise UnittestException(f"Internal XML document is populated with data.") if self.TestsuiteCount != 1: ex = UnittestException(f"The Ant + JUnit4 format requires exactly one test suite.") ex.add_note(f"Found {self.TestsuiteCount} test suites.") raise ex testsuite = firstValue(self._testsuites) rootElement = Element("testsuite") rootElement.attrib["name"] = self._name if self._startTime is not None: rootElement.attrib["timestamp"] = f"{self._startTime.isoformat()}" if self._duration is not None: rootElement.attrib["time"] = f"{self._duration.total_seconds():.6f}" rootElement.attrib["tests"] = str(self._tests) rootElement.attrib["failures"] = str(self._failed) rootElement.attrib["errors"] = str(self._errored) rootElement.attrib["skipped"] = str(self._skipped) # if self._assertionCount is not None: # rootElement.attrib["assertions"] = f"{self._assertionCount}" if testsuite._hostname is not None: rootElement.attrib["hostname"] = testsuite._hostname self._xmlDocument = ElementTree(rootElement) for testclass in testsuite._testclasses.values(): for tc in testclass._testcases.values(): self._GenerateTestcase(tc, rootElement)
[docs] def _GenerateTestcase(self, testcase: Testcase, parentElement: _Element) -> None: """ Generate the internal XML data structure for a test case. This method generates the XML element (``<testcase>``) and recursively calls other generated methods. :param testcase: The test case to convert to an XML data structures. :param parentElement: The parent XML data structure element, this data structure part will be added to. """ testcaseElement = SubElement(parentElement, "testcase") if testcase.Classname is not None: testcaseElement.attrib["classname"] = testcase.Classname testcaseElement.attrib["name"] = testcase._name if testcase._duration is not None: testcaseElement.attrib["time"] = f"{testcase._duration.total_seconds():.6f}" if testcase._assertionCount is not None: testcaseElement.attrib["assertions"] = f"{testcase._assertionCount}" if testcase._status is TestcaseStatus.Passed: pass elif testcase._status is TestcaseStatus.Failed: failureElement = SubElement(testcaseElement, "failure") elif testcase._status is TestcaseStatus.Skipped: skippedElement = SubElement(testcaseElement, "skipped") else: errorElement = SubElement(testcaseElement, "error")