天天看点

Linux expect 用法安装用法例子

expect是建立在tcl基础上的一个工具,它用来让一些需要交互的任务自动化地完成。

因为expect是基于tcl的,所以需要你的系统中安装有tcl

如何检查?

[root@dev ~]# whereis tcl

tcl: /usr/lib/tcl8.4 /usr/share/tcl8.4

如果看不到结果,请先安装tcl

> 安装tcl

解压tcl安装包后

cd tcl8.4.11/unix/

./configure --prefix=/usr/local/tcl/ --enable-shared

make && make install

> 安装expect

解压expect安装包后

cd expect-5.43

./configure --prefix=/usr/local/expect/ --with-tcl=/usr/local/tcl/lib/ --with-tclinclude=/opt/tcl8.4.11/generic/ --enable-shared

注意:指定的/opt/tcl8.4.11/generic/ 为我们上面解压的tcl目录

> 创建连接符号

ln -s /usr/local/expect/bin/expect /usr/bin/expect

> 查看连接符号

ls -l /usr/bin/expect

lrwxrwxrwx. 1 root root 28 9月   8 11:21 /usr/bin/expect -> /usr/local/expect/bin/expect

这个符号链接将在编写expect脚本文件时用到,例如在expect文件头部会指定用于执行该脚本的shell 

#!/usr/bin/expect 

> 测试

[root@localhost opt]# expect

expect1.1> exit

[root@localhost opt]# 

这样就可以开始运行expect脚本了。

1. [#!/usr/bin/expect] 

这一行告诉操作系统脚本里的代码使用那一个shell来执行。这里的expect其实和linux下的bash、windows下的cmd是一类东西。 

注意:这一行需要在脚本的第一行。 

2. [set timeout 30] 

基本上认识英文的都知道这是设置超时时间的,现在你只要记住他的计时单位是:秒。timeout -1 为永不超时,默认情况下,timeout是10秒;

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运行进程加个壳,用来传递交互指令。 

spawn后面加上需要执行的shell命令,比如说spawn sudo touch testfile

4. [expect "password:"] 

这里的expect也是expect的一个内部命令,有点晕吧,expect的shell命令和内部命令是一样的,但不是一个功能,习惯就好了。这个命令的意思是判断上次输出结果里是否包含“password:”的字符串,如果有则立即返回,否则就等待一段时间后返回,这里等待时长就是前面设置的30秒 

5. [send "ispass\r"] 

这里就是执行交互动作,与手工输入密码的动作等效。 

温馨提示: 命令字符串结尾别忘记加上“\r”,如果出现异常等待的状态可以核查一下。 

6. [interact] 

执行完成后保持交互状态,把控制权交给控制台,这个时候就可以手工操作了。如果没有这一句登录完成后会退出,而不是留在远程终端上。如果你只是登录过去执行 

7. $argv 参数数组

expect脚本可以接受从bash传递过来的参数.可以使用[lindex $argv n]获得,n从0开始,分别表示第一个,第二个,第三个....参数

其中,$argc为命令行参数的个数,$argv0为脚本名字本身,$argv为命令行参数。[lrange $argv 0 0]表示第1个参数,[lrange $argv 0 4]为第一个到第五个参数。与c语言不一样的地方在于,$argv不包含脚本名字本身。

8. send和send_user

send会将expect脚本中需要的信息发送给spawn启动的那个进程,而send_user只是回显用户发出的信息,类似于shell中的echo而已。

9. 如果你在第一行(#!那行)使用-d (debug参数),可以在运行的时候输出一些很有用的信息

比如你会看见

argv[0] = /usr/bin/expect argv[1] = -d argv[2] = ./launch.exp argv[3] = 1 argv[4] = 2 argv[5] = 3

使用这些也可以完成参数传递

10. exp_continue的用法

expect {

        -re "permission denied, please try again." {

                send_user "error:permission denied.\n"

                exit

        }

        -re "are you sure you want to continue connecting (yes/no)?" {

                send "yes\r";exp_continue

        -re "assword:" {

                send "$loginpass\r";exp_continue

        -re "connection refused" {

        timeout {

        eof {

}

使用exp_continue后,会重新从当前expect块的开始重新执行,可以简单理解问while循环的continue

下面的脚本例子,实现了:登录一个linux服务器,执行 df -h 等命令

vi expectexample.sh 后输入下面的脚本内容保存后,使用chmod +x expectexample.sh 命令为脚本赋予可执行权限。

#!/usr/bin/expect -f

#-------------------------------------------- set the variable,you can modify the value

set ipaddr [lrange $argv 0 0]

set port [lrange $argv 1 1]

set loginuser [lrange $argv 2 2]

set loginpass [lrange $argv 3 3]

set cmd_prompt "]#|~]?"

# 超时时间,单位(秒)

set timeout 3600

#-------------------------------------------- login by ssh

spawn ssh -p $port $loginuser@$ipaddr

                -re "connection refused" {

                send_user "error:connection refused.\n"

                send "yes\r"

                exp_continue

                send "$loginpass\r"

        -re $cmd_prompt {

                send "\r"

#-------------------------------------------- now,we do some commands

                send "df -h\r"

                send "free -m\r"

                send "uptime\r"

send "cd /opt\r"

send "mkdir shanhy123\r"

send "cd shanhy123\r"

send "echo 123 > 123.txt\r"

send "/opt/xs.sh\r"

send "exit\r"

expect eof

#interact

-------------------------------------------------

其中 xs.sh 为我写的一个测试的 shell 脚本,里面故意做了一个sleep 模拟耗时操作,代码如下:

#!/bin/bash

sleep 10

echo abc > /opt/abc.txt

注意:expect脚本必须以interact或expect eof结束,执行自动化任务通常expect eof就够了。