# vyatta bash configuration mode completion # **** License **** # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License version 2 as # published by the Free Software Foundation. # # This program is distributed in the hope that it will be useful, but # WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU # General Public License for more details. # # A copy of the GNU General Public License is available as # `/usr/share/common-licenses/GPL' in the Debian GNU/Linux distribution # or on the World Wide Web at `http://www.gnu.org/copyleft/gpl.html'. # You can also obtain it by writing to the Free Software Foundation, # Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, # MA 02110-1301, USA. # # Author: Vyatta # Description: bash completion for Vyatta configuration commands # # **** End License **** # Turn on history logging export HISTCONTROL= export HISTFILESIZE=10000 export HISTSIZE=10000 export HISTTIMEFORMAT='%FT%T%z ' export PROMPT_COMMAND='history -a' # remove colon from completion word seperators export COMP_WORDBREAKS=${COMP_WORDBREAKS/:/} builtin set histappend=1 # only do this if we are going into configure mode if [ "$_OFR_CONFIGURE" != "ok" ]; then return 0 fi if [ -r /etc/default/vyatta ]; then source /etc/default/vyatta fi declare -a op_functions op_functions=( /opt/vyatta/share/vyatta-op/functions/interpreter/* ) for file in "${op_functions[@]}"; do source $file done declare -a cfg_functions cfg_functions=( /opt/vyatta/share/vyatta-cfg/functions/interpreter/* ) for file in "${cfg_functions[@]}"; do source $file done # function for shell api vyatta_cli_shell_api () { local noeval='' if [ "$1" == NOEVAL ]; then noeval=true shift fi local outstr if ! outstr=$(${vyatta_sbindir}/my_cli_shell_api -- "$@"); then # display the error output (if any) and then fail if [ -n "$outstr" ]; then echo "$outstr" fi return 1 fi # eval the output (if any) if [ -n "$outstr" ]; then if [ -n "$noeval" ]; then echo "$outstr" else eval "$outstr" fi fi return 0 } # set up the session environment ## note: this can not use vyatta_cli_shell_api() above since it "declares" ## env vars. eval "$(${vyatta_sbindir}/my_cli_shell_api getSessionEnv $$)" declare is_set=0 declare last_idx=0 declare -a comp_words=() # commands to unalias declare -a unalias_cmds=( clear configure date debug edit exit load merge \ no run set show save terminal undebug up top ) for cmd in "${unalias_cmds[@]}"; do unalias $cmd >& /dev/null done ### Top level command completions ### # do op mode completion vyatta_run_complete () { local restore_shopts=$( shopt -p extglob nullglob | tr \\n \; ) shopt -s extglob nullglob COMP_WORDS=( "${COMP_WORDS[@]:1}" ) (( COMP_CWORD -= 1 )) _vyatta_op_expand eval $restore_shopts } vyatta_loadsave_complete() { # Generate completion help for the "load" and "save" commands local restore_shopts=$( shopt -p extglob nullglob | tr \\n \; ) shopt -s extglob nullglob # Only provide completions after command name has been typed, but # before any characters of the command argument have been entered. # File name completion, and completion of the various URL formats # is not supported yet. # if [ $COMP_CWORD -eq 1 -a -z "${COMP_WORDS[1]}" ]; then local command=$(vyatta_cfg_expand_top_level ${COMP_WORDS[0]}) echo echo "Possible completions:" if [ "$command" = "load" ]; then echo -e " \t\t\t\tLoad from system config file" echo -e " \t\t\t\tLoad from file on local machine" echo -e " scp://:@/\tLoad from file on remote machine" echo -e " ftp://:@/\tLoad from file on remote machine" echo -e " http:///\t\t\tLoad from file on remote machine" echo -e " tftp:///\t\t\tLoad from file on remote machine" elif [ "$command" = "merge" ]; then echo -e " \t\t\t\tMerge from system config file" echo -e " \t\t\t\tMerge from file on local machine" echo -e " scp://:@/\tMerge from file on remote machine" echo -e " ftp://:@/\tMerge from file on remote machine" echo -e " http:///\t\t\tMerge from file on remote machine" echo -e " tftp:///\t\t\tMerge from file on remote machine" elif [ "$command" = "save" ]; then echo -e " \t\t\t\tSave to system config file" echo -e " \t\t\t\tSave to file on local machine" echo -e " scp://:@/\tSave to file on remote machine" echo -e " ftp://:@/\tSave to file on remote machine" echo -e " tftp:///\t\t\tSave to file on remote machine" fi COMPREPLY=( "" " " ) else COMPREPLY=() fi eval $restore_shopts } vyatta_config_loadkey() { # don't load if there are uncommitted changes. if vyatta_cli_shell_api sessionChanged; then echo "Cannot load: configuration modified." echo "Commit or discard the changes before loading a config file." return 1 fi # return to top level. reset_edit_level ${vyatta_sbindir}/vyatta-load-user-key.pl "$@" } vyatta_loadkey_complete() { case "$COMP_CWORD" in 1) if [ -z "${COMP_WORDS[1]}" ]; then COMPREPLY=( $(getent passwd | awk -F: '$7 == "/bin/vbash" { print $1}') ) else COMPREPLY=( $(compgen -u -- ${COMP_WORDS[1]} ) ) fi ;; 2) if [ -z "${COMP_WORDS[2]}" ]; then echo echo "Possible completions:" echo -e " \t\t\t\tLoad from file on local machine" echo -e " scp://@/\tLoad from file on remote machine" echo -e " ftp://@/\tLoad from file on remote machine" echo -e " http:///\t\t\tLoad from file on remote machine" echo -e " tftp:///\t\t\tLoad from file on remote machine" COMPREPLY=() else COMPREPLY=( $(compgen -f -- ${COMP_WORDS[2]} ) ) fi ;; esac } print_commit_log () { local -a array eval "array=($(${vyatta_sbindir}/vyatta-config-mgmt.pl --action=show-commit-log-brief))" local count=0 for i in "${array[@]}"; do i=${i//_/ } echo -e " $count\t$i" (( count++ )) done } vyatta_rollback_complete () { # Generate completion help for the "rollback" command local restore_shopts=$( shopt -p extglob nullglob | tr \\n \; ) shopt -s extglob nullglob # Only provide completions after command name has been typed, but # before any characters of the command argument have been entered. if [ $COMP_CWORD -eq 1 -a -z "${COMP_WORDS[1]}" ]; then echo echo "Possible completions:" echo -e " \tRollback to revision N (currently requires reboot)" echo -e "\n Revisions:" print_commit_log COMPREPLY=( "" " " ) else COMPREPLY=() fi eval $restore_shopts } vyatta_compare_complete () { # Generate completion help for the "compare" command local restore_shopts=$( shopt -p extglob nullglob | tr \\n \; ) shopt -s extglob nullglob # Only provide completions after command name has been typed, but # before any characters of the command argument have been entered. if [ $COMP_CWORD -eq 1 -a -z "${COMP_WORDS[1]}" ]; then echo echo "Possible completions:" echo -e " \tCompare working & active configurations" echo -e " saved\t\tCompare working & saved configurations" echo -e " \t\tCompare working with revision N" echo -e " \tCompare revision N with M" echo -e "\n Revisions:" print_commit_log COMPREPLY=( "" " " ) elif [ $COMP_CWORD -eq 2 -a -z "${COMP_WORDS[2]}" ]; then if [[ "${COMP_WORDS[1]}" == "saved" ]]; then echo "Possible completions:" echo -e " \tCompare working and saved configurations" COMPREPLY=( "" " " ) eval $restore_shopts return fi echo echo "Possible completions:" echo -e " \tCompare working revision N" echo -e " \t\tCompare revision N with M" echo -e "\n Revisions:" print_commit_log COMPREPLY=( "" " " ) elif [ $COMP_CWORD -eq 1 ]; then COMPREPLY=$(compgen -W saved -- ${COMP_WORDS[$COMP_CWORD]}) else COMPREPLY=() fi eval $restore_shopts } vyatta_commit_complete () { # Generate completion help for the "commit-confirm" command local restore_shopts=$( shopt -p extglob nullglob | tr \\n \; ) shopt -s extglob nullglob # Only provide completions after command name has been typed, but # before any characters of the command argument have been entered. if [ $COMP_CWORD -eq 1 -a -z "${COMP_WORDS[1]}" ]; then echo echo "Possible completions:" if [ "${COMP_WORDS[0]}" = "commit" ]; then echo -e " \tCommit working configuration" elif [ "${COMP_WORDS[0]}" = "commit-confirm" ]; then echo -e " \tCommit, rollback/reboot in 10 minutes if no confirm" echo -e " \t\tCommit, rollback/reboot in N minutes if no confirm" fi echo -e " comment\tComment for commit log" COMPREPLY=( "" " " ) elif [ "${COMP_WORDS[1]}" = "comment" ]; then echo echo "Possible completions:" echo -e " \tText comment for commit log (e.g. \"add user bob\")" COMPREPLY=( "" " " ) elif [ $COMP_CWORD -eq 1 ]; then COMPREPLY=$(compgen -W comment -- ${COMP_WORDS[$COMP_CWORD]}) else COMPREPLY=() fi eval $restore_shopts } ### End Top level command completions ### declare vyatta_cfg_help="" declare vyatta_cfg_type="" declare vyatta_cfg_tag=0 declare vyatta_cfg_multi=0 declare -a vyatta_cfg_allowed=() declare vyatta_cfg_comp_help="" declare -a vyatta_cfg_val_type=() declare -a vyatta_cfg_val_help=() declare -a _get_help_text_items=() declare -a _get_help_text_helps=() get_help_text () { vyatta_help_text="\\nPossible completions:" for (( idx = 0; idx < ${#_get_help_text_items[@]}; idx++ )); do vyatta_help_text+="\\n\\x20\\x20" if [ ${#_get_help_text_items[idx]} -lt 6 ]; then vyatta_help_text+="${_get_help_text_items[idx]}\\t\\t" elif [ ${#_get_help_text_items[idx]} -lt 14 ]; then vyatta_help_text+="${_get_help_text_items[idx]}\\t" else vyatta_help_text+="${_get_help_text_items[idx]}\\n\\x20\\x20\\t\\t" fi vyatta_help_text+="${_get_help_text_helps[idx]}" done if [ -n "$vyatta_cfg_comp_help" ]; then local hstr=${vyatta_cfg_comp_help//\'/\'\\\\\\\'\'} vyatta_help_text+="\\n\\nDetailed information:\\n" local sIFS=$IFS IFS=' ' local chstr=$(echo -en "$hstr\n" \ | while read comp_help_line; do echo "vyatta_help_text+=' $comp_help_line\\n';" done) eval "$chstr" IFS=$sIFS fi } get_value_format_string () { local vtype=$1 if [[ $vtype = !* ]]; then echo -n '!' vtype="${vtype#!}" fi case "$vtype" in _*) echo -n "${vtype#_}" ;; txt) echo -n '' ;; u32) echo -n '<0-4294967295>' ;; u32:*) echo -n "<${vtype##u32:}>" ;; range) echo -n "-" ;; ipv4) echo -n '' ;; ipv6) echo -n '' ;; ipv4net) echo -n '' ;; ipv6net) echo -n '' ;; ipv4range) echo -n '-' ;; ipv6range) echo -n '-' ;; bool) echo -n '' ;; macaddr) echo -n '' ;; *) echo -n "$vtype" ;; esac } declare -a vyatta_completions declare vyatta_help_text="\\nNo help text available" declare vyatta_do_help=false vyatta_do_complete () { if $vyatta_do_help; then printf "$vyatta_help_text" COMPREPLY=( "" " " ) else local -a f_comps=() local cword= if (( ${#COMP_WORDS[@]} > 0 )); then cword=${COMP_WORDS[COMP_CWORD]} fi get_prefix_filtered_list "$cword" vyatta_completions f_comps local estr="COMPREPLY=( " for w in "${f_comps[@]}"; do estr="$estr\"$w\" " done estr="${estr})" eval "$estr" fi vyatta_help_text="\\nNo help text available" } vyatta_simple_complete () { # when this function is called, it is expected that: # * "vyatta_help_text" is filled with the help text. # * "vyatta_completions" is an array of "filtered" possible completions # (i.e., only those starting with the current last component). if $vyatta_do_help; then printf "$vyatta_help_text" COMPREPLY=( "" " " ) else COMPREPLY=( "${vyatta_completions[@]}" ) fi vyatta_help_text="\\nNo help text available" } generate_pipe_help () { _get_help_text_items=( "${_vyatta_pipe_completions[@]}" \ "${_vyatta_pipe_noncompletions[@]}" ) _get_help_text_helps=() for comp in "${_get_help_text_items[@]}"; do _get_help_text_helps+=("$(_vyatta_pipe_help "$comp")") done get_help_text } ### Expand config compwords ### vyatta_config_expand_compwords () { local cmd=$1 cmd=$(vyatta_cfg_expand_top_level $cmd) expanded_api_args=( "$cmd" ) local _cli_shell_api_last_comp_val='' local _cli_shell_api_comp_help='' local -a _cli_shell_api_comp_values=() local -a _cli_shell_api_hitems=() local -a _cli_shell_api_hstrs=() local -a path=( "$cmd" ) for arg in "${@:2}"; do _cli_shell_api_comp_values=() _cli_shell_api_last_comp_val='' if [[ -z $arg ]]; then path=( "${expanded_api_args[@]}" '' ) else path=( "${expanded_api_args[@]}" "$arg" ) fi vyatta_cli_shell_api getCompletionEnv "${path[@]}" if [[ "${#_cli_shell_api_comp_values[@]}" == "1" ]]; then if [[ "$_cli_shell_api_last_comp_val" == 'true' ]]; then arg=$arg else arg=${_cli_shell_api_comp_values[0]} fi expanded_api_args+=( "$arg" ) else expanded_api_args+=( "$arg" ) fi done } # env variables for shell api completion declare _cli_shell_api_last_comp_val='' declare _cli_shell_api_comp_help='' declare -a _cli_shell_api_comp_values=() declare -a _cli_shell_api_hitems=() declare -a _cli_shell_api_hstrs=() vyatta_config_complete () { local restore_shopts=$( shopt -p extglob nullglob | tr \\n \; ) shopt -s extglob nullglob if [ "$COMP_LINE" == "$VYATTA_COMP_LINE" ]; then VYATTA_COMP_LINE=$VYATTA_COMP_LINE_EMPTY vyatta_do_help=true else VYATTA_COMP_LINE=$COMP_LINE vyatta_do_help=false fi # handle pipe if _vyatta_pipe_completion "${COMP_WORDS[@]}"; then generate_pipe_help vyatta_completions=( "${_vyatta_pipe_completions[@]}" ) vyatta_do_complete eval $restore_shopts return fi if (( ${#COMP_WORDS[@]} < 2 )); then _get_help_text_items=( "${_vyatta_cfg_cmds[@]}" ) _get_help_text_helps=( "${_vyatta_cfg_helps[@]}" ) if (( ${#COMP_WORDS[@]} == 1 )); then declare -a fitems=() declare -a fstrs=() get_prefix_filtered_list2 "${COMP_WORDS[0]}" \ _get_help_text_items fitems _get_help_text_helps fstrs _get_help_text_items=( "${fitems[@]}" ) _get_help_text_helps=( "${fstrs[@]}" ) fi get_help_text vyatta_completions=( "${_get_help_text_items[@]}" ) vyatta_do_complete eval $restore_shopts return fi local command=${COMP_WORDS[0]} ### Expand top level commands command=$(vyatta_cfg_expand_top_level $command) local last_comp="${COMP_WORDS[COMP_CWORD]}" # handle "exit" if [ "$command" == "exit" ]; then if (( COMP_CWORD > 1 )); then COMPREPLY=() eval $restore_shopts return fi _get_help_text_items=("discard") _get_help_text_helps=("Discard any changes") get_help_text vyatta_completions=("discard") vyatta_do_complete eval $restore_shopts return fi local -a api_args=("${COMP_WORDS[@]:0:$[$COMP_CWORD+1]}") # handle "copy" and "rename" if [ "$command" == "copy" -o "$command" == "rename" ]; then # Syntax of copy and rename commands are: # # copy/rename to # # where and are configuration parameters # in the tree at the current edit level. # # If parsing index 1 or 2 (i.e. or ), # fall through this test to the parameter parsing code below. if (( COMP_CWORD == 3 )); then # If parsing index 3, there's only one option. _get_help_text_items=("to") _get_help_text_helps=("Set destination") get_help_text vyatta_completions=("to") vyatta_do_complete eval $restore_shopts return elif (( COMP_CWORD > 3 && COMP_CWORD < 6 )); then # If parsing index 4 or 5, start completion at . api_args=("$command" "${COMP_WORDS[@]:4}") elif (( COMP_CWORD > 5 )); then # If parsing after index 5, there are no more valid parameters COMPREPLY=() eval $restore_shopts return fi fi local -a expanded_api_args vyatta_config_expand_compwords "${api_args[@]}" api_args=( "${expanded_api_args[@]}" ) if ! vyatta_cli_shell_api getCompletionEnv "${api_args[@]}"; then # invalid completion eval $restore_shopts return fi vyatta_cfg_comp_help=$_cli_shell_api_comp_help _get_help_text_helps=( "${_cli_shell_api_hstrs[@]}" ) if $_cli_shell_api_last_comp_val; then # last component is a "value". need to do the following: # use comp_help if exists # prefix filter comp_values # replace any <*> in comp_values with "" # convert help items to <...> representation _get_help_text_items=() for ((i = 0; i < ${#_cli_shell_api_hitems[@]}; i++)); do local t=$(get_value_format_string "${_cli_shell_api_hitems[i]}") _get_help_text_items+=("$t") done vyatta_completions=() for ((i = 0; i < ${#_cli_shell_api_comp_values[@]}; i++)); do if [ -z "$last_comp" ] \ && [[ "${_cli_shell_api_comp_values[i]}" = \<*\> ]]; then vyatta_completions+=("") elif [ -z "$last_comp" ] \ || [[ "${_cli_shell_api_comp_values[i]}" = "$last_comp"* ]]; then vyatta_completions+=("${_cli_shell_api_comp_values[i]}") fi done else _get_help_text_items=( "${_cli_shell_api_hitems[@]}" ) vyatta_completions=( "${_cli_shell_api_comp_values[@]}" ) fi get_help_text vyatta_simple_complete eval $restore_shopts } if ! vyatta_cli_shell_api setupSession; then echo 'Failed to set up config session' builtin exit 1 fi # disallow 'Ctrl-D' exit, since we need special actions on 'exit' builtin set -o ignoreeof 1 reset_edit_level export VYATTA_COMP_LINE_EMPTY=VYATTA_COMP_LINE_EMPTY export VYATTA_COMP_LINE=$VYATTA_COMP_LINE_EMPTY # readline bindings case "$-" in *i*) bind 'set show-all-if-ambiguous on' if ! bind -p |grep -q '\\C-x\\C-t'; then bind '"\C-x\C-t": kill-region' fi if ! bind -p |grep -q '\\C-x\\C-o'; then bind '"\C-x\C-o": copy-region-as-kill' fi ;; esac # note: now that we're using bash's new "empty completion" (-E), it becomes # necessary to capture the "default completion" (-D) as well in order to # provide completion "within the first word". (see below for -E and -D # assignments.) however, this changes the previous behavior that uses # "filename completion" as default completion. # # since we explicitly specify the completion function for each vyatta command, # the "default completion" only applies in two scenarios: # 1. "within" the first word, and # 2. after any non-vyatta commands that do not have completion functions. # # therefore, to provide the previous behavior, just detect scenario 2 above # and use filename completion. vyatta_config_default_complete () { local wc=${#COMP_WORDS[@]} if (( wc < 2 )); then vyatta_config_complete else # after the first word => cannot be vyatta command so use original default compopt -o filenames _filedir_xspec fi } _vyatta_cfg_init # Local Variables: # mode: shell-script # sh-indentation: 4 # End: