Shell 编程 免交互 expect
程序员文章站
2023-11-09 17:57:04
本篇主要写一些 shell 脚本免交互 expect 的使用。 ......
本篇主要写一些shell
脚本免交互expect
的使用。
概述
expect
是建立在tcl
基础上的一个工具,expect
是用来进行自动化控制和测试的工具。主要解决shell
脚本中不可交互的问题。
安装
- 使用此工具前需先安装
yum install -y expect
基本命令
send
向进程发送字符串,用于模拟用户的输入
该命令不能自动回车换行,一般要加
\r
(回车)
expect
expect
的一个内部命令,判断上次输出结果里是否包含指定的字符串,如果有则立即返回,否则就等待超时时间后返回。只能捕捉由
spawn
启动的进程的输出
spawn
- 启动进程,并跟踪后续交互信息
interact
- 执行完成后保持交互状态,把控制权交给控制台
timeout
指定超时时间,过期则继续执行后续指令
单位是:秒
timeout -1
永不超时默认情况下,
timeout
是10
秒
exp_continue
- 允许
expect
继续向下执行指令
send_user
- 回显命令,相当于
echo
$argv 参数数组
-
expect
脚本可以接受从bash
传递的参数.可以使用[lindex $argv n]
获得,n
从0
开始,分别表示第一个,第二个,第三个...参数
expect 脚本
expect
脚本必须以interact
或expect eof
结束,执行自动化任务通常expect eof
就够了expect eof
是在等待结束标志。由spawn
启动的命令在结束时会产生一个eof
标记,expect eof
即在等待这个标记
expect 语法
- 单分支
expect "password:" {send "mypassword\r";}
- 多分支
expect "aaa" {send"aaa\r"} expect "aaa" {send"aaa\r"} expect "aaa" {send"aaa\r"}
send
命令不具备回车换行功能,一般要加\r
或\n
expect { "aaa" {send "aaa\r"} "bbb" {send "bbb\r"} "ccc" {send "ccc\r"} }
只要配置
aaa
或bbb
或ccc
中的任何一个,执行相应的send
语句后退出该expect
语句
expect { "aaa" {send "aaa";exp_continue} "bbb" {send "bbb";exp_continue} "ccc" {send "ccc"} }
exp_continue
表示继续后面的匹配,如果匹配了aaa
,执行完send
语句后还要继续向下匹配bbb
执行方式
- 基本语法结构
spawn 命令 expect "提示信息" send "代替人工输入的字符串\r"
- 直接执行
#!/usr/bin/expect # 超时时间 set timeout 20 log_file test.log log_user 1 # 参数传入 set hostname [lindex $argv 0] set password [lindex $argv 1] # 追踪命令 spawn ssh root@$hostname # 捕捉信息并匹配,免交互执行 expect { "(yes/no)" {send "yes\r";exp_continue} "*password" {send "$password\r"} } # 控制权交给控制台执行 interact
[root@host01 ~]# yum install expect -y [root@host01 ~]# vim ssh.sh [root@host01 ~]# chmod +x ssh.sh [root@host01 ~]# ./ssh.sh 192.168.28.129 000000 spawn ssh root@192.168.28.129 the authenticity of host '192.168.28.129 (192.168.28.129)' can't be established. ecdsa key fingerprint is sha256:qmztjt0pibuskf9p3gfyf3ueogzbws08si7j0ebe/ci. ecdsa key fingerprint is md5:ef:e6:06:22:8a:0f:24:00:f8:af:a5:59:5b:a2:b8:b1. are you sure you want to continue connecting (yes/no)? yes warning: permanently added '192.168.28.129' (ecdsa) to the list of known hosts. root@192.168.28.129's password: last login: thu oct 17 09:35:35 2019 [root@host02 ~]#
- 嵌入执行
#!/bin/bash hostname=$1 password=$2 /usr/bin/expect <<-eof spawn ssh root@$hostname expect { "(yes/no)" {send "yes\r";exp_continue} "*password" {send "$password\r"} } expect "*]#" send "exit\r" expect eof eof
-eof
只能容错制表符tab
[root@host01 ~]# vim ssh.sh [root@host01 ~]# ./ssh.sh 192.168.28.129 000000 spawn ssh root@192.168.28.129 root@192.168.28.129's password: last login: thu oct 17 09:38:23 2019 from 192.168.28.128 [root@host02 ~]# exit logout connection to 192.168.28.129 closed. [root@host01 ~]#
案例1 useradd
#!/bin/bash username=$1 password=$2 useradd $username /usr/bin/expect << eof spawn passwd $username expect "new password:" send "$password\r" expect "retype new password:" send "$password\r" expect eof eof
[root@host01 ~]# vim useradd.sh [root@host01 ~]# chmod +x useradd.sh [root@host01 ~]# ./useradd.sh zhangsan 000000 spawn passwd zhangsan changing password for user zhangsan. new password: bad password: the password is a palindrome retype new password: passwd: all authentication tokens updated successfully.
案例2 ssh
#!/usr/bin/expect # 超时时间 set timeout 20 log_file test.log log_user 1 # 参数传入 set hostname [lindex $argv 0] set password [lindex $argv 1] # 追踪命令 spawn ssh root@$hostname # 捕捉信息并匹配,免交互执行 expect { "connection refused" exit "name or service not known" exit "(yes/no)" {send "yes\r";exp_continue} "*password" {send "$password\r"} } # 控制权交给控制台执行 interact exit
[root@host02 ~]# systemctl stop sshd
[root@host01 ~]# ./ssh.sh 192.168.28.129 000000 spawn ssh root@192.168.28.129 ssh: connect to host 192.168.28.129 port 22: connection refused [root@host01 ~]#
[root@host02 ~]# systemctl start sshd
[root@host01 ~]# ./ssh.sh host02 000000 spawn ssh root@host02 ssh: could not resolve hostname host02: name or service not known [root@host01 ~]#
[root@host01 ~]# ./ssh.sh 192.168.28.129 000000 spawn ssh root@192.168.28.129 root@192.168.28.129's password: last login: thu oct 17 09:49:38 2019 [root@host02 ~]#
下一篇: 数据库索引详解