2020-10-18 14:52:13 +05:30
|
|
|
{%- let SECTION = "# =============================================================================\n#" -%}
|
|
|
|
{%- let NOT_CONFIGURED = "# -- not configured --" -%}
|
|
|
|
|
2020-11-11 00:41:26 +05:30
|
|
|
"""Initialize zoxide on Xonsh."""
|
|
|
|
|
2020-10-18 14:52:13 +05:30
|
|
|
import os
|
|
|
|
import os.path
|
2020-10-20 23:29:28 +05:30
|
|
|
import subprocess
|
|
|
|
import sys
|
2020-11-11 00:41:26 +05:30
|
|
|
{%- if cmd.is_some() %}
|
|
|
|
from builtins import aliases # type: ignore # pylint: disable=no-name-in-module
|
|
|
|
{%- endif %}
|
|
|
|
{%- if hook != Hook::None %}
|
|
|
|
from builtins import events # type: ignore # pylint: disable=no-name-in-module
|
|
|
|
{%- endif %}
|
2020-10-20 23:29:28 +05:30
|
|
|
from subprocess import CalledProcessError
|
2020-11-11 00:41:26 +05:30
|
|
|
from typing import AnyStr, List, Optional
|
|
|
|
|
|
|
|
import xonsh.dirstack # type: ignore
|
2020-10-18 14:52:13 +05:30
|
|
|
|
|
|
|
{{ SECTION }}
|
|
|
|
# Utility functions for zoxide.
|
|
|
|
#
|
|
|
|
|
2020-11-11 00:41:26 +05:30
|
|
|
|
2020-10-18 14:52:13 +05:30
|
|
|
def __zoxide_pwd() -> str:
|
2020-11-11 00:41:26 +05:30
|
|
|
"""pwd based on the value of _ZO_RESOLVE_SYMLINKS."""
|
2020-10-18 14:52:13 +05:30
|
|
|
{%- if resolve_symlinks %}
|
2020-11-11 00:41:26 +05:30
|
|
|
pwd = os.getcwd()
|
2020-10-18 14:52:13 +05:30
|
|
|
{%- else %}
|
2020-11-11 00:41:26 +05:30
|
|
|
pwd = os.getenv("PWD")
|
|
|
|
if pwd is None:
|
|
|
|
raise Exception("$PWD not found in env")
|
2020-10-18 14:52:13 +05:30
|
|
|
{%- endif %}
|
2020-11-11 00:41:26 +05:30
|
|
|
return pwd
|
|
|
|
|
|
|
|
|
|
|
|
def __zoxide_cd(path: Optional[AnyStr] = None):
|
|
|
|
"""cd + custom logic based on the value of _ZO_ECHO."""
|
|
|
|
if path is None:
|
|
|
|
args = []
|
|
|
|
elif isinstance(path, bytes):
|
|
|
|
args = [path.decode("utf-8")]
|
|
|
|
elif isinstance(path, str):
|
|
|
|
args = [path]
|
|
|
|
_, exc, _ = xonsh.dirstack.cd(args)
|
|
|
|
if exc is not None:
|
|
|
|
raise Exception(exc)
|
|
|
|
{%- if echo %}
|
|
|
|
print(__zoxide_pwd())
|
|
|
|
{%- endif %}
|
|
|
|
|
|
|
|
|
|
|
|
class ZoxideSilentException(Exception):
|
|
|
|
"""Exit without complaining."""
|
|
|
|
|
|
|
|
|
|
|
|
def __zoxide_errhandler(func):
|
|
|
|
"""Print exception and exit with error code 1."""
|
|
|
|
|
|
|
|
def wrapper(args: List[str]):
|
|
|
|
try:
|
|
|
|
func(args)
|
|
|
|
return 0
|
|
|
|
except ZoxideSilentException:
|
|
|
|
return 1
|
|
|
|
except Exception as exc: # pylint: disable=broad-except
|
|
|
|
print(f"zoxide: {exc}", file=sys.stderr)
|
|
|
|
return 1
|
|
|
|
|
|
|
|
return wrapper
|
2020-10-18 14:52:13 +05:30
|
|
|
|
|
|
|
|
|
|
|
{{ SECTION }}
|
|
|
|
# Hook configuration for zoxide.
|
|
|
|
#
|
|
|
|
|
|
|
|
# Initialize hook to add new entries to the database.
|
|
|
|
{%- match hook %}
|
|
|
|
{%- when Hook::None %}
|
|
|
|
{{ NOT_CONFIGURED }}
|
|
|
|
|
|
|
|
{%- when Hook::Prompt %}
|
2020-11-11 00:41:26 +05:30
|
|
|
@events.on_post_prompt # type: ignore # pylint:disable=undefined-variable
|
2020-10-18 14:52:13 +05:30
|
|
|
|
|
|
|
{%- when Hook::Pwd %}
|
2020-11-11 00:41:26 +05:30
|
|
|
@events.on_chdir # type: ignore # pylint:disable=undefined-variable
|
2020-10-18 14:52:13 +05:30
|
|
|
|
|
|
|
{%- endmatch %}
|
2020-11-11 00:41:26 +05:30
|
|
|
def __zoxide_hook(**_kwargs):
|
|
|
|
"""Hook to add new entries to the database."""
|
|
|
|
pwd = __zoxide_pwd()
|
|
|
|
subprocess.run(["zoxide", "add", pwd], check=False)
|
|
|
|
|
2020-10-18 14:52:13 +05:30
|
|
|
|
|
|
|
{{ SECTION }}
|
|
|
|
# When using zoxide with --no-aliases, alias these internal functions as
|
|
|
|
# desired.
|
|
|
|
#
|
|
|
|
|
2020-10-20 23:29:28 +05:30
|
|
|
|
2020-11-11 00:41:26 +05:30
|
|
|
@__zoxide_errhandler
|
|
|
|
def __zoxide_z(args: List[str]):
|
|
|
|
"""Jump to a directory using only keywords."""
|
|
|
|
if args == []:
|
|
|
|
__zoxide_cd()
|
|
|
|
elif args == ["-"]:
|
|
|
|
__zoxide_cd("-")
|
|
|
|
elif len(args) == 1 and os.path.isdir(args[0]):
|
|
|
|
__zoxide_cd(args[0])
|
|
|
|
else:
|
2020-10-20 23:29:28 +05:30
|
|
|
try:
|
2020-11-11 00:41:26 +05:30
|
|
|
__zoxide_cmd = subprocess.run(
|
|
|
|
["zoxide", "query", "--"] + args, check=True, stdout=subprocess.PIPE
|
|
|
|
)
|
|
|
|
except CalledProcessError as exc:
|
|
|
|
raise ZoxideSilentException() from exc
|
2020-10-20 23:29:28 +05:30
|
|
|
|
2020-11-11 00:41:26 +05:30
|
|
|
__zoxide_result = __zoxide_cmd.stdout[:-1]
|
2020-10-20 23:29:28 +05:30
|
|
|
__zoxide_cd(__zoxide_result)
|
2020-10-18 14:52:13 +05:30
|
|
|
|
2020-10-20 23:29:28 +05:30
|
|
|
|
2020-11-11 00:41:26 +05:30
|
|
|
def __zoxide_zi(args: List[str]):
|
|
|
|
"""Jump to a directory using interactive search."""
|
2020-10-20 23:29:28 +05:30
|
|
|
try:
|
2020-11-11 00:41:26 +05:30
|
|
|
__zoxide_cmd = subprocess.run(
|
|
|
|
["zoxide", "query", "-i", "--"] + args, check=True, stdout=subprocess.PIPE
|
|
|
|
)
|
|
|
|
except CalledProcessError as exc:
|
|
|
|
raise ZoxideSilentException() from exc
|
2020-10-20 23:29:28 +05:30
|
|
|
|
2020-11-11 00:41:26 +05:30
|
|
|
__zoxide_result = __zoxide_cmd.stdout[:-1]
|
2020-10-20 23:29:28 +05:30
|
|
|
__zoxide_cd(__zoxide_result)
|
|
|
|
|
2020-10-18 14:52:13 +05:30
|
|
|
|
2020-11-11 00:41:26 +05:30
|
|
|
def __zoxide_za(args: List[str]):
|
|
|
|
"""Add a new entry to the database."""
|
|
|
|
try:
|
|
|
|
subprocess.run(["zoxide", "add"] + args, check=True)
|
|
|
|
except CalledProcessError as exc:
|
|
|
|
raise ZoxideSilentException from exc
|
|
|
|
|
|
|
|
|
|
|
|
def __zoxide_zq(args: List[str]):
|
|
|
|
"""Query an entry from the database using only keywords."""
|
|
|
|
try:
|
|
|
|
subprocess.run(["zoxide", "query"] + args, check=True)
|
|
|
|
except CalledProcessError as exc:
|
|
|
|
raise ZoxideSilentException from exc
|
|
|
|
|
|
|
|
|
|
|
|
def __zoxide_zqi(args: List[str]):
|
|
|
|
"""Query an entry from the database using interactive selection."""
|
|
|
|
try:
|
|
|
|
subprocess.run(["zoxide", "query", "-i"] + args, check=True)
|
|
|
|
except CalledProcessError as exc:
|
|
|
|
raise ZoxideSilentException from exc
|
|
|
|
|
2020-10-18 14:52:13 +05:30
|
|
|
|
2020-11-11 00:41:26 +05:30
|
|
|
def __zoxide_zr(args: List[str]):
|
|
|
|
"""Remove an entry from the database using the exact path."""
|
|
|
|
try:
|
|
|
|
subprocess.run(["zoxide", "remove"] + args, check=True)
|
|
|
|
except CalledProcessError as exc:
|
|
|
|
raise ZoxideSilentException from exc
|
2020-10-18 14:52:13 +05:30
|
|
|
|
|
|
|
|
2020-11-11 00:41:26 +05:30
|
|
|
def __zoxide_zri(args: List[str]):
|
|
|
|
"""Remove an entry from the database using interactive selection."""
|
|
|
|
try:
|
|
|
|
subprocess.run(["zoxide", "remove", "-i"] + args, check=True)
|
|
|
|
except CalledProcessError as exc:
|
|
|
|
raise ZoxideSilentException from exc
|
2020-10-18 14:52:13 +05:30
|
|
|
|
|
|
|
|
|
|
|
{{ SECTION }}
|
|
|
|
# Convenient aliases for zoxide. Disable these using --no-aliases.
|
|
|
|
#
|
|
|
|
|
|
|
|
{%- match cmd %}
|
|
|
|
{%- when Some with (cmd) %}
|
|
|
|
|
2020-11-11 00:41:26 +05:30
|
|
|
aliases["{{cmd}}"] = __zoxide_z
|
|
|
|
aliases["{{cmd}}i"] = __zoxide_zi
|
|
|
|
aliases["{{cmd}}a"] = __zoxide_za
|
|
|
|
aliases["{{cmd}}q"] = __zoxide_zq
|
|
|
|
aliases["{{cmd}}qi"] = __zoxide_zqi
|
|
|
|
aliases["{{cmd}}r"] = __zoxide_zr
|
|
|
|
aliases["{{cmd}}ri"] = __zoxide_zri
|
2020-10-18 14:52:13 +05:30
|
|
|
|
|
|
|
{%- when None %}
|
|
|
|
{{ NOT_CONFIGURED }}
|
|
|
|
|
|
|
|
{%- endmatch %}
|
|
|
|
|
|
|
|
{{ SECTION }}
|
|
|
|
# To initialize zoxide with xonsh, add the following line to your xonsh
|
|
|
|
# configuration file (usually ~/.xonshrc):
|
|
|
|
#
|
|
|
|
# execx($(zoxide init xonsh), 'exec', __xonsh__.ctx, filename='zoxide')
|