| """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 |