expect在Linux下的应用

先来看一段代码:

  1. #!/usr/bin/expect
  2. set timeout 30
  3. spawn ssh -l root 192.168.1.188
  4. expect "password:"
  5. send "abc123456\r"
  6. interact

对这6行进行下解释:

第1行[#!/usr/bin/expect]
这一行告诉操作系统脚本里的代码使用那一个shell来执行。这里的expect其实和linux下的bash、windows下的cmd是一类东西。 这一行需要在脚本的第一行。
第2行[set timeout 30]
基本上认识英文的都知道这是设置超时时间的,现在你只要记住他的计时单位是:秒
第3行 [spawn ssh -l username 192.168.1.188]
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 "abc123456\r"]
这里就是执行交互动作,与手工输入密码的动作等效。
温馨提示: 命令字符串结尾别忘记加上“\r”,如果出现异常等待的状态可以核查一下。
第6行[interact]
执行完成后保持交互状态,把控制权交给控制台,这个时候就可以手工操作了。如果没有这一句登录完成后会退出,而不是留在远程终端上。

在Linux下,执行ssh登陆或者是scp复制文件的时候可以使用到该命令,实现自动输入密码。

案例1:远程执行命令

  1. #!/usr/bin/expect -f
  2. set timeout 30
  3. spawn ssh -l root 192.168.1.188
  4. expect {
  5. "yes/no" { send "yes\r";exp_continue }
  6. "password:" { send "abc123456\r" }
  7. }
  8. expect -re "\](\$|#) "
  9. send "bash /root/test.sh \r"
  10. expect -re "\](\$|#) "
  11. send "exit\r"

该代码的功能是执行远程服务器上的/root/test.sh脚本,参数“-re”是匹配正则表达式。可以写多行send,执行多个命令。该脚本的另一种写法

  1. #!/bin/bash
  2. expect -c "
  3. spawn ssh [email protected] \"ifconfig;whoami;pwd;echo hello;\"
  4. expect {
  5. \"*assword\" {set timeout 30; send \"abc123456\r\";}
  6. \"yes/no\" {send \"yes\r\"; exp_continue;}
  7. }
  8. expect eof
  9. "

参数“-c”是指在命令行执行expect。把要执行的命令用分号隔开了。并且执行过程中没有shell提示符的,有兴趣的你可以测试下。

案例2:批量处理

在需要管理大量的Linux服务器时,可以这样做:

1.host.list

  1. 192.168.1.1 22 root abc1
  2. 192.168.1.2 22 root abc2
  3. 192.168.1.3 22 root abc3
  4. 192.168.1.4 22 root abc4
  5. 192.168.1.5 22 root abc5
  6. 192.168.1.6 22 root abc6
  7. 192.168.1.7 22 root abc7
  8. 192.168.1.8 22 root abc8

我们把服务器的IP地址、端口号、用户名、密码一起放在host.list文件里,按行排列。

下面来看看脚本文件main.exp

  1. #!/usr/bin/expect -f
  2. set ipaddress [lindex $argv 0]
  3. set port [lindex $argv 1]
  4. set username [lindex $argv 2]
  5. set passwd [lindex $argv 3]
  6. set timeout 30
  7. spawn ssh $ipaddress -p$port -l$username
  8. expect {
  9. "yes/no" { send "yes\r";exp_continue }
  10. "password:" { send "$passwd\r" }
  11. }
  12. expect -re "\](\$|#) "
  13. send "bash /root/test.sh \r"
  14. expect -re "\](\$|#) "
  15. send "exit\r"

来做一个while循环:whi.sh

  1. #!/bin/bash
  2. host="host.list"
  3. while read line
  4. do
  5. expect main.exp $line
  6. done < $host

把以上脚本都赋予执行权限,就可以了。

案例3:scp文件传输

  1. expect -c "
  2. spawn scp [email protected]:/root/file1.tgz /root
  3. expect {
  4. \"*assword\" {set timeout 300; send \"abc123456\r\";}
  5. \"yes/no\" {send \"yes\r\"; exp_continue;}
  6. }
  7. expect eof
  8. "

同样的原理,也可以做循环进行批量管理,还可以更简化脚本,也不多说明了,有兴趣的同学可以多尝试下。使用expect进行自动输入密码登陆,基本多用于批量管理。也还算方便。关于批量管理有多种方法,可以在主机之间建立ssh信任机制,也可以免密码登陆管理等。还可以使用sshpass(外部命令)这个命令进行带密码在脚本里执行自动输入密码。

相关推荐