You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 

60 lines
1.7 KiB

import sys
import time
if sys.version_info >= (3,3):
_monotonic = time.monotonic
else:
_monotonic = time.time # not really monotonic vOv
class ConstantRateLimit(object):
def __init__(self, times, seconds, exit_wait=False, sleep_func=time.sleep):
"""Context manager for enforcing constant rate on code inside the block .
`rate = seconds / times`
:param times: times to execute per...
:type times: :class:`int`
:param seconds: ...seconds
:type seconds: :class:`int`
:param exit_wait: whether to automatically call :meth:`wait` before exiting the block
:type exit_wait: :class:`bool`
:param sleep_func: Sleep function in seconds. Default: :func:`time.sleep`
:type sleep_func: :class:`bool`
Example:
.. code:: python
with RateLimiter(1, 5) as r:
# code taking 1s
r.wait() # blocks for 4s
# code taking 7s
r.wait() # doesn't block
# code taking 1s
r.wait() # blocks for 4s
"""
self.__dict__.update(locals())
self.rate = float(seconds) / times
self.sleep_func = sleep_func
def __enter__(self):
self._update_ref()
return self
def __exit__(self, etype, evalue, traceback):
if self.exit_wait:
self.wait()
def _update_ref(self):
self._ref = _monotonic() + self.rate
def wait(self):
"""Blocks until the rate is met"""
now = _monotonic()
if now < self._ref:
delay = max(0, self._ref - now)
self.sleep_func(delay)
self._update_ref()