5

我们正在将代码部署到我们的应用程序服务器环境中,该过程的一部分是在服务器上创建许多 cron 作业。当代码被推送时,我们的部署脚本会使用以下内容毫无问题地创建所需的 cron 作业:

CRON_FILE=$SCRIPT_DIR/cron.txt 
if [[ ! -f "$CRON_FILE" ]]; then
        printf "Cron template file missing!\n\n"
        exit 1
fi
while read LINE || [[ -n "$LINE" ]]; do
        printf "\n> Adding cron job \"$LINE\"\n"
        crontab -l | { cat; echo "$LINE"; } | crontab -
done < $CRON_FILE

问题是在初始部署之后,其他部署会创建重复的 cron 作业。

有关如何检测 cron 作业是否已经存在的任何指示?

4

4 回答 4

8

添加 cron 作业时,请包含带有唯一标签的注释。稍后您可以使用该唯一标签来确定 cron 作业是否存在,也可以“卸载”该 cron 作业。

我一直这样做。我有一个可重用的脚本:

#!/bin/sh
#
# Usage: 
# 1. Put this script somewhere in your project
# 2. Edit "$0".crontab file, it should look like this, 
#    but without the # in front of the lines
#0  *   *   *   *   stuff_you_want_to_do
#15 */5 *   *   *   stuff_you_want_to_do
#*  *   1,2 *   *   and_so_on
# 3. To install the crontab, simply run the script
# 4. To remove the crontab, run ./crontab.sh --remove
# 

cd $(dirname "$0")

test "$1" = --remove && mode=remove || mode=add

cron_unique_label="# $PWD"

crontab="$0".crontab
crontab_bak=$crontab.bak
test -f $crontab || cp $crontab.sample $crontab

crontab_exists() {
    crontab -l 2>/dev/null | grep -x "$cron_unique_label" >/dev/null 2>/dev/null
}

# if crontab is executable
if type crontab >/dev/null 2>/dev/null; then
    if test $mode = add; then
        if ! crontab_exists; then
            crontab -l > $crontab_bak
            echo 'Appending to crontab:'
            cat $crontab
            crontab -l 2>/dev/null | { cat; echo; echo $cron_unique_label; cat $crontab; echo; } | crontab -
        else
            echo 'Crontab entry already exists, skipping ...'
            echo
        fi
        echo "To remove previously added crontab entry, run: $0 --remove"
        echo
    elif test $mode = remove; then
        if crontab_exists; then
            echo Removing crontab entry ...
            crontab -l 2>/dev/null | sed -e "\?^$cron_unique_label\$?,/^\$/ d" | crontab -
        else
            echo Crontab entry does not exist, nothing to do.
        fi
    fi
fi

将脚本保存crontab.sh在您的项目目录中,并crontab.sh.crontab使用您的 cron 作业定义创建一个,例如:

0 0 * * * echo hello world
0 0 * * * date
  • 要安装您的 cron 作业,只需运行./crontab.sh
  • 该脚本可以安全地多次运行:它将检测唯一标签是否已经存在并跳过再次添加您的 cron 作业
  • 要卸载 cron 作业,请运行./crontab.sh --remove

我也把它放在 GitHub 上:https ://github.com/janosgyerik/crontab-script

解释sed -e "\?^$cron_unique_label\$?,/^\$/ d"

  • 最简单的表达式基本上是:sed -e '/start/,/end/ d'
  • 意思是:删除匹配开始模式和结束模式的行之间的内容,包括包含模式的行
  • 脚本sed用双引号而不是单引号来引用命令,因为它需要扩展$cron_unique_labelshell变量的值
  • 开始模式\?^$cron_unique_label\$?使用一对?而不是/包围模式,因为$cron_unique_labelcontains /,这会导致问题
  • 开头?必须用反斜杠转义,但老实说我不知道​​为什么。
  • ^匹配行首和行尾$$必须转义,否则shell会扩展$?shell变量的值
  • 结束模式/^\$/比较简单,它匹配一个行首后跟一个行尾,也就是一个空行,并且$必须再次转义
  • 最后dsed命令,删除匹配的行,有效地将其从 的内容中删除crontab -l,我们可以通过管道传递到crontab -
于 2013-07-11T18:26:19.770 回答
5

奇怪,但非常彻底的答案。IMO 过于复杂。

对于任何来此寻求 2017 年答案的人来说,这是一个不错的班轮:

crontab -l | grep 'match-your-cronjob-search' || (crontab -l 2>/dev/null; echo "* * * * * /bin/cronjobCommandYouWant >> /dev/null 2>&1") | crontab -

对我们来说没有欺骗 crons 非常有用。

这是原始海报中的编辑脚本:

    CRON_FILE=$SCRIPT_DIR/cron.txt 
if [[ ! -f "$CRON_FILE" ]]; then
        printf "Cron template file missing!\n\n"
        exit 1
fi
while read LINE || [[ -n "$LINE" ]]; do
        printf "\n> Adding cron job \"$LINE\"\n"
        crontab -l | grep "$LINE" || (crontab -l 2>/dev/null; echo "$LINE") | crontab -
done < $CRON_FILE
于 2017-06-21T23:13:03.163 回答
1

您的脚本 janos 很棒,运行良好,并且正是我正在寻找的一个小故障。我无法管理多个 xxx.crontab 模板。您的脚本运行良好,我将它添加到我的引导例程中并稍作修改,这样我就可以传递 xx.crontab 文件名的第一个参数 $1 和第二个参数 $2 可以删除。我有父 shell 脚本,然后有条件地决定,我想要添加/删除的所有或组合的 crontab 文件。

这里包含我修改的脚本:

#!/bin/sh
#
# Usage:
# 1. Put this script somewhere in your project
# 2. Edit "$1".crontab file, it should look like this,
#    but without the # in front of the lines
#0  *   *   *   *   stuff_you_want_to_do
#15 */5 *   *   *   stuff_you_want_to_do
#*  *   1,2 *   *   and_so_on
# 3. To install the crontab, run ./crontab.sh <nameOfCronTabfile>
# 4. To remove the crontab, run ./crontab.sh <nameOfCronTabfile> --remove

cd $(dirname "$0")

test "$2" = --remove && mode=remove || mode=add

cron_unique_label="# cmID:$PWD|$1#"

crontab="$1".crontab
crontab_bak=$crontab.bak
test -f $crontab || cp $crontab.sample $crontab

crontab_exists() {
    crontab -l 2>/dev/null | grep -x "$cron_unique_label" >/dev/null 2>/dev/null
}

# if crontab is executable
if type crontab >/dev/null 2>/dev/null; then
    if test $mode = add; then
        if ! crontab_exists; then
            crontab -l > $crontab_bak
            echo 'Appending to crontab:'
            echo '-----------------------------------------------'
            cat $crontab
            crontab -l 2>/dev/null | { cat; echo; echo $cron_unique_label; cat $crontab; echo "# cm #"; } | crontab -
        else
            echo 'Crontab entry already exists, skipping ...'
            echo
        fi
        echo '-----------------------------------------------'
        echo "To remove previously added crontab entry, run: $0 $1 --remove"
        echo
    elif test $mode = remove; then
        if crontab_exists; then
            echo 'Removing crontab entry ...'
            crontab -l 2>/dev/null | sed -e "\?^$cron_unique_label\$?,/^# cm #\$/ d" | crontab -
        else
            echo 'Crontab entry does not exist, nothing to do.'
        fi
    fi
fi

更新: 对不起,没有注意到用于删除 crontab 的弱空行模式。只需删除找到 crontab id 之后的所有内容,包括手动添加的 crontab。将空行结束模式更改为小结束标记。所以它将添加 crontabs:

crontab -l 2>/dev/null | { cat; echo; echo $cron_unique_label; cat $crontab; echo "# cm #"; } | crontab -

..并仅删除此cron:

crontab -l 2>/dev/null | sed -e "\?^$cron_unique_label\$?,/^# cm #\$/ d" | crontab -
于 2014-06-26T23:26:25.210 回答
0

我们也可以在代码中做到这一点:首先在 bash 中,查看手册:man 2 open:

int fd = open('/tmp/1.txt', O_CREAT|O_EXCL)
if (fd === -1) {
   print "job exist"
   exit(1)
}
//unlink()
// your code

js版本:

const fs = require('fs')
try {
const fd = fs.openSync('/tmp/1.txt', 'wx')
} catch(err) {
  console.log('job exist')
  process.exit(0)
}

fs.unlinkSync('/tmp/1.txt')
fs.appendFileSync(fd, 'pid', 'utf8');
于 2020-01-16T21:48:46.493 回答