mirror of
https://gitea.com/mcereda/oam.git
synced 2026-02-08 21:34:25 +00:00
11 KiB
11 KiB
Bourne Again SHell
Table of contents
- TL;DR
- Startup files loading order
- Functions
- Substitutions
- Here documents
- Keys combinations
- Check if a script is sourced by another
- Gotchas
- Further readings
- Sources
TL;DR
Shortcuts:
| Shortcut | Action |
|---|---|
!! |
Insert the previous command in the current prompt. |
Alt + . |
Insert the last argument in the current prompt. |
Ctrl + L |
Clear the terminal. |
Ctrl + R |
Search backwards in the history one step at a time. |
Ctrl + Z |
Send the current foreground task to background. |
Get help:
# Get a brief summary about commands.
help 'nano'
# Get detailed information about commands.
man 'parallel'
Session management:
# Clean the console.
clear
# Print the current directory.
pwd
# Change the current directory.
cd
cd /bin
cd ..
# Create local variables.
VAR_NAME="value"
# Convert local variables in environment ones.
export VAR_NAME
export VAR_NAME="value"
# Deletes variables.
unset MY_FRIENDS
# Add directories to the current executables locations.
export PATH="${PATH}:/home/user/bin"
export PATH="/home/user/bin:${PATH}"
# Show the path of executables in $PATH.
which 'redis-cli'
# Show the path, man pages, source code, etc of executables in $PATH.
whereis nano
# List existing aliases.
alias
# Create aliases.
alias redo='$(history -p !!)'
# Remove aliases.
unalias redo
# Print all environment variables.
env
# Print all local *and* environment variables.
set
( set -o posix ; set )
# Print exported variables only.
export -p
# Logout after 3 minutes of inactivity.
TMOUT=180
Piping:
# Use the output of a command as the input of another.
tail 'file.txt' | grep 'search'
# Save the output of command 'a' as 'file.txt'.
# This *overwrites* already existing files with that name.
a > 'file.txt'
# Append the output of command 'a' to 'file.txt'.
a >> 'file.txt'
Arrays:
# Declare arrays.
ARRAY=(
"first_element"
"second_element" "nth_element"
)
# Get the length of arrays.
# A.K.A. number of elements.
ARRAY_LEN=${#ARRAY[@]}
# Access all elements in arrays with referencing.
echo ${ARRAY[@]}
echo ${ARRAY[*]}
# Access the last value of arrays.
echo ${ARRAY[-1]}
echo ${ARRAY: -1}
# Get a slice of 4 elements from an array.
# Start from the element with index number 2.
echo ${ARRAY:2:4}
Functions:
# Declare functions.
functionName () { … }
function functionName { … }
# Declare functions on a single line.
functionName () { command1 ; … ; command N ; }
# Get all the arguments in input.
echo $@
Error management:
# Run a command or function on exit, kill or error.
trap "rm -f $tempfile" EXIT SIGTERM ERR
trap function-name EXIT SIGTERM ERR
# Disable CTRL-C.
trap "" SIGINT
# Re-enable CTRL-C.
trap - SIGINT
Job control:
# Print a list of background tasks.
jobs
# Bring a background task in the foreground.
fg
fg 'task_number'
Other snippets:
# Copy and paste *on Linux*.
echo "Hello my friend!" | xclip \
&& xclip -o >> pasted_text.txt
# Copy and paste *on Darwin*.
echo "Hello my friend!" | pbcopy \
&& pbpaste >> pasted_text.txt
# Of all the arguments in input, return only those which are existing directories.
DIRECTORIES=()
for (( I = $# ; I >= 0 ; I-- )); do
if [[ -d ${@[$I]} ]]; then
DIRECTORIES+=${@[$I]}
else
local COMMAND="${@:1: 1-$I}"
break
fi
done
# Bash 3 and `sh` have no built-in means to convert case of a string, but the
# `awk`, `sed` or `tr` tools can be used instead.
echo $(echo "$name" | tr '[:upper:]' '[:lower:]' )
echo $(tr '[:upper:]' '[:lower:]' <<< "$name")
# Bash 5 has a special parameter expansion for upper- and lowercasing strings.
echo ${name,,}
echo ${name^^}
# Leverage brace expansion to write less duplicated stuff.
mv /tmp/readme.md{,.backup} # = mv /tmp/readme.md /tmp/readme.md.backup
cp a{1,2,3}.txt backup-dir # = cp a1.txt a2.txt a3.txt backup-dir
cp a{1..3}.txt backup-dir # = cp a1.txt a2.txt a3.txt backup-dir
# Add a clock to the top-right part of the terminal.
while sleep 1
do
tput sc;
tput cup 0 $(($(tput cols)-29))
date
tput rc
done &
# Show a binary clock.
watch -n 1 'echo "obase=2; $(date +%s)" | bc'
# Fork bomb.
:(){ :|: & };:
Startup files loading order
On startup:
- (if login shell)
/etc/profile - (if interactive and non login shell)
/etc/bashrc - (if login shell)
~/.bash_profile - (if login shell and
~/.bash_profilenot found)~/.bash_login - (if login shell and no
~/.bash_profilenor~/.bash_loginfound)~/.profile - (if interactive and non login shell)
~/.bashrc
Upon exit:
- (if login shell)
~/.bash_logout - (if login shell)
/etc/bash_logout
Functions
A function automatically returns the exit code of the last command in it.
Substitutions
!! (command substitution)
Substitutes !! with the last command in your history
$ echo 'hallo!'
hallo!
$ !!
echo 'hallo!'
hallo!
$ sudo !!
sudo echo 'hallo!'
[sudo] password for user:
hallo!
^^ (caret substitution)
Re-runs a command replacing a string.
$ sudo apt search tmux
…
$ ^search^show
sudo apt show tmux
…
Here documents
A Here document (heredoc) is a type of redirection that allows you to pass multiple lines of input to a command.
[COMMAND] <<[-] 'DELIMITER'
HERE-DOCUMENT
DELIMITER
- the first line must start with an optional command followed by the special redirection operator
<<and the delimiting identifier - one can use any string as a delimiting identifier, the most commonly used being
EOForEND - if the delimiting identifier is unquoted, the shell will substitute all variables, commands and special characters before passing the here-document lines to the command
- appending a minus sign to the redirection operator (
<<-), will cause all leading tab characters to be ignored
this allows one to use indentation when writing here-documents in shell scripts
leading whitespace characters are not allowed, only tabs are - the here-document block can contain strings, variables, commands and any other type of input
- the last line must end with the delimiting identifier
white space in front of the delimiter is not allowed
$ cat << EOF
The current working directory is: $PWD
You are logged in as: $(whoami)
EOF
The current working directory is: /home/user
You are logged in as: user
$ cat <<-'EOF' | sed 's/l/e/g' > file.txt
Hello
World
EOF
$ cat file.txt
Heeeo
Wored
Keys combinations
Ctrl+L: clear the screen (instead of typingclear)Ctrl+R: reverse search your Bash history for a command that you have already run and wish to run again
Check if a script is sourced by another
(return 0 2>/dev/null) \
&& echo "this script is not sourced" \
|| echo "this script is sourced"
Gotchas
Exist statuses of killed commands
The exit status of a killed command is 128 + n if the command was killed by signal n:
$ pgrep tryme.sh
880
$ kill -9 880
$ echo $?
137
Go incognito
See How do I open an incognito bash session on unix.stackexchange.com
HISTFILE=
You can also avoid recording a single command simply preceding it with space
echo $RECORDED
echo $NOT_RECORDED
Further readings
Sources
All the references in the further readings section, plus the following: