2014-08-12 10:10:52 -04:00
|
|
|
# -*- coding: utf-8 -*-
|
2015-01-06 08:21:39 -05:00
|
|
|
# Copyright 2014, 2015 OpenMarket Ltd
|
2014-08-12 10:10:52 -04:00
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
|
2014-08-12 22:14:34 -04:00
|
|
|
|
2014-08-12 10:10:52 -04:00
|
|
|
from twisted.internet import defer, reactor
|
|
|
|
|
2015-05-08 11:32:18 -04:00
|
|
|
from .logcontext import preserve_context_over_deferred
|
2014-08-12 10:10:52 -04:00
|
|
|
|
2014-11-20 12:26:36 -05:00
|
|
|
|
2014-08-12 10:10:52 -04:00
|
|
|
def sleep(seconds):
|
|
|
|
d = defer.Deferred()
|
|
|
|
reactor.callLater(seconds, d.callback, seconds)
|
2015-05-08 11:32:18 -04:00
|
|
|
return preserve_context_over_deferred(d)
|
2014-10-29 13:02:22 -04:00
|
|
|
|
2014-11-20 12:26:36 -05:00
|
|
|
|
2014-10-29 13:02:22 -04:00
|
|
|
def run_on_reactor():
|
|
|
|
""" This will cause the rest of the function to be invoked upon the next
|
|
|
|
iteration of the main loop
|
|
|
|
"""
|
2014-11-14 06:16:50 -05:00
|
|
|
return sleep(0)
|
2015-04-27 08:59:37 -04:00
|
|
|
|
|
|
|
|
2015-05-08 11:27:36 -04:00
|
|
|
class ObservableDeferred(object):
|
|
|
|
"""Wraps a deferred object so that we can add observer deferreds. These
|
|
|
|
observer deferreds do not affect the callback chain of the original
|
|
|
|
deferred.
|
|
|
|
|
|
|
|
If consumeErrors is true errors will be captured from the origin deferred.
|
2015-04-27 08:59:37 -04:00
|
|
|
"""
|
|
|
|
|
2015-05-08 11:27:36 -04:00
|
|
|
__slots__ = ["_deferred", "_observers", "_result"]
|
|
|
|
|
|
|
|
def __init__(self, deferred, consumeErrors=False):
|
|
|
|
object.__setattr__(self, "_deferred", deferred)
|
|
|
|
object.__setattr__(self, "_result", None)
|
|
|
|
object.__setattr__(self, "_observers", [])
|
|
|
|
|
|
|
|
def callback(r):
|
|
|
|
self._result = (True, r)
|
|
|
|
while self._observers:
|
|
|
|
try:
|
|
|
|
self._observers.pop().callback(r)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
return r
|
|
|
|
|
|
|
|
def errback(f):
|
|
|
|
self._result = (False, f)
|
|
|
|
while self._observers:
|
|
|
|
try:
|
|
|
|
self._observers.pop().errback(f)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
if consumeErrors:
|
|
|
|
return None
|
|
|
|
else:
|
|
|
|
return f
|
|
|
|
|
|
|
|
deferred.addCallbacks(callback, errback)
|
2015-04-27 08:59:37 -04:00
|
|
|
|
2015-05-08 11:27:36 -04:00
|
|
|
def observe(self):
|
|
|
|
if not self._result:
|
|
|
|
d = defer.Deferred()
|
|
|
|
self._observers.append(d)
|
|
|
|
return d
|
|
|
|
else:
|
|
|
|
success, res = self._result
|
|
|
|
return defer.succeed(res) if success else defer.fail(res)
|
2015-04-27 08:59:37 -04:00
|
|
|
|
2015-05-08 11:27:36 -04:00
|
|
|
def __getattr__(self, name):
|
|
|
|
return getattr(self._deferred, name)
|
2015-04-27 08:59:37 -04:00
|
|
|
|
2015-05-08 11:27:36 -04:00
|
|
|
def __setattr__(self, name, value):
|
|
|
|
setattr(self._deferred, name, value)
|