from __future__ import annotations import os import sys from datetime import datetime from pathlib import Path from subprocess import CalledProcessError from typing import Callable from unittest.mock import MagicMock import pytest from pytest_mock import MockerFixture from virtualenv.app_data import AppDataDiskFolder from virtualenv.seed.wheels.acquire import download_wheel, get_wheel, pip_wheel_env_run from virtualenv.seed.wheels.embed import BUNDLE_FOLDER, get_embed_wheel from virtualenv.seed.wheels.periodic_update import dump_datetime from virtualenv.seed.wheels.util import Wheel, discover_wheels @pytest.fixture(autouse=True) def _fake_release_date(mocker): mocker.patch("virtualenv.seed.wheels.periodic_update.release_date_for_wheel_path", return_value=None) def test_pip_wheel_env_run_could_not_find(session_app_data, mocker): mocker.patch("virtualenv.seed.wheels.acquire.from_bundle", return_value=None) with pytest.raises(RuntimeError, match="could not find the embedded pip"): pip_wheel_env_run([], session_app_data, os.environ) def test_download_wheel_bad_output(mocker, for_py_version, session_app_data): """if the download contains no match for what wheel was downloaded, pick one that matches from target""" distribution = "setuptools" p_open = mocker.MagicMock() mocker.patch("virtualenv.seed.wheels.acquire.Popen", return_value=p_open) p_open.communicate.return_value = "", "" p_open.returncode = 0 embed = get_embed_wheel(distribution, for_py_version) as_path = mocker.MagicMock() available = discover_wheels(BUNDLE_FOLDER, "setuptools", None, for_py_version) as_path.iterdir.return_value = [i.path for i in available] result = download_wheel( distribution, f"=={embed.version}", for_py_version, [], session_app_data, as_path, os.environ, ) assert result.path == embed.path def test_download_fails(mocker, for_py_version, session_app_data): p_open = mocker.MagicMock() mocker.patch("virtualenv.seed.wheels.acquire.Popen", return_value=p_open) p_open.communicate.return_value = "out", "err" p_open.returncode = 1 as_path = mocker.MagicMock() with pytest.raises(CalledProcessError) as context: download_wheel("pip", "==1", for_py_version, [], session_app_data, as_path, os.environ), exc = context.value assert exc.output == "out" assert exc.stderr == "err" assert exc.returncode == 1 assert [ sys.executable, "-m", "pip", "download", "--progress-bar", "off", "--disable-pip-version-check", "--only-binary=:all:", "--no-deps", "--python-version", for_py_version, "-d", str(as_path), "pip==1", ] == exc.cmd @pytest.fixture() def downloaded_wheel(mocker): wheel = Wheel.from_path(Path("setuptools-0.0.0-py2.py3-none-any.whl")) return wheel, mocker.patch("virtualenv.seed.wheels.acquire.download_wheel", return_value=wheel) @pytest.mark.parametrize("version", ["bundle", "0.0.0"]) def test_get_wheel_download_called(mocker, for_py_version, session_app_data, downloaded_wheel, version): distribution = "setuptools" write = mocker.patch("virtualenv.app_data.via_disk_folder.JSONStoreDisk.write") wheel = get_wheel(distribution, version, for_py_version, [], True, session_app_data, False, os.environ) assert wheel is not None assert wheel.name == downloaded_wheel[0].name assert downloaded_wheel[1].call_count == 1 assert write.call_count == 1 @pytest.mark.parametrize("version", ["embed", "pinned"]) def test_get_wheel_download_not_called(mocker, for_py_version, session_app_data, downloaded_wheel, version): distribution = "setuptools" expected = get_embed_wheel(distribution, for_py_version) if version == "pinned": version = expected.version write = mocker.patch("virtualenv.app_data.via_disk_folder.JSONStoreDisk.write") wheel = get_wheel(distribution, version, for_py_version, [], True, session_app_data, False, os.environ) assert wheel is not None assert wheel.name == expected.name assert downloaded_wheel[1].call_count == 0 assert write.call_count == 0 def test_get_wheel_download_cached( tmp_path: Path, mocker: MockerFixture, for_py_version: str, downloaded_wheel: tuple[Wheel, MagicMock], time_freeze: Callable[[datetime], None], ) -> None: time_freeze(datetime.now()) from virtualenv.app_data.via_disk_folder import JSONStoreDisk app_data = AppDataDiskFolder(folder=str(tmp_path)) expected = downloaded_wheel[0] write = mocker.spy(JSONStoreDisk, "write") # 1st call, not cached, download is called wheel = get_wheel(expected.distribution, expected.version, for_py_version, [], True, app_data, False, os.environ) assert wheel is not None assert wheel.name == expected.name assert downloaded_wheel[1].call_count == 1 assert write.call_count == 1 # 2nd call, cached, download is not called wheel = get_wheel(expected.distribution, expected.version, for_py_version, [], True, app_data, False, os.environ) assert wheel is not None assert wheel.name == expected.name assert downloaded_wheel[1].call_count == 1 assert write.call_count == 1 wrote_json = write.call_args[0][1] assert wrote_json == { "completed": None, "periodic": None, "started": None, "versions": [ { "filename": expected.name, "release_date": None, "found_date": dump_datetime(datetime.now()), "source": "download", }, ], }