yesh
yesh

Reputation: 2070

Better way of parsing the input

My shell script takes in a set of input variables some are optional. The input should follow the following signature.

myscript.sh var1 var2 [-x var3] [-y var4]`.

The -x and -y are optional inputs and these options can occur at any location (between var1 & var 2 or at the start) but var3 will always preceed var4. Also var1 will preceed var 2.

The script I have does a lot of if else checks to cover all the probabilities. Something like this

if [ $1 == "-x" ]; then
## Then check if $3 == "-y" 
## Assign values to VAR1 , VAR2, VAR3, VAR4 
## else check for other possibilities
fi

I was wondering if there was a better way of doing this and assigning the values to variables?

Upvotes: 0

Views: 108

Answers (4)

F. Hauri  - Give Up GitHub
F. Hauri - Give Up GitHub

Reputation: 70742

with a test to ensure that optionals arguments, if provided, are provided only once.

#!/bin/bash

die() {
    echo >&2 $@
    exit 1
}

declare -a vars_1_2
var1='' var2='' var3='' var4=''
while [ "$1" ];do
    case $1 in
        -x ) [ "$var3" ] && die "Argument '-x' provided twice!"
             shift
             var3=$1
             shift
             ;;
        -y ) [ "$var4" ] && die "Argument '-y' provided twice!" 
             shift
             var4=$1
             shift
             ;;
        * ) vars_1_2+=($1) ; shift ;;
    esac
done
[ ${#vars_1_2[@]} -ne 2 ] && \
    die "Needed exactly 2 positional arguments. Got ${#vars_1_2[@]}."
var1=${vars_1_2[0]}
var2=${vars_1_2[1]}
set | grep ^var

make:

$ ./parsargs.sh 12 33 -x 45 -y 67 -x 2
Argument '-x' provided twice!

$ ./parsargs.sh 12 -x 45 -y 67 
Needed exactly 2 positional arguments. Got 1.

$ ./parsargs.sh 12 34 -x 45 -y 67 
var1=12
var2=34
var3=45
var4=67
vars_1_2=([0]="12" [1]="34")

Upvotes: 1

F. Hauri  - Give Up GitHub
F. Hauri - Give Up GitHub

Reputation: 70742

As Charles said: this work only in bash (>2 ;), This another purpose seem more compact:

#!/bin/bash

declare -a vars_1_2
var1='' var2='' var3='' var4=''
while [ "$1" ];do
    case $1 in
        -x ) shift ; var3=$1 ; shift ;;
        -y ) shift ; var4=$1 ; shift ;;
        * ) vars_1_2+=($1) ; shift ;;
    esac
done
if [ ${#vars_1_2[@]} -ne 2 ] ;then
    echo "Needed exactly 2 positional arguments. Got ${#vars_1_2[@]}." >&2
    exit 1
    fi
var1=${vars_1_2[0]}
var2=${vars_1_2[1]}
set | grep ^var

Let probe this code:

$ ./parsargs.sh
Needed at least 2 positional arguments. Got 0.

$ ./parsargs.sh 12
Needed at least 2 positional arguments. Got 1.

$ ./parsargs.sh 12 34
var1=12
var2=34
var3=
var4=
vars_1_2=([0]="12" [1]="34")

$ ./parsargs.sh 12 34 56
Needed at least 2 positional arguments. Got 3.

$ ./parsargs.sh 12 34 -x 56
var1=12
var2=34
var3=56
var4=
vars_1_2=([0]="12" [1]="34")

$ ./parsargs.sh 12 34 -x 56 -y 78
var1=12
var2=34
var3=56
var4=78
vars_1_2=([0]="12" [1]="34")

$ ./parsargs.sh 12 34 -x 56 -y 78 -z 9
Needed at least 2 positional arguments. Got 4.

$ ./parsargs.sh -x 56 12 -y 78 34
var1=12
var2=34
var3=56
var4=78
vars_1_2=([0]="12" [1]="34")

Upvotes: 1

Charles Duffy
Charles Duffy

Reputation: 295308

Easy enough, without even needing getopt. Note that the following uses bash-only syntax, so you should start your script with #!/bin/bash, not #!/bin/sh.

args=()
while (( $# )); do
  if [[ $1 = -x ]] ; then
    x_val=$2
    shift; shift
  elif [[ $1 = -y ]] ; then
    y_val=$2
    shift; shift
  elif (( ${#args[@]} < 2 )); then
    args+=( "$1" )
    shift
  else
    printf 'Unhandled argument: %q\n' "$1" >&2
    exit 1
  fi
done
if (( ${#args[@]} < 2 )) ; then
  echo "Needed at least 2 positional arguments; got only ${#args[@]}." >&2
  exit 1
fi

See also the relevant BashFAQ entry.

Upvotes: 2

Gilles Qu&#233;not
Gilles Qu&#233;not

Reputation: 185015

Take a look to getopts

man getopts

And if you want long GNU style switchs, see

getopts_long_example

and

getopts_long

And a recommended doc to read before anything : http://mywiki.wooledge.org/BashFAQ/035

Upvotes: 3

Related Questions