Skip to content

Commit 47a90a5

Browse files
committed
mkvenv: always pass locally-installed packages to pip
Let pip decide whether a new version should be installed or the current one is okay. This ensures that the virtual environment is updated (either upgraded or downgraded) whenever a new version of a package is requested. The hardest part here is figuring out if a package is installed in the venv (which also has to be done twice to account for the presence of either setuptools in Python <3.8, or importlib in Python >=3.8). Suggested-by: Peter Maydell <[email protected]> Cc: John Snow <[email protected]> Signed-off-by: Paolo Bonzini <[email protected]>
1 parent e8e4298 commit 47a90a5

File tree

1 file changed

+74
-2
lines changed

1 file changed

+74
-2
lines changed

python/scripts/mkvenv.py

+74-2
Original file line numberDiff line numberDiff line change
@@ -553,6 +553,74 @@ def pkgname_from_depspec(dep_spec: str) -> str:
553553
return match.group(0)
554554

555555

556+
def _get_path_importlib(package: str) -> Optional[str]:
557+
# pylint: disable=import-outside-toplevel
558+
# pylint: disable=no-name-in-module
559+
# pylint: disable=import-error
560+
try:
561+
# First preference: Python 3.8+ stdlib
562+
from importlib.metadata import ( # type: ignore
563+
PackageNotFoundError,
564+
distribution,
565+
)
566+
except ImportError as exc:
567+
logger.debug("%s", str(exc))
568+
# Second preference: Commonly available PyPI backport
569+
from importlib_metadata import ( # type: ignore
570+
PackageNotFoundError,
571+
distribution,
572+
)
573+
574+
try:
575+
return str(distribution(package).locate_file("."))
576+
except PackageNotFoundError:
577+
return None
578+
579+
580+
def _get_path_pkg_resources(package: str) -> Optional[str]:
581+
# pylint: disable=import-outside-toplevel
582+
# Bundled with setuptools; has a good chance of being available.
583+
import pkg_resources
584+
585+
try:
586+
return str(pkg_resources.get_distribution(package).location)
587+
except pkg_resources.DistributionNotFound:
588+
return None
589+
590+
591+
def _get_path(package: str) -> Optional[str]:
592+
try:
593+
return _get_path_importlib(package)
594+
except ImportError as exc:
595+
logger.debug("%s", str(exc))
596+
597+
try:
598+
return _get_path_pkg_resources(package)
599+
except ImportError as exc:
600+
logger.debug("%s", str(exc))
601+
raise Ouch(
602+
"Neither importlib.metadata nor pkg_resources found. "
603+
"Use Python 3.8+, or install importlib-metadata or setuptools."
604+
) from exc
605+
606+
607+
def _path_is_prefix(prefix: Optional[str], path: str) -> bool:
608+
try:
609+
return (
610+
prefix is not None and os.path.commonpath([prefix, path]) == prefix
611+
)
612+
except ValueError:
613+
return False
614+
615+
616+
def _is_system_package(package: str) -> bool:
617+
path = _get_path(package)
618+
return path is not None and not (
619+
_path_is_prefix(sysconfig.get_path("purelib"), path)
620+
or _path_is_prefix(sysconfig.get_path("platlib"), path)
621+
)
622+
623+
556624
def _get_version_importlib(package: str) -> Optional[str]:
557625
# pylint: disable=import-outside-toplevel
558626
# pylint: disable=no-name-in-module
@@ -741,8 +809,12 @@ def _do_ensure(
741809
for spec in dep_specs:
742810
matcher = distlib.version.LegacyMatcher(spec)
743811
ver = _get_version(matcher.name)
744-
if ver is None or not matcher.match(
745-
distlib.version.LegacyVersion(ver)
812+
if (
813+
ver is None
814+
# Always pass installed package to pip, so that they can be
815+
# updated if the requested version changes
816+
or not _is_system_package(matcher.name)
817+
or not matcher.match(distlib.version.LegacyVersion(ver))
746818
):
747819
absent.append(spec)
748820
else:

0 commit comments

Comments
 (0)