zoukankan      html  css  js  c++  java
  • inotify+rsync

    一、环境准备

    操作系统:CentOS release 6.8 (Final) x86_64

    服务器IP:

    rsync_server(数据源)192.168.80.16
    rsync_client(目标端)192.168.80.140

    同步目录:

    rsync_server       /app/rsync_server
    rsync_client       /app/rsync_client 

    二、安装及配置rsync

    客户端配置(目标端)

    1、安装rsync

    # yum -y install rsync xinetd
    # cp /etc/xinetd.d/rsync{,.bak}
    # vim /etc/xinetd.d/rsync
    service rsync
    {
            disable = no            #修改为no
            flags           = IPv6
            socket_type     = stream
            wait            = no
            user            = root
            server          = /usr/bin/rsync
            server_args     = --daemon
            log_on_failure  += USERID
    }
    # /etc/init.d/xinetd start 

    2、配置rsync

    # vim /etc/rsyncd.conf    #创建配置文件
    
    logfile = /var/log/rsyncd.log    #日志文件位置,启动rsync后自动产生这个文件,无需提前创建
    pidfile = /var/run/rsyncd.pid    #pid文件的存放位置
    lockfile = /var/run/rsync.lock   #支持max connections参数的锁文件
    secretsfile = /etc/rsync_client.pwd    #用户认证配置文件,里面保存用户名称和密码,后面会创建这个文件
    motdfile = /etc/rsyncd.Motd    #rsync启动时欢迎信息页面文件位置(文件内容自定义)
    [backup]   #自定义名称
    path = /    #rsync服务端数据目录路径
    comment = rsync_client    #模块名称与[app_rsync_client]自定义名称相同
    uid = root    #设置rsync运行权限为root
    gid = root    #设置rsync运行权限为root
    port =873
    use chroot = no    #默认为true,修改为no,增加对目录文件软连接的备份
    read only = no    设置rsync服务端文件为读写权限
    list = no    #不显示rsync服务端资源列表
    mac connections = 200
    timeout = 600
    auth users = ronghui    #执行数据同步的用户名,可以设置多个,用英文状态下逗号隔开
    hosts allow = 192.168.80.16   #允许进行数据同步的客户端IP地址,可以设置多个,用英文状态下逗号隔开
    hosts deny = 0.0.0.0/0    #禁止数据同步的客户端IP地址,可以设置多个,用英文状态下逗号隔开,先允许后拒绝

    3、配置rsync同步的账户密码

    # vim /etc/rsync.pass    #配置文件,添加以下内容
    ronghui:ronghui123    #格式,用户名:密码,可以设置多个,每行一个用户名:密码

    4、赋权启动rsync

    # chmod 600 /etc/rsyncd.conf 
    # chmod 600 /etc/rsync.pass 
    # /etc/init.d/xinetd restart

    服务端配置(数据源)

    1、安装rsync

    # yum install rsync xinetd
    # vim /etc/xinetd.d/rsync
    service rsync
    {
            disable = no    #修改为no
            flags           = IPv6
            socket_type     = stream
            wait            = no
            user            = root
            server          = /usr/bin/rsync
            server_args     = --daemon
            log_on_failure  += USERID
    }
    

    2、配置rsync同步的账户密码

    # vim /etc/passwd.txt
    ronghui123
    
    # chmod 600 /etc/passwd.txt
    

    3、测试手动同步

    
    在rsync_server的/test目录下创建文件test666.txt,在rsync_server端运行同步命令同步数据:
    
    rsync -avH --port=873 --progress --delete  /test ronghui@192.168.80.140::backup --password-file=/etc/passwd.txt
    
    注释:
    /test             #数据源的目录
    -password-file=/etc/passwd.txt #数据源的密码文件
    ronghui@192.168.80.140::backup #ronghui目标端rsync服务端配置的用户名,backup目标端rsync服务端配置的模块名称
    
    检查客户端rsync_client目录
    
    # ls /app/rsync_client/
    test.txt

    三、安装Inotify-tools实时触发rsync进行同步

    这里可以参考github上的官方wiki文档(包含安装及配置使用示例)
    https://github.com/rvoicilas/inotify-tools/wiki

    1、下载安装Inotify-tools

    # uname -r        #Linux下支持inotify的内核最小为2.6.13
    2.6.32-642.el6.x86_64
    
    # 安装前要先下载epel源
    # yum install inotify-tools -y
    
    查看其程序是否安装成功
    # rpm -qa inotify-tools
    inotify-tools-3.14-1.el6.x86_64
    
    查看程序包含的文件
    #rpm -ql inotify-tools
    /usr/bin/inotifywait
    /usr/bin/inotifywatch
    /usr/lib64/libinotifytools.so.0
    /usr/lib64/libinotifytools.so.0.4.1
    /usr/share/doc/inotify-tools-3.14
    /usr/share/doc/inotify-tools-3.14/AUTHORS
    /usr/share/doc/inotify-tools-3.14/COPYING
    /usr/share/doc/inotify-tools-3.14/ChangeLog
    /usr/share/doc/inotify-tools-3.14/NEWS
    /usr/share/doc/inotify-tools-3.14/README
    /usr/share/man/man1/inotifywait.1.gz
    /usr/share/man/man1/inotifywatch.1.gz
    

    2、配置inotify-tools

    # sysctl -a|egrep -i "max_queued_events|max_user_watches|max_user_instances"    #修改inotify默认参数(inotify默认内核参数值太小)
    fs.inotify.max_user_instances = 128
    fs.inotify.max_user_watches = 8192
    fs.inotify.max_queued_events = 16384
    fs.epoll.max_user_watches = 201420
    
    # vim /etc/sysctl.conf 添加
    fs.inotify.max_queued_events = 99999999
    fs.inotify.max_user_watches = 99999999
    fs.inotify.max_user_instances = 65535
    
    #sysctl  -p   参数立即生效
    
    # cat /proc/sys/fs/inotify/{max_user_instances,max_user_watches,max_queued_events}  #检查参数是否生效
    65535
    99999999
    99999999
    
    注释:
        max_queued_events:inotify队列最大长度,如果值太小,会出现"** Event Queue Overflow **"错误,导致监控文件不准确
        max_user_watches:要同步的文件包含多少目录,可以用:find /app/rsync_server/ -type d | wc -l 统计,必须保证max_user_watches值大于统计结果(这里/app/rsync_server/为同步文件目录)
        max_user_instances:每个用户创建inotify实例最大值

    3、创建实时同步脚本

     # vim /usr/local/inotify/rsync.sh
    #!/bin/bash
    src_dir="/test"
    dst_dir="backup"
    exclude_dir="/usr/local/inotify/exclude.list"
    rsync_user="ronghui"
    rsync_passwd="/etc/passwd.txt"
    dst_ip="192.168.80.140"
    rsync_command(){
    rsync -avH --port=873 --progress --delete $src_dir $rsync_user@$ip::$dst_dir --password-file=$rsync_passwd
    }
    for ip in $dst_ip;do
    rsync_command
    done
    /usr/bin/inotifywait -mrq --timefmt '%d/%m/%y %H:%M' --format '%T %w%f%e' -e close_write,modify,delete,create,attrib,move $src_dir
    | while read file;do
    for ip in $dst_ip;do
    rsync_command
    echo "${file} was rsynced" >> /tmp/rsync.log 2>&1
    done
    done 注释: src_dir="/test" #源服务器同步目录 dst_dir="backup" #目标服务器rsync同步目录模块名称 rsync_user="rronghui" #目标服务器rsync同步用户名 rsync_passwd="/etc/passwd.txt" #目标服务器rsync同步用户的密码在源服务器的存放路径 dst_ip="192.168.80.140" #目标服务器ip,多个ip用空格分开
    
    ##赋权,添加开机启动
    
    # chmod +x /usr/local/inotify/rsync.sh
    # touch /usr/local/inotify/exclude.list
    # vim /etc/rc.d/rc.local
    nohup /bin/sh /usr/local/inotify/rsync.sh &
    # nohup /bin/sh /usr/local/inotify/rsync.sh &
    

    4、测试

    在rsync_server(数据源)192.168.80.16的/test创建文件
    # cd /app/rsync_server
    # touch test{1..9}
    # touch test{a..j}
    # ls
    test1  test2  test3  test4  test5  test6  test7  test8  test9  testa  testb  testc  testd  teste  testf  testg  testh  testi  testj
    
    在rsync_client(目标端)192.168.80.140上查看已经同步
    # cd /app/rsync_client
    # ls
    test1  test2  test3  test4  test5  test6  test7  test8  test9  testa  testb  testc  testd  teste  testf  testg  testh  testi  testj

    如果以上测试都通过,说明inotify实时触发rsync同步脚本运行正常。
    至此,Linux下Rsync+Inotify-tools实现数据实时同步完成。如果要双向同步可以把以上反过来部署一次。

    FAQ

    Q1:
    #rsync -avH --port=873 --progress --delete /app/rsync_client/ rsync@192.168.0.45::app_rsync_client --password-file=/etc/passwd.txt

    @ERROR: auth failed on module app_rsync_client
    rsync error: error starting client-server protocol (code 5) at main.c(1503) [sender=3.0.6]

    A:如果出现这个错误,请详细检查配置文件是否有误,建议删掉无用的注释

    Q2:
    #rsync -avH --port=873 --progress --delete /app/rsync_client rsync@192.168.0.45::app_rsync_client --password-file=/etc/passwd.txt

    sending incremental file list
    rsync: link_stat "/app/rsync_client" failed: No such file or directory (2)

    A:检查客户端及服务端文件夹是否存在,这里应该还有一个坑,就是这里是在服务端(数据源)同步,目录应该指向“/app/rsync_client”

    # vim /usr/local/inotify/rsync.sh
    #!/bin/bash
    src_dir="/test"
    dst_dir="backup"
    exclude_dir="/usr/local/inotify/exclude.list"
    rsync_user="ronghui"
    rsync_passwd="/etc/passwd.txt"
    dst_ip="192.168.80.140"
    rsync_command(){
    rsync -avH --port=873 --progress --delete $src_dir $rsync_user@$ip::$dst_dir --password-file=$rsync_passwd
    }
    for ip in $dst_ip;do
    rsync_command
    done
    /usr/bin/inotifywait -mrq --timefmt '%d/%m/%y %H:%M' --format '%T %w%f%e' -e close_write,modify,delete,create,attrib,move $src_dir
    | while read file;do
    for ip in $dst_ip;do
    rsync_command
    echo "${file} was rsynced" >> /tmp/rsync.log 2>&1
    done
    done

    #!/bin/bash

    src=/data/                           # 需要同步的源路径
    des=data                             # 目标服务器上 rsync --daemon 发布的名称,rsync --daemon这里就不做介绍了,网上搜一下,比较简单。
    rsync_passwd_file=/etc/rsyncd.passwd            # rsync验证的密码文件
    ip1=192.168.0.18                 # 目标服务器1
    ip2=192.168.0.19                 # 目标服务器2
    user=root                            # rsync --daemon定义的验证用户名
    cd ${src}                            
    # 此方法中,由于rsync同步的特性,这里必须要先cd到源目录,inotify再监听 ./ 才能rsync同步后目录结构一致,有兴趣的同学可以进行各种尝试观看其效果
    /usr/local/bin/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
     
     
     

    附录:

    Rsync的命令格式可以为以下六种:

    1
    2
    3
    4
    5
    6
    rsync [OPTION]... SRC DEST
    rsync [OPTION]... SRC [USER@]HOST:DEST
    rsync [OPTION]... [USER@]HOST:SRC DEST
    rsync [OPTION]... [USER@]HOST::SRC DEST
    rsync [OPTION]... SRC [USER@]HOST::DEST
    rsync [OPTION]... rsync://[USER@]HOST[:PORT]/SRC [DEST]

    对应于以上六种命令格式,rsync有六种不同的工作模式:

      1)拷贝本地文件。当SRC和DES路径信息都不包含有单个冒号”:”分隔符时就启动这种工作模式。如:rsync -a /data /backup

      2)使用一个远程shell程序(如rsh、ssh)来实现将本地机器的内容拷贝到远程机器。当DST路径地址包含单个冒号”:”分隔符时启动该模式。如:rsync -avz *.c foo:src

      3)使用一个远程shell程序(如rsh、ssh)来实现将远程机器的内容拷贝到本地机器。当SRC地址路径包含单个冒号”:”分隔符时启动该模式。如:rsync -avz foo:src/bar /data

      4)从远程rsync服务器中拷贝文件到本地机。当SRC路径信息包含”::”分隔符时启动该模式。如:rsync -av root@172.16.78.192::www /databack

      rsync -vzrtopg --progress --delete ronghui@192.168.80.16::module_test /test2 --password-file=/etc/rsync_client.pwd

      5)从本地机器拷贝文件到远程rsync服务器中。当DST路径信息包含”::”分隔符时启动该模式。如:rsync -av /databack root@172.16.78.192::www

      rsync -av --port=873 --progress --delete /test ronghui@192.168.80.140::backup --password-file=/etc/passwd.txt

      6)列远程机的文件列表。这类似于rsync传输,不过只要在命令中省略掉本地机信息即可。如:rsync -v rsync://172.16.78.192/www

    rsync参数的具体解释如下:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    -v, --verbose 详细模式输出
    -q, --quiet 精简输出模式
    -c, --checksum 打开校验开关,强制对文件传输进行校验
    -a, --archive 归档模式,表示以递归方式传输文件,并保持所有文件属性,等于-rlptgoD
    -r, --recursive 对子目录以递归模式处理
    -R, --relative 使用相对路径信息
    -b, --backup 创建备份,也就是对于目的已经存在有同样的文件名时,将老的文件重新命名为~filename。可以使用--suffix选项来指定不同的备份文件前缀。
    --backup-dir 将备份文件(如~filename)存放在在目录下。
    -suffix=SUFFIX 定义备份文件前缀
    -u, --update 仅仅进行更新,也就是跳过所有已经存在于DST,并且文件时间晚于要备份的文件。(不覆盖更新的文件)
    -l, --links 保留软链结
    -L, --copy-links 想对待常规文件一样处理软链结
    --copy-unsafe-links 仅仅拷贝指向SRC路径目录树以外的链结
    --safe-links 忽略指向SRC路径目录树以外的链结
    -H, --hard-links 保留硬链结
    -p, --perms 保持文件权限
    -o, --owner 保持文件属主信息
    -g, --group 保持文件属组信息
    -D, --devices 保持设备文件信息
    -t, --times 保持文件时间信息
    -S, --sparse 对稀疏文件进行特殊处理以节省DST的空间
    -n, --dry-run现实哪些文件将被传输
    -W, --whole-file 拷贝文件,不进行增量检测
    -x, --one-file-system 不要跨越文件系统边界
    -B, --block-size=SIZE 检验算法使用的块尺寸,默认是700字节
    -e, --rsh=COMMAND 指定使用rsh、ssh方式进行数据同步
    --rsync-path=PATH 指定远程服务器上的rsync命令所在路径信息
    -C, --cvs-exclude 使用和CVS一样的方法自动忽略文件,用来排除那些不希望传输的文件
    --existing 仅仅更新那些已经存在于DST的文件,而不备份那些新创建的文件
    --delete 删除那些DST中SRC没有的文件
    --delete-excluded 同样删除接收端那些被该选项指定排除的文件
    --delete-after 传输结束以后再删除
    --ignore-errors 及时出现IO错误也进行删除
    --max-delete=NUM 最多删除NUM个文件
    --partial 保留那些因故没有完全传输的文件,以是加快随后的再次传输
    --force 强制删除目录,即使不为空
    --numeric-ids 不将数字的用户和组ID匹配为用户名和组名
    --timeout=TIME IP超时时间,单位为秒
    -I, --ignore-times 不跳过那些有同样的时间和长度的文件
    --size-only 当决定是否要备份文件时,仅仅察看文件大小而不考虑文件时间
    --modify-window=NUM 决定文件是否时间相同时使用的时间戳窗口,默认为0
    -T --temp-dir=DIR 在DIR中创建临时文件
    --compare-dest=DIR 同样比较DIR中的文件来决定是否需要备份
    -P 等同于 --partial
    --progress 显示备份过程
    -z, --compress 对备份的文件在传输时进行压缩处理
    --exclude=PATTERN 指定排除不需要传输的文件模式
    --include=PATTERN 指定不排除而需要传输的文件模式
    --exclude-from=FILE 排除FILE中指定模式的文件
    --include-from=FILE 不排除FILE指定模式匹配的文件
    --version 打印版本信息
    --address 绑定到特定的地址
    --config=FILE 指定其他的配置文件,不使用默认的rsyncd.conf文件
    --port=PORT 指定其他的rsync服务端口
    --blocking-io 对远程shell使用阻塞IO
    -stats 给出某些文件的传输状态
    --progress 在传输时现实传输过程
    --log-format=formAT 指定日志文件格式
    --password-file=FILE 从FILE中得到密码
    --bwlimit=KBPS 限制I/O带宽,KBytes per second
    -h, --help 显示帮助信息
  • 相关阅读:
    SpringJMS解析2-JmsTemplate
    SpringJMS解析1-使用示例
    SpringHttpInvoker解析3-客户端实现
    算法笔记_084:蓝桥杯练习 11-1实现strcmp函数(Java)
    算法笔记_083:蓝桥杯练习 合并石子(Java)
    算法笔记_082:蓝桥杯练习 12-1三角形(Java)
    算法笔记_081:蓝桥杯练习 算法提高 矩阵乘法(Java)
    算法笔记_080:蓝桥杯练习 队列操作(Java)
    算法笔记_079:蓝桥杯练习 区间k大数查询(Java)
    算法笔记_078:蓝桥杯练习 最大最小公倍数(Java)
  • 原文地址:https://www.cnblogs.com/ronghui/p/9206428.html
Copyright © 2011-2022 走看看