OnlineJudge/utils/throttling.py

73 lines
2.3 KiB
Python
Raw Permalink Normal View History

2017-05-08 17:29:01 +08:00
import time
class TokenBucket:
2017-12-23 22:27:53 +08:00
"""
注意对于单个key的操作不是线程安全的
"""
def __init__(self, key, capacity, fill_rate, default_capacity, redis_conn):
"""
:param capacity: 最大容量
:param fill_rate: 填充速度/每秒
:param default_capacity: 初始容量
:param redis_conn: redis connection
"""
self._key = key
self._capacity = capacity
self._fill_rate = fill_rate
self._default_capacity = default_capacity
self._redis_conn = redis_conn
2017-05-08 17:29:01 +08:00
2017-12-23 22:27:53 +08:00
self._last_capacity_key = "last_capacity"
self._last_timestamp_key = "last_timestamp"
2017-05-08 17:29:01 +08:00
2017-12-23 22:27:53 +08:00
def _init_key(self):
self._last_capacity = self._default_capacity
now = time.time()
self._last_timestamp = now
return self._default_capacity, now
2017-05-08 17:29:01 +08:00
@property
2017-12-23 22:27:53 +08:00
def _last_capacity(self):
last_capacity = self._redis_conn.hget(self._key, self._last_capacity_key)
if last_capacity is None:
return self._init_key()[0]
else:
return float(last_capacity)
2017-05-08 17:29:01 +08:00
2017-12-23 22:27:53 +08:00
@_last_capacity.setter
def _last_capacity(self, value):
self._redis_conn.hset(self._key, self._last_capacity_key, value)
2017-05-08 17:29:01 +08:00
@property
2017-12-23 22:27:53 +08:00
def _last_timestamp(self):
return float(self._redis_conn.hget(self._key, self._last_timestamp_key))
2017-05-08 17:29:01 +08:00
2017-12-23 22:27:53 +08:00
@_last_timestamp.setter
def _last_timestamp(self, value):
self._redis_conn.hset(self._key, self._last_timestamp_key, value)
2017-05-08 17:29:01 +08:00
2017-12-23 22:27:53 +08:00
def _try_to_fill(self, now):
delta = self._fill_rate * (now - self._last_timestamp)
return min(self._last_capacity + delta, self._capacity)
2017-05-08 17:29:01 +08:00
2017-12-23 22:27:53 +08:00
def consume(self, num=1):
"""
消耗 num token返回是否成功
:param num:
:return: result: bool, wait_time: float
"""
# print("capacity ", self.fill(time.time()))
if self._last_capacity >= num:
self._last_capacity -= num
return True, 0
else:
now = time.time()
cur_num = self._try_to_fill(now)
if cur_num >= num:
self._last_capacity = cur_num - num
self._last_timestamp = now
return True, 0
else:
return False, (num - cur_num) / self._fill_rate