#!/bin/bash # # Bash completion generated for '{{name}}' at {{date}}. # # The original template lives here: # https://github.com/trentm/node-dashdash/blob/master/etc/dashdash.bash_completion.in # # # Copyright 2016 Trent Mick # Copyright 2016 Joyent, Inc. # # # A generic Bash completion driver script. # # This is meant to provide a re-usable chunk of Bash to use for # "etc/bash_completion.d/" files for individual tools. Only the "Configuration" # section with tool-specific info need differ. Features: # # - support for short and long opts # - support for knowing which options take arguments # - support for subcommands (e.g. 'git log <TAB>' to show just options for the # log subcommand) # - does the right thing with "--" to stop options # - custom optarg and arg types for custom completions # - (TODO) support for shells other than Bash (tcsh, zsh, fish?, etc.) # # # Examples/design: # # 1. Bash "default" completion. By default Bash's 'complete -o default' is # enabled. That means when there are no completions (e.g. if no opts match # the current word), then you'll get Bash's default completion. Most notably # that means you get filename completion. E.g.: # $ tool ./<TAB> # $ tool READ<TAB> # # 2. all opts and subcmds: # $ tool <TAB> # $ tool -v <TAB> # assuming '-v' doesn't take an arg # $ tool -<TAB> # matching opts # $ git lo<TAB> # matching subcmds # # Long opt completions are given *without* the '=', i.e. we prefer space # separated because that's easier for good completions. # # 3. long opt arg with '=' # $ tool --file=<TAB> # $ tool --file=./d<TAB> # We maintain the "--file=" prefix. Limitation: With the attached prefix # the 'complete -o filenames' doesn't know to do dirname '/' suffixing. Meh. # # 4. envvars: # $ tool $<TAB> # $ tool $P<TAB> # Limitation: Currently only getting exported vars, so we miss "PS1" and # others. # # 5. Defer to other completion in a subshell: # $ tool --file $(cat ./<TAB> # We get this from 'complete -o default ...'. # # 6. Custom completion types from a provided bash function. # $ tool --profile <TAB> # complete available "profiles" # # # Dev Notes: # - compgen notes, from http://unix.stackexchange.com/questions/151118/understand-compgen-builtin-command # - https://www.gnu.org/software/bash/manual/html_node/Programmable-Completion-Builtins.html # # Debugging this completion: # 1. Uncomment the "_{{name}}_log_file=..." line. # 2. 'tail -f /var/tmp/dashdash-completion.log' in one terminal. # 3. Re-source this bash completion file. #_{{name}}_log=/var/tmp/dashdash-completion.log function _{{name}}_completer { # ---- cmd definition {{spec}} # ---- locals declare -a argv # ---- support functions function trace { [[ -n "$_{{name}}_log" ]] && echo "$*" >&2 } function _dashdash_complete { local idx context idx=$1 context=$2 local shortopts longopts optargs subcmds allsubcmds argtypes shortopts="$(eval "echo \${cmd${context}_shortopts}")" longopts="$(eval "echo \${cmd${context}_longopts}")" optargs="$(eval "echo \${cmd${context}_optargs}")" subcmds="$(eval "echo \${cmd${context}_subcmds}")" allsubcmds="$(eval "echo \${cmd${context}_allsubcmds}")" IFS=', ' read -r -a argtypes <<< "$(eval "echo \${cmd${context}_argtypes}")" trace "" trace "_dashdash_complete(idx=$idx, context=$context)" trace " shortopts: $shortopts" trace " longopts: $longopts" trace " optargs: $optargs" trace " subcmds: $subcmds" trace " allsubcmds: $allsubcmds" # Get 'state' of option parsing at this COMP_POINT. # Copying "dashdash.js#parse()" behaviour here. local state= local nargs=0 local i=$idx local argtype local optname local prefix local word local dashdashseen= while [[ $i -lt $len && $i -le $COMP_CWORD ]]; do argtype= optname= prefix= word= arg=${argv[$i]} trace " consider argv[$i]: '$arg'" if [[ "$arg" == "--" && $i -lt $COMP_CWORD ]]; then trace " dashdash seen" dashdashseen=yes state=arg word=$arg elif [[ -z "$dashdashseen" && "${arg:0:2}" == "--" ]]; then arg=${arg:2} if [[ "$arg" == *"="* ]]; then optname=${arg%%=*} val=${arg##*=} trace " long opt: optname='$optname' val='$val'" state=arg argtype=$(echo "$optargs" | awk -F "-$optname=" '{print $2}' | cut -d' ' -f1) word=$val prefix="--$optname=" else optname=$arg val= trace " long opt: optname='$optname'" state=longopt word=--$optname if [[ "$optargs" == *"-$optname="* && $i -lt $COMP_CWORD ]]; then i=$(( $i + 1 )) state=arg argtype=$(echo "$optargs" | awk -F "-$optname=" '{print $2}' | cut -d' ' -f1) word=${argv[$i]} trace " takes arg (consume argv[$i], word='$word')" fi fi elif [[ -z "$dashdashseen" && "${arg:0:1}" == "-" ]]; then trace " short opt group" state=shortopt word=$arg local j=1 while [[ $j -lt ${#arg} ]]; do optname=${arg:$j:1} trace " consider index $j: optname '$optname'" if [[ "$optargs" == *"-$optname="* ]]; then argtype=$(echo "$optargs" | awk -F "-$optname=" '{print $2}' | cut -d' ' -f1) if [[ $(( $j + 1 )) -lt ${#arg} ]]; then state=arg word=${arg:$(( $j + 1 ))} trace " takes arg (rest of this arg, word='$word', argtype='$argtype')" elif [[ $i -lt $COMP_CWORD ]]; then state=arg i=$(( $i + 1 )) word=${argv[$i]} trace " takes arg (word='$word', argtype='$argtype')" fi break fi j=$(( $j + 1 )) done elif [[ $i -lt $COMP_CWORD && -n "$arg" ]] && $(echo "$allsubcmds" | grep -w "$arg" >/dev/null); then trace " complete subcmd: recurse _dashdash_complete" _dashdash_complete $(( $i + 1 )) "${context}__${arg/-/_}" return else trace " not an opt or a complete subcmd" state=arg word=$arg nargs=$(( $nargs + 1 )) if [[ ${#argtypes[@]} -gt 0 ]]; then argtype="${argtypes[$(( $nargs - 1 ))]}" if [[ -z "$argtype" ]]; then # If we have more args than argtypes, we use the # last type. argtype="${argtypes[@]: -1:1}" fi fi fi trace " state=$state prefix='$prefix' word='$word'" i=$(( $i + 1 )) done trace " parsed: state=$state optname='$optname' argtype='$argtype' prefix='$prefix' word='$word' dashdashseen=$dashdashseen" local compgen_opts= if [[ -n "$prefix" ]]; then compgen_opts="$compgen_opts -P $prefix" fi case $state in shortopt) compgen $compgen_opts -W "$shortopts $longopts" -- "$word" ;; longopt) compgen $compgen_opts -W "$longopts" -- "$word" ;; arg) # If we don't know what completion to do, then emit nothing. We # expect that we are running with: # complete -o default ... # where "default" means: "Use Readline's default completion if # the compspec generates no matches." This gives us the good filename # completion, completion in subshells/backticks. # # We cannot support an argtype="directory" because # compgen -S '/' -A directory -- "$word" # doesn't give a satisfying result. It doesn't stop at the trailing '/' # so you cannot descend into dirs. if [[ "${word:0:1}" == '$' ]]; then # By default, Bash will complete '$<TAB>' to all envvars. Apparently # 'complete -o default' does *not* give us that. The following # gets *close* to the same completions: '-A export' misses envvars # like "PS1". trace " completing envvars" compgen $compgen_opts -P '$' -A export -- "${word:1}" elif [[ -z "$argtype" ]]; then # Only include opts in completions if $word is not empty. # This is to avoid completing the leading '-', which foils # using 'default' completion. if [[ -n "$dashdashseen" ]]; then trace " completing subcmds, if any (no argtype, dashdash seen)" compgen $compgen_opts -W "$subcmds" -- "$word" elif [[ -z "$word" ]]; then trace " completing subcmds, if any (no argtype, empty word)" compgen $compgen_opts -W "$subcmds" -- "$word" else trace " completing opts & subcmds (no argtype)" compgen $compgen_opts -W "$shortopts $longopts $subcmds" -- "$word" fi elif [[ $argtype == "none" ]]; then # We want *no* completions, i.e. some way to get the active # 'complete -o default' to not do filename completion. trace " completing 'none' (hack to imply no completions)" echo "##-no-completion- -results-##" elif [[ $argtype == "file" ]]; then # 'complete -o default' gives the best filename completion, at least # on Mac. trace " completing 'file' (let 'complete -o default' handle it)" echo "" elif ! type complete_$argtype 2>/dev/null >/dev/null; then trace " completing '$argtype' (fallback to default b/c complete_$argtype is unknown)" echo "" else trace " completing custom '$argtype'" completions=$(complete_$argtype "$word") if [[ -z "$completions" ]]; then trace " no custom '$argtype' completions" # These are in ascii and "dictionary" order so they sort # correctly. echo "##-no-completion- -results-##" else echo $completions fi fi ;; *) trace " unknown state: $state" ;; esac } trace "" trace "-- $(date)" #trace "\$IFS: '$IFS'" #trace "\$@: '$@'" #trace "COMP_WORDBREAKS: '$COMP_WORDBREAKS'" trace "COMP_CWORD: '$COMP_CWORD'" trace "COMP_LINE: '$COMP_LINE'" trace "COMP_POINT: $COMP_POINT" # Guard against negative COMP_CWORD. This is a Bash bug at least on # Mac 10.10.4's bash. See # <https://lists.gnu.org/archive/html/bug-bash/2009-07/msg00125.html>. if [[ $COMP_CWORD -lt 0 ]]; then trace "abort on negative COMP_CWORD" exit 1; fi # I don't know how to do array manip on argv vars, # so copy over to argv array to work on them. shift # the leading '--' i=0 len=$# while [[ $# -gt 0 ]]; do argv[$i]=$1 shift; i=$(( $i + 1 )) done trace "argv: '${argv[@]}'" trace "argv[COMP_CWORD-1]: '${argv[$(( $COMP_CWORD - 1 ))]}'" trace "argv[COMP_CWORD]: '${argv[$COMP_CWORD]}'" trace "argv len: '$len'" _dashdash_complete 1 "" } # ---- mainline # Note: This if-block to help work with 'compdef' and 'compctl' is # adapted from 'npm completion'. if type complete &>/dev/null; then function _{{name}}_completion { local _log_file=/dev/null [[ -z "$_{{name}}_log" ]] || _log_file="$_{{name}}_log" COMPREPLY=($(COMP_CWORD="$COMP_CWORD" \ COMP_LINE="$COMP_LINE" \ COMP_POINT="$COMP_POINT" \ _{{name}}_completer -- "${COMP_WORDS[@]}" \ 2>$_log_file)) || return $? } complete -o default -F _{{name}}_completion {{name}} elif type compdef &>/dev/null; then function _{{name}}_completion { local _log_file=/dev/null [[ -z "$_{{name}}_log" ]] || _log_file="$_{{name}}_log" compadd -- $(COMP_CWORD=$((CURRENT-1)) \ COMP_LINE=$BUFFER \ COMP_POINT=0 \ _{{name}}_completer -- "${words[@]}" \ 2>$_log_file) } compdef _{{name}}_completion {{name}} elif type compctl &>/dev/null; then function _{{name}}_completion { local cword line point words si read -Ac words read -cn cword let cword-=1 read -l line read -ln point local _log_file=/dev/null [[ -z "$_{{name}}_log" ]] || _log_file="$_{{name}}_log" reply=($(COMP_CWORD="$cword" \ COMP_LINE="$line" \ COMP_POINT="$point" \ _{{name}}_completer -- "${words[@]}" \ 2>$_log_file)) || return $? } compctl -K _{{name}}_completion {{name}} fi ## ## This is a Bash completion file for the '{{name}}' command. You can install ## with either: ## ## cp FILE /usr/local/etc/bash_completion.d/{{name}} # Mac ## cp FILE /etc/bash_completion.d/{{name}} # Linux ## ## or: ## ## cp FILE > ~/.{{name}}.completion ## echo "source ~/.{{name}}.completion" >> ~/.bashrc ##