#!/bin/sh
# Copyright (c) International Business Machines Corp., 2001
# Copyright (c) 2017 Petr Vorel <pvorel@suse.cz>
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License as
# published by the Free Software Foundation; either version 2 of
# the License, or (at your option) any later version.
#
# This program is distributed in the hope that it would be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write the Free Software Foundation,
# Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
#
# Author: Manoj Iyer <manjo@mail.utexas.edu>
TST_CNT=3
TST_TESTFUNC=do_test
TST_NEEDS_TMPDIR=1
TST_SETUP=setup
TST_CLEANUP=cleanup
. tst_test.sh
. daemonlib.sh
SYSLOG_STARTED=
CROND_STARTED=
LOGS=
grep_logs()
{
local pattern="$1"
local fail_msg="$2"
local pass_msg="${3:-}"
local n="${4:-10}"
local lines=10
local out=out.$$
local err=err.$$
local i ret
for i in $(seq 1 $n); do
if [ "$LOGS" ]; then
tail -n $lines $LOGS | grep "$pattern" > $out 2> $err
else
journalctl -n $lines | grep "$pattern" > $out 2> $err
fi
ret=$?
[ $ret -eq 0 ] && break
sleep 1
done
if [ $ret -ne 0 ]; then
tst_res TFAIL "$fail_msg: `cat $err`"
else
[ "$pass_msg" ] && tst_res TPASS "$pass_msg"
fi
}
create_crontab()
{
local crontab=cronjob.cron
local script=$1
local out=out.$$
tst_res TINFO "creating crontab: $script"
cat > $crontab <<EOF
* * * * * $script
EOF
tst_res TINFO "installing crontab file"
crontab $crontab > $out 2>&1
if [ $? -ne 0 ]; then
tst_brk TBROK "crontab: error while installing crontab file: `cat $out`"
return 1
fi
return 0
}
remove_crontab()
{
local out=out.$$
tst_res TINFO "removing crontab file"
crontab -r > $out 2>&1
if [ $? -ne 0 ]; then
tst_brk TBROK "crontab: error while removing crontab file `cat $out`"
return 1
fi
return 0
}
create_hello_script()
{
local script=$1
cat > $script <<EOF
#!/bin/sh
echo "Hello Hell"
exit 0
EOF
chmod +x $script
}
install_cron_test()
{
local script=$PWD/cronprg.sh
local cron_out=$PWD/tst1_cron.out
local err=err.log
local sleep_sec
local ts_min1 ts_min2 fail
tst_res TINFO "test install cron job"
cat > $script <<EOF
#! /bin/sh
DATE=\`LC_ALL=C date\`
echo "Hello Hell today is \$DATE" > $cron_out 2>&1
exit 0
EOF
chmod +x $script
create_crontab $script 2> $err
if [ $? -ne 0 ]; then
tst_brk TBROK "crontab: error while creating cron job: `cat $err`"
else
tst_res TINFO "cron job installed successfully"
fi
grep_logs 'crontab.*REPLACE' \
"cron activity not recorded" \
"cron activity logged"
# Sleep 3s after next minute since the loop below sleeps for 62 seconds, we
# should start this 5-iteration loop closely following the start of a
# minute.
sleep_sec=$((123-`date +%-S`))
tst_res TINFO "sleep for ${sleep_sec}s"
sleep $sleep_sec
# $script executed by the cron job will record the date and time into file
# $cron_out. Get the minute recorded by the program, sleep to allow the cron
# job to update file after 1m, and check if the value is advanced by 1.
for i in $(seq 1 5); do
tst_res TINFO "loop: $i: start"
if [ ! -f "$cron_out" ]; then
tst_res TFAIL "loop $i: file $cron_out doesn't exist"
fail=1
break
fi
ts_min1=$(awk '{print $8}' $cron_out | awk -F: '{printf("%d", $2);}')
# wait for the cron job to update output file
sleep 62
# Check the time recorded in output file, this should be 1 minute ahead of
# what was recored earlier.
ts_min2=$(awk '{print $8}' $cron_out | awk -F: '{printf("%d", $2);}')
if [ "x${ts_min1}" = "x" ] || [ "x${ts_min2}" = "x" ]; then
tst_res TFAIL "loop $i: failed to get time: ts_min1: $ts_min1, ts_min2: $ts_min2"
fail=1
break
fi
[ $ts_min1 -eq 59 ] && ts_min1=0 || ts_min1=$(( $ts_min1+1 ))
if [ $ts_min2 -ne $ts_min1 ]; then
tst_res TFAIL "loop $i: failed to update every minute: expected: $ts_min1, received: $ts_min2"
fail=1
break
fi
done
if [ ! "$fail" ]; then
grep_logs "CMD ($script)" \
"failed to install cron job installed and execute it" \
"cron job installed and executed" 1
fi
remove_crontab
}
remove_cron_job_test()
{
local script=$PWD/cronprg.sh
tst_res TINFO "test remove cron job"
create_hello_script $script
create_crontab $script
grep_logs 'crontab.*REPLACE' \
"crontab activity not recorded"
remove_crontab && grep_logs DELETE \
"crontab activity not recorded" \
"crontab removed the cron job" 1
}
list_cron_jobs_test()
{
local script=$PWD/cronprg.sh
local out=cron.out
tst_res TINFO "test list installed cron jobs"
create_hello_script $script
create_crontab $script
tst_res TINFO "crontab: listing cron jobs"
crontab -l | grep "$script" > $out 2>&1 || \
tst_brk TBROK "crontab failed while listing installed cron jobs: `cat $out`"
remove_crontab
crontab -l > $out 2>&1
if [ $? -ne 0 ]; then
grep -q "no crontab for" $out
if [ $? -ne 0 ]; then
tst_res TFAIL "crontab failed removing cron job: `cat $out`"
else
tst_res TPASS "crontab did not list any cron jobs"
fi
else
tst_res TFAIL "crontab failed removing cron job: `cat $out`"
fi
}
setup()
{
if [ "$SYSLOG_DAEMON" ]; then
status_daemon $SYSLOG_DAEMON
if [ $? -ne 0 ]; then
restart_daemon $SYSLOG_DAEMON
SYSLOG_STARTED=1
fi
fi
if [ "$CROND_DAEMON" ]; then
status_daemon $CROND_DAEMON
if [ $? -ne 0 ]; then
restart_daemon $CROND_DAEMON
CROND_STARTED=1
fi
fi
for f in /var/log/syslog /var/log/messages /var/log/cron /var/log/cron.log; do
[ -f "$f" ] && LOGS="$f $LOGS"
done
}
cleanup()
{
[ "$SYSLOG_STARTED" = "1" ] && stop_daemon $SYSLOG_DAEMON
[ "$CROND_STARTED" = "1" ] && stop_daemon $CROND_DAEMON
}
do_test()
{
case $1 in
1) install_cron_test;;
2) remove_cron_job_test;;
3) list_cron_jobs_test;;
esac
}
tst_run