2019-03-24 14:14:00 +01:00
|
|
|
from __future__ import print_function
|
2008-10-28 21:15:48 -07:00
|
|
|
|
|
|
|
import time
|
|
|
|
from twisted.internet import task
|
|
|
|
|
|
|
|
class TimeoutError(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
class PollComplete(Exception):
|
|
|
|
pass
|
|
|
|
|
2019-05-15 08:17:44 +02:00
|
|
|
class PollMixin(object):
|
2009-06-22 19:07:31 -07:00
|
|
|
_poll_should_ignore_these_errors = []
|
2008-10-28 21:15:48 -07:00
|
|
|
|
2009-07-08 21:47:29 -07:00
|
|
|
def poll(self, check_f, pollinterval=0.01, timeout=1000):
|
2008-10-28 21:15:48 -07:00
|
|
|
# Return a Deferred, then call check_f periodically until it returns
|
|
|
|
# True, at which point the Deferred will fire.. If check_f raises an
|
|
|
|
# exception, the Deferred will errback. If the check_f does not
|
|
|
|
# indicate success within timeout= seconds, the Deferred will
|
|
|
|
# errback. If timeout=None, no timeout will be enforced, and the loop
|
|
|
|
# will poll forever (or really until Trial times out).
|
|
|
|
cutoff = None
|
|
|
|
if timeout is not None:
|
|
|
|
cutoff = time.time() + timeout
|
|
|
|
lc = task.LoopingCall(self._poll, check_f, cutoff)
|
|
|
|
d = lc.start(pollinterval)
|
|
|
|
def _convert_done(f):
|
|
|
|
f.trap(PollComplete)
|
|
|
|
return None
|
|
|
|
d.addErrback(_convert_done)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def _poll(self, check_f, cutoff):
|
|
|
|
if cutoff is not None and time.time() > cutoff:
|
|
|
|
raise TimeoutError("PollMixin never saw %s return True" % check_f)
|
|
|
|
if check_f():
|
|
|
|
raise PollComplete()
|
2009-06-22 19:07:31 -07:00
|
|
|
# since PollMixin is mostly used for unit tests, quit if we see any
|
|
|
|
# logged errors. This should give us a nice fast failure, instead of
|
|
|
|
# waiting for a timeout. Tests which use flushLoggedErrors() will
|
|
|
|
# need to warn us by putting the error types they'll be ignoring in
|
|
|
|
# self._poll_should_ignore_these_errors
|
|
|
|
if hasattr(self, "_observer") and hasattr(self._observer, "getErrors"):
|
|
|
|
errs = []
|
|
|
|
for e in self._observer.getErrors():
|
|
|
|
if not e.check(*self._poll_should_ignore_these_errors):
|
|
|
|
errs.append(e)
|
|
|
|
if errs:
|
2019-03-24 14:14:00 +01:00
|
|
|
print(errs)
|
2009-06-22 19:07:31 -07:00
|
|
|
self.fail("Errors snooped, terminating early")
|
2008-10-28 21:15:48 -07:00
|
|
|
|