Skip to content

Instantly share code, notes, and snippets.

@unixfox
Last active February 7, 2021 15:49
Show Gist options
  • Save unixfox/f84a7de927fb587eb4cdf9ba3cfc756e to your computer and use it in GitHub Desktop.
Save unixfox/f84a7de927fb587eb4cdf9ba3cfc756e to your computer and use it in GitHub Desktop.
nench but without CPU test and speedtest
#!/usr/bin/env bash
##########
# nench.sh ("new bench.sh")
# =========================
# current version at https://github.com/n-st/nench
# - loosely based on the established freevps.us/bench.sh
# - includes CPU and ioping measurements
# - reduced number of speedtests (9 x 100 MB), while retaining useful European
# and North American POPs
# - runs IPv6 speedtest by default (if the server has IPv6 connectivity)
# Run using `curl -s bench.wget.racing | bash`
# or `wget -qO- bench.wget.racing | bash`
# - list of possibly required packages: curl,gawk,coreutils,util-linux,procps,ioping
##########
command_exists()
{
command -v "$@" > /dev/null 2>&1
}
Bps_to_MiBps()
{
awk '{ printf "%.2f MiB/s\n", $0 / 1024 / 1024 } END { if (NR == 0) { print "error" } }'
}
B_to_MiB()
{
awk '{ printf "%.0f MiB\n", $0 / 1024 / 1024 } END { if (NR == 0) { print "error" } }'
}
redact_ip()
{
case "$1" in
*.*)
printf '%s.xxxx\n' "$(printf '%s\n' "$1" | cut -d . -f 1-3)"
;;
*:*)
printf '%s:xxxx\n' "$(printf '%s\n' "$1" | cut -d : -f 1-3)"
;;
esac
}
finish()
{
printf '\n'
rm -f test_$$
exit
}
# make sure the dd test file is always deleted, even when the script is
# interrupted while dd is running
trap finish EXIT INT TERM
command_benchmark()
{
if [ "$1" = "-q" ]
then
QUIET=1
shift
fi
if command_exists "$1"
then
( time "$gnu_dd" if=/dev/zero bs=1M count=500 2> /dev/null | \
"$@" > /dev/null ) 2>&1
else
if [ "$QUIET" -ne 1 ]
then
unset QUIET
printf '[command `%s` not found]\n' "$1"
fi
return 1
fi
}
dd_benchmark()
{
# returns IO speed in B/s
# Temporarily override locale to deal with non-standard decimal separators
# (e.g. "," instead of ".").
# The awk script assumes bytes/second if the suffix is !~ [TGMK]B. Call me
# if your storage system does more than terabytes per second; I'll want to
# see that.
LC_ALL=C "$gnu_dd" if=/dev/zero of=test_$$ bs=64k count=16k conv=fdatasync 2>&1 | \
awk -F, '
{
io=$NF
}
END {
if (io ~ /TB\/s/) {printf("%.0f\n", 1000*1000*1000*1000*io)}
else if (io ~ /GB\/s/) {printf("%.0f\n", 1000*1000*1000*io)}
else if (io ~ /MB\/s/) {printf("%.0f\n", 1000*1000*io)}
else if (io ~ /KB\/s/) {printf("%.0f\n", 1000*io)}
else { printf("%.0f", 1*io)}
}'
rm -f test_$$
}
download_benchmark()
{
curl --max-time 10 -so /dev/null -w '%{speed_download}\n' "$@"
}
if ! command_exists curl
then
printf '%s\n' 'This script requires curl, but it could not be found.' 1>&2
exit 1
fi
if command_exists gdd
then
gnu_dd='gdd'
elif command_exists dd
then
gnu_dd='dd'
else
printf '%s\n' 'This script requires dd, but it could not be found.' 1>&2
exit 1
fi
if ! "$gnu_dd" --version > /dev/null 2>&1
then
printf '%s\n' 'It seems your system only has a non-GNU version of dd.'
printf '%s\n' 'dd write tests disabled.'
gnu_dd=''
fi
printf '%s\n' '-------------------------------------------------'
printf ' nench.sh v2019.07.20 -- https://git.io/nench.sh\n'
date -u '+ benchmark timestamp: %F %T UTC'
printf '%s\n' '-------------------------------------------------'
printf '\n'
if ! command_exists ioping
then
curl -s --max-time 10 -o ioping.static http://wget.racing/ioping.static
chmod +x ioping.static
ioping_cmd="./ioping.static"
else
ioping_cmd="ioping"
fi
# Basic info
if [ "$(uname)" = "Linux" ]
then
printf 'Processor: '
awk -F: '/model name/ {name=$2} END {print name}' /proc/cpuinfo | sed 's/^[ \t]*//;s/[ \t]*$//'
printf 'CPU cores: '
awk -F: '/model name/ {core++} END {print core}' /proc/cpuinfo
printf 'Frequency: '
awk -F: ' /cpu MHz/ {freq=$2} END {print freq " MHz"}' /proc/cpuinfo | sed 's/^[ \t]*//;s/[ \t]*$//'
printf 'RAM: '
free -h | awk 'NR==2 {print $2}'
if [ "$(swapon -s | wc -l)" -lt 2 ]
then
printf 'Swap: -\n'
else
printf 'Swap: '
free -h | awk '/Swap/ {printf $2}'
printf '\n'
fi
else
# we'll assume FreeBSD, might work on other BSDs too
printf 'Processor: '
sysctl -n hw.model
printf 'CPU cores: '
sysctl -n hw.ncpu
printf 'Frequency: '
grep -Eo -- '[0-9.]+-MHz' /var/run/dmesg.boot | tr -- '-' ' ' | sort -u
printf 'RAM: '
sysctl -n hw.physmem | B_to_MiB
if [ "$(swapinfo | wc -l)" -lt 2 ]
then
printf 'Swap: -\n'
else
printf 'Swap: '
swapinfo -k | awk 'NR>1 && $1!="Total" {total+=$2} END {print total*1024}' | B_to_MiB
fi
fi
printf 'Kernel: '
uname -s -r -m
printf '\n'
printf 'Disks:\n'
if command_exists lsblk && [ -n "$(lsblk)" ]
then
lsblk --nodeps --noheadings --output NAME,SIZE,ROTA --exclude 1,2,11 | sort | awk '{if ($3 == 0) {$3="SSD"} else {$3="HDD"}; printf("%-3s%8s%5s\n", $1, $2, $3)}'
elif [ -r "/var/run/dmesg.boot" ]
then
awk '/(ad|ada|da|vtblk)[0-9]+: [0-9]+.B/ { print $1, $2/1024, "GiB" }' /var/run/dmesg.boot | sort -u
elif command_exists df
then
df -h --output=source,fstype,size,itotal | awk 'NR == 1 || /^\/dev/'
else
printf '[ no data available ]'
fi
printf '\n'
# ioping
printf 'ioping: seek rate\n '
"$ioping_cmd" -DR -w 5 . | tail -n 1
printf 'ioping: sequential read speed\n '
"$ioping_cmd" -DRL -w 5 . | tail -n 2 | head -n 1
printf '\n'
# dd disk test
printf 'dd: sequential write speed\n'
if [ -z "$gnu_dd" ]
then
printf ' %s\n' '[disabled due to missing GNU dd]'
else
io1=$( dd_benchmark )
printf ' 1st run: %s\n' "$(printf '%d\n' "$io1" | Bps_to_MiBps)"
io2=$( dd_benchmark )
printf ' 2nd run: %s\n' "$(printf '%d\n' "$io2" | Bps_to_MiBps)"
io3=$( dd_benchmark )
printf ' 3rd run: %s\n' "$(printf '%d\n' "$io3" | Bps_to_MiBps)"
# Calculating avg I/O (better approach with awk for non int values)
ioavg=$( awk 'BEGIN{printf("%.0f", ('"$io1"' + '"$io2"' + '"$io3"')/3)}' )
printf ' average: %s\n' "$(printf '%d\n' "$ioavg" | Bps_to_MiBps)"
fi
printf '\n'
printf '%s\n' '-------------------------------------------------'
# delete downloaded ioping binary if script has been run straight from a pipe
# (rather than a downloaded file)
[ -t 0 ] || rm -f ioping.static
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment