Having a console in a single-threaded Python script

前端 未结 2 958
攒了一身酷
攒了一身酷 2020-12-15 14:44

I would like to have an interactive console in a single-threaded script that has several TCP connections open. This means I can\'t just have a standard input blocking the th

相关标签:
2条回答
  • 2020-12-15 15:24

    You can subclass InteractiveConsole (from the builtin 'code' module) and override the push() method with a wrapper that redirects stdout/stderr to a StringIO instance before forwarding to the base InteractiveConsole's push() method. Your wrapper can return a 2-tuple (more, result) where 'more' indicates whether InteractiveConsole expects more input, and 'result' is whatever InteractiveConsole.push() wrote to your StringIO instance.

    It sounds harder than it is. Here's the basic premise:

    import sys
    from cStringIO import StringIO
    from code import InteractiveConsole
    from contextlib import contextmanager
    
    __all__ = ['Interpreter']
    
    
    @contextmanager
    def std_redirector(stdin=sys.stdin, stdout=sys.stdin, stderr=sys.stderr):
        """Temporarily redirect stdin/stdout/stderr"""
    
        tmp_fds = stdin, stdout, stderr
        orig_fds = sys.stdin, sys.stdout, sys.stderr
        sys.stdin, sys.stdout, sys.stderr = tmp_fds
        yield
        sys.stdin, sys.stdout, sys.stderr = orig_fds
    
    
    class Interpreter(InteractiveConsole):
        """Remote-friendly InteractiveConsole subclass
    
        This class behaves just like InteractiveConsole, except that it
        returns all output as a string rather than emitting to stdout/stderr
    
        """
        banner = ("Python %s\n%s\n" % (sys.version, sys.platform) +
                  'Type "help", "copyright", "credits" or "license" '
                  'for more information.\n')
    
        ps1 = getattr(sys, "ps1", ">>> ")
        ps2 = getattr(sys, "ps2", "... ")
    
    
        def __init__(self, locals=None):
            InteractiveConsole.__init__(self, locals=locals)
            self.output = StringIO()
            self.output = StringIO()
    
        def push(self, command):
            """Return the result of executing `command`
    
            This function temporarily redirects stdout/stderr and then simply
            forwards to the base class's push() method.  It returns a 2-tuple
            (more, result) where `more` is a boolean indicating whether the
            interpreter expects more input [similar to the base class push()], and
            `result` is the captured output (if any) from running `command`.
    
            """
            self.output.reset()
            self.output.truncate()
            with std_redirector(stdout=self.output, stderr=self.output):
                try:
                    more = InteractiveConsole.push(self, command)
                    result = self.output.getvalue()
                except (SyntaxError, OverflowError):
                    pass
                return more, result
    

    Check out this complete example, which accepts input from a UDP socket:

    • http://files.evadeflow.com/pyrrepl.zip

    Start two consoles and run server.py in one, client.py in the other. What you see in client.py should be indistinguishable from python's regular interactive interpreter, even though all commands are being round-tripped to server.py for evaluation.

    Of course, using sockets like this is terribly insecure, but it illustrates how to evaluate an external input asynchronously. You should be able to adapt it to your situation, as long as you trust the input source. Things get 'interesting' when someone types:

    while True: continue
    

    But that's another problem entirely... :-)

    0 讨论(0)
  • 2020-12-15 15:34

    Either single-threaded or multi-threaded will do, but if you choose not to use threads, you will need to use polling (in C this may be done using poll(2), for example) and check for whether the console and/or the TCP connections have input ready.

    0 讨论(0)
提交回复
热议问题