我有一个嵌入在 bash 脚本中的期望脚本:
#! /bin/bash
# this function runs a command like 'ssh' and provides the password
function with_password {
expect << END
spawn $2
expect *assword:*
send -- $1
interact
wait
END
}
# run "long_running_command" on the remote server
with_password my_password "ssh my_user@some-server long_running_command"
# rsync some data to the remote server
with_password my_password "rsync /some/dir my_user@some-server:/remote/dir"
# run some other random command
with_password my_password "ssh my_user@some-server some_other_command"
问题是,有时脚本会在等待生成的命令时挂起。如果我将命令wait
从 expect 脚本中移除,则命令将继续在远程服务器上运行,但 bash 脚本将继续运行,我无法知道它何时完成。
为什么我的预期脚本似乎是随机挂起的?
答案1
interact
问题出在这里文档中的命令。expect
我需要等待 EOF:
#! /bin/bash
# this function runs a command like 'ssh' and provides the password
function with_password {
expect << END
set timeout 900
spawn $2
expect *assword:*
send -- $1
expect EOF
END
}
# run "long_running_command" on the remote server
with_password my_password "ssh my_user@some-server long_running_command"
# rsync some data to the remote server
with_password my_password "rsync /some/dir my_user@some-server:/remote/dir"
# run some other random command
with_password my_password "ssh my_user@some-server some_other_command"
interact
当 ssh/rsync 命令在正确的时间关闭输入时,该命令有时似乎有效(可能?),但它并不可靠。
答案2
那么使用 ssh 密钥进行无密码登录怎么样?
您可以使用创建密钥,ssh-keygen
然后将您的公钥放在远程端:~/.ssh/authorized_keys
。
您甚至可以使用以下方式创建密钥密码(如果您想添加某种安全性)您可以在 shell 中通过ssh-agent bash
输入一次来加载它,它会将您解锁的密钥保存在内存中,以便以后使用密码无提示登录。
答案3
您缺少此处的最后一个文档分隔符。
expect << END
spawn $2
expect *assword:*
send -- $1
interact
wait
END