| 
									
										
										
										
											2020-03-08 14:17:39 +00:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class RateLimiter | 
					
						
							|  |  |  |   include Redisable | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   FAMILIES = { | 
					
						
							|  |  |  |     follows: { | 
					
						
							|  |  |  |       limit: 400, | 
					
						
							|  |  |  |       period: 24.hours.freeze, | 
					
						
							|  |  |  |     }.freeze, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     statuses: { | 
					
						
							|  |  |  |       limit: 300, | 
					
						
							|  |  |  |       period: 3.hours.freeze, | 
					
						
							|  |  |  |     }.freeze, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 12:40:08 +00:00
										 |  |  |     reports: { | 
					
						
							|  |  |  |       limit: 400, | 
					
						
							|  |  |  |       period: 24.hours.freeze, | 
					
						
							| 
									
										
										
										
											2020-03-08 14:17:39 +00:00
										 |  |  |     }.freeze, | 
					
						
							|  |  |  |   }.freeze | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def initialize(by, options = {}) | 
					
						
							|  |  |  |     @by     = by | 
					
						
							|  |  |  |     @family = options[:family] | 
					
						
							|  |  |  |     @limit  = FAMILIES[@family][:limit] | 
					
						
							|  |  |  |     @period = FAMILIES[@family][:period].to_i | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def record! | 
					
						
							|  |  |  |     count = redis.get(key) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if count.nil? | 
					
						
							|  |  |  |       redis.set(key, 0) | 
					
						
							|  |  |  |       redis.expire(key, (@period - (last_epoch_time % @period) + 1).to_i) | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     raise Mastodon::RateLimitExceededError if count.present? && count.to_i >= @limit | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     redis.incr(key) | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def rollback! | 
					
						
							|  |  |  |     redis.decr(key) | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def to_headers(now = Time.now.utc) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       'X-RateLimit-Limit' => @limit.to_s, | 
					
						
							|  |  |  |       'X-RateLimit-Remaining' => (@limit - (redis.get(key) || 0).to_i).to_s, | 
					
						
							|  |  |  |       'X-RateLimit-Reset' => (now + (@period - now.to_i % @period)).iso8601(6), | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   private | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def key | 
					
						
							|  |  |  |     @key ||= "rate_limit:#{@by.id}:#{@family}:#{(last_epoch_time / @period).to_i}" | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def last_epoch_time | 
					
						
							|  |  |  |     @last_epoch_time ||= Time.now.to_i | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end |