2019-03-14 16:37:45 +01:00
<!DOCTYPE html>
2020-05-31 16:16:16 +02:00
< html lang = "en" class = "dark-gray bg-white sans-serif lh-copy" >
< title > RRDtool looks nicer... - OpenBSD Amsterdam< / title >
2019-03-14 16:37:45 +01:00
< meta charset = "UTF-8" >
2020-05-31 16:16:16 +02:00
< meta name = "generator" content = "romanzolotarev.com/ssg" >
< meta name = "viewport" content = "width=device-width, initial-scale=1" >
< meta name = "discountegg" content = "OBSD-EGG-2020" >
< link rel = "icon" type = "image/png" href = "/favicon.png" >
< link rel = "stylesheet" href = "/style.css" >
< body class = "mw7 w-100 center pa3" >
< div class = "cf" >
< div class = "fl w-100 w-50-l w-50-m" > < a href = "/" title = "OpenBSD Amsterdam Logo" > < img class = "w3 br-100" src = "/images/logo.png" alt = "OpenBSD Amsterdam Logo" > < / a > < / div >
< div class = "fl w-100 w-50-l w-50-m mt2 tr" > < a href = "servers.html" > 364 VMs deployed< / a > 🐡 < / div >
< / div >
2019-03-14 16:37:45 +01:00
< h1 id = "RRDtool%20looks%20nicer..." > RRDtool looks nicer...< / h1 >
2020-05-31 16:16:16 +02:00
< p class = "f7" > March 14, 2019< / p >
< p > < a href = "https://openbsd.amsterdam/" > OpenBSD Amsterdam< / a > was in search of a lightweight toolset to keep track of resource usage, at a minimum the CPU load generated by the < a href = "https://man.openbsd.org/vmm.4" > vmm(4)< / a > / < a href = "https://man.openbsd.org/vmd.8" > vmd(8)< / a > hosts and the traffic from and to the hosts. A couple of weeks ago we ended up with a workable < a href = "https://openbsd.amsterdam/blog/using-mrtg-again.html" > MRTG Setup< / a > . While it worked, it didn' t look very pretty.< / p >
2019-03-14 16:37:45 +01:00
< p > In a moment of clarity, we thought about using < a href = "https://oss.oetiker.ch/rrdtool/" > RRDtool< / a > . Heck, why shouldn' t we give it a try? From the previous tooling, we already had some required building blocks in place to make < a href = "https://oss.oetiker.ch/mrtg/" > MRTG< / a > understand the CPU Cores and uptime from < a href = "https://openbsd.org/" > OpenBSD< / a > .< / p >
2020-05-31 16:16:16 +02:00
< p > Before we start:< / p >
< pre >
# < b > pkg_add rrdtool< / b >
#
< / pre >
2019-03-14 16:37:45 +01:00
< p > We decided to split the collection of the different OIDs (SNMP Object Identifiers) into three different scripts, which < a href = "https://man.openbsd.org/cron.1" > cron(1)< / a > calls, from a wrapper script.< / p >
< ul >
< li > uptime.sh< / li >
< li > cpu_load.sh< / li >
< li > interface.sh< / li >
< / ul >
< h3 id = "uptime.sh" > uptime.sh< / h3 >
2020-05-31 16:16:16 +02:00
< pre >
#!/bin/sh
test -n "$1" || exit 1
HOST="$1"
COMMUNITY="public"
UPTIMEINFO="/tmp/${HOST}-uptime.txt"
2019-03-14 16:37:45 +01:00
TICKS=$(snmpctl snmp get ${HOST} community ${COMMUNITY} oid hrSystemUptime.0 | cut -d= -f2)
2020-05-31 16:16:16 +02:00
DAYS=$(echo "${TICKS}/8640000" | bc -l)
HOURS=$(echo "0.${DAYS##*.} * 24" | bc -l)
MINUTES=$(echo "0.${HOURS##*.} * 60" | bc -l)
SECS=$(echo "0.${MINUTES##*.} * 60" | bc -l)
test -n "$DAYS" & & printf '%s days, ' "${DAYS%.*}" > ${UPTIMEINFO}
printf '%02d\\:%02d\\:%02d\n' "${HOURS%.*}" "${MINUTES%.*}" "${SECS%.*}" >> ${UPTIMEINFO}
< / pre >
2019-03-14 16:37:45 +01:00
< p > This is a seperate script, due to the uptime usage of both hosts in both graphs. < / p >
2020-05-31 16:16:16 +02:00
< p > The origins for this script can be found detailled in our < a href = "https://openbsd.amsterdam/blog/using-mrtg-again.html" > MRTG Setup< / a > .< / p >
2019-03-14 16:37:45 +01:00
< h3 id = "cpu_load.sh" > cpu_load.sh< / h3 >
2020-05-31 16:16:16 +02:00
< pre >
test -n "$1" || exit 1
HOST="$1"
COMMUNITY="public"
RRDFILES="/var/rrdtool"
IMAGES="/var/www/htdocs"
WATERMARK="OpenBSD Amsterdam - https://obsda.ms"
RRDTOOL="/usr/local/bin/rrdtool"
CPUINFO="/tmp/${HOST}-cpu.txt"
2019-03-14 16:37:45 +01:00
UPTIME=$(cat /tmp/${HOST}-uptime.txt)
2020-05-31 16:16:16 +02:00
NOW=$(date "+%Y-%m-%d %H:%M:%S %Z" | sed 's/:/\\:/g')
2019-03-14 16:37:45 +01:00
2020-05-31 16:16:16 +02:00
if ! test -f "${RRDFILES}/${HOST}-cpu.rrd"
2019-03-14 16:37:45 +01:00
then
2020-05-31 16:16:16 +02:00
echo "Creating ${RRDFILES}/${HOST}-cpu.rrd"
2019-03-14 16:37:45 +01:00
${RRDTOOL} create ${RRDFILES}/${HOST}-cpu.rrd \
2020-05-31 16:16:16 +02:00
--step 300 \
DS:ds0:GAUGE:600:U:U \
RRA:MAX:0.5:1:20000
2019-03-14 16:37:45 +01:00
fi
2020-05-31 16:16:16 +02:00
snmpctl snmp walk ${HOST} community ${COMMUNITY} oid hrProcessorLoad | cut -d= -f2 > ${CPUINFO}
CORES=$(grep -cv "^0$" ${CPUINFO})
CPU_LOAD_SUM=$(awk '{sum += $1} END {print sum}' ${CPUINFO})
CPU_LOAD=$(echo "scale=2; ${CPU_LOAD_SUM}/${CORES}" | bc -l)
2019-03-14 16:37:45 +01:00
${RRDTOOL} update ${RRDFILES}/${HOST}-cpu.rrd N:${CPU_LOAD}
${RRDTOOL} graph ${IMAGES}/${HOST}-cpu.png \
2020-05-31 16:16:16 +02:00
--start -43200 \
--title "${HOST} - CPU" \
--vertical-label "% CPU Used" \
--watermark "${WATERMARK}" \
DEF:CPU=${RRDFILES}/${HOST}-cpu.rrd:ds0:AVERAGE \
AREA:CPU#FFCC00 \
LINE2:CPU#CC0033:"CPU" \
GPRINT:CPU:MAX:"Max\:%2.2lf %s" \
GPRINT:CPU:AVERAGE:"Average\:%2.2lf %s" \
GPRINT:CPU:LAST:" Current\:%2.2lf %s\n" \
COMMENT:"\\n" \
COMMENT:" SUM CPU Load / Active Cores = % CPU Used\n" \
COMMENT:" Up for ${UPTIME} at ${NOW}"
< / pre >
2019-03-14 16:37:45 +01:00
< p > On the first run, < a href = "https://oss.oetiker.ch/rrdtool/" > RRDtool< / a > will create the .rrd file. On every subsequent run, it will update the file with the collected values and update the graph.< / p >
2020-05-31 16:16:16 +02:00
< p > The origins for this script can be found detailled in our < a href = "https://openbsd.amsterdam/blog/using-mrtg-again.html" > MRTG Setup< / a > .< / p >
2019-03-14 16:37:45 +01:00
< h3 id = "interface.sh" > interface.sh< / h3 >
2020-05-31 16:16:16 +02:00
< pre >
test -n "$1" || exit 1
test -n "$2" || exit 1
HOST="$1"
INTERFACE="$2"
COMMUNITY="public"
RRDFILES="/var/rrdtool"
IMAGES="/var/www/htdocs"
WATERMARK="OpenBSD Amsterdam - https://obsda.ms"
RRDTOOL="/usr/local/bin/rrdtool"
2019-03-14 16:37:45 +01:00
UPTIME=$(cat /tmp/${HOST}-uptime.txt)
2020-05-31 16:16:16 +02:00
NOW=$(date "+%Y-%m-%d %H:%M:%S %Z" | sed 's/:/\\:/g')
2019-03-14 16:37:45 +01:00
2020-05-31 16:16:16 +02:00
if ! test -f "${RRDFILES}/${HOST}-${INTERFACE}.rrd"
2019-03-14 16:37:45 +01:00
then
2020-05-31 16:16:16 +02:00
echo "Creating ${RRDFILES}/${HOST}-${INTERFACE}.rrd"
2019-03-14 16:37:45 +01:00
${RRDTOOL} create ${RRDFILES}/${HOST}-${INTERFACE}.rrd \
2020-05-31 16:16:16 +02:00
--step 300 \
DS:ds0:COUNTER:600:0:1250000000 \
DS:ds1:COUNTER:600:0:1250000000 \
RRA:AVERAGE:0.5:1:600 \
RRA:AVERAGE:0.5:6:700 \
RRA:AVERAGE:0.5:24:775 \
RRA:AVERAGE:0.5:288:797 \
RRA:MAX:0.5:1:600 \
RRA:MAX:0.5:6:700 \
RRA:MAX:0.5:24:775 \
RRA:MAX:0.5:288:797
2019-03-14 16:37:45 +01:00
fi
IN=$(snmpctl snmp get ${HOST} community ${COMMUNITY} oid ifInOctets.${INTERFACE} | cut -d= -f2)
OUT=$(snmpctl snmp get ${HOST} community ${COMMUNITY} oid ifOutOctets.${INTERFACE} | cut -d= -f2)
DESCR=$(snmpctl snmp get ${HOST} community ${COMMUNITY} oid ifDescr.${INTERFACE} | cut -d= -f2 | tr
2020-05-31 16:16:16 +02:00
-d '"')
2019-03-14 16:37:45 +01:00
${RRDTOOL} update ${RRDFILES}/${HOST}-${INTERFACE}.rrd N:${IN}:${OUT}
${RRDTOOL} graph ${IMAGES}/${HOST}-${INTERFACE}.png \
2020-05-31 16:16:16 +02:00
--start -43200 \
--title "${HOST} - ${DESCR}" \
--vertical-label "Bits per Second" \
--watermark "${WATERMARK}" \
DEF:IN=${RRDFILES}/${HOST}-${INTERFACE}.rrd:ds0:AVERAGE \
DEF:OUT=${RRDFILES}/${HOST}-${INTERFACE}.rrd:ds1:AVERAGE \
CDEF:IN_CDEF="IN,8,*" \
CDEF:OUT_CDEF="OUT,8,*" \
AREA:IN_CDEF#00FF00:"In " \
GPRINT:IN_CDEF:MAX:"Max\:%5.2lf %s" \
GPRINT:IN_CDEF:AVERAGE:"Average\:%5.2lf %s" \
GPRINT:IN_CDEF:LAST:" Current\:%5.2lf %s\n" \
LINE2:OUT_CDEF#0000FF:"Out" \
GPRINT:OUT_CDEF:MAX:"Max\:%5.2lf %s" \
GPRINT:OUT_CDEF:AVERAGE:"Average\:%5.2lf %s" \
GPRINT:OUT_CDEF:LAST:" Current\:%5.2lf %s\n" \
COMMENT:"\\n" \
COMMENT:" Up for ${UPTIME} at ${NOW}"
< / pre >
2019-03-14 16:37:45 +01:00
< p > To pinpoint the network interface you want to measure the bandwith for, this command prints the available interfaces:< / p >
2020-05-31 16:16:16 +02:00
< pre >
snmpctl snmp walk [host] community [community] oid ifDescr
< / pre >
2019-03-14 16:37:45 +01:00
< p > This will output a list like:< / p >
2020-05-31 16:16:16 +02:00
< pre >
ifDescr.1="em0"
ifDescr.2="em1"
ifDescr.3="enc0"
ifDescr.4="lo0"
ifDescr.5="bridge880"
ifDescr.6="vlan880"
ifDescr.13="pflog0"
ifDescr.669="tap0"
ifDescr.670="tap1"
< / pre >
2019-03-14 16:37:45 +01:00
< p > The number behind < code > ifDescr< / code > is the one that you need to feed to < strong > interface.sh< / strong > , for example:< / p >
2020-05-31 16:16:16 +02:00
< pre >
# interface.sh 5
< / pre >
2019-03-14 16:37:45 +01:00
< p > Finally the < strong > wrapper.sh< / strong > script calls all the aforementioned scripts:< / p >
2020-05-31 16:16:16 +02:00
< pre >
#!/bin/sh
SCRIPTS="/var/rrdtool"
2019-03-14 16:37:45 +01:00
for i in $(jot 2 1); do ${SCRIPTS}/uptime.sh host${i}.domain.tld; done
for i in $(jot 2 1); do ${SCRIPTS}/cpu_load.sh host${i}.domain.tld; done
${SCRIPTS}/interface.sh host1.domain.tld 12
${SCRIPTS}/interface.sh host2.domain.tld 11
2020-05-31 16:16:16 +02:00
< / pre >
2019-03-14 16:37:45 +01:00
< p > The resulting graphs:< / p >
< p > < img src = "https://ops.high5.nl/obsda.ms/s1.obsda.ms-cpu.png" alt = "" / >
< img src = "https://ops.high5.nl/obsda.ms/s1.obsda.ms-12.png" alt = "" / > < / p >
2020-05-31 16:16:16 +02:00
< p > To serve the graphs we use < a href = "http://man.openbsd.org/httpd.8" > httpd(8)< / a > with the following config:
< pre >
server " default" {
listen on * port 80
location " / .well-known/ acme-challenge/ *" {
root " / acme"
request strip 2
}
location * {
block return 302 " https:/ / $HTTP< em > HOST$REQUEST< / em > URI"
}
}< / p >
< p > server " default" {
listen on * tls port 443
tls {
certificate " / etc/ ssl/ default-fullchain.pem"
key " / etc/ ssl/ private/ default.key"
}
location " / .well-known/ acme-challenge/ *" {
root " / acme"
request strip 2
}
root " / htdocs"
}
< / pre >
All the scripts can be found in our < a href = "https://git.high5.nl/rrdtool" > Git Repository< / a > .< / p >
2019-03-14 16:37:45 +01:00
< p > You can follow us on < a href = "https://twitter.com/@OpenBSDAms" > Twitter< / a > and < a href = "https://bsd.network/@OpenBSDAms" > Mastodon< / a > .< / p >
2020-05-31 16:16:16 +02:00
< p class = "footer f7 mt4" >
© 2018– 2020 OpenBSD Amsterdam is a project of < a href = "https://high5.nl/" > High5!< / a >
2019-03-14 16:37:45 +01:00
< / body >
< / html >