Is this a bash-specific method of calling a script: VARIABLE=value bash somescript

I have worked all my life in HPUX/ksh shell, so bash these days seems to be throwing new ways of doing many things (many of which don't seem to be POSIX compliant - for example 'mkdir -p /root/{abc,def,ghi,jkl}' - apparently this is a coding guideline best-practice in bash / OpenStack - but that's a different rant altogether).

Anyways, I have been seeing this sort of script execution in multiple places in OpenStack code:

ENABLE_VCENTER=true bash /root/stack_install.sh

Can someone explain how this works - I know that the shell will set the variable and then invoke the shell script (which is the explanation the classic nooB mistake for why variable assignment shouldn't be with spaces - eg. HELLO= world). But what exactly happens internally?

Also, why doesn't it seem to work for me when I try this command:

HELLO=world echo $HELLO

I get the following output from the shell:

[ ubuntuOne : /home/bumblebee ]
> HELLO=world echo $HELLO


[ ubuntuOne : /home/bumblebee ]
> 

What am I missing here?

Thanks for your replies.


Solution 1:

HELLO=world echo $HELLO

In the above, bash does parameter expansion before the statement is executed. That means that $HELLO is replaced with nothing (or whatever the current value of HELLO is) before the statement is executed (which would include setting HELLO to world).

Observe:

$ HELLO=world bash -c 'echo $HELLO'
world

Notice that echo $HELLO is inside single quotes. Single quotes prevent parameter expansion. That means that HELLO is not evaluated until the bash subshell starts up and sees it as an argument. By that time, HELLO will have been set equal to world.

By contrast, consider this line in which the single quotes have been replaced with double quotes:

$ HELLO=world bash -c "echo $HELLO"

Here nothing prints because double-quotes do not suppress parameter expansion. Thus, the bash command is given two arguments -c and echo. Hence, nothing prints.

All the gory details

man bash explains, in detail, what happens when commands such as the above are processed:

When a simple command is executed, the shell performs the following expansions, assignments, and redirections, from left to right.

   1.     The words that  the  parser  has  marked  as  variable
          assignments  (those  preceding  the  command name) and
          redirections are saved for later processing.

   2.     The words that are not variable assignments  or  redi‐
          rections  are  expanded.   If  any  words remain after
          expansion, the first word is taken to be the  name  of
          the command and the remaining words are the arguments.

   3.     Redirections  are  performed  as described above under
          REDIRECTION.

   4.     The text after  the  =  in  each  variable  assignment
          undergoes  tilde  expansion, parameter expansion, com‐
          mand substitution,  arithmetic  expansion,  and  quote
          removal before being assigned to the variable.

Consider again the command:

HELLO=world echo $HELLO

The argument to echo undergoes parameter expansion in step 2 while bash does not even begin to assign the new value to HELLO until step 4.