SKILL.md
定时任务管理
概述
Cron 定时任务配置、日志监控、故障排查等技能。
Crontab 基础
管理命令
# 编辑当前用户的 crontab
crontab -e
# 查看当前用户的 crontab
crontab -l
# 删除当前用户的 crontab
crontab -r
管理其他用户的 crontab(需要 root)
crontab -u username -e
crontab -u username -l
### 时间格式
┌───────────── 分钟 (0-59)
│ ┌───────────── 小时 (0-23)
│ │ ┌───────────── 日 (1-31)
│ │ │ ┌───────────── 月 (1-12)
│ │ │ │ ┌───────────── 星期 (0-7, 0和7都是周日)
│ │ │ │ │
-
-
-
-
- command
### 特殊字符
- # 任意值
, # 列表 (1,3,5)
- # 范围 (1-5)
/ # 步长 (*/5 每5分钟)
示例
0 # 每小时整点
/15 * # 每15分钟
0 9-17 * # 9点到17点每小时
0 0 1-5 # 工作日零点
0 0 1,15 # 每月1号和15号零点
### 特殊时间字符串
@reboot # 系统启动时
@yearly # 每年 (0 0 1 1 *)
@monthly # 每月 (0 0 1 )
@weekly # 每周 (0 0 0)
@daily # 每天 (0 0 *)
@hourly # 每小时 (0 )
## 配置文件
### 用户 crontab
位置
/var/spool/cron/crontabs/username # Debian/Ubuntu
/var/spool/cron/username # CentOS/RHEL
格式
SHELL=/bin/bash
PATH=/usr/local/bin:/usr/bin:/bin
MAILTO=admin@example.com
任务
0 2 * /usr/local/bin/backup.sh
### 系统 crontab
/etc/crontab
SHELL=/bin/bash
PATH=/sbin:/bin:/usr/sbin:/usr/bin
MAILTO=root
格式:多了用户字段
分 时 日 月 周 用户 命令
0 2 * root /usr/local/bin/backup.sh
### cron.d 目录
/etc/cron.d/myapp
SHELL=/bin/bash
PATH=/usr/local/bin:/usr/bin:/bin
0 appuser /opt/myapp/hourly-task.sh
0 2 * root /opt/myapp/daily-backup.sh
### 预定义目录
/etc/cron.hourly/ # 每小时执行
/etc/cron.daily/ # 每天执行
/etc/cron.weekly/ # 每周执行
/etc/cron.monthly/ # 每月执行
放入可执行脚本即可
chmod +x /etc/cron.daily/myscript
## 最佳实践
### 脚本模板
#!/bin/bash
/usr/local/bin/cron-task.sh
日志文件
LOG_FILE="/var/log/cron-task.log"
锁文件(防止重复执行)
LOCK_FILE="/var/run/cron-task.lock"
检查锁
if [ -f "$LOCK_FILE" ]; then
echo "$(date): Task already running" >> "$LOG_FILE"
exit 1
fi
创建锁
trap "rm -f $LOCK_FILE" EXIT
touch "$LOCK_FILE"
记录开始
echo "$(date): Task started" >> "$LOG_FILE"
执行任务
/path/to/actual/command >> "$LOG_FILE" 2>&1
EXIT_CODE=$?
记录结束
echo "$(date): Task finished with exit code $EXIT_CODE" >> "$LOG_FILE"
exit $EXIT_CODE
### Crontab 条目
推荐写法
1. 使用绝对路径
2. 重定向输出
3. 添加注释
每日备份 - 凌晨2点
0 2 * /usr/local/bin/backup.sh >> /var/log/backup.log 2>&1
每5分钟健康检查
/5 * /usr/local/bin/healthcheck.sh > /dev/null 2>&1
每周日志清理 - 周日凌晨3点
0 3 0 /usr/local/bin/cleanup-logs.sh >> /var/log/cleanup.log 2>&1
### 环境变量
在 crontab 中设置
SHELL=/bin/bash
PATH=/usr/local/bin:/usr/bin:/bin
HOME=/home/user
MAILTO=admin@example.com
或在脚本中加载
#!/bin/bash
source /etc/profile
source ~/.bashrc
## 日志与监控
### 查看日志
系统日志
grep CRON /var/log/syslog # Debian/Ubuntu
grep CRON /var/log/cron # CentOS/RHEL
实时监控
tail -f /var/log/syslog | grep CRON
查看邮件(如果配置了 MAILTO)
cat /var/mail/username
### 调试技巧
手动测试脚本
/usr/local/bin/myscript.sh
模拟 cron 环境
env -i /bin/bash --noprofile --norc -c '/usr/local/bin/myscript.sh'
检查 cron 服务状态
systemctl status cron # Debian/Ubuntu
systemctl status crond # CentOS/RHEL
## 常见场景
### 场景 1:数据库备份
每天凌晨2点备份 MySQL
0 2 * /usr/bin/mysqldump -u root -p'password' database | gzip > /backup/db_$(date +\%Y\%m\%d).sql.gz
注意:% 需要转义为 \%
### 场景 2:日志轮转
每天压缩并清理7天前的日志
0 0 find /var/log/myapp -name ".log" -mtime +7 -delete
0 1 gzip /var/log/myapp/.log.1
### 场景 3:监控告警
每5分钟检查服务状态
/5 * /usr/local/bin/check-service.sh || /usr/local/bin/send-alert.sh
### 场景 4:使用 flock 防止重复
使用 flock 确保单实例运行
/5 * /usr/bin/flock -n /var/lock/mytask.lock /usr/local/bin/mytask.sh
## 故障排查
问题
排查方法
任务不执行
检查 cron 服务状态、日志
权限错误
检查脚本权限、用户权限
环境变量问题
在脚本中设置 PATH
命令找不到
使用绝对路径
输出丢失
重定向到日志文件
检查 cron 服务
systemctl status cron
检查用户是否被禁止
cat /etc/cron.allow
cat /etc/cron.deny
检查语法
crontab -l | grep -v '^#' | while read line; do
echo "Checking: $line"
done