既然没有遇到过,做好准备总是好的。这是自己送给自己的话,现在运维做自动话越来越多,自己就学以下,记录笔记。目前主流的有puppet、Expect、pssh等等,今天就用Expect做自动部署和日常管理维护。
一、Expect简介
expect是一种能够按照脚本内容里面设定的方式与交互式程序进行“会话”的程序。根据脚本内容,Expect可以知道程序会提示或反馈什么内容以及什么是正确的应答。它是一种可以提供“分支和嵌套结构”来引导程序流程的解释型脚本语言。
我们熟知的shell编程功能虽然很强大,但是不能实现有交互功能的多机器之前的操作,例如ssh和scp等。而expect可以帮助我们来实现。
二、安装
[root@Server ~]# yum -y install expect
三、Expect使用
这里使用Expect批量管理和部署服务器大致分为两个步骤,使用for循环读取服务器IP、密码列表并取值,远程执行命令。如下需求,在两台服务器上执行自己命令mkdir /tmp/`date +%Y%m%d`,看下面实现方法。
首先定义一个expect登录脚本:
1、login.exp,内容如下:
1 [root@Server ~]# vi /data/sh/login.exp 2 3 #!/usr/bin/expect -f 4 set ip [lindex $argv 0 ]#读取ip 5 set passwd [lindex $argv 1 ]#读取密码 6 set command [lindex $argv 2]#命令 7 set timeout 10#登录后下次执行命令间隔 8 spawn ssh root@$ip#spawn 意思是执行命令,expect内命令,shell中不存在 9 expect { 10 "yes/no" { send "yes ";exp_continue }#选择yes 11 "password:" { send "$passwd " }#读取 12 } 13 expect "*#*" { send "$command " }#执行命令 14 expect eof
2、创建批量执行脚本auto_exec.sh
[root@Server ~]# vi auto_exec.sh #!/bin/sh CMD="$*" for i in `awk '{print $1}' passwd.txt`#fou循环读取ip do j=`awk -v I="$i" '{if(I==$1)print $2}' passwd.txt`#密码 expect /data/sh/login.exp $i $j "$CMD"#执行 done
3、建立批量IP、密码文件
1 [root@Server ~]# vi passwd.txt 2 3 192.168.17.135 123456 4 192.168.17.128 123456 5 192.168.17.136 123456
四、测试脚本
直接执行:
[root@Server ~]# /bin/sh auto_exec.sh "mkdir -p /tmp/`date +%Y%m%d`"
登录三台服务器在/tmp下有日期文件
五、SCP远程拷贝
如果需要远程推送文件,重新建立文件login.scp相关参数和auto_exec.sh变量:
1、login.scp内容如下:
[root@Server ~]# vi login.scp #!/usr/bin/expect -f set ip [lindex $argv 0 ] set passwd [lindex $argv 1 ] set src_file [lindex $argv 2] set des_dir [lindex $argv 3] set timeout 1 spawn scp -r $src_file root@$ip:$des_dir expect { "yes/no" { send "yes ";exp_continue } "password:" { send "$passwd " } } expect "#*" expect eof
2、auto_exec.sh脚本内容如下:
[root@Server ~]# vi auto_exec.sh #!/bin/sh read -p "Please Enter insert Source File or DIR: " src_file echo ====================================================== sleep 1 read -p "Please Enter insert Destination DIR: " des_dir for i in `awk '{print $1}' passwd.txt` do j=`awk -v I="$i" '{if(I==$1)print $2}' passwd.txt` expect login.scp $i $j $src_file $des_dir done
密码保持不变即可。
[root@Server ~]# /bin/sh auto_exec.sh Please Enter insert Source File or DIR: login.scp#本地文件 ====================================================== Please Enter insert Destination DIR: /home spawn scp -r login.scp root@192.168.17.135:/home#发送到客户端的文件
六、一键安装expect、scp批量auto_exec.sh脚本:
#!/bin/sh if [ ! -e /usr/bin/expect ];then yum install expect -y fi #Judge passwd.txt exist if [ ! -e ./passwd.txt ];then echo -e "The passwd.txt is not exist......Please touch ./passwd.txt ,Content Example: 192.168.1.11 passwd1 192.168.1.12 passwd2" sleep 2 &&exit 0 fi #Auto Tuoch login.exp File cat >login.exp <<EOF #!/usr/bin/expect -f set ip [lindex $argv 0 ] set passwd [lindex $argv 1 ] set src_file [lindex $argv 2] set des_dir [lindex $argv 3] set timeout 1 spawn scp -r $src_file root@$ip:$des_dir expect { "yes/no" { send "yes ";exp_continue } "password:" { send "$passwd " } } expect "#*" expect eof EOF ##Auto exec shell scripts read -p "Please Enter insert Source File or DIR: " src_file echo ====================================================== sleep 1 read -p "Please Enter insert Destination DIR: " des_dir for i in `awk '{print $1}' passwd.txt` do j=`awk -v I="$i" '{if(I==$1)print $2}' passwd.txt` expect ./login.exp $i $j $src_file $des_dir
就是以上几个脚本的合成。
以下是expect用法
1. [#!/usr/bin/expect] 这一行告诉操作系统脚本里的代码使用那一个shell来执行。这里的expect其实和linux下的bash、windows下的cmd是一类东西。 注意:这一行需要在脚本的第一行。 2. [set timeout 30] 基本上认识英文的都知道这是设置超时时间的,现在你只要记住他的计时单位是:秒 。timeout -1 为永不超时 3. [spawn ssh -l username 192.168.1.1] spawn是进入expect环境后才可以执行的expect内部命令,如果没有装expect或者直接在默认的SHELL下执行是找不到spawn命令的。所以不要用 “which spawn“之类的命令去找spawn命令。好比windows里的dir就是一个内部命令,这个命令由shell自带,你无法找到一个dir.com 或 dir.exe 的可执行文件。 它主要的功能是给ssh运行进程加个壳,用来传递交互指令。 4. [expect "password:"] 这里的expect也是expect的一个内部命令,有点晕吧,expect的shell命令和内部命令是一样的,但不是一个功能,习惯就好了。这个命令的意思是判断上次输出结果里是否包含“password:”的字符串,如果有则立即返回,否则就等待一段时间后返回,这里等待时长就是前面设置的30秒 5. [send "ispass "] 这里就是执行交互动作,与手工输入密码的动作等效。 温馨提示: 命令字符串结尾别忘记加上“ ”,如果出现异常等待的状态可以核查一下。 6. [interact] 执行完成后保持交互状态,把控制权交给控制台,这个时候就可以手工操作了。如果没有这一句登录完成后会退出,而不是留在远程终端上。如果你只是登录过去执行 7.$argv 参数数组 expect脚本可以接受从bash传递过来的参数.可以使用[lindex $argv n]获得,n从0开始,分别表示第一个,第二个,第三个....参数
参考来源:http://www.linuxidc.com/Linux/2013-08/88660.htm