自动交互脚本之expect使用记录

之前一直没怎么用这个命令,意外用了一下,还不错,那这个是干嘛的呢

我们或多或少会远程登录其他服务器,需要执行某项任务,通常需要手动接入,输入密码啊,等等

那我们如何有效的自动执行呢,expect可以解决咯

参考链接:http://blog.51cto.com/balich/1720804

                  https://www.cnblogs.com/lixigang/articles/4849527.html

一、expect的介绍

expect是一个简单的工具语言,如要工作就是进行自动化的人机交互。它的作者对Expect的定义:是一个实现自动交互功能的软件套件(a software suite for automating interactive tools),使用expect,它能帮助我们在合适的情景下进行合适的交互。

expect的核心是spawn、 expect、 send、 set

1 spawn         ## 调用要执行的命令
2 expect        ##等待命令提示信息的出现,也就是捕捉用户输入的提示
3 send         ##发送需要交互的值,替代了用户手动输入内容
4 set          ##设置变量值
5 interact       ##执行完成后保持交互状态,把控制权交给控制台,这个时候就可以手工操作了。如果没有这一句登录完成后会退出,而不是留在远程终端上。
6 expect eof     # 这个一定要加,与spawn对应表示捕获终端输出信息终止,类似于if....endif
7 expect脚本必须以interact或expect eof结束,执行自动化任务通常expect eof表示脚本执行完成。

二、expect的安装与使用

系统:centos 7     # 可以直接用yum安装

1.安装

yum install  -y expect

2.案例说明

2.1)自动登录系统并执行任务

vim login.exp
#
!/usr/bin/expect set host "192.168.68.25" set passwd "123456" ##设置变量 spawn ssh root@$host ##调用执行命令 expect { ##等待命令出现 "yes/no" {send "yes\r"; exp_continue} ##发送交互值,yes "password:" {send "$passwd\r"} ##输入密码 } expect "]*" ##获取当前的终端的命令行提示符 send "uname -a\r" ##执行uname -a 命令,\r是换行 expect "]*" send "exit\r" ##执行完命令后,退出当前终端。 expect eof

脚本执行过程

[root@localhost shell]#
[root@localhost shell]# ./login.exp
spawn ssh [email protected]
[email protected]'s password:
Last login: Mon May 21 23:00:39 2018 from 192.168.68.26
[root@localhost ~]# uname -a
Linux localhost.localdomain 3.10.0-693.21.1.el7.x86_64 #1 SMP Wed Mar 7 19:03:37 UTC 2018 x86_64 x86_64 x86_64 GNU/Linux
[root@localhost ~]# exit
logout
Connection to 192.168.68.25 closed.

2.2 使用传递参数

#!/usr/bin/expect               
set user [lindex $argv 0]          #参数1,用户名
set passwd [lindex $argv 1]     #参数2,用户密码
set host [lindex $argv 2]          #参数3,远程主机ip或者主机名
set cmd [lindex $argv 3]          #参数4,执行命令
spawn ssh $user@$host
expect {
        "yes/no" {send "yes\r"; exp_continue}
        "password:" {send "$passwd\r"}
        }
expect "]*"
send "$cmd\r"
expect "]*"
send "exit\r"
expect eof

脚本的执行是在脚本后面依次写对应的参数,用户名、用户密码、主机ip、执行命令,参数之间使用空格隔开

执行结果如下:

[root@localhost shell]# 

[root@localhost shell]# ./login.exp   root  123456  192.168.68.25  'uname -a'
spawn ssh [email protected]
[email protected]'s password:
Last login: Mon May 21 23:33:46 2018 from 192.168.68.26
[root@localhost ~]# uname -a
Linux localhost.localdomain 3.10.0-693.21.1.el7.x86_64 #1 SMP Wed Mar 7 19:03:37 UTC 2018 x86_64 x86_64 x86_64 GNU/Linux
[root@localhost ~]# exit
logout
Connection to 192.168.68.25 closed.

2.3 打印服务器的内网和外网ip   返回到本地文件

脚本如下:

存放ip的文件:

vim ip.list

192.168.68.25

实现脚本:

 
 
vim   true.sh
#!/bin/bash


set -e List=`cat ip.txt` for IpList in ${List} do export IpList expect -f 520.sh |grep -E -o "^[0-9]+\.[0-9]+\.[0-9]+\.[0-9]+"|awk '{print $1}' |tr "\n" " " >> /tmp/520.txt echo >> /tmp/520.txt #打印换行符 done

解释:
expect -f 表示执行命令文件
grep -E -o "^[0-9]+\.[0-9]+\.[0-9]+\.[0-9]+" 表示获取以xx.xx.xx.xx开头的,类似于获取以ip开头的
tr "\n" " " 表示把换行符替换为空格,多行合并为一行


vim 520.exp
#!/usr/expect/bin/expect -f spawn -noecho ssh root@$env(IpList) #获取true.sh里面的ip set timeout 60 expect "assword*" { send "123456\r" } expect "root*" { send "ip a |grep 'eno16777736'|grep inet|cut -d '/' -f 1|cut -d ' ' -f 6 \r" #获取内网ip send "ip a |grep 'eno16777736'|grep inet|cut -d '/' -f 1|cut -d ' ' -f 6 \r" #获取外网ip send "logout \r" send "logout \r" } expect eof

最终的结果,可以在/tmp/520.txt 里面去查看

更多的综合实例,需要自己实践啦,相信实践是检验理论的唯一标准

猜你喜欢

转载自www.cnblogs.com/tianfen/p/9067792.html
今日推荐