twisted - interrupt callback via KeyboardInterrupt

后端 未结 2 1287
情歌与酒
情歌与酒 2021-01-13 16:37

I\'m currently repeating a task in a for loop inside a callback using Twisted, but would like the reactor to break the loop in the callback (one) if the user issues a Keyboa

相关标签:
2条回答
  • 2021-01-13 17:15

    I got this working dandy. The fired SIGINT sets a flag running for any running task in my code, and additionally calls reactor.callFromThread(reactor.stop) to stop any twisted running code:

    #!/usr/bin/env python
    
    import sys
    import twisted
    import re
    from twisted.internet import reactor, defer, task
    import signal
    
    
    def one(result, token):
        print "Start one()"
        for i in xrange(1000):
            print i
            if token.running is False:
                raise KeyboardInterrupt()
                #reactor.callFromThread(reactor.stop) # this doesn't work
        print "End one()"
    
    def oneErrorHandler(failure):
        print "INTERRUPTING one(): Unkown Exception"
        import traceback
        print traceback.format_exc()
        reactor.stop()
    
    def oneKeyboardInterruptHandler(failure):
        failure.trap(KeyboardInterrupt)
        print "INTERRUPTING one(): KeyboardInterrupt"
        reactor.stop()
    
    def repeatingTask(token):
        d = defer.Deferred()
        d.addCallback(one, token)
        d.addErrback(oneKeyboardInterruptHandler)
        d.addErrback(oneErrorHandler)
        d.callback('result')
    
    class Token(object):
        def __init__(self):
            self.running = True
    
    def sayBye():
        print "bye bye."
    
    
    if __name__ == '__main__':
    
        token = Token()
    
        def customHandler(signum, stackframe):
            print "Got signal: %s" % signum
            token.running = False                # to stop my code
            reactor.callFromThread(reactor.stop) # to stop twisted code when in the reactor loop
        signal.signal(signal.SIGINT, customHandler)
    
        t2 = task.LoopingCall(reactor.callLater, 0, repeatingTask, token)
        t2.start(5) 
    
        reactor.addSystemEventTrigger('during', 'shutdown', sayBye)
    
        print "STARTING REACTOR..."
        reactor.run()
    
    0 讨论(0)
  • 2021-01-13 17:31

    This is intentional to avoid (semi-)preemption, since Twisted is a cooperative multitasking system. Ctrl-C is handled in Python with a SIGINT handler installed by the interpreter at startup. The handler sets a flag when it is invoked. After each byte code is executed, the interpreter checks the flag. If it is set, KeyboardInterrupt is raised at that point.

    The reactor installs its own SIGINT handler. This replaces the behavior of the interpreter's handler. The reactor's handler initiates reactor shutdown. Since it doesn't raise an exception, it doesn't interrupt whatever code is running. The loop (or whatever) gets to finish, and when control is returned to the reactor, shutdown proceeds.

    If you'd rather have Ctrl-C (ie SIGINT) raise KeyboardInterrupt, then you can just restore Python's SIGINT handler using the signal module:

    signal.signal(signal.SIGINT, signal.default_int_handler)
    

    Note, however, that if you send a SIGINT while code from Twisted is running, rather than your own application code, the behavior is undefined, as Twisted does not expect to be interrupted by KeyboardInterrupt.

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