-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathprocess_coverage.py
executable file
·90 lines (76 loc) · 4.38 KB
/
process_coverage.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
#!/usr/bin/python3
#######################################################
# You should only use the coverage annotations for
# a) lines around errors and warnings,
# b) code solely existing for debugging purposes,
# c) calls of `TryNextMethod` leading to external code.
#######################################################
# This script applies three rules to all coverage*.json files below the current working directory:
# a) Remove all coverage information of files not below the current working directory
# (this technically includes "tst/testall.g" because it is not referenced by an absolute path).
# b) Strip the current working directory from the paths of files below the current working directory.
# c) Ignore lines with one of the "COVERAGE_IGNORE_..." annotations (see below).
import json
import re
import os
from pathlib import Path
regex_block_start = re.compile("^\s*# COVERAGE_IGNORE_BLOCK_START$")
regex_block_end = re.compile("^\s*# COVERAGE_IGNORE_BLOCK_END$")
regex_next_line = re.compile("^\s*# COVERAGE_IGNORE_NEXT_LINE$")
regex_return_void = re.compile("^\s*return;$")
for coverage_filename in Path("../").glob("**/coverage*.json"):
print("processing coverage file " + str(coverage_filename))
new_data = {}
new_data["coverage"] = {}
with open(coverage_filename) as json_file:
data = json.load(json_file)
files = data["coverage"]
for filename, lines_covered in files.items():
if filename.startswith(os.getcwd() + "/"):
print(" processing code file " + filename)
# ignored lines will be delete from lines_covered in-place
new_data["coverage"][filename.replace(os.getcwd() + "/", "", 1)] = lines_covered
line_number = 0
ignoring = False
ignored_lines = []
with open(filename) as file:
while True:
line_number += 1
line = file.readline()
if not line:
break
if regex_block_start.match(line) is not None:
if ignoring:
print("Error in line " + str(line_number) + ": start ignoring while already ignoring")
exit(1)
ignoring = True
if regex_block_end.match(line) is not None:
if not ignoring:
print("Error in line " + str(line_number) + ": end ignoring while not ignoring")
exit(1)
ignoring = False
if regex_next_line.match(line) is not None:
if ignoring:
print("Error in line " + str(line_number) + ": ignoring next line while already ignoring block")
exit(1)
ignored_lines.append(line_number + 1)
# ignore empty return statements in precompiled code
# those are inserted automatically by GAP and are never executed
if "precompiled_categories" in filename and regex_return_void.match(line) is not None:
if ignoring:
print("Error in line " + str(line_number) + ": ignoring empty return statement while already ignoring block")
exit(1)
ignored_lines.append(line_number)
if ignoring:
ignored_lines.append(line_number)
if ignoring:
print("Error: ignoring until the end of the file")
exit(1)
for line_number in ignored_lines:
if str(line_number) in lines_covered:
if lines_covered[str(line_number)] != "0":
print("Error in line " + str(line_number) + ": an ignored line is covered")
exit(1)
del lines_covered[str(line_number)]
with open(coverage_filename, "w") as outfile:
json.dump(new_data, outfile, indent=0)