# ==================================================================================================================== #
# _____ ____ _ _ ____ _ #
# _ __ _ _| ____| _ \ / \ / \ | _ \ ___ _ __ ___ _ __| |_ ___ #
# | '_ \| | | | _| | | | |/ _ \ / _ \ | |_) / _ \ '_ \ / _ \| '__| __/ __| #
# | |_) | |_| | |___| |_| / ___ \ / ___ \ _| _ < __/ |_) | (_) | | | |_\__ \ #
# | .__/ \__, |_____|____/_/ \_\/_/ \_(_)_| \_\___| .__/ \___/|_| \__|___/ #
# |_| |___/ |_| #
# ==================================================================================================================== #
# 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 CTest JUnit dialect.
"""
[docs]
@export
@InheritDocString(ju_Testclass, merge=True)
class Testclass(ju_Testclass):
"""
This is a derived implementation for the CTest JUnit dialect.
"""
[docs]
@export
@InheritDocString(ju_Testsuite, merge=True)
class Testsuite(ju_Testsuite):
"""
This is a derived implementation for the CTest JUnit 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 CTest JUnit dialect.
:param testsuite: Test suite from unified data model.
:returns: Test suite from JUnit specific data model (CTest JUnit 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 CTest JUnit 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 CTest JUnit dialect.
:param testsuiteSummary: Test suite summary from unified data model.
:returns: Test suite summary from JUnit specific data model (CTest JUnit 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 CTest JUnit XML file format.
This class reads, validates and transforms an XML file in the CTest JUnit 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 CTest JUnit 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 CTest JUnit dialect.
"""
xmlSchemaFile = "CTest-JUnit.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 CTest 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 CTest JUnit 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)
rootElement.attrib["disabled"] = "0" # TODO: find a value
# if self._assertionCount is not None:
# rootElement.attrib["assertions"] = f"{self._assertionCount}"
rootElement.attrib["hostname"] = str(testsuite._hostname) # TODO: find a value
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}"
testcaseElement.attrib["status"] = "run" # TODO: find a value
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")