-
Notifications
You must be signed in to change notification settings - Fork 41
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add
--install-tab-completion
option for installing shell completion
The user gets four options to choose from bash, zsh, fish and auto (auto detects shell using `shellingham`). To activate shell completion run: kart config --install-completion auto Replace `auto` with desired shell eg: kart config --install-completion zsh Completion will take effect once you restart the terminal.
- Loading branch information
1 parent
0724ed4
commit 5b3145a
Showing
10 changed files
with
256 additions
and
21 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,157 @@ | ||
import os | ||
import re | ||
import sys | ||
from enum import Enum | ||
from pathlib import Path | ||
from typing import Optional, Tuple, Any, Optional | ||
|
||
import click | ||
from click.shell_completion import _SOURCE_BASH, _SOURCE_ZSH, _SOURCE_FISH | ||
|
||
try: | ||
import shellingham | ||
except ImportError: | ||
shellingham = None | ||
|
||
|
||
class Shells(str, Enum): | ||
bash = "bash" | ||
zsh = "zsh" | ||
fish = "fish" | ||
|
||
|
||
_completion_scripts = { | ||
"bash": _SOURCE_BASH, | ||
"zsh": _SOURCE_ZSH, | ||
"fish": _SOURCE_FISH, | ||
} | ||
|
||
_invalid_ident_char_re = re.compile(r"[^a-zA-Z0-9_]") | ||
|
||
|
||
def get_completion_script(*, prog_name: str, complete_var: str, shell: str) -> str: | ||
cf_name = _invalid_ident_char_re.sub("", prog_name.replace("-", "_")) | ||
script = _completion_scripts.get(shell) | ||
if script is None: | ||
click.echo(f"Shell {shell} not supported.", err=True) | ||
sys.exit(1) | ||
return ( | ||
script | ||
% dict( | ||
complete_func="_{}_completion".format(cf_name), | ||
complete_var=complete_var, | ||
prog_name=prog_name, | ||
) | ||
).strip() | ||
|
||
|
||
def install_helper( | ||
prog_name, complete_var, shell, completion_path, rc_path, completion_init_lines | ||
): | ||
rc_path.parent.mkdir(parents=True, exist_ok=True) | ||
rc_content = "" | ||
if rc_path.is_file(): | ||
rc_content = rc_path.read_text() | ||
for line in completion_init_lines: | ||
if line not in rc_content: | ||
rc_content += f"\n{line}" | ||
rc_content += "\n" | ||
rc_path.write_text(rc_content) | ||
# Install completion | ||
completion_path.parent.mkdir(parents=True, exist_ok=True) | ||
script_content = get_completion_script( | ||
prog_name=prog_name, complete_var=complete_var, shell=shell | ||
) | ||
completion_path.write_text(script_content) | ||
return completion_path | ||
|
||
|
||
def install_bash(*, prog_name: str, complete_var: str, shell: str) -> Path: | ||
# Ref: https://github.com/scop/bash-completion#faq | ||
# It seems bash-completion is the official completion system for bash: | ||
# Ref: https://www.gnu.org/software/bash/manual/html_node/A-Programmable-Completion-Example.html | ||
# But installing in the locations from the docs doesn't seem to have effect | ||
completion_path = Path.home() / f".bash_completions/{prog_name}.sh" | ||
bashrc_path = Path.home() / ".bashrc" | ||
completion_init_lines = [f"source {completion_path}"] | ||
return install_helper( | ||
prog_name, | ||
complete_var, | ||
shell, | ||
completion_path, | ||
bashrc_path, | ||
completion_init_lines, | ||
) | ||
|
||
|
||
def install_zsh(*, prog_name: str, complete_var: str, shell: str) -> Path: | ||
# Setup Zsh and load ~/.zfunc | ||
completion_path = Path.home() / f".zfunc/_{prog_name}" | ||
zshrc_path = Path.home() / ".zshrc" | ||
completion_init_lines = [ | ||
"autoload -Uz compinit", | ||
"zstyle ':completion:*' menu select", | ||
"fpath+=~/.zfunc; compinit", | ||
] | ||
return install_helper( | ||
prog_name, | ||
complete_var, | ||
shell, | ||
completion_path, | ||
zshrc_path, | ||
completion_init_lines, | ||
) | ||
|
||
|
||
def install_fish(*, prog_name: str, complete_var: str, shell: str) -> Path: | ||
path_obj = Path.home() / f".config/fish/completions/{prog_name}.fish" | ||
parent_dir: Path = path_obj.parent | ||
parent_dir.mkdir(parents=True, exist_ok=True) | ||
script_content = get_completion_script( | ||
prog_name=prog_name, complete_var=complete_var, shell=shell | ||
) | ||
path_obj.write_text(f"{script_content}\n") | ||
return path_obj | ||
|
||
|
||
def install( | ||
shell: Optional[str] = None, | ||
prog_name: Optional[str] = None, | ||
complete_var: Optional[str] = None, | ||
) -> Tuple[str, Path]: | ||
prog_name = prog_name or click.get_current_context().find_root().info_name | ||
assert prog_name | ||
if complete_var is None: | ||
complete_var = "_{}_COMPLETE".format(prog_name.replace("-", "_").upper()) | ||
if shell is None and shellingham is not None: | ||
shell, _ = shellingham.detect_shell() | ||
if shell == "bash": | ||
installed_path = install_bash( | ||
prog_name=prog_name, complete_var=complete_var, shell=shell | ||
) | ||
return shell, installed_path | ||
elif shell == "zsh": | ||
installed_path = install_zsh( | ||
prog_name=prog_name, complete_var=complete_var, shell=shell | ||
) | ||
return shell, installed_path | ||
elif shell == "fish": | ||
installed_path = install_fish( | ||
prog_name=prog_name, complete_var=complete_var, shell=shell | ||
) | ||
return shell, installed_path | ||
else: | ||
click.echo(f"Shell {shell} is not supported.") | ||
raise click.exceptions.Exit(1) | ||
|
||
|
||
def install_callback(ctx: click.Context, param: click.Parameter, value: Any) -> Any: | ||
if not value or ctx.resilient_parsing: | ||
return value | ||
if value == "auto": | ||
shell, path = install() | ||
else: | ||
shell, path = install(shell=value) | ||
click.secho(f"{shell} completion installed in {path}", fg="green") | ||
click.echo("Completion will take effect once you restart the terminal") | ||
sys.exit(0) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -11,4 +11,5 @@ pytest-helpers-namespace | |
pytest-benchmark[aspect] | ||
pytest-profiling | ||
pytest-shard | ||
pytest-mock | ||
html5lib |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.