Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

What's a concise way to check that environment variables are set in a Unix shell script?

Tags:

bash

shell

unix

I've got a few Unix shell scripts where I need to check that certain environment variables are set before I start doing stuff, so I do this sort of thing:

if [ -z "$STATE" ]; then     echo "Need to set STATE"     exit 1 fi    if [ -z "$DEST" ]; then     echo "Need to set DEST"     exit 1 fi 

which is a lot of typing. Is there a more elegant idiom for checking that a set of environment variables is set?

EDIT: I should mention that these variables have no meaningful default value - the script should error out if any are unset.

like image 793
AndrewR Avatar asked Nov 21 '08 01:11

AndrewR


People also ask

How do you check if an environment variable is set?

In the command window that opens, enter echo %VARIABLE%. Replace VARIABLE with the name of the environment variable you set earlier. For example, to check if MARI_CACHE is set, enter echo %MARI_CACHE%. If the variable is set, its value is displayed in the command window.

How do you check if a variable is defined in Unix?

To find out if a bash variable is defined: Return true if a bash variable is unset or set to the empty string: if [ -z ${my_variable+x} ]; Also try: [ -z ${my_bash_var+y} ] && echo "\$my_bash_var not defined" Determine if a bash variable is set or not : [[ ! -z ${PURGEIMAGE+z} ]] && echo "Set" || echo "Not defined"

How do you set an environment variable in UNIX shell script?

To set an environment variable everytime, use the export command in the . bashrc file (or the appropriate initialization file for your shell). To set an environment variable from a script, use the export command in the script, and then source the script.


2 Answers

Parameter Expansion

The obvious answer is to use one of the special forms of parameter expansion:

: ${STATE?"Need to set STATE"} : ${DEST:?"Need to set DEST non-empty"} 

Or, better (see section on 'Position of double quotes' below):

: "${STATE?Need to set STATE}" : "${DEST:?Need to set DEST non-empty}" 

The first variant (using just ?) requires STATE to be set, but STATE="" (an empty string) is OK — not exactly what you want, but the alternative and older notation.

The second variant (using :?) requires DEST to be set and non-empty.

If you supply no message, the shell provides a default message.

The ${var?} construct is portable back to Version 7 UNIX and the Bourne Shell (1978 or thereabouts). The ${var:?} construct is slightly more recent: I think it was in System III UNIX circa 1981, but it may have been in PWB UNIX before that. It is therefore in the Korn Shell, and in the POSIX shells, including specifically Bash.

It is usually documented in the shell's man page in a section called Parameter Expansion. For example, the bash manual says:

${parameter:?word} 

Display Error if Null or Unset. If parameter is null or unset, the expansion of word (or a message to that effect if word is not present) is written to the standard error and the shell, if it is not interactive, exits. Otherwise, the value of parameter is substituted.

The Colon Command

I should probably add that the colon command simply has its arguments evaluated and then succeeds. It is the original shell comment notation (before '#' to end of line). For a long time, Bourne shell scripts had a colon as the first character. The C Shell would read a script and use the first character to determine whether it was for the C Shell (a '#' hash) or the Bourne shell (a ':' colon). Then the kernel got in on the act and added support for '#!/path/to/program' and the Bourne shell got '#' comments, and the colon convention went by the wayside. But if you come across a script that starts with a colon, now you will know why.


Position of double quotes

blong asked in a comment:

Any thoughts on this discussion? https://github.com/koalaman/shellcheck/issues/380#issuecomment-145872749

The gist of the discussion is:

… However, when I shellcheck it (with version 0.4.1), I get this message:

In script.sh line 13: : ${FOO:?"The environment variable 'FOO' must be set and non-empty"}   ^-- SC2086: Double quote to prevent globbing and word splitting. 

Any advice on what I should do in this case?

The short answer is "do as shellcheck suggests":

: "${STATE?Need to set STATE}" : "${DEST:?Need to set DEST non-empty}" 

To illustrate why, study the following. Note that the : command doesn't echo its arguments (but the shell does evaluate the arguments). We want to see the arguments, so the code below uses printf "%s\n" in place of :.

$ mkdir junk $ cd junk $ > abc $ > def $ > ghi $  $ x="*" $ printf "%s\n" ${x:?You must set x}    # Careless; not recommended abc def ghi $ unset x $ printf "%s\n" ${x:?You must set x}    # Careless; not recommended bash: x: You must set x $ printf "%s\n" "${x:?You must set x}"  # Careful: should be used bash: x: You must set x $ x="*" $ printf "%s\n" "${x:?You must set x}"  # Careful: should be used * $ printf "%s\n" ${x:?"You must set x"}  # Not quite careful enough abc def ghi $ x= $ printf "%s\n" ${x:?"You must set x"}  # Not quite careful enough bash: x: You must set x $ unset x $ printf "%s\n" ${x:?"You must set x"}  # Not quite careful enough bash: x: You must set x $  

Note how the value in $x is expanded to first * and then a list of file names when the overall expression is not in double quotes. This is what shellcheck is recommending should be fixed. I have not verified that it doesn't object to the form where the expression is enclosed in double quotes, but it is a reasonable assumption that it would be OK.

like image 191
Jonathan Leffler Avatar answered Sep 18 '22 10:09

Jonathan Leffler


Try this:

[ -z "$STATE" ] && echo "Need to set STATE" && exit 1; 
like image 38
David Schlosnagle Avatar answered Sep 20 '22 10:09

David Schlosnagle