Skip to content

Latest commit

 

History

History
1485 lines (1110 loc) · 43.6 KB

Linux_Shell.md

File metadata and controls

1485 lines (1110 loc) · 43.6 KB

Bash Shell

Table Of Contents

1. Symbolic Links
2. Secure Shell (SSH)
3. Using SCP
4. find Command
5. vi Editor
6. Usefull commands

  • Bash command line shortcuts
Ctrl+K  Cut from curser till the end of line
Ctrl+U  Cut from curser till beginning of line
Ctrl+W  Cut words backwards
Ctrl+Y  Paste from clilp board
Ctrl+x+e Continue editing current line in text editor (uses $EDITOR)
Alt+.   Paste previous command's last argument
  • Bash is "Bourne Again Shell". Bourne shell is developed by GNU.

  • Different shells

    1. sh - Bourne Shell
    2. bash - Bourne Again Shell
    3. csh - C shell
    4. tcsh - Turbo C shell
    5. ksh - Korn shell
  • /etc/shells file lists all available shells on the systems

  • Unoffical Bash Strict Mode

    #!/bin/bash
    set -euo pipefail
    IFS=$'\n\t'
  • The set -e option instructs bash to immediately exit if any command has a non-zero exit status.

  • set -u affects variables. When set, a reference to any variable you haven't previously defined - with the exceptions of $* and $@ - is an error, and causes the program to immediately exit.

  • set -o pipefail setting prevents errors in a pipeline from being masked. If any command in a pipeline fails, that return code will be used as the return code of the whole pipeline.

  • The IFS (Internal Field Separator) variable controls what Bash calls word splitting.

#!/bin/bash
IFS=$' '
items="a b c"
for x in $items; do
    echo "$x"
done

IFS=$'\n'
for y in $items; do
    echo "$y"
done
  • set +u disables this variable strictness, and set -u enables it.

  • if a reference is made at runtime to an undefined variable, bash has a syntax for declaring a default value, using the :- operator:

bar=${foo:-alpha}  #bar is set to "alpha"
empty_str=${undef_var:-}  #empty_str is set to ""
$ cat /etc/shells
# /etc/shells: valid login shells
/bin/sh
/bin/bash
/bin/rbash
/bin/dash
  • cat /etc/issue will show which Linux distro the container is running,

  • Last filed of every user record in /etc/passwd indicates default shell for that user.

  • "#!" is called sha bang.

  • /bin/sh is a link to bash shell in Linux.

  • shell command line provides two modes to edit the commands in command line.

    1. emacs mode
    2. vi mode
# show current options, default is emac mode
$ set | grep SHELLOPTS
SHELLOPTS=braceexpand:emacs:hashall:histexpand:history:interactive-comments:monitor

# change to vi mode
$ set -o vi
$ set | grep SHELLOPTS
SHELLOPTS=braceexpand:hashall:histexpand:history:interactive-comments:monitor:vi

# go back to emacs mode
$ set -o emacs

customize the readline default key bindings by adding your own bindings in a ~/.inputrc File. Global default file is /etc/inputrc.

Search command history by pressing Ctrl+R Repeat previous command: !! or !-1 In emacs mode, press Ctlr+P Execute command that starts with specific word: ! Clear history: history -c Get command from previous argument to current command: !!:$ Get first argument from previous command: !^ Substitue specific argument: !cp:2

# display timestamp with history command
$ export HISTTIMEFORMAT='%F %T '
$ history | more

# control history size is HISTSIZE in ~/.bash_profile
HISTSIZE=450
HISTFILESIZE=450

# change history file name using HISTFILE in ~/.bash_profile
HISTFILE=/root/.mycommands

# Eliminate consecutive repeated history entires
$ export HISTCONTROL=ignoredups

# Erase duplicate across history
$ export HISTCONTROL=erasedups

# force history not to remember command
$ export HISTCONTROL=ignorespace

# Ignore specific command from history
$ export HISTIGNORE="pwd:ls:ls -ltr:"

# disable history
$ export HISTSIZE=0

By default history is stored in ~/.bash_history

Command Description
!! Repeats the previous command
!10 Repeat the 10 th command from the history
!-2 Repeat the 2 nd command (from the last) from the history
!string Repeat the command that starts with “string” from the history
!?string Repeat the command that contains the word “string” from the history
^str1^str2^ Substitute str1 in the previous command with str2 and execute it
!!:$ Gets the last argument from the previous command
!string:n Gets the nth argument from the command that starts with “string” from the history.
# find all files that changed within last 24 hours
$ find / -ctime -1 > /tmp/changed-file-list.txt &

# suspend current job
$ find / -ctime -1 > /tmp/changed-file-list.txt
$ bg
$ %2 &

# view all background jobs
$ jobs

# move background job to foreground
$ fg

# kill background job using kill %job-number
$ kill %2

# username - hostname - full path of CWD
$ export PS1="\u@\h \w$ "

$ export PS2="continue-> "
# dark blue prompt
$ export PS1="\e[1;34m\u@\h \w$ \e[m"
  • \e[ - indicates beginning of color prompt
  • x;ym - indicate color code
  • \e[m - indicate end of color prompt

Color codes:

  • Black 0;30
  • Blue 0;34
  • Green 0;32
  • Cyan 0;36
  • Red 0;31
  • Purple 0;35
  • Brown 0;33 Replace 0 with 1 for dark color

Another way is to use following color codes:

  • Black \033[30m
  • Blue \033[34m
  • Green \033[32m
  • Cyan \033[36m
  • Red \033[31m
  • Purple \033[35m
  • Brown \033[33m

Sequence of execution of bash startup files

  • /etc/profile -> execute next step
  • ~/.bash_profile -> end
  • ~/.bash_login -> end
  • ~/.profile -> end

~/.bashrc is not directly executed by bash. However, ~/.bash_profile typically execute ~/.bashrc.

if [ -f ~/.bashrc ]; then
    . ~/.bashrc
fi

when concatenate the value of a variable with another value, use the ${} referencing form.

/etc/profile checks all *.sh file under /etc/profile.d and executes them.

"$VAR" - If quoted, it is considered as one word $VAR - iF not quoted, it will be split into multiple words

states="CA NY UT TX"
for i in $states; do
    echo $i
done
Output
------
CA
NY
UT

for i in "$states"; do
    echo $i
done
Output
------
CA NY UT TX
  • Executing a shell script like ./script.sh will spawn a new bash shell and it executes the script.

  • Positional parameters are of the form $1 to $N. When N is more than single digit, it must be enclosed in braces like ${N}. $1, $2 ... ${10}, ${11}

Command Description
$0 Script Name
$1 First parameter
$2 Second parameter
$* All parameters
$@ All parameters
$# Total number of parameters
$$ PID of the shell
$! PID of most recently executed background process
$? Exit status of most recently executed command
$- All options set using bash set builtin command
$_ Gives last argument to previous command. At the shell startup, it gives absolute filename of the shell script being executed.
# display all parameters using following single line code-snippet
printf '"%b"\n' "$@" | cat -n

$* and $@ are same except when double quoted. "$*" is converted to "${1}x${2}x${3}x..."

# get range of parameters ${@:$start:$count}
$ start=2
$ count=3
$ ${@:$start:$count}
$ for i in "${@:$start:$count}"; do
$    echo $i
$ done
  • shift command moves the arguments (positional parameters) to the left by n positions. If no argument is specified, the argument will be moved by 1.

  • echo -n Do not output trailing newline

  • echo -e Enable interpretation of backslash escapes

  • single quotes '', will not expand variable values, i.e $ symbol will be treated as a literal value.

  • Declare variables using typeset and declare builtins. typeset is deprecated.

# declare integer
$ declare -i variablename

# readonly variable
$ declare -r variablename

# array
$ declare -a array

# associative array
$ declare -A array
  • Inside shell script use let for arithmetic expressions.
# both are same, no need to have $ on RHS
let total=total+3
let total=$total+3
  • ((expression)) tells bash to evaluate contents as an expression. This is very much like let. However within (()), you can use spaces. (()) allows pre- and post- increment/decrement:
$ ((total++))
$ ((total--))
$ ((++total))
  • expr is a unix command (not a bash builtin) which can be used to evaluate an expression.

  • Bash does not understand floating point arithmetic. It treats numbers containing a decimal point as strings.

Number Comparison

Operator Description
-eq Equal to
-ne Not Equal to
-gt Greater than
-ge Greater than or equal to
-lt Less than
-le Less than or equal to

String Comparison

Operator Description
= Equal to
== Equal to
!= Not Equal to
<
>
-z Zero byte? Is string empty?
-n Not empty?
if [ conditional express ]; then
    statement1
    statement2
else
    statement3
fi

if [ conditional express ]; then
    statement1
    statement2
elif [ conditional expression2 ]; then
    statement3
else
    statement4
fi
  • In a shell
    • 0 means last command executed successfully
    • 1 means last command execution failed

File Test Operators

Operator Description
-e File exists
-f It's a regular file
-d It's a directory
-b It's a block device
-c It's a character device
-s File is not empty
-p It's a pipe
-S It's a socket
-h It's a symbolic link
-t Check if FD is opened in terminal
-r File read permission
-w
-x
-u suid set on the file
-g sgid set on the file
-k Sticky bit set on file
-O You own the file
-G File group id and my group id are same
-N fiel got modified since last read
f1 -nt f2 f1 newer than f2
f1 -ot f2 f1 is older than f2
f1 -ef f2 Both files are hard linked to same file
  • -a does 'and' comparison inside conditional expression
  • -o does 'or' comparison inside conditional expression
  • ! is a logical NOT
  • && is used to execute sequence of commands only when previous command is successfully executed with a return status of zero.
  • || execute exactly one command in a sequence of commands.
if [ $total -ge 50 -a $total -le 100 ]; then
if [ "$input" == "apple" -o "$input" == "orange" ]; then

passwd=/etc/passwd
group=/etc/group

if [ -f $passwd ] && x=`wc -l $passwd|awk '{print $1}'` && \
   [ -f $group ] && y=`wc -l $group|awk '{print $1}'` && let total=$x+$y
then
    echo "Total Lines in $passwd and $group files are: $total"
fi

[ -f /.datafile ] || touch /.datafile
  • [[ ]] - Extended Test Command is an advanced variation of [ ]
    • Allows pattern matching
    • =~ regular expresison matching
    if [[ $name ~= ^b ]];
for varname in list
do
    commands ##Body of the loop
done

# loop through files and directories
cd ~
for item in *
do
    echo "Item $((i++)) : $item"
done
  • If list is missing from for statement, bash uses positional parameters that were passed into the shell.
# Use range of numbers using brace expansion
for num in {1..10}

# Range of number with increment
for num in {1..10..2}

# Bash for loop using C syntax
for (( expr1; expr2; expr3 ))
do
    commands
done
# While loop
while expression
do
    commands
done

# until loop
until expression
do
    commands #body of the loop
done
  • bash until loop is useful as a way of waiting for certain events to occur.
case var in
pattern1 )
    command1
    command2
    ...
    ;;
pattern2 )
    command3
    command4
    ...
    ;;
esac
  • Each set of command in case statement should end with two semi-colons (;;)

  • case terminator can be ;;, ;& and ;;&

  • Bash has a formatted print command printf

read -a arrayname
read -p prompt
read -s # This doesn't echo the value as you enter
read -t 10 # read waits for 10 sec and times out
read -N 5  # return after reading exactly 5 char
read -r    # read \ literally
read       # if no variable, store value in $REPLY internal variable
  • The bash shell has a hash table where the commands you execute are stored. When you execute the same command next time, bash won't search the PATH variable, instead it will pick up the command from the hash table and run it.

  • Use type command to identify the type of a particular command. A command can be an alias, a shell builtin, a shell keyword, or an external program.

type lw
type pwd
  • ulimit control the amount of resources that can be assigned to processes that are started by bash shell.
# view all current limits
ulimit -a
  • shopt stands for shell options. View all shell options and their current status.

  • truncate is a command line utility that can be found in most Linux distros. It is used to shrink the size of a file to a desired size.

  • The truncation process basically removes all the contents of the file. It does not however remove the file itself, but it leaves it on the disk as a zero byte file. This allows the file to re-used or be continually used by other programs while keeping the overall size in check. This process is also referred to as “zero out a file” or to “empty a file“. The truncation process also will preserve the inode of the file.

  • noclobber is a feature in Linux which is used to prevent accidental overwriting of files. If noclobber is set or enabled on your system, the above I/O redirection methods will throw an error.

$ truncate -s 0 /var/log/syslog

# Empty file using I/O redirection
$ > /var/log/syslog

$ cp /dev/null /var/log/syslog
$ cat /dev/null > /var/log/syslog

Bash substring removal

# Deletes shortest match of substring from front of $string
${string#substring}

# Deletes longest match of substring from front of $string
${string##substring}

Bash Cli shortcut

1. Symbolic Links

  • A symlink (also called a symbolic link) is a type of file in Linux that points to another file or a folder on your computer.

  • A hard link cannot be created for a folder or file in a different file system.

  • A symlink can be created using ln link command. By default ln create hard link. -s flag specifies that the link should be soft link.

ln -s <path to file/folder TO BE LINKED> <path to link TO BE CREATED>

# example
ln -s /home/aamir/flight_test_report.txt tp.txt
ln -s /home/aamir/flight_test_report.txt flight_reports/tp.txt

# symlink for folder
ln -s /home/aamir/workspace ws

# check if a file is symlink
$ ls -l host_ws
lrwxrwxrwx 1 aamir aamir 22 Mar  3 10:09 host_ws -> /home/aamir/workspace/

# Remove symlink
unlink <path-to-symlink>

# Can also use rm
rm <path-to-symlink>

The main benefit of rm over unlink is that we can remove multiple links at once.

# find broken symlink
find /home/aamir -xtype l

# delete broken symlink
find /home/aamir -xtype l -delete

2. Secure Shell (SSH) Server Installation

Table Of Contents

SSH is a cryptographic network protocol used for a secure connection between a client and a server.

$ sudo apt update
$ sudo apt install openssh-server -y

# verify that installation is successful
$ sudo systemctl status ssh

# Allow Uncomplicated Firewall(UFW) firewall
$ sudo ufw allow ssh

# connect to ssh server
$ ssh user_name@ip_address

To determine the public IP address of the machine you’re trying to SSH to, simply visit the following URL: https://api.ipify.org

# To disable SSH
$ sudo systemctl stop ssh

# To start again
$ sudo systemctl start ssh

OpenSSH client-side configuration file is config is stored in .ssh directory under user's home directory. The config file must be readable and writable only by the user and not accessible by others:

$ touch ~/.ssh/config
$ chmod 600 ~/.ssh/config

3. Secure Transfer Files Using SCP

Table Of Contents

SCP (secure copy) allows to securely copy files and directories between two locations.

scp [options] source destination

SCP options
-P: Remote host ssh port -p: Preserve file modification and access time -q: Suppress progress meter and non-error messages -C: Force scp to compress the data -r: copy directories recursively

# Copy local file to remote
$ scp file.txt [email protected]:/remote/directory

# Save file with different name
$ scp file.txt [email protected]:/remote/directory/newfilename.txt

# if remote is listening on different port
$ scp -P 2322 file.txt [email protected]:/remote/directory

# Copy dirctory
$ scp -r /local/directory [email protected]:/remote/directory

# Copy remote file to local system
$ scp [email protected]:/remote/file.txt /local/directory

# Copy file between two remote systems
$ scp [email protected]:/files/file.txt [email protected]:/files

4. find command

Table Of Contents

  • find command can be used to find files by permission, users, groups, file type, date, size and other possible criteria.
# find files in current working directory
$ find . -name test.txt
$ find /home -name test.txt

# ignore case
$ find /home -iname test.txt

# find directories
$ find / -type d -name Test

# find files
$ find . -type f -name test.cpp

# find all cpp files
$ find . -type f -name "*.cpp"

# find file with 777 permission
$ find . -type f -perm 0777 -print

# without 777 permission
$ find / -type f ! -perm 777

# find read only files
$ find . -perm /u=r

# find all executable file
$ find . -perm /a=x

# find all 777 and chmod to 644
$ find / -type f -perm 0777 -print -exec chmod 644 {} \;

# find files by size
$ find / -size +100MB

$ find . -type f ! -name "*.*" | xargs -o rm
$ find . -type f ! -name "*.*" -exec rm {} \;

TBD: https://www.tecmint.com/35-practical-examples-of-linux-find-command/ What are files with SGID bit and Sticky bit?? SUID files

5. vi Editor

Table Of Contents

  • vim is the most popular command-line text editor. vim works in

    • Normal mode,
    • Insert mode,
    • Visual mode,
    • Command mode and
    • Replace mode.
  • To get into Normal mode, press ESC key.

  • Move the cursor

    h move one character left
    j move one row down
    k move one row up
    l move one character right
    4j move 4 rows down
    6k move 6 rows up
  • Basic word movements:
    w move to beginning of next word
    b move to previous beginning of word
    e move to end of word
    W move to beginning of next word after a whitespace
    B move to beginning of previous word before a whitespace
    E move to end of word before a whitespace

    r replace a single character
    x delete a single character
    u undo changes
    ctrl-r undo undo
  • Insert commands include:
    i for ’insert’, this immediately switches vim to insert mode
    a for ’append’, this moves the cursor after the current character and enters insert mode
    o inserts a new line below the current line and enters insert mode on the new line

    I moves the cursor to the beginning of the line and enters insert mode
    A moves the cursor to the end of the line and enters insert mode
    O inserts a new line above the current one and enters insert mode on the new line
  • Visual mode commands
    v enter visual mode, this will also mark a starting selection point
      Move the cursor to the desired end selection point; vim will provide a visual highlight of the text selection

    V enter visual line mode, this will make text selections by line
    <C-V> to enter visual block mode, this will make text selections by blocks; moving the cursor will make rectangle selections of the text

    visual: Enter by pressing v
    block-visual: select any rectangular region. Enter by pressing <ctrl>+v
    linewise-visual: always select full lines. Enter by pressing <shift>+v
  • Command mode has a wide variety of commands and can do things that normal mode can’t do as easily. To enter command mode type ’:’ from normal mode
    : Enters command mode
    % Means across all lines
    s Means substitute
    /foo is regex to find things to replace
    /bar/ is regex to replace things with
    /g means global, otherwise it would only execute once per line

  • Replace mode
In normal mode press ‘R’ (capital R) to enter replace mode

Deleting Lines

    dd delete current line
   5dd delete 5 lines from current line
   :[start],[end]d Delete a range of lines
   :3,5d   delete line 3-5
   . current line
   $ last line
   % All lines
   :.,$d  From current line to end of file
   :.,1d  From current ilne to beginning of file
   %d     Delete all lines

Delete Lines containing pattern

  • The global command (g) tells te delete command (d) to delete all lines containing the <pattern>
:g/<pattern>/d   Delete all lines containing the <pattern>
:g!/<pattern>/d  Delete all lines not containing the <pattern>
:g/^#/d          Delete all comment lines from Bash script
:g/^$/d          Delete all blank lines
:g/^\s*$/d       Delete all blank lines with zero or more whitespace character
  • To delete all lines in the current file use %d command in normal mode.
  • To delete from current line till the end use ,$d.
  • To delete from line 3 to 6 use 3,6d.
  • general form of substitute command
:[range]s/{pattern}/{string}/[flags] [count]

# count is a positive integer that multiplies the command
  • When no range is specified the substitute command operates only in current line.

  • The substitude command looks for the pattern as a string, not a whole word.

  • To browse search history, enter :s and use arrow up/down keys.

command Description
:s (:substitude) find and replace text
:s/pattern/str/g replace all occurrencesd in current line (g flag)
:%s/pattern/str/g search and replace in entire file (% as range)
:%s/pattern//g delete all instance of pattern
:s|pattern|str| Use | instead of /
:%s/pattern/str/gc To confirm each substitution, (c flag)
:%s/^foo.\*/vim/gc regular expression
:3,10s/pat/str/g all lines starting from 3 to 10
:.,$s/pat/str/ current line to last line
:.,+4s/pat/str/g current line and next four lines
:s/pattern/str/gi ignore case (i flag)
:s/pattern/str/gI case sensitive (I flag)
:s/\<pat\>/str/g find whole word pat
:5,20s/^/#/ comment lines from 5 to 20
:5,20s/^#// uncomment lines from 5 to 20
:%s/ab|cd/aa replace ab or cd with aa
:%s/\s\+$//g remove trailing whitespace at the end of each line

Useful tips

S (SHIFT+s) Start writing on a line at correct indentation
%           Jump to matching bracket/brace
>>, <<      Indent one line
>, <        Indent multiple lines
=G          Fix indentation in the whole file
:tabnew     Create a new tab
gt          Go to next tab
gT          Go to previous tab
:tabo       Close all other tabs beside the active one

6. Usefull Commands

Killin a bunch of processes

# Kill processes
$ kill -9 $(ps -aux | grep main.py | awk '{print $2}')

# or better use pgrep and pkill
$ kill -9 $(pgrep main.py)

# remove grep output from ps output
$ ps -aux | grep main.* | grep -v grep

Generate a sequence of numbers

#
$ seq LAST
$ seq FIRST LAST
$ seq FIRST INCREMENT LAST
$ seq 10  # 1 to 10
$ seq 0 2 10  # 0 2 4 6 8 10

Check OS name and version

$ cat /etc/os-release

# Linux kernel version
$ uname -r

xargs

  • xargs is used to build and execute command lines from standard input. While tools like grep can accept standard input as parameters, many other tools cannot. Using xargs allows tools like echo, rm, mkdir etc to accept standard input as arguments.

  • xargs reads items from standard iput as separated by blanks and executes a command once for each argument.

$ echo "one two three" | xargs mkdir
  • Use -d option to change delimeter

  • The most common usage of xargs is to use it with find command.

# find files older than two weeks
$ find /tmp -mtime +14 | xargs rm
  • find command supports the -exec option that allow arbitrary commands to be performed on found files.

  • -t option print each command that will be executed.

$ echo "one two three" | xargs -t rm
  • -p option will print the command to be executd and prompt the user to run it.

  • Use -I option to run multiple commands.

$ echo "one two three" | xargs -I % sh -c 'echo %; mkdir %'

Create multiple files and directories

$ mkdir {cs,files,masters,draft,static}   # directories.
$ touch -- 'file with spaces' '-a' '-l' 'filename'    # And some files:

Create Application Icon in Ubuntu

vi anaconda-navigator.desktop

#!/usr/bin/env xdg-open
[Desktop Entry]
Name=Anaconda
Version=2.0
Type=Application
Exec=/path/to/anaconda-navigator
Icon=/path/to/selected/icon
Comment=Open Anaconda Navigator
Terminal=false

mv anaconda-navigator.desktop ~/.local/share/applications/

Debian

  • Creating and merging patches in Linux
$ diff -u file.old file.new >file.patch

$ patch -p0 file.old <file.patch
  • git diff will generate a file in the same format as what diff -u would do and git apply can do the same as patch.

  • Version control tools and their companies

    subversion (svn) => Apache git => bazaar (bzr) => canonical mercurial (hg) =>

  • /etc/debian_version contains the version number for the installed Debian system.

  • The apt-show-versions program (from the Debian package of the same name) checks the list of installed packages and identifies the available versions.

  • /etc/apt/sources.list file (and /etc/apt/sources.list.d/ directory) will show where the installed Debian packages likely came from.

  • Ext4, the default filesystem for Debian, is a good compromise, based on the three previous versions of filesystems historically used in Linux (ext, ext2 and ext3). Ext4 overcomes certain limitations of ext3 and is particularly appropriate for very large capacity hard drives

Bash

  • POSIX standardized the Unix libraries and utilities, including the shell. The standard shell was primarily based on the 1988 version of the Korn shell, with some C shell features and a bit of invention to fill in the gaps. bash was begun as part of the GNU Project’s effort to produce a complete POSIX system.

  • pwd -P displays your physical location

  • cd -P dir take symbolic link dir and navigate to physical location.

  • The bash builtin type command searches your environment (including aliases, keywords, functions, builtins, directories in $PATH, and the command hash table) for executable commands matching its arguments and displays the type and location of any matches.

  • type -a <cmd-name> display all matches.

  • The which command is similar but only searches your $PATH (and csh aliases).

  • apropos searches manpage names and descriptions for regular expressions supplied as arguments. This is the same as man -k:

CMD Description
ls -A Skip . and ..
ls -F Show type of file with trailing type designators
ls -r Reverse sort order
ls -R Recurse through subdirectories
ls -S Sort by file size
ls -d list information about directory itself, rather than list contents of directory
ls -C to preserve formatting when redirecting ls output
  • Use a backslash in front of the command name to avoid any alias. \ls -d .v*/

  • In echo *, * is expanded by the shell to everything in the current directory, which results in a list similar to what you’d get with ls.

  • Unquoted text and even text enclosed in double quotes is subject to shell expansion and substitution. Enclose a string in single quotes unless it contains elements that you want the shell to interpolate.

  • enable -a will list all builtins and their enabled or disabled status.

  • help displays help about shell builtins. help cd.

  • $- is a string listing of all the current shell option flags. On Ubuntu it displays himBHs

option desciption
h Cache location of binaries in the $PATH. Speeds up execution, but fails if you move binaries around during the shell session.
i The current shell is interactive
m Job control is enabled
B Brace expansion is enabled
H History substitution like !-1
  • Ubuntu Personal Package Archive (PPA).

  • If you use double quotes (""), some shell substitutions do take place (variable, arithmetic, and tilde expansions and command substitutions).

  • printf is a builtin command which can be use for more control over the formatting and placement of output.

  • echo -n or echo -e 'hi\c' produce output without default newline that echo provides.

  • redirect standard output and standard error to same file

$ both >& outile  # OR
$ both &> outfile # OR
$ both > outfile 2>&1 # older, more verbose
$ both 2> outfile 1>&2

# redirect both STDERR and STDOUT and append them to the specified file.
$ ls &>> /tmp/ls.out
  • head and tail, along with cat, grep, sort, cut, and uniq, are some of the most commonly used Unix text processing tools out there.

  • tail -n +3 lines skip first 2 lines of the file

$ tail -n 10 file # offset relative to end of file
$ tail -n +10 file # offset relative to start of file
  • Use braces ({ }) to group these commands together; then redirection applies to the output from all commands in the group.
{ pwd; ls; cd ../elsewhere; pwd; ls; } > /tmp/all.out
  • () tell bash to run the commands in a subshell, then redirect the output of the entire subshell’s execution.
(pwd; ls; cd ../elsewhere; pwd; ls) > /tmp/all.out
  • Use the tee command to split the output into two identical streams, one that is written to a file and the other that is written to standard output, so as to continue the sending of data along the pipes.

  • The tee command writes the output to the filename(s) specified as its parameter and also writes that same output to standard out.

$ uniq | tee /tmp/uniq.txt | awk -f transform.awk
$ find / -name '*.c' -print 2>&1 | tee /tmp/all.c.src
  • The $() encloses a command that is run in a subshell. The output from that command is substituted in place of the $() phrase. Newlines cause the output to become several parameters on the command line.

  • One important difference between standard output and standard error is that standard output is buffered but standard error is unbuffered; that is, every character is written individually, and they aren’t collected together and written as a bunch.

  • As of the 4.x versions of bash, there is a shortcut syntax for redirecting both standard output and standard error into a pipe.

# redirect both stdin and stderr
$ somecmd |& othercmd
  • tee -a append instead of replace.

  • The noclobber option tells bash not to overwrite any existing files when you redirect output.

$ set -o noclobber # don't override
$ set +o noclobber # override existing file
  • Use >| to redirect your output. Even if noclobber is set, bash ignores its setting and overwrites the file.

  • Use a here-document with the << characters, redirecting the text from the command line rather than from a file. When put into a shell script, the script file then contains the data along with the script.

$ cat ext
#
# here is a "here" document
#
grep $1 <<EOF
mike x.123
joe x.234
sue x.555
pete x.818
sara x.822
bill x.919
EOF
$

$ ext bill
bill x.919
  • EOF is just an arbitrary string (you can choose what you like)

  • <<EOF can be replaced with <<\EOF, or <<'EOF', or even <<E\OF to tell bash to stop parameter expansion, command substitution and arithmetic expansion.

  • Use <<-, and then you can use tab characters (only!) at the beginning of lines to indent this portion of your shell script.

  • In its simplest form, a read statement with no arguments will read user input and place it into the shell variable REPLY.

$ read -p # print prompt
$ read -t # set timeout in seconds
$ read -s # don't echo the typed char
  • Use bash’s builtin select construct to generate a menu, then have the user choose by typing the number of the selection
stty -echo # turn echo off
stty sane  # turn on echo
  • The shell variable $? is set with a nonzero value if the command fails. We recommend using only 0 to 127 because the shell uses 128+N to denote killed by signal N.

  • (( )) evaluates an arithmetic expression

cd mytmp
if (( $? == 0 )); then rm * ; fi

if (( 4+15 == 9 )); then echo "match"; else echo "not match"; fi
  • Separating two commands by the double ampersands tells bash to run the first command and then to run the second command only if the first command succeeds.
$ (( 1 )) && echo "hello" # output: hello
$ (( 0 )) && echo "hello" # output: -
  • set -e will cause the shell to exit when a command fails.

  • If you want to run a job in the background and expect to exit the shell before the job completes, then you need to nohup the job.

  • The nohup command simply sets up the child process to ignore hangup signals. You can still kill the job with the kill command, because kill sends a SIGTERM signal, not a SIGHUP signal.

  • You don’t use the dollar sign on the variable name to assign it a value, but you do use the dollar sign to get the value of the variable. (The exception to this is using variables inside a $(( )) expression.)

  • Perl usually comes with pod2* programs to convert POD to HTML, LaTeX, manpage, text, and usage files.

  • If you want to see a list of all the exported variables, just type the command env (or use the builtin export -p) for a list of each variable and its value.

  • Use the shell special variable $* to refer to all of your arguments, and use that in a for loop

ls $1 # treated as ls my
ls ${1} => "my c file" treated as ls my c file
ls "${1}" => "my c file" is treated as a single variable
  • Difference between $* &amp; $@
# input
music.mp3 another music.mp3 third.mp3
script *.mp3

$*    # music.mp3 another music.mp3 third.mp3
"$*"  # "music.mp3 another music.mp3 third.mp3"
"$@"  #

${#}       # gives the number of arguments
${#VAR}    # gives the length of the value in the variable VAR
${VAR#alt} # does a certain kind of substitution
  • Left unquoted, $* and $@ give you the same thing. A reference to $* inside of quotes gives the entire list inside one set of quotes, as we just saw. But a reference to $@ inside of quotes returns not one string but a list of quoted strings, one for each argument.

  • Use shift to remove an argument after you’ve handled it.

  • Use the ${:-} syntax when referring to the parameter, and use it to supply a default value.

  • Use the assignment operator in the shell variable reference the first time you refer to it to assign a value to the variable if it doesn’t already have one

FILEDIR=${1:-/tmp}
cd ${HOME:=/tmp} # doesn't work with positional variables
${1:-default}
cd ${BASE:="$(pwd)"}
  • Parameter expansion means that we could use other shell variables in this expression, as in ${BASE:=${HOME}}.

  • Tilde expansion means that we can use an expression like ~bob, and it will expand that to refer to the home directory of the user bob.

  • Command substitution will run the commands and take their output as the value for the variable. Commands are enclosed in the single parentheses syntax, $(cmds).

  • Arithmetic expansion means that we can do integer arithmetic, using the $(( )) echo ${BASE:=/home/uid$((ID+1))}

Inside ${...} Action taken
name:number:number Return a substring of name starting at number with length number
#name Return length of string
name#pattern Remove (shortest) front-anchored pattern
name##pattern Remove (longest) front-anchored pattern
name%pattern Remove (shortest) rear-anchored pattern
name%%pattern Remove (longest) rear-anchored pattern
name/pattern/string Replace first occurence
name//pattern/string Replace all occurances
  • MYVAR=${MYVAR#-} remove -ve sign from the number

  • FILE=${FULLPATHOFFILE##*/} get the basename without using basename external executable.

FILE=$(basename $MYIMAGEFILE .jpg)

# same as above
FILE=${MYIMAGEFILE%/} # remove trailing slash
FILE=${FILE##*/}      # remove everything upto last /
FILE=${FILE%.jpg}     # remove .jpg suffix if present
command description
basename
dirname
  • Bash array initialization
MYARRAY=(first second third fourth)

echo ${MYARRAY[0]} and ${MYARRAY[2]}

${FN,,} # convert FN value to all lowercase
${FN^^} # convert FN value to all uppercase
${FN~~} # toggle case
declare -l lc
declare -u UP
declare -c Ca

while read TXT; do
	UP="${TXT}"
	lc="${TXT}"
	Ca="${TXT}"

	echo $TXT $UP $lc $Ca
done
  • ${ARRAY[@]} references all the elements of the array at once.

  • Use $(( )) or let for integer arithmetic expressions.

COUNT=$((COUNT + 1))
let COUNT+='5'
  • let statement is a bash builtin and its arguments will undergo word expansion.

  • The $ sign is not needed inside the double parentheses $(( )).

  • In bash, the semicolon serves the same purpose as a newline—it ends a statement.

  • The double parentheses (( )) are strictly for arithmetic expressions. The square brackets can also test for file characteristics, but the syntax is much less streamlined for arithmetic expressions.

  • Use the operators for logical AND (-a) and OR (-o) to combine more than one test in an expression.

# both read and write
if [ -r $FILE -a -w $FILE ]

if [ -r "$FN" -a \( -f "$FN" -o -p "$FN" \) ]
  • Use the -eq operator for numeric comparisons and the equality primary = (or ==) for string comparisons.

  • This is the opposite of Perl, in which eq, ne, etc. are the string operators, while ==, !=, etc. are numeric.

  • Use the double-bracket compound statement in an if statement to enable shell-style pattern matches on the righthand side of the equality operator:

if [[ "${MYFILENAME}" == *.jpg ]]
  • in the double-bracket if [[ ]] syntax the equals sign is a more powerful string comparator.
shopt -s extglob
if [[ "$FN" == *.@(jpg|jpeg) ]]
  • The shopt -s command is the way to turn on shell options. The extglob option deals with extended pattern matching (or globbing).

  • Use the regular expression matching of the =~ operator. Once it has matched the string, the various parts of the pattern are available in the shell variable $BASH_REMATCH.

while read lineoftext
do
    process that line
done < file.input

# or using cat
cat file.input |
while read lineoftext
do
    process that line
done
for (( i=0 ; i < 10 ; i++ )) ; do echo $i ; done
  • Use the seq command to generate your floating-point values
for fp in $(seq 1.0 .01 1.1); do
    echo $fp;
done

# or this one, which is more efficient then above one
# as seq and while will be run in parellel
seq 1.0 .01 1.1 | while read fp; do echo $fp; done
  • Use the case statement for a multiway branch
case $FN in
    *.gif) gif2png $FN
        ;;
    *.png) pngOK $FN
        ;;
    *.jpg) jpg2gif $FN
        ;;
    *.tif | *.TIFF) tif2jpg $FN
        ;;
    *) printf "File not supported: %s" $FN
        ;;
esac
  • The select statement will display input list of words, each preceded by a number, and the user will be prompted for input.

  • The bash environment variable $PS3 is the prompt used by select.

grep options description
-h don't display filename
-c just count, not actual lines
-l display just the filename
-q quite output
-i ignore case
-v Invert the search
  • !! history operator lets you repeat the previous command without retyping it.

  • zgrep, zcat, or gzcat

  • zgrep is simply a grep that understands various compressed and uncompressed file types

  • awk has a built-in variable called NF that holds the number of fields found on the current line, $NF always refers to the last field.

ls -l | awk '/^total/{next} {sum += $5}; END {print sum}'
  • For any line of input matching that regex, the associated block of code will be executed.

  • The next command, which ends processing on this line of input and starts over with next line of input.

  • awk’s associative arrays are same as hashes or dictionaries in other languages

sort option description
sort -r sort in reverse order
sort -f fold lower- and uppercase (ignore case)
sort -n data as numbers
sort -u remove duplicate
sort -t . use . char as separator between fields
  • A stable sort preserves the original order in the sorted data when the sort fields are equal. Linux and Solaris do not default to a stable sort, but NetBSD does.

  • cut -c12-15 cut from column 12 to 15

  • cut -c58- from column 58 till end of line

$ tar cf tarball_name.tar directory_of_files
$ gzip tarball_name.tar
  • GNU tar use -Z for compress (obsolete), -z for gzip (safest) or -j for bzip2 (highest compression)
$ tar czf tarball.tgz directory_of_files
$ zip -r zipfile_name directory_of_files
# list contents
$ tar tf tarball.tar

# extract contents
$ tar xf tarball.tar

# GNU tar, list contents
$ tar tzf tarball.tar.gz

# GNU tar, extract contents
$ tar xzf tarball.tar.gz
  • Some tar programs strip the leading / by default (e.g., GNU tar) or optionally. That’s a much safer way to create a tarball.

  • Use the tr command to translate one character to another.

tr ';' ',' <be.fore >af.ter
tr 'A-Z' 'a-z'
echo "hello; hope ur fine: What's ur name?" | tr ';:.!?' ',*'
  • a semicolon has special meaning to bash, so if we didn’t quote it bash would break our command into two commands, resulting in an error.

  • Just make sure that both arguments end up with the same number of characters. If the second argument is shorter, its last character will be repeated to match the length of the first argument. If the first argument is shorter, the second argument will be truncated to match the length of the first.

  • a very simplistic encoding of a text message using a simple substitution cypher that offsets each character by 13 places (i.e., ROT13). An interesting characteristic of ROT13 is that the same process is used to both encipher and decipher the text.

  • Use the -d option on tr to delete the character(s) in the supplied list.

  • fmt and pr command

  • use the $LESS variable with ~/.lessfilter and ~/.lesspipe files. less takes options from the $LESS variable,

find . -name '*.mp3' -print -exec mv '{}' ~/songs \;