Как получить завершение bash для работы с псевдонимами?

Case in point:

Я на mac с bash v3.2.17, я использую git, установленный через macports с вариантом bash_completion.

когда я типа git checkout m<tab>. например, я завершаю его до master.

тем не менее, у меня есть псевдоним для git checkout, gco. Когда я набираю gco m<tab>, Я не получаю имя филиала автозаполненным.

В идеале я хотел бы, чтобы автозаполнение просто волшебно работало для всех моих псевдонимов. Возможно ли это? В противном случае, я бы хотел вручную настройте его для каждого псевдонима. Так как же мне поступить?

10 ответов


как указано в комментариях выше,

complete -o default -o nospace -F _git_checkout gco

больше не будет работать. Тем не менее, есть


Я также столкнулся с этой проблемой и придумал этот фрагмент кода. Это автоматически даст вам завершение для всех псевдонимов. Запустите его после объявления всех (или любого) псевдонима.

# wrap_alias takes three arguments:
# : The name of the alias
# : The command used in the alias
# : The arguments in the alias all in one string
# Generate a wrapper completion function (completer) for an alias
# based on the command and the given arguments, if there is a
# completer for the command, and set the wrapper as the completer for
# the alias.
function wrap_alias() {
  [[ "$#" == 3 ]] || return 1

  local alias_name=""
  local aliased_command=""
  local alias_arguments=""
  local num_alias_arguments=$(echo "$alias_arguments" | wc -w)

  # The completion currently being used for the aliased command.
  local completion=$(complete -p $aliased_command 2> /dev/null)

  # Only a completer based on a function can be wrapped so look for -F
  # in the current completion. This check will also catch commands
  # with no completer for which $completion will be empty.
  echo $completion | grep -q -- -F || return 0

  local namespace=alias_completion::

  # Extract the name of the completion function from a string that
  # looks like: something -F function_name something
  # First strip the beginning of the string up to the function name by
  # removing "* -F " from the front.
  local completion_function=${completion##* -F }
  # Then strip " *" from the end, leaving only the function name.
  completion_function=${completion_function%% *}

  # Try to prevent an infinite loop by not wrapping a function
  # generated by this function. This can happen when the user runs
  # this twice for an alias like ls='ls --color=auto' or alias l='ls'
  # and alias ls='l foo'
  [[ "${completion_function#$namespace}" != $completion_function ]] && return 0

  local wrapper_name="${namespace}${alias_name}"

  eval "
function ${wrapper_name}() {
  let COMP_CWORD+=$num_alias_arguments
  args=( \"${alias_arguments}\" )
  COMP_WORDS=( $aliased_command ${args[@]} ${COMP_WORDS[@]:1} )
  $completion_function
  }
"

  # To create the new completion we use the old one with two
  # replacements:
  # 1) Replace the function with the wrapper.
  local new_completion=${completion/-F * /-F $wrapper_name }
  # 2) Replace the command being completed with the alias.
  new_completion="${new_completion% *} $alias_name"

  eval "$new_completion"
}

# For each defined alias, extract the necessary elements and use them
# to call wrap_alias.
eval "$(alias -p | sed -e 's/alias \([^=][^=]*\)='\''\([^ ][^ ]*\) *\(.*\)'\''/wrap_alias   '\'''\'' /')"

unset wrap_alias

на git-completion.bash есть строка:

complete -o default -o nospace -F _git git

глядя на эту строку (и функцию _git), вы можете добавить эту строку в свой .bash_profile:

complete -o default -o nospace -F _git_checkout gco

у меня есть псевдонимы g= 'git', и в сочетании с моими псевдонимами git я печатаю такие вещи, как

$ g co <branchname>

более простым решением для моего конкретного случая использования было добавить одну строку в Git-завершение.

прямо под этой строкой:

__git_complete git _git

я добавил эту строку для обработки моего единственного псевдонима "g":

__git_complete g _git

В идеале я хотел бы, чтобы автозаполнение просто волшебно работало для всех моих псевдонимов. Возможно ли это?

Да, это возможно с complete-alias проект (в Linux). Поддержка Mac является экспериментальной, но пользователи сообщили об успехе.


эта страница форума показывает решение.

поставить эти строки в свой .bashrc или .bash_profile:

# Author.: Ole J
# Date...: 23.03.2008
# License: Whatever

# Wraps a completion function
# make-completion-wrapper <actual completion function> <name of new func.>
#                         <command name> <list supplied arguments>
# eg.
#   alias agi='apt-get install'
#   make-completion-wrapper _apt_get _apt_get_install apt-get install
# defines a function called _apt_get_install (that's ) that will complete
# the 'agi' alias. (complete -F _apt_get_install agi)
#
function make-completion-wrapper () {
    local function_name=""
    local arg_count=$(($#-3))
    local comp_function_name=""
    shift 2
    local function="
function $function_name {
    ((COMP_CWORD+=$arg_count))
    COMP_WORDS=( "$@" ${COMP_WORDS[@]:1} )
    "$comp_function_name"
    return 0
}"
    eval "$function"
}

# and now the commands that are specific to this SO question

alias gco='git checkout'

# we create a _git_checkout_mine function that will do the completion for "gco"
# using the completion function "_git"
make-completion-wrapper _git _git_checkout_mine git checkout

# we tell bash to actually use _git_checkout_mine to complete "gco"
complete -o bashdefault -o default -o nospace -F _git_checkout_mine gco

это решение похоже на balshetzer это, но только это на самом деле работает для меня. (у сценария бальшетцера были проблемы с некоторыми из моих псевдонимов.)


вы также можете попробовать использовать псевдонимы в Git. Например, в моем , У меня есть раздел, который выглядит так:

[alias]
        co = checkout

таким образом, вы можете ввести git co m<TAB>, и это должно расшириться до git co master, который является .


еще один вариант-использовать . Чтобы создать gco псевдоним git checkout просто оставлю это здесь:

_xfunc git __git_complete gco _git_checkout

затем в ~/.bashrc вы должны поставить только псевдоним себе:

alias gco='git checkout'

две линии. Вот и все.

объяснение:

на ~/bash_completion получает источник в конце основного сценария bash_completion. В gentoo я нашел основной скрипт в /usr/share/bash-completion/bash_completion.

на _xfunc git немного заботится о поиск git-completion файл для вас, поэтому вам не нужно ничего добавлять в ~/.bashrc.

принятый ответ требует, чтобы вы скопировали .git-completion.sh и источник его из своего ~/.bashrc файл, который я нахожу хромым.


PS: Я все еще пытаюсь выяснить, как не источник всей git-completion скрипт в мою среду bash. Пожалуйста, прокомментируйте или отредактируйте, если найдете способ.


вы просто должны найти и повторяющиеся строки, а не псевдоним.

Я alias d-m="docker-machine". На словах,d-m будет псевдонимом для docker-machine.

Итак, на Mac (через brew) файлы завершения находятся в cd `brew --prefix`/etc/bash_completion.d/.
Для моего случая я отредактировал файл под названием docker-machine.
На самом дне было написано:--9-->

complete -F _docker_machine docker-machine

поэтому я просто добавил еще одну строку с моим псевдонимом:

complete -F _docker_machine docker-machine
complete -F _docker_machine d-m

сначала найдите исходную команду завершения. Пример:

$ complete | grep git

complete -o bashdefault -o default -o nospace -F __git_wrap__git_main git

теперь добавьте их в свой скрипт запуска (например,~/.bashrc):

# copy the original statement, but replace the last command (git) with your alias (g)
complete -o bashdefault -o default -o nospace -F __git_wrap__git_main g

# load dynamically loaded completion functions (may not be required)
_completion_loader git

на _completion_loader строка может не потребоваться. Но в некоторых ситуациях функция завершения загружается динамически только после ввода команды и нажатия кнопки TAB в первый раз. Поэтому, если вы не использовали исходную команду, и попробуйте псевдоним + TAB, вы можете получить ошибку, как "bash: завершение: функция '_docker' не найдено."