-
Notifications
You must be signed in to change notification settings - Fork 14.5k
/
Copy pathsummarize_junit_failures.py
executable file
·117 lines (88 loc) · 3.77 KB
/
summarize_junit_failures.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
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
#!/usr/bin/env python
# Licensed to the Apache Software Foundation (ASF) under one
# or more contributor license agreements. See the NOTICE file
# distributed with this work for additional information
# regarding copyright ownership. The ASF licenses this file
# to you 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.
from __future__ import annotations
import os
import re
import sys
import xml.etree.ElementTree as ET
# Do not use anything outsdie of standard library in this file!
from functools import lru_cache
from pathlib import Path
TEXT_RED = "\033[31m"
TEXT_YELLOW = "\033[33m"
TEXT_RESET = "\033[0m"
@lru_cache(maxsize=None)
def translate_classname(classname):
# The pytest xunit output has "classname" in the python sense as 'dir.subdir.package.Class' -- we want to
# convert that back in to a pytest "selector" of dir/subdir/package.py::Class
if not classname:
return None
context = Path.cwd()
parts = classname.split(".")
for offset, component in enumerate(parts, 1): # noqa: B007
candidate = context / component
if candidate.is_dir():
context = candidate
else:
candidate = context / (component + ".py")
if candidate.is_file():
context = candidate
break
parts = parts[offset:]
val = str(context.relative_to(Path.cwd()))
if parts:
val += "::" + ".".join(parts)
return val
def translate_name(testcase):
classname = translate_classname(testcase.get("classname"))
name = testcase.get("name")
if not classname:
# Some times (i.e. collect error) the classname is empty and we only have a name
return translate_classname(name)
return f"{classname}::{name}"
def summarize_file(input, test_type, backend):
root = ET.parse(input)
# <testsuite errors="0" failures="0" hostname="1dea0db81c88" name="pytest" skipped="2" tests="640"
# time="98.678">
testsuite = root.find(".//testsuite")
fail_message_parts = []
num_failures = int(testsuite.get("failures"))
if num_failures:
fail_message_parts.append(f"{num_failures} failure{'' if num_failures == 1 else 's'}")
num_errors = int(testsuite.get("errors"))
if num_errors:
fail_message_parts.append(f"{num_errors} error{'' if num_errors == 1 else 's'}")
if not fail_message_parts:
return
print(f"\n{TEXT_RED}==== {test_type} {backend}: {', '.join(fail_message_parts)} ===={TEXT_RESET}\n")
for testcase in testsuite.findall(".//testcase[error]"):
case_name = translate_name(testcase)
for err in testcase.iterfind("error"):
print(f'{case_name}: {TEXT_YELLOW}{err.get("message")}{TEXT_RESET}')
for testcase in testsuite.findall(".//testcase[failure]"):
case_name = translate_name(testcase)
for failure in testcase.iterfind("failure"):
print(f'{case_name}: {TEXT_YELLOW}{failure.get("message")}{TEXT_RESET}')
if __name__ == "__main__":
fname_pattern = re.compile("^test_result-(?P<test_type>.*?)-(?P<backend>.*).xml$")
for fname in sys.argv[1:]:
match = fname_pattern.match(os.path.basename(fname))
if not match:
exit(f"I cannot understand the name format of {fname!r}")
with open(fname) as fh:
summarize_file(fh, **match.groupdict())