-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathruntests.py
executable file
·172 lines (143 loc) · 4.74 KB
/
runtests.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
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
from __future__ import annotations
import subprocess
from subprocess import Popen
from sys import argv, executable, exit
# Slow test suites
CMDLINE = "PythonCmdline"
PEP561 = "PEP561Suite"
EVALUATION = "PythonEvaluation"
DAEMON = "testdaemon"
STUBGEN_CMD = "StubgenCmdLine"
STUBGEN_PY = "StubgenPythonSuite"
MYPYC_RUN = "TestRun"
MYPYC_RUN_MULTI = "TestRunMultiFile"
MYPYC_EXTERNAL = "TestExternal"
MYPYC_COMMAND_LINE = "TestCommandLine"
MYPYC_SEPARATE = "TestRunSeparate"
MYPYC_MULTIMODULE = "multimodule" # Subset of mypyc run tests that are slow
ERROR_STREAM = "ErrorStreamSuite"
print(executable)
ALL_NON_FAST = [
CMDLINE,
PEP561,
EVALUATION,
DAEMON,
STUBGEN_CMD,
STUBGEN_PY,
MYPYC_RUN,
MYPYC_RUN_MULTI,
MYPYC_EXTERNAL,
MYPYC_COMMAND_LINE,
MYPYC_SEPARATE,
ERROR_STREAM,
]
# This must be enabled by explicitly including 'pytest-extra' on the command line
PYTEST_OPT_IN = [PEP561]
# These must be enabled by explicitly including 'mypyc-extra' on the command line.
MYPYC_OPT_IN = [MYPYC_RUN, MYPYC_RUN_MULTI, MYPYC_SEPARATE]
# These mypyc test filters cover most slow test cases
MYPYC_SLOW = [MYPYC_RUN_MULTI, MYPYC_COMMAND_LINE, MYPYC_SEPARATE, MYPYC_MULTIMODULE]
# We split the pytest run into three parts to improve test
# parallelization. Each run should have tests that each take a roughly similar
# time to run.
cmds = {
# Self type check
# We do a no-strict compatible check, this is needed because generating a baseline from the head of upstream
# sound very laborious, strict check is in tox.ini
"self": [
executable,
"-m",
"mypy",
"--config-file",
"mypy_self_check.ini",
"--baseline-file=",
"-p",
"mypy",
"-p",
"mypyc",
],
# Lint
"lint": ["mypy", "-V"],
# Fast test cases only (this is the bulk of the test suite)
"pytest-fast": ["pytest", "-q", "-k", f"not ({' or '.join(ALL_NON_FAST)})"],
# Test cases that invoke mypy (with small inputs)
"pytest-cmdline": [
"pytest",
"-q",
"-k",
" or ".join([CMDLINE, EVALUATION, STUBGEN_CMD, STUBGEN_PY]),
],
# Test cases that may take seconds to run each
"pytest-slow": [
"pytest",
"-q",
"-k",
" or ".join([DAEMON, MYPYC_EXTERNAL, MYPYC_COMMAND_LINE, ERROR_STREAM]),
],
"mypyc-fast": ["pytest", "-q", "mypyc", "-k", f"not ({' or '.join(MYPYC_SLOW)})"],
# Test cases that might take minutes to run
"pytest-extra": ["pytest", "-q", "-k", " or ".join(PYTEST_OPT_IN)],
# Mypyc tests that aren't run by default, since they are slow and rarely
# fail for commits that don't touch mypyc
"mypyc-extra": ["pytest", "-q", "-k", " or ".join(MYPYC_OPT_IN)],
}
# Stop run immediately if these commands fail
FAST_FAIL = ["self", "lint"]
EXTRA_COMMANDS = ("pytest-extra", "mypyc-fast", "mypyc-extra")
DEFAULT_COMMANDS = [cmd for cmd in cmds if cmd not in EXTRA_COMMANDS]
assert all(cmd in cmds for cmd in FAST_FAIL)
def run_cmd(name: str) -> int:
status = 0
cmd = cmds[name]
print(f"run {name}: {cmd}")
proc = subprocess.run(cmd, stderr=subprocess.STDOUT)
if proc.returncode:
print("\nFAILED: %s" % name)
status = proc.returncode
if name in FAST_FAIL:
exit(status)
return status
def start_background_cmd(name: str) -> Popen:
cmd = cmds[name]
proc = subprocess.Popen(cmd, stderr=subprocess.STDOUT, stdout=subprocess.PIPE)
return proc
def wait_background_cmd(name: str, proc: Popen) -> int:
output = proc.communicate()[0]
status = proc.returncode
print(f"run {name}: {cmds[name]}")
if status:
print(output.decode().rstrip())
print("\nFAILED:", name)
if name in FAST_FAIL:
exit(status)
return status
def main() -> None:
prog, *args = argv
if not set(args).issubset(cmds):
print("usage:", prog, " ".join(f"[{k}]" for k in cmds))
print()
print(
"Run the given tests. If given no arguments, run everything except"
+ " pytest-extra and mypyc-extra."
)
exit(1)
if not args:
args = DEFAULT_COMMANDS.copy()
status = 0
if "self" in args and "lint" in args:
# Perform lint and self check in parallel as it's faster.
proc = start_background_cmd("lint")
cmd_status = run_cmd("self")
if cmd_status:
status = cmd_status
cmd_status = wait_background_cmd("lint", proc)
if cmd_status:
status = cmd_status
args = [arg for arg in args if arg not in ("self", "lint")]
for arg in args:
cmd_status = run_cmd(arg)
if cmd_status:
status = cmd_status
exit(status)
if __name__ == "__main__":
main()