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
|