Fork me on Github
Fork me on Github

Joe Dog Software

Proudly serving the Internets since 1999

sh Scripting Cheat Sheet

If you’re like me, well, then I feel sorry for you. But if you’re like  me, then perl is the first item out of your scripting toolbox.  Unfortunately there are moments when you are forced to use sh. And since  we’d rather use perl, sh syntax leaves our brains faster than thirteen year old can hack a Windows system. This article provides a brief overview of some of the things we love to forget.

Condition Operators

The good news is: sh has a lot of condition operators. The bad news is: sh has a lot of condition operators. Who can remember  them all? It helps to have a cheat sheet.

Binary operators

  -eq (arg1 is equal to arg2)
  -ne (arg1 is not equal to arg2)
  -lt (arg1 is less than arg2)
  -le (arg1 is less than or equal to arg2)
  -gt (arg1 is greater than arg2)
  -ge (arg1 is greater than or equal to arg2)

String operators

  -z string (length of string is zero)
  -n string (length of string is non-zero)
  string1 == string2 (strings are equal)
  string1 != string2 (string are not equal)
  string1 < string2 (string1 sorts before string2)
  string1 > string2 (string1 sorts after string2)

File operators

  -a file (file exists)
  -b file (file exists and is a block special)
  -c file (file exists and is a character special)
  -d file (file exists and is a directory)
  -e file (file exists)
  -f file (file exists and is a regular file)
  -g file (file exists and is set-group-id)
  -h file (file exists and is a symbolic link)
  -k file (file exists and is sticky)
  -p file (file exists and is a named pipe)
  -r file (file exists and is readable)
  -s file (file exists and has a size greater than zero)
  -t fd   (file descriptor is open and refers to a terminal)
  -u file (file exists and is set-user-id)
  -w file (file exists and is writable)
  -x file (file exists and is executable)
  -O file (file exists and is owned by the current user)
  -G file (file exists and is owned by the current user group)
  -L file (file exists and is a symbolic link)
  -S file (file exists and is a socket)
  -N file (file exists and has been modified since last read)
  file1 -nt file2 (file1 is newer by modification date than file2)
  file1 -ot file2 (file1 is older by modification date than file2)
  file1 -ef file2 (file1 and file2 have the same device and inode)

Condition Logic

If you’ve gotten this far, you’ve either mastered English or  Google Translate. “If, then” is a common English logic construct. If he looks like a douche and he talks like a douche, then he’s probably a douche. Most computer languages skip the superfluous “then.” Not sh! If you skip it, your program will explode in several million pieces. (Or not, but whatever).

Here’s a sh conditional:

if [ $opt -eq 1 ]; then
  HOST="www.joedog.org"
elif [ $opt -eq 2 ]; then
  HOST="ftp.joedog.org"
else
  HOST="ssl.joedog.org"
fi

Your author tested that logic and it works like a charm. But what happens if opt is unitialized? Let’s say we want to assign a value  at run time: opt=$1 # $0 is the program’s name, $1 is the first arg. If a user runs your script without providing an argument, your program is gonna explode is several million pieces. (Or not, …)

Ben $ sh haha  
haha: line 4: [: -eq: unary operator expected
haha: line 6: [: -eq: unary operator expected

It’s pretty hard to evaluate nothing. We can avoid this problem by ensuring $opt evaluates to something. It’s best set it explicitly, but we can also provide it with a default value. Check this out:

if [ ${opt:-0} -eq 1 ] ; then
  HOST="www.joedog.org"
elif [ ${opt:-0} -eq 2 ] ; then
  HOST="ftp.joedog.org"
else
  HOST="ssl.joedog.org"
fi

Logical Boolean Operators

Do the business people in your company have exceptions for every rule? “Do this, no that, except when, but if…” If that’s the case, you’re gonna need yourself some boolean operators.

  && - logical AND  if [ condition1 ] && [ condition2 ]
  || - logical OR   if [ condition1 ] || [ condition2 ]
  !  - logical NOT  if [ ! condition ]

What? You want more explicit examples? Sheesh.

A=10
B=11

if [ $A -eq 10 ] && [ $B -eq 11 ] ;
then
  echo "Whoo hoo!"
fi

if [ $A -eq 11 ] || [ $B -eq 11 ] ;
then
  echo "Whoo hoo!"
fi

if [ ! $A -eq 11 ];
then
  echo "Whoo hoo!"
fi

Ben $ sh papa

Whoo hoo!

Whoo hoo!

Whoo hoo!

sh Arrays

Arrays are ordered lists of values. Each value can be accessed by the value’s index in the list. Imagine a key chain, the keys are the values and the chain is the array. Rather than track every key you own, just remember where you put the chain. Where did I put that?

Unfortunately, sh arrays aren’t quite as convenient as a key chain. It is necessary to maintain individual variables but strategic name selection can make access easier. Here’s how we build a sh array:

#!/bin/sh

a="Homer Marge Bart Lisa"
for CHARACTER in $a ; do
  echo $CHARACTER
done

Ben $ sh haha

Homer

Marge

Bart

Lisa

That was convenient. Unfortunately, we can’t reference each value by its index. The perl motto applies to sh as well: “There’s more than one way to do it!” Or something like that. Here’s an array implementation with index references:

#!/bin/sh

a0="Homer"
a1="Marge"
a2="Bart"
a3="Lisa"

# I used standard C array indexing which starts
# at zero. I could have just as easily started at
# 1 or 1000. 

# 
# Now we'll loop through the "array" and access its
# values with "eval" 

i=0
y=4
while [ $i -lt $y ]
do
  eval echo "a${i} = ${a${i}}"
  let i=$i+1
done

Ben $ sh haha

a0 = Homer

a1 = Marge

a2 = Bart

a3 = Lisa

I know what you’re saying, “meh!” Fortunately, there are alternatives to this kludge. If you’re scripting on Linux, your sh is actually bash which has much better support for arrays. Let’s bashify this script, shall we?

#!/bin/sh 
SIMPSON[0]="Homer"
SIMPSON[1]="Marge"
SIMPSON[2]="Bart"
SIMPSON[3]="Lisa"

SIMPSONS=${SIMPSON[@]}
for CHARACTER in $SIMPSONS ; do
  echo $CHARACTER
done

Ben $ sh haha

Homer

Marge

Bart

Lisa

At the risk of insulting your intelligence, you can reference Marge in this manner:

echo ${SIMPSON[1]}

Happy Hacking