-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathnoxfile.py
206 lines (170 loc) · 6.22 KB
/
noxfile.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
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
from __future__ import annotations
import argparse
import functools
import os
import re
import subprocess
import sys
import tempfile
from pathlib import Path
import nox
os.environ["PDM_IGNORE_SAVED_PYTHON"] = "1"
CI = os.environ.get("CI") is not None
ROOT = Path(".")
MAIN_BRANCH_NAME = "master"
PYTHON_VERSIONS = ["3.9", "3.11", "3.12"]
PYTHON_DEFAULT_VERSION = PYTHON_VERSIONS[-1]
nox.options.default_venv_backend = "venv"
nox.options.stop_on_first_error = True
nox.options.reuse_existing_virtualenvs = not CI
if CI:
# In CI, use Python interpreter provided by GitHub Actions
PYTHON_VERSIONS = [sys.executable]
def install(session: nox.Session, *groups, dev: bool = True, editable: bool = False, no_self=False, no_default=False):
other_args = []
if not dev:
other_args.append("--prod")
if not editable:
other_args.append("--no-editable")
if no_self:
other_args.append("--no-self")
if no_default:
other_args.append("--no-default")
for group in groups:
other_args.extend(["--group", group])
session.run("pdm", "install", "--check", *other_args, external=True)
@functools.lru_cache
def _list_files() -> list[Path]:
file_list = []
for cmd in (
["git", "ls-files"],
["git", "ls-files", "--others", "--exclude-standard"],
):
cmd_result = subprocess.run(cmd, check=True, text=True, capture_output=True)
file_list.extend(cmd_result.stdout.splitlines())
return [Path(p) for p in file_list]
def list_files(suffix: str | None = None) -> list[Path]:
"""List all non-files not-ignored by git."""
file_paths = _list_files()
if suffix is not None:
file_paths = [p for p in file_paths if p.suffix == suffix]
return file_paths
def run_readable(session, mode="check"):
session.run(
"docker",
"run",
"--platform",
"linux/amd64",
"--rm",
"-v",
f"{ROOT.absolute()}:/data",
"-w",
"/data",
"ghcr.io/bobheadxi/readable:v0.5.0@sha256:423c133e7e9ca0ac20b0ab298bd5dbfa3df09b515b34cbfbbe8944310cc8d9c9",
mode,
"![.]**/*.md",
external=True,
)
def run_shellcheck(session, mode="check"):
shellcheck_cmd = [
"docker",
"run",
"--platform",
"linux/amd64", # while this image is multi-arch, we cannot use digest with multi-arch images
"--rm",
"-v",
f"{ROOT.absolute()}:/mnt",
"-w",
"/mnt",
"-q",
"koalaman/shellcheck:0.9.0@sha256:a527e2077f11f28c1c1ad1dc784b5bc966baeb3e34ef304a0ffa72699b01ad9c",
]
files = list_files(suffix=".sh")
if not files:
session.log("No shell files found")
return
shellcheck_cmd.extend(files)
if mode == "fmt":
with tempfile.NamedTemporaryFile(mode="w+") as diff_file:
session.run(
*shellcheck_cmd,
"--format=diff",
external=True,
stdout=diff_file,
success_codes=[0, 1],
)
diff_file.seek(0)
diff = diff_file.read()
if len(diff.splitlines()) > 1: # ignore single-line message
session.log("Applying shellcheck patch:\n%s", diff)
subprocess.run(
["patch", "-p1"],
input=diff,
text=True,
check=True,
)
session.run(*shellcheck_cmd, external=True)
@nox.session(name="format", python=PYTHON_DEFAULT_VERSION, tags=["format", "check"])
def format_(session):
"""Lint the code and apply fixes in-place whenever possible."""
install(session, "lint", no_self=True, no_default=True)
session.run("ruff", "check", "--fix", ".")
run_shellcheck(session, mode="fmt")
run_readable(session, mode="fmt")
session.run("ruff", "format", ".")
@nox.session(python=PYTHON_DEFAULT_VERSION, tags=["lint", "check"])
def lint(session):
"""Run linters in readonly mode."""
# "test" group is required for mypy to work against test files
install(session, "lint", "test")
session.run("ruff", "check", "--diff", "--unsafe-fixes", ".")
session.run("ruff", "format", "--diff", ".")
session.run("mypy", ".")
session.run("codespell", ".")
run_shellcheck(session, mode="check")
run_readable(session, mode="check")
@nox.session(python=PYTHON_VERSIONS, tags=["test", "check"])
def test(session):
install(session, "test")
session.run("pytest", "-vv", "-n", "auto", *session.posargs)
@nox.session(python=PYTHON_DEFAULT_VERSION)
def make_release(session):
install(session, "release", no_self=True, no_default=True)
parser = argparse.ArgumentParser()
def version(value):
if not re.match(r"\d+\.\d+\.\d+(?:(?:a|b|rc)\d+)?", value):
raise argparse.ArgumentTypeError("Invalid version format")
return value
parser.add_argument(
"release_version",
help="Release version in semver format (e.g. 1.2.3)",
type=version,
)
parser.add_argument(
"--draft",
action="store_true",
help="Create a draft release",
)
parsed_args = parser.parse_args(session.posargs)
local_changes = subprocess.check_output(["git", "diff", "--stat"])
if local_changes:
session.error("Uncommitted changes detected")
current_branch = subprocess.check_output(["git", "rev-parse", "--abbrev-ref", "HEAD"], text=True).strip()
if current_branch != MAIN_BRANCH_NAME:
session.warn(f"Releasing from a branch {current_branch!r}, while main branch is {MAIN_BRANCH_NAME!r}")
if not parsed_args.draft:
session.error("Only draft releases are allowed from non-main branch")
session.run("towncrier", "build", "--yes", "--version", parsed_args.release_version)
if parsed_args.draft:
tag = f"draft/v{parsed_args.release_version}"
message = f"Draft release {tag}"
else:
tag = f"v{parsed_args.release_version}"
message = f"release {tag}"
session.log(
f"CHANGELOG updated, please review changes, and execute when ready:\n"
f" git commit -m {message!r}\n"
f" git push origin {current_branch}\n"
f" git tag {tag}\n"
f" git push origin {tag}\n"
)