#!/usr/bin/env -S uv run # /// script # dependencies = ["nox>=2025.2.9"] # /// from __future__ import annotations import argparse import contextlib import os from pathlib import Path from typing import TYPE_CHECKING if TYPE_CHECKING: from collections.abc import Generator import nox nox.needs_version = ">=2025.2.9" nox.options.default_venv_backend = "uv|virtualenv" @nox.session(reuse_venv=True) def lint(session: nox.Session) -> None: """ Lint the codebase (except for clang-format/tidy). """ session.install("pre-commit") session.run("pre-commit", "run", "-a", *session.posargs) @nox.session def tests(session: nox.Session) -> None: """ Run the tests (requires a compiler). """ tmpdir = session.create_tmp() session.install("cmake") session.install("-r", "tests/requirements.txt") session.run( "cmake", "-S.", f"-B{tmpdir}", "-DPYBIND11_WERROR=ON", "-DDOWNLOAD_CATCH=ON", "-DDOWNLOAD_EIGEN=ON", *session.posargs, ) session.run("cmake", "--build", tmpdir) session.run("cmake", "--build", tmpdir, "--config=Release", "--target", "check") @nox.session def tests_packaging(session: nox.Session) -> None: """ Run the packaging tests. """ session.install("-r", "tests/requirements.txt", "pip") session.run("pytest", "tests/extra_python_package", *session.posargs) @nox.session(reuse_venv=True, default=False) def docs(session: nox.Session) -> None: """ Build the docs. Pass --non-interactive to avoid serving. """ parser = argparse.ArgumentParser() parser.add_argument( "-b", dest="builder", default="html", help="Build target (default: html)" ) args, posargs = parser.parse_known_args(session.posargs) serve = args.builder == "html" and session.interactive extra_installs = ["sphinx-autobuild"] if serve else [] session.install("-r", "docs/requirements.txt", *extra_installs) session.chdir("docs") shared_args = ( "-n", # nitpicky mode "-T", # full tracebacks f"-b={args.builder}", ".", f"_build/{args.builder}", *posargs, ) if serve: session.run( "sphinx-autobuild", "--open-browser", "--ignore=.build", *shared_args ) else: session.run("sphinx-build", "--keep-going", *shared_args) @nox.session(reuse_venv=True, default=False) def make_changelog(session: nox.Session) -> None: """ Inspect the closed issues and make entries for a changelog. """ session.install_and_run_script("tools/make_changelog.py") @nox.session(reuse_venv=True, default=False) def build(session: nox.Session) -> None: """ Build SDist and wheel. """ session.install("build") session.log("Building normal files") session.run("python", "-m", "build", *session.posargs) @contextlib.contextmanager def preserve_file(filename: Path) -> Generator[str, None, None]: """ Causes a file to be stored and preserved when the context manager exits. """ old_stat = filename.stat() old_file = filename.read_text(encoding="utf-8") try: yield old_file finally: filename.write_text(old_file, encoding="utf-8") os.utime(filename, (old_stat.st_atime, old_stat.st_mtime)) @nox.session(reuse_venv=True) def build_global(session: nox.Session) -> None: """ Build global SDist and wheel. """ installer = ["--installer=uv"] if session.venv_backend == "uv" else [] session.install("build", "tomlkit") session.log("Building pybind11-global files") pyproject = Path("pyproject.toml") with preserve_file(pyproject): newer_txt = session.run("python", "tools/make_global.py", silent=True) assert isinstance(newer_txt, str) pyproject.write_text(newer_txt, encoding="utf-8") session.run( "python", "-m", "build", *installer, *session.posargs, )