Skip to content

Commit

Permalink
airbyte-ci: embed junit xml reports into user-facing html report (#34923
Browse files Browse the repository at this point in the history
)
  • Loading branch information
postamar authored Feb 7, 2024
1 parent e06243b commit fccc1d0
Show file tree
Hide file tree
Showing 8 changed files with 126 additions and 5 deletions.
1 change: 1 addition & 0 deletions airbyte-ci/connectors/pipelines/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -640,6 +640,7 @@ E.G.: running Poe tasks on the modified internal packages of the current branch:

| Version | PR | Description |
| ------- | ---------------------------------------------------------- | -------------------------------------------------------------------------------------------------------------------------- |
| 4.1.0 | [#34923](https://github.com/airbytehq/airbyte/pull/34923) | Include gradle test reports in HTML connector test report. |
| 4.0.0 | [#34736](https://github.com/airbytehq/airbyte/pull/34736) | Run poe tasks declared in internal poetry packages. |
| 3.10.4 | [#34867](https://github.com/airbytehq/airbyte/pull/34867) | Remove connector ops team |
| 3.10.3 | [#34836](https://github.com/airbytehq/airbyte/pull/34836) | Add check for python registry publishing enabled for certified python sources. |
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@ class IntegrationTests(GradleTask):
gradle_task_name = "integrationTestJava"
mount_connector_secrets = True
bind_to_docker_host = True
with_test_report = True

@property
def default_params(self) -> STEP_PARAMS:
Expand Down Expand Up @@ -75,6 +76,7 @@ class UnitTests(GradleTask):
title = "Java Connector Unit Tests"
gradle_task_name = "test"
bind_to_docker_host = True
with_test_report = True


def _create_integration_step_args_factory(context: ConnectorContext) -> Callable:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -168,6 +168,10 @@
<span class="std">Standard error:</span>
<pre>{{ step_result.stderr }}</pre>
{% endif %}
{% if step_result.report %}
<span class="std">Report:</span>
<pre lang="xml">{{ step_result.report }}</pre>
{% endif %}
</div>
</div>
</div>
Expand Down
Original file line number Diff line number Diff line change
@@ -1,12 +1,14 @@
#
# Copyright (c) 2023 Airbyte, Inc., all rights reserved.
#

import html
import re
from abc import ABC
from typing import Any, ClassVar, List, Optional, Tuple

import pipelines.dagger.actions.system.docker
from dagger import CacheSharingMode, CacheVolume
import xmltodict
from dagger import CacheSharingMode, CacheVolume, Container, QueryError
from pipelines.airbyte_ci.connectors.context import ConnectorContext
from pipelines.consts import AMAZONCORRETTO_IMAGE
from pipelines.dagger.actions import secrets
Expand All @@ -33,6 +35,7 @@ class GradleTask(Step, ABC):
gradle_task_name: ClassVar[str]
bind_to_docker_host: ClassVar[bool] = False
mount_connector_secrets: ClassVar[bool] = False
with_test_report: ClassVar[bool] = False
accept_extra_params = True

@property
Expand Down Expand Up @@ -196,3 +199,84 @@ async def _run(self, *args: Any, **kwargs: Any) -> StepResult:
)
)
return await self.get_step_result(gradle_container)

async def get_step_result(self, container: Container) -> StepResult:
step_result = await super().get_step_result(container)
return StepResult(
step=step_result.step,
status=step_result.status,
stdout=step_result.stdout,
stderr=step_result.stderr,
report=await self._collect_test_report(container),
output_artifact=step_result.output_artifact,
)

async def _collect_test_report(self, gradle_container: Container) -> Optional[str]:
if not self.with_test_report:
return None

junit_xml_path = f"{self.context.connector.code_directory}/build/test-results/{self.gradle_task_name}"
testsuites = []
try:
junit_xml_dir = await gradle_container.directory(junit_xml_path)
for file_name in await junit_xml_dir.entries():
if file_name.endswith(".xml"):
junit_xml = await junit_xml_dir.file(file_name).contents()
# This will be embedded in the HTML report in a <pre lang="xml"> block.
# The java logging backend will have already taken care of masking any secrets.
# Nothing to do in that regard.
try:
if testsuite := xmltodict.parse(junit_xml):
testsuites.append(testsuite)
except Exception as e:
self.context.logger.error(str(e))
self.context.logger.warn(f"Failed to parse junit xml file {file_name}.")
except QueryError as e:
self.context.logger.error(str(e))
self.context.logger.warn(f"Failed to retrieve junit test results from {junit_xml_path} gradle container.")
return None
return render_junit_xml(testsuites)


MAYBE_STARTS_WITH_XML_TAG = re.compile("^ *<")
ESCAPED_ANSI_COLOR_PATTERN = re.compile(r"\?\[m|\?\[[34][0-9]m")


def render_junit_xml(testsuites: List[Any]) -> str:
"""Renders the JUnit XML report as something readable in the HTML test report."""
# Transform the dict contents.
indent = " "
for testsuite in testsuites:
testsuite = testsuite.get("testsuite")
massage_system_out_and_err(testsuite, indent, 4)
if testcases := testsuite.get("testcase"):
if not isinstance(testcases, list):
testcases = [testcases]
for testcase in testcases:
massage_system_out_and_err(testcase, indent, 5)
# Transform back to XML string.
# Try to respect the JUnit XML test result schema.
root = {"testsuites": {"testsuite": testsuites}}
xml = xmltodict.unparse(root, pretty=True, short_empty_elements=True, indent=indent)
# Escape < and > and so forth to make them render properly, but not in the log messages.
# These lines will already have been escaped by xmltodict.unparse.
lines = xml.splitlines()
for idx, line in enumerate(lines):
if MAYBE_STARTS_WITH_XML_TAG.match(line):
lines[idx] = html.escape(line)
return "\n".join(lines)


def massage_system_out_and_err(d: dict, indent: str, indent_levels: int) -> None:
"""Makes the system-out and system-err text prettier."""
if d:
for key in ["system-out", "system-err"]:
if s := d.get(key):
lines = s.splitlines()
s = ""
for line in lines:
stripped = line.strip()
if stripped:
s += "\n" + indent * indent_levels + ESCAPED_ANSI_COLOR_PATTERN.sub("", line.strip())
s = s + "\n" + indent * (indent_levels - 1) if s else None
d[key] = s
1 change: 1 addition & 0 deletions airbyte-ci/connectors/pipelines/pipelines/models/steps.py
Original file line number Diff line number Diff line change
Expand Up @@ -71,6 +71,7 @@ class Result:
created_at: datetime = field(default_factory=datetime.utcnow)
stderr: Optional[str] = None
stdout: Optional[str] = None
report: Optional[str] = None
exc_info: Optional[Exception] = None
output_artifact: Any = None

Expand Down
26 changes: 24 additions & 2 deletions airbyte-ci/connectors/pipelines/poetry.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 3 additions & 1 deletion airbyte-ci/connectors/pipelines/pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ build-backend = "poetry.core.masonry.api"

[tool.poetry]
name = "pipelines"
version = "3.10.5"
version = "4.1.0"
description = "Packaged maintained by the connector operations team to perform CI for connectors' pipelines"
authors = ["Airbyte <contact@airbyte.io>"]

Expand All @@ -31,6 +31,7 @@ tomli = "^2.0.1"
tomli-w = "^1.0.0"
types-requests = "2.28.2"
dpath = "^2.1.6"
xmltodict = "^0.13.0"

[tool.poetry.group.dev.dependencies]
freezegun = "^1.2.2"
Expand All @@ -43,6 +44,7 @@ mypy = "^1.7.1"
ruff = "^0.1.9"
types-toml = "^0.10.8"
types-requests = "2.28.2"
types-xmltodict = "^0.13.0"

[tool.poetry.scripts]
airbyte-ci = "pipelines.cli.airbyte_ci:airbyte_ci"
Expand Down
5 changes: 5 additions & 0 deletions build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -267,6 +267,11 @@ subprojects { subproj ->
// Swallow the logs when running in airbyte-ci, rely on test reports instead.
showStandardStreams = !System.getenv().containsKey("RUN_IN_AIRBYTE_CI")
}
reports {
junitXml {
outputPerTestCase = true
}
}

// Set the timezone to UTC instead of picking up the host machine's timezone,
// which on a developer's laptop is more likely to be PST.
Expand Down

0 comments on commit fccc1d0

Please sign in to comment.