8

我使用 cmd 模块编写了一个 Python 3.5 应用程序。我想实现的最后一件事是正确处理 CTRL-C (sigint) 信号。我希望它的行为或多或少像 Bash 那样:

  • 在光标所在的点打印 ^C
  • 清除缓冲区以便删除输入文本
  • 跳到下一行,打印提示,等待输入

基本上:

/test $ bla bla bla|
# user types CTRL-C
/test $ bla bla bla^C
/test $ 

这是作为可运行示例的简化代码:

import cmd
import signal


class TestShell(cmd.Cmd):
    def __init__(self):
        super().__init__()

        self.prompt = '$ '

        signal.signal(signal.SIGINT, handler=self._ctrl_c_handler)
        self._interrupted = False

    def _ctrl_c_handler(self, signal, frame):
        print('^C')
        self._interrupted = True

    def precmd(self, line):
        if self._interrupted:
            self._interrupted = False
            return ''

        if line == 'EOF':
            return 'exit'

        return line

    def emptyline(self):
        pass

    def do_exit(self, line):
        return True


TestShell().cmdloop()

这几乎可以工作。当我按 CTRL-C 时,^C 会打印在光标处,但我仍然必须按 Enter。然后,该precmd方法注意到self._interrupted处理程序设置的标志,并返回一个空行。这是我所能接受的,但我想以某种方式不必按该输入。

我想我需要以某种方式强制input()返回,有人有想法吗?

4

1 回答 1

7

我发现了一些用 Ctrl-C 实现你想要的行为的 hacky 方法。

设置use_rawinput=False和替换stdin

这个坚持(或多或少......)到cmd.Cmd. 不幸的是,它禁用了 readline 支持。

您可以设置use_rawinput为 false 并传递一个不同的类似文件的对象来替换stdin. Cmd.__init__()在实践中,只有readline()在这个对象上被调用。因此,您可以创建一个包装器来stdin捕获KeyboardInterrupt并执行您想要的行为:

class _Wrapper:

    def __init__(self, fd):
        self.fd = fd

    def readline(self, *args):
        try:
            return self.fd.readline(*args)
        except KeyboardInterrupt:
            print()
            return '\n'


class TestShell(cmd.Cmd):

    def __init__(self):
        super().__init__(stdin=_Wrapper(sys.stdin))
        self.use_rawinput = False
        self.prompt = '$ '

    def precmd(self, line):
        if line == 'EOF':
            return 'exit'
        return line

    def emptyline(self):
        pass

    def do_exit(self, line):
        return True


TestShell().cmdloop()

当我在终端上运行它时,Ctrl-C 显示^C并切换到新行。

猴子补丁input()

如果您想要 的结果input(),除了您想要 Ctrl-C 的不同行为之外,一种方法是使用不同的函数而不是input()

def my_input(*args):   # input() takes either no args or one non-keyword arg
    try:
        return input(*args)
    except KeyboardInterrupt:
        print('^C')   # on my system, input() doesn't show the ^C
        return '\n'

但是,如果你只是盲目地设置input = my_input,你会得到无限递归,因为my_input()will call input(),它现在就是它本身。但这是可以修复的,您可以__builtins__在模块中修补字典以在以下期间cmd使用修改后的input()方法Cmd.cmdloop()

def input_swallowing_interrupt(_input):
    def _input_swallowing_interrupt(*args):
        try:
            return _input(*args)
        except KeyboardInterrupt:
            print('^C')
            return '\n'
    return _input_swallowing_interrupt


class TestShell(cmd.Cmd):

    def cmdloop(self, *args, **kwargs):
        old_input_fn = cmd.__builtins__['input']
        cmd.__builtins__['input'] = input_swallowing_interrupt(old_input_fn)
        try:
            super().cmdloop(*args, **kwargs)
        finally:
            cmd.__builtins__['input'] = old_input_fn

    # ...

请注意,这会更改所有对象input(),而不仅仅是对象。如果你不能接受,你可以……</p> CmdTestShell

复制Cmd.cmdloop()源码并修改

由于您将其子类化,因此您可以cmdloop()做任何您想做的事情。“任何你想要的”可以包括复制部分Cmd.cmdloop()和重写其他部分。要么用input()对另一个函数的调用替换调用,要么KeyboardInterrupt在重写的cmdloop().

如果您担心底层实现会随着新版本的 Python 发生变化,您可以将整个cmd模块复制到一个新模块中,然后更改您想要的内容。

于 2016-05-22T21:51:51.730 回答