之前已经使用过expect脚本,可以远程登录机器并执行命令,其实expect脚本也可以传输文件。本案例的需求就是通过expect脚本来同步文件,从而实现代码上线。
生产环境中一个业务通常跑在多台服务器上,也就是所谓的负载均衡,那么这些机器上运行的代码必须要保持一致。如何保持一致呢?有两种方案。
1.共享式
如果机器不多,可以使用NFS实现,如果要求稳定性最好是使用专业的存储设备(NAS、SAN等)。这种架构的优点是方便维护,比如有代码更新时,只需要更新一台机器上的代码,其它机器上则都会跟着更新。缺点是,机器多了之后,共享存储会成为瓶颈,而且共享存储这里是一个单点隐患,不出问题还好,如果出问题了,那整个业务都会挂掉。
2.分布式
既然共享式有不少缺点,那么就选择分布式,即把代码存到每一台web服务器的本地磁盘上。这样做的好处是没有存储性能问题,没有资源争抢冲突,没有单点隐患。缺点是,每次代码更新需要对所有web服务器进行更新,比较繁琐。
本案例采用分布式,具体要求如下:
1)提供一个所有web服务器的IP地址列表 ip.list ;
2)假设所有web服务器上有一个普通用户user01 ,密码是SGs2ox6uj ,该用户为同步代码用户;
3)每次代码上线会提供一个要同步的文件列表 file.list 。
参考脚本如下:
#!/bin/bash
#代码更新
read -p "你是否已经更新了文件列表 ./file.list ?确认请输入y/Y,否则按其他任意键退出"
#如果按回车,也会退出
if [ -z $c ]
then
exit 1
fi
if [ $c == "y" -o $c == "Y" ]
then
echo "脚本将在2秒后执行"
for i in 1 2
do
echo -n "."
sleep 1
done
echo
else
exit 1
fi
#判断有无 ./rsync.exp 文件
[ -f ./rsync.exp ] && rm -rf ./rsync.exp
#定义rsync.exp
cat > ./rsync.exp << EOF
#!/usr/bin/expect
set passwd "SGs2ox6uj"
set host [lindex \$argv 0]
set file [lindex \$argv 1]
spawn rsync -avR --file-from=\$file / user01@\$host:/
expect()
{
"yes/no" {send "yes\r"}
"password:" {send \$passwd\r}
}
expect eof
EOF
chmod a+x ./rsync.exp
#定义检测文件是否存在
file_exist()
{
if [ ! -f $1 ]
then
echo "文件$1不存在,请检查"
exit 1
fi
}
file_exist ./ip.list
file_exist ./file.list
for ip in `cat ./ip.list`
do
./rsync.exp $ip ./file.list
done
rm -f ./rsync.exp
说明:要想使用该脚本,需要先在所有的web服务器上创建user01用户并设置好密码,并且保证user01用户对web服务器上代码所在目录有写权限。