20

我有一个要在 Python (2.6.5) 中运行的脚本,它遵循以下逻辑:

  • 提示用户输入密码。它看起来像(“输入密码:”)(*注意:输入不会回显到屏幕)
  • 输出无关信息
  • 提示用户响应(“Blah Blah filename.txt blah blah (Y/N)?:”)

最后一个提示行包含我需要解析的文本(filename.txt)。提供的响应无关紧要(只要我可以解析该行,程序实际上可以在不提供响应的情况下退出这里)。

我的要求有点类似于Wrapping an interactive command line application in a Python script,但那里的响应似乎有点令人困惑,即使 OP 提到它不适合他,我的仍然挂起。

通过环顾四周,我得出的结论subprocess是最好的方法是这样做,但我遇到了一些问题。这是我的 Popen 行:

p = subprocess.Popen("cmd", shell=True, stdout=subprocess.PIPE,
stderr=subprocess.STDOUT, stdin=subprocess.PIPE)
  • 当我调用 aread()readline()onstdout时,提示是屏幕上的打印机并且它挂起。

  • 如果我调用write("password\n")for stdin,则提示将写入屏幕并挂起。中的文本write()未写入(我没有将光标移动到新行)。

  • 如果我打电话p.communicate("password\n"),与 write() 相同的行为

我在这里寻找一些关于最佳输入方式的想法,stdin如果你觉得慷慨的话,可能如何解析输出中的最后一行,尽管我最终可能会弄清楚。

4

2 回答 2

15

如果您正在与子进程生成的程序进行通信,您应该查看Python 中 subprocess.PIPE 上的非阻塞读取。我的应用程序遇到了类似的问题,发现使用队列是与子进程进行持续通信的最佳方式。

至于从用户那里获取值,您始终可以使用 raw_input() 内置函数来获取响应,对于密码,请尝试使用该getpass模块从用户那里获取非回显密码。然后,您可以解析这些响应并将它们写入您的子进程的标准输入。

我最终做了类似于以下的事情:

import sys
import subprocess
from threading import Thread

try:
    from Queue import Queue, Empty
except ImportError:
    from queue import Queue, Empty  # Python 3.x


def enqueue_output(out, queue):
    for line in iter(out.readline, b''):
        queue.put(line)
    out.close()


def getOutput(outQueue):
    outStr = ''
    try:
        while True: # Adds output from the Queue until it is empty
            outStr+=outQueue.get_nowait()

    except Empty:
        return outStr

p = subprocess.Popen("cmd", stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE, shell=False, universal_newlines=True)

outQueue = Queue()
errQueue = Queue()

outThread = Thread(target=enqueue_output, args=(p.stdout, outQueue))
errThread = Thread(target=enqueue_output, args=(p.stderr, errQueue))

outThread.daemon = True
errThread.daemon = True

outThread.start()
errThread.start()

try:
    someInput = raw_input("Input: ")
except NameError:
    someInput = input("Input: ")

p.stdin.write(someInput)
errors = getOutput(errQueue)
output = getOutput(outQueue)

一旦创建了队列并启动了线程,就可以循环获取用户的输入、进程的错误和输出,以及处理并将它们显示给用户。

于 2012-11-19T16:51:57.740 回答
1

对于简单的任务,使用线程可能有点过分。相反,可以使用 os.spawnvpe。它将作为进程生成脚本外壳。您将能够与脚本进行交互通信。在此示例中,我将密码作为参数传递,显然这不是一个好主意。

import os
import sys
from getpass import unix_getpass

def cmd(cmd):
    cmd = cmd.split()
    code = os.spawnvpe(os.P_WAIT, cmd[0], cmd, os.environ)
    if code == 127:
        sys.stderr.write('{0}: command not found\n'.format(cmd[0]))
    return code

password = unix_getpass('Password: ')
cmd_run = './run.sh --password {0}'.format(password)
cmd(cmd_run)

pattern = raw_input('Pattern: ')
lines = []
with open('filename.txt', 'r') as fd:
    for line in fd:
        if pattern in line:
            lines.append(line)

# manipulate lines
于 2015-04-06T20:41:16.173 回答