From a08c4be3f8018beea2bb34120e91e5c70cef22db Mon Sep 17 00:00:00 2001 From: Tzu-ping Chung Date: Thu, 23 Apr 2020 12:01:05 +0800 Subject: [PATCH 1/4] Basic abstraction and one migration --- src/pip/_internal/metadata/__init__.py | 15 +++++++ src/pip/_internal/metadata/base.py | 29 +++++++++++++ src/pip/_internal/metadata/pkg_resources.py | 46 +++++++++++++++++++++ src/pip/_internal/self_outdated_check.py | 10 ++--- tests/unit/test_self_check_outdated.py | 12 +++++- 5 files changed, 104 insertions(+), 8 deletions(-) create mode 100644 src/pip/_internal/metadata/__init__.py create mode 100644 src/pip/_internal/metadata/base.py create mode 100644 src/pip/_internal/metadata/pkg_resources.py diff --git a/src/pip/_internal/metadata/__init__.py b/src/pip/_internal/metadata/__init__.py new file mode 100644 index 00000000000..84e91d6aebc --- /dev/null +++ b/src/pip/_internal/metadata/__init__.py @@ -0,0 +1,15 @@ +from pip._internal.utils.typing import MYPY_CHECK_RUNNING + +if MYPY_CHECK_RUNNING: + from typing import List, Optional + + from .base import BaseEnvironment + + +def get_environment(paths=None): + # type: (Optional[List[str]]) -> BaseEnvironment + from .pkg_resources import Environment + + if paths is None: + return Environment.default() + return Environment.from_paths(paths) diff --git a/src/pip/_internal/metadata/base.py b/src/pip/_internal/metadata/base.py new file mode 100644 index 00000000000..6d387220fab --- /dev/null +++ b/src/pip/_internal/metadata/base.py @@ -0,0 +1,29 @@ +from pip._internal.utils.typing import MYPY_CHECK_RUNNING + +if MYPY_CHECK_RUNNING: + from typing import List, Optional + + +class BaseDistribution: + @property + def installer(self): + # type: () -> str + raise NotImplementedError() + + +class BaseEnvironment: + """An environment containing distributions to introspect.""" + + @classmethod + def default(cls): + # type: () -> BaseEnvironment + raise NotImplementedError() + + @classmethod + def from_paths(cls, paths): + # type: (List[str]) -> BaseEnvironment + raise NotImplementedError() + + def get_distribution(self, name): + # type: (str) -> Optional[BaseDistribution] + raise NotImplementedError() diff --git a/src/pip/_internal/metadata/pkg_resources.py b/src/pip/_internal/metadata/pkg_resources.py new file mode 100644 index 00000000000..02c32876a0f --- /dev/null +++ b/src/pip/_internal/metadata/pkg_resources.py @@ -0,0 +1,46 @@ +from pip._vendor import pkg_resources + +from pip._internal.utils.packaging import get_installer +from pip._internal.utils.typing import MYPY_CHECK_RUNNING + +from .base import BaseDistribution, BaseEnvironment + +if MYPY_CHECK_RUNNING: + from typing import List, Optional + + +class Distribution(BaseDistribution): + def __init__(self, dist): + # type: (pkg_resources.Distribution) -> None + self._dist = dist + + @property + def installer(self): + # type: () -> str + # TODO: Move get_installer() implementation here. + return get_installer(self._dist) + + +class Environment(BaseEnvironment): + def __init__(self, ws): + # type: (pkg_resources.WorkingSet) -> None + self._ws = ws + + @classmethod + def default(cls): + # type: () -> BaseEnvironment + return cls(pkg_resources.working_set) + + @classmethod + def from_paths(cls, paths): + # type: (List[str]) -> BaseEnvironment + return cls(pkg_resources.WorkingSet(paths)) + + def get_distribution(self, name): + # type: (str) -> Optional[BaseDistribution] + req = pkg_resources.Requirement(name) + try: + dist = self._ws.find(req) + except pkg_resources.DistributionNotFound: + return None + return Distribution(dist) diff --git a/src/pip/_internal/self_outdated_check.py b/src/pip/_internal/self_outdated_check.py index c22f06afe87..1819886591c 100644 --- a/src/pip/_internal/self_outdated_check.py +++ b/src/pip/_internal/self_outdated_check.py @@ -10,10 +10,10 @@ from pip._internal.index.collector import LinkCollector from pip._internal.index.package_finder import PackageFinder +from pip._internal.metadata import get_environment from pip._internal.models.selection_prefs import SelectionPreferences from pip._internal.utils.filesystem import adjacent_tmp_file, check_path_owner, replace -from pip._internal.utils.misc import ensure_dir, get_distribution, get_installed_version -from pip._internal.utils.packaging import get_installer +from pip._internal.utils.misc import ensure_dir, get_installed_version from pip._internal.utils.typing import MYPY_CHECK_RUNNING if MYPY_CHECK_RUNNING: @@ -103,10 +103,8 @@ def was_installed_by_pip(pkg): This is used not to display the upgrade message when pip is in fact installed by system package manager, such as dnf on Fedora. """ - dist = get_distribution(pkg) - if not dist: - return False - return "pip" == get_installer(dist) + dist = get_environment().get_distribution(pkg) + return dist is not None and "pip" == dist.installer def pip_self_version_check(session, options): diff --git a/tests/unit/test_self_check_outdated.py b/tests/unit/test_self_check_outdated.py index 42c4c452726..35a94a36b0f 100644 --- a/tests/unit/test_self_check_outdated.py +++ b/tests/unit/test_self_check_outdated.py @@ -57,6 +57,14 @@ def get_metadata_lines(self, name): raise NotImplementedError('nope') +class MockEnvironment(object): + def __init__(self, installer): + self.installer = installer + + def get_distribution(self, name): + return MockDistribution(self.installer) + + def _options(): ''' Some default options that we pass to self_outdated_check.pip_self_version_check ''' @@ -97,8 +105,8 @@ def test_pip_self_version_check(monkeypatch, stored_time, installed_ver, pretend.call_recorder(lambda *a, **kw: None)) monkeypatch.setattr(logger, 'debug', pretend.call_recorder(lambda s, exc_info=None: None)) - monkeypatch.setattr(self_outdated_check, 'get_distribution', - lambda name: MockDistribution(installer)) + monkeypatch.setattr(self_outdated_check, 'get_environment', + lambda: MockEnvironment(installer)) fake_state = pretend.stub( state={"last_check": stored_time, 'pypi_version': installed_ver}, From 7e3610461d5966d5e676dd7dc6d9071af6c30e09 Mon Sep 17 00:00:00 2001 From: Tzu-ping Chung Date: Thu, 23 Apr 2020 12:40:31 +0800 Subject: [PATCH 2/4] Some more yet unused implementation --- src/pip/_internal/metadata/base.py | 47 ++++++++++++++++++++- src/pip/_internal/metadata/pkg_resources.py | 30 ++++++++++++- 2 files changed, 74 insertions(+), 3 deletions(-) diff --git a/src/pip/_internal/metadata/base.py b/src/pip/_internal/metadata/base.py index 6d387220fab..312a900b00a 100644 --- a/src/pip/_internal/metadata/base.py +++ b/src/pip/_internal/metadata/base.py @@ -1,15 +1,36 @@ +from pip._internal.utils.misc import stdlib_pkgs # TODO: Move definition here. from pip._internal.utils.typing import MYPY_CHECK_RUNNING if MYPY_CHECK_RUNNING: - from typing import List, Optional + from typing import Container, Iterator, List, Optional class BaseDistribution: + @property + def canonical_name(self): + # type: () -> str + raise NotImplementedError() + @property def installer(self): # type: () -> str raise NotImplementedError() + @property + def editable(self): + # type: () -> bool + raise NotImplementedError() + + @property + def local(self): + # type: () -> bool + raise NotImplementedError() + + @property + def in_usersite(self): + # type: () -> bool + raise NotImplementedError() + class BaseEnvironment: """An environment containing distributions to introspect.""" @@ -27,3 +48,27 @@ def from_paths(cls, paths): def get_distribution(self, name): # type: (str) -> Optional[BaseDistribution] raise NotImplementedError() + + def iter_distributions(self): + # type: () -> Iterator[BaseDistribution] + raise NotImplementedError() + + def iter_installed_distributions( + self, + local_only=True, # type: bool + skip=stdlib_pkgs, # type: Container[str] + include_editables=True, # type: bool + editables_only=False, # type: bool + user_only=False, # type: bool + ): + # type: (...) -> Iterator[BaseDistribution] + it = self.iter_distributions() + if local_only: + it = (d for d in it if d.local) + if not include_editables: + it = (d for d in it if not d.editable) + if editables_only: + it = (d for d in it if d.editable) + if user_only: + it = (d for d in it if d.in_usersite) + return (d for d in it if d.canonical_name not in skip) diff --git a/src/pip/_internal/metadata/pkg_resources.py b/src/pip/_internal/metadata/pkg_resources.py index 02c32876a0f..5462fecde87 100644 --- a/src/pip/_internal/metadata/pkg_resources.py +++ b/src/pip/_internal/metadata/pkg_resources.py @@ -1,12 +1,14 @@ from pip._vendor import pkg_resources +from pip._vendor.packaging.utils import canonicalize_name +from pip._internal.utils.misc import dist_in_usersite, dist_is_editable, dist_is_local from pip._internal.utils.packaging import get_installer from pip._internal.utils.typing import MYPY_CHECK_RUNNING from .base import BaseDistribution, BaseEnvironment if MYPY_CHECK_RUNNING: - from typing import List, Optional + from typing import Iterator, List, Optional class Distribution(BaseDistribution): @@ -14,12 +16,31 @@ def __init__(self, dist): # type: (pkg_resources.Distribution) -> None self._dist = dist + @property + def canonical_name(self): + # type: () -> str + return canonicalize_name(self._dist.project_name) + @property def installer(self): # type: () -> str - # TODO: Move get_installer() implementation here. return get_installer(self._dist) + @property + def editable(self): + # type: () -> bool + return dist_is_editable(self._dist) + + @property + def local(self): + # type: () -> bool + return dist_is_local(self._dist) + + @property + def in_usersite(self): + # type: () -> bool + return dist_in_usersite(self._dist) + class Environment(BaseEnvironment): def __init__(self, ws): @@ -44,3 +65,8 @@ def get_distribution(self, name): except pkg_resources.DistributionNotFound: return None return Distribution(dist) + + def iter_distributions(self): + # type: () -> Iterator[BaseDistribution] + for dist in self._ws: + yield Distribution(dist) From 349bb730ded98f906875c928c7e3d9902fca7d09 Mon Sep 17 00:00:00 2001 From: Tzu-ping Chung Date: Tue, 19 Jan 2021 04:19:32 +0800 Subject: [PATCH 3/4] Move dist-related logic from misc into metadata --- src/pip/_internal/metadata/base.py | 13 +++ src/pip/_internal/metadata/pkg_resources.py | 42 +++++-- src/pip/_internal/utils/misc.py | 120 ++++---------------- tests/unit/test_utils.py | 28 +++-- 4 files changed, 85 insertions(+), 118 deletions(-) diff --git a/src/pip/_internal/metadata/base.py b/src/pip/_internal/metadata/base.py index 312a900b00a..0d4de00764d 100644 --- a/src/pip/_internal/metadata/base.py +++ b/src/pip/_internal/metadata/base.py @@ -47,10 +47,12 @@ def from_paths(cls, paths): def get_distribution(self, name): # type: (str) -> Optional[BaseDistribution] + """Given a requirement name, return the installed distributions.""" raise NotImplementedError() def iter_distributions(self): # type: () -> Iterator[BaseDistribution] + """Iterate through installed distributions.""" raise NotImplementedError() def iter_installed_distributions( @@ -62,6 +64,17 @@ def iter_installed_distributions( user_only=False, # type: bool ): # type: (...) -> Iterator[BaseDistribution] + """Return a list of installed distributions. + + :param local_only: If True (default), only return installations + local to the current virtualenv, if in a virtualenv. + :param skip: An iterable of canonicalized project names to ignore; + defaults to ``stdlib_pkgs``. + :param include_editables: If False, don't report editables. + :param editables_only: If True, only report editables. + :param user_only: If True, only report installations in the user + site directory. + """ it = self.iter_distributions() if local_only: it = (d for d in it if d.local) diff --git a/src/pip/_internal/metadata/pkg_resources.py b/src/pip/_internal/metadata/pkg_resources.py index 5462fecde87..1e73b79f40e 100644 --- a/src/pip/_internal/metadata/pkg_resources.py +++ b/src/pip/_internal/metadata/pkg_resources.py @@ -1,7 +1,7 @@ from pip._vendor import pkg_resources from pip._vendor.packaging.utils import canonicalize_name -from pip._internal.utils.misc import dist_in_usersite, dist_is_editable, dist_is_local +from pip._internal.utils import misc # TODO: Move definition here. from pip._internal.utils.packaging import get_installer from pip._internal.utils.typing import MYPY_CHECK_RUNNING @@ -29,17 +29,17 @@ def installer(self): @property def editable(self): # type: () -> bool - return dist_is_editable(self._dist) + return misc.dist_is_editable(self._dist) @property def local(self): # type: () -> bool - return dist_is_local(self._dist) + return misc.dist_is_local(self._dist) @property def in_usersite(self): # type: () -> bool - return dist_in_usersite(self._dist) + return misc.dist_in_usersite(self._dist) class Environment(BaseEnvironment): @@ -57,14 +57,42 @@ def from_paths(cls, paths): # type: (List[str]) -> BaseEnvironment return cls(pkg_resources.WorkingSet(paths)) + def _search_distribution(self, name): + # type: (str) -> Optional[BaseDistribution] + """Find a distribution matching the ``name`` in the environment. + + This searches from *all* distributions available in the environment, to + match the behavior of ``pkg_resources.get_distribution()``. + """ + canonical_name = canonicalize_name(name) + for dist in self.iter_distributions(): + if dist.canonical_name == canonical_name: + return dist + return None + def get_distribution(self, name): # type: (str) -> Optional[BaseDistribution] - req = pkg_resources.Requirement(name) + + # Search the distribution by looking through the working set. + dist = self._search_distribution(name) + if dist: + return dist + + # If distribution could not be found, call working_set.require to + # update the working set, and try to find the distribution again. + # This might happen for e.g. when you install a package twice, once + # using setup.py develop and again using setup.py install. Now when + # running pip uninstall twice, the package gets removed from the + # working set in the first uninstall, so we have to populate the + # working set again so that pip knows about it and the packages gets + # picked up and is successfully uninstalled the second time too. try: - dist = self._ws.find(req) + # We didn't pass in any version specifiers, so this can never + # raise pkg_resources.VersionConflict. + self._ws.require(name) except pkg_resources.DistributionNotFound: return None - return Distribution(dist) + return self._search_distribution(name) def iter_distributions(self): # type: () -> Iterator[BaseDistribution] diff --git a/src/pip/_internal/utils/misc.py b/src/pip/_internal/utils/misc.py index a3bd49b9139..0cee9156574 100644 --- a/src/pip/_internal/utils/misc.py +++ b/src/pip/_internal/utils/misc.py @@ -18,7 +18,6 @@ from itertools import filterfalse, tee, zip_longest from pip._vendor import pkg_resources -from pip._vendor.packaging.utils import canonicalize_name # NOTE: retrying is not annotated in typeshed as on 2017-07-17, which is # why we ignore the type on this import. @@ -402,86 +401,20 @@ def get_installed_distributions( paths=None # type: Optional[List[str]] ): # type: (...) -> List[Distribution] - """ - Return a list of installed Distribution objects. - - If ``local_only`` is True (default), only return installations - local to the current virtualenv, if in a virtualenv. - - ``skip`` argument is an iterable of lower-case project names to - ignore; defaults to stdlib_pkgs - - If ``include_editables`` is False, don't report editables. - - If ``editables_only`` is True , only report editables. - - If ``user_only`` is True , only report installations in the user - site directory. - - If ``paths`` is set, only report the distributions present at the - specified list of locations. - """ - if paths: - working_set = pkg_resources.WorkingSet(paths) - else: - working_set = pkg_resources.working_set - - if local_only: - local_test = dist_is_local - else: - def local_test(d): - return True - - if include_editables: - def editable_test(d): - return True - else: - def editable_test(d): - return not dist_is_editable(d) - - if editables_only: - def editables_only_test(d): - return dist_is_editable(d) - else: - def editables_only_test(d): - return True - - if user_only: - user_test = dist_in_usersite - else: - def user_test(d): - return True - - return [d for d in working_set - if local_test(d) and - d.key not in skip and - editable_test(d) and - editables_only_test(d) and - user_test(d) - ] - - -def _search_distribution(req_name): - # type: (str) -> Optional[Distribution] - """Find a distribution matching the ``req_name`` in the environment. - - This searches from *all* distributions available in the environment, to - match the behavior of ``pkg_resources.get_distribution()``. - """ - # Canonicalize the name before searching in the list of - # installed distributions and also while creating the package - # dictionary to get the Distribution object - req_name = canonicalize_name(req_name) - packages = get_installed_distributions( - local_only=False, - skip=(), - include_editables=True, - editables_only=False, - user_only=False, - paths=None, + """Return a list of installed Distribution objects. + + Left for compatibility until direct pkg_resources uses are refactored out. + """ + from pip._internal.metadata import get_environment + from pip._internal.metadata.pkg_resources import Distribution as _Dist + dists = get_environment(paths).iter_installed_distributions( + local_only=local_only, + skip=skip, + include_editables=include_editables, + editables_only=editables_only, + user_only=user_only, ) - pkg_dict = {canonicalize_name(p.key): p for p in packages} - return pkg_dict.get(req_name) + return [cast(_Dist, dist)._dist for dist in dists] def get_distribution(req_name): @@ -490,26 +423,15 @@ def get_distribution(req_name): This searches from *all* distributions available in the environment, to match the behavior of ``pkg_resources.get_distribution()``. - """ - # Search the distribution by looking through the working set - dist = _search_distribution(req_name) - - # If distribution could not be found, call working_set.require - # to update the working set, and try to find the distribution - # again. - # This might happen for e.g. when you install a package - # twice, once using setup.py develop and again using setup.py install. - # Now when run pip uninstall twice, the package gets removed - # from the working set in the first uninstall, so we have to populate - # the working set again so that pip knows about it and the packages - # gets picked up and is successfully uninstalled the second time too. - if not dist: - try: - pkg_resources.working_set.require(req_name) - except pkg_resources.DistributionNotFound: - return None - return _search_distribution(req_name) + Left for compatibility until direct pkg_resources uses are refactored out. + """ + from pip._internal.metadata import get_environment + from pip._internal.metadata.pkg_resources import Distribution as _Dist + dist = get_environment().get_distribution(req_name) + if dist is None: + return None + return cast(_Dist, dist)._dist def egg_link_path(dist): diff --git a/tests/unit/test_utils.py b/tests/unit/test_utils.py index 9c43d553143..4caf4cc754b 100644 --- a/tests/unit/test_utils.py +++ b/tests/unit/test_utils.py @@ -196,21 +196,21 @@ def require(self, name): pass workingset = MockWorkingSet(( - Mock(test_name="global", key="global"), - Mock(test_name="editable", key="editable"), - Mock(test_name="normal", key="normal"), - Mock(test_name="user", key="user"), + Mock(test_name="global", project_name="global"), + Mock(test_name="editable", project_name="editable"), + Mock(test_name="normal", project_name="normal"), + Mock(test_name="user", project_name="user"), )) workingset_stdlib = MockWorkingSet(( - Mock(test_name='normal', key='argparse'), - Mock(test_name='normal', key='wsgiref') + Mock(test_name='normal', project_name='argparse'), + Mock(test_name='normal', project_name='wsgiref') )) workingset_freeze = MockWorkingSet(( - Mock(test_name='normal', key='pip'), - Mock(test_name='normal', key='setuptools'), - Mock(test_name='normal', key='distribute') + Mock(test_name='normal', project_name='pip'), + Mock(test_name='normal', project_name='setuptools'), + Mock(test_name='normal', project_name='distribute') )) def dist_is_editable(self, dist): @@ -290,9 +290,13 @@ def test_freeze_excludes(self, mock_dist_is_editable, @pytest.mark.parametrize( "working_set, req_name", itertools.chain( - itertools.product([workingset], (d.key for d in workingset)), itertools.product( - [workingset_stdlib], (d.key for d in workingset_stdlib), + [workingset], + (d.project_name for d in workingset), + ), + itertools.product( + [workingset_stdlib], + (d.project_name for d in workingset_stdlib), ), ), ) @@ -312,7 +316,7 @@ def test_get_distribution( with patch("pip._vendor.pkg_resources.working_set", working_set): dist = get_distribution(req_name) assert dist is not None - assert dist.key == req_name + assert dist.project_name == req_name @patch('pip._vendor.pkg_resources.working_set', workingset) def test_get_distribution_nonexist( From 11e37aa6e1b3afd6a8235327b64122b8398232a1 Mon Sep 17 00:00:00 2001 From: Tzu-ping Chung Date: Tue, 19 Jan 2021 08:13:57 +0800 Subject: [PATCH 4/4] Separate default and ad-hoc environment APIs pkg_resources performs annoying caching that needs to be worked around in some parts of pip. This makes it easier to represent the difference between environments backend by WorkingSet() and working_set. --- src/pip/_internal/metadata/__init__.py | 23 ++++++++++++++++++--- src/pip/_internal/metadata/base.py | 2 +- src/pip/_internal/metadata/pkg_resources.py | 2 +- src/pip/_internal/self_outdated_check.py | 4 ++-- src/pip/_internal/utils/misc.py | 13 ++++++++---- tests/unit/test_self_check_outdated.py | 2 +- 6 files changed, 34 insertions(+), 12 deletions(-) diff --git a/src/pip/_internal/metadata/__init__.py b/src/pip/_internal/metadata/__init__.py index 84e91d6aebc..da2c4355dfc 100644 --- a/src/pip/_internal/metadata/__init__.py +++ b/src/pip/_internal/metadata/__init__.py @@ -6,10 +6,27 @@ from .base import BaseEnvironment -def get_environment(paths=None): +def get_default_environment(): + # type: () -> BaseEnvironment + """Get the default representation for the current environment. + + This returns an Environment instance from the chosen backend. The default + Environment instance should be built from ``sys.path`` and may use caching + to share instance state accorss calls. + """ + from .pkg_resources import Environment + + return Environment.default() + + +def get_environment(paths): # type: (Optional[List[str]]) -> BaseEnvironment + """Get a representation of the environment specified by ``paths``. + + This returns an Environment instance from the chosen backend based on the + given import paths. The backend must build a fresh instance representing + the state of installed distributions when this function is called. + """ from .pkg_resources import Environment - if paths is None: - return Environment.default() return Environment.from_paths(paths) diff --git a/src/pip/_internal/metadata/base.py b/src/pip/_internal/metadata/base.py index 0d4de00764d..955682545cc 100644 --- a/src/pip/_internal/metadata/base.py +++ b/src/pip/_internal/metadata/base.py @@ -42,7 +42,7 @@ def default(cls): @classmethod def from_paths(cls, paths): - # type: (List[str]) -> BaseEnvironment + # type: (Optional[List[str]]) -> BaseEnvironment raise NotImplementedError() def get_distribution(self, name): diff --git a/src/pip/_internal/metadata/pkg_resources.py b/src/pip/_internal/metadata/pkg_resources.py index 1e73b79f40e..d9db2955159 100644 --- a/src/pip/_internal/metadata/pkg_resources.py +++ b/src/pip/_internal/metadata/pkg_resources.py @@ -54,7 +54,7 @@ def default(cls): @classmethod def from_paths(cls, paths): - # type: (List[str]) -> BaseEnvironment + # type: (Optional[List[str]]) -> BaseEnvironment return cls(pkg_resources.WorkingSet(paths)) def _search_distribution(self, name): diff --git a/src/pip/_internal/self_outdated_check.py b/src/pip/_internal/self_outdated_check.py index 1819886591c..e8c8282cbf9 100644 --- a/src/pip/_internal/self_outdated_check.py +++ b/src/pip/_internal/self_outdated_check.py @@ -10,7 +10,7 @@ from pip._internal.index.collector import LinkCollector from pip._internal.index.package_finder import PackageFinder -from pip._internal.metadata import get_environment +from pip._internal.metadata import get_default_environment from pip._internal.models.selection_prefs import SelectionPreferences from pip._internal.utils.filesystem import adjacent_tmp_file, check_path_owner, replace from pip._internal.utils.misc import ensure_dir, get_installed_version @@ -103,7 +103,7 @@ def was_installed_by_pip(pkg): This is used not to display the upgrade message when pip is in fact installed by system package manager, such as dnf on Fedora. """ - dist = get_environment().get_distribution(pkg) + dist = get_default_environment().get_distribution(pkg) return dist is not None and "pip" == dist.installer diff --git a/src/pip/_internal/utils/misc.py b/src/pip/_internal/utils/misc.py index 0cee9156574..5214e9dd79b 100644 --- a/src/pip/_internal/utils/misc.py +++ b/src/pip/_internal/utils/misc.py @@ -405,9 +405,14 @@ def get_installed_distributions( Left for compatibility until direct pkg_resources uses are refactored out. """ - from pip._internal.metadata import get_environment + from pip._internal.metadata import get_default_environment, get_environment from pip._internal.metadata.pkg_resources import Distribution as _Dist - dists = get_environment(paths).iter_installed_distributions( + + if paths is None: + env = get_default_environment() + else: + env = get_environment(paths) + dists = env.iter_installed_distributions( local_only=local_only, skip=skip, include_editables=include_editables, @@ -426,9 +431,9 @@ def get_distribution(req_name): Left for compatibility until direct pkg_resources uses are refactored out. """ - from pip._internal.metadata import get_environment + from pip._internal.metadata import get_default_environment from pip._internal.metadata.pkg_resources import Distribution as _Dist - dist = get_environment().get_distribution(req_name) + dist = get_default_environment().get_distribution(req_name) if dist is None: return None return cast(_Dist, dist)._dist diff --git a/tests/unit/test_self_check_outdated.py b/tests/unit/test_self_check_outdated.py index 35a94a36b0f..2e8663a9c01 100644 --- a/tests/unit/test_self_check_outdated.py +++ b/tests/unit/test_self_check_outdated.py @@ -105,7 +105,7 @@ def test_pip_self_version_check(monkeypatch, stored_time, installed_ver, pretend.call_recorder(lambda *a, **kw: None)) monkeypatch.setattr(logger, 'debug', pretend.call_recorder(lambda s, exc_info=None: None)) - monkeypatch.setattr(self_outdated_check, 'get_environment', + monkeypatch.setattr(self_outdated_check, 'get_default_environment', lambda: MockEnvironment(installer)) fake_state = pretend.stub(