我正在尝试编写一个python脚本,通过scp将文件从远程服务器复制到本地目录。
因为我在OpenELEC发行版上运行它(最小的HTPC linux发行版,除了userhome之外的只读文件系统使安装python ssh模块变得不切实际),我这样做很难看,只是将文件名传递给scp命令通过os.system。
SCPCopy = "scp -c blowfish -C user@host:\"" + pipes.quote(file) + "\" /storage/downloads/incoming/"
SCPCopy = SCPCopy.replace('\n','')
os.system(SCPCopy)
除了包含撇号的文件名外,这是有效的。
下面是一个在带有撇号的文件中传递给os.system的示例:
scp -c blowfish -C user@host:"'/media/sdi1/home/data/bob'"'"'s file.avi'" /storage/downloads/incoming/
错误:
sh: -c: line 0: unexpected EOF while looking for matching `''
sh: -c: line 1: syntax error: unexpected end of file
看起来pipes.quote(x)正在逃避撇号(应该如此),但显然语法仍然不正确。我已经尝试过丢弃pipes.quote(x)并用/来替换撇号,但这也没有让我到任何地方。
答案 0 :(得分:6)
由于scp
基于SSH
,因此您提供给它的文件名也会在远程端进行shell转义。因此你需要逃脱两次。
shell正确转义的cmdline:
scp -c blowfish -C user@host:"\"/media/sdi1/home/data/bob's file\"" /storage/.../
要创建一个python字符串,我们必须再添加一个转义级别。为了保持理智,我们可以使用三重引号:
"""scp -c blowfish -C user@host:"\"/media/sdi1/home/data/bob's file\"" /storage/.../"""
如果您以编程方式执行此操作(例如使用已弃用的pipes.quote
),则根本不要触摸文件名(在上面的示例中,您在文件名周围添加了撇号)。
fp = "/media/sdi1/home/data/bob's file.avi"
fp = "user@host:" + pipes.quote(pipes.quote(fp))
cmdline = "scp -c blowfish -C " + fp + " /storage/downloads/incoming/"
os.system(cmdline)
这无疑令人困惑。对于一个简单的模型,pipes.quote
的整个点是转义输入,以便shell将输入解析为一个单词,等于输入强>
以下是更通用的正确方法(并产生相同的结果):
fp = "/media/sdi1/home/data/bob's file.avi"
# the filepath argument escaped for ssh/scp on the remote side
fp = pipes.quote(fp)
commandargs = ["scp", "-c", "blowfish", "-C", "user@host:"+fp, "/storage/downloads/incoming/"]
# escape all words for the local shell, and then concatenate space-separated
cmdline = " ".join(map(pipes.quote, commandargs))
os.system(cmdline)
它更清楚地表达了意图:控制shell将解析的单词。
但为什么首先要开始使用shell?我们不需要一个,可以在本地保存逃逸。要使用我们的args直接生成进程,请使用os.exec*
系列中的命令。
fp = pipes.quote("/media/sdi1/home/data/bob's file.avi")
commandargs = ["scp", "-c", "blowfish", "-C", "user@host:"+fp, "/storage/downloads/incoming/"]
if os.fork() == 0:
os.execvp("scp", commandargs)