734 lines
		
	
	
		
			23 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			734 lines
		
	
	
		
			23 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """Exceptions used throughout package.
 | ||
| 
 | ||
| This module MUST NOT try to import from anything within `pip._internal` to
 | ||
| operate. This is expected to be importable from any/all files within the
 | ||
| subpackage and, thus, should not depend on them.
 | ||
| """
 | ||
| 
 | ||
| import configparser
 | ||
| import contextlib
 | ||
| import locale
 | ||
| import logging
 | ||
| import pathlib
 | ||
| import re
 | ||
| import sys
 | ||
| from itertools import chain, groupby, repeat
 | ||
| from typing import TYPE_CHECKING, Dict, Iterator, List, Optional, Union
 | ||
| 
 | ||
| from pip._vendor.requests.models import Request, Response
 | ||
| from pip._vendor.rich.console import Console, ConsoleOptions, RenderResult
 | ||
| from pip._vendor.rich.markup import escape
 | ||
| from pip._vendor.rich.text import Text
 | ||
| 
 | ||
| if TYPE_CHECKING:
 | ||
|     from hashlib import _Hash
 | ||
|     from typing import Literal
 | ||
| 
 | ||
|     from pip._internal.metadata import BaseDistribution
 | ||
|     from pip._internal.req.req_install import InstallRequirement
 | ||
| 
 | ||
| logger = logging.getLogger(__name__)
 | ||
| 
 | ||
| 
 | ||
| #
 | ||
| # Scaffolding
 | ||
| #
 | ||
| def _is_kebab_case(s: str) -> bool:
 | ||
|     return re.match(r"^[a-z]+(-[a-z]+)*$", s) is not None
 | ||
| 
 | ||
| 
 | ||
| def _prefix_with_indent(
 | ||
|     s: Union[Text, str],
 | ||
|     console: Console,
 | ||
|     *,
 | ||
|     prefix: str,
 | ||
|     indent: str,
 | ||
| ) -> Text:
 | ||
|     if isinstance(s, Text):
 | ||
|         text = s
 | ||
|     else:
 | ||
|         text = console.render_str(s)
 | ||
| 
 | ||
|     return console.render_str(prefix, overflow="ignore") + console.render_str(
 | ||
|         f"\n{indent}", overflow="ignore"
 | ||
|     ).join(text.split(allow_blank=True))
 | ||
| 
 | ||
| 
 | ||
| class PipError(Exception):
 | ||
|     """The base pip error."""
 | ||
| 
 | ||
| 
 | ||
| class DiagnosticPipError(PipError):
 | ||
|     """An error, that presents diagnostic information to the user.
 | ||
| 
 | ||
|     This contains a bunch of logic, to enable pretty presentation of our error
 | ||
|     messages. Each error gets a unique reference. Each error can also include
 | ||
|     additional context, a hint and/or a note -- which are presented with the
 | ||
|     main error message in a consistent style.
 | ||
| 
 | ||
|     This is adapted from the error output styling in `sphinx-theme-builder`.
 | ||
|     """
 | ||
| 
 | ||
|     reference: str
 | ||
| 
 | ||
|     def __init__(
 | ||
|         self,
 | ||
|         *,
 | ||
|         kind: 'Literal["error", "warning"]' = "error",
 | ||
|         reference: Optional[str] = None,
 | ||
|         message: Union[str, Text],
 | ||
|         context: Optional[Union[str, Text]],
 | ||
|         hint_stmt: Optional[Union[str, Text]],
 | ||
|         note_stmt: Optional[Union[str, Text]] = None,
 | ||
|         link: Optional[str] = None,
 | ||
|     ) -> None:
 | ||
|         # Ensure a proper reference is provided.
 | ||
|         if reference is None:
 | ||
|             assert hasattr(self, "reference"), "error reference not provided!"
 | ||
|             reference = self.reference
 | ||
|         assert _is_kebab_case(reference), "error reference must be kebab-case!"
 | ||
| 
 | ||
|         self.kind = kind
 | ||
|         self.reference = reference
 | ||
| 
 | ||
|         self.message = message
 | ||
|         self.context = context
 | ||
| 
 | ||
|         self.note_stmt = note_stmt
 | ||
|         self.hint_stmt = hint_stmt
 | ||
| 
 | ||
|         self.link = link
 | ||
| 
 | ||
|         super().__init__(f"<{self.__class__.__name__}: {self.reference}>")
 | ||
| 
 | ||
|     def __repr__(self) -> str:
 | ||
|         return (
 | ||
|             f"<{self.__class__.__name__}("
 | ||
|             f"reference={self.reference!r}, "
 | ||
|             f"message={self.message!r}, "
 | ||
|             f"context={self.context!r}, "
 | ||
|             f"note_stmt={self.note_stmt!r}, "
 | ||
|             f"hint_stmt={self.hint_stmt!r}"
 | ||
|             ")>"
 | ||
|         )
 | ||
| 
 | ||
|     def __rich_console__(
 | ||
|         self,
 | ||
|         console: Console,
 | ||
|         options: ConsoleOptions,
 | ||
|     ) -> RenderResult:
 | ||
|         colour = "red" if self.kind == "error" else "yellow"
 | ||
| 
 | ||
|         yield f"[{colour} bold]{self.kind}[/]: [bold]{self.reference}[/]"
 | ||
|         yield ""
 | ||
| 
 | ||
|         if not options.ascii_only:
 | ||
|             # Present the main message, with relevant context indented.
 | ||
|             if self.context is not None:
 | ||
|                 yield _prefix_with_indent(
 | ||
|                     self.message,
 | ||
|                     console,
 | ||
|                     prefix=f"[{colour}]×[/] ",
 | ||
|                     indent=f"[{colour}]│[/] ",
 | ||
|                 )
 | ||
|                 yield _prefix_with_indent(
 | ||
|                     self.context,
 | ||
|                     console,
 | ||
|                     prefix=f"[{colour}]╰─>[/] ",
 | ||
|                     indent=f"[{colour}]   [/] ",
 | ||
|                 )
 | ||
|             else:
 | ||
|                 yield _prefix_with_indent(
 | ||
|                     self.message,
 | ||
|                     console,
 | ||
|                     prefix="[red]×[/] ",
 | ||
|                     indent="  ",
 | ||
|                 )
 | ||
|         else:
 | ||
|             yield self.message
 | ||
|             if self.context is not None:
 | ||
|                 yield ""
 | ||
|                 yield self.context
 | ||
| 
 | ||
|         if self.note_stmt is not None or self.hint_stmt is not None:
 | ||
|             yield ""
 | ||
| 
 | ||
|         if self.note_stmt is not None:
 | ||
|             yield _prefix_with_indent(
 | ||
|                 self.note_stmt,
 | ||
|                 console,
 | ||
|                 prefix="[magenta bold]note[/]: ",
 | ||
|                 indent="      ",
 | ||
|             )
 | ||
|         if self.hint_stmt is not None:
 | ||
|             yield _prefix_with_indent(
 | ||
|                 self.hint_stmt,
 | ||
|                 console,
 | ||
|                 prefix="[cyan bold]hint[/]: ",
 | ||
|                 indent="      ",
 | ||
|             )
 | ||
| 
 | ||
|         if self.link is not None:
 | ||
|             yield ""
 | ||
|             yield f"Link: {self.link}"
 | ||
| 
 | ||
| 
 | ||
| #
 | ||
| # Actual Errors
 | ||
| #
 | ||
| class ConfigurationError(PipError):
 | ||
|     """General exception in configuration"""
 | ||
| 
 | ||
| 
 | ||
| class InstallationError(PipError):
 | ||
|     """General exception during installation"""
 | ||
| 
 | ||
| 
 | ||
| class UninstallationError(PipError):
 | ||
|     """General exception during uninstallation"""
 | ||
| 
 | ||
| 
 | ||
| class MissingPyProjectBuildRequires(DiagnosticPipError):
 | ||
|     """Raised when pyproject.toml has `build-system`, but no `build-system.requires`."""
 | ||
| 
 | ||
|     reference = "missing-pyproject-build-system-requires"
 | ||
| 
 | ||
|     def __init__(self, *, package: str) -> None:
 | ||
|         super().__init__(
 | ||
|             message=f"Can not process {escape(package)}",
 | ||
|             context=Text(
 | ||
|                 "This package has an invalid pyproject.toml file.\n"
 | ||
|                 "The [build-system] table is missing the mandatory `requires` key."
 | ||
|             ),
 | ||
|             note_stmt="This is an issue with the package mentioned above, not pip.",
 | ||
|             hint_stmt=Text("See PEP 518 for the detailed specification."),
 | ||
|         )
 | ||
| 
 | ||
| 
 | ||
| class InvalidPyProjectBuildRequires(DiagnosticPipError):
 | ||
|     """Raised when pyproject.toml an invalid `build-system.requires`."""
 | ||
| 
 | ||
|     reference = "invalid-pyproject-build-system-requires"
 | ||
| 
 | ||
|     def __init__(self, *, package: str, reason: str) -> None:
 | ||
|         super().__init__(
 | ||
|             message=f"Can not process {escape(package)}",
 | ||
|             context=Text(
 | ||
|                 "This package has an invalid `build-system.requires` key in "
 | ||
|                 f"pyproject.toml.\n{reason}"
 | ||
|             ),
 | ||
|             note_stmt="This is an issue with the package mentioned above, not pip.",
 | ||
|             hint_stmt=Text("See PEP 518 for the detailed specification."),
 | ||
|         )
 | ||
| 
 | ||
| 
 | ||
| class NoneMetadataError(PipError):
 | ||
|     """Raised when accessing a Distribution's "METADATA" or "PKG-INFO".
 | ||
| 
 | ||
|     This signifies an inconsistency, when the Distribution claims to have
 | ||
|     the metadata file (if not, raise ``FileNotFoundError`` instead), but is
 | ||
|     not actually able to produce its content. This may be due to permission
 | ||
|     errors.
 | ||
|     """
 | ||
| 
 | ||
|     def __init__(
 | ||
|         self,
 | ||
|         dist: "BaseDistribution",
 | ||
|         metadata_name: str,
 | ||
|     ) -> None:
 | ||
|         """
 | ||
|         :param dist: A Distribution object.
 | ||
|         :param metadata_name: The name of the metadata being accessed
 | ||
|             (can be "METADATA" or "PKG-INFO").
 | ||
|         """
 | ||
|         self.dist = dist
 | ||
|         self.metadata_name = metadata_name
 | ||
| 
 | ||
|     def __str__(self) -> str:
 | ||
|         # Use `dist` in the error message because its stringification
 | ||
|         # includes more information, like the version and location.
 | ||
|         return "None {} metadata found for distribution: {}".format(
 | ||
|             self.metadata_name,
 | ||
|             self.dist,
 | ||
|         )
 | ||
| 
 | ||
| 
 | ||
| class UserInstallationInvalid(InstallationError):
 | ||
|     """A --user install is requested on an environment without user site."""
 | ||
| 
 | ||
|     def __str__(self) -> str:
 | ||
|         return "User base directory is not specified"
 | ||
| 
 | ||
| 
 | ||
| class InvalidSchemeCombination(InstallationError):
 | ||
|     def __str__(self) -> str:
 | ||
|         before = ", ".join(str(a) for a in self.args[:-1])
 | ||
|         return f"Cannot set {before} and {self.args[-1]} together"
 | ||
| 
 | ||
| 
 | ||
| class DistributionNotFound(InstallationError):
 | ||
|     """Raised when a distribution cannot be found to satisfy a requirement"""
 | ||
| 
 | ||
| 
 | ||
| class RequirementsFileParseError(InstallationError):
 | ||
|     """Raised when a general error occurs parsing a requirements file line."""
 | ||
| 
 | ||
| 
 | ||
| class BestVersionAlreadyInstalled(PipError):
 | ||
|     """Raised when the most up-to-date version of a package is already
 | ||
|     installed."""
 | ||
| 
 | ||
| 
 | ||
| class BadCommand(PipError):
 | ||
|     """Raised when virtualenv or a command is not found"""
 | ||
| 
 | ||
| 
 | ||
| class CommandError(PipError):
 | ||
|     """Raised when there is an error in command-line arguments"""
 | ||
| 
 | ||
| 
 | ||
| class PreviousBuildDirError(PipError):
 | ||
|     """Raised when there's a previous conflicting build directory"""
 | ||
| 
 | ||
| 
 | ||
| class NetworkConnectionError(PipError):
 | ||
|     """HTTP connection error"""
 | ||
| 
 | ||
|     def __init__(
 | ||
|         self,
 | ||
|         error_msg: str,
 | ||
|         response: Optional[Response] = None,
 | ||
|         request: Optional[Request] = None,
 | ||
|     ) -> None:
 | ||
|         """
 | ||
|         Initialize NetworkConnectionError with  `request` and `response`
 | ||
|         objects.
 | ||
|         """
 | ||
|         self.response = response
 | ||
|         self.request = request
 | ||
|         self.error_msg = error_msg
 | ||
|         if (
 | ||
|             self.response is not None
 | ||
|             and not self.request
 | ||
|             and hasattr(response, "request")
 | ||
|         ):
 | ||
|             self.request = self.response.request
 | ||
|         super().__init__(error_msg, response, request)
 | ||
| 
 | ||
|     def __str__(self) -> str:
 | ||
|         return str(self.error_msg)
 | ||
| 
 | ||
| 
 | ||
| class InvalidWheelFilename(InstallationError):
 | ||
|     """Invalid wheel filename."""
 | ||
| 
 | ||
| 
 | ||
| class UnsupportedWheel(InstallationError):
 | ||
|     """Unsupported wheel."""
 | ||
| 
 | ||
| 
 | ||
| class InvalidWheel(InstallationError):
 | ||
|     """Invalid (e.g. corrupt) wheel."""
 | ||
| 
 | ||
|     def __init__(self, location: str, name: str):
 | ||
|         self.location = location
 | ||
|         self.name = name
 | ||
| 
 | ||
|     def __str__(self) -> str:
 | ||
|         return f"Wheel '{self.name}' located at {self.location} is invalid."
 | ||
| 
 | ||
| 
 | ||
| class MetadataInconsistent(InstallationError):
 | ||
|     """Built metadata contains inconsistent information.
 | ||
| 
 | ||
|     This is raised when the metadata contains values (e.g. name and version)
 | ||
|     that do not match the information previously obtained from sdist filename,
 | ||
|     user-supplied ``#egg=`` value, or an install requirement name.
 | ||
|     """
 | ||
| 
 | ||
|     def __init__(
 | ||
|         self, ireq: "InstallRequirement", field: str, f_val: str, m_val: str
 | ||
|     ) -> None:
 | ||
|         self.ireq = ireq
 | ||
|         self.field = field
 | ||
|         self.f_val = f_val
 | ||
|         self.m_val = m_val
 | ||
| 
 | ||
|     def __str__(self) -> str:
 | ||
|         return (
 | ||
|             f"Requested {self.ireq} has inconsistent {self.field}: "
 | ||
|             f"expected {self.f_val!r}, but metadata has {self.m_val!r}"
 | ||
|         )
 | ||
| 
 | ||
| 
 | ||
| class InstallationSubprocessError(DiagnosticPipError, InstallationError):
 | ||
|     """A subprocess call failed."""
 | ||
| 
 | ||
|     reference = "subprocess-exited-with-error"
 | ||
| 
 | ||
|     def __init__(
 | ||
|         self,
 | ||
|         *,
 | ||
|         command_description: str,
 | ||
|         exit_code: int,
 | ||
|         output_lines: Optional[List[str]],
 | ||
|     ) -> None:
 | ||
|         if output_lines is None:
 | ||
|             output_prompt = Text("See above for output.")
 | ||
|         else:
 | ||
|             output_prompt = (
 | ||
|                 Text.from_markup(f"[red][{len(output_lines)} lines of output][/]\n")
 | ||
|                 + Text("".join(output_lines))
 | ||
|                 + Text.from_markup(R"[red]\[end of output][/]")
 | ||
|             )
 | ||
| 
 | ||
|         super().__init__(
 | ||
|             message=(
 | ||
|                 f"[green]{escape(command_description)}[/] did not run successfully.\n"
 | ||
|                 f"exit code: {exit_code}"
 | ||
|             ),
 | ||
|             context=output_prompt,
 | ||
|             hint_stmt=None,
 | ||
|             note_stmt=(
 | ||
|                 "This error originates from a subprocess, and is likely not a "
 | ||
|                 "problem with pip."
 | ||
|             ),
 | ||
|         )
 | ||
| 
 | ||
|         self.command_description = command_description
 | ||
|         self.exit_code = exit_code
 | ||
| 
 | ||
|     def __str__(self) -> str:
 | ||
|         return f"{self.command_description} exited with {self.exit_code}"
 | ||
| 
 | ||
| 
 | ||
| class MetadataGenerationFailed(InstallationSubprocessError, InstallationError):
 | ||
|     reference = "metadata-generation-failed"
 | ||
| 
 | ||
|     def __init__(
 | ||
|         self,
 | ||
|         *,
 | ||
|         package_details: str,
 | ||
|     ) -> None:
 | ||
|         super(InstallationSubprocessError, self).__init__(
 | ||
|             message="Encountered error while generating package metadata.",
 | ||
|             context=escape(package_details),
 | ||
|             hint_stmt="See above for details.",
 | ||
|             note_stmt="This is an issue with the package mentioned above, not pip.",
 | ||
|         )
 | ||
| 
 | ||
|     def __str__(self) -> str:
 | ||
|         return "metadata generation failed"
 | ||
| 
 | ||
| 
 | ||
| class HashErrors(InstallationError):
 | ||
|     """Multiple HashError instances rolled into one for reporting"""
 | ||
| 
 | ||
|     def __init__(self) -> None:
 | ||
|         self.errors: List["HashError"] = []
 | ||
| 
 | ||
|     def append(self, error: "HashError") -> None:
 | ||
|         self.errors.append(error)
 | ||
| 
 | ||
|     def __str__(self) -> str:
 | ||
|         lines = []
 | ||
|         self.errors.sort(key=lambda e: e.order)
 | ||
|         for cls, errors_of_cls in groupby(self.errors, lambda e: e.__class__):
 | ||
|             lines.append(cls.head)
 | ||
|             lines.extend(e.body() for e in errors_of_cls)
 | ||
|         if lines:
 | ||
|             return "\n".join(lines)
 | ||
|         return ""
 | ||
| 
 | ||
|     def __bool__(self) -> bool:
 | ||
|         return bool(self.errors)
 | ||
| 
 | ||
| 
 | ||
| class HashError(InstallationError):
 | ||
|     """
 | ||
|     A failure to verify a package against known-good hashes
 | ||
| 
 | ||
|     :cvar order: An int sorting hash exception classes by difficulty of
 | ||
|         recovery (lower being harder), so the user doesn't bother fretting
 | ||
|         about unpinned packages when he has deeper issues, like VCS
 | ||
|         dependencies, to deal with. Also keeps error reports in a
 | ||
|         deterministic order.
 | ||
|     :cvar head: A section heading for display above potentially many
 | ||
|         exceptions of this kind
 | ||
|     :ivar req: The InstallRequirement that triggered this error. This is
 | ||
|         pasted on after the exception is instantiated, because it's not
 | ||
|         typically available earlier.
 | ||
| 
 | ||
|     """
 | ||
| 
 | ||
|     req: Optional["InstallRequirement"] = None
 | ||
|     head = ""
 | ||
|     order: int = -1
 | ||
| 
 | ||
|     def body(self) -> str:
 | ||
|         """Return a summary of me for display under the heading.
 | ||
| 
 | ||
|         This default implementation simply prints a description of the
 | ||
|         triggering requirement.
 | ||
| 
 | ||
|         :param req: The InstallRequirement that provoked this error, with
 | ||
|             its link already populated by the resolver's _populate_link().
 | ||
| 
 | ||
|         """
 | ||
|         return f"    {self._requirement_name()}"
 | ||
| 
 | ||
|     def __str__(self) -> str:
 | ||
|         return f"{self.head}\n{self.body()}"
 | ||
| 
 | ||
|     def _requirement_name(self) -> str:
 | ||
|         """Return a description of the requirement that triggered me.
 | ||
| 
 | ||
|         This default implementation returns long description of the req, with
 | ||
|         line numbers
 | ||
| 
 | ||
|         """
 | ||
|         return str(self.req) if self.req else "unknown package"
 | ||
| 
 | ||
| 
 | ||
| class VcsHashUnsupported(HashError):
 | ||
|     """A hash was provided for a version-control-system-based requirement, but
 | ||
|     we don't have a method for hashing those."""
 | ||
| 
 | ||
|     order = 0
 | ||
|     head = (
 | ||
|         "Can't verify hashes for these requirements because we don't "
 | ||
|         "have a way to hash version control repositories:"
 | ||
|     )
 | ||
| 
 | ||
| 
 | ||
| class DirectoryUrlHashUnsupported(HashError):
 | ||
|     """A hash was provided for a version-control-system-based requirement, but
 | ||
|     we don't have a method for hashing those."""
 | ||
| 
 | ||
|     order = 1
 | ||
|     head = (
 | ||
|         "Can't verify hashes for these file:// requirements because they "
 | ||
|         "point to directories:"
 | ||
|     )
 | ||
| 
 | ||
| 
 | ||
| class HashMissing(HashError):
 | ||
|     """A hash was needed for a requirement but is absent."""
 | ||
| 
 | ||
|     order = 2
 | ||
|     head = (
 | ||
|         "Hashes are required in --require-hashes mode, but they are "
 | ||
|         "missing from some requirements. Here is a list of those "
 | ||
|         "requirements along with the hashes their downloaded archives "
 | ||
|         "actually had. Add lines like these to your requirements files to "
 | ||
|         "prevent tampering. (If you did not enable --require-hashes "
 | ||
|         "manually, note that it turns on automatically when any package "
 | ||
|         "has a hash.)"
 | ||
|     )
 | ||
| 
 | ||
|     def __init__(self, gotten_hash: str) -> None:
 | ||
|         """
 | ||
|         :param gotten_hash: The hash of the (possibly malicious) archive we
 | ||
|             just downloaded
 | ||
|         """
 | ||
|         self.gotten_hash = gotten_hash
 | ||
| 
 | ||
|     def body(self) -> str:
 | ||
|         # Dodge circular import.
 | ||
|         from pip._internal.utils.hashes import FAVORITE_HASH
 | ||
| 
 | ||
|         package = None
 | ||
|         if self.req:
 | ||
|             # In the case of URL-based requirements, display the original URL
 | ||
|             # seen in the requirements file rather than the package name,
 | ||
|             # so the output can be directly copied into the requirements file.
 | ||
|             package = (
 | ||
|                 self.req.original_link
 | ||
|                 if self.req.is_direct
 | ||
|                 # In case someone feeds something downright stupid
 | ||
|                 # to InstallRequirement's constructor.
 | ||
|                 else getattr(self.req, "req", None)
 | ||
|             )
 | ||
|         return "    {} --hash={}:{}".format(
 | ||
|             package or "unknown package", FAVORITE_HASH, self.gotten_hash
 | ||
|         )
 | ||
| 
 | ||
| 
 | ||
| class HashUnpinned(HashError):
 | ||
|     """A requirement had a hash specified but was not pinned to a specific
 | ||
|     version."""
 | ||
| 
 | ||
|     order = 3
 | ||
|     head = (
 | ||
|         "In --require-hashes mode, all requirements must have their "
 | ||
|         "versions pinned with ==. These do not:"
 | ||
|     )
 | ||
| 
 | ||
| 
 | ||
| class HashMismatch(HashError):
 | ||
|     """
 | ||
|     Distribution file hash values don't match.
 | ||
| 
 | ||
|     :ivar package_name: The name of the package that triggered the hash
 | ||
|         mismatch. Feel free to write to this after the exception is raise to
 | ||
|         improve its error message.
 | ||
| 
 | ||
|     """
 | ||
| 
 | ||
|     order = 4
 | ||
|     head = (
 | ||
|         "THESE PACKAGES DO NOT MATCH THE HASHES FROM THE REQUIREMENTS "
 | ||
|         "FILE. If you have updated the package versions, please update "
 | ||
|         "the hashes. Otherwise, examine the package contents carefully; "
 | ||
|         "someone may have tampered with them."
 | ||
|     )
 | ||
| 
 | ||
|     def __init__(self, allowed: Dict[str, List[str]], gots: Dict[str, "_Hash"]) -> None:
 | ||
|         """
 | ||
|         :param allowed: A dict of algorithm names pointing to lists of allowed
 | ||
|             hex digests
 | ||
|         :param gots: A dict of algorithm names pointing to hashes we
 | ||
|             actually got from the files under suspicion
 | ||
|         """
 | ||
|         self.allowed = allowed
 | ||
|         self.gots = gots
 | ||
| 
 | ||
|     def body(self) -> str:
 | ||
|         return "    {}:\n{}".format(self._requirement_name(), self._hash_comparison())
 | ||
| 
 | ||
|     def _hash_comparison(self) -> str:
 | ||
|         """
 | ||
|         Return a comparison of actual and expected hash values.
 | ||
| 
 | ||
|         Example::
 | ||
| 
 | ||
|                Expected sha256 abcdeabcdeabcdeabcdeabcdeabcdeabcdeabcdeabcde
 | ||
|                             or 123451234512345123451234512345123451234512345
 | ||
|                     Got        bcdefbcdefbcdefbcdefbcdefbcdefbcdefbcdefbcdef
 | ||
| 
 | ||
|         """
 | ||
| 
 | ||
|         def hash_then_or(hash_name: str) -> "chain[str]":
 | ||
|             # For now, all the decent hashes have 6-char names, so we can get
 | ||
|             # away with hard-coding space literals.
 | ||
|             return chain([hash_name], repeat("    or"))
 | ||
| 
 | ||
|         lines: List[str] = []
 | ||
|         for hash_name, expecteds in self.allowed.items():
 | ||
|             prefix = hash_then_or(hash_name)
 | ||
|             lines.extend(
 | ||
|                 ("        Expected {} {}".format(next(prefix), e)) for e in expecteds
 | ||
|             )
 | ||
|             lines.append(
 | ||
|                 "             Got        {}\n".format(self.gots[hash_name].hexdigest())
 | ||
|             )
 | ||
|         return "\n".join(lines)
 | ||
| 
 | ||
| 
 | ||
| class UnsupportedPythonVersion(InstallationError):
 | ||
|     """Unsupported python version according to Requires-Python package
 | ||
|     metadata."""
 | ||
| 
 | ||
| 
 | ||
| class ConfigurationFileCouldNotBeLoaded(ConfigurationError):
 | ||
|     """When there are errors while loading a configuration file"""
 | ||
| 
 | ||
|     def __init__(
 | ||
|         self,
 | ||
|         reason: str = "could not be loaded",
 | ||
|         fname: Optional[str] = None,
 | ||
|         error: Optional[configparser.Error] = None,
 | ||
|     ) -> None:
 | ||
|         super().__init__(error)
 | ||
|         self.reason = reason
 | ||
|         self.fname = fname
 | ||
|         self.error = error
 | ||
| 
 | ||
|     def __str__(self) -> str:
 | ||
|         if self.fname is not None:
 | ||
|             message_part = f" in {self.fname}."
 | ||
|         else:
 | ||
|             assert self.error is not None
 | ||
|             message_part = f".\n{self.error}\n"
 | ||
|         return f"Configuration file {self.reason}{message_part}"
 | ||
| 
 | ||
| 
 | ||
| _DEFAULT_EXTERNALLY_MANAGED_ERROR = f"""\
 | ||
| The Python environment under {sys.prefix} is managed externally, and may not be
 | ||
| manipulated by the user. Please use specific tooling from the distributor of
 | ||
| the Python installation to interact with this environment instead.
 | ||
| """
 | ||
| 
 | ||
| 
 | ||
| class ExternallyManagedEnvironment(DiagnosticPipError):
 | ||
|     """The current environment is externally managed.
 | ||
| 
 | ||
|     This is raised when the current environment is externally managed, as
 | ||
|     defined by `PEP 668`_. The ``EXTERNALLY-MANAGED`` configuration is checked
 | ||
|     and displayed when the error is bubbled up to the user.
 | ||
| 
 | ||
|     :param error: The error message read from ``EXTERNALLY-MANAGED``.
 | ||
|     """
 | ||
| 
 | ||
|     reference = "externally-managed-environment"
 | ||
| 
 | ||
|     def __init__(self, error: Optional[str]) -> None:
 | ||
|         if error is None:
 | ||
|             context = Text(_DEFAULT_EXTERNALLY_MANAGED_ERROR)
 | ||
|         else:
 | ||
|             context = Text(error)
 | ||
|         super().__init__(
 | ||
|             message="This environment is externally managed",
 | ||
|             context=context,
 | ||
|             note_stmt=(
 | ||
|                 "If you believe this is a mistake, please contact your "
 | ||
|                 "Python installation or OS distribution provider. "
 | ||
|                 "You can override this, at the risk of breaking your Python "
 | ||
|                 "installation or OS, by passing --break-system-packages."
 | ||
|             ),
 | ||
|             hint_stmt=Text("See PEP 668 for the detailed specification."),
 | ||
|         )
 | ||
| 
 | ||
|     @staticmethod
 | ||
|     def _iter_externally_managed_error_keys() -> Iterator[str]:
 | ||
|         # LC_MESSAGES is in POSIX, but not the C standard. The most common
 | ||
|         # platform that does not implement this category is Windows, where
 | ||
|         # using other categories for console message localization is equally
 | ||
|         # unreliable, so we fall back to the locale-less vendor message. This
 | ||
|         # can always be re-evaluated when a vendor proposes a new alternative.
 | ||
|         try:
 | ||
|             category = locale.LC_MESSAGES
 | ||
|         except AttributeError:
 | ||
|             lang: Optional[str] = None
 | ||
|         else:
 | ||
|             lang, _ = locale.getlocale(category)
 | ||
|         if lang is not None:
 | ||
|             yield f"Error-{lang}"
 | ||
|             for sep in ("-", "_"):
 | ||
|                 before, found, _ = lang.partition(sep)
 | ||
|                 if not found:
 | ||
|                     continue
 | ||
|                 yield f"Error-{before}"
 | ||
|         yield "Error"
 | ||
| 
 | ||
|     @classmethod
 | ||
|     def from_config(
 | ||
|         cls,
 | ||
|         config: Union[pathlib.Path, str],
 | ||
|     ) -> "ExternallyManagedEnvironment":
 | ||
|         parser = configparser.ConfigParser(interpolation=None)
 | ||
|         try:
 | ||
|             parser.read(config, encoding="utf-8")
 | ||
|             section = parser["externally-managed"]
 | ||
|             for key in cls._iter_externally_managed_error_keys():
 | ||
|                 with contextlib.suppress(KeyError):
 | ||
|                     return cls(section[key])
 | ||
|         except KeyError:
 | ||
|             pass
 | ||
|         except (OSError, UnicodeDecodeError, configparser.ParsingError):
 | ||
|             from pip._internal.utils._log import VERBOSE
 | ||
| 
 | ||
|             exc_info = logger.isEnabledFor(VERBOSE)
 | ||
|             logger.warning("Failed to read %s", config, exc_info=exc_info)
 | ||
|         return cls(None)
 | 
