-
-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathnoxfile.py
117 lines (98 loc) · 4.4 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
# Copyright (c) 2020 OpenCyphal
# This software is distributed under the terms of the MIT License.
# Author: Pavel Kirienko <pavel@opencyphal.org>
# type: ignore
import os
import sys
import shutil
from pathlib import Path
import nox
ROOT_DIR = Path(__file__).resolve().parent
DEPS_DIR = ROOT_DIR / "tests" / "deps"
assert DEPS_DIR.is_dir(), "Invalid configuration"
PYTHONS = ["3.8", "3.9", "3.10", "3.11"]
@nox.session(python=False)
def clean(session):
wildcards = [
"dist",
"build",
"htmlcov",
".coverage*",
".*cache",
".*compiled",
"*.egg-info",
"*.log",
"*.tmp",
".nox",
]
for w in wildcards:
for f in Path.cwd().glob(w):
session.log(f"Removing: {f}")
shutil.rmtree(f, ignore_errors=True)
@nox.session(python=PYTHONS, reuse_venv=True)
def test(session):
# First, while the environment is clean, install just the tool alone and ensure that it is minimally functional.
# This is needed to catch errors caused by accidental reliance on test dependencies in the main codebase.
# Also in this way we validate that the executable entry points are configured and installed correctly.
session.install("-e", ".")
session.run_always("yakut", "--help", silent=True)
# now, install the optional [joystick] dependency to test MIDI controller/joystick support
session.install("-e", ".[joystick]")
# Now we can install dependencies for the full integration test suite.
session.install(
"pytest ~= 7.4",
"pytest-asyncio ~= 0.21.0",
"coverage ~= 7.4",
)
# The test suite generates a lot of temporary files, so we change the working directory.
# We have to symlink the original setup.cfg as well if we run tools from the new directory.
tmp_dir = Path(session.create_tmp()).resolve()
os.chdir(tmp_dir)
session.log(f"Working directory: {Path.cwd()}")
fn = "setup.cfg"
if not (tmp_dir / fn).exists():
(tmp_dir / fn).symlink_to(ROOT_DIR / fn)
if sys.platform.startswith("linux"):
# Enable packet capture for the Python executable. This is necessary for commands that rely on low-level
# network packet capture, such as the Monitor when used with Cyphal/UDP.
# We do it here because the sudo may ask the user for the password; doing that from the suite is inconvenient.
session.run("sudo", "setcap", "cap_net_raw+eip", str(Path(session.bin, "python").resolve()), external=True)
src_dirs = [(ROOT_DIR / t) for t in ["yakut", "tests"]]
# Run PyTest and make a code coverage report.
# Positional arguments passed to Nox after "--" are forwarded to PyTest as-is.
session.run(
"pytest",
*map(str, src_dirs),
# Log format cannot be specified in setup.cfg https://github.com/pytest-dev/pytest/issues/3062
r"--log-file-format=%(asctime)s %(levelname)-3.3s %(name)s: %(message)s",
*session.posargs,
env={
"PYTHONPATH": str(DEPS_DIR),
"PATH": os.pathsep.join([session.env["PATH"], str(DEPS_DIR)]),
},
)
# The coverage threshold is intentionally set low for interactive runs because when running locally
# in a reused virtualenv the DSDL compiler run may be skipped to save time, resulting in a reduced coverage.
# Some features are not available on Windows so the coverage threshold is set low for it.
if session.posargs or session.interactive or sys.platform.startswith("win"):
fail_under = 1
else:
fail_under = 70
session.run("coverage", "combine")
session.run("coverage", "report", f"--fail-under={fail_under}")
if session.interactive:
session.run("coverage", "html")
report_file = Path.cwd().resolve() / "htmlcov" / "index.html"
session.log(f"COVERAGE REPORT: file://{report_file}")
# MyPy has to be run in the same session because:
# 1. It requires access to the code generated by the test suite.
# 2. It has to be run separately per Python version we support.
# If the interpreter is not CPython, this may need to be conditionally disabled.
session.install("mypy ~= 1.8")
session.run("mypy", "--strict", *map(str, src_dirs))
@nox.session(reuse_venv=True)
def lint(session):
session.install("pylint ~= 3.0.3")
session.run("pylint", "yakut", "tests")
session.install("black ~= 23.12")
session.run("black", "--check", ".")