12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364 |
- # 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,
- reports: {
- limit: 400,
- period: 24.hours.freeze,
- }.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
|