使用popen和专用的TTY Python运行交互式Bash

我需要在一个独立的Python进程中运行一个交互的Bash实例,并使用它自己的TTY(我不能使用pexpect)。 我使用这个代码片段,我通常看到类似的程序中使用:

master, slave = pty.openpty() p = subprocess.Popen(["/bin/bash", "-i"], stdin=slave, stdout=slave, stderr=slave) os.close(slave) x = os.read(master, 1026) print x subprocess.Popen.kill(p) os.close(master) 

但是当我运行它时,我得到以下输出:

 $ ./pty_try.py bash: cannot set terminal process group (10790): Inappropriate ioctl for device bash: no job control in this shell 

运行Strace显示一些错误:

 ... readlink("/usr/bin/python2.7", 0x7ffc8db02510, 4096) = -1 EINVAL (Invalid argument) ... ioctl(3, SNDCTL_TMR_TIMEBASE or SNDRV_TIMER_IOCTL_NEXT_DEVICE or TCGETS, 0x7ffc8db03590) = -1 ENOTTY (Inappropriate ioctl for device) ... readlink("./pty_try.py", 0x7ffc8db00610, 4096) = -1 EINVAL (Invalid argument) 

代码片段看起来很简单,Bash没有得到它需要的东西? 这里可能是什么问题?

Solutions Collecting From Web of "使用popen和专用的TTY Python运行交互式Bash"

这是在子流程中运行交互式命令的解决方案。 它使用伪终端使标准输出非阻塞(也有一些命令需要一个tty设备,例如bash)。 它使用select来处理输入和输出到子进程。

 #!/usr/bin/env python # -*- coding: utf-8 -*- import os import sys import select import termios import tty import pty from subprocess import Popen command = 'bash' # command = 'docker run -it --rm centos /bin/bash'.split() # save original tty setting then set it to raw mode old_tty = termios.tcgetattr(sys.stdin) tty.setraw(sys.stdin.fileno()) # open pseudo-terminal to interact with subprocess master_fd, slave_fd = pty.openpty() # use os.setsid() make it run in a new process group, or bash job control will not be enabled p = Popen(command, preexec_fn=os.setsid, stdin=slave_fd, stdout=slave_fd, stderr=slave_fd, universal_newlines=True) while p.poll() is None: r, w, e = select.select([sys.stdin, master_fd], [], []) if sys.stdin in r: d = os.read(sys.stdin.fileno(), 10240) os.write(master_fd, d) elif master_fd in r: o = os.read(master_fd, 10240) if o: os.write(sys.stdout.fileno(), o) # restore tty settings back termios.tcsetattr(sys.stdin, termios.TCSADRAIN, old_tty) 

这是最后解决方案(由qarma建议):

 libc = ctypes.CDLL('libc.so.6') master, slave = pty.openpty() p = subprocess.Popen(["/bin/bash", "-i"], preexec_fn=libc.setsid, stdin=slave, stdout=slave, stderr=slave) os.close(slave) ... do stuff here ... x = os.read(master, 1026) print x 

您将不得不通过输入来读取,然后打印x。