用python实现远程复制 (scp + expect )
2015-10-23 01:06
609 查看
scp 功能很强大,但需要人工输入 password, 当然可以通过把 公钥保存在远程主机的 ~/.ssh 目录中,而后就不用输入password,但这需要配置.
用 sshpass 可能在命令输入 password, 但 需要用 “sudo apt-get install sshpass” 安装
如果不想用上面两种方法,可以用 expect 编写脚本可以帮助我们自动交互
虽然 python 也提供 pexpect 模块,但既然 expect 很简单,为何不直接用 os.system() 去执行呢?
下面是我编写的类,实现了远程复制
[html] view plaincopy
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了
[html] view plaincopy
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 每一条语句都是顺序执行
[html] view plaincopy
因为scp 可能先返回 (yes/no)? 再 返回 password:, 也可能直接返回password:, 考虑顺序关系,上面语句的层次关系其实如下:
[html] view plaincopy
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
[html] view plaincopy
expect "100%%"
expect eof
5) expect 是部分匹配,所以不要担心自己不知道程序的完整输出
用 sshpass 可能在命令输入 password, 但 需要用 “sudo apt-get install sshpass” 安装
如果不想用上面两种方法,可以用 expect 编写脚本可以帮助我们自动交互
虽然 python 也提供 pexpect 模块,但既然 expect 很简单,为何不直接用 os.system() 去执行呢?
下面是我编写的类,实现了远程复制
[html] view plaincopy
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了
[html] view plaincopy
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 每一条语句都是顺序执行
[html] view plaincopy
因为scp 可能先返回 (yes/no)? 再 返回 password:, 也可能直接返回password:, 考虑顺序关系,上面语句的层次关系其实如下:
[html] view plaincopy
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
[html] view plaincopy
expect "100%%"
expect eof
5) expect 是部分匹配,所以不要担心自己不知道程序的完整输出
相关文章推荐
- python svn
- 2013年度Python Git工具
- 使用python获得git中分支存成list
- GitPython git python 的开发库
- Python的高级Git库 Gittle
- leetcode之Sliding Window Maximum
- Python实现自动登录/登出校园网网关
- leetcode之Kth Largest Element in an Array
- leetcode之Maximum Gap
- python学习——基础(七)
- Python知识(1)----基础入门和进阶总结。
- python例子-Django之传递json到js脚本中。
- 谈谈 Python 程序的运行原理
- python问与答常识
- fedora下python3 安装tkinter和pygame
- python用法笔记(数组(list、touple、dict)、字符串)
- Python学习笔记 -- 第六章 文件操作
- opencv-python学习一--人脸检测
- python获取cookies
- python