Bash Scripting and Shell Programming
Bash Scripting and Shell Programming
Shebang
If a script does not contain a shebang the commands are executed, using your
Shell.
Different shells have slightly varying syntax.
#!/bin/csh
echo "This script uses csh as the interpreter."
#!/bin/ksh
echo "This script uses ksh as the interpreter."
#!/bin/zsh
echo "This script uses zsh as the interpreter."
Execution
sleepy.sh
#!/bin/bash
sleep 90
$ ./sleepy.sh &
[1] 16796
$ ps -fp 16796
UID PID PPID c STIME TTY TIME CMD
jacks 16796 16725 0 22:50 pts/0 00:00:00
/bin/bash ./sleepy.sh
$
$ /tmp//sleepy.sh &
[1] 16804
$ ps -fp 16804
UID PID PPID c STIME TTY TIME CMD
jacks 16804 16725 0 22:51 pts/0 00:00:00
/bin/bash /tmp/sleepy.sh
$
$ ps -ef | grep 16804 | grep -v grep
jacks 16804 16725 0 22:51 pts/0 00:00:00
/bin/bash /tmp/sleepy.sh
jacks 16804 16725 0 22:51 pts/0 00:00:00
sleep 90
$ pstree -p 16804
sleepy.sh(16804)-----sleep(16805)
$
Variable
Storage locations that have a name
Name-value pairs
Syntax:
VARIABLE_NAME="Value"
Variable Usage
#!/bin/bash
MY_SHELL="bash"
echo "I like the $MY_SHELL shell."
#!/bin/bash
MY_SHELL="bash"
echo "I like the ${MY_SHELL} shell."
The curly brace syntax is optional, unless you need to immediately precede
or follow the variable with additional data.
Example
Lets say you want to add the letter 'ing' to the value
#!/bin/bash
MY_SHELL="bash"
echo "I like the ${MY_SHELL}ing on my keyboard."
Note
you can't do 'echo "I like the $MY_SHELLing on my keyboard." '
Because the shell will interpret as variable name MY_SHELLing
#!/bin/bash
SERVER_NAME=$(hostname)
echo "You are running this script on ${SERVER_NAME}."
#!/bin/bash
SERVER_NAME=`hostname`
echo "You are running this script on ${SERVER_NAME}."
Variables name
Here some example of valid and invalid for variable names
Note
Also is a convention that variables names be uppercase
Valid:
FIRST3LETTERS="ABC"
FIRST_THREE_LETTERS="ABC"
firstThreeLetters="ABC"
Invalid:
3LETTERS="ABC"
first-three-letters="ABC"
first@Three@Letters="ABC"
TESTS
For test a condition
Syntax:
[ condition-to-test-for]
example:
The If statement
Syntax:
if [ condition-is-true ]
then
command 1
command 2
command n
fi
An example:
#!/bin/bash
MY_SHELL="bash"
if [ "$MY_SHELL" = "bash" ]
then
echo "You seem to like the bash shell."
fi
An example:
#!/bin/bash
MY_SHELL="csh"
if [ "$MY_SHELL" = "bash" ]
then
echo "You seem to like the bash shell."
else
echo "You don't seem to like the bash shell."
fi
if [ condition-is-true ]
then
command n
elif [ condition-is-true ]
then
command n
else
command n
fi
Note
You can also test for mutiple conditions using the 'elif'
An example:
#!/bin/bash
MY_SHELL="csh"
if [ "$MY_SHELL" = "bash" ]
then
echo "You seem to like the bash shell."
elif [ "$MY_SHELL" = "csh" ]
then
echo "You seem to like the csh shell."
else
echo "You don't seem to like the bash or csh shell."
fi
For loop
For iterate N times you can use the for loop
Syntax:
An example:
#!/bin/bash
for COLOR in red green blue
do
echo "COLOR: $COLOR"
done
Output:
COLOR:red
COLOR:green
COLOR:blue
Example
Positional Parameters
The value of parameter can be accessed by special syntax like n, beginwith0 that will contain tha
name of the program, until many parameters was passed
Example
Will be contain
var value
$0 "script.sh"
$1 "parameter1"
$2 "parameter2"
$3 "parameter3"
Note
You can assign this Positional parameter like
USER=$1
#!/bin/bash
Note
Using the $@ you can pass multiple user, how many needed.
Syntax:
Sumary 1
#!/path/to/interpreter
VARIABLE_NAME="Value"
$VARIABLE_NAME
${VARIABLE_NAME}
VARIABLE_NAME=$(command)
If
if [ condition-is-true ]
then
commands
elif[ condition-is-true ]
then
commands
else
commands
fi
For loop
Positional Parameters:
$0, $1, $2 ... $9
$@
Comments # This will be ignored
To accept input use read
Example
> $ ls /note/here
echo "$?"
Output:
ls: cannot access '/note/here': No such file or directory
2
Note this example only check for an error, but you can
implement some error handling.
Note
If the mkdir fails the cp command will not be executed
|| = OR
Note
If the cp test.text fails, the cp test.txt will be executed.
But if the first commands executed with rc=0, the second will NOT be executed.
Example AND:
#!/bin/bash
HOST="google.com"
ping -c 1 $HOST && echo "$HOST reachable."
Note
In this example, if the ping command exits with a 0 exit status, then "google.com reachable" will
be echoed to the screen.
Example OR:
#!/bin/bash
HOST="google.com"
ping -c 1 $HOST || echo "$HOST unreachable."
Note
In this example, if the ping command fails, then "google.com unreachable" will be echoed to the
screen.
If the ping succeeds, the echo will not be executed.
The semicolon
Separate commands with a semicolon to ensure they all get executed. No matter if the previous
command fails, the next will execute.
Same as:
cp test.txt /tmp/bak
cp test.txt /tmp/
Exit Command
To explicitly define the return code you can use exit.
The exit allow number from 0 to 255.
exit 0
exit 1
exit 2
exit 255
The default values is that of the last command executed.
Sumary Exit
All command return an exit status
0-255 is the range allowed for exit status
0= success
Other than 0 = error condition
$? contains the exit StatusDecision making - if, &&,||
exit
Using man
When you type man <command> for search for a patter you can:
Creating a function
# Explicitly
function function-name() {
# Code goes here.
}
Another way
# Implicitly is the same but withou the keyword function
function-name() {
# Code goes here.
}
Note
To call or execute a function, you just need to call the name.
#!/bin/bash
function hello() {
echo "Hello!"
}
hello
#!/bin/bash
function hello() {
echo "Hello!"
now
}
function now() {
echo "It's $(date +%r)"
}
hello
Note
you don't need the parenthesis, just te name, if you need to provide
some arguments (parameters), they will follow the name
with a space as separator.
For loop of all parameter you can use the same strategy of the for loop with $@
#!/bin/bash
function hello() {
for NAME in $@
do
echo "Hello $NAME"
done
}
hello Jason Dan Ryan
Variable Scope
By default, variable are global
Variables have to be defined before used.
Note
One attention will be on the declaration point, if you try to use a variable that wasn't declared
before you will get a null point.
my_function
GLOBAL_VAR=1
The variable are global but my_function will not get access To
because was declared after my_function call.
Other case:
#!/bin/bash
my_function() {
GLOBAL_VAR=1
}
# GLOBAL_VAR not available yet
echo $GLOBAL_VAR
my_function
# GLOBAL_VAR is NOW available
echo $GLOBAL_VAR
Note
On the first echo you don't get any value because the execution of the function has not yet take
place.
And is on the execution that declaration will create the GLOBAL_VAR.
Local Variables
Can only be accessed within the function
Create using the local keyword.
local LOCAL_VAR=1
Only function can have local variables.
Best practice to keep variables local in functions.
function backup_file() {
if [ -f $1 ]
then
local BACK="/tmp/$(basename $(1)).$(date +%F).$$"
echo "Backing up $1 to $(BACK)"
# The exit status of the function will
# be the exit status of the cp command.
cp $1 $BACK
else
# The file does not exist.
return 1
fi
}
backup_file /etc/hosts
if [ $? -eq 0 ]
then
echo "Backup succeeded!"
fi
Case Statement
Alternative to if statement
if[ "$VAR"="one"]
elif[ "$VAR"="two"]
elif[ "$VAR"="three"]
elif[ "$VAR"="four"]
case "$VAR" in
pattern_1)
# Commands go here.
;;
pattern_N)
# Commands go here
;;
esac
If the pattern matches the commands will be executed until reach a ;;
An example:
case "$1" in
start)
/usr/sbin/sshd
;;
stop)
kill $(cat /var/run/sshd.pid)
;;
esac
Note
if you pass START with capital letters nothing will happen Because
The bash is case sensitive.
A modified version will take case about the any other case like a default
case "$1" in
start)
/usr/sbin/sshd
;;
stop)
kill $(cat /var/run/sshd.pid)
;;
*)
echo "Usage: $0 start|stop" ; exit 1
;;
esac
Note
The *) will be executed if any other pattern matches.
Another improvement
case "$1" in
start|START)
/usr/sbin/sshd
;;
stop|STOP)
kill $(cat /var/run/sshd.pid)
;;
*)
echo "Usage: $0 start|stop" ; exit 1
;;
esac
The | will be a or, so in this case both start and START will be a match for the first, and stop or
STOP also will be a match for the second
doing this we will take care of the lower and upper case.
Asking for
The input is stored in the variable ANSWER
Here we are using the character class to filter
While Loop
While loop is a loop that repeats a series of commands fo as long
the condition be true, if the condition fails with a exit status
different from 0 the loop will stop.
while [ CONDITION_IS_TRUE ]
do
command 1
command 2
command n
done
Note
If the condition is never true, than the commands inside the while never will be executed too.
Infinite loop:
Note
If the condition never changes from true to false inside the while loop, than you never break the
loop, this will be a infinite loop.
Than you can use control + C to interrupt, or kill the process.
If you want a condition that will be always true, you can use true keyword.
while true
do
command N
sleep 1
done
INDEX=1
while [ $INDEX -lt 6 ]
do
echo "Creating project-${INDEX}"
mkdir /usr/local/project-${INDEX}
((INDEX++))
done
Note
the >/dev/null is for to discard the output, and get only the return code.
Pipe a command
More complex
FS_NUM=1
grep xfs /etc/fstab | while read FS MP REST
do
echo "${FS_NUM}: file system: ${FS}"
echo "${FS_NUM}:mount point: ${MP}"
((FS_NUM++))
done
read: read [-ers] [-a array] [-d delim] [-i text] [-n nchars] [-N nchars]
[-p prompt] [-t timeout] [-u fd] [name ...]
Read a line from the standard input and split it into fields.
Reads a single line from the standard input, or from file descriptor FD
if the -u option is supplied. The line is split into fields as with word
splitting, and the first word is assigned to the first NAME, the second
word to the second NAME, and so on, with any leftover words assigned to
the last NAME. Only the characters found in $IFS are recognized as word
delimiters.
If no NAMEs are supplied, the line read is stored in the REPLY variable.
Options:
-a array assign the words read to sequential indices of the array
variable ARRAY, starting at zero
-d delim continue until the first character of DELIM is read, rather
than newline
-e use Readline to obtain the line in an interactive shell
-i text use TEXT as the initial text for Readline
-n nchars return after reading NCHARS characters rather than waiting
for a newline, but honor a delimiter if fewer than
Exit Status:
The return code is zero, unless end-of-file is encountered, read times out
(in which case it's greater than 128), a variable assignment err
Continue keyword
mysql -BNe 'show databases' | while read DB
do
db-backed-up-recently $DB
if [ "$?" -eq "0" ]
then
continue
fi
backup $DB
done
Any command that follow the continue statement in the loop will be executed. Execution
continues back at the top of the loop and the
while condition is examined again.
Here we are looping through a list of MySQL databases, the -B option to MySQL disables the
ASCII table output that MySQL normally displays.
The -N option suppresses the column names in the output. Hidding the headers.
The -e option causes MySQL to execute the command that follow it.
MySQL is showing a database per line of output (the show databases)
That is piped to a while loop
the read assigns the input to the DB variable. First we check if the database has been backed up
recently this is a script ( db-backed-up-recently). If return 0, the database is passed to it has
backed up in the las 24 hr, otherwise returns a 1.
we use a if to check the return code of that script
if the database was backed up, we call the continue, restarting the process.
when this if get a false we execute the last of the lines below the continue.
References
reference
bash-scripting course Udemy
https://linuxhint.com/bash_read_command/#:~:text=Read is a bash builtin,taking input from
standard input.