Bash
From Attie's Wiki
Contents |
Useful Arguments
-e |
treat any non-zero return as an error, and quit |
-u |
treat any use of an unset variable as an error, and quit |
-v |
print unexpanded commands as they are executed |
-x |
print expanded commands as they are executed |
example
#!/bin/bash -eu echo "hi" false echo "there"
Built-in Variables
$? |
The return code of the last process |
$! |
The PID of the most recently started process |
Print Commands
# print commands as they are executed set -v # print commands as they are executed, but expand the variables before printing set -x
Command Not Found
function command_not_found_handle { echo "Hello" return 127 }
Heredoc Without Expanding Variables
cat - > /etc/profile.d/mymotd.sh <<'EOF' hostname=`uname -n` echo -e "Hostname is $hostname" EOF
Line-by-Line
Use the IFS variable (Internal Field Separator), or:
ls -l | while read -r line; do echo $line done
Maths
a=5 b=3 c=$((a+b)) echo $a $b $c
Function List
Get a list of Perl functions, with their line number
cat gitweb.cgi | grep -n ^sub | sed -re 's/([0-9]+): ?sub ([^ {]+).*/\1:\t\2/' | sort -k2 | less
Parent's PID
function ppid { ps -Af 2>/dev/null | awk -v mypid=$(echo $$) '{if($2==mypid)print $3}' }
runSplice
This will run the given command, n times, where n is the number of records in the given files. Each iteration will run the given command, appending the line from file A followed to the line from file B.
function runSplice { IFS=$'\n' A=( $(cat $1) ); shift B=( $(cat $1) ); shift unset IFS ALEN=${#A[@]} BLEN=${#B[@]} if [ ${ALEN} -ne ${BLEN} ]; then echo "Files don't have the same numer of records..." return 1 fi echo "Running splice on ${ALEN} records" echo "Command: $@ {A} {B}" for i in $(seq 0 $((${ALEN} - 1))); do echo "$@ \"${A[${i}]}\" \"${B[${i}]}\"" $@ "${A[${i}]}" "${B[${i}]}" done }
Say you have a list of files a b c
, and you want to rename them A B C
. You would of course have more files to do than 3...
Do the following:
Contents of a.list
a b c
Contents of b.list
A B C
Now execute:
runSplice a.list b.list mv
The following commands will be executed for you:
mv "a" "A" mv "b" "B" mv "c" "C"
If conditions
Primary | Meaning |
---|---|
[ -a FILE ] |
True if FILE exists. |
[ -b FILE ] |
True if FILE exists and is a block-special file. |
[ -c FILE ] |
True if FILE exists and is a character-special file. |
[ -d FILE ] |
True if FILE exists and is a directory. |
[ -e FILE ] |
True if FILE exists. |
[ -f FILE ] |
True if FILE exists and is a regular file. |
[ -g FILE ] |
True if FILE exists and its SGID bit is set. |
[ -h FILE ] |
True if FILE exists and is a symbolic link. |
[ -k FILE ] |
True if FILE exists and its sticky bit is set. |
[ -p FILE ] |
True if FILE exists and is a named pipe (FIFO). |
[ -r FILE ] |
True if FILE exists and is readable. |
[ -s FILE ] |
True if FILE exists and has a size greater than zero. |
[ -t FD ] |
True if file descriptor FD is open and refers to a terminal. |
[ -u FILE ] |
True if FILE exists and its SUID (set user ID) bit is set. |
[ -w FILE ] |
True if FILE exists and is writable. |
[ -x FILE ] |
True if FILE exists and is executable. |
[ -O FILE ] |
True if FILE exists and is owned by the effective user ID. |
[ -G FILE ] |
True if FILE exists and is owned by the effective group ID. |
[ -L FILE ] |
True if FILE exists and is a symbolic link. |
[ -N FILE ] |
True if FILE exists and has been modified since it was last read. |
[ -S FILE ] |
True if FILE exists and is a socket. |
[ FILE1 -nt FILE2 ] |
True if FILE1 has been changed more recently than FILE2, or if FILE1 exists and FILE2 does not. |
[ FILE1 -ot FILE2 ] |
True if FILE1 is older than FILE2, or is FILE2 exists and FILE1 does not. |
[ FILE1 -ef FILE2 ] |
True if FILE1 and FILE2 refer to the same device and inode numbers. |
[ -o OPTIONNAME ] |
True if shell option "OPTIONNAME" is enabled. |
[ -z STRING ] |
True of the length if "STRING" is zero. |
[ -n STRING ] or [ STRING ] |
True if the length of "STRING" is non-zero. |
[ STRING1 == STRING2 ] |
True if the strings are equal. "=" may be used instead of "==" for strict POSIX compliance. |
[ STRING1 != STRING2 ] |
True if the strings are not equal. |
[ STRING1 < STRING2 ] |
True if "STRING1" sorts before "STRING2" lexicographically in the current locale. |
[ STRING1 > STRING2 ] |
True if "STRING1" sorts after "STRING2" lexicographically in the current locale. |
[ ARG1 OP ARG2 ] |
"OP" is one of -eq, -ne, -lt, -le, -gt or -ge. These arithmetic binary operators return true if "ARG1" is equal to, not equal to, less than, less than or equal to, greater than, or greater than or equal to "ARG2", respectively. "ARG1" and "ARG2" are integers. |
Parse JSON
Note: This script may require the following argument added to the awk instances:
-F ''
#!/bin/bash cat json_file | \ # extract the top-most objects from the JSON fold -1 | \ awk 'BEGIN{ depth=0; } { if ($1 == "{") depth++; else if ($1 == "}") { depth--; if (depth == 0) printf "}\n"; } if (depth > 0) printf $1; } END{ printf "\n" }' | \ # strip of the curly braces from each line sed -re 's/^\{//' -e 's/\}$//' | \ # process line-by-line while read -r line; do # skip blank lines if [ "${line}" == "" ]; then continue fi echo "JSON Line: ${line}" # get a list of values/elements (take into account the posibility of a comma _inside_ a string items=$(echo "${line}" | \ fold -1 | \ awk 'BEGIN{ t = 0; q = 0; e = 0; } { if (q > 0) { if (!e && $1 == "\"") q--; } else if (t > 0) { if (!e && $1 == "'\''") t--; } else if (!e && $1 == "\"") { q++; } else if (!e && $1 == "'\''") { t++; } if (!q && !t && $1 == ",") { printf "\n"; } else { printf $1; } if ($1 == "\\") e = 1; else e = 0; } END{ printf "\n" }') echo "${items}" | while read -r item; do echo " JSON Item: ${item}" value=$(echo "${item}" | \ fold -1 | \ awk 'BEGIN{ q=0; e=0; f=0; } { if (f) { printf $1; } else if (!e) { if ($1 == "\"") { if (!q) { q = 1; } else { q = 0; } } else if ($1 == ":") { f = 1; } } if ($1 == "\\") { e = 1; } else { e = 0; } }' | \ sed -re 's/^"//' -e 's/"$//') echo " Value: ${value}" done done