使用Python通过ssh将文件捕获到远程bash脚本

时间:2012-11-07 19:26:48

标签: python bash shlex

我正在使用带有shlex的subprocess.popen来使用ssh调用远程bash脚本。这个命令在bash上运行得很好。但是一旦我尝试将它转换为python并使用subprocess.popen进行shlex,它就会出错。

远程bash脚本:

#!/bin/bash
tmp="";     
while read -r line;
do
    tmp="$tmp $line\n";
done;
echo $tmp;

BASH CMD RESULT (在命令行上调用远程bash脚本)

$> ssh x.x.x.x cat < /tmp/bef69a1d-e580-5780-8963-6a9b950e529f.txt " | /path/to/bash/script.sh;"
Bar\n
$> 

Python代码

import shlex
import subprocess

fn = '/tmp/bef69a1d-e580-5780-8963-6a9b950e529f.txt'
s = """                                                                    
ssh x.x.x.x cat < {localfile} '| /path/to/bash/script.sh;'
""".format(localfile=fn)

print s

lexer = shlex.shlex(s)                                                     
lexer.quotes = "'"                                                         
lexer.whitespace_split = True                                              
sbash = list(lexer)                                                        
print sbash                                                                

# print buildCmd                                                           
proc=subprocess.Popen(sbash,stdout=subprocess.PIPE,stderr=subprocess.PIPE) 
out,err=proc.communicate()                                                 

print "Out: " + out                                                        
print "Err: " + err                                                        

PYTHON SCRIPT RESULT

$> python rt.py

    ssh x.x.x.x cat < /tmp/bef69a1d-e580-5780-8963-6a9b950e529f.txt '| /path/to/bash/script.sh'
['ssh', 'x.x.x.x', 'cat', '<', '/tmp/bef69a1d-e580-5780-8963-6a9b950e529f.txt', "'| /path/to/bash/script.sh'"]
Out: 
Err: bash: /tmp/bef69a1d-e580-5780-8963-6a9b950e529f.txt: No such file or directory
$>

我错过了什么?

2 个答案:

答案 0 :(得分:4)

问题是您在命令中使用了shell重定向,但是在使用子进程时没有生成shell。

考虑以下(非常简单)的程序:

import sys
print sys.argv

现在,如果我们运行它就像你正在运行ssh(假设存在foofile.txt),我们得到:

python argcheck.py ssh cat < foofile.txt " | /path/to/bash/script.sh;"
['argcheck.py', 'ssh', 'cat', ' | /path/to/bash/script.sh;']

请注意,< foofile.txt永远不会使用python的命令行参数。那是因为bash解析器截获<及其后面的文件,并将该文件的内容重定向到程序的stdin。换句话说,ssh正在从stdin读取文件。您希望使用python将文件传递给stdin ssh

s = """                                                                    
ssh x.x.x.x cat '| /path/to/bash/script.sh;'
"""

#<snip>

proc=subprocess.Popen(sbash,stdout=subprocess.PIPE,stderr=subprocess.PIPE,
                            stdin=subprocess.PIPE)
out,err=proc.communicate(open(fn).read())

可能会起作用。


以下适用于我:

import subprocess
from subprocess import PIPE

with open('foo.h') as f:
    p = subprocess.Popen(['ssh','mgilson@XXXXX','cat','| cat'],stdin=f,stdout=PIPE,stderr=PIPE)
    out,err = p.communicate()
    print out
    print '#'*80
    print err

bash中的等效命令:

ssh mgilson@XXXXX cat < foo.h '| cat'

其中foo.h是我本地计算机上的文件。

答案 1 :(得分:0)

为什么不放弃中间人并使用Paramiko