Skip to content

Commit

Permalink
Properly check for created coverage files
Browse files Browse the repository at this point in the history
An empty `ReportFile` is boolean-coerced to `False`, so we always need to explicitly compare it to `None`.

This primarily fixes the `JetBrainsXML` parser always emitting empty `Report`s.
Also fixes cobertura failing on empty filenames, and other smaller fixes.
  • Loading branch information
Swatinem committed Oct 31, 2024
1 parent f153619 commit cea8c11
Show file tree
Hide file tree
Showing 8 changed files with 81 additions and 11 deletions.
13 changes: 9 additions & 4 deletions services/report/languages/cobertura.py
Original file line number Diff line number Diff line change
Expand Up @@ -64,6 +64,8 @@ def from_xml(xml: Element, report_builder_session: ReportBuilderSession) -> None

for _class in xml.iter("class"):
filename = _class.attrib["filename"]
if not filename:
continue
_file = report_builder_session.create_coverage_file(filename, do_fix_path=False)
assert (
_file is not None
Expand Down Expand Up @@ -194,14 +196,17 @@ def from_xml(xml: Element, report_builder_session: ReportBuilderSession) -> None

# path rename
path_fixer = report_builder_session.path_fixer
path_name_fixing = []
source_path_list = get_sources_to_attempt(xml)
path_name_fixing = []

for _class in xml.iter("class"):
filename = _class.attrib["filename"]
fixed_name = path_fixer(filename, bases_to_try=source_path_list)
path_name_fixing.append((filename, fixed_name))

_set = set(("dist-packages", "site-packages"))
report_builder_session.resolve_paths(
sorted(path_name_fixing, key=lambda a: _set & set(a[0].split("/")))
# paths with `X-packages` should be sorted to the end
path_name_fixing.sort(
key=lambda a: "/dist-packages/" in a[0] or "/site-packages/" in a[0]
)

report_builder_session.resolve_paths(path_name_fixing)
2 changes: 1 addition & 1 deletion services/report/languages/gap.py
Original file line number Diff line number Diff line change
Expand Up @@ -53,5 +53,5 @@ def from_string(string: bytes, report_builder_session: ReportBuilderSession) ->
)

# append last file
if _file:
if _file is not None:
report_builder_session.append(_file)
7 changes: 3 additions & 4 deletions services/report/languages/jetbrainsxml.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,10 +21,9 @@ def process(
def from_xml(xml: Element, report_builder_session: ReportBuilderSession) -> None:
file_by_id: dict[str, ReportFile] = {}
for f in xml.iter("File"):
_file = report_builder_session.create_coverage_file(
f.attrib["Name"].replace("\\", "/")
)
if _file:
filename = f.attrib["Name"].replace("\\", "/")
_file = report_builder_session.create_coverage_file(filename)
if _file is not None:
file_by_id[str(f.attrib["Index"])] = _file

for statement in xml.iter("Statement"):
Expand Down
2 changes: 1 addition & 1 deletion services/report/languages/lcov.py
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ def from_txt(reports: bytes, report_builder_session: ReportBuilderSession) -> No
# http://ltp.sourceforge.net/coverage/lcov/geninfo.1.php
# merge same files
for string in reports.split(b"\nend_of_record"):
if _file := _process_file(string, report_builder_session):
if (_file := _process_file(string, report_builder_session)) is not None:
report_builder_session.append(_file)


Expand Down
2 changes: 1 addition & 1 deletion services/report/languages/mono.py
Original file line number Diff line number Diff line change
Expand Up @@ -45,5 +45,5 @@ def from_xml(xml: Element, report_builder_session: ReportBuilderSession) -> None
)

for _file in files.values():
if _file:
if _file is not None:

Check warning on line 48 in services/report/languages/mono.py

View check run for this annotation

Codecov Notifications / codecov/patch

services/report/languages/mono.py#L48

Added line #L48 was not covered by tests
report_builder_session.append(_file)
21 changes: 21 additions & 0 deletions services/report/languages/tests/unit/test_cobertura.py
Original file line number Diff line number Diff line change
Expand Up @@ -531,3 +531,24 @@ def test_use_source_for_filename_if_multiple_sources_first_and_second_works(self
# doesnt use the source as we dont know which one
assert "/here/source" in processed_report["report"]["files"]
assert "/here/file" in processed_report["report"]["files"]


def test_empty_filename():
xml = """
<coverage>
<class filename="" name="">
<line number="1" hits="1" />
</class>
<class filename="non-empty" name="">
<line number="1" hits="1" />
</class>
</coverage>
"""
report_builder_session = create_report_builder_session()
cobertura.from_xml(
etree.fromstring(xml),
report_builder_session,
)
report = report_builder_session.output_report()

assert not report.is_empty()
22 changes: 22 additions & 0 deletions services/report/languages/tests/unit/test_jetbrainsxml.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
import xml.etree.cElementTree as etree

from services.report.languages import jetbrainsxml

from . import create_report_builder_session


def test_simple_jetbrainsxml():
xml = """
<Root ReportType="DetailedXml">
<File Index="1" Name="/_/src/testhost.x86/UnitTestClient.cs"/>
<Statement FileIndex="1" Line="1" Column="1" EndLine="1" EndColumn="10" Covered="True" />
</Root>
"""
report_builder_session = create_report_builder_session()
jetbrainsxml.from_xml(
etree.fromstring(xml),
report_builder_session,
)
report = report_builder_session.output_report()

assert not report.is_empty()
23 changes: 23 additions & 0 deletions services/report/tests/unit/test_process.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,29 @@
folder = here.parent


@pytest.mark.skip(reason="this is supposed to be invoked manually")
def test_manual():
# The intention of this test is to easily reproduce production problems with real reports.
# So download the relevant report, fill in its filename below, comment out the `skip` annotation,
# and run this test directly.
filename = "..."
with open(filename, "rb") as d:
contents = d.read()

Check warning on line 35 in services/report/tests/unit/test_process.py

View check run for this annotation

Codecov Notifications / codecov/patch

services/report/tests/unit/test_process.py#L33-L35

Added lines #L33 - L35 were not covered by tests

parsed_report = LegacyReportParser().parse_raw_report_from_bytes(contents)
master = Report()
result = process.process_raw_upload(

Check warning on line 39 in services/report/tests/unit/test_process.py

View check run for this annotation

Codecov Notifications / codecov/patch

services/report/tests/unit/test_process.py#L37-L39

Added lines #L37 - L39 were not covered by tests
commit_yaml=None,
report=master,
raw_reports=parsed_report,
flags=[],
session=Session(),
)
master = result.report

Check warning on line 46 in services/report/tests/unit/test_process.py

View check run for this annotation

Codecov Notifications / codecov/patch

services/report/tests/unit/test_process.py#L46

Added line #L46 was not covered by tests

assert not master.is_empty()

Check warning on line 48 in services/report/tests/unit/test_process.py

View check run for this annotation

Codecov Notifications / codecov/patch

services/report/tests/unit/test_process.py#L48

Added line #L48 was not covered by tests


class TestProcessRawUpload(BaseTestCase):
def readjson(self, filename):
with open(folder / filename, "r") as d:
Expand Down

0 comments on commit cea8c11

Please sign in to comment.