Class: SimpleThrottle
- Inherits:
-
Object
- Object
- SimpleThrottle
- Defined in:
- lib/simple_throttle.rb
Overview
Create a simple throttle that can be used to limit the number of request for a resouce per time period. These objects are thread safe.
Constant Summary collapse
- LUA_SCRIPT =
Server side Lua script that maintains the throttle in redis. The throttle is stored as a list of timestamps in milliseconds. When the script is invoked it will scan the oldest entries removing any that should be expired from the list. If the list is below the specified limit then the current entry will be added. The list is marked to expire with the oldest entry so there’s no need to cleanup the lists.
<<~LUA local list_key = KEYS[1] local limit = tonumber(ARGV[1]) local ttl = tonumber(ARGV[2]) local now = ARGV[3] local pause_to_recover = tonumber(ARGV[4]) local amount = tonumber(ARGV[5]) local size = redis.call('llen', list_key) if size >= limit then local expired = tonumber(now) - ttl while size > 0 do local t = redis.call('lpop', list_key) if tonumber(t) > expired then redis.call('lpush', list_key, t) break end size = size - 1 end end if pause_to_recover > 0 then limit = limit + 1 end if size + amount > limit then amount = (limit - size) + 1 end if size < limit then for i = 1, amount do redis.call('rpush', list_key, now) end redis.call('pexpire', list_key, ttl) end return size + amount LUA
Instance Attribute Summary collapse
-
#limit ⇒ Object
readonly
Returns the value of attribute limit.
-
#name ⇒ Object
readonly
Returns the value of attribute name.
-
#ttl ⇒ Object
readonly
Returns the value of attribute ttl.
Class Method Summary collapse
-
.[](name) ⇒ SimpleThrottle
Returns a globally defined throttle with the specfied name.
-
.add(name, ttl:, limit:, pause_to_recover: false, redis: nil) ⇒ void
Add a global throttle that can be referenced later with the [] method.
-
.redis ⇒ Redis
Return the Redis instance where the throttles are stored.
-
.set_redis(client = nil, &block) ⇒ void
Set the Redis instance to use for maintaining the throttle.
Instance Method Summary collapse
-
#allowed! ⇒ Boolean
Returns true if the limit for the throttle has not been reached yet.
-
#increment!(amount = 1) ⇒ Integer
Increment the throttle by the specified and return the current size.
-
#initialize(name, ttl:, limit:, pause_to_recover: false, redis: nil) ⇒ SimpleThrottle
constructor
Create a new throttle.
-
#peek ⇒ Integer
Peek at the current number for throttled calls being tracked.
-
#reset! ⇒ void
Reset a throttle back to zero.
-
#wait_time ⇒ Float
Returns when the next resource call should be allowed.
Constructor Details
#initialize(name, ttl:, limit:, pause_to_recover: false, redis: nil) ⇒ SimpleThrottle
Create a new throttle.
141 142 143 144 145 146 147 148 |
# File 'lib/simple_throttle.rb', line 141 def initialize(name, ttl:, limit:, pause_to_recover: false, redis: nil) @name = name.to_s @name = name.dup.freeze unless name.frozen? @limit = limit.to_i @ttl = ttl.to_f @pause_to_recover = !!pause_to_recover @redis = redis end |
Instance Attribute Details
#limit ⇒ Object (readonly)
Returns the value of attribute limit.
130 131 132 |
# File 'lib/simple_throttle.rb', line 130 def limit @limit end |
#name ⇒ Object (readonly)
Returns the value of attribute name.
130 131 132 |
# File 'lib/simple_throttle.rb', line 130 def name @name end |
#ttl ⇒ Object (readonly)
Returns the value of attribute ttl.
130 131 132 |
# File 'lib/simple_throttle.rb', line 130 def ttl @ttl end |
Class Method Details
.[](name) ⇒ SimpleThrottle
Returns a globally defined throttle with the specfied name.
78 79 80 81 82 |
# File 'lib/simple_throttle.rb', line 78 def [](name) if defined?(@throttles) && @throttles @throttles[name.to_s] end end |
.add(name, ttl:, limit:, pause_to_recover: false, redis: nil) ⇒ void
This method returns an undefined value.
Add a global throttle that can be referenced later with the [] method. This can be used to configure global throttles that you want to setup once and then use in multiple places.
67 68 69 70 71 72 |
# File 'lib/simple_throttle.rb', line 67 def add(name, ttl:, limit:, pause_to_recover: false, redis: nil) @lock.synchronize do @throttles ||= {} @throttles[name.to_s] = new(name, limit: limit, ttl: ttl, pause_to_recover: pause_to_recover, redis: redis) end end |
.redis ⇒ Redis
Return the Redis instance where the throttles are stored.
100 101 102 103 104 105 106 107 |
# File 'lib/simple_throttle.rb', line 100 def redis @redis_client ||= Redis.new if @redis_client.is_a?(Proc) @redis_client.call else @redis_client end end |
.set_redis(client = nil, &block) ⇒ void
This method returns an undefined value.
Set the Redis instance to use for maintaining the throttle. This can either be set with a hard coded value or by the value yielded by a block. If the block form is used it will be invoked at runtime to get the instance. Use this method if your Redis instance isn’t constant (for example if you’re in a forking environment and re-initialize connections on fork)
93 94 95 |
# File 'lib/simple_throttle.rb', line 93 def set_redis(client = nil, &block) @redis_client = (client || block) # rubocop:disable Style/RedundantParentheses end |
Instance Method Details
#allowed! ⇒ Boolean
Returns true if the limit for the throttle has not been reached yet. This method will also track the throttled resource as having been invoked on each call.
154 155 156 157 |
# File 'lib/simple_throttle.rb', line 154 def allowed! size = increment! size <= limit end |
#increment!(amount = 1) ⇒ Integer
Increment the throttle by the specified and return the current size. Because how the throttle is implemented in Redis, the return value will always max out at the throttle limit + 1 or, if the pause to recover option is set, limit + 2.
165 166 167 168 169 170 171 172 173 174 175 |
# File 'lib/simple_throttle.rb', line 165 def increment!(amount = 1) pause_to_recover_arg = (@pause_to_recover ? 1 : 0) time_ms = (Time.now.to_f * 1000).round ttl_ms = (ttl * 1000).ceil self.class.send( :execute_lua_script, redis: redis_client, keys: [redis_key], args: [limit, ttl_ms, time_ms, pause_to_recover_arg, amount] ) end |
#peek ⇒ Integer
Peek at the current number for throttled calls being tracked.
187 188 189 190 191 |
# File 'lib/simple_throttle.rb', line 187 def peek = redis_client.lrange(redis_key, 0, -1).collect(&:to_i) = ((Time.now.to_f - ttl) * 1000).ceil .count { |t| t > } end |
#reset! ⇒ void
This method returns an undefined value.
Reset a throttle back to zero.
180 181 182 |
# File 'lib/simple_throttle.rb', line 180 def reset! redis_client.del(redis_key) end |
#wait_time ⇒ Float
Returns when the next resource call should be allowed. Note that this doesn’t guarantee that calling allow! will return true if the wait time is zero since other processes or threads can claim the resource.
198 199 200 201 202 203 204 205 206 207 |
# File 'lib/simple_throttle.rb', line 198 def wait_time if peek < limit 0.0 else first = redis_client.lindex(redis_key, 0).to_f / 1000.0 delta = Time.now.to_f - first delta = 0.0 if delta < 0 ttl - delta end end |