我有一个#!/bin/bash文件。周目录。
有办法测试它是否有效吗?不能等一个星期
我在Debian 6上使用root
我有一个#!/bin/bash文件。周目录。
有办法测试它是否有效吗?不能等一个星期
我在Debian 6上使用root
当前回答
我使用的解决方案如下:
编辑crontab(使用命令:crontab -e)以频繁地运行作业 根据需要(每1分钟或5分钟一次) 修改应该使用cron执行的shell脚本,将输出打印到某个文件中(例如:echo "Working fine" >> output.txt) 使用命令tail -f output.txt检查output.txt文件,它将把最新的内容打印到该文件中,因此您可以跟踪脚本的执行
其他回答
sudo run-parts --test /var/spool/cron/crontabs/
crontabs/目录中的文件需要由owner - octal 700可执行
资料来源:man cron和NNRooth's
有点超出了你的问题范围……但我是这么做的
“如何测试cron作业?”问题与“如何测试在其他程序启动的非交互上下文中运行的脚本?”密切相关。在cron中,触发器是一些时间条件,但许多其他*nix工具以非交互的方式启动脚本或脚本片段,这些脚本运行的条件通常包含一些意想不到的东西,并导致破坏,直到错误被清除。(另见:https://stackoverflow.com/a/17805088/237059)
有一个解决这个问题的通用方法是有帮助的。
我最喜欢的技术之一是使用我写的名为“crontest”的脚本。它从cron中在GNU屏幕会话中启动目标命令,因此您可以附加一个单独的终端来查看正在发生什么,与脚本交互,甚至使用调试器。
为此,您可以在crontab条目中使用“all stars”,并将crontest指定为命令行的第一个命令,例如:
* * * * * crontest /command/to/be/tested --param1 --param2
所以现在cron将每分钟运行您的命令,但crontest将确保一次只运行一个实例。如果该命令需要时间运行,您可以执行“screen -x”来附加并观察它的运行。如果命令是一个脚本,你可以在顶部放一个“read”命令,让它停止并等待屏幕附件完成(加载后按回车键)
如果你的命令是一个bash脚本,你可以这样做:
* * * * * crontest --bashdb /command/to/be/tested --param1 --param2
现在,如果你附加了"screen -x",你将面对一个交互式bashdb会话,你可以逐步检查代码,检查变量等。
#!/bin/bash
# crontest
# See https://github.com/Stabledog/crontest for canonical source.
# Test wrapper for cron tasks. The suggested use is:
#
# 1. When adding your cron job, use all 5 stars to make it run every minute
# 2. Wrap the command in crontest
#
#
# Example:
#
# $ crontab -e
# * * * * * /usr/local/bin/crontest $HOME/bin/my-new-script --myparams
#
# Now, cron will run your job every minute, but crontest will only allow one
# instance to run at a time.
#
# crontest always wraps the command in "screen -d -m" if possible, so you can
# use "screen -x" to attach and interact with the job.
#
# If --bashdb is used, the command line will be passed to bashdb. Thus you
# can attach with "screen -x" and debug the remaining command in context.
#
# NOTES:
# - crontest can be used in other contexts, it doesn't have to be a cron job.
# Any place where commands are invoked without an interactive terminal and
# may need to be debugged.
#
# - crontest writes its own stuff to /tmp/crontest.log
#
# - If GNU screen isn't available, neither is --bashdb
#
crontestLog=/tmp/crontest.log
lockfile=$(if [[ -d /var/lock ]]; then echo /var/lock/crontest.lock; else echo /tmp/crontest.lock; fi )
useBashdb=false
useScreen=$( if which screen &>/dev/null; then echo true; else echo false; fi )
innerArgs="$@"
screenBin=$(which screen 2>/dev/null)
function errExit {
echo "[-err-] $@" | tee -a $crontestLog >&2
}
function log {
echo "[-stat-] $@" >> $crontestLog
}
function parseArgs {
while [[ ! -z $1 ]]; do
case $1 in
--bashdb)
if ! $useScreen; then
errExit "--bashdb invalid in crontest because GNU screen not installed"
fi
if ! which bashdb &>/dev/null; then
errExit "--bashdb invalid in crontest: no bashdb on the PATH"
fi
useBashdb=true
;;
--)
shift
innerArgs="$@"
return 0
;;
*)
innerArgs="$@"
return 0
;;
esac
shift
done
}
if [[ -z $sourceMe ]]; then
# Lock the lockfile (no, we do not wish to follow the standard
# advice of wrapping this in a subshell!)
exec 9>$lockfile
flock -n 9 || exit 1
# Zap any old log data:
[[ -f $crontestLog ]] && rm -f $crontestLog
parseArgs "$@"
log "crontest starting at $(date)"
log "Raw command line: $@"
log "Inner args: $@"
log "screenBin: $screenBin"
log "useBashdb: $( if $useBashdb; then echo YES; else echo no; fi )"
log "useScreen: $( if $useScreen; then echo YES; else echo no; fi )"
# Were building a command line.
cmdline=""
# If screen is available, put the task inside a pseudo-terminal
# owned by screen. That allows the developer to do a "screen -x" to
# interact with the running command:
if $useScreen; then
cmdline="$screenBin -D -m "
fi
# If bashdb is installed and --bashdb is specified on the command line,
# pass the command to bashdb. This allows the developer to do a "screen -x" to
# interactively debug a bash shell script:
if $useBashdb; then
cmdline="$cmdline $(which bashdb) "
fi
# Finally, append the target command and params:
cmdline="$cmdline $innerArgs"
log "cmdline: $cmdline"
# And run the whole schlock:
$cmdline
res=$?
log "Command result: $res"
echo "[-result-] $(if [[ $res -eq 0 ]]; then echo ok; else echo fail; fi)" >> $crontestLog
# Release the lock:
9<&-
fi
在摆弄了cron中一些不能立即兼容的东西之后,我发现下面的方法很适合调试:
crontab -e
* * * * * /path/to/prog var1 var2 &>>/tmp/cron_debug_log.log
这将每分钟运行一次任务,您可以简单地查看/tmp/cron_debug_log.log文件以了解发生了什么。
这并不是您可能正在寻找的“救火工作”,但在调试最初在cron中无法工作的脚本时,这对我帮助很大。
只要做cron所做的,以root身份运行以下命令:
run-parts -v /etc/cron.weekly
... 或者下一个,如果你收到"Not a directory: -v"错误:
run-parts /etc/cron.weekly -v
选项-v在运行脚本之前打印脚本名称。
除此之外,你还可以用:
http://pypi.python.org/pypi/cronwrap
结束你的cron,在成功或失败时给你发一封电子邮件。