Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add initial support for plugins #12985

Draft
wants to merge 5 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions src/pip/_internal/cli/main.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@
from pip._internal.commands import create_command
from pip._internal.exceptions import PipError
from pip._internal.utils import deprecation
from pip._internal.utils.plugins import load_plugins

logger = logging.getLogger(__name__)

Expand Down Expand Up @@ -77,4 +78,6 @@ def main(args: Optional[List[str]] = None) -> int:
logger.debug("Ignoring error %s when setting locale", e)
command = create_command(cmd_name, isolated=("--isolated" in cmd_args))

load_plugins()

return command.main(cmd_args)
83 changes: 83 additions & 0 deletions src/pip/_internal/models/plugin.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,83 @@
import abc
import logging
from pathlib import Path
from types import ModuleType
from typing import List, Optional

logger = logging.getLogger(__name__)

PLUGIN_HOOK_PRE_DOWNLOAD = "pre_download"
PLUGIN_HOOK_PRE_EXTRACT = "pre_extract"
SUPPORTED_PLUGIN_HOOKS = [PLUGIN_HOOK_PRE_DOWNLOAD, PLUGIN_HOOK_PRE_EXTRACT]


class Plugin(metaclass=abc.ABCMeta):
@abc.abstractmethod
def provided_hooks(self) -> List[str]:
raise NotImplementedError

@property
@abc.abstractmethod
def name(self) -> str:
raise NotImplementedError


class LoadedPlugin(Plugin):
def __init__(self, name: str, loaded_module: ModuleType):
self._pre_download = None
self._pre_extract = None
if not hasattr(loaded_module, "provided_hooks"):
raise ValueError(
f"Ignoring plugin {name} due to missing provided_hooks definition"
)
for hook in loaded_module.provided_hooks():
if hook == PLUGIN_HOOK_PRE_DOWNLOAD:
if not hasattr(loaded_module, "pre_download"):
raise ValueError(
f'Plugin "{name}" wants to register a pre-download hook but '
"does not define a pre_download method"
)
self._pre_download = loaded_module.pre_download
elif hook == PLUGIN_HOOK_PRE_EXTRACT:
if not hasattr(loaded_module, "pre_extract"):
raise ValueError(
f'Plugin "{name}" wants to register a pre-extract hook but '
"does not define a pre_extract method"
)
self._pre_extract = loaded_module.pre_extract
else:
raise ValueError(
f'Plugin "{name}" wants to register a hook of unknown type:'
'"{hook}"'
)

self._name = name
self._module = loaded_module

def provided_hooks(self) -> List[str]:
return self._module.provided_hooks()

@property
def name(self) -> str:
return self._name

def pre_download(self, url: str, filename: str, digest: str) -> None:
# contract: `pre_download` raises `ValueError` to terminate
# the operation that intends to download `filename` from `url`
# with hash `digest`
if self._pre_download is not None:
self._pre_download(url=url, filename=filename, digest=digest)

def pre_extract(self, dist: Path) -> None:
# contract: `pre_extract` raises `ValueError` to terminate
# the operation that intends to unarchive `dist`
if self._pre_extract is not None:
self._module.pre_extract(dist)


def plugin_from_module(name: str, loaded_module: ModuleType) -> Optional[LoadedPlugin]:
try:
return LoadedPlugin(name, loaded_module)
except ValueError as e:
logger.warning("Ignoring plugin %s due to error: %s", name, e)
return None
8 changes: 8 additions & 0 deletions src/pip/_internal/operations/install/wheel.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
from base64 import urlsafe_b64encode
from email.message import Message
from itertools import chain, filterfalse, starmap
from pathlib import Path
from typing import (
IO,
TYPE_CHECKING,
Expand Down Expand Up @@ -52,6 +53,7 @@
from pip._internal.models.scheme import SCHEME_KEYS, Scheme
from pip._internal.utils.filesystem import adjacent_tmp_file, replace
from pip._internal.utils.misc import StreamWrapper, ensure_dir, hash_file, partition
from pip._internal.utils.plugins import plugin_pre_extract_hook
from pip._internal.utils.unpacking import (
current_umask,
is_within_directory,
Expand Down Expand Up @@ -727,6 +729,12 @@ def install_wheel(
direct_url: Optional[DirectUrl] = None,
requested: bool = False,
) -> None:
try:
plugin_pre_extract_hook(Path(wheel_path))
except ValueError as e:
raise InstallationError(
f"Could not unpack file {wheel_path} due to plugin:\n{e}"
)
with ZipFile(wheel_path, allowZip64=True) as z:
with req_error_context(req_description):
_install_wheel(
Expand Down
16 changes: 16 additions & 0 deletions src/pip/_internal/operations/prepare.py
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,7 @@
hide_url,
redact_auth_from_requirement,
)
from pip._internal.utils.plugins import plugin_pre_download_hook
from pip._internal.utils.temp_dir import TempDirectory
from pip._internal.utils.unpacking import unpack_file
from pip._internal.vcs import vcs
Expand Down Expand Up @@ -461,6 +462,14 @@ def _complete_partial_requirements(
for req in partially_downloaded_reqs:
assert req.link
links_to_fully_download[req.link] = req
try:
plugin_pre_download_hook(
url=req.link.url, filename=req.link.filename, digest=req.link.hash
)
except ValueError as e:
raise InstallationError(
f"Could not install requirement {req} due to plugin:\n{e}"
)

batch_download = self._batch_download(
links_to_fully_download.keys(),
Expand Down Expand Up @@ -595,6 +604,9 @@ def _prepare_linked_requirement(
local_file = None
elif link.url not in self._downloaded:
try:
plugin_pre_download_hook(
url=req.link.url, filename=req.link.filename, digest=req.link.hash
)
local_file = unpack_url(
link,
req.source_dir,
Expand All @@ -608,6 +620,10 @@ def _prepare_linked_requirement(
f"Could not install requirement {req} because of HTTP "
f"error {exc} for URL {link}"
)
except ValueError as e:
raise InstallationError(
f"Could not install requirement {req} due to plugin:\n{e}"
)
else:
file_path = self._downloaded[link.url]
if hashes:
Expand Down
84 changes: 84 additions & 0 deletions src/pip/_internal/utils/plugins.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,84 @@
import contextlib
import logging
from importlib.metadata import EntryPoint, entry_points
from pathlib import Path
from typing import Iterator, List

from pip._internal.models.plugin import LoadedPlugin, plugin_from_module

logger = logging.getLogger(__name__)
_loaded_plugins: List[LoadedPlugin] = []


def iter_entry_points(group_name: str) -> List[EntryPoint]:
# Only Python >= 3.10 supports the `EntryPoints` class, so we return
# a list of `EntryPoint` instead.
groups = entry_points()
if hasattr(groups, "select"):
# New interface in Python 3.10 and newer versions of the
# importlib_metadata backport.
return list(groups.select(group=group_name))
else:
assert hasattr(groups, "get")
# Older interface, deprecated in Python 3.10 and recent
# importlib_metadata, but we need it in Python 3.8 and 3.9.
return groups.get(group_name, [])


def load_plugins() -> None:
for entrypoint in iter_entry_points(group_name="pip.plugins"):
try:
module = entrypoint.load()
except ModuleNotFoundError:
logger.warning("Tried to load plugin %s but failed", entrypoint.name)
continue
plugin = plugin_from_module(entrypoint.name, module)
if plugin is not None:
_loaded_plugins.append(plugin)


@contextlib.contextmanager
def _only_raise_value_error(plugin_name: str) -> Iterator[None]:
try:
yield
except ValueError as e:
raise ValueError(f"Plugin {plugin_name}: {e}") from e
except Exception as e:
logger.warning(
"Plugin %s raised an unexpected exception type: %s",
plugin_name,
{e.__class__.__name__},
)
raise ValueError(f"Plugin {plugin_name}: {e}") from e


def plugin_pre_download_hook(url: str, filename: str, digest: str) -> None:
"""Call the pre-download hook of all loaded plugins

This function should be called right before a distribution is downloaded.
It will go through all the loaded plugins and call their `pre_download(url)`
function.
Only ValueError will be raised. If the plugin (incorrectly) raises another
exception type, this function will wrap it as a ValueError and log
a warning.
"""

for p in _loaded_plugins:
with _only_raise_value_error(p.name):
p.pre_download(url=url, filename=filename, digest=digest)


def plugin_pre_extract_hook(dist: Path) -> None:
"""Call the pre-extract hook of all loaded plugins

This function should be called right before a distribution is extracted.
It will go through all the loaded plugins and call their `pre_extract(dist)`
function.
Only ValueError will be raised. If the plugin (incorrectly) raises another
exception type, this function will wrap it as a ValueError and log
a warning.
"""

for p in _loaded_plugins:
with _only_raise_value_error(p.name):
p.pre_extract(dist)
7 changes: 7 additions & 0 deletions src/pip/_internal/utils/unpacking.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
import sys
import tarfile
import zipfile
from pathlib import Path
from typing import Iterable, List, Optional
from zipfile import ZipInfo

Expand All @@ -19,6 +20,7 @@
ZIP_EXTENSIONS,
)
from pip._internal.utils.misc import ensure_dir
from pip._internal.utils.plugins import plugin_pre_extract_hook

logger = logging.getLogger(__name__)

Expand Down Expand Up @@ -312,6 +314,11 @@ def unpack_file(
content_type: Optional[str] = None,
) -> None:
filename = os.path.realpath(filename)
try:
plugin_pre_extract_hook(Path(filename))
except ValueError as e:
raise InstallationError(f"Could not unpack file {filename} due to plugin:\n{e}")

if (
content_type == "application/zip"
or filename.lower().endswith(ZIP_EXTENSIONS)
Expand Down
Loading