我正在使用带有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
$>
我错过了什么?
答案 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?