扭曲:如何在延迟上中断回调链

时间:2016-05-25 15:23:54

标签: python twisted deferred

我的问题:我有一个延迟回调和错误。我需要在特定的错误后停止该过程。换句话说,如果调用errback的特定函数,我需要得到它的返回值,并且不要处理以下回调。

from twisted.internet import defer
from twisted.python import failure, util


class Test (object):

@classmethod
def handleFailure(self, f):
    print "handleFailure"
    f.trap(RuntimeError)
    return '0', 'erro'

@classmethod
def handleResult(self, result, message):
    print "handleResult of %s. Old results %s, %s: " % (message, result[0], result[1])
    return 1, 'ok'

@classmethod
def doFailure (self, result, message):
    print "handleResult of %s. Old results %s, %s: " % (message, result[0], result[1])
    raise RuntimeError, "whoops! we encountered an error"


@classmethod
def deferredExample(self):
    d = defer.Deferred()
    # 1o. call without error
    d.addCallback(Test.handleResult, 'call 1')
    # 2o. call without error
    d.addCallback(Test.handleResult, 'call 1')
    # 3o. call causes the failure
    d.addCallback(Test.doFailure,    'call 3')
    # the failure calls the error back 
    d.addErrback (Test.handleFailure) # - A -
    # after error back, the next call back is called
    d.addCallback(Test.handleResult, 'call 4')  # - B -
    # and the last call back is called   
    d.addCallback(Test.handleResult, 'call 5')  # - C -

    d.callback("success")
    return d.result[0], d.result[1]

if __name__ == '__main__': 
#    behindTheScenes("success")
    print "\n-------------------------------------------------\n"
    global num; num = 0
    tst = Test()
    rtn1, rtn2 = tst.deferredExample()
    print "RTN: %s %s" % (rtn1, rtn2)

这段代码是一个简单的版本,而不是我需要的东西。在-A-过程之后,我需要使用-B-和-C-,最后,响应为&#34; 0,错误&#34;,而不是&#34; 1,ok&#39;。< / p>

我目前的回报:

handleResult of call 1. Old results s, u: 
handleResult of call 1. Old results 1, ok: 
handleResult of call 3. Old results 1, ok: 
handleFailure
handleResult of call 4. Old results 0, erro: 
handleResult of call 5. Old results 1, ok: 
RTN: 1 ok

我希望:

handleResult of call 1. Old results s, u: 
handleResult of call 1. Old results 1, ok: 
handleResult of call 3. Old results 1, ok: 
handleFailure
RTN: 0 erro

我该怎么做?

提前致谢

1 个答案:

答案 0 :(得分:1)

这是一个非常有趣的问题。 Twisted没有给出任何记录的API,用于在发生错误后修改回调链。如果你真的想在errback中断处理,你总是可以引发错误而不是陷阱。

但是如果你真的想在某些条件下从Deferred中删除所有回调,你可以使用简单的hack。回调保持为延迟作为简单对象属性,请参见此处:https://github.com/twisted/twisted/blob/twisted-16.2.0/twisted/internet/defer.py#L288如果您只是将此属性重置为空列表,则将删除所有回调并停止处理。

@classmethod
def handleFailure(self, f, d):
    print "handleFailure"
    f.trap(RuntimeError)
    d.callbacks = []
    return '0', 'erro'

这可以按照您的喜好工作,但看起来有点hacky所以我很乐意阅读其他用户的想法。这里有更多关于这个问题的讨论:Stopping Twisted from swallowing exceptions讨论表明你所做的事情有点违背扭曲的精神。