getops: condição para restringir opções de entrada

getops: condição para restringir opções de entrada

Eu tenho um script de shell que aceita 4 opções de entrada (x:m:b:v). Mas quero restringir minha opção de entrada.

Condição 1

1. Accept option x or x and b
   eg: sh script -x <arg> (or)             ;will execute function a defined in script
       sh script -x <arg> -b               ;will execute the functions a and b

Condição 2

2. Accept option m or m and b
   eg: sh script -m <arg> (or)     ;execute function c
       sh script -m <arg1> -b   ;should execute function for c and b

Condição 3

3. Option both x and m should not be passed together.
   This will print usage(how to use the input options) function.

Condição 4

4. Option v. This should not be passed with any other option.
   eg: sh script -v <arg>  ; executs the function for v

Eu sei que isso pode ser alcançado com condições if e else, mas não consigo obter a lógica das condições.

Além disso, se nenhum dos argumentos de entrada for passado (ou) se todos os argumentos de entrada forem passados, ele deverá executar a função de uso.

Responder1

Como exemplo para ilustrar meu comentário acima, aqui está um trecho do FreeBSD /usr/src/bin/cp/cp.c:

while ((ch = getopt(argc, argv, "HLPRafilnprsvx")) != -1)
        switch (ch) {
        case 'H':
                Hflag = 1;
                Lflag = 0;
                break;
        case 'L':
                Lflag = 1;
                Hflag = 0;
                break;
        ...
        case 'l':
                lflag = 1;
                break;
        ...
        case 's':
                sflag = 1;
                break;
        case 'v':
                vflag = 1;
                break;
        case 'x':
                fts_options |= FTS_XDEV;
                break;
        default:
                usage();
                break;
        }

Observe que o acima define muitas variáveis ​​booleanas como Hflag, Lflag, lflag, sflag, etc. Isso não apenas permite que o código teste relações booleanas entre sinalizadores, mas também permite que a presença de um sinalizador ative outro ou substitua outro. Observe também que a defaultcláusula da switchinstrução força uma usage()chamada quando o usuário especifica um sinalizador de opção indefinido.

Quanto às exclusões de sinalizadores, como proibir mflag&& xflag, um caso semelhante aparece em cp.c:

if (lflag && sflag)
        errx(1, "the -l and -s options may not be specified together");

Aqui está uma possível tradução desse código de C para bashcom adaptação adicional ao seu exemplo:

flagb=
flagm=
flagv=
flagx=

while getopts "bm:vx:" ch
do

        case "$ch" in
        [bv])
                eval flag$ch=1
                printf 'flag%s set\n' "$ch"
                ;;
        [mx])
                eval flag$ch=1
                eval arg$ch=\"$OPTARG\"
                printf 'flag%s set\n' "$ch"
                printf 'arg%s is "%s"\n' "$ch" "$OPTARG"
                ;;
        -)
                break
                ;;
        esac

done

[[ $OPTIND > 1 ]] && shift $(($OPTIND-1))

if [[ "$flagv" ]]
then
        if [[ -n "$flagb$flagm$flagx" ]]
        then
                printf 'error: -v cannot be used with any other flags\n' >&2
                exit 2
        fi
fi

if [[ "$flagm$flagx" = "11" ]]
then
        printf 'error: -m and -x cannot be used together\n' >&2
        exit 2
fi

if [[ "$flagb" ]] && ! [[ "$flagm$flagx" ]]
then
        printf 'error: -b requires either -m or -x\n' >&2
        exit 2
fi

[[ "$flagb" ]] && printf 'b flag is set\n'
[[ "$flagm" ]] && printf 'm flag is set, arg is "%s"\n' "$argm"
[[ "$flagv" ]] && printf 'v flag is set\n'
[[ "$flagx" ]] && printf 'x flag is set, arg is "%s"\n' "$argx"

[[ "$@" ]] && {
        printf 'remaining arguments:'
        printf ' "%s"' "$@"
        printf '\n'
}

informação relacionada