inotify与rsync实现实时同步记录文档

安装

  • 安装rsync
yum -y install rsync
  • 安装inotify-tools

这是一个实时监听文件变换的工具

wget -O /etc/yum.repos.d/epel.repo http://mirrors.aliyun.com/repo/epel-7.repo
yum -y install inotify-tools

配置

  • 配置rsync

这个配置客户端与服务端都要设置。

#启动rsync
systemctl start rsyncd.service
#设置rsyncd开机启动
systemctl enable rsyncd.service 
# 设置同步的密码
vim /etc/rsync.pass
demo:123456
  • 配置服务端rsync
vim /etc/rsyncd.conf
uid = root  #配置同步的用户
gid = root  #配置同步的组
[log]   
#保存的位置
path= /storage/cf_elk/logstash/log
read only = no
  • 配置客户端rsync

vim /etc/rsyncd.conf
uid = root
gid = root
#以上两个用户组需与服务端一一对应,然后保存退出
#在运行脚本的目录下配置一下file文件,文件内容为同步的目录.下面会用到。
vim file
/usr/local/src/logs/
#然后运行一下命令测试一下是否可同步有无报错
rsync -avz --password-file=/etc/rsync.pass /usr/local/src/logs/  root@127.0.0.1::log
# rsync+参数+密码配置+同步路径+同步用户@IP地址::对应服务端的模块

如果在同步的时候有所报错的话那就创建一个目录的用户,然后授权给这个用户。服务端与客户端的文件用户需一同。命令:seradd rsync && chown -R rsync:rsync 目录或者文件

  • 配置监听脚本

在上面手动执行rsync的时候没有问题的情况下开始配置哦。配置脚本如下:

#!/bin/bash
#这是一个日志同步脚本
src=/usr/local/src/logs/                # 需要同步的源路径
des=log                             # 目标服务器上 rsync --daemon模块名称
rsync_passwd_file=/etc/rsync.pass            # rsync验证的密码文件
ip1=127.0.0.1               # 目标服务器1
#ip2=192.168.0.19                 # 目标服务器2
user=root                           # rsync --daemon定义的验证用户名
cd ${src}                            
# 此方法中,由于rsync同步的特性,这里必须要先cd到源目录,inotify再监听 ./ 才能rsync同步后目录结构一致,
inotifywait -mrq --format  '%Xe %w%f' -e modify,create,delete,attrib,close_write,move ./  | while read file
# 把监控到有发生更改的"文件路径列表"循环
do
        INO_EVENT=$(echo $file | awk '{print $1}')      # 把inotify输出切割 把事件类型部分赋值给INO_EVENT
        INO_FILE=$(echo $file | awk '{print $2}')       # 把inotify输出切割 把文件路径部分赋值给INO_FILE
        echo "-------------------------------$(date)------------------------------------"
        echo $file
        #增加、修改、写入完成、移动进事件
        #增、改放在同一个判断,因为他们都肯定是针对文件的操作,即使是新建目录,要同步的也只是一个空目录,不会影响速度。
        if [[ $INO_EVENT =~ 'CREATE' ]] || [[ $INO_EVENT =~ 'MODIFY' ]] || [[ $INO_EVENT =~ 'CLOSE_WRITE' ]] || [[ $INO_EVENT =~ 'MOVED_TO' ]]         # 判断事件类型
        then
                echo 'CREATE or MODIFY or CLOSE_WRITE or MOVED_TO'
                rsync -avzcR --password-file=${rsync_passwd_file} $(dirname ${INO_FILE}) ${user}@${ip1}::${des} &&
# INO_FILE变量代表路径哦  -c校验文件内容
                rsync -avzcR --password-file=${rsync_passwd_file} $(dirname ${INO_FILE}) ${user}@${ip2}::${des}
#仔细看 上面的rsync同步命令 源是用了$(dirname ${INO_FILE})变量 即每次只针对性的同步发生改变的文件的目录(只同步目标文件的方法在生产环境的某些极端
#环境下会漏文件 现在可以在不漏文件下也有不错的速度 做到平衡)
#然后用-R参数把源的目录结构递归到目标后面 保证目录结构一致性
        fi
        #删除、移动出事件
        if [[ $INO_EVENT =~ 'DELETE' ]] || [[ $INO_EVENT =~ 'MOVED_FROM' ]]
        then
                echo 'DELETE or MOVED_FROM'
                rsync -avzR --delete --password-file=${rsync_passwd_file} $(dirname ${INO_FILE}) ${user}@${ip1}::${des} 
                #&& rsync -avzR --delete --password-file=${rsync_passwd_file} $(dirname ${INO_FILE}) ${user}@${ip2}::${des}
#看rsync命令 如果直接同步已删除的路径${INO_FILE}会报no such or directory错误 所以这里同步的源是被删文件或目录的上一级路径
#并加上--delete来删除目标上有而源中没有的文件,这里不能做到指定文件删除,如果删除的路径越靠近根,则同步的目录月多,同步删除的操作就越花时间。
#这里有更好方法的同学,欢迎交流。
        fi
        #修改属性事件 指 touch chgrp chmod chown等操作
        if [[ $INO_EVENT =~ 'ATTRIB' ]]
        then
                echo 'ATTRIB'
                if [ ! -d "$INO_FILE" ]
# 如果修改属性的是目录 则不同步,因为同步目录会发生递归扫描,等此目录下的文件发生同步时,rsync会顺带更新此目录。
                then
                        rsync -avzcR --password-file=${rsync_passwd_file} $(dirname ${INO_FILE}) ${user}@${ip1}::${des}       
                    #    &&   rsync -avzcR --password-file=${rsync_passwd_file} $(dirname ${INO_FILE}) ${user}@${ip2}::${des}
                fi
        fi
done
> 脚本中用到的 `inotifywait`,可能你需要改一下路径,我是yum 安装的,所以可以直接用如果是其他方式安装的可能就不太一样。然后脚本中的路径、IP地址、用户名都需改一下哦。des,密码路径的话如果上面都与我一样的话那就无需更改了哦。
  • 运行脚本
nohup bash rsync.sh > rsync.logs 2>&1 &

查看一下运行日志是否有报错,然后去服务端查看一下大小是否与客户端的目录大小一致,如果只是差一点的话问题不大。根据你线上的环境来定哦。实在不确定是否同步可以酌情选即可文件看一下哈。

参考链接

inotify实时同步
rsync问题参考

posted @ 2020-12-11 18:10  自在拉基  阅读(369)  评论(0编辑  收藏  举报