在 Python 中 scp 文件的最 Pythonic 方式是什么?我知道的唯一路线是
os.system('scp "%s" "%s:%s"' % (localfile, remotehost, remotefile) )
这是一种 hack,在类 Linux 系统之外无法使用,并且需要 Pexpect 模块的帮助以避免密码提示,除非您已经为远程主机设置了无密码 SSH。
我知道 Twisted 的 conch
,但我宁愿避免自己通过低级 ssh 模块实现 scp。
我知道 paramiko
,一个支持 SSH 和 SFTP 的 Python 模块;但它不支持SCP。
背景:我连接到一个不支持 SFTP 但支持 SSH/SCP 的路由器,所以 SFTP 不是一个选项。
编辑:这是 How to copy a file to a remote server in Python using SCP or SSH? 的副本。 但是,该问题没有给出处理 Python 内部密钥的特定于 scp 的答案。我希望有一种方法来运行类似的代码
import scp
client = scp.Client(host=host, user=user, keyfile=keyfile)
# or
client = scp.Client(host=host, user=user)
client.use_system_keys()
# or
client = scp.Client(host=host, user=user, password=password)
# and then
client.transfer('/etc/local/filename', '/etc/remote/filename')
试试 Python scp module for Paramiko。它非常易于使用。请参见以下示例:
import paramiko
from scp import SCPClient
def createSSHClient(server, port, user, password):
client = paramiko.SSHClient()
client.load_system_host_keys()
client.set_missing_host_key_policy(paramiko.AutoAddPolicy())
client.connect(server, port, user, password)
return client
ssh = createSSHClient(server, port, user, password)
scp = SCPClient(ssh.get_transport())
然后调用 scp.get()
或 scp.put()
进行 SCP 操作。
您可能有兴趣尝试 Pexpect (source code)。这将允许您处理输入密码的交互式提示。
这是来自主网站的示例用法(用于 ftp)的片段:
# This connects to the openbsd ftp site and
# downloads the recursive directory listing.
import pexpect
child = pexpect.spawn ('ftp ftp.openbsd.org')
child.expect ('Name .*: ')
child.sendline ('anonymous')
child.expect ('Password:')
child.sendline ('noah@example.com')
child.expect ('ftp> ')
child.sendline ('cd pub')
child.expect('ftp> ')
child.sendline ('get ls-lR.gz')
child.expect('ftp> ')
child.sendline ('bye')
找不到直接的答案,而且这个“scp.Client”模块不存在。相反,this 适合我:
from paramiko import SSHClient
from scp import SCPClient
ssh = SSHClient()
ssh.load_system_host_keys()
ssh.connect('example.com')
with SCPClient(ssh.get_transport()) as scp:
scp.put('test.txt', 'test2.txt')
scp.get('test2.txt')
您也可以查看 paramiko。还没有 scp 模块,但它完全支持 sftp。
[编辑] 对不起,错过了你提到 paramiko 的那一行。以下模块只是 paramiko 的 scp 协议的实现。如果您不想使用 paramiko 或 conch(我所知道的唯一用于 python 的 ssh 实现),您可以重新设计它以使用管道在常规 ssh 会话上运行。
如果你在 win32 上安装 putty,你会得到一个 pscp(putty scp)。
所以你也可以在 win32 上使用 os.system hack。
(您可以使用 putty-agent 进行密钥管理)
抱歉,这只是一个 hack(但你可以将它包装在 python 类中)
截至今天,最好的解决方案可能是AsyncSSH
https://asyncssh.readthedocs.io/en/latest/#scp-client
async with asyncssh.connect('host.tld') as conn:
await asyncssh.scp((conn, 'example.txt'), '.', recurse=True)
您可以使用包子进程和命令调用来使用 shell 中的 scp 命令。
from subprocess import call
cmd = "scp user1@host1:files user2@host2:files"
call(cmd.split(" "))
from fabric import Connection
with Connection(host="hostname",
user="admin",
connect_kwargs={"key_filename": "/home/myuser/.ssh/private.key"}
) as c:
c.get('/foo/bar/file.txt', '/tmp/')
import paramiko
client = paramiko.SSHClient()
client.set_missing_host_key_policy(paramiko.AutoAddPolicy())
client.connect('<IP Address>', username='<User Name>',password='' ,key_filename='<.PEM File path')
#Setup sftp connection and transmit this script
print ("copying")
sftp = client.open_sftp()
sftp.put(<Source>, <Destination>)
sftp.close()
自从提出这个问题以来已经有一段时间了,与此同时,出现了另一个可以处理这个问题的库:您可以使用 Plumbum 库中包含的 copy 函数:
import plumbum
r = plumbum.machines.SshMachine("example.net")
# this will use your ssh config as `ssh` from shell
# depending on your config, you might also need additional
# params, eg: `user="username", keyfile=".ssh/some_key"`
fro = plumbum.local.path("some_file")
to = r.path("/path/to/destination/")
plumbum.path.utils.copy(fro, to)
SshMachine
确实可以访问 ssh-agent
,所以如果您只是想避免每次都输入它,这是一种方法。你也可以在 plumbum 的 github 页面上提交功能请求。
如果您在 *nix 上,您可以使用 sshpass
sshpass -p password scp -o User=username -o StrictHostKeyChecking=no src dst:/path
我前段时间整理了一个依赖于 paramiko 的 python SCP 复制脚本。它包括处理与私钥或 SSH 密钥代理的连接的代码,并回退到密码身份验证。
http://code.activestate.com/recipes/576810-copy-files-over-ssh-using-paramiko/
scp
最终实际上调用了仅适用于 *nix 的scp
命令行。scp -t
或scp -f
的远程调用('to' 和 'from' 模式,存在用于此服务器端目的)。这本质上就是 scp 的工作方式——它依赖于服务器上存在的另一个 scp 副本。这篇文章解释得很好:blogs.oracle.com/janp/entry/how_the_scp_protocol_worksimport paramiko
from scp import SCPClient
2. 这是 scp.get() 命令的示例:scp.get(r'/nfs_home/appers/xxxx/test2.txt', r'C:\Users\xxxx\Desktop\MR_Test')
scp.SCPException: scp Protocol not available
。我正在尝试将文件从 Windows 复制到 Mac。相同的命令在终端上运行良好。