我正在尝试编写一个脚本将提供凭据到一个vpn的一个警告,其中VPN要求我提供动态口令。
这是我的脚本:
#! /usr/bin/expect
set vpn [ lindex $argv 0]
set group [ lindex $argv 1]
set user [ lindex $argv 2]
set secret [ lindex $argv 3]
log_user 2
spawn nmcli con up $vpn --ask
expect {
"GROUP:" {
send "$group\r"
exp_continue
}
"Username:" {
send "$user\r"
exp_continue
}
Password: {
send_user "\nProvide RSA OTP:"
expect_user -re ":(.*)\r"
set otp $expect_out(0,string) <--------- Error!!!
set pass "$secret$otp"
send "$pass\r"
exp_continue
}
"Connection successfully activated" {
send_user "Connected to VPN\r"
}
"Login failed" {
send_user "Login failed\r"
exp_continue
}
"already active" {
send_user "Already connected to VPN\r"
}
}
exit
我添加了一个箭头,指向我认为是问题根源的地方,因为当我运行此脚本时,它看起来像是expect_out包含Password:
。根据我在手册页中读到的内容,我认为每次进行新的匹配时缓冲区都会被清除,因此它包含最近的expect子句(在本例中为expect_user -re ":(.*)\r"
)中匹配的字符串,但似乎我错了。我没有看到任何说明说需要像interact
那样使用不同的函数访问expect_user内容,所以我不确定我的密码去哪里了?
谢谢
发布于 2020-04-02 17:31:39
使用send_user
时,您将看不到所写内容的回显,这与发送到派生命令时不同,因此除非用户显式键入:
,否则您永远不会匹配它。此外,您将读取换行符,而不是回车符。所以请使用
expect_user -re "(.*)\n"
来获取用户输入,$expect_out(1,string)
(1而不是0)将包含输入。
您观察到的是expect_user
中的超时,这就是为什么变量不会更新,仍然包含旧的匹配。
为了防止不可见的超时,您可以设置一个“全局”超时检查,从
proc abort { } { send_user "Timeout!" ; exit 2 }
expect_before timeout abort
expect_before
是在每个expect
之前完成的,而且似乎也是在每个expect_user
之前完成的。
https://stackoverflow.com/questions/60996173
复制相似问题