0

我有以下python代码:

import pty
import subprocess
os=subprocess.os
from subprocess import PIPE
import time
import resource

pipe=subprocess.Popen(["cat"], stdin=PIPE, stdout=PIPE, stderr=PIPE, \
                      close_fds=True)
skip=[f.fileno() for f in (pipe.stdin, pipe.stdout, pipe.stderr)]
pid, child_fd = pty.fork()
if(pid==0):
    max_fd=resource.getrlimit(resource.RLIMIT_NOFILE)[0]
    fd=3
    while fd<max_fd:
        if(fd not in skip):
            try:
                os.close(fd)
            except OSError:
                pass
            fd+=1
        enviroment=os.environ.copy()
        enviroment.update({"FD": str(pipe.stdin.fileno())})
        os.execvpe("zsh", ["-i", "-s"], enviroment)
else:
    os.write(child_fd, "echo a >&$FD\n")
    time.sleep(1)
    print pipe.stdout.read(2)

我怎样才能重写它,使它不会使用Popenand cat?我需要一种方法从在交互式 shell 中运行的 shell 函数传递数据,该函数不会与其他函数创建的数据混合(所以我不能使用 stdout 或 stderr)。

4

1 回答 1

1

好的,我想我现在已经解决了你的问题,看看你可以采取两种不同的方法。

如果您绝对想为子进程中的 shell 提供一个已经打开的文件描述符,那么您可以将Popen()of替换为catos.pipe(). 这将为您提供一对连接的真实文件描述符(不是 Pythonfile对象)。写入第二个文件描述符的任何内容都可以从第一个文件描述符中读取,替换您的 jury-rigged cat-pipe。(虽然“猫管”说起来很有趣……)。socket.socketpair()如果您需要双向对,也可以使用套接字对 ( ) 来实现相同的目的。

或者,您可以通过使用命名管道(又名 FIFO)进一步简化您的生活。如果您不熟悉该工具,命名管道是位于文件系统命名空间中的单向管道。该os.mkfifo()函数将在文件系统上创建管道。然后,您可以打开管道以在您的主进程中读取,并打开它以从您的 shell 子进程写入/直接输出到它。这应该会简化您的代码并打开使用现有库(如Pexpect)与 shell 交互的选项。

于 2010-09-23T03:26:05.863 回答