七十四、expect脚本同步文件、expect脚本指定host和要同步的文件、构建文件分发系统、批量远程执行命令

一、expect脚本同步文件

自动同步文件,在一台机器上同步文件到另一台机器上去。核心命令是rsync

[root@MRX sbin]# vim 4.expect     路径:/usr/local/sbin/

#!/usr/bin/expect

set passwd "1346"

spawn rsync -av [email protected]:/tmp/12.txt /tmp/

#把远程机器上的12.txt同步到这台机器上来。远程机器上需要安装rsync这个命令。从远程到本机。

expect {

"yes/no" { send "yes\r"}

"password:" { send "$passwd\r" }

#同理

}

expect eof

#停留在远程机器上几秒再退出,不然还没传输就退出来了

[root@MRX sbin]# chmod a+x 4.expect

[root@MRX sbin]# ./4.expect

spawn rsync -av [email protected]:/tmp/12.txt /tmp/

[email protected]'s password:

receiving incremental file list

12.txt


sent 42 bytes  received 91 bytes  88.67 bytes/sec

total size is 5  speedup is 0.04


二、expect脚本指定host和要同步的文件

指定host和要同步的文件

[root@MRX sbin]# vim 5.expect     路径:/usr/local/sbin/

#!/usr/bin/expect

set passwd "1346"

set host [lindex $argv 0]

#第一个参数是host

set file [lindex $argv 1]

#第二个参数是file,要同步的文件

spawn rsync -av $file root@$host:$file

#最终形成了一个脚本,从本机到远程。file一定要写绝对路径。

expect {

"yes/no" { send "yes\r"}

"password:" { send "$passwd\r" }

}

expect eof

[root@MRX sbin]# chmod a+x 5.expect

[root@MRX sbin]# ./5.expect 192.168.93.129 "/tmp/12.txt"

spawn rsync -av /tmp/12.txt [email protected]:/tmp/12.txt

[email protected]'s password:

sending incremental file list


sent 31 bytes  received 12 bytes  86.00 bytes/sec

total size is 5  speedup is 0.12

这种只适合同步一个文件。


三、构建文件分发系统

需求背景

对于大公司而言,肯定时不时会有网站或者配置文件更新,而且使用的机器肯定也是好多台,少则几台,多则几十甚至上百台。所以,自动同步文件是至关重要的。

实现思路

首先要有一台模板机器,把要分发的文件准备好,然后只要使用expect脚本批量把需要同步的文件分发到目标机器即可。

核心命令:

rsync -av --files-from=list.txt  /  root@host:/       //list.txt:文件列表;files-from,能把文件列表里的文件都同步上去,文件列表里的路径,要用绝对路径。

[root@MRX sbin]# vim rsync.expect       路径:/usr/local/sbin/

#!/usr/bin/expect

set passwd "1346"

set host [lindex $argv 0]

set file [lindex $argv 1]

#此时这里的file代表的就是list这个文件列表,而不是单一的一个文件。

spawn rsync -avR --files-from=$file / root@$host:/   #源目录是根,目标目录也是根。-R,不能保证对方机器有没有这个路径,就加-R自动创建。

expect {

"yes/no" { send "yes\r"}

"password:" { send "$passwd\r" }

}

expect eof

还要定义list.txt。文件列表里的路径,要保证对方机器上也有这个路径,文件是要同步的,路径一定要有。也可以加个-R自动创建。

示例:

[root@MRX sbin]# vim /tmp/list.txt

/tmp/12.txt

/tmp/111.txt

/root/1.txt

/root/123/2.txt

[root@MRX sbin]# vim /tmp/ip.list   #因为机器可能不止一台,所以还要定义一个ip的列表

192.168.93.129

做这个expect脚本的前提是两台机器的密码是一样的,如果不一样就只能挨个定义每台机器的密码,这样做又有风险,如果expect脚本泄露,就很危险。解决方法:做密钥认证,所以expect脚本中password那一步就可以省略,但是yes/no不能省略。

[root@MRX sbin]# vim rsync.sh

#!/bin/bash

for ip in `cat /tmp/ip.list`

do

    echo $ip

    ./rsync.expect $ip /tmp/list.txt

done

[root@MRX sbin]# chmod a+x rsync.expect

[root@MRX sbin]# sh -x rsync.sh

++ cat /tmp/ip.list

+ for ip in '`cat /tmp/ip.list`'

+ echo 192.168.93.129

192.168.93.129

+ ./rsync.expect 192.168.93.129 /tmp/list.txt

spawn rsync -avR --files-from=/tmp/list.txt / [email protected]:/

[email protected]'s password:

building file list ... done

root/

root/1.txt

root/123/

root/123/2.txt

tmp/

tmp/111.txt


sent 400 bytes  received 78 bytes  956.00 bytes/sec

total size is 109  speedup is 0.23


四、批量远程执行命令

这个文件头一定要写对,不能写/bin/bash,不然这些命令无法解析。

[root@MRX sbin]# vim exe.expect        路径:/usr/local/sbin/

#!/usr/bin/expect

set host [lindex $argv 0]

set passwd "1346"

set cm [lindex $argv 1]

#第二个参数,cm,也就是要执行的命令作为第二个参数

spawn ssh root@$host

expect {

"yes/no" { send "yes\r"}

"password:" { send "$passwd\r" }

}

expect "]*"

send "$cm\r"

expect "]*"

send "exit\r"

[root@MRX sbin]# chmod a+x exe.expect

[root@MRX sbin]# vim exe.sh

#!/bin/bash

  for ip in `cat /tmp/ip.list`

do

  echo $ip

  ./exe.expect $ip "w;free-m;ls /tmp"

done

[root@MRX sbin]# sh exe.sh