diff --git a/news/0741cad6-3007-47f8-9c53-984e9116c7ff.bugfix.rst b/news/0741cad6-3007-47f8-9c53-984e9116c7ff.bugfix.rst new file mode 100644 index 00000000000..d292fe37260 --- /dev/null +++ b/news/0741cad6-3007-47f8-9c53-984e9116c7ff.bugfix.rst @@ -0,0 +1 @@ +Fix: include sub-actions in tab completion diff --git a/src/pip/_internal/cli/autocompletion.py b/src/pip/_internal/cli/autocompletion.py index f3f70ac8553..9a2645bc7bc 100644 --- a/src/pip/_internal/cli/autocompletion.py +++ b/src/pip/_internal/cli/autocompletion.py @@ -101,6 +101,10 @@ def autocomplete() -> None: if option[1] and option[0][:2] == "--": opt_label += "=" print(opt_label) + + for handler_name in subcommand.handler_map(): + if handler_name.startswith(current): + print(handler_name) else: # show main parser options only when necessary diff --git a/src/pip/_internal/cli/base_command.py b/src/pip/_internal/cli/base_command.py index bc1ab65949d..6d2e123c7af 100644 --- a/src/pip/_internal/cli/base_command.py +++ b/src/pip/_internal/cli/base_command.py @@ -7,7 +7,7 @@ import sys import traceback from optparse import Values -from typing import List, Optional, Tuple +from typing import Callable, List, Optional, Tuple from pip._vendor.rich import reconfigure from pip._vendor.rich import traceback as rich_traceback @@ -229,3 +229,9 @@ def _main(self, args: List[str]) -> int: options.cache_dir = None return self._run_wrapper(level_number, options, args) + + def handler_map(self) -> dict[str, Callable[[Values, list[str]], None]]: + """ + map of names to handler actions for commands with sub-actions + """ + return {} diff --git a/src/pip/_internal/commands/cache.py b/src/pip/_internal/commands/cache.py index ad65641edb2..293cf1343b2 100644 --- a/src/pip/_internal/commands/cache.py +++ b/src/pip/_internal/commands/cache.py @@ -1,7 +1,7 @@ import os import textwrap from optparse import Values -from typing import Any, List +from typing import Callable, List from pip._internal.cli.base_command import Command from pip._internal.cli.status_codes import ERROR, SUCCESS @@ -49,8 +49,8 @@ def add_options(self) -> None: self.parser.insert_option_group(0, self.cmd_opts) - def run(self, options: Values, args: List[str]) -> int: - handlers = { + def handler_map(self) -> dict[str, Callable[[Values, list[str]], None]]: + return { "dir": self.get_cache_dir, "info": self.get_cache_info, "list": self.list_cache_items, @@ -58,15 +58,18 @@ def run(self, options: Values, args: List[str]) -> int: "purge": self.purge_cache, } + def run(self, options: Values, args: List[str]) -> int: + handler_map = self.handler_map() + if not options.cache_dir: logger.error("pip cache commands can not function since cache is disabled.") return ERROR # Determine action - if not args or args[0] not in handlers: + if not args or args[0] not in handler_map: logger.error( "Need an action (%s) to perform.", - ", ".join(sorted(handlers)), + ", ".join(sorted(handler_map)), ) return ERROR @@ -74,20 +77,20 @@ def run(self, options: Values, args: List[str]) -> int: # Error handling happens here, not in the action-handlers. try: - handlers[action](options, args[1:]) + handler_map[action](options, args[1:]) except PipError as e: logger.error(e.args[0]) return ERROR return SUCCESS - def get_cache_dir(self, options: Values, args: List[Any]) -> None: + def get_cache_dir(self, options: Values, args: List[str]) -> None: if args: raise CommandError("Too many arguments") logger.info(options.cache_dir) - def get_cache_info(self, options: Values, args: List[Any]) -> None: + def get_cache_info(self, options: Values, args: List[str]) -> None: if args: raise CommandError("Too many arguments") @@ -129,7 +132,7 @@ def get_cache_info(self, options: Values, args: List[Any]) -> None: logger.info(message) - def list_cache_items(self, options: Values, args: List[Any]) -> None: + def list_cache_items(self, options: Values, args: List[str]) -> None: if len(args) > 1: raise CommandError("Too many arguments") @@ -161,7 +164,7 @@ def format_for_abspath(self, files: List[str]) -> None: if files: logger.info("\n".join(sorted(files))) - def remove_cache_items(self, options: Values, args: List[Any]) -> None: + def remove_cache_items(self, options: Values, args: List[str]) -> None: if len(args) > 1: raise CommandError("Too many arguments") @@ -188,7 +191,7 @@ def remove_cache_items(self, options: Values, args: List[Any]) -> None: logger.verbose("Removed %s", filename) logger.info("Files removed: %s (%s)", len(files), format_size(bytes_removed)) - def purge_cache(self, options: Values, args: List[Any]) -> None: + def purge_cache(self, options: Values, args: List[str]) -> None: if args: raise CommandError("Too many arguments") diff --git a/src/pip/_internal/commands/configuration.py b/src/pip/_internal/commands/configuration.py index 1a1dc6b6cd8..56754c5b0d7 100644 --- a/src/pip/_internal/commands/configuration.py +++ b/src/pip/_internal/commands/configuration.py @@ -2,7 +2,7 @@ import os import subprocess from optparse import Values -from typing import Any, List, Optional +from typing import Any, Callable, List, Optional from pip._internal.cli.base_command import Command from pip._internal.cli.status_codes import ERROR, SUCCESS @@ -93,8 +93,8 @@ def add_options(self) -> None: self.parser.insert_option_group(0, self.cmd_opts) - def run(self, options: Values, args: List[str]) -> int: - handlers = { + def handler_map(self) -> dict[str, Callable[[Values, list[str]], None]]: + return { "list": self.list_values, "edit": self.open_in_editor, "get": self.get_name, @@ -103,11 +103,14 @@ def run(self, options: Values, args: List[str]) -> int: "debug": self.list_config_values, } + def run(self, options: Values, args: List[str]) -> int: + handler_map = self.handler_map() + # Determine action - if not args or args[0] not in handlers: + if not args or args[0] not in handler_map: logger.error( "Need an action (%s) to perform.", - ", ".join(sorted(handlers)), + ", ".join(sorted(handler_map)), ) return ERROR @@ -131,7 +134,7 @@ def run(self, options: Values, args: List[str]) -> int: # Error handling happens here, not in the action-handlers. try: - handlers[action](options, args[1:]) + handler_map[action](options, args[1:]) except PipError as e: logger.error(e.args[0]) return ERROR diff --git a/src/pip/_internal/commands/index.py b/src/pip/_internal/commands/index.py index 2e2661bba71..622dc1d8dc5 100644 --- a/src/pip/_internal/commands/index.py +++ b/src/pip/_internal/commands/index.py @@ -1,6 +1,6 @@ import logging from optparse import Values -from typing import Any, Iterable, List, Optional +from typing import Any, Callable, Iterable, List, Optional from pip._vendor.packaging.version import Version @@ -45,11 +45,14 @@ def add_options(self) -> None: self.parser.insert_option_group(0, index_opts) self.parser.insert_option_group(0, self.cmd_opts) - def run(self, options: Values, args: List[str]) -> int: - handlers = { + def handler_map(self) -> dict[str, Callable[[Values, list[str]], None]]: + return { "versions": self.get_available_package_versions, } + def run(self, options: Values, args: List[str]) -> int: + handler_map = self.handler_map() + logger.warning( "pip index is currently an experimental command. " "It may be removed/changed in a future release " @@ -57,10 +60,10 @@ def run(self, options: Values, args: List[str]) -> int: ) # Determine action - if not args or args[0] not in handlers: + if not args or args[0] not in handler_map: logger.error( "Need an action (%s) to perform.", - ", ".join(sorted(handlers)), + ", ".join(sorted(handler_map)), ) return ERROR @@ -68,7 +71,7 @@ def run(self, options: Values, args: List[str]) -> int: # Error handling happens here, not in the action-handlers. try: - handlers[action](options, args[1:]) + handler_map[action](options, args[1:]) except PipError as e: logger.error(e.args[0]) return ERROR diff --git a/tests/functional/test_completion.py b/tests/functional/test_completion.py index a52b135c8b0..549da6ece3c 100644 --- a/tests/functional/test_completion.py +++ b/tests/functional/test_completion.py @@ -421,3 +421,28 @@ def test_completion_uses_same_executable_name( expect_stderr=deprecated_python, ) assert executable_name in result.stdout + + +@pytest.mark.parametrize( + "subcommand, handler_prefix, expected", + [ + ("cache", "d", "dir"), + ("cache", "in", "info"), + ("cache", "l", "list"), + ("cache", "re", "remove"), + ("cache", "pu", "purge"), + ("config", "li", "list"), + ("config", "e", "edit"), + ("config", "ge", "get"), + ("config", "se", "set"), + ("config", "unse", "unset"), + ("config", "d", "debug"), + ("index", "ve", "versions"), + ], +) +def test_completion_for_action_handler( + subcommand: str, handler_prefix: str, expected: str, autocomplete: DoAutocomplete +) -> None: + res, _ = autocomplete(f"pip {subcommand} {handler_prefix}", cword="2") + + assert [expected] == res.stdout.split()