diff --git a/.bash_completion.d/docker_machine.bash b/.bash_completion.d/docker-machine.bash similarity index 81% rename from .bash_completion.d/docker_machine.bash rename to .bash_completion.d/docker-machine.bash index 1c9f2aba946d96104a6cfbfff685ef0d982f0804..feb08247f3c46d21108ce879e16d11aa77c6f8e1 100644 --- a/.bash_completion.d/docker_machine.bash +++ b/.bash_completion.d/docker-machine.bash @@ -62,7 +62,7 @@ _docker_machine_value_of_option() { } # Returns `key` if we are currently completing the value of a map option -# (`key=value`) which matches the extglob passed in as an argument. +# (`key=value`) which matches the glob passed in as an argument. # This function is needed for key-specific argument completions. _docker_machine_map_key_of_current_option() { local glob="$1" @@ -83,9 +83,43 @@ _docker_machine_map_key_of_current_option() { [ "${words[$glob_pos]}" = "=" ] && ((glob_pos--)) # --option=key=value syntax - [[ ${words[$glob_pos]} == @($glob) ]] && echo "$key" + [[ ${words[$glob_pos]} == $glob ]] && echo "$key" } +# Finds the position of the first word that is neither option nor an option's argument. +# If there are options that require arguments, you need to pass a glob describing +# those options, e.g. "--option1|-o|--option2". +# Use this function to restrict completions to exact positions after the options. +_docker_machine_pos_first_nonflag() { + local argument_flags=$1 + + local counter=$((${subcommand_pos:-${command_pos}} + 1)) + while [ "$counter" -le "$cword" ]; do + if [ -n "$argument_flags" ] && eval "case '${words[$counter]}' in $argument_flags) true ;; *) false ;; esac"; then + (( counter++ )) + # eat "=" in case of --option=arg syntax + [ "${words[$counter]}" = "=" ] && (( counter++ )) + else + case "${words[$counter]}" in + -*) + ;; + *) + break + ;; + esac + fi + + # Bash splits words at "=", retaining "=" as a word, examples: + # "--debug=false" => 3 words, "--log-opt syslog-facility=daemon" => 4 words + while [ "${words[$counter + 1]}" = "=" ] ; do + counter=$(( counter + 2)) + done + + (( counter++ )) + done + + echo $counter +} # --- completion functions --------------------------------------------------- _docker_machine_active() { @@ -208,6 +242,21 @@ _docker_machine_ls() { fi } +_docker_machine_mount() { + if [[ "${cur}" == -* ]]; then + COMPREPLY=($(compgen -W "--help --unmount -u" -- "${cur}")) + else + local pos=$(_docker_machine_pos_first_nonflag) + if [ "$cword" -eq "$pos" ]; then + # We can't complete remote filesystems. All we can do here is to complete the machine. + COMPREPLY=($(compgen -W "$(_docker_machine_machines --filter state=Running)" -S: -- "${cur}")) + _docker_machine_nospace + elif [ "$cword" -eq "$((pos + 1))" ]; then + _filedir -d + fi + fi +} + _docker_machine_provision() { if [[ "${cur}" == -* ]]; then COMPREPLY=($(compgen -W "--help" -- "${cur}")) @@ -218,7 +267,7 @@ _docker_machine_provision() { _docker_machine_regenerate_certs() { if [[ "${cur}" == -* ]]; then - COMPREPLY=($(compgen -W "--force -f --help" -- "${cur}")) + COMPREPLY=($(compgen -W "--client-certs --force -f --help" -- "${cur}")) else COMPREPLY=($(compgen -W "$(_docker_machine_machines --filter state=Running)" -- "${cur}")) fi @@ -250,7 +299,7 @@ _docker_machine_ssh() { _docker_machine_scp() { if [[ "${cur}" == -* ]]; then - COMPREPLY=($(compgen -W "--delta -d --help --recursive -r" -- "${cur}")) + COMPREPLY=($(compgen -W "--delta -d --help --quiet -q --recursive -r" -- "${cur}")) else _filedir # It would be really nice to ssh to the machine and ls to complete @@ -329,7 +378,7 @@ _docker_machine_docker_machine() { _docker_machine() { COMPREPLY=() - local commands=(active config create env inspect ip kill ls provision regenerate-certs restart rm ssh scp start status stop upgrade url version help) + local commands=(active config create env inspect ip kill ls mount provision regenerate-certs restart rm ssh scp start status stop upgrade url version help) local flags=(--debug --native-ssh --github-api-token --bugsnag-api-token --help --version) local wants_dir=(--storage-path) @@ -343,7 +392,7 @@ _docker_machine() { local cur prev words cword _get_comp_words_by_ref -n : cur prev words cword local i - local command=docker-machine + local command=docker-machine command_pos=0 for (( i=1; i < ${cword}; ++i)); do local word=${words[i]} @@ -352,6 +401,7 @@ _docker_machine() { (( ++i )) elif [[ " ${commands[*]} " =~ " ${word} " ]]; then command=${word} + command_pos=$i fi done diff --git a/.bash_completion.d/docker_compose.bash b/.bash_completion.d/docker_compose.bash deleted file mode 100644 index faa537cb35d2fd9071dda5c055d7c51eeda14169..0000000000000000000000000000000000000000 --- a/.bash_completion.d/docker_compose.bash +++ /dev/null @@ -1,661 +0,0 @@ -#!/bin/bash -# -# bash completion for docker-compose -# -# This work is based on the completion for the docker command. -# -# This script provides completion of: -# - commands and their options -# - service names -# - filepaths -# -# To enable the completions either: -# - place this file in /etc/bash_completion.d -# or -# - copy this file to e.g. ~/.docker-compose-completion.sh and add the line -# below to your .bashrc after bash completion features are loaded -# . ~/.docker-compose-completion.sh - - -__docker_compose_q() { - docker-compose 2>/dev/null "${top_level_options[@]}" "$@" -} - -# Transforms a multiline list of strings into a single line string -# with the words separated by "|". -__docker_compose_to_alternatives() { - local parts=( $1 ) - local IFS='|' - echo "${parts[*]}" -} - -# Transforms a multiline list of options into an extglob pattern -# suitable for use in case statements. -__docker_compose_to_extglob() { - local extglob=$( __docker_compose_to_alternatives "$1" ) - echo "@($extglob)" -} - -# Determines whether the option passed as the first argument exist on -# the commandline. The option may be a pattern, e.g. `--force|-f`. -__docker_compose_has_option() { - local pattern="$1" - for (( i=2; i < $cword; ++i)); do - if [[ ${words[$i]} =~ ^($pattern)$ ]] ; then - return 0 - fi - done - return 1 -} - -# Returns `key` if we are currently completing the value of a map option (`key=value`) -# which matches the extglob passed in as an argument. -# This function is needed for key-specific completions. -__docker_compose_map_key_of_current_option() { - local glob="$1" - - local key glob_pos - if [ "$cur" = "=" ] ; then # key= case - key="$prev" - glob_pos=$((cword - 2)) - elif [[ $cur == *=* ]] ; then # key=value case (OSX) - key=${cur%=*} - glob_pos=$((cword - 1)) - elif [ "$prev" = "=" ] ; then - key=${words[$cword - 2]} # key=value case - glob_pos=$((cword - 3)) - else - return - fi - - [ "${words[$glob_pos]}" = "=" ] && ((glob_pos--)) # --option=key=value syntax - - [[ ${words[$glob_pos]} == @($glob) ]] && echo "$key" -} - -# suppress trailing whitespace -__docker_compose_nospace() { - # compopt is not available in ancient bash versions - type compopt &>/dev/null && compopt -o nospace -} - -# Extracts all service names from the compose file. -___docker_compose_all_services_in_compose_file() { - __docker_compose_q config --services -} - -# All services, even those without an existing container -__docker_compose_services_all() { - COMPREPLY=( $(compgen -W "$(___docker_compose_all_services_in_compose_file)" -- "$cur") ) -} - -# All services that are defined by a Dockerfile reference -__docker_compose_services_from_build() { - COMPREPLY=( $(compgen -W "$(__docker_compose_q ps --services --filter "source=build")" -- "$cur") ) -} - -# All services that are defined by an image -__docker_compose_services_from_image() { - COMPREPLY=( $(compgen -W "$(__docker_compose_q ps --services --filter "source=image")" -- "$cur") ) -} - -# The services for which at least one paused container exists -__docker_compose_services_paused() { - names=$(__docker_compose_q ps --services --filter "status=paused") - COMPREPLY=( $(compgen -W "$names" -- "$cur") ) -} - -# The services for which at least one running container exists -__docker_compose_services_running() { - names=$(__docker_compose_q ps --services --filter "status=running") - COMPREPLY=( $(compgen -W "$names" -- "$cur") ) -} - -# The services for which at least one stopped container exists -__docker_compose_services_stopped() { - names=$(__docker_compose_q ps --services --filter "status=stopped") - COMPREPLY=( $(compgen -W "$names" -- "$cur") ) -} - - -_docker_compose_build() { - case "$prev" in - --build-arg) - COMPREPLY=( $( compgen -e -- "$cur" ) ) - __docker_compose_nospace - return - ;; - esac - - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--build-arg --force-rm --help --memory --no-cache --pull" -- "$cur" ) ) - ;; - *) - __docker_compose_services_from_build - ;; - esac -} - - -_docker_compose_bundle() { - case "$prev" in - --output|-o) - _filedir - return - ;; - esac - - COMPREPLY=( $( compgen -W "--push-images --help --output -o" -- "$cur" ) ) -} - - -_docker_compose_config() { - COMPREPLY=( $( compgen -W "--help --quiet -q --resolve-image-digests --services --volumes" -- "$cur" ) ) -} - - -_docker_compose_create() { - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--build --force-recreate --help --no-build --no-recreate" -- "$cur" ) ) - ;; - *) - __docker_compose_services_all - ;; - esac -} - - -_docker_compose_docker_compose() { - case "$prev" in - --tlscacert|--tlscert|--tlskey) - _filedir - return - ;; - --file|-f) - _filedir "y?(a)ml" - return - ;; - --project-directory) - _filedir -d - return - ;; - $(__docker_compose_to_extglob "$top_level_options_with_args") ) - return - ;; - esac - - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "$top_level_boolean_options $top_level_options_with_args --help -h --no-ansi --verbose --version -v" -- "$cur" ) ) - ;; - *) - COMPREPLY=( $( compgen -W "${commands[*]}" -- "$cur" ) ) - ;; - esac -} - - -_docker_compose_down() { - case "$prev" in - --rmi) - COMPREPLY=( $( compgen -W "all local" -- "$cur" ) ) - return - ;; - --timeout|-t) - return - ;; - esac - - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help --rmi --timeout -t --volumes -v --remove-orphans" -- "$cur" ) ) - ;; - esac -} - - -_docker_compose_events() { - case "$prev" in - --json) - return - ;; - esac - - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help --json" -- "$cur" ) ) - ;; - *) - __docker_compose_services_all - ;; - esac -} - - -_docker_compose_exec() { - case "$prev" in - --index|--user|-u) - return - ;; - esac - - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "-d --help --index --privileged -T --user -u" -- "$cur" ) ) - ;; - *) - __docker_compose_services_running - ;; - esac -} - - -_docker_compose_help() { - COMPREPLY=( $( compgen -W "${commands[*]}" -- "$cur" ) ) -} - -_docker_compose_images() { - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help -q" -- "$cur" ) ) - ;; - *) - __docker_compose_services_all - ;; - esac -} - -_docker_compose_kill() { - case "$prev" in - -s) - COMPREPLY=( $( compgen -W "SIGHUP SIGINT SIGKILL SIGUSR1 SIGUSR2" -- "$(echo $cur | tr '[:lower:]' '[:upper:]')" ) ) - return - ;; - esac - - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help -s" -- "$cur" ) ) - ;; - *) - __docker_compose_services_running - ;; - esac -} - - -_docker_compose_logs() { - case "$prev" in - --tail) - return - ;; - esac - - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--follow -f --help --no-color --tail --timestamps -t" -- "$cur" ) ) - ;; - *) - __docker_compose_services_all - ;; - esac -} - - -_docker_compose_pause() { - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help" -- "$cur" ) ) - ;; - *) - __docker_compose_services_running - ;; - esac -} - - -_docker_compose_port() { - case "$prev" in - --protocol) - COMPREPLY=( $( compgen -W "tcp udp" -- "$cur" ) ) - return; - ;; - --index) - return; - ;; - esac - - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help --index --protocol" -- "$cur" ) ) - ;; - *) - __docker_compose_services_all - ;; - esac -} - - -_docker_compose_ps() { - local key=$(__docker_compose_map_key_of_current_option '--filter') - case "$key" in - source) - COMPREPLY=( $( compgen -W "build image" -- "${cur##*=}" ) ) - return - ;; - status) - COMPREPLY=( $( compgen -W "paused restarting running stopped" -- "${cur##*=}" ) ) - return - ;; - esac - - case "$prev" in - --filter) - COMPREPLY=( $( compgen -W "source status" -S "=" -- "$cur" ) ) - __docker_compose_nospace - return; - ;; - esac - - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help -q --services --filter" -- "$cur" ) ) - ;; - *) - __docker_compose_services_all - ;; - esac -} - - -_docker_compose_pull() { - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help --ignore-pull-failures --parallel --quiet" -- "$cur" ) ) - ;; - *) - __docker_compose_services_from_image - ;; - esac -} - - -_docker_compose_push() { - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help --ignore-push-failures" -- "$cur" ) ) - ;; - *) - __docker_compose_services_all - ;; - esac -} - - -_docker_compose_restart() { - case "$prev" in - --timeout|-t) - return - ;; - esac - - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help --timeout -t" -- "$cur" ) ) - ;; - *) - __docker_compose_services_running - ;; - esac -} - - -_docker_compose_rm() { - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--force -f --help --stop -s -v" -- "$cur" ) ) - ;; - *) - if __docker_compose_has_option "--stop|-s" ; then - __docker_compose_services_all - else - __docker_compose_services_stopped - fi - ;; - esac -} - - -_docker_compose_run() { - case "$prev" in - -e) - COMPREPLY=( $( compgen -e -- "$cur" ) ) - __docker_compose_nospace - return - ;; - --entrypoint|--label|-l|--name|--user|-u|--volume|-v|--workdir|-w) - return - ;; - esac - - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "-d --entrypoint -e --help --label -l --name --no-deps --publish -p --rm --service-ports -T --user -u --volume -v --workdir -w" -- "$cur" ) ) - ;; - *) - __docker_compose_services_all - ;; - esac -} - - -_docker_compose_scale() { - case "$prev" in - =) - COMPREPLY=("$cur") - return - ;; - --timeout|-t) - return - ;; - esac - - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help --timeout -t" -- "$cur" ) ) - ;; - *) - COMPREPLY=( $(compgen -S "=" -W "$(___docker_compose_all_services_in_compose_file)" -- "$cur") ) - __docker_compose_nospace - ;; - esac -} - - -_docker_compose_start() { - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help" -- "$cur" ) ) - ;; - *) - __docker_compose_services_stopped - ;; - esac -} - - -_docker_compose_stop() { - case "$prev" in - --timeout|-t) - return - ;; - esac - - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help --timeout -t" -- "$cur" ) ) - ;; - *) - __docker_compose_services_running - ;; - esac -} - - -_docker_compose_top() { - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help" -- "$cur" ) ) - ;; - *) - __docker_compose_services_running - ;; - esac -} - - -_docker_compose_unpause() { - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--help" -- "$cur" ) ) - ;; - *) - __docker_compose_services_paused - ;; - esac -} - - -_docker_compose_up() { - case "$prev" in - =) - COMPREPLY=("$cur") - return - ;; - --exit-code-from) - __docker_compose_services_all - return - ;; - --scale) - COMPREPLY=( $(compgen -S "=" -W "$(___docker_compose_all_services_in_compose_file)" -- "$cur") ) - __docker_compose_nospace - return - ;; - --timeout|-t) - return - ;; - esac - - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--abort-on-container-exit --always-recreate-deps --build -d --exit-code-from --force-recreate --help --no-build --no-color --no-deps --no-recreate --no-start --renew-anon-volumes -V --remove-orphans --scale --timeout -t" -- "$cur" ) ) - ;; - *) - __docker_compose_services_all - ;; - esac -} - - -_docker_compose_version() { - case "$cur" in - -*) - COMPREPLY=( $( compgen -W "--short" -- "$cur" ) ) - ;; - esac -} - - -_docker_compose() { - local previous_extglob_setting=$(shopt -p extglob) - shopt -s extglob - - local commands=( - build - bundle - config - create - down - events - exec - help - images - kill - logs - pause - port - ps - pull - push - restart - rm - run - scale - start - stop - top - unpause - up - version - ) - - # Options for the docker daemon that have to be passed to secondary calls to - # docker-compose executed by this script. - # Other global otions that are not relevant for secondary calls are defined in - # `_docker_compose_docker_compose`. - local top_level_boolean_options=" - --skip-hostname-check - --tls - --tlsverify - " - local top_level_options_with_args=" - --file -f - --host -H - --project-directory - --project-name -p - --tlscacert - --tlscert - --tlskey - " - - COMPREPLY=() - local cur prev words cword - _get_comp_words_by_ref -n : cur prev words cword - - # search subcommand and invoke its handler. - # special treatment of some top-level options - local command='docker_compose' - local top_level_options=() - local counter=1 - - while [ $counter -lt $cword ]; do - case "${words[$counter]}" in - $(__docker_compose_to_extglob "$top_level_boolean_options") ) - local opt=${words[counter]} - top_level_options+=($opt) - ;; - $(__docker_compose_to_extglob "$top_level_options_with_args") ) - local opt=${words[counter]} - local arg=${words[++counter]} - top_level_options+=($opt $arg) - ;; - -*) - ;; - *) - command="${words[$counter]}" - break - ;; - esac - (( counter++ )) - done - - local completions_func=_docker_compose_${command//-/_} - declare -F $completions_func >/dev/null && $completions_func - - eval "$previous_extglob_setting" - return 0 -} - -complete -F _docker_compose docker-compose docker-compose.exe diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 606b7c09cc4b016b6fdab3da6fc78e403f8d80f4..83ad13c98f8e578a6486512da7879bf777f7abfc 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -14,6 +14,6 @@ repos: sha: v0.6.0 hooks: - id: shell-lint - exclude: &excluded_shellscripts \.bash_completion\.d/(docker_machine\.bash|fabric-completion.bash|docker_compose.bash)|\.travis/travis\.sh + exclude: &excluded_shellscripts \.bash_completion\.d/(docker-machine\.bash|fabric-completion.bash|docker_compose.bash)|\.travis/travis\.sh - id: shellcheck exclude: *excluded_shellscripts diff --git a/Documents/bin/gen-bash-completion b/Documents/bin/gen-bash-completion index 85ad36408b961a1508f0d396d4b09553a50c2a7e..83b18521eae51f5502e2709683fe4daa43132aa6 100755 --- a/Documents/bin/gen-bash-completion +++ b/Documents/bin/gen-bash-completion @@ -1,5 +1,13 @@ #!/bin/sh set -eu -! command -v pandoc > /dev/null || pandoc --bash-completion > "$HOME/.bash_completion.d/pandoc" -! command -v pipenv > /dev/null || pipenv --completion > "$HOME/.bash_completion.d/pipenv" +gen_completion () { + executable="$1" + ! command -v "$executable" > /dev/null || "$@" > "$HOME/.bash_completion.d/$executable" +} + +gen_completion pandoc --bash-completion +gen_completion pipenv --completion +gen_completion kops completion bash +gen_completion kubectl completion bash +gen_completion minikube completion bash