是否有命令或现有脚本可以让我一次查看 *NIX 系统的所有计划的 cron 作业?我希望它包含所有用户 crontab,/etc/crontab
以及/etc/cron.d
. 也很高兴看到run-parts
in运行的特定命令/etc/crontab
。
理想情况下,我希望以漂亮的列形式输出并以某种有意义的方式排序。
然后,我可以合并来自多个服务器的这些列表,以查看整体“事件时间表”。
我正要自己写这样的脚本,但如果有人已经麻烦了......
您必须以 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
我最终编写了一个脚本(我试图自学 bash 脚本的更精细的点,所以这就是为什么你在这里看不到 Perl 之类的东西)。这不是一件简单的事情,但它可以满足我的大部分需求。它使用 Kyle 的建议来查找单个用户的 crontab,但也处理/etc/crontab
(包括run-parts
in /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}"
在 Ubuntu 或 debian 下,您可以查看 crontab /var/spool/cron/crontabs/
,然后每个用户的文件都在那里。当然,这仅适用于特定于用户的 crontab。
对于 Redhat 6/7 和 Centos,crontab 位于/var/spool/cron/
.
这将显示所有用户的所有 crontab 条目。
sed 's/^\([^:]*\):.*$/crontab -u \1 -l 2>\&1/' /etc/passwd | sh | grep -v "no crontab for"
取决于您的 linux 版本,但我使用:
tail -n 1000 /var/spool/cron/*
作为根。非常简单而且非常简短。
给我这样的输出:
==> /var/spool/cron/root <==
15 2 * * * /bla
==> /var/spool/cron/my_user <==
*/10 1 * * * /path/to/script
Kyle Burton 的答案的一个小改进,改进了输出格式:
#!/bin/bash
for user in $(cut -f1 -d: /etc/passwd)
do echo $user && crontab -u $user -l
echo " "
done
getent passwd | cut -d: -f1 | perl -e'while(<>){chomp;$l = `crontab -u $_ -l 2>/dev/null`;print "$_\n$l\n" if $l}'
这避免了直接与 passwd 混淆,跳过没有 cron 条目的用户,对于那些拥有它们的用户,它会打印出用户名以及他们的 crontab。
不过主要是把它放在这里,这样我以后可以找到它,以防我需要再次搜索它。
从 ROOT 用户获取列表。
for user in $(cut -f1 -d: /etc/passwd); do echo $user; sudo crontab -u $user -l; done
如果您使用 NIS 检查集群,根据马特的回答 /var/spool/cron/tabs,查看用户是否有 crontab 条目的唯一方法是。
grep -v "#" -R /var/spool/cron/tabs
这个脚本在 CentOS 中对我有用,可以列出环境中的所有 cron:
sudo cat /etc/passwd | sed 's/^\([^:]*\):.*$/sudo crontab -u \1 -l 2>\&1/' | grep -v "no crontab for" | sh
我喜欢上面简单的单行答案:
对于 $(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)。
for user in $(cut -f1 -d: /etc/passwd);
do
echo $user; crontab -u $user -l;
done
下面从没有 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)上使用它。
取决于你的 cron 版本。在 FreeBSD 上使用 Vixie cron,我可以这样做:
(cd /var/cron/tabs && grep -vH ^# *)
如果我想要更多的制表符分隔,我可能会做这样的事情:
(cd /var/cron/tabs && grep -vH ^# * | sed "s/:/ /")
这是 sed 替换部分中的文字选项卡。
/etc/passwd
循环访问用户并crontab -l -u $user
为每个用户执行操作可能更加独立于系统。
虽然许多答案产生了有用的结果,但我认为为这项任务维护复杂脚本的忙碌是不值得的。这主要是因为大多数发行版使用不同的 cron 守护进程。
$ \cat ~jaroslav/bin/ls-crons
#!/bin/bash
getent passwd | awk -F: '{ print $1 }' | xargs -I% sh -c 'crontab -l -u % | sed "/^$/d; /^#/d; s/^/% /"' 2>/dev/null
echo
cat /etc/crontab /etc/anacrontab 2>/dev/null | sed '/^$/d; /^#/d;'
echo
run-parts --list /etc/cron.hourly;
run-parts --list /etc/cron.daily;
run-parts --list /etc/cron.weekly;
run-parts --list /etc/cron.monthly;
$ sudo ls-cron
$ sudo ~jaroslav/bin/ls-crons
jaroslav */5 * * * * mv ~/java_error_in_PHPSTORM* ~/tmp 2>/dev/null
jaroslav 5 */24 * * * ~/bin/Find-home-files
jaroslav * 7 * * * cp /T/fortrabbit/ssh-config/fapps.tsv /home/jaroslav/reference/fortrabbit/fapps
jaroslav */8 1 * * * make -C /T/fortrabbit/ssh-config discover-apps # >/dev/null
jaroslav */7 * * * * getmail -r jazzoslav -r fortrabbit 2>/dev/null
jaroslav */1 * * * * /home/jaroslav/bin/checkmail
jaroslav * 9-18 * * * getmail -r fortrabbit 2>/dev/null
SHELL=/bin/bash
PATH=/sbin:/bin:/usr/sbin:/usr/bin
MAILTO=root
HOME=/
SHELL=/bin/sh
PATH=/sbin:/bin:/usr/sbin:/usr/bin
MAILTO=root
RANDOM_DELAY=45
START_HOURS_RANGE=3-22
1 5 cron.daily nice run-parts /etc/cron.daily
7 25 cron.weekly nice run-parts /etc/cron.weekly
@monthly 45 cron.monthly nice run-parts /etc/cron.monthly
/etc/cron.hourly/0anacron
/etc/cron.daily/logrotate
/etc/cron.daily/man-db
/etc/cron.daily/mlocate
/etc/cron.weekly/mdadm
/etc/cron.weekly/pfl
SHELL=/bin/sh
PATH=/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin
17 * * * * root cd / && run-parts --report /etc/cron.hourly
25 6 * * * root test -x /usr/sbin/anacron || ( cd / && run-parts --report /etc/cron.daily )
47 6 * * 7 root test -x /usr/sbin/anacron || ( cd / && run-parts --report /etc/cron.weekly )
52 6 1 * * root test -x /usr/sbin/anacron || ( cd / && run-parts --report /etc/cron.monthly )
/etc/cron.hourly/btrfs-quota-cleanup
/etc/cron.hourly/ntpdate-debian
/etc/cron.daily/apport
/etc/cron.daily/apt-compat
/etc/cron.daily/apt-show-versions
/etc/cron.daily/aptitude
/etc/cron.daily/bsdmainutils
/etc/cron.daily/dpkg
/etc/cron.daily/logrotate
/etc/cron.daily/man-db
/etc/cron.daily/mlocate
/etc/cron.daily/passwd
/etc/cron.daily/popularity-contest
/etc/cron.daily/ubuntu-advantage-tools
/etc/cron.daily/update-notifier-common
/etc/cron.daily/upstart
/etc/cron.weekly/apt-xapian-index
/etc/cron.weekly/man-db
/etc/cron.weekly/update-notifier-common
Ubuntu:
巴布亚新几内亚:
感谢这个非常有用的脚本。我在旧系统上运行它时遇到了一些小问题(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 上进行了测试,没有任何问题。希望这可以帮助!
您可以为所有用户列表编写:
sudo crontab -u userName -l
,
你也可以去
cd /etc/cron.daily/
ls -l
cat filename
此文件将列出时间表
cd /etc/cron.d/
ls -l
cat filename
建立在@Kyle 之上
for user in $(tail -n +11 /etc/passwd | cut -f1 -d:); do echo $user; crontab -u $user -l; done
为了避免通常在 /etc/passwd 顶部的注释,
在 macOS 上
for user in $(dscl . -list /users | cut -f1 -d:); do echo $user; crontab -u $user -l; done
我认为下面会有一个更好的班轮。例如,如果您在 NIS 或 LDAP 中有用户,他们就不会在 /etc/passwd 中。这将为您提供每个已登录用户的 crontab。
for I in `lastlog | grep -v Never | cut -f1 -d' '`; do echo $I ; crontab -l -u $I ; done
很抱歉并感谢 yukondude。
我试图总结时间设置以便于阅读,虽然这不是一个完美的工作,而且我不会触及“每个星期五”或“仅在星期一”的东西。
这是版本 10 - 现在是:
我现在在这里完整地发布脚本。
https://gist.github.com/myshkin-uk/d667116d3e2d689f23f18f6cd3c71107
由于它是遍历文件(/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
安全(如果它包含空格怎么办?在这样的文件中不太可能,但你永远不会知道)。
我倾向于使用以下小命令来列出单个用户的所有作业以及带有现代 bash 控制台的基于 Unix 的操作系统上的所有用户:
1. 单用户
echo "Jobs owned by $USER" && crontab -l -u $USER
2.所有用户
for wellknownUser in $(cut -f1 -d: /etc/passwd);
do
echo "Jobs owned by $wellknownUser";
crontab -l -u $wellknownUser;
echo -e "\n";
sleep 2; # (optional sleep 2 seconds) while drinking a coffee
done
我在一个衬里脚本下制作了它,它可以为我列出所有用户的所有 cron 作业。
cat /etc/passwd |awk -F ':' '{print $1}'|while read a;do crontab -l -u ${a} ; done
对我来说,看看 /var/spool/cron/crontabs 是最好的方法
此脚本将 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