4

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

远程 bash 脚本:

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

BASH CMD 结果(在命令行调用远程 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脚本结果

$> 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
$>

我错过了什么?

4

2 回答 2

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 解析器拦截了<和 后面的文件,并将该文件的内容重定向到程序的标准输入。换句话说,ssh就是从标准输入读取文件。您还希望将文件传递给stdin使用sshpython。

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我的本地机器上的文件在哪里。

于 2012-11-07T19:51:40.593 回答
0

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

于 2012-11-07T20:50:57.107 回答