123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134 |
- """A single place for constructing and exposing the main parser
- """
- import os
- import subprocess
- import sys
- from typing import List, Optional, Tuple
- from pip._internal.build_env import get_runnable_pip
- from pip._internal.cli import cmdoptions
- from pip._internal.cli.parser import ConfigOptionParser, UpdatingDefaultsHelpFormatter
- from pip._internal.commands import commands_dict, get_similar_commands
- from pip._internal.exceptions import CommandError
- from pip._internal.utils.misc import get_pip_version, get_prog
- __all__ = ["create_main_parser", "parse_command"]
- def create_main_parser() -> ConfigOptionParser:
- """Creates and returns the main parser for pip's CLI"""
- parser = ConfigOptionParser(
- usage="\n%prog <command> [options]",
- add_help_option=False,
- formatter=UpdatingDefaultsHelpFormatter(),
- name="global",
- prog=get_prog(),
- )
- parser.disable_interspersed_args()
- parser.version = get_pip_version()
- # add the general options
- gen_opts = cmdoptions.make_option_group(cmdoptions.general_group, parser)
- parser.add_option_group(gen_opts)
- # so the help formatter knows
- parser.main = True # type: ignore
- # create command listing for description
- description = [""] + [
- f"{name:27} {command_info.summary}"
- for name, command_info in commands_dict.items()
- ]
- parser.description = "\n".join(description)
- return parser
- def identify_python_interpreter(python: str) -> Optional[str]:
- # If the named file exists, use it.
- # If it's a directory, assume it's a virtual environment and
- # look for the environment's Python executable.
- if os.path.exists(python):
- if os.path.isdir(python):
- # bin/python for Unix, Scripts/python.exe for Windows
- # Try both in case of odd cases like cygwin.
- for exe in ("bin/python", "Scripts/python.exe"):
- py = os.path.join(python, exe)
- if os.path.exists(py):
- return py
- else:
- return python
- # Could not find the interpreter specified
- return None
- def parse_command(args: List[str]) -> Tuple[str, List[str]]:
- parser = create_main_parser()
- # Note: parser calls disable_interspersed_args(), so the result of this
- # call is to split the initial args into the general options before the
- # subcommand and everything else.
- # For example:
- # args: ['--timeout=5', 'install', '--user', 'INITools']
- # general_options: ['--timeout==5']
- # args_else: ['install', '--user', 'INITools']
- general_options, args_else = parser.parse_args(args)
- # --python
- if general_options.python and "_PIP_RUNNING_IN_SUBPROCESS" not in os.environ:
- # Re-invoke pip using the specified Python interpreter
- interpreter = identify_python_interpreter(general_options.python)
- if interpreter is None:
- raise CommandError(
- f"Could not locate Python interpreter {general_options.python}"
- )
- pip_cmd = [
- interpreter,
- get_runnable_pip(),
- ]
- pip_cmd.extend(args)
- # Set a flag so the child doesn't re-invoke itself, causing
- # an infinite loop.
- os.environ["_PIP_RUNNING_IN_SUBPROCESS"] = "1"
- returncode = 0
- try:
- proc = subprocess.run(pip_cmd)
- returncode = proc.returncode
- except (subprocess.SubprocessError, OSError) as exc:
- raise CommandError(f"Failed to run pip under {interpreter}: {exc}")
- sys.exit(returncode)
- # --version
- if general_options.version:
- sys.stdout.write(parser.version)
- sys.stdout.write(os.linesep)
- sys.exit()
- # pip || pip help -> print_help()
- if not args_else or (args_else[0] == "help" and len(args_else) == 1):
- parser.print_help()
- sys.exit()
- # the subcommand name
- cmd_name = args_else[0]
- if cmd_name not in commands_dict:
- guess = get_similar_commands(cmd_name)
- msg = [f'unknown command "{cmd_name}"']
- if guess:
- msg.append(f'maybe you meant "{guess}"')
- raise CommandError(" - ".join(msg))
- # all the args without the subcommand
- cmd_args = args[:]
- cmd_args.remove(cmd_name)
- return cmd_name, cmd_args
|