mirror of
https://github.com/markqvist/rnsh.git
synced 2025-01-23 04:31:01 -05:00
115 lines
3.3 KiB
Python
115 lines
3.3 KiB
Python
import uuid
|
|
import time
|
|
from types import TracebackType
|
|
from typing import Type
|
|
|
|
import rnsh.retry
|
|
from contextlib import AbstractContextManager
|
|
import logging
|
|
logging.getLogger().setLevel(logging.DEBUG)
|
|
|
|
|
|
class State(AbstractContextManager):
|
|
def __init__(self, delay: float):
|
|
self.delay = delay
|
|
self.retry_thread = rnsh.retry.RetryThread(self.delay / 10.0)
|
|
self.tries = 0
|
|
self.callbacks = 0
|
|
self.timed_out = False
|
|
self.tag = str(uuid.uuid4())
|
|
self.got_tag = None
|
|
assert self.retry_thread.is_alive()
|
|
|
|
def cleanup(self):
|
|
self.retry_thread.wait()
|
|
assert self.tries != 0
|
|
self.retry_thread.close()
|
|
assert not self.retry_thread.is_alive()
|
|
|
|
def retry(self, tag, tries):
|
|
self.tries = tries
|
|
self.got_tag = tag
|
|
self.callbacks += 1
|
|
return self.tag
|
|
|
|
def timeout(self, tag, tries):
|
|
self.tries = tries
|
|
self.got_tag = tag
|
|
self.timed_out = True
|
|
self.callbacks += 1
|
|
|
|
def __exit__(self, __exc_type: Type[BaseException], __exc_value: BaseException,
|
|
__traceback: TracebackType) -> bool:
|
|
self.cleanup()
|
|
return False
|
|
|
|
|
|
def test_retry_timeout():
|
|
|
|
with State(0.1) as state:
|
|
state.retry_thread.begin(try_limit=3,
|
|
wait_delay=state.delay,
|
|
try_callback=state.retry,
|
|
timeout_callback=state.timeout)
|
|
|
|
assert state.tries == 1
|
|
assert state.callbacks == 1
|
|
assert state.got_tag is None
|
|
assert not state.timed_out
|
|
time.sleep(state.delay / 2.0)
|
|
time.sleep(state.delay)
|
|
assert state.tries == 2
|
|
assert state.callbacks == 2
|
|
assert state.got_tag == state.tag
|
|
assert not state.timed_out
|
|
time.sleep(state.delay)
|
|
assert state.tries == 3
|
|
assert state.callbacks == 3
|
|
assert state.got_tag == state.tag
|
|
assert not state.timed_out
|
|
|
|
# check timeout
|
|
time.sleep(state.delay)
|
|
assert state.tries == 3
|
|
assert state.callbacks == 4
|
|
assert state.got_tag == state.tag
|
|
assert state.timed_out
|
|
|
|
# check no more callbacks
|
|
time.sleep(state.delay * 3.0)
|
|
assert state.callbacks == 4
|
|
assert state.tries == 3
|
|
|
|
|
|
def test_retry_complete():
|
|
with State(0.01) as state:
|
|
state.retry_thread.begin(try_limit=3,
|
|
wait_delay=state.delay,
|
|
try_callback=state.retry,
|
|
timeout_callback=state.timeout)
|
|
|
|
assert state.tries == 1
|
|
assert state.callbacks == 1
|
|
assert state.got_tag is None
|
|
assert not state.timed_out
|
|
time.sleep(state.delay / 2.0)
|
|
time.sleep(state.delay)
|
|
assert state.tries == 2
|
|
assert state.callbacks == 2
|
|
assert state.got_tag == state.tag
|
|
assert not state.timed_out
|
|
|
|
state.retry_thread.complete(state.tag)
|
|
|
|
time.sleep(state.delay)
|
|
assert state.tries == 2
|
|
assert state.callbacks == 2
|
|
assert state.got_tag == state.tag
|
|
assert not state.timed_out
|
|
|
|
# check no more callbacks
|
|
time.sleep(state.delay * 3.0)
|
|
assert state.callbacks == 2
|
|
assert state.tries == 2
|
|
|