2

目标

我想让我的 curses Python 应用程序在 Linux 机器的第一个物理控制台 (TTY1) 上显示它的输出,方法是将其添加到/etc/inittab、重新加载 inittelinit q等等。

我想避免使用 IO 重定向从以下方式启动时使用 IO 重定向的骇人听闻的方式/etc/inittab

1:2345:respawn:/path/to/app.py > /dev/tty1 < /dev/tty1

我所追求的是在我的应用程序中本地执行它,类似于getty它的方式,即您使用命令行参数来告诉它要听哪个 TTY:

S0:2345:respawn:/sbin/getty -L ttyS1 115200 vt100

示例代码

为简单起见,假设我编写了这个非常复杂的应用程序,当它被调用时,它会使用 ncurses 例程打印一些内容。

import curses

class CursesApp(object):

  def __init__(self, stdscr):
    self.stdscr = stdscr
    # Code producing some output, accepting user input, etc.
    # ...    

curses.wrapper(CursesApp)

我已经拥有的代码可以满足我的所有需求,只是它只在运行它的终端上显示它的输出。在没有我上面提到的 hacky 重定向的情况下从 inittab 调用时,它可以工作,但在 TTY1 上没有输出。

我知道 init 不会自行重定向输入和输出,所以这是意料之中的。

我需要如何修改现有代码以将其输出发送到请求的 TTY 而不是 STDOUT?

PS。我不是在问如何添加对命令行参数的支持,我已经有了这个,但为了简洁起见,将它从代码示例中删除了。

4

1 回答 1

2

这是相当简单的。只需打开终端设备一次输入,一次输出;然后将输入描述符复制到活动进程的文件描述符 0,并将输出描述符复制到文件描述符 1 和 2。然后关闭 TTY 的其他句柄:

import os
import sys

with open('/dev/tty6', 'rb') as inf, open('/dev/tty6', 'wb') as outf:
    os.dup2(inf.fileno(), 0)
    os.dup2(outf.fileno(), 1)
    os.dup2(outf.fileno(), 2)

我使用在 TTY6 上运行的cmd模块对此进行了测试:

import cmd
cmd.Cmd().cmdloop()

完美运行。使用 curses 从它们的外观中可以明显看出缺少一些东西:TERM环境变量:

os.environ['TERM'] = 'linux'

在导入之前执行所有这些语句curses,它应该可以工作。

于 2016-03-19T14:50:14.457 回答