Skip to content

Commit 6af7739

Browse files
authored
Merge pull request #10157 from uranusjr/metadata-refactor-2107
Replace more pkg_resources usages
2 parents 26778e9 + d4d2445 commit 6af7739

24 files changed

+488
-429
lines changed

news/10157.process.rst

Lines changed: 7 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
``pip freeze``, ``pip list``, and ``pip show`` no longer normalize underscore
2+
(``_``) in distribution names to dash (``-``). This is a side effect of the
3+
migration to ``importlib.metadata``, since the underscore-dash normalization
4+
behavior is non-standard and specific to setuptools. This should not affect
5+
other parts of pip (for example, when feeding the ``pip freeze`` result back
6+
into ``pip install``) since pip internally performs standard PEP 503
7+
normalization independently to setuptools.

news/9825.process.rst

Lines changed: 0 additions & 2 deletions
This file was deleted.

src/pip/_internal/build_env.py

Lines changed: 17 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -14,11 +14,13 @@
1414
from typing import TYPE_CHECKING, Iterable, Iterator, List, Optional, Set, Tuple, Type
1515

1616
from pip._vendor.certifi import where
17-
from pip._vendor.pkg_resources import Requirement, VersionConflict, WorkingSet
17+
from pip._vendor.packaging.requirements import Requirement
18+
from pip._vendor.packaging.version import Version
1819

1920
from pip import __file__ as pip_location
2021
from pip._internal.cli.spinners import open_spinner
2122
from pip._internal.locations import get_platlib, get_prefixed_libs, get_purelib
23+
from pip._internal.metadata import get_environment
2224
from pip._internal.utils.subprocess import call_subprocess
2325
from pip._internal.utils.temp_dir import TempDirectory, tempdir_kinds
2426

@@ -167,14 +169,20 @@ def check_requirements(self, reqs):
167169
missing = set()
168170
conflicting = set()
169171
if reqs:
170-
ws = WorkingSet(self._lib_dirs)
171-
for req in reqs:
172-
try:
173-
if ws.find(Requirement.parse(req)) is None:
174-
missing.add(req)
175-
except VersionConflict as e:
176-
conflicting.add((str(e.args[0].as_requirement()),
177-
str(e.args[1])))
172+
env = get_environment(self._lib_dirs)
173+
for req_str in reqs:
174+
req = Requirement(req_str)
175+
dist = env.get_distribution(req.name)
176+
if not dist:
177+
missing.add(req_str)
178+
continue
179+
if isinstance(dist.version, Version):
180+
installed_req_str = f"{req.name}=={dist.version}"
181+
else:
182+
installed_req_str = f"{req.name}==={dist.version}"
183+
if dist.version not in req.specifier:
184+
conflicting.add((installed_req_str, req_str))
185+
# FIXME: Consider direct URL?
178186
return conflicting, missing
179187

180188
def install_requirements(

src/pip/_internal/cli/autocompletion.py

Lines changed: 6 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@
99

1010
from pip._internal.cli.main_parser import create_main_parser
1111
from pip._internal.commands import commands_dict, create_command
12-
from pip._internal.utils.misc import get_installed_distributions
12+
from pip._internal.metadata import get_default_environment
1313

1414

1515
def autocomplete() -> None:
@@ -45,11 +45,13 @@ def autocomplete() -> None:
4545
"uninstall",
4646
]
4747
if should_list_installed:
48+
env = get_default_environment()
4849
lc = current.lower()
4950
installed = [
50-
dist.key
51-
for dist in get_installed_distributions(local_only=True)
52-
if dist.key.startswith(lc) and dist.key not in cwords[1:]
51+
dist.canonical_name
52+
for dist in env.iter_installed_distributions(local_only=True)
53+
if dist.canonical_name.startswith(lc)
54+
and dist.canonical_name not in cwords[1:]
5355
]
5456
# if there are no dists installed, fall back to option completion
5557
if installed:

src/pip/_internal/commands/list.py

Lines changed: 6 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -152,7 +152,7 @@ def run(self, options, args):
152152

153153
skip = set(stdlib_pkgs)
154154
if options.excludes:
155-
skip.update(options.excludes)
155+
skip.update(canonicalize_name(n) for n in options.excludes)
156156

157157
packages: "_ProcessedDists" = [
158158
cast("_DistWithLatestInfo", d)
@@ -199,7 +199,7 @@ def get_not_required(self, packages, options):
199199
dep_keys = {
200200
canonicalize_name(dep.name)
201201
for dist in packages
202-
for dep in dist.iter_dependencies()
202+
for dep in (dist.iter_dependencies() or ())
203203
}
204204

205205
# Create a set to remove duplicate packages, and cast it to a list
@@ -252,10 +252,10 @@ def output_package_listing(self, packages, options):
252252
elif options.list_format == 'freeze':
253253
for dist in packages:
254254
if options.verbose >= 1:
255-
write_output("%s==%s (%s)", dist.canonical_name,
255+
write_output("%s==%s (%s)", dist.raw_name,
256256
dist.version, dist.location)
257257
else:
258-
write_output("%s==%s", dist.canonical_name, dist.version)
258+
write_output("%s==%s", dist.raw_name, dist.version)
259259
elif options.list_format == 'json':
260260
write_output(format_for_json(packages, options))
261261

@@ -297,7 +297,7 @@ def format_for_columns(pkgs, options):
297297
for proj in pkgs:
298298
# if we're working on the 'outdated' list, separate out the
299299
# latest_version and type
300-
row = [proj.canonical_name, str(proj.version)]
300+
row = [proj.raw_name, str(proj.version)]
301301

302302
if running_outdated:
303303
row.append(str(proj.latest_version))
@@ -318,7 +318,7 @@ def format_for_json(packages, options):
318318
data = []
319319
for dist in packages:
320320
info = {
321-
'name': dist.canonical_name,
321+
'name': dist.raw_name,
322322
'version': str(dist.version),
323323
}
324324
if options.verbose >= 1:

src/pip/_internal/commands/show.py

Lines changed: 109 additions & 96 deletions
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,14 @@
1+
import csv
12
import logging
23
import os
3-
from email.parser import FeedParser
44
from optparse import Values
5-
from typing import Dict, Iterator, List
5+
from typing import Iterator, List, NamedTuple, Optional
66

7-
from pip._vendor import pkg_resources
87
from pip._vendor.packaging.utils import canonicalize_name
98

109
from pip._internal.cli.base_command import Command
1110
from pip._internal.cli.status_codes import ERROR, SUCCESS
11+
from pip._internal.metadata import BaseDistribution, get_default_environment
1212
from pip._internal.utils.misc import write_output
1313

1414
logger = logging.getLogger(__name__)
@@ -50,98 +50,111 @@ def run(self, options, args):
5050
return SUCCESS
5151

5252

53-
def search_packages_info(query):
54-
# type: (List[str]) -> Iterator[Dict[str, str]]
53+
class _PackageInfo(NamedTuple):
54+
name: str
55+
version: str
56+
location: str
57+
requires: List[str]
58+
required_by: List[str]
59+
installer: str
60+
metadata_version: str
61+
classifiers: List[str]
62+
summary: str
63+
homepage: str
64+
author: str
65+
author_email: str
66+
license: str
67+
entry_points: List[str]
68+
files: Optional[List[str]]
69+
70+
71+
def search_packages_info(query: List[str]) -> Iterator[_PackageInfo]:
5572
"""
5673
Gather details from installed distributions. Print distribution name,
5774
version, location, and installed files. Installed files requires a
5875
pip generated 'installed-files.txt' in the distributions '.egg-info'
5976
directory.
6077
"""
61-
installed = {}
62-
for p in pkg_resources.working_set:
63-
installed[canonicalize_name(p.project_name)] = p
78+
env = get_default_environment()
6479

80+
installed = {
81+
dist.canonical_name: dist
82+
for dist in env.iter_distributions()
83+
}
6584
query_names = [canonicalize_name(name) for name in query]
6685
missing = sorted(
6786
[name for name, pkg in zip(query, query_names) if pkg not in installed]
6887
)
6988
if missing:
7089
logger.warning('Package(s) not found: %s', ', '.join(missing))
7190

72-
def get_requiring_packages(package_name):
73-
# type: (str) -> List[str]
74-
canonical_name = canonicalize_name(package_name)
91+
def _get_requiring_packages(current_dist: BaseDistribution) -> List[str]:
7592
return [
76-
pkg.project_name for pkg in pkg_resources.working_set
77-
if canonical_name in
78-
[canonicalize_name(required.name) for required in
79-
pkg.requires()]
93+
dist.metadata["Name"] or "UNKNOWN"
94+
for dist in installed.values()
95+
if current_dist.canonical_name in {
96+
canonicalize_name(d.name) for d in dist.iter_dependencies()
97+
}
8098
]
8199

82-
for dist in [installed[pkg] for pkg in query_names if pkg in installed]:
83-
package = {
84-
'name': dist.project_name,
85-
'version': dist.version,
86-
'location': dist.location,
87-
'requires': [dep.project_name for dep in dist.requires()],
88-
'required_by': get_requiring_packages(dist.project_name)
89-
}
90-
file_list = None
91-
metadata = ''
92-
if isinstance(dist, pkg_resources.DistInfoDistribution):
93-
# RECORDs should be part of .dist-info metadatas
94-
if dist.has_metadata('RECORD'):
95-
lines = dist.get_metadata_lines('RECORD')
96-
paths = [line.split(',')[0] for line in lines]
97-
paths = [os.path.join(dist.location, p) for p in paths]
98-
file_list = [os.path.relpath(p, dist.location) for p in paths]
99-
100-
if dist.has_metadata('METADATA'):
101-
metadata = dist.get_metadata('METADATA')
100+
def _files_from_record(dist: BaseDistribution) -> Optional[Iterator[str]]:
101+
try:
102+
text = dist.read_text('RECORD')
103+
except FileNotFoundError:
104+
return None
105+
return (row[0] for row in csv.reader(text.splitlines()))
106+
107+
def _files_from_installed_files(dist: BaseDistribution) -> Optional[Iterator[str]]:
108+
try:
109+
text = dist.read_text('installed-files.txt')
110+
except FileNotFoundError:
111+
return None
112+
return (p for p in text.splitlines(keepends=False) if p)
113+
114+
for query_name in query_names:
115+
try:
116+
dist = installed[query_name]
117+
except KeyError:
118+
continue
119+
120+
try:
121+
entry_points_text = dist.read_text('entry_points.txt')
122+
entry_points = entry_points_text.splitlines(keepends=False)
123+
except FileNotFoundError:
124+
entry_points = []
125+
126+
files_iter = _files_from_record(dist) or _files_from_installed_files(dist)
127+
if files_iter is None:
128+
files: Optional[List[str]] = None
102129
else:
103-
# Otherwise use pip's log for .egg-info's
104-
if dist.has_metadata('installed-files.txt'):
105-
paths = dist.get_metadata_lines('installed-files.txt')
106-
paths = [os.path.join(dist.egg_info, p) for p in paths]
107-
file_list = [os.path.relpath(p, dist.location) for p in paths]
108-
109-
if dist.has_metadata('PKG-INFO'):
110-
metadata = dist.get_metadata('PKG-INFO')
111-
112-
if dist.has_metadata('entry_points.txt'):
113-
entry_points = dist.get_metadata_lines('entry_points.txt')
114-
package['entry_points'] = entry_points
115-
116-
if dist.has_metadata('INSTALLER'):
117-
for line in dist.get_metadata_lines('INSTALLER'):
118-
if line.strip():
119-
package['installer'] = line.strip()
120-
break
121-
122-
# @todo: Should pkg_resources.Distribution have a
123-
# `get_pkg_info` method?
124-
feed_parser = FeedParser()
125-
feed_parser.feed(metadata)
126-
pkg_info_dict = feed_parser.close()
127-
for key in ('metadata-version', 'summary',
128-
'home-page', 'author', 'author-email', 'license'):
129-
package[key] = pkg_info_dict.get(key)
130-
131-
# It looks like FeedParser cannot deal with repeated headers
132-
classifiers = []
133-
for line in metadata.splitlines():
134-
if line.startswith('Classifier: '):
135-
classifiers.append(line[len('Classifier: '):])
136-
package['classifiers'] = classifiers
137-
138-
if file_list:
139-
package['files'] = sorted(file_list)
140-
yield package
141-
142-
143-
def print_results(distributions, list_files=False, verbose=False):
144-
# type: (Iterator[Dict[str, str]], bool, bool) -> bool
130+
files = sorted(os.path.relpath(p, dist.location) for p in files_iter)
131+
132+
metadata = dist.metadata
133+
134+
yield _PackageInfo(
135+
name=dist.raw_name,
136+
version=str(dist.version),
137+
location=dist.location or "",
138+
requires=[req.name for req in dist.iter_dependencies()],
139+
required_by=_get_requiring_packages(dist),
140+
installer=dist.installer,
141+
metadata_version=dist.metadata_version or "",
142+
classifiers=metadata.get_all("Classifier", []),
143+
summary=metadata.get("Summary", ""),
144+
homepage=metadata.get("Home-page", ""),
145+
author=metadata.get("Author", ""),
146+
author_email=metadata.get("Author-email", ""),
147+
license=metadata.get("License", ""),
148+
entry_points=entry_points,
149+
files=files,
150+
)
151+
152+
153+
def print_results(
154+
distributions: Iterator[_PackageInfo],
155+
list_files: bool,
156+
verbose: bool,
157+
) -> bool:
145158
"""
146159
Print the information from installed distributions found.
147160
"""
@@ -151,31 +164,31 @@ def print_results(distributions, list_files=False, verbose=False):
151164
if i > 0:
152165
write_output("---")
153166

154-
write_output("Name: %s", dist.get('name', ''))
155-
write_output("Version: %s", dist.get('version', ''))
156-
write_output("Summary: %s", dist.get('summary', ''))
157-
write_output("Home-page: %s", dist.get('home-page', ''))
158-
write_output("Author: %s", dist.get('author', ''))
159-
write_output("Author-email: %s", dist.get('author-email', ''))
160-
write_output("License: %s", dist.get('license', ''))
161-
write_output("Location: %s", dist.get('location', ''))
162-
write_output("Requires: %s", ', '.join(dist.get('requires', [])))
163-
write_output("Required-by: %s", ', '.join(dist.get('required_by', [])))
167+
write_output("Name: %s", dist.name)
168+
write_output("Version: %s", dist.version)
169+
write_output("Summary: %s", dist.summary)
170+
write_output("Home-page: %s", dist.homepage)
171+
write_output("Author: %s", dist.author)
172+
write_output("Author-email: %s", dist.author_email)
173+
write_output("License: %s", dist.license)
174+
write_output("Location: %s", dist.location)
175+
write_output("Requires: %s", ', '.join(dist.requires))
176+
write_output("Required-by: %s", ', '.join(dist.required_by))
164177

165178
if verbose:
166-
write_output("Metadata-Version: %s",
167-
dist.get('metadata-version', ''))
168-
write_output("Installer: %s", dist.get('installer', ''))
179+
write_output("Metadata-Version: %s", dist.metadata_version)
180+
write_output("Installer: %s", dist.installer)
169181
write_output("Classifiers:")
170-
for classifier in dist.get('classifiers', []):
182+
for classifier in dist.classifiers:
171183
write_output(" %s", classifier)
172184
write_output("Entry-points:")
173-
for entry in dist.get('entry_points', []):
185+
for entry in dist.entry_points:
174186
write_output(" %s", entry.strip())
175187
if list_files:
176188
write_output("Files:")
177-
for line in dist.get('files', []):
178-
write_output(" %s", line.strip())
179-
if "files" not in dist:
180-
write_output("Cannot locate installed-files.txt")
189+
if dist.files is None:
190+
write_output("Cannot locate RECORD or installed-files.txt")
191+
else:
192+
for line in dist.files:
193+
write_output(" %s", line.strip())
181194
return results_printed

src/pip/_internal/metadata/__init__.py

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,14 @@
22

33
from .base import BaseDistribution, BaseEnvironment
44

5+
__all__ = [
6+
"BaseDistribution",
7+
"BaseEnvironment",
8+
"get_default_environment",
9+
"get_environment",
10+
"get_wheel_distribution",
11+
]
12+
513

614
def get_default_environment() -> BaseEnvironment:
715
"""Get the default representation for the current environment.

0 commit comments

Comments
 (0)