-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathgetopts.sh
124 lines (123 loc) · 4.62 KB
/
getopts.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
#!/bin/bash
#
# Helper utility to simplify using Bash's getopts utility, and make it usable with functions.
#
# This is lightly forked from https://gist.github.com/dimo414/93776b78a38791ed1bd1bad082d08009
# to use ProfileGem's namespace.
#
# Example usage:
# foo() {
# local _usage=... # optional usage string
# eval "$(pg::getopts 'ab:f:v')" # provide a standard getopts optstring
# echo "f is $f" # opts are now local variables
# if (( a )); then # check boolean flags with (( ... ))
# echo "Saw -a"
# fi
# echo "$@" # opts are removed from $@, positional args remain
# }
#
# A getopts optstring consists of a series of letters, each denoting a single-letter option, such
# as `-v`. Letters followed by a : denote options that accept an argument. For example 'ab:f:v'
# means "Accept -a, -b, -f, and -v as options. Options -b and -f must be followed by an argument."
#
# No-arg options are set to 0 by default and 1 if passed as an argument, allowing concise
# testing with (( ... )).
#
# Options that accept an argument are set to the empty string by default, and otherwise
# set to the value passed as an argument. To check if a (non-empty) argument was passed
# use [[ -n "$..." ]].
#
# Numeric options, such as -0 which is often used to configure NUL-byte delimited output,
# are also supported but are prefixed with "o" to distinguish from the positional arguments.
#
# All parsed arguments are shift-ed out of $@, leaving any subsequent positional arguments
# in-place. A -- argument can be used to halt option parsing early, e.g. `-a -- -b` will
# only parse -a and leave -b as an argument.
#
# pg::getopts takes two optional arguments, min_args and max_args, which constrain the
# number of positional arguments. For example `eval $(pg::getopts '...' 2 4)` requires at
# least two but no more than four positional args.
#
# Parsing errors cause the calling function to return with exit code 2. If a _usage
# variable is in scope its contents will be included in the error message.
#
# The output of this function is intended to be passed to eval, therefore we try to
# minimize what it prints and delegate most of the heavy lifting to a separate function.
# Expected output:
# local OPTIND=1 a b c=0 # vars are dynamically parsed
# _parse_opts_helper a:b:c 0 '' "$@" || return
# shift $((OPTIND - 1)); OPTIND=1
pg::getopts() {
local i char last_char var vars=() optstring=${1:-} min_args=${2:-0} max_args=${3:-}
optstring="${optstring#:}" # ensure string is not prefixed with :
if ! [[ "$optstring" =~ ^[a-zA-Z0-9:]*$ ]] || [[ "$optstring" == *::* ]]; then
pg::err "Invalid optstring: $optstring"
echo 'return 2' # for eval-ing
return 2
fi
for (( i=${#optstring}-1 ; i >= 0 ; i-- )); do
char=${optstring:i:1}
if [[ "$char" != ":" ]]; then
var="$char"
if [[ "$var" =~ [0-9] ]]; then
# prefix with 'o' so numeric flags aren't confused for positional args
var="o${var}"
fi
if [[ "$last_char" == ":" ]]; then
vars+=("$var")
else
vars+=("${var}=0")
fi
fi
last_char=$char
done
# Do as little work as possible here, as it will be eval-ed by the caller.
echo "local OPTIND=1 ${vars[*]}"
printf 'pg::_getopts_helper %q %q %q "$@" || return\n' "$optstring" "$min_args" "$max_args"
# shellcheck disable=SC2016
echo 'shift $((OPTIND - 1)); OPTIND=1'
}
# Actual parser implementation; assumes all variables it sets are local,
# which pg::getopts sets up. Do not call directly.
pg::_getopts_helper() {
local OPTARG opt failed=0
# $1 and $3 can be empty strings, so ? instead of :?
local optstring=${1?optstring} min_args=${2:?min_args} max_args=${3?max_args}
shift 3
# ensure optstring _is_ prefixed with :
while getopts ":${optstring#:}" opt; do
case "${opt}" in
[?:])
case "${opt}" in
:) pg::err "Option '-${OPTARG}' requires an argument" ;;
[?]) pg::err "Unknown option '-${OPTARG}'" ;;
esac
failed=1
break
;;
*)
if [[ "$optstring" != *"${opt}:"* ]]; then
OPTARG=1
fi
if [[ "$opt" =~ [0-9] ]]; then
opt="o${opt}"
fi
printf -v "$opt" '%s' "$OPTARG"
;;
esac
done
local pos_args=$(( $# - OPTIND + 1 ))
if (( pos_args < min_args )); then
pg::err "Insufficient arguments; minimum ${min_args}"
failed=1
elif [[ -n "$max_args" ]] && (( pos_args > max_args )); then
pg::err "Too many arguments; maximum ${max_args}"
failed=1
fi
if (( failed )); then
if [[ -n "${_usage:-}" ]]; then
pg::err "Usage: $_usage"
fi
return 2
fi
}