#!/usr/bin/env bash # Scaling/ performance tests # CLI/Netconf/Restconf # Lists (and leaf-lists) # Add, get and delete entries # Magic line must be first in script (see README.md) s="$_" ; . ./lib.sh || if [ "$s" = $0 ]; then exit 0; else return 0; fi # Which format to use as datastore format internally : ${format:=xml} # Number of list/leaf-list entries in file : ${perfnr:=20000} # Number of requests made get/put : ${perfreq:=10} # time function (this is a mess to get right on freebsd/linux) # -f %e gives elapsed wall clock time but is not available on all systems # so we use time -p for POSIX compliance and awk to get wall clock time # Note sometimes time -p is used and sometimes $TIMEFN, cant get it to work same everywhere # time function (this is a mess to get right on freebsd/linux) : ${TIMEFN:=time -p} # portability: 2>&1 | awk '/real/ {print $2}' if ! $TIMEFN true; then err "A working time function" "'$TIMEFN' does not work"; fi APPNAME=example cfg=$dir/scaling-conf.xml fyang=$dir/scaling.yang fconfig=$dir/large.xml fconfigonly=$dir/config.xml # only config for test ftest=$dir/test.xml fconfig2=$dir/large2.xml # leaf-list foutput=$dir/output.xml cat < $fyang module scaling{ yang-version 1.1; namespace "urn:example:clixon"; prefix ex; container x { list y { key "a"; leaf a { type int32; } leaf b { type int32; } } leaf-list c { type string; } } } EOF cat < $cfg $cfg $dir ${YANG_INSTALLDIR} $fyang /usr/local/var/$APPNAME/$APPNAME.sock /usr/local/var/example/$APPNAME.pidfile $dir false $format example /usr/local/lib/example/cli /usr/local/lib/example/clispec 0 ietf-netconf:startup EOF new "test params: -f $cfg" if [ $BE -ne 0 ]; then new "kill old backend" sudo clixon_backend -zf $cfg if [ $? -ne 0 ]; then err fi new "start backend -s init -f $cfg -- -s" start_backend -s init -f $cfg -- -s fi new "waiting" wait_backend # Check this later with committed data new "generate config with $perfnr list entries" echo -n "" > $fconfigonly for (( i=0; i<$perfnr; i++ )); do echo -n "$i$i" >> $fconfigonly done echo -n "" >> $fconfigonly # No CR echo -n "$DEFAULTHELLO" > $fconfig cat $fconfigonly >> $fconfig echo "]]>]]>" >> $fconfig # Now take large config file and write it via netconf to candidate new "test time exists" expectpart "$(time -p ls)" 0 new "netconf write large config" expecteof_file "time -p $clixon_netconf -qf $cfg" 0 "$fconfig" "^]]>]]>$" 2>&1 | awk '/real/ {print $2}' # Here, there are $perfnr entries in candidate # Now commit it from candidate to running new "netconf commit large config" expecteof "time -p $clixon_netconf -qf $cfg" 0 "$DEFAULTHELLO]]>]]>" "^]]>]]>$" 2>&1 | awk '/real/ {print $2}' new "Check running-db contents" echo "$DEFAULTHELLO]]>]]>" | $clixon_netconf -qf $cfg > $foutput # Create a file to compare with echo -n "" > $ftest cat $fconfigonly >> $ftest echo -n "]]>]]>" >> $ftest ret=$(diff $ftest $foutput) if [ $? -ne 0 ]; then err1 "Matching running-db with $fconfigonly" fi # Now commit it again from candidate (validation takes time when # comparing to existing) # Having a large db, get and put single entries many times # Note same entries in the range alreay there, db has same size # NETCONF get 1 key index # Note this is done by streaming input into one single netconf client. it is much # slower if it is started and stopped for each request. (next) new "netconf get $perfreq small config 1 key index" { time -p for (( i=0; i<$perfreq; i++ )); do rnd=$(( ( RANDOM % $perfnr ) )) echo "$DEFAULTHELLO]]>]]>" done | $clixon_netconf -qf $cfg > /dev/null; } 2>&1 | awk '/real/ {print $2}' new "netconf get $perfreq small config 1 key index start/stop" { time -p for (( i=0; i<$perfreq; i++ )); do rnd=$(( ( RANDOM % $perfnr ) )) echo "$DEFAULTHELLO]]>]]>" | $clixon_netconf -qf $cfg > /dev/null; done } 2>&1 | awk '/real/ {print $2}' # NETCONF get 1 key and one non-key index new "netconf get $perfreq small config 1 key + 1 non-key index" { time -p for (( i=0; i<$perfreq; i++ )); do rnd=$(( ( RANDOM % $perfnr ) )) echo "$DEFAULTHELLO]]>]]>" done | $clixon_netconf -qf $cfg > /dev/null; } 2>&1 | awk '/real/ {print $2}' # NETCONF add new "netconf add $perfreq small config" { time -p for (( i=0; i<$perfreq; i++ )); do rnd=$(( ( RANDOM % $perfnr ) )) echo "$DEFAULTHELLO$rnd$rnd]]>]]>" done | $clixon_netconf -qf $cfg > /dev/null; } 2>&1 | awk '/real/ {print $2}' # Instead of many small entries, get one large in netconf and restconf # cli? new "netconf get large config" expecteof "time -p $clixon_netconf -qf $cfg" 0 "$DEFAULTHELLO]]>]]>" "^00112233" 2>&1 | awk '/real/ {print $2}' # Delete entries (last since entries are removed from db) new "netconf discard-changes" expecteof "$clixon_netconf -qf $cfg" 0 "$DEFAULTHELLO]]>]]>" "^]]>]]>$" new "netconf delete $perfreq small config" { time -p for (( i=0; i<$perfreq; i++ )); do rnd=$(( ( RANDOM % $perfnr ) )) echo "$DEFAULTHELLO$rnd]]>]]>" done | $clixon_netconf -qf $cfg > /dev/null; } 2>&1 | awk '/real/ {print $2}' #new "netconf discard-changes" expecteof "$clixon_netconf -qf $cfg" 0 "$DEFAULTHELLO]]>]]>" "^]]>]]>$" # Now do leaf-lists istead of leafs #new "generate leaf-list config" echo -n "$DEFAULTHELLOreplace" > $fconfig2 for (( i=0; i<$perfnr; i++ )); do echo -n "$i" >> $fconfig2 done echo "]]>]]>" >> $fconfig2 new "netconf replace large list-leaf config" expecteof_file "time -p $clixon_netconf -qf $cfg" 0 "$fconfig2" "^]]>]]>$" 2>&1 | awk '/real/ {print $2}' new "netconf commit large leaf-list config" expecteof "time -p $clixon_netconf -qf $cfg" 0 "$DEFAULTHELLO]]>]]>" "^]]>]]>$" 2>&1 | awk '/real/ {print $2}' new "netconf add $perfreq small leaf-list config" { time -p for (( i=0; i<$perfreq; i++ )); do rnd=$(( ( RANDOM % $perfnr ) )) echo "$DEFAULTHELLO$rnd]]>]]>" done | $clixon_netconf -qf $cfg > /dev/null; } 2>&1 | awk '/real/ {print $2}' new "netconf add small leaf-list config" expecteof "time -p $clixon_netconf -qf $cfg" 0 "$DEFAULTHELLOx]]>]]>" "^]]>]]>$" 2>&1 | awk '/real/ {print $2}' new "netconf commit small leaf-list config" expecteof "time -p $clixon_netconf -qf $cfg" 0 "$DEFAULTHELLO]]>]]>" "^]]>]]>$" 2>&1 | awk '/real/ {print $2}' new "netconf get large leaf-list config" expecteof "time -p $clixon_netconf -qf $cfg" 0 "$DEFAULTHELLO]]>]]>" "^01" 2>&1 | awk '/real/ {print $2}' if [ $BE -ne 0 ]; then new "Kill backend" # Check if premature kill pid=$(pgrep -u root -f clixon_backend) if [ -z "$pid" ]; then err "backend already dead" fi # kill backend stop_backend -f $cfg fi rm -rf $dir # unset conditional parameters unset format unset perfnr unset perfreq new "endtest" endtest