Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

What's the meaning of a ! before a command in the shell?

What is the purpose of a shell command (part of a shell script) starting with an exclamation mark?

Concrete example:

In foo.sh:

#!/usr/bin/env bash set -e ! docker stop foo ! docker rm -f foo # ... other stuff 

I know that without the space the exclamation mark is used for history replacements and ! <expression> according to the man page can be used to evaluate "True if expr is false". But in the example context that does not make sense to me.

like image 356
sthzg Avatar asked Nov 13 '17 09:11

sthzg


People also ask

What is the meaning of symbol in shell?

It's called pipe . It gives the output of the first command as the input to the second command.

What does before a command mean in Linux?

simply negates the exit status of a command. It is just one of the many exceptions in the list of commands whose exit status set -e will ignore.

What is the purpose of an & after a command?

The & symbol instructs commands to run in a background process and immediately returns to the command line for additional commands.

What does $() mean in shell script?

$() – the command substitution. ${} – the parameter substitution/variable expansion.


1 Answers

TL;DR: This is just by-passing the set -e flag in the specific line where you are using it.


Adding add to hek2mgl's correct and useful answer.

You have:

set -e ! command 

Bash Reference Manual → Pipelines describes:

Each command in a pipeline is executed in its own subshell. The exit status of a pipeline is the exit status of the last command in the pipeline (...). If the reserved word ‘!’ precedes the pipeline, the exit status is the logical negation of the exit status as described above. The shell waits for all commands in the pipeline to terminate before returning a value.

This means that ! preceding a command is negating the exit status of it:

$ echo 23 23 $ echo $? 0                 # But $ ! echo 23 23 $ echo $? 1 

Or:

$ echo 23 && echo "true" || echo "fail" 23 true $ ! echo 23 && echo "true" || echo "fail" 23 fail 

The exit status is useful in many ways. In your script, used together with set -e makes the script exit whenever a command returns a non-zero status.

Thus, when you have:

set -e command1 command2 

If command1 returns a non-zero status, the script will finish and won't proceed to command2.

However, there is also an interesting point to mention, described in 4.3.1 The Set Builtin:

-e

Exit immediately if a pipeline (see Pipelines), which may consist of a single simple command (see Simple Commands), a list (see Lists), or a compound command (see Compound Commands) returns a non-zero status. The shell does not exit if the command that fails is part of the command list immediately following a while or until keyword, part of the test in an if statement, part of any command executed in a && or || list except the command following the final && or ||, any command in a pipeline but the last, or if the command’s return status is being inverted with !. If a compound command other than a subshell returns a non-zero status because a command failed while -e was being ignored, the shell does not exit. A trap on ERR, if set, is executed before the shell exits.


Taking all of these into consideration, when you have:

set -e ! command1 command2 

What you are doing is to by-pass the set -e flag in the command1. Why?

  • if command1 runs properly, it will return a zero status. ! will negate it, but set -e won't trigger an exit by the because it comes from a return status inverted with !, as described above.
  • if command1 fails, it will return a non-zero status. ! will negate it, so the line will end up returning a zero status and the script will continue normally.
like image 129
fedorqui 'SO stop harming' Avatar answered Oct 20 '22 10:10

fedorqui 'SO stop harming'