Class: Rack::Attack::Throttle

Inherits:
Object
  • Object
show all
Defined in:
lib/rack/attack/throttle.rb

Constant Summary collapse

MANDATORY_OPTIONS =
[:limit, :period].freeze

Instance Attribute Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(name, options, block) ⇒ Throttle

Returns a new instance of Throttle.



7
8
9
10
11
12
13
14
15
# File 'lib/rack/attack/throttle.rb', line 7

def initialize(name, options, block)
  @name, @block = name, block
  MANDATORY_OPTIONS.each do |opt|
    raise ArgumentError.new("Must pass #{opt.inspect} option") unless options[opt]
  end
  @limit  = options[:limit]
  @period = options[:period].respond_to?(:call) ? options[:period] : options[:period].to_i
  @type   = options.fetch(:type, :throttle)
end

Instance Attribute Details

#blockObject (readonly)

Returns the value of attribute block.



6
7
8
# File 'lib/rack/attack/throttle.rb', line 6

def block
  @block
end

#limitObject (readonly)

Returns the value of attribute limit.



6
7
8
# File 'lib/rack/attack/throttle.rb', line 6

def limit
  @limit
end

#nameObject (readonly)

Returns the value of attribute name.



6
7
8
# File 'lib/rack/attack/throttle.rb', line 6

def name
  @name
end

#periodObject (readonly)

Returns the value of attribute period.



6
7
8
# File 'lib/rack/attack/throttle.rb', line 6

def period
  @period
end

#typeObject (readonly)

Returns the value of attribute type.



6
7
8
# File 'lib/rack/attack/throttle.rb', line 6

def type
  @type
end

Instance Method Details

#cacheObject



17
18
19
# File 'lib/rack/attack/throttle.rb', line 17

def cache
  Rack::Attack.cache
end

#matched_by?(request) ⇒ Boolean

Returns:

  • (Boolean)


21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
# File 'lib/rack/attack/throttle.rb', line 21

def matched_by?(request)
  discriminator = block.call(request)
  return false unless discriminator

  current_period = period.respond_to?(:call) ? period.call(request) : period
  current_limit  = limit.respond_to?(:call) ? limit.call(request) : limit
  key            = "#{name}:#{discriminator}"
  count          = cache.count(key, current_period)
  epoch_time     = cache.last_epoch_time

  data = {
    :count => count,
    :period => current_period,
    :limit => current_limit,
    :epoch_time => epoch_time
  }

  (request.env['rack.attack.throttle_data'] ||= {})[name] = data

  (count > current_limit).tap do |throttled|
    if throttled
      request.env['rack.attack.matched']             = name
      request.env['rack.attack.match_discriminator'] = discriminator
      request.env['rack.attack.match_type']          = type
      request.env['rack.attack.match_data']          = data
      Rack::Attack.instrument(request)
    end
  end
end