七十四、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