forked from pypa/bandersnatch
-
Notifications
You must be signed in to change notification settings - Fork 0
/
test_runner.py
163 lines (129 loc) · 4.92 KB
/
test_runner.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
#!/usr/bin/env python
"""
bandersnatch CI run script - Will either drive `tox` or run an Integration Test
- Rewritten in Python for easier dev contributions + Windows support
Integration Tests will go off and hit PyPI + pull allowlisted packages
then check for expected outputs to exist
"""
import json
from configparser import ConfigParser
from os import environ
from pathlib import Path
from shutil import rmtree, which
from subprocess import run
from sys import exit
from tempfile import gettempdir
from src.bandersnatch.utils import hash
BANDERSNATCH_EXE = Path(
which("bandersnatch") or which("bandersnatch.exe") or "bandersnatch"
)
CI_CONFIG = Path("src/bandersnatch/tests/ci.conf")
EOP = "[CI ERROR]:"
MIRROR_ROOT = Path(f"{gettempdir()}/pypi")
MIRROR_BASE = MIRROR_ROOT / "web"
TGZ_SHA256 = "b6114554fb312f9b0bdeaf6a7498f7da05fc17b9250c0449ed796fac9ab663e2"
TOX_EXE = Path(which("tox") or "tox")
# Make Global so we can check exists before delete
A_BLACK_WHL = (
MIRROR_BASE
/ "packages"
/ "1b"
/ "cf"
/ "f1c3925f35a63bede25a17ecfed4529bf12e14fdf592917b36c17caf0b30"
/ "black-24.4.1-cp312-cp312-macosx_11_0_arm64.whl"
)
def check_ci(suppress_errors: bool = False) -> int:
black_index = MIRROR_BASE / "simple/b/black/index.html"
pyaib_index = MIRROR_BASE / "simple/p/pyaib/index.html"
pyaib_json_index = MIRROR_BASE / "simple/p/pyaib/index.v1_json"
pyaib_json = MIRROR_BASE / "json/pyaib"
pyaib_tgz = (
MIRROR_BASE
/ "packages"
/ "0c"
/ "af"
/ "0389466685844d95c6f1f857008d4931d14c7937ac8dba689639ccf0cc54"
/ "pyaib-2.1.0.tar.gz"
)
if not suppress_errors and not pyaib_index.exists():
print(f"{EOP} No pyaib simple API index exists @ {pyaib_index}")
return 69
if not suppress_errors and not pyaib_json.exists():
print(f"{EOP} No pyaib JSON API file exists @ {pyaib_json}")
return 70
if not suppress_errors and not pyaib_tgz.exists():
print(f"{EOP} No pyaib tgz file exists @ {pyaib_tgz}")
return 71
pyaib_tgz_sha256 = hash(pyaib_tgz)
if not suppress_errors and pyaib_tgz_sha256 != TGZ_SHA256:
print(f"{EOP} Bad pyaib 1.0.0 sha256: {pyaib_tgz_sha256} != {TGZ_SHA256}")
return 72
if not suppress_errors and black_index.exists():
print(f"{EOP} {black_index} exists ... delete failed?")
return 73
if not suppress_errors and A_BLACK_WHL.exists():
print(f"{EOP} {A_BLACK_WHL} exists ... delete failed?")
return 74
if not suppress_errors and not pyaib_json_index.exists():
print(f"{EOP} {pyaib_json_index} does not exist ...")
return 75
else:
with pyaib_json_index.open("r") as fp:
json.load(fp) # Check it's valid JSON
rmtree(MIRROR_ROOT)
print("Bandersnatch PyPI CI finished successfully!")
return 0
def do_ci(conf: Path, suppress_errors: bool = False) -> int:
if not conf.exists():
print(f"CI config {conf} does not exist for bandersnatch run")
return 2
print("Starting CI bandersnatch mirror ...")
cmds = (str(BANDERSNATCH_EXE), "--config", str(conf), "--debug", "mirror")
print(f"bandersnatch cmd: {' '.join(cmds)}")
run(cmds, check=not suppress_errors)
print(f"Checking if {A_BLACK_WHL} exists")
if not A_BLACK_WHL.exists():
print(f"{EOP} {A_BLACK_WHL} does not exist after mirroring ...")
if not suppress_errors:
return 68
print("Starting to deleting black from mirror ...")
del_cmds = (
str(BANDERSNATCH_EXE),
"--config",
str(conf),
"--debug",
"delete",
"black",
)
print(f"bandersnatch delete cmd: {' '.join(cmds)}")
run(del_cmds, check=not suppress_errors)
return check_ci(suppress_errors)
def platform_config() -> Path:
"""Ensure the CI_CONFIG is correct for the platform we're running on"""
platform_ci_conf = MIRROR_ROOT / "ci.conf"
cp = ConfigParser()
cp.read(str(CI_CONFIG))
print(f"Setting CI directory={MIRROR_ROOT}")
cp["mirror"]["directory"] = str(MIRROR_ROOT)
with platform_ci_conf.open("w") as pccfp:
cp.write(pccfp)
return platform_ci_conf
def main() -> int:
if "TOXENV" not in environ:
print("No TOXENV set. Exiting!")
return 1
# GitHub Actions does not have a nice way to ignore failures
# like TravisCI has. So will start with ignoring all 3.10-dev failures
# and maybe remove this once we get everything to pass
suppress_errors = bool(environ.get("SUPPRESS_ERRORS", False))
if environ["TOXENV"] != "INTEGRATION":
returncode = run((str(TOX_EXE),)).returncode
if not suppress_errors:
return returncode
return 0
else:
print("Running Ingtegration tests due to TOXENV set to INTEGRATION")
MIRROR_ROOT.mkdir(exist_ok=True)
return do_ci(platform_config(), suppress_errors)
if __name__ == "__main__":
exit(main())