diff --git a/nox/virtualenv.py b/nox/virtualenv.py index b2897dce..6726642b 100644 --- a/nox/virtualenv.py +++ b/nox/virtualenv.py @@ -19,12 +19,10 @@ import functools import json import os -import platform import re import shutil import subprocess import sys -import sysconfig from pathlib import Path from socket import gethostbyname from typing import TYPE_CHECKING, Any, ClassVar @@ -50,8 +48,6 @@ "UV_SYSTEM_PYTHON", ] ) -_SYSTEM = platform.system() -_IS_MINGW = sysconfig.get_platform().startswith("mingw") def find_uv() -> tuple[bool, str]: @@ -336,7 +332,7 @@ def _clean_location(self) -> bool: def bin_paths(self) -> list[str]: """Returns the location of the conda env's bin folder.""" # see https://github.com/conda/conda/blob/f60f0f1643af04ed9a51da3dd4fa242de81e32f4/conda/activate.py#L563-L572 - if _SYSTEM == "Windows": + if sys.platform.startswith("win"): return [ self.location, os.path.join(self.location, "Library", "mingw-w64", "bin"), @@ -578,12 +574,12 @@ def _resolved_interpreter(self) -> str: # The rest of this is only applicable to Windows, so if we don't have # an interpreter by now, raise. - if _SYSTEM != "Windows": + if not sys.platform.startswith("win"): self._resolved = InterpreterNotFound(self.interpreter) raise self._resolved # Allow versions of the form ``X.Y-32`` for Windows. - match = re.match(r"^\d\.\d+-32?$", cleaned_interpreter) + match = re.match(r"^\d\.\d+-32?$", cleaned_interpreter) # type: ignore[unreachable] if match: # preserve the "-32" suffix, as the Python launcher expects # it. @@ -607,7 +603,7 @@ def _resolved_interpreter(self) -> str: @property def bin_paths(self) -> list[str]: """Returns the location of the virtualenv's bin folder.""" - if _SYSTEM == "Windows" and not _IS_MINGW: + if sys.platform.startswith("win"): return [os.path.join(self.location, "Scripts")] return [os.path.join(self.location, "bin")] diff --git a/noxfile.py b/noxfile.py index 75dd7e3f..7c23c90f 100644 --- a/noxfile.py +++ b/noxfile.py @@ -18,14 +18,13 @@ import contextlib import functools import os -import platform import shutil import sqlite3 import sys import nox -ON_WINDOWS_CI = "CI" in os.environ and platform.system() == "Windows" +ON_WINDOWS_CI = "CI" in os.environ and sys.platform.startswith("win32") nox.needs_version = ">=2024.4.15" nox.options.default_venv_backend = "uv|virtualenv" diff --git a/tests/test_command.py b/tests/test_command.py index 62c783c1..9f5d7aeb 100644 --- a/tests/test_command.py +++ b/tests/test_command.py @@ -39,12 +39,12 @@ PYTHON = sys.executable skip_on_windows_primary_console_session = pytest.mark.skipif( - platform.system() == "Windows" and "SECONDARY_CONSOLE_SESSION" not in os.environ, + sys.platform.startswith("win") and "SECONDARY_CONSOLE_SESSION" not in os.environ, reason="On Windows, this test must run in a separate console session.", ) only_on_windows = pytest.mark.skipif( - platform.system() != "Windows", reason="Only run this test on Windows." + not sys.platform.startswith("win"), reason="Only run this test on Windows." ) @@ -130,7 +130,7 @@ def test_run_verbosity_failed_command( @pytest.mark.skipif( - platform.system() == "Windows", + sys.platform.startswith("win"), reason="See https://github.com/python/cpython/issues/85815", ) def test_run_non_str() -> None: @@ -294,7 +294,7 @@ def enable_ctrl_c(*, enabled: bool) -> None: def interrupt_process(proc: subprocess.Popen[Any]) -> None: """Send SIGINT or CTRL_C_EVENT to the process.""" - if platform.system() == "Windows": + if sys.platform.startswith("win"): # Disable Ctrl-C so we don't terminate ourselves. enable_ctrl_c(enabled=False) @@ -310,7 +310,7 @@ def command_with_keyboard_interrupt( monkeypatch: pytest.MonkeyPatch, marker: Any ) -> None: """Monkeypatch Popen.communicate to raise KeyboardInterrupt.""" - if platform.system() == "Windows": + if sys.platform.startswith("win"): # Enable Ctrl-C because the child inherits the setting from us. enable_ctrl_c(enabled=True) diff --git a/tests/test_virtualenv.py b/tests/test_virtualenv.py index 1e281fa2..36dd8ffc 100644 --- a/tests/test_virtualenv.py +++ b/tests/test_virtualenv.py @@ -40,7 +40,7 @@ from nox.virtualenv import CondaEnv, ProcessEnv, VirtualEnv -IS_WINDOWS = nox.virtualenv._SYSTEM == "Windows" +IS_WINDOWS = sys.platform.startswith("win") HAS_CONDA = shutil.which("conda") is not None HAS_UV = shutil.which("uv") is not None RAISE_ERROR = "RAISE_ERROR" @@ -253,7 +253,7 @@ def test_conda_env_create_verbose( assert kwargs["log"] -@mock.patch("nox.virtualenv._SYSTEM", new="Windows") +@mock.patch("sys.platform", new="win32") def test_condaenv_bin_windows(make_conda: Callable[..., tuple[CondaEnv, Any]]) -> None: venv, dir_ = make_conda() assert [ @@ -413,7 +413,7 @@ def test_bin_paths( assert dir_.join("bin").strpath == venv.bin -@mock.patch("nox.virtualenv._SYSTEM", new="Windows") +@mock.patch("sys.platform", new="win32") def test_bin_windows( make_one: Callable[..., tuple[VirtualEnv | ProcessEnv, Any]], ) -> None: @@ -906,7 +906,7 @@ def test__resolved_interpreter_none( ("2.7.15", "python2.7"), ], ) -@mock.patch("nox.virtualenv._SYSTEM", new="Linux") +@mock.patch("sys.platform", new="linux") @mock.patch.object(shutil, "which", return_value=True) def test__resolved_interpreter_numerical_non_windows( which: mock.Mock, @@ -921,7 +921,7 @@ def test__resolved_interpreter_numerical_non_windows( @pytest.mark.parametrize("input_", ["2.", "2.7."]) -@mock.patch("nox.virtualenv._SYSTEM", new="Linux") +@mock.patch("sys.platform", new="linux") @mock.patch.object(shutil, "which", return_value=False) def test__resolved_interpreter_invalid_numerical_id( which: mock.Mock, @@ -936,7 +936,7 @@ def test__resolved_interpreter_invalid_numerical_id( which.assert_called_once_with(input_) -@mock.patch("nox.virtualenv._SYSTEM", new="Linux") +@mock.patch("sys.platform", new="linux") @mock.patch.object(shutil, "which", return_value=False) def test__resolved_interpreter_32_bit_non_windows( which: mock.Mock, make_one: Callable[..., tuple[VirtualEnv, Any]] @@ -948,7 +948,7 @@ def test__resolved_interpreter_32_bit_non_windows( which.assert_called_once_with("3.6-32") -@mock.patch("nox.virtualenv._SYSTEM", new="Linux") +@mock.patch("sys.platform", new="linux") @mock.patch.object(shutil, "which", return_value=True) def test__resolved_interpreter_non_windows( which: mock.Mock, make_one: Callable[..., tuple[VirtualEnv, Any]] @@ -961,7 +961,7 @@ def test__resolved_interpreter_non_windows( which.assert_called_once_with("python3.6") -@mock.patch("nox.virtualenv._SYSTEM", new="Windows") +@mock.patch("sys.platform", new="win32") @mock.patch.object(shutil, "which") def test__resolved_interpreter_windows_full_path( which: mock.Mock, make_one: Callable[..., tuple[VirtualEnv, Any]] @@ -983,7 +983,7 @@ def test__resolved_interpreter_windows_full_path( ("2.7-32", r"c:\python27\python.exe"), ], ) -@mock.patch("nox.virtualenv._SYSTEM", new="Windows") +@mock.patch("sys.platform", new="win32") @mock.patch.object(subprocess, "run") @mock.patch.object(shutil, "which") def test__resolved_interpreter_windows_pyexe( @@ -1018,7 +1018,7 @@ def special_run(cmd: str, *args: str, **kwargs: object) -> TextProcessResult: which.assert_has_calls([mock.call(input_), mock.call("py")]) -@mock.patch("nox.virtualenv._SYSTEM", new="Windows") +@mock.patch("sys.platform", new="win32") @mock.patch.object(subprocess, "run") @mock.patch.object(shutil, "which") def test__resolved_interpreter_windows_pyexe_fails( @@ -1027,8 +1027,8 @@ def test__resolved_interpreter_windows_pyexe_fails( # Establish that if the py launcher fails, we give the right error. venv, _ = make_one(interpreter="python3.6") - # Trick the nox.virtualenv._SYSTEM into thinking that it cannot find python3.6 - # (it likely will on Unix). Also, when the nox.virtualenv._SYSTEM looks for the + # Trick the nox.virtualenv into thinking that it cannot find python3.6 + # (it likely will on Unix). Also, when the nox.virtualenv looks for the # py launcher, give it a dummy that fails. def special_run(cmd: str, *args: str, **kwargs: object) -> TextProcessResult: # noqa: ARG001 return TextProcessResult("", 1) @@ -1043,7 +1043,7 @@ def special_run(cmd: str, *args: str, **kwargs: object) -> TextProcessResult: # which.assert_has_calls([mock.call("python3.6"), mock.call("py")]) -@mock.patch("nox.virtualenv._SYSTEM", new="Windows") +@mock.patch("sys.platform", new="win32") @mock.patch("nox.virtualenv.UV_PYTHON_SUPPORT", new=False) def test__resolved_interpreter_windows_path_and_version( make_one: Callable[..., tuple[VirtualEnv, Any]], @@ -1072,7 +1072,7 @@ def test__resolved_interpreter_windows_path_and_version( @pytest.mark.parametrize("input_", ["2.7", "python3.7", "goofy"]) @pytest.mark.parametrize("sysfind_result", [r"c:\python37-x64\python.exe", None]) @pytest.mark.parametrize("sysexec_result", ["3.7.3\\n", RAISE_ERROR]) -@mock.patch("nox.virtualenv._SYSTEM", new="Windows") +@mock.patch("sys.platform", new="win32") @mock.patch("nox.virtualenv.UV_PYTHON_SUPPORT", new=False) def test__resolved_interpreter_windows_path_and_version_fails( input_: str, @@ -1096,7 +1096,7 @@ def test__resolved_interpreter_windows_path_and_version_fails( print(venv._resolved_interpreter) -@mock.patch("nox.virtualenv._SYSTEM", new="Windows") +@mock.patch("sys.platform", new="win32") @mock.patch.object(shutil, "which") def test__resolved_interpreter_not_found( which: mock.Mock, make_one: Callable[..., tuple[VirtualEnv, Any]] @@ -1113,7 +1113,7 @@ def test__resolved_interpreter_not_found( print(venv._resolved_interpreter) -@mock.patch("nox.virtualenv._SYSTEM", new="Windows") +@mock.patch("sys.platform", new="win32") @mock.patch("nox.virtualenv.locate_via_py", new=lambda _: None) # type: ignore[misc] # noqa: PT008 def test__resolved_interpreter_nonstandard( make_one: Callable[..., tuple[VirtualEnv, Any]], @@ -1126,7 +1126,7 @@ def test__resolved_interpreter_nonstandard( print(venv._resolved_interpreter) -@mock.patch("nox.virtualenv._SYSTEM", new="Linux") +@mock.patch("sys.platform", new="linux") @mock.patch.object(shutil, "which", return_value=True) def test__resolved_interpreter_cache_result( which: mock.Mock, make_one: Callable[..., tuple[VirtualEnv, Any]] @@ -1142,7 +1142,7 @@ def test__resolved_interpreter_cache_result( assert which.call_count == 1 -@mock.patch("nox.virtualenv._SYSTEM", new="Linux") +@mock.patch("sys.platform", new="linux") @mock.patch.object(shutil, "which", return_value=None) def test__resolved_interpreter_cache_failure( which: mock.Mock, make_one: Callable[..., tuple[VirtualEnv, Any]]