如何列出所有用户的所有cron作业?


854

是否有命令或现有脚本可以让我立即查看* NIX系统的所有计划cron作业?我希望它包含所有用户crontabs以及中的/etc/crontab所有内容/etc/cron.d。看看run-partsin中运行的特定命令也将很不错/etc/crontab

理想情况下,我希望输出以漂亮的列形式并以某种有意义的方式排序。

然后,我可以合并来自多个服务器的这些清单,以查看总体的“事件时间表”。

我本打算自己写这样的脚本,但是如果有人已经麻烦了...


Answers:


1131

您将必须以root用户身份运行,但是:

for user in $(cut -f1 -d: /etc/passwd); do crontab -u $user -l; done

将遍历列出其crontab的每个用户名。crontab由各自的用户拥有,因此您将无法看到其他用户的crontab,而无论他们是他们还是root。


编辑 是否要知道crontab属于哪个用户,请使用echo $user

for user in $(cut -f1 -d: /etc/passwd); do echo $user; crontab -u $user -l; done

52
在NIS或LDAP中定义用户时不起作用。您需要使用for user in $(getent passwd | cut -f1 -d: ); do echo $user; crontab -u $user -l; done
休伯特·卡里奥

9
更新了此内容以排除评论并禁止显示“用户无crontab ...”消息:for user in $(cut -f1 -d: /etc/passwd); do crontab -u $user -l 2>/dev/null | grep -v '^#'; done
Jonathan

38
在/ var / spool / cron中查看文件会更容易吗?
graywh

1
我们需要用getent命令替换LDAP和/ etc / passwd:for user in $(getent passwd | awk -F : '{print $1}'); do echo $user; crontab -u $user -l; done
Toby Batch

9
有关在cronjobs什么/etc/cron.hourly//etc/cron.daily//etc/cron.weekly//etc/cron.monthly/...?
阿卜杜勒2015年

309

我最终写了一个脚本(我想教自己bash脚本的精髓,所以这就是为什么您在这里看不到Perl之类的原因)。这并不是一件简单的事情,但是它满足了我的大部分需求。它采用了凯尔的建议用于查找个人用户的crontab,而且还与涉及/etc/crontab(其中包括推出的脚本run-parts/etc/cron.hourly/etc/cron.daily等),在和工作/etc/cron.d目录。它需要所有这些内容,并将它们合并到如下所示的显示中:

mi     h    d  m  w  user      command
09,39  *    *  *  *  root      [ -d /var/lib/php5 ] && find /var/lib/php5/ -type f -cmin +$(/usr/lib/php5/maxlifetime) -print0 | xargs -r -0 rm
47     */8  *  *  *  root      rsync -axE --delete --ignore-errors / /mirror/ >/dev/null
17     1    *  *  *  root      /etc/cron.daily/apt
17     1    *  *  *  root      /etc/cron.daily/aptitude
17     1    *  *  *  root      /etc/cron.daily/find
17     1    *  *  *  root      /etc/cron.daily/logrotate
17     1    *  *  *  root      /etc/cron.daily/man-db
17     1    *  *  *  root      /etc/cron.daily/ntp
17     1    *  *  *  root      /etc/cron.daily/standard
17     1    *  *  *  root      /etc/cron.daily/sysklogd
27     2    *  *  7  root      /etc/cron.weekly/man-db
27     2    *  *  7  root      /etc/cron.weekly/sysklogd
13     3    *  *  *  archiver  /usr/local/bin/offsite-backup 2>&1
32     3    1  *  *  root      /etc/cron.monthly/standard
36     4    *  *  *  yukon     /home/yukon/bin/do-daily-stuff
5      5    *  *  *  archiver  /usr/local/bin/update-logs >/dev/null

请注意,它显示了用户,并且按小时和分钟显示或多或少的排序,以便我可以看到每日时间表。

到目前为止,我已经在Ubuntu,Debian和Red Hat AS上对其进行了测试。

#!/bin/bash

# System-wide crontab file and cron job directory. Change these for your system.
CRONTAB='/etc/crontab'
CRONDIR='/etc/cron.d'

# Single tab character. Annoyingly necessary.
tab=$(echo -en "\t")

# Given a stream of crontab lines, exclude non-cron job lines, replace
# whitespace characters with a single space, and remove any spaces from the
# beginning of each line.
function clean_cron_lines() {
    while read line ; do
        echo "${line}" |
            egrep --invert-match '^($|\s*#|\s*[[:alnum:]_]+=)' |
            sed --regexp-extended "s/\s+/ /g" |
            sed --regexp-extended "s/^ //"
    done;
}

# Given a stream of cleaned crontab lines, echo any that don't include the
# run-parts command, and for those that do, show each job file in the run-parts
# directory as if it were scheduled explicitly.
function lookup_run_parts() {
    while read line ; do
        match=$(echo "${line}" | egrep -o 'run-parts (-{1,2}\S+ )*\S+')

        if [[ -z "${match}" ]] ; then
            echo "${line}"
        else
            cron_fields=$(echo "${line}" | cut -f1-6 -d' ')
            cron_job_dir=$(echo  "${match}" | awk '{print $NF}')

            if [[ -d "${cron_job_dir}" ]] ; then
                for cron_job_file in "${cron_job_dir}"/* ; do  # */ <not a comment>
                    [[ -f "${cron_job_file}" ]] && echo "${cron_fields} ${cron_job_file}"
                done
            fi
        fi
    done;
}

# Temporary file for crontab lines.
temp=$(mktemp) || exit 1

# Add all of the jobs from the system-wide crontab file.
cat "${CRONTAB}" | clean_cron_lines | lookup_run_parts >"${temp}" 

# Add all of the jobs from the system-wide cron directory.
cat "${CRONDIR}"/* | clean_cron_lines >>"${temp}"  # */ <not a comment>

# Add each user's crontab (if it exists). Insert the user's name between the
# five time fields and the command.
while read user ; do
    crontab -l -u "${user}" 2>/dev/null |
        clean_cron_lines |
        sed --regexp-extended "s/^((\S+ +){5})(.+)$/\1${user} \3/" >>"${temp}"
done < <(cut --fields=1 --delimiter=: /etc/passwd)

# Output the collected crontab lines. Replace the single spaces between the
# fields with tab characters, sort the lines by hour and minute, insert the
# header line, and format the results as a table.
cat "${temp}" |
    sed --regexp-extended "s/^(\S+) +(\S+) +(\S+) +(\S+) +(\S+) +(\S+) +(.*)$/\1\t\2\t\3\t\4\t\5\t\6\t\7/" |
    sort --numeric-sort --field-separator="${tab}" --key=2,1 |
    sed "1i\mi\th\td\tm\tw\tuser\tcommand" |
    column -s"${tab}" -t

rm --force "${temp}"

38
没什么,但是它对/ etc / crontab和/etc/cron.d/中的系统cron作业没有任何作用。我的脚本就是处理这些内容并在最后格式化所有内容。
yukondude

10
yukondude-您应该考虑将其放在github上,即使只是要点。
凯尔·伯顿

3
试图复制粘贴并运行它,但是失败了:showcrons.sh:第59行:意外标记附近的语法错误已<' showcrons.sh: line 59: 完成<<(cut --fields = 1 --delimiter =:/ etc / passwd)'
Fraggle

2
@KyleBurton似乎已经有至少8个gist
gists

9
警告:此脚本缺少来自/etc/anacrontab
ck_

191

在Ubuntu或debian下,您可以按查看crontab /var/spool/cron/crontabs/,然后在其中查看每个用户的文件。当然,这仅适用于特定于用户的crontab。

对于Redhat 6/7和Centos,crontab位于下方/var/spool/cron/


7
这也可以在RedHat(/ var / spool / cron)上使用,并且比编写/运行脚本更容易,尤其是在使用Ldap之类的帐户来管理帐户时。+1
user49913

6
这对我来说比其他任何答案都更有帮助。使用此方法,您还可以查看不再存在的用户的crontab,从而为OP提供了所有cron作业。
安德鲁·恩斯利

1
此方法的另一个好处:我的服务器使用LDAP,因此大多数用户不在/etc/passwd。国际海事组织,这应该是公认的答案,而不是所有的蛮力解决方案。
米克尔

1
在这里使用Suse Linux很好。
Bret

谢谢,对于AWS EC2安装实例也是如此,此信息非常有帮助!
伊万·马里亚诺维奇

34

这将显示所有用户的所有crontab条目。

sed 's/^\([^:]*\):.*$/crontab -u \1 -l 2>\&1/' /etc/passwd | grep -v "no crontab for" | sh

30

取决于您的linux版本,但我使用:

tail -n 1000 /var/spool/cron/*

作为根。很简单很短。

给我这样的输出:

==> /var/spool/cron/root <==
15 2 * * * /bla

==> /var/spool/cron/my_user <==
*/10 1 * * * /path/to/script

4
使用tail -n +1 /var/spool/cron/*列出文件的所有内容。
汉斯·金泽尔

4
...或者sudo sh -c 'tail -n +1 /var/spool/cron/*'如果您不想成为根。我的OCD迫使我研究为什么我不能按书面要求对这个命令进行伪装。这是因为普通用户无权访问/ var / spool / cron dir,并且该glob被解释为原义的星形字符,显然不存在。
戴尔·安德森

另外,cd /var/spool/cron/cron/ && grep . *也将打印在每一个cron作业前相应的用户名
的Jakub Kukul


14
getent passwd | cut -d: -f1 | perl -e'while(<>){chomp;$l = `crontab -u $_ -l 2>/dev/null`;print "$_\n$l\n" if $l}'

这样可以避免直接与passwd混淆,跳过没有cron条目的用户,对于有cron条目的用户,它会打印出用户名以及crontab。

不过,通常将其放到这里,这样我以后可以找到它,以防万一我需要再次搜索它。


1
它还列出了中不存在的LDAP用户/etc/passwd。上面的Matt解决方案更适合这种特殊情况,但是很高兴知道该命令存在。
米克尔

11

如果使用NIS检查集群,则唯一的方法就是根据Matt的答案/ var / spool / cron / tabs查看用户是否具有crontab条目。

grep -v "#" -R  /var/spool/cron/tabs


10

这个脚本在CentOS中为我工作,列出了环境中的所有cron:

sudo cat /etc/passwd | sed 's/^\([^:]*\):.*$/sudo crontab -u \1 -l 2>\&1/' | grep -v "no crontab for" | sh

2
太棒了!我添加了一些变体来查看cron作业所属的用户,并在结果之间cat /etc/passwd | sed 's/^\([^:]*\):.*$/echo "\ncrontab for \1:"; sudo crontab -u \1 -l 2>\&1/' | grep -v "no crontab for" | sh

9

我喜欢上面简单的一线回答:

对于$(cut -f1-d:/ etc / passwd)中的用户; 做crontab -u $ user -l; 做完了

但是,Solaris没有-u标志,并且不打印其检查的用户,因此您可以像这样修改它:

for user in $(cut -f1 -d: /etc/passwd); do echo User:$user; crontab -l $user 2>&1 | grep -v crontab; done

当不允许某个帐户使用cron等时,您将获得用户列表,而不会出现crontab引发的错误。请注意,在Solaris中,角色也可以位于/ etc / passwd中(请参见/ etc / user_attr)。



凉。TIL不用于。
squarism


7

以下内容删除了没有crontab的用户的注释,空白行和错误。您所剩下的只是一个清晰的用户及其工作清单。

注意sudo第二行中的使用。如果您已经是root用户,请将其删除。

for USER in $(cut -f1 -d: /etc/passwd); do \
USERTAB="$(sudo crontab -u "$USER" -l 2>&1)";  \
FILTERED="$(echo "$USERTAB"| grep -vE '^#|^$|no crontab for|cannot use this program')";  \
if ! test -z "$FILTERED"; then  \
echo "# ------ $(tput bold)$USER$(tput sgr0) ------";  \
echo "$FILTERED";  \
echo "";  \
fi;  \
done

输出示例:

# ------ root ------
0 */6 * * * /usr/local/bin/disk-space-notify.sh
45 3 * * * /opt/mysql-backups/mysql-backups.sh
5 7 * * * /usr/local/bin/certbot-auto renew --quiet --no-self-upgrade

# ------ sammy ------
55 * * * * wget -O - -q -t 1 https://www.example.com/cron.php > /dev/null

我在Ubuntu(12至16)和Red Hat(5至7)上​​使用它。


5

取决于您的cron版本。在FreeBSD上使用Vixie cron,我可以执行以下操作:

(cd /var/cron/tabs && grep -vH ^# *) 

如果我希望它不再使用制表符,则可以执行以下操作:

(cd /var/cron/tabs && grep -vH ^# * | sed "s/:/      /")

这是sed替换部分中的文字选项卡。

遍历用户/etc/passwdcrontab -l -u $user为每个用户做事可能与系统更不相关。


4

感谢这个非常有用的脚本。在旧系统(Red Hat Enterprise 3,它处理不同的egrep和字符串中的制表符)以及在/etc/cron.d/中没有任何内容的其他系统上运行该脚本时,我遇到了一些小问题。因此,这里有一个补丁程序可以使其在这种情况下工作:

2a3,4
> #See:  http://stackoverflow.com/questions/134906/how-do-i-list-all-cron-jobs-for-all-users
>
27c29,30
<         match=$(echo "${line}" | egrep -o 'run-parts (-{1,2}\S+ )*\S+')
---
>         #match=$(echo "${line}" | egrep -o 'run-parts (-{1,2}\S+ )*\S+')
>         match=$(echo "${line}" | egrep -o 'run-parts.*')
51c54,57
< cat "${CRONDIR}"/* | clean_cron_lines >>"${temp}"  # */ <not a comment>
---
> sys_cron_num=$(ls /etc/cron.d | wc -l | awk '{print $1}')
> if [ "$sys_cron_num" != 0 ]; then
>       cat "${CRONDIR}"/* | clean_cron_lines >>"${temp}"  # */ <not a comment>
> fi
67c73
<     sed "1i\mi\th\td\tm\tw\tuser\tcommand" |
---
>     sed "1i\mi${tab}h${tab}d${tab}m${tab}w${tab}user${tab}command" |

我不确定第一个egrep中的更改是否是一个好主意,但是好了,此脚本已经在RHEL3、4、5和Debian5上进行了测试,没有任何问题。希望这可以帮助!


3

在@Kyle上构建

for user in $(tail -n +11 /etc/passwd | cut -f1 -d:); do echo $user; crontab -u $user -l; done

为了避免通常在/ etc / passwd顶部的注释,

在macosx上

for user in $(dscl . -list /users | cut -f1 -d:); do echo $user; crontab -u $user -l; done    

3
您不应该不grep -v '^#依赖魔术数字11吗?
rr-

1
Red Hat / CentOS发行版不会在用户的crontab的开始处写有用的提示,因此删掉前11行将消除其内容。如果Ubuntu用户已经编辑了自己的crontab并删除了所有内容,那也是一样。
戴尔·安德森



3

您可以为所有用户列表编写:

sudo crontab -u userName -l

,

你也可以去

cd /etc/cron.daily/
ls -l
cat filename

该文件将列出时间表

cd /etc/cron.d/
ls -l
cat filename

3

致歉,并感谢yukondude。

尽管这不是一项完美的工作,但我试图总结一下计时设置以便于阅读,并且我不会碰“每个星期五”或“仅星期一”之类的内容。

这是版本10-现在:

  • 运行快得多
  • 具有可选的进度字符,因此您可以进一步提高速度。
  • 使用分隔线分隔标题和输出。
  • 当可以总结所有遇到的时间间隔时,以紧凑格式输出。
  • 接受一年中月份的Jan ... Dec描述符
  • 一周中的某几天接受Mon ... Sun描述符
  • 尝试处理缺失的anacron的debian风格虚拟化
  • 尝试处理crontab行,这些行使用“ [-x ...]”在预测试可执行性之后运行文件
  • 尝试处理crontab行,这些行使用“ -v”命令在预测试可执行性之后运行文件
  • 允许使用间隔范围和列表。
  • 支持特定于用户的/ var / spool crontab文件中的运行部件用法。

我现在在这里完整发布脚本。

https://gist.github.com/myshkin-uk/d667116d3e2d689f23f18f6cd3c71107


2

由于这是遍历文件(/etc/passwd)并执行操作的问题,因此我缺少关于如何逐行(和/或逐字段)读取文件(数据流,变量)的正确方法。)?

while IFS=":" read -r user _
do
   echo "crontab for user ${user}:"
   crontab -u "$user" -l
done < /etc/passwd

这将用作字段分隔符/etc/passwd逐行读取:。通过说read -r user _,我们使$user第一个字段保留,_其余字段保留(忽略字段只是一个垃圾变量)。

这样,我们便可以crontab -u使用$user为安全起见引用的变量进行调用(如果它包含空格,则在该文件中不太可能,但是您永远不会知道)。


1

在Solaris上,对于特定的已知用户名:

crontab -l username

所有其他* Nix将需要-u修饰符:

crontab -u username -l

要一次在Solaris上一次获得所有用户的工作,就像上面的其他帖子一样:

for user in $(cut -f1 -d: /etc/passwd); do crontab -l $user 2>/dev/null; done


-2

此脚本将Crontab输出到文件,并且还列出所有确认没有crontab条目的用户:

for user in $(cut -f1 -d: /etc/passwd); do 
  echo $user >> crontab.bak
  echo "" >> crontab.bak
  crontab -u $user -l >> crontab.bak 2>> > crontab.bak
done

//,RobFrost,您确定此脚本(如此处编写的)始终有效吗?
弥敦道(Nathan Basanese)'16

请参阅为什么不阅读带有“ for”的行。另外,在您发布此问题之前,已经回答了很多次。
fedorqui'SO停止伤害
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.