| import logging |
| from typing import TYPE_CHECKING, Iterable, Optional, Set, Tuple |
|
|
| from pip._internal.build_env import BuildEnvironment |
| from pip._internal.distributions.base import AbstractDistribution |
| from pip._internal.exceptions import InstallationError |
| from pip._internal.metadata import BaseDistribution |
| from pip._internal.utils.subprocess import runner_with_spinner_message |
|
|
| if TYPE_CHECKING: |
| from pip._internal.index.package_finder import PackageFinder |
|
|
| logger = logging.getLogger(__name__) |
|
|
|
|
| class SourceDistribution(AbstractDistribution): |
| """Represents a source distribution. |
| |
| The preparation step for these needs metadata for the packages to be |
| generated, either using PEP 517 or using the legacy `setup.py egg_info`. |
| """ |
|
|
| @property |
| def build_tracker_id(self) -> Optional[str]: |
| """Identify this requirement uniquely by its link.""" |
| assert self.req.link |
| return self.req.link.url_without_fragment |
|
|
| def get_metadata_distribution(self) -> BaseDistribution: |
| return self.req.get_dist() |
|
|
| def prepare_distribution_metadata( |
| self, |
| finder: "PackageFinder", |
| build_isolation: bool, |
| check_build_deps: bool, |
| ) -> None: |
| |
| self.req.load_pyproject_toml() |
|
|
| |
| should_isolate = self.req.use_pep517 and build_isolation |
| if should_isolate: |
| |
| |
| self._prepare_build_backend(finder) |
| |
| |
| |
| |
| |
| |
| |
| self.req.isolated_editable_sanity_check() |
| |
| self._install_build_reqs(finder) |
| |
| should_check_deps = self.req.use_pep517 and check_build_deps |
| if should_check_deps: |
| pyproject_requires = self.req.pyproject_requires |
| assert pyproject_requires is not None |
| conflicting, missing = self.req.build_env.check_requirements( |
| pyproject_requires |
| ) |
| if conflicting: |
| self._raise_conflicts("the backend dependencies", conflicting) |
| if missing: |
| self._raise_missing_reqs(missing) |
| self.req.prepare_metadata() |
|
|
| def _prepare_build_backend(self, finder: "PackageFinder") -> None: |
| |
| |
| pyproject_requires = self.req.pyproject_requires |
| assert pyproject_requires is not None |
|
|
| self.req.build_env = BuildEnvironment() |
| self.req.build_env.install_requirements( |
| finder, pyproject_requires, "overlay", kind="build dependencies" |
| ) |
| conflicting, missing = self.req.build_env.check_requirements( |
| self.req.requirements_to_check |
| ) |
| if conflicting: |
| self._raise_conflicts("PEP 517/518 supported requirements", conflicting) |
| if missing: |
| logger.warning( |
| "Missing build requirements in pyproject.toml for %s.", |
| self.req, |
| ) |
| logger.warning( |
| "The project does not specify a build backend, and " |
| "pip cannot fall back to setuptools without %s.", |
| " and ".join(map(repr, sorted(missing))), |
| ) |
|
|
| def _get_build_requires_wheel(self) -> Iterable[str]: |
| with self.req.build_env: |
| runner = runner_with_spinner_message("Getting requirements to build wheel") |
| backend = self.req.pep517_backend |
| assert backend is not None |
| with backend.subprocess_runner(runner): |
| return backend.get_requires_for_build_wheel() |
|
|
| def _get_build_requires_editable(self) -> Iterable[str]: |
| with self.req.build_env: |
| runner = runner_with_spinner_message( |
| "Getting requirements to build editable" |
| ) |
| backend = self.req.pep517_backend |
| assert backend is not None |
| with backend.subprocess_runner(runner): |
| return backend.get_requires_for_build_editable() |
|
|
| def _install_build_reqs(self, finder: "PackageFinder") -> None: |
| |
| |
| |
| if ( |
| self.req.editable |
| and self.req.permit_editable_wheels |
| and self.req.supports_pyproject_editable |
| ): |
| build_reqs = self._get_build_requires_editable() |
| else: |
| build_reqs = self._get_build_requires_wheel() |
| conflicting, missing = self.req.build_env.check_requirements(build_reqs) |
| if conflicting: |
| self._raise_conflicts("the backend dependencies", conflicting) |
| self.req.build_env.install_requirements( |
| finder, missing, "normal", kind="backend dependencies" |
| ) |
|
|
| def _raise_conflicts( |
| self, conflicting_with: str, conflicting_reqs: Set[Tuple[str, str]] |
| ) -> None: |
| format_string = ( |
| "Some build dependencies for {requirement} " |
| "conflict with {conflicting_with}: {description}." |
| ) |
| error_message = format_string.format( |
| requirement=self.req, |
| conflicting_with=conflicting_with, |
| description=", ".join( |
| f"{installed} is incompatible with {wanted}" |
| for installed, wanted in sorted(conflicting_reqs) |
| ), |
| ) |
| raise InstallationError(error_message) |
|
|
| def _raise_missing_reqs(self, missing: Set[str]) -> None: |
| format_string = ( |
| "Some build dependencies for {requirement} are missing: {missing}." |
| ) |
| error_message = format_string.format( |
| requirement=self.req, missing=", ".join(map(repr, sorted(missing))) |
| ) |
| raise InstallationError(error_message) |
|
|