Como configuro o preenchimento bash para argumentos de comando?

33

Quando abro a linha de comando e digito git , recebo um monte de conclusões para argumentos do git, como adicionar, confirmar, mesclar e assim por diante.

Como adiciono esse tipo de conclusão aos meus próprios aplicativos?

Até agora, consegui especificar o primeiro nível de conclusão assim:

$ complete -W "asd asdf" ./test.py

$ ./test.py asd
asd asdf

Mas ele tenta continuar depois disso:

$ ./test.py asdf asd
asd asdf

Eu tentei definir uma conclusão para o segundo nível executando complete -W "lol" "./test.py asd" , mas isso não funciona.

Qual é a maneira correta de configurá-los? E uma vez que eu descobri todos os comandos, onde devo colocá-los? Executar uma lista grande desses comandos em .bashrc , toda vez que um terminal é aberto, não parece muito eficiente (além disso, não quero mexer com os arquivos dos usuários se não precisar).

    
por Stefano Palazzo 13.01.2012 / 18:13

1 resposta

33

Todas as conclusões do bash são armazenadas em /etc/bash_completion.d/ . Então, se você está construindo software com bash_completion, valeria a pena fazer com que o deb / make install descartasse um arquivo com o nome do software naquele diretório. Aqui está um exemplo de script de conclusão do bash para o Rsync:

# bash completion for rsync

have rsync &&
_rsync()
{
    # TODO: _split_longopt

    local cur prev shell i userhost path   

    COMPREPLY=()
    cur='_get_cword'
    prev=${COMP_WORDS[COMP_CWORD-1]}

    _expand || return 0

    case "$prev" in
    --@(config|password-file|include-from|exclude-from))
        _filedir
        return 0
        ;;
    -@(T|-temp-dir|-compare-dest))
        _filedir -d
        return 0
        ;;
    -@(e|-rsh))
        COMPREPLY=( $( compgen -W 'rsh ssh' -- "$cur" ) )
        return 0
        ;;
    esac

    case "$cur" in
    -*)
        COMPREPLY=( $( compgen -W '-v -q  -c -a -r -R -b -u -l -L -H \
            -p -o -g -D -t -S -n -W -x -B -e -C -I -T -P \
            -z -h -4 -6 --verbose --quiet --checksum \
            --archive --recursive --relative --backup \
            --backup-dir --suffix= --update --links \
            --copy-links --copy-unsafe-links --safe-links \
            --hard-links --perms --owner --group --devices\
            --times --sparse --dry-run --whole-file \
            --no-whole-file --one-file-system \
            --block-size= --rsh= --rsync-path= \
            --cvs-exclude --existing --ignore-existing \
            --delete --delete-excluded --delete-after \
            --ignore-errors --max-delete= --partial \
            --force --numeric-ids --timeout= \
            --ignore-times --size-only --modify-window= \
            --temp-dir= --compare-dest= --compress \
            --exclude= --exclude-from= --include= \
            --include-from= --version --daemon --no-detach\
            --address= --config= --port= --blocking-io \
            --no-blocking-io --stats --progress \
            --log-format= --password-file= --bwlimit= \
            --write-batch= --read-batch= --help' -- "$cur" ))
        ;;
    *:*)
        # find which remote shell is used
        shell=ssh
        for (( i=1; i < COMP_CWORD; i++ )); do
            if [[ "${COMP_WORDS[i]}" == -@(e|-rsh) ]]; then
                shell=${COMP_WORDS[i+1]}
                break
            fi
        done
        if [[ "$shell" == ssh ]]; then
            # remove backslash escape from :
            cur=${cur/\:/:}
            userhost=${cur%%?(\):*}
            path=${cur#*:}
            # unescape spaces
            path=${path//\\\\ / }
            if [ -z "$path" ]; then
                # default to home dir of specified
                # user on remote host
                path=$(ssh -o 'Batchmode yes' $userhost pwd 2>/dev/null)
            fi
            # escape spaces; remove executables, aliases, pipes
            # and sockets; add space at end of file names
            COMPREPLY=( $( ssh -o 'Batchmode yes' $userhost \
                command ls -aF1d "$path*" 2>/dev/null | \
                sed -e 's/ /\\\\ /g' -e 's/[*@|=]$//g' \
                -e 's/[^\/]$/& /g' ) )
        fi
        ;;
    *)  
        _known_hosts_real -c -a "$cur"
        _filedir
        ;;
    esac

    return 0
} &&
complete -F _rsync $nospace $filenames rsync

# Local variables:
# mode: shell-script
# sh-basic-offset: 4
# sh-indent-comment: t
# indent-tabs-mode: nil
# End:
# ex: ts=4 sw=4 et filetype=sh

Provavelmente vale a pena revisar um dos arquivos de conclusão do bash que melhor corresponde ao seu programa. Um dos exemplos mais simples é o arquivo rrdtool .

    
por Marco Ceppi 13.01.2012 / 18:28