diff --git a/coverage/lcovreport.py b/coverage/lcovreport.py index d0e636b47b..7f7953ddc9 100644 --- a/coverage/lcovreport.py +++ b/coverage/lcovreport.py @@ -64,41 +64,50 @@ def lcov_arcs( missing_arcs = analysis.missing_branch_arcs() for line in lines: - if line in branch_stats: - # The meaning of a BRDA: line is not well explained in the lcov - # documentation. Based on what genhtml does with them, however, - # the interpretation is supposed to be something like this: - # BRDA: , , , - # where is the source line number of the *origin* of the - # branch; is an arbitrary number which distinguishes multiple - # control flow operations on a single line; is an arbitrary - # number which distinguishes the possible destinations of the specific - # control flow operation identified by + ; and is - # either the hit count for + + or "-" meaning - # that + was never *reached*. must be >= 1, - # and , , must be >= 0. - - # This is only one possible way to map our sets of executed and - # not-executed arcs to BRDA codes. It seems to produce reasonable - # results when fed through genhtml. - + if line not in branch_stats: + continue + + # This is only one of several possible ways to map our sets of executed + # and not-executed arcs to BRDA codes. It seems to produce reasonable + # results when fed through genhtml. + _, taken = branch_stats[line] + + if taken == 0: + # When _none_ of the out arcs from 'line' were executed, + # this probably means means 'line' was never executed at all. + # Cross-check with the line stats. + assert len(executed_arcs[line]) == 0 + if line in analysis.missing: + hit = "-" # indeed, never reached + else: + hit = "0" # I am not prepared to swear this is impossible + destinations = [ + (int(dst < 0), abs(dst), hit) for dst in missing_arcs[line] + ] + else: # Q: can we get counts of the number of times each arc was executed? - # branch_stats has "total" and "taken" counts for each branch, but it - # doesn't have "taken" broken down by destination. - destinations = {} - for dst in executed_arcs[line]: - destinations[(int(dst < 0), abs(dst))] = 1 - for dst in missing_arcs[line]: - destinations[(int(dst < 0), abs(dst))] = 0 - - if all(v == 0 for v in destinations.values()): - # When _none_ of the out arcs from 'line' were executed, presume - # 'line' was never reached. - for branch, _ in enumerate(sorted(destinations.keys())): - outfile.write(f"BRDA:{line},0,{branch},-\n") + # branch_stats has "total" and "taken" counts for each branch, + # but it doesn't have "taken" broken down by destination. + destinations = [ + (int(dst < 0), abs(dst), "1") for dst in executed_arcs[line] + ] + destinations.extend( + (int(dst < 0), abs(dst), "0") for dst in missing_arcs[line] + ) + + destinations.sort() + n_exits = sum( + 1 for is_exit, _, _ in destinations if is_exit + ) + for is_exit, dst, hit in destinations: + if is_exit: + if n_exits == 1: + branch = "to exit" + else: + branch = f"to exit {dst}" else: - for branch, (_, hit) in enumerate(sorted(destinations.items())): - outfile.write(f"BRDA:{line},0,{branch},{hit}\n") + branch = f"to line {dst}" + outfile.write(f"BRDA:{line},0,{branch},{hit}\n") # Summary of the branch coverage. brf = sum(t for t, k in branch_stats.values()) diff --git a/tests/test_lcov.py b/tests/test_lcov.py index 65671f3fcb..5c2061dc2d 100644 --- a/tests/test_lcov.py +++ b/tests/test_lcov.py @@ -160,8 +160,8 @@ def is_it_x(x): DA:5,0 LF:4 LH:1 - BRDA:2,0,0,- - BRDA:2,0,1,- + BRDA:2,0,to line 3,- + BRDA:2,0,to line 5,- BRF:2 BRH:0 end_of_record @@ -203,8 +203,8 @@ def test_is_it_x(self): DA:5,0 LF:4 LH:1 - BRDA:2,0,0,- - BRDA:2,0,1,- + BRDA:2,0,to line 3,- + BRDA:2,0,to line 5,- BRF:2 BRH:0 end_of_record @@ -247,8 +247,8 @@ def test_half_covered_branch(self) -> None: DA:6,0 LF:4 LH:3 - BRDA:3,0,0,1 - BRDA:3,0,1,0 + BRDA:3,0,to line 4,1 + BRDA:3,0,to line 6,0 BRF:2 BRH:1 end_of_record @@ -315,11 +315,78 @@ def test_excluded_lines(self) -> None: DA:6,1 LF:4 LH:3 - BRDA:3,0,0,0 - BRDA:3,0,1,1 + BRDA:3,0,to line 4,0 + BRDA:3,0,to line 6,1 BRF:2 BRH:1 end_of_record """) actual_result = self.get_lcov_report_content() assert expected_result == actual_result + + def test_exit_branches(self) -> None: + self.make_file("runme.py", """\ + def foo(a): + if a: + print(f"{a!r} is truthy") + foo(True) + foo(False) + foo([]) + foo([0]) + """) + cov = coverage.Coverage(source=".", branch=True) + self.start_import_stop(cov, "runme") + cov.lcov_report() + expected_result = textwrap.dedent("""\ + SF:runme.py + DA:1,1 + DA:2,1 + DA:3,1 + DA:4,1 + DA:5,1 + DA:6,1 + DA:7,1 + LF:7 + LH:7 + BRDA:2,0,to line 3,1 + BRDA:2,0,to exit,1 + BRF:2 + BRH:2 + end_of_record + """) + actual_result = self.get_lcov_report_content() + assert expected_result == actual_result + + def test_multiple_exit_branches(self) -> None: + self.make_file("runme.py", """\ + def foo(a): + if any(x > 0 for x in a): + print(f"{a!r} has positives") + foo([]) + foo([0]) + foo([0,1]) + foo([0,-1]) + """) + cov = coverage.Coverage(source=".", branch=True) + self.start_import_stop(cov, "runme") + cov.lcov_report() + expected_result = textwrap.dedent("""\ + SF:runme.py + DA:1,1 + DA:2,1 + DA:3,1 + DA:4,1 + DA:5,1 + DA:6,1 + DA:7,1 + LF:7 + LH:7 + BRDA:2,0,to line 3,1 + BRDA:2,0,to exit 1,1 + BRDA:2,0,to exit 2,1 + BRF:2 + BRH:2 + end_of_record + """) + actual_result = self.get_lcov_report_content() + assert expected_result == actual_result