Skip to content

Commit c70ecd4

Browse files
committed
cleanup: move terminal summary code to terminal plugin
Fixes #5067.
1 parent b549438 commit c70ecd4

File tree

4 files changed

+157
-160
lines changed

4 files changed

+157
-160
lines changed

src/_pytest/skipping.py

Lines changed: 0 additions & 125 deletions
Original file line numberDiff line numberDiff line change
@@ -183,128 +183,3 @@ def pytest_report_teststatus(report):
183183
return "xfailed", "x", "XFAIL"
184184
elif report.passed:
185185
return "xpassed", "X", "XPASS"
186-
187-
188-
# called by the terminalreporter instance/plugin
189-
190-
191-
def pytest_terminal_summary(terminalreporter):
192-
tr = terminalreporter
193-
if not tr.reportchars:
194-
return
195-
196-
lines = []
197-
for char in tr.reportchars:
198-
action = REPORTCHAR_ACTIONS.get(char, lambda tr, lines: None)
199-
action(terminalreporter, lines)
200-
201-
if lines:
202-
tr._tw.sep("=", "short test summary info")
203-
for line in lines:
204-
tr._tw.line(line)
205-
206-
207-
def show_simple(terminalreporter, lines, stat):
208-
failed = terminalreporter.stats.get(stat)
209-
if failed:
210-
config = terminalreporter.config
211-
for rep in failed:
212-
verbose_word = _get_report_str(config, rep)
213-
pos = _get_pos(config, rep)
214-
lines.append("%s %s" % (verbose_word, pos))
215-
216-
217-
def show_xfailed(terminalreporter, lines):
218-
xfailed = terminalreporter.stats.get("xfailed")
219-
if xfailed:
220-
config = terminalreporter.config
221-
for rep in xfailed:
222-
verbose_word = _get_report_str(config, rep)
223-
pos = _get_pos(config, rep)
224-
lines.append("%s %s" % (verbose_word, pos))
225-
reason = rep.wasxfail
226-
if reason:
227-
lines.append(" " + str(reason))
228-
229-
230-
def show_xpassed(terminalreporter, lines):
231-
xpassed = terminalreporter.stats.get("xpassed")
232-
if xpassed:
233-
config = terminalreporter.config
234-
for rep in xpassed:
235-
verbose_word = _get_report_str(config, rep)
236-
pos = _get_pos(config, rep)
237-
reason = rep.wasxfail
238-
lines.append("%s %s %s" % (verbose_word, pos, reason))
239-
240-
241-
def folded_skips(skipped):
242-
d = {}
243-
for event in skipped:
244-
key = event.longrepr
245-
assert len(key) == 3, (event, key)
246-
keywords = getattr(event, "keywords", {})
247-
# folding reports with global pytestmark variable
248-
# this is workaround, because for now we cannot identify the scope of a skip marker
249-
# TODO: revisit after marks scope would be fixed
250-
if (
251-
event.when == "setup"
252-
and "skip" in keywords
253-
and "pytestmark" not in keywords
254-
):
255-
key = (key[0], None, key[2])
256-
d.setdefault(key, []).append(event)
257-
values = []
258-
for key, events in d.items():
259-
values.append((len(events),) + key)
260-
return values
261-
262-
263-
def show_skipped(terminalreporter, lines):
264-
tr = terminalreporter
265-
skipped = tr.stats.get("skipped", [])
266-
if skipped:
267-
fskips = folded_skips(skipped)
268-
if fskips:
269-
verbose_word = _get_report_str(terminalreporter.config, report=skipped[0])
270-
for num, fspath, lineno, reason in fskips:
271-
if reason.startswith("Skipped: "):
272-
reason = reason[9:]
273-
if lineno is not None:
274-
lines.append(
275-
"%s [%d] %s:%d: %s"
276-
% (verbose_word, num, fspath, lineno + 1, reason)
277-
)
278-
else:
279-
lines.append("%s [%d] %s: %s" % (verbose_word, num, fspath, reason))
280-
281-
282-
def shower(stat):
283-
def show_(terminalreporter, lines):
284-
return show_simple(terminalreporter, lines, stat)
285-
286-
return show_
287-
288-
289-
def _get_report_str(config, report):
290-
_category, _short, verbose = config.hook.pytest_report_teststatus(
291-
report=report, config=config
292-
)
293-
return verbose
294-
295-
296-
def _get_pos(config, rep):
297-
nodeid = config.cwd_relative_nodeid(rep.nodeid)
298-
return nodeid
299-
300-
301-
REPORTCHAR_ACTIONS = {
302-
"x": show_xfailed,
303-
"X": show_xpassed,
304-
"f": shower("failed"),
305-
"F": shower("failed"),
306-
"s": show_skipped,
307-
"S": show_skipped,
308-
"p": shower("passed"),
309-
"E": shower("error"),
310-
}

src/_pytest/terminal.py

Lines changed: 122 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -874,6 +874,128 @@ def summary_stats(self):
874874
self.write_line(msg, **markup)
875875

876876

877+
def pytest_terminal_summary(terminalreporter):
878+
tr = terminalreporter
879+
if not tr.reportchars:
880+
return
881+
882+
lines = []
883+
for char in tr.reportchars:
884+
action = REPORTCHAR_ACTIONS.get(char, lambda tr, lines: None)
885+
action(terminalreporter, lines)
886+
887+
if lines:
888+
tr._tw.sep("=", "short test summary info")
889+
for line in lines:
890+
tr._tw.line(line)
891+
892+
893+
def show_simple(terminalreporter, lines, stat):
894+
failed = terminalreporter.stats.get(stat)
895+
if failed:
896+
config = terminalreporter.config
897+
for rep in failed:
898+
verbose_word = _get_report_str(config, rep)
899+
pos = _get_pos(config, rep)
900+
lines.append("%s %s" % (verbose_word, pos))
901+
902+
903+
def show_xfailed(terminalreporter, lines):
904+
xfailed = terminalreporter.stats.get("xfailed")
905+
if xfailed:
906+
config = terminalreporter.config
907+
for rep in xfailed:
908+
verbose_word = _get_report_str(config, rep)
909+
pos = _get_pos(config, rep)
910+
lines.append("%s %s" % (verbose_word, pos))
911+
reason = rep.wasxfail
912+
if reason:
913+
lines.append(" " + str(reason))
914+
915+
916+
def show_xpassed(terminalreporter, lines):
917+
xpassed = terminalreporter.stats.get("xpassed")
918+
if xpassed:
919+
config = terminalreporter.config
920+
for rep in xpassed:
921+
verbose_word = _get_report_str(config, rep)
922+
pos = _get_pos(config, rep)
923+
reason = rep.wasxfail
924+
lines.append("%s %s %s" % (verbose_word, pos, reason))
925+
926+
927+
def folded_skips(skipped):
928+
d = {}
929+
for event in skipped:
930+
key = event.longrepr
931+
assert len(key) == 3, (event, key)
932+
keywords = getattr(event, "keywords", {})
933+
# folding reports with global pytestmark variable
934+
# this is workaround, because for now we cannot identify the scope of a skip marker
935+
# TODO: revisit after marks scope would be fixed
936+
if (
937+
event.when == "setup"
938+
and "skip" in keywords
939+
and "pytestmark" not in keywords
940+
):
941+
key = (key[0], None, key[2])
942+
d.setdefault(key, []).append(event)
943+
values = []
944+
for key, events in d.items():
945+
values.append((len(events),) + key)
946+
return values
947+
948+
949+
def show_skipped(terminalreporter, lines):
950+
tr = terminalreporter
951+
skipped = tr.stats.get("skipped", [])
952+
if skipped:
953+
fskips = folded_skips(skipped)
954+
if fskips:
955+
verbose_word = _get_report_str(terminalreporter.config, report=skipped[0])
956+
for num, fspath, lineno, reason in fskips:
957+
if reason.startswith("Skipped: "):
958+
reason = reason[9:]
959+
if lineno is not None:
960+
lines.append(
961+
"%s [%d] %s:%d: %s"
962+
% (verbose_word, num, fspath, lineno + 1, reason)
963+
)
964+
else:
965+
lines.append("%s [%d] %s: %s" % (verbose_word, num, fspath, reason))
966+
967+
968+
def shower(stat):
969+
def show_(terminalreporter, lines):
970+
return show_simple(terminalreporter, lines, stat)
971+
972+
return show_
973+
974+
975+
def _get_report_str(config, report):
976+
_category, _short, verbose = config.hook.pytest_report_teststatus(
977+
report=report, config=config
978+
)
979+
return verbose
980+
981+
982+
def _get_pos(config, rep):
983+
nodeid = config.cwd_relative_nodeid(rep.nodeid)
984+
return nodeid
985+
986+
987+
REPORTCHAR_ACTIONS = {
988+
"x": show_xfailed,
989+
"X": show_xpassed,
990+
"f": shower("failed"),
991+
"F": shower("failed"),
992+
"s": show_skipped,
993+
"S": show_skipped,
994+
"p": shower("passed"),
995+
"E": shower("error"),
996+
}
997+
998+
877999
def build_summary_stats_line(stats):
8781000
known_types = (
8791001
"failed passed skipped deselected xfailed xpassed warnings error".split()

testing/test_skipping.py

Lines changed: 0 additions & 35 deletions
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,6 @@
66

77
import pytest
88
from _pytest.runner import runtestprotocol
9-
from _pytest.skipping import folded_skips
109
from _pytest.skipping import MarkEvaluator
1110
from _pytest.skipping import pytest_runtest_setup
1211

@@ -749,40 +748,6 @@ def test_though(self):
749748
result.stdout.fnmatch_lines(["*2 skipped*"])
750749

751750

752-
def test_skip_reasons_folding():
753-
path = "xyz"
754-
lineno = 3
755-
message = "justso"
756-
longrepr = (path, lineno, message)
757-
758-
class X(object):
759-
pass
760-
761-
ev1 = X()
762-
ev1.when = "execute"
763-
ev1.skipped = True
764-
ev1.longrepr = longrepr
765-
766-
ev2 = X()
767-
ev2.when = "execute"
768-
ev2.longrepr = longrepr
769-
ev2.skipped = True
770-
771-
# ev3 might be a collection report
772-
ev3 = X()
773-
ev3.when = "collect"
774-
ev3.longrepr = longrepr
775-
ev3.skipped = True
776-
777-
values = folded_skips([ev1, ev2, ev3])
778-
assert len(values) == 1
779-
num, fspath, lineno, reason = values[0]
780-
assert num == 3
781-
assert fspath == path
782-
assert lineno == lineno
783-
assert reason == message
784-
785-
786751
def test_skipped_reasons_functional(testdir):
787752
testdir.makepyfile(
788753
test_one="""

testing/test_terminal.py

Lines changed: 35 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@
1818
from _pytest.reports import BaseReport
1919
from _pytest.terminal import _plugin_nameversions
2020
from _pytest.terminal import build_summary_stats_line
21+
from _pytest.terminal import folded_skips
2122
from _pytest.terminal import getreportopt
2223
from _pytest.terminal import TerminalReporter
2324

@@ -1524,3 +1525,37 @@ def test_xdist_normal(self, many_files, testdir, monkeypatch):
15241525
monkeypatch.delenv("PYTEST_DISABLE_PLUGIN_AUTOLOAD", raising=False)
15251526
output = testdir.runpytest("-n2")
15261527
output.stdout.re_match_lines([r"[\.E]{40} \s+ \[100%\]"])
1528+
1529+
1530+
def test_skip_reasons_folding():
1531+
path = "xyz"
1532+
lineno = 3
1533+
message = "justso"
1534+
longrepr = (path, lineno, message)
1535+
1536+
class X(object):
1537+
pass
1538+
1539+
ev1 = X()
1540+
ev1.when = "execute"
1541+
ev1.skipped = True
1542+
ev1.longrepr = longrepr
1543+
1544+
ev2 = X()
1545+
ev2.when = "execute"
1546+
ev2.longrepr = longrepr
1547+
ev2.skipped = True
1548+
1549+
# ev3 might be a collection report
1550+
ev3 = X()
1551+
ev3.when = "collect"
1552+
ev3.longrepr = longrepr
1553+
ev3.skipped = True
1554+
1555+
values = folded_skips([ev1, ev2, ev3])
1556+
assert len(values) == 1
1557+
num, fspath, lineno, reason = values[0]
1558+
assert num == 3
1559+
assert fspath == path
1560+
assert lineno == lineno
1561+
assert reason == message

0 commit comments

Comments
 (0)