Spaces:
Runtime error
Runtime error
""" | |
shared options and groups | |
The principle here is to define options once, but *not* instantiate them | |
globally. One reason being that options with action='append' can carry state | |
between parses. pip parses general options twice internally, and shouldn't | |
pass on state. To be consistent, all options will follow this design. | |
""" | |
# The following comment should be removed at some point in the future. | |
# mypy: strict-optional=False | |
import importlib.util | |
import logging | |
import os | |
import textwrap | |
from functools import partial | |
from optparse import SUPPRESS_HELP, Option, OptionGroup, OptionParser, Values | |
from textwrap import dedent | |
from typing import Any, Callable, Dict, Optional, Tuple | |
from pip._vendor.packaging.utils import canonicalize_name | |
from pip._internal.cli.parser import ConfigOptionParser | |
from pip._internal.exceptions import CommandError | |
from pip._internal.locations import USER_CACHE_DIR, get_src_prefix | |
from pip._internal.models.format_control import FormatControl | |
from pip._internal.models.index import PyPI | |
from pip._internal.models.target_python import TargetPython | |
from pip._internal.utils.hashes import STRONG_HASHES | |
from pip._internal.utils.misc import strtobool | |
logger = logging.getLogger(__name__) | |
def raise_option_error(parser: OptionParser, option: Option, msg: str) -> None: | |
""" | |
Raise an option parsing error using parser.error(). | |
Args: | |
parser: an OptionParser instance. | |
option: an Option instance. | |
msg: the error text. | |
""" | |
msg = f"{option} error: {msg}" | |
msg = textwrap.fill(" ".join(msg.split())) | |
parser.error(msg) | |
def make_option_group(group: Dict[str, Any], parser: ConfigOptionParser) -> OptionGroup: | |
""" | |
Return an OptionGroup object | |
group -- assumed to be dict with 'name' and 'options' keys | |
parser -- an optparse Parser | |
""" | |
option_group = OptionGroup(parser, group["name"]) | |
for option in group["options"]: | |
option_group.add_option(option()) | |
return option_group | |
def check_dist_restriction(options: Values, check_target: bool = False) -> None: | |
"""Function for determining if custom platform options are allowed. | |
:param options: The OptionParser options. | |
:param check_target: Whether or not to check if --target is being used. | |
""" | |
dist_restriction_set = any( | |
[ | |
options.python_version, | |
options.platforms, | |
options.abis, | |
options.implementation, | |
] | |
) | |
binary_only = FormatControl(set(), {":all:"}) | |
sdist_dependencies_allowed = ( | |
options.format_control != binary_only and not options.ignore_dependencies | |
) | |
# Installations or downloads using dist restrictions must not combine | |
# source distributions and dist-specific wheels, as they are not | |
# guaranteed to be locally compatible. | |
if dist_restriction_set and sdist_dependencies_allowed: | |
raise CommandError( | |
"When restricting platform and interpreter constraints using " | |
"--python-version, --platform, --abi, or --implementation, " | |
"either --no-deps must be set, or --only-binary=:all: must be " | |
"set and --no-binary must not be set (or must be set to " | |
":none:)." | |
) | |
if check_target: | |
if dist_restriction_set and not options.target_dir: | |
raise CommandError( | |
"Can not use any platform or abi specific options unless " | |
"installing via '--target'" | |
) | |
def _path_option_check(option: Option, opt: str, value: str) -> str: | |
return os.path.expanduser(value) | |
def _package_name_option_check(option: Option, opt: str, value: str) -> str: | |
return canonicalize_name(value) | |
class PipOption(Option): | |
TYPES = Option.TYPES + ("path", "package_name") | |
TYPE_CHECKER = Option.TYPE_CHECKER.copy() | |
TYPE_CHECKER["package_name"] = _package_name_option_check | |
TYPE_CHECKER["path"] = _path_option_check | |
########### | |
# options # | |
########### | |
help_: Callable[..., Option] = partial( | |
Option, | |
"-h", | |
"--help", | |
dest="help", | |
action="help", | |
help="Show help.", | |
) | |
debug_mode: Callable[..., Option] = partial( | |
Option, | |
"--debug", | |
dest="debug_mode", | |
action="store_true", | |
default=False, | |
help=( | |
"Let unhandled exceptions propagate outside the main subroutine, " | |
"instead of logging them to stderr." | |
), | |
) | |
isolated_mode: Callable[..., Option] = partial( | |
Option, | |
"--isolated", | |
dest="isolated_mode", | |
action="store_true", | |
default=False, | |
help=( | |
"Run pip in an isolated mode, ignoring environment variables and user " | |
"configuration." | |
), | |
) | |
require_virtualenv: Callable[..., Option] = partial( | |
Option, | |
"--require-virtualenv", | |
"--require-venv", | |
dest="require_venv", | |
action="store_true", | |
default=False, | |
help=( | |
"Allow pip to only run in a virtual environment; " | |
"exit with an error otherwise." | |
), | |
) | |
override_externally_managed: Callable[..., Option] = partial( | |
Option, | |
"--break-system-packages", | |
dest="override_externally_managed", | |
action="store_true", | |
help="Allow pip to modify an EXTERNALLY-MANAGED Python installation", | |
) | |
python: Callable[..., Option] = partial( | |
Option, | |
"--python", | |
dest="python", | |
help="Run pip with the specified Python interpreter.", | |
) | |
verbose: Callable[..., Option] = partial( | |
Option, | |
"-v", | |
"--verbose", | |
dest="verbose", | |
action="count", | |
default=0, | |
help="Give more output. Option is additive, and can be used up to 3 times.", | |
) | |
no_color: Callable[..., Option] = partial( | |
Option, | |
"--no-color", | |
dest="no_color", | |
action="store_true", | |
default=False, | |
help="Suppress colored output.", | |
) | |
version: Callable[..., Option] = partial( | |
Option, | |
"-V", | |
"--version", | |
dest="version", | |
action="store_true", | |
help="Show version and exit.", | |
) | |
quiet: Callable[..., Option] = partial( | |
Option, | |
"-q", | |
"--quiet", | |
dest="quiet", | |
action="count", | |
default=0, | |
help=( | |
"Give less output. Option is additive, and can be used up to 3" | |
" times (corresponding to WARNING, ERROR, and CRITICAL logging" | |
" levels)." | |
), | |
) | |
progress_bar: Callable[..., Option] = partial( | |
Option, | |
"--progress-bar", | |
dest="progress_bar", | |
type="choice", | |
choices=["on", "off"], | |
default="on", | |
help="Specify whether the progress bar should be used [on, off] (default: on)", | |
) | |
log: Callable[..., Option] = partial( | |
PipOption, | |
"--log", | |
"--log-file", | |
"--local-log", | |
dest="log", | |
metavar="path", | |
type="path", | |
help="Path to a verbose appending log.", | |
) | |
no_input: Callable[..., Option] = partial( | |
Option, | |
# Don't ask for input | |
"--no-input", | |
dest="no_input", | |
action="store_true", | |
default=False, | |
help="Disable prompting for input.", | |
) | |
keyring_provider: Callable[..., Option] = partial( | |
Option, | |
"--keyring-provider", | |
dest="keyring_provider", | |
choices=["auto", "disabled", "import", "subprocess"], | |
default="auto", | |
help=( | |
"Enable the credential lookup via the keyring library if user input is allowed." | |
" Specify which mechanism to use [disabled, import, subprocess]." | |
" (default: disabled)" | |
), | |
) | |
proxy: Callable[..., Option] = partial( | |
Option, | |
"--proxy", | |
dest="proxy", | |
type="str", | |
default="", | |
help="Specify a proxy in the form scheme://[user:passwd@]proxy.server:port.", | |
) | |
retries: Callable[..., Option] = partial( | |
Option, | |
"--retries", | |
dest="retries", | |
type="int", | |
default=5, | |
help="Maximum number of retries each connection should attempt " | |
"(default %default times).", | |
) | |
timeout: Callable[..., Option] = partial( | |
Option, | |
"--timeout", | |
"--default-timeout", | |
metavar="sec", | |
dest="timeout", | |
type="float", | |
default=15, | |
help="Set the socket timeout (default %default seconds).", | |
) | |
def exists_action() -> Option: | |
return Option( | |
# Option when path already exist | |
"--exists-action", | |
dest="exists_action", | |
type="choice", | |
choices=["s", "i", "w", "b", "a"], | |
default=[], | |
action="append", | |
metavar="action", | |
help="Default action when a path already exists: " | |
"(s)witch, (i)gnore, (w)ipe, (b)ackup, (a)bort.", | |
) | |
cert: Callable[..., Option] = partial( | |
PipOption, | |
"--cert", | |
dest="cert", | |
type="path", | |
metavar="path", | |
help=( | |
"Path to PEM-encoded CA certificate bundle. " | |
"If provided, overrides the default. " | |
"See 'SSL Certificate Verification' in pip documentation " | |
"for more information." | |
), | |
) | |
client_cert: Callable[..., Option] = partial( | |
PipOption, | |
"--client-cert", | |
dest="client_cert", | |
type="path", | |
default=None, | |
metavar="path", | |
help="Path to SSL client certificate, a single file containing the " | |
"private key and the certificate in PEM format.", | |
) | |
index_url: Callable[..., Option] = partial( | |
Option, | |
"-i", | |
"--index-url", | |
"--pypi-url", | |
dest="index_url", | |
metavar="URL", | |
default=PyPI.simple_url, | |
help="Base URL of the Python Package Index (default %default). " | |
"This should point to a repository compliant with PEP 503 " | |
"(the simple repository API) or a local directory laid out " | |
"in the same format.", | |
) | |
def extra_index_url() -> Option: | |
return Option( | |
"--extra-index-url", | |
dest="extra_index_urls", | |
metavar="URL", | |
action="append", | |
default=[], | |
help="Extra URLs of package indexes to use in addition to " | |
"--index-url. Should follow the same rules as " | |
"--index-url.", | |
) | |
no_index: Callable[..., Option] = partial( | |
Option, | |
"--no-index", | |
dest="no_index", | |
action="store_true", | |
default=False, | |
help="Ignore package index (only looking at --find-links URLs instead).", | |
) | |
def find_links() -> Option: | |
return Option( | |
"-f", | |
"--find-links", | |
dest="find_links", | |
action="append", | |
default=[], | |
metavar="url", | |
help="If a URL or path to an html file, then parse for links to " | |
"archives such as sdist (.tar.gz) or wheel (.whl) files. " | |
"If a local path or file:// URL that's a directory, " | |
"then look for archives in the directory listing. " | |
"Links to VCS project URLs are not supported.", | |
) | |
def trusted_host() -> Option: | |
return Option( | |
"--trusted-host", | |
dest="trusted_hosts", | |
action="append", | |
metavar="HOSTNAME", | |
default=[], | |
help="Mark this host or host:port pair as trusted, even though it " | |
"does not have valid or any HTTPS.", | |
) | |
def constraints() -> Option: | |
return Option( | |
"-c", | |
"--constraint", | |
dest="constraints", | |
action="append", | |
default=[], | |
metavar="file", | |
help="Constrain versions using the given constraints file. " | |
"This option can be used multiple times.", | |
) | |
def requirements() -> Option: | |
return Option( | |
"-r", | |
"--requirement", | |
dest="requirements", | |
action="append", | |
default=[], | |
metavar="file", | |
help="Install from the given requirements file. " | |
"This option can be used multiple times.", | |
) | |
def editable() -> Option: | |
return Option( | |
"-e", | |
"--editable", | |
dest="editables", | |
action="append", | |
default=[], | |
metavar="path/url", | |
help=( | |
"Install a project in editable mode (i.e. setuptools " | |
'"develop mode") from a local project path or a VCS url.' | |
), | |
) | |
def _handle_src(option: Option, opt_str: str, value: str, parser: OptionParser) -> None: | |
value = os.path.abspath(value) | |
setattr(parser.values, option.dest, value) | |
src: Callable[..., Option] = partial( | |
PipOption, | |
"--src", | |
"--source", | |
"--source-dir", | |
"--source-directory", | |
dest="src_dir", | |
type="path", | |
metavar="dir", | |
default=get_src_prefix(), | |
action="callback", | |
callback=_handle_src, | |
help="Directory to check out editable projects into. " | |
'The default in a virtualenv is "<venv path>/src". ' | |
'The default for global installs is "<current dir>/src".', | |
) | |
def _get_format_control(values: Values, option: Option) -> Any: | |
"""Get a format_control object.""" | |
return getattr(values, option.dest) | |
def _handle_no_binary( | |
option: Option, opt_str: str, value: str, parser: OptionParser | |
) -> None: | |
existing = _get_format_control(parser.values, option) | |
FormatControl.handle_mutual_excludes( | |
value, | |
existing.no_binary, | |
existing.only_binary, | |
) | |
def _handle_only_binary( | |
option: Option, opt_str: str, value: str, parser: OptionParser | |
) -> None: | |
existing = _get_format_control(parser.values, option) | |
FormatControl.handle_mutual_excludes( | |
value, | |
existing.only_binary, | |
existing.no_binary, | |
) | |
def no_binary() -> Option: | |
format_control = FormatControl(set(), set()) | |
return Option( | |
"--no-binary", | |
dest="format_control", | |
action="callback", | |
callback=_handle_no_binary, | |
type="str", | |
default=format_control, | |
help="Do not use binary packages. Can be supplied multiple times, and " | |
'each time adds to the existing value. Accepts either ":all:" to ' | |
'disable all binary packages, ":none:" to empty the set (notice ' | |
"the colons), or one or more package names with commas between " | |
"them (no colons). Note that some packages are tricky to compile " | |
"and may fail to install when this option is used on them.", | |
) | |
def only_binary() -> Option: | |
format_control = FormatControl(set(), set()) | |
return Option( | |
"--only-binary", | |
dest="format_control", | |
action="callback", | |
callback=_handle_only_binary, | |
type="str", | |
default=format_control, | |
help="Do not use source packages. Can be supplied multiple times, and " | |
'each time adds to the existing value. Accepts either ":all:" to ' | |
'disable all source packages, ":none:" to empty the set, or one ' | |
"or more package names with commas between them. Packages " | |
"without binary distributions will fail to install when this " | |
"option is used on them.", | |
) | |
platforms: Callable[..., Option] = partial( | |
Option, | |
"--platform", | |
dest="platforms", | |
metavar="platform", | |
action="append", | |
default=None, | |
help=( | |
"Only use wheels compatible with <platform>. Defaults to the " | |
"platform of the running system. Use this option multiple times to " | |
"specify multiple platforms supported by the target interpreter." | |
), | |
) | |
# This was made a separate function for unit-testing purposes. | |
def _convert_python_version(value: str) -> Tuple[Tuple[int, ...], Optional[str]]: | |
""" | |
Convert a version string like "3", "37", or "3.7.3" into a tuple of ints. | |
:return: A 2-tuple (version_info, error_msg), where `error_msg` is | |
non-None if and only if there was a parsing error. | |
""" | |
if not value: | |
# The empty string is the same as not providing a value. | |
return (None, None) | |
parts = value.split(".") | |
if len(parts) > 3: | |
return ((), "at most three version parts are allowed") | |
if len(parts) == 1: | |
# Then we are in the case of "3" or "37". | |
value = parts[0] | |
if len(value) > 1: | |
parts = [value[0], value[1:]] | |
try: | |
version_info = tuple(int(part) for part in parts) | |
except ValueError: | |
return ((), "each version part must be an integer") | |
return (version_info, None) | |
def _handle_python_version( | |
option: Option, opt_str: str, value: str, parser: OptionParser | |
) -> None: | |
""" | |
Handle a provided --python-version value. | |
""" | |
version_info, error_msg = _convert_python_version(value) | |
if error_msg is not None: | |
msg = "invalid --python-version value: {!r}: {}".format( | |
value, | |
error_msg, | |
) | |
raise_option_error(parser, option=option, msg=msg) | |
parser.values.python_version = version_info | |
python_version: Callable[..., Option] = partial( | |
Option, | |
"--python-version", | |
dest="python_version", | |
metavar="python_version", | |
action="callback", | |
callback=_handle_python_version, | |
type="str", | |
default=None, | |
help=dedent( | |
"""\ | |
The Python interpreter version to use for wheel and "Requires-Python" | |
compatibility checks. Defaults to a version derived from the running | |
interpreter. The version can be specified using up to three dot-separated | |
integers (e.g. "3" for 3.0.0, "3.7" for 3.7.0, or "3.7.3"). A major-minor | |
version can also be given as a string without dots (e.g. "37" for 3.7.0). | |
""" | |
), | |
) | |
implementation: Callable[..., Option] = partial( | |
Option, | |
"--implementation", | |
dest="implementation", | |
metavar="implementation", | |
default=None, | |
help=( | |
"Only use wheels compatible with Python " | |
"implementation <implementation>, e.g. 'pp', 'jy', 'cp', " | |
" or 'ip'. If not specified, then the current " | |
"interpreter implementation is used. Use 'py' to force " | |
"implementation-agnostic wheels." | |
), | |
) | |
abis: Callable[..., Option] = partial( | |
Option, | |
"--abi", | |
dest="abis", | |
metavar="abi", | |
action="append", | |
default=None, | |
help=( | |
"Only use wheels compatible with Python abi <abi>, e.g. 'pypy_41'. " | |
"If not specified, then the current interpreter abi tag is used. " | |
"Use this option multiple times to specify multiple abis supported " | |
"by the target interpreter. Generally you will need to specify " | |
"--implementation, --platform, and --python-version when using this " | |
"option." | |
), | |
) | |
def add_target_python_options(cmd_opts: OptionGroup) -> None: | |
cmd_opts.add_option(platforms()) | |
cmd_opts.add_option(python_version()) | |
cmd_opts.add_option(implementation()) | |
cmd_opts.add_option(abis()) | |
def make_target_python(options: Values) -> TargetPython: | |
target_python = TargetPython( | |
platforms=options.platforms, | |
py_version_info=options.python_version, | |
abis=options.abis, | |
implementation=options.implementation, | |
) | |
return target_python | |
def prefer_binary() -> Option: | |
return Option( | |
"--prefer-binary", | |
dest="prefer_binary", | |
action="store_true", | |
default=False, | |
help="Prefer older binary packages over newer source packages.", | |
) | |
cache_dir: Callable[..., Option] = partial( | |
PipOption, | |
"--cache-dir", | |
dest="cache_dir", | |
default=USER_CACHE_DIR, | |
metavar="dir", | |
type="path", | |
help="Store the cache data in <dir>.", | |
) | |
def _handle_no_cache_dir( | |
option: Option, opt: str, value: str, parser: OptionParser | |
) -> None: | |
""" | |
Process a value provided for the --no-cache-dir option. | |
This is an optparse.Option callback for the --no-cache-dir option. | |
""" | |
# The value argument will be None if --no-cache-dir is passed via the | |
# command-line, since the option doesn't accept arguments. However, | |
# the value can be non-None if the option is triggered e.g. by an | |
# environment variable, like PIP_NO_CACHE_DIR=true. | |
if value is not None: | |
# Then parse the string value to get argument error-checking. | |
try: | |
strtobool(value) | |
except ValueError as exc: | |
raise_option_error(parser, option=option, msg=str(exc)) | |
# Originally, setting PIP_NO_CACHE_DIR to a value that strtobool() | |
# converted to 0 (like "false" or "no") caused cache_dir to be disabled | |
# rather than enabled (logic would say the latter). Thus, we disable | |
# the cache directory not just on values that parse to True, but (for | |
# backwards compatibility reasons) also on values that parse to False. | |
# In other words, always set it to False if the option is provided in | |
# some (valid) form. | |
parser.values.cache_dir = False | |
no_cache: Callable[..., Option] = partial( | |
Option, | |
"--no-cache-dir", | |
dest="cache_dir", | |
action="callback", | |
callback=_handle_no_cache_dir, | |
help="Disable the cache.", | |
) | |
no_deps: Callable[..., Option] = partial( | |
Option, | |
"--no-deps", | |
"--no-dependencies", | |
dest="ignore_dependencies", | |
action="store_true", | |
default=False, | |
help="Don't install package dependencies.", | |
) | |
ignore_requires_python: Callable[..., Option] = partial( | |
Option, | |
"--ignore-requires-python", | |
dest="ignore_requires_python", | |
action="store_true", | |
help="Ignore the Requires-Python information.", | |
) | |
no_build_isolation: Callable[..., Option] = partial( | |
Option, | |
"--no-build-isolation", | |
dest="build_isolation", | |
action="store_false", | |
default=True, | |
help="Disable isolation when building a modern source distribution. " | |
"Build dependencies specified by PEP 518 must be already installed " | |
"if this option is used.", | |
) | |
check_build_deps: Callable[..., Option] = partial( | |
Option, | |
"--check-build-dependencies", | |
dest="check_build_deps", | |
action="store_true", | |
default=False, | |
help="Check the build dependencies when PEP517 is used.", | |
) | |
def _handle_no_use_pep517( | |
option: Option, opt: str, value: str, parser: OptionParser | |
) -> None: | |
""" | |
Process a value provided for the --no-use-pep517 option. | |
This is an optparse.Option callback for the no_use_pep517 option. | |
""" | |
# Since --no-use-pep517 doesn't accept arguments, the value argument | |
# will be None if --no-use-pep517 is passed via the command-line. | |
# However, the value can be non-None if the option is triggered e.g. | |
# by an environment variable, for example "PIP_NO_USE_PEP517=true". | |
if value is not None: | |
msg = """A value was passed for --no-use-pep517, | |
probably using either the PIP_NO_USE_PEP517 environment variable | |
or the "no-use-pep517" config file option. Use an appropriate value | |
of the PIP_USE_PEP517 environment variable or the "use-pep517" | |
config file option instead. | |
""" | |
raise_option_error(parser, option=option, msg=msg) | |
# If user doesn't wish to use pep517, we check if setuptools and wheel are installed | |
# and raise error if it is not. | |
packages = ("setuptools", "wheel") | |
if not all(importlib.util.find_spec(package) for package in packages): | |
msg = ( | |
f"It is not possible to use --no-use-pep517 " | |
f"without {' and '.join(packages)} installed." | |
) | |
raise_option_error(parser, option=option, msg=msg) | |
# Otherwise, --no-use-pep517 was passed via the command-line. | |
parser.values.use_pep517 = False | |
use_pep517: Any = partial( | |
Option, | |
"--use-pep517", | |
dest="use_pep517", | |
action="store_true", | |
default=None, | |
help="Use PEP 517 for building source distributions " | |
"(use --no-use-pep517 to force legacy behaviour).", | |
) | |
no_use_pep517: Any = partial( | |
Option, | |
"--no-use-pep517", | |
dest="use_pep517", | |
action="callback", | |
callback=_handle_no_use_pep517, | |
default=None, | |
help=SUPPRESS_HELP, | |
) | |
def _handle_config_settings( | |
option: Option, opt_str: str, value: str, parser: OptionParser | |
) -> None: | |
key, sep, val = value.partition("=") | |
if sep != "=": | |
parser.error(f"Arguments to {opt_str} must be of the form KEY=VAL") # noqa | |
dest = getattr(parser.values, option.dest) | |
if dest is None: | |
dest = {} | |
setattr(parser.values, option.dest, dest) | |
if key in dest: | |
if isinstance(dest[key], list): | |
dest[key].append(val) | |
else: | |
dest[key] = [dest[key], val] | |
else: | |
dest[key] = val | |
config_settings: Callable[..., Option] = partial( | |
Option, | |
"-C", | |
"--config-settings", | |
dest="config_settings", | |
type=str, | |
action="callback", | |
callback=_handle_config_settings, | |
metavar="settings", | |
help="Configuration settings to be passed to the PEP 517 build backend. " | |
"Settings take the form KEY=VALUE. Use multiple --config-settings options " | |
"to pass multiple keys to the backend.", | |
) | |
build_options: Callable[..., Option] = partial( | |
Option, | |
"--build-option", | |
dest="build_options", | |
metavar="options", | |
action="append", | |
help="Extra arguments to be supplied to 'setup.py bdist_wheel'.", | |
) | |
global_options: Callable[..., Option] = partial( | |
Option, | |
"--global-option", | |
dest="global_options", | |
action="append", | |
metavar="options", | |
help="Extra global options to be supplied to the setup.py " | |
"call before the install or bdist_wheel command.", | |
) | |
no_clean: Callable[..., Option] = partial( | |
Option, | |
"--no-clean", | |
action="store_true", | |
default=False, | |
help="Don't clean up build directories.", | |
) | |
pre: Callable[..., Option] = partial( | |
Option, | |
"--pre", | |
action="store_true", | |
default=False, | |
help="Include pre-release and development versions. By default, " | |
"pip only finds stable versions.", | |
) | |
disable_pip_version_check: Callable[..., Option] = partial( | |
Option, | |
"--disable-pip-version-check", | |
dest="disable_pip_version_check", | |
action="store_true", | |
default=False, | |
help="Don't periodically check PyPI to determine whether a new version " | |
"of pip is available for download. Implied with --no-index.", | |
) | |
root_user_action: Callable[..., Option] = partial( | |
Option, | |
"--root-user-action", | |
dest="root_user_action", | |
default="warn", | |
choices=["warn", "ignore"], | |
help="Action if pip is run as a root user. By default, a warning message is shown.", | |
) | |
def _handle_merge_hash( | |
option: Option, opt_str: str, value: str, parser: OptionParser | |
) -> None: | |
"""Given a value spelled "algo:digest", append the digest to a list | |
pointed to in a dict by the algo name.""" | |
if not parser.values.hashes: | |
parser.values.hashes = {} | |
try: | |
algo, digest = value.split(":", 1) | |
except ValueError: | |
parser.error( | |
"Arguments to {} must be a hash name " # noqa | |
"followed by a value, like --hash=sha256:" | |
"abcde...".format(opt_str) | |
) | |
if algo not in STRONG_HASHES: | |
parser.error( | |
"Allowed hash algorithms for {} are {}.".format( # noqa | |
opt_str, ", ".join(STRONG_HASHES) | |
) | |
) | |
parser.values.hashes.setdefault(algo, []).append(digest) | |
hash: Callable[..., Option] = partial( | |
Option, | |
"--hash", | |
# Hash values eventually end up in InstallRequirement.hashes due to | |
# __dict__ copying in process_line(). | |
dest="hashes", | |
action="callback", | |
callback=_handle_merge_hash, | |
type="string", | |
help="Verify that the package's archive matches this " | |
"hash before installing. Example: --hash=sha256:abcdef...", | |
) | |
require_hashes: Callable[..., Option] = partial( | |
Option, | |
"--require-hashes", | |
dest="require_hashes", | |
action="store_true", | |
default=False, | |
help="Require a hash to check each requirement against, for " | |
"repeatable installs. This option is implied when any package in a " | |
"requirements file has a --hash option.", | |
) | |
list_path: Callable[..., Option] = partial( | |
PipOption, | |
"--path", | |
dest="path", | |
type="path", | |
action="append", | |
help="Restrict to the specified installation path for listing " | |
"packages (can be used multiple times).", | |
) | |
def check_list_path_option(options: Values) -> None: | |
if options.path and (options.user or options.local): | |
raise CommandError("Cannot combine '--path' with '--user' or '--local'") | |
list_exclude: Callable[..., Option] = partial( | |
PipOption, | |
"--exclude", | |
dest="excludes", | |
action="append", | |
metavar="package", | |
type="package_name", | |
help="Exclude specified package from the output", | |
) | |
no_python_version_warning: Callable[..., Option] = partial( | |
Option, | |
"--no-python-version-warning", | |
dest="no_python_version_warning", | |
action="store_true", | |
default=False, | |
help="Silence deprecation warnings for upcoming unsupported Pythons.", | |
) | |
# Features that are now always on. A warning is printed if they are used. | |
ALWAYS_ENABLED_FEATURES = [ | |
"no-binary-enable-wheel-cache", # always on since 23.1 | |
] | |
use_new_feature: Callable[..., Option] = partial( | |
Option, | |
"--use-feature", | |
dest="features_enabled", | |
metavar="feature", | |
action="append", | |
default=[], | |
choices=[ | |
"fast-deps", | |
"truststore", | |
] | |
+ ALWAYS_ENABLED_FEATURES, | |
help="Enable new functionality, that may be backward incompatible.", | |
) | |
use_deprecated_feature: Callable[..., Option] = partial( | |
Option, | |
"--use-deprecated", | |
dest="deprecated_features_enabled", | |
metavar="feature", | |
action="append", | |
default=[], | |
choices=[ | |
"legacy-resolver", | |
], | |
help=("Enable deprecated functionality, that will be removed in the future."), | |
) | |
########## | |
# groups # | |
########## | |
general_group: Dict[str, Any] = { | |
"name": "General Options", | |
"options": [ | |
help_, | |
debug_mode, | |
isolated_mode, | |
require_virtualenv, | |
python, | |
verbose, | |
version, | |
quiet, | |
log, | |
no_input, | |
keyring_provider, | |
proxy, | |
retries, | |
timeout, | |
exists_action, | |
trusted_host, | |
cert, | |
client_cert, | |
cache_dir, | |
no_cache, | |
disable_pip_version_check, | |
no_color, | |
no_python_version_warning, | |
use_new_feature, | |
use_deprecated_feature, | |
], | |
} | |
index_group: Dict[str, Any] = { | |
"name": "Package Index Options", | |
"options": [ | |
index_url, | |
extra_index_url, | |
no_index, | |
find_links, | |
], | |
} | |