用python实现远程复制 (scp + expect )

scp 功能很强大,但需要人工输入 password, 当然可以通过把 公钥保存在远程主机的 ~/.ssh 目录中,而后就不用输入password,但这需要配置.

用 sshpass 可能在命令输入 password, 但 需要用 “sudo apt-get install sshpass” 安装

如果不想用上面两种方法,可以用 expect 编写脚本可以帮助我们自动交互

虽然 python 也提供 pexpect  模块,但既然 expect 很简单,为何不直接用 os.system() 去执行呢?

下面是我编写的类,实现了远程复制 

  1. class RemoteShell:  
  2.   
  3.     def __init__(self, host, user, pwd):  
  4.         self.host = host  
  5.         self.user  = user  
  6.         self.pwd  = pwd  
  7.   
  8.   
  9.     def put(self, local_path, remote_path):  
  10. scp_put = '''  
  11. spawn scp %s %s@%s:%s  
  12. expect "(yes/no)?" {  
  13. send "yes\r"  
  14. expect "password:"  
  15. send "%s\r"  
  16. } "password:" {send "%s\r"}  
  17. expect eof  
  18. exit'''  
  19.         os.system("echo '%s' > scp_put.cmd" % (scp_put % (os.path.expanduser(local_path), self.user, self.host, remote_path, self.pwd, self.pwd)))  
  20.         os.system('expect scp_put.cmd')  
  21.         os.system('rm scp_put.cmd')  
class RemoteShell:

    def __init__(self, host, user, pwd):
        self.host = host
        self.user  = user
        self.pwd  = pwd


    def put(self, local_path, remote_path):
scp_put = '''
spawn scp %s %s@%s:%s
expect "(yes/no)?" {
send "yes\r"
expect "password:"
send "%s\r"
} "password:" {send "%s\r"}
expect eof
exit'''
        os.system("echo '%s' > scp_put.cmd" % (scp_put % (os.path.expanduser(local_path), self.user, self.host, remote_path, self.pwd, self.pwd)))
        os.system('expect scp_put.cmd')
        os.system('rm scp_put.cmd')

但发现每次文件都没有复制完,我想看expect 究竟做了什么事情,还好 expect 提供 -d 参数给出更多的信息。

最后发现是被复制文件太大,expect 超时退出了

在脚本前加入 “set timeout -1" 就OK了


  1. scp_put = '''  
  2. set timeout -1  
  3. spawn scp %s %s@%s:%s  
  4. expect "(yes/no)?" {  
  5. send "yes\r"  
  6. expect "password:"  
  7. send "%s\r"  
  8. } "password:" {send "%s\r"}  
  9. expect eof  
  10. exit'''  
scp_put = '''
set timeout -1
spawn scp %s %s@%s:%s
expect "(yes/no)?" {
send "yes\r"
expect "password:"
send "%s\r"
} "password:" {send "%s\r"}
expect eof
exit'''

总结

1) expect 每一条语句都是顺序执行

  1.   
</pre> 因为scp 可能先返回 (yes/no)? 再 返回 password:, 也可能直接返回password:, 考虑顺序关系,上面语句的层次关系其实如下: <div class="dp-highlighter bg_html"><div class="bar"><div class="tools"><strong>[html]</strong> <a target=_blank title="view plain" class="ViewSource" href="http://blog.csdn.net/span76/article/details/11575231#">view plain</a><a target=_blank title="copy" class="CopyToClipboard" href="http://blog.csdn.net/span76/article/details/11575231#">copy</a><a target=_blank title="print" class="PrintSource" href="http://blog.csdn.net/span76/article/details/11575231#">print</a><a target=_blank title="?" class="About" href="http://blog.csdn.net/span76/article/details/11575231#">?</a></div></div><ol class="dp-xml"><li class="alt"><span><span>expect "(yes/no)?" {   send "yes\r"  </span></span></li><li><span>                       expect "password:"  </span></li><li class="alt"><span>                       send "%s\r"  </span></li><li><span>                    }   </span></li><li class="alt"><span>       "password:" {send "%s\r"}  </span></li></ol></div><pre class="html" style="display: none;" name="code">expect "(yes/no)?" {   send "yes\r"
                       expect "password:"
                       send "%s\r"
                    } 
       "password:" {send "%s\r"}

2) 每当 spawn 的程序有输出的时候的,expect都会去匹配, 如果匹配不上,就等下次有输出,再次执行当前的 expect, 直到超时 (我用 expect -d 去追踪,得到的结论);当然可以设置没有超时 "set timeout -1"


3) 如果  expect 退出, 被它 spawn 的程序会被 kill 掉


4) spawn 结束的时候,它向标准输出的的 eof 会被 expect 检测到,正好作为 expect 脚本退出的时机。

对于 scp 可以先检测 100%,因为 scp 会输出复制进度,再检测 eof

  1. expect "100%%"  
  2. expect eof  
expect "100%%"
expect eof


5) expect 是部分匹配,所以不要担心自己不知道程序的完整输出

版权声明:本文为博主原创文章,未经博主允许不得转载。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值