使用expect运行动态脚本
在平时的工作中,如果接手的环境多了之后,每天去尝试连接服务器,都是例行的步骤,时间长了之后就会感觉这些工作都是繁琐重复的工作,其实我们可以尝试让工作更简化,更高效一些。
比如我们设定下面的场景,
我们存在服务器A,这个服务器可以连接到网络环境中的其它机器,我们假定这个机器就是中控机。
通过中控机连接到各个服务器环境,有下面几个步骤,
1)连接到某一台服务器B
2)查看系统的版本信息
3)查看系统的内核信息
4)切换到Oracle用户下
5)查看服务器所使用的Oracle版本
因为切换用户的原因,所以单纯使用ssh来执行命令,第4,5步就没法完成,这个时候我们可以有几种思路来做,一种是通过连接工具录制脚本,比如secureCRT录制一段脚本,操作一遍之后,以后每次执行就需要重放脚本即可。
如果受到工具的限制,或者觉得录制脚本也不够动态,比如有100台服务器,我们就需要录制100个脚本,查找,运行相比而言工作量还是不少,这个时候我们可以使用Linux中的expect命令来DIY
expect是RHEL6版本中自带的一个实用工具,可以校验上个命令执行后的结果集中的关键字,来灵活的运行各种批量处理任务。
expect的交互方式有两种,一种需要以expect eof结尾,另外一种是interact结尾。
这两种方式可以打个比方,比如我们把家比作服务器A,朋友的家比作服务器B,有一天我去给朋友送一个东西,可以把这个比作在服务器B需要的的操作
如果我给朋友送完东西之后,想在朋友家里吃个晚饭,然后一起看一晚上球赛,第二天再回家,这种情况就有点类似interact的方式。
如果我给朋友送完东西之后就回家,朋友邀请吃晚饭也婉拒,这种方式就类似expect eof.
我们来看几个简单的例子。
我们假设下面的脚本为test.sh
需要在脚本前声明 #!/usr/bin/expect,然后使用spawn来启动ssh连接,然后切换到oracle用户,查看系统的版本信息,内核信息,然后查看oracle的版本信息,最后退出,返回到最开始的session
#!/usr/bin/expect
spawn ssh 10.127.133.45
expect "#"
send "su - oracle\r"
send "cat /etc/issue \r"
send "uname -a \r"
send "sqlplus -v \r"
send "exit \r"
send "exit \r"
expect eof
运行命令 ./test.sh, 命令运行的输出结果如下:
spawn ssh 10.xxxx.xxxx.45
Last login: Wed Aug 5 22:06:45 2015 from 10.xxxx.133.xxxx
[root@xxxx_xxxx_45 ~]# su - oracle
[oracle@BX_xxxx_45 ~]$ cat /etc/issue
Red Hat Enterprise Linux Server release 6.3 (Santiago)
Kernel \r on an \m
[oracle@xxxx_xxxx_45 ~]$ uname -a
Linux xxxx_xxxx 2.6.32-279.el6.x86_64 #1 SMP Wed Jun 13 18:24:36 EDT 2012 x86_64 x86_64 x86_64 GNU/Linux
[oracle@xxxx_xxxx_45 ~]$ sqlplus -v
SQL*Plus: Release 11.2.0.4.0 Production
[oracle@xxxx_xxxx_45 ~]$ exit
logout
[root@xxxx_xxxx_45 ~]# exit
logout
Connection to 10.xxxx.xxxx.45 closed.
可以看到其实就是连接到目标环境中,执行完命令之后返回原来的session.
再来看一个interact的例子,比如我们存在大量的用户,需要通过快捷方式登录到指定的机器上,查看服务器名,然后连入oracle用户,查看数据库版本。
#!/usr/bin/expect
spawn ssh 10.xxxx.0.63
expect "#"
send "su - oracle\r"
send "cat /etc/issue \r"
send "uname -a \r"
send "cat /etc/hosts|grep `hostname` \r"
send "sqlplus -v \r"
interact
命令的运行结果如下:
spawn ssh 10.127.0.63
Last login: Wed Aug 5 22:43:58 2015 from 10.127.133.86
[root@xxxx ~]# su - oracle
[oracle@xxxx ~]$ cat /etc/issue
Red Hat Enterprise Linux Server release 5.3 (Tikanga)
Kernel \r on an \m
[oracle@xxxx ~]$ uname -a
Linux xxxx.com 2.6.18-194.el5 #1 SMP Tue Mar 16 21:52:39 EDT 2010 x86_64 x86_64 x86_64 GNU/Linux
[oracle@xxxx ~]$ cat /etc/hosts|grep `hostname` ;
10.127.xxxx.63 xxxx.com #Primary
10.127.xxxx.82 xxxx.com #Standby
[oracle@xxxx ~]$ sqlplus -v ;
ps -ef|grep smon
SQL*Plus: Release 11.2.0.3.0 Production
这个时候我们已经连入了那个目标环境,可以继续在这个基础上进行其它的操作。
在这个基础上稍微扩展一下,我们可以把expect和shell变量联系起来
比如我们有100台服务器,我们只需要输入IP地址即可,然后后面会自动去切换用户,检查内核参数,系统参数等等。
只需要修改下面的两个部分即可,整个脚本一下子就动态起来了。
#!/usr/bin/expect
set ip_addr [lindex $argv 0]
spawn ssh $ip_addr
expect "#"
send "su - oracle\r"
send "cat /etc/issue \r"
send "uname -a \r"
send "cat /etc/hosts|grep `hostname` ; \r"
send "sqlplus -v ;\r"
send "ps -ef|grep smon \r"
interact