Commit cc556e6f authored by Luke Duncalfe's avatar Luke Duncalfe Committed by Kerri Miller

Add read state access to ApplicationRateLimiter

parent 2ba756c4
...@@ -64,45 +64,47 @@ module Gitlab ...@@ -64,45 +64,47 @@ module Gitlab
# be throttled. # be throttled.
# #
# @param key [Symbol] Key attribute registered in `.rate_limits` # @param key [Symbol] Key attribute registered in `.rate_limits`
# @option scope [Array<ActiveRecord>] Array of ActiveRecord models to scope throttling to a specific request (e.g. per user per project) # @param scope [Array<ActiveRecord>] Array of ActiveRecord models to scope throttling to a specific request (e.g. per user per project)
# @option threshold [Integer] Optional threshold value to override default one registered in `.rate_limits` # @param threshold [Integer] Optional threshold value to override default one registered in `.rate_limits`
# @option users_allowlist [Array<String>] Optional list of usernames to exclude from the limit. This param will only be functional if Scope includes a current user. # @param users_allowlist [Array<String>] Optional list of usernames to exclude from the limit. This param will only be functional if Scope includes a current user.
# @param peek [Boolean] Optional. When true the key will not be incremented but the current throttled state will be returned.
# #
# @return [Boolean] Whether or not a request should be throttled # @return [Boolean] Whether or not a request should be throttled
def throttled?(key, **options) def throttled?(key, scope:, threshold: nil, users_allowlist: nil, peek: false)
raise InvalidKeyError unless rate_limits[key] raise InvalidKeyError unless rate_limits[key]
return if scoped_user_in_allowlist?(options) return false if scoped_user_in_allowlist?(scope, users_allowlist)
threshold_value = options[:threshold] || threshold(key) threshold_value = threshold || threshold(key)
threshold_value > 0 &&
increment(key, options[:scope]) > threshold_value
end
# Increments a cache key that is based on the current time and interval. return false if threshold_value == 0
# So that when time passes to the next interval, the key changes and the count starts again from 0.
#
# Based on https://github.com/rack/rack-attack/blob/886ba3a18d13c6484cd511a4dc9b76c0d14e5e96/lib/rack/attack/cache.rb#L63-L68
#
# @param key [Symbol] Key attribute registered in `.rate_limits`
# @option scope [Array<ActiveRecord>] Array of ActiveRecord models to scope throttling to a specific request (e.g. per user per project)
#
# @return [Integer] incremented value
def increment(key, scope)
interval_value = interval(key)
interval_value = interval(key)
# `period_key` is based on the current time and interval so when time passes to the next interval
# the key changes and the rate limit count starts again from 0.
# Based on https://github.com/rack/rack-attack/blob/886ba3a18d13c6484cd511a4dc9b76c0d14e5e96/lib/rack/attack/cache.rb#L63-L68
period_key, time_elapsed_in_period = Time.now.to_i.divmod(interval_value) period_key, time_elapsed_in_period = Time.now.to_i.divmod(interval_value)
cache_key = cache_key(key, scope, period_key)
cache_key = "#{action_key(key, scope)}:#{period_key}" value = if peek
# We add a 1 second buffer to avoid timing issues when we're at the end of a period read(cache_key)
expiry = interval_value - time_elapsed_in_period + 1 else
increment(cache_key, interval_value, time_elapsed_in_period)
end
::Gitlab::Redis::RateLimiting.with do |redis| value > threshold_value
redis.pipelined do end
redis.incr(cache_key)
redis.expire(cache_key, expiry) # Returns the current rate limited state without incrementing the count.
end.first #
end # @param key [Symbol] Key attribute registered in `.rate_limits`
# @param scope [Array<ActiveRecord>] Array of ActiveRecord models to scope throttling to a specific request (e.g. per user per project)
# @param threshold [Integer] Optional threshold value to override default one registered in `.rate_limits`
# @param users_allowlist [Array<String>] Optional list of usernames to exclude from the limit. This param will only be functional if Scope includes a current user.
#
# @return [Boolean] Whether or not a request is currently throttled
def peek(key, scope:, threshold: nil, users_allowlist: nil)
throttled?(key, peek: true, scope: scope, threshold: threshold, users_allowlist: users_allowlist)
end end
# Logs request using provided logger # Logs request using provided logger
...@@ -150,7 +152,28 @@ module Gitlab ...@@ -150,7 +152,28 @@ module Gitlab
action[setting] if action action[setting] if action
end end
def action_key(key, scope) # Increments the rate limit count and returns the new count value.
def increment(cache_key, interval_value, time_elapsed_in_period)
# We add a 1 second buffer to avoid timing issues when we're at the end of a period
expiry = interval_value - time_elapsed_in_period + 1
::Gitlab::Redis::RateLimiting.with do |redis|
redis.pipelined do
redis.incr(cache_key)
redis.expire(cache_key, expiry)
end.first
end
end
# Returns the rate limit count.
# Will be 0 if there is no data in the cache.
def read(cache_key)
::Gitlab::Redis::RateLimiting.with do |redis|
redis.get(cache_key).to_i
end
end
def cache_key(key, scope, period_key)
composed_key = [key, scope].flatten.compact composed_key = [key, scope].flatten.compact
serialized = composed_key.map do |obj| serialized = composed_key.map do |obj|
...@@ -161,20 +184,20 @@ module Gitlab ...@@ -161,20 +184,20 @@ module Gitlab
end end
end.join(":") end.join(":")
"application_rate_limiter:#{serialized}" "application_rate_limiter:#{serialized}:#{period_key}"
end end
def application_settings def application_settings
Gitlab::CurrentSettings.current_application_settings Gitlab::CurrentSettings.current_application_settings
end end
def scoped_user_in_allowlist?(options) def scoped_user_in_allowlist?(scope, users_allowlist)
return unless options[:users_allowlist].present? return unless users_allowlist.present?
scoped_user = [options[:scope]].flatten.find { |s| s.is_a?(User) } scoped_user = [scope].flatten.find { |s| s.is_a?(User) }
return unless scoped_user return unless scoped_user
scoped_user.username.downcase.in?(options[:users_allowlist]) scoped_user.username.downcase.in?(users_allowlist)
end end
end end
......
...@@ -2,37 +2,37 @@ ...@@ -2,37 +2,37 @@
require 'spec_helper' require 'spec_helper'
RSpec.describe Gitlab::ApplicationRateLimiter do RSpec.describe Gitlab::ApplicationRateLimiter, :clean_gitlab_redis_rate_limiting do
let_it_be(:user) { create(:user) } let_it_be(:user) { create(:user) }
let_it_be(:project) { create(:project) } let_it_be(:project) { create(:project) }
subject { described_class } let(:rate_limits) do
{
describe '.throttled?', :clean_gitlab_redis_rate_limiting do test_action: {
let(:rate_limits) do threshold: 1,
{ interval: 2.minutes
test_action: { },
threshold: 1, another_action: {
interval: 2.minutes threshold: 2,
}, interval: 3.minutes
another_action: {
threshold: 2,
interval: 3.minutes
}
} }
end }
end
before do subject { described_class }
allow(described_class).to receive(:rate_limits).and_return(rate_limits)
end before do
allow(described_class).to receive(:rate_limits).and_return(rate_limits)
end
describe '.throttled?' do
context 'when the key is invalid' do context 'when the key is invalid' do
context 'is provided as a Symbol' do context 'is provided as a Symbol' do
context 'but is not defined in the rate_limits Hash' do context 'but is not defined in the rate_limits Hash' do
it 'raises an InvalidKeyError exception' do it 'raises an InvalidKeyError exception' do
key = :key_not_in_rate_limits_hash key = :key_not_in_rate_limits_hash
expect { subject.throttled?(key) }.to raise_error(Gitlab::ApplicationRateLimiter::InvalidKeyError) expect { subject.throttled?(key, scope: [user]) }.to raise_error(Gitlab::ApplicationRateLimiter::InvalidKeyError)
end end
end end
end end
...@@ -42,7 +42,7 @@ RSpec.describe Gitlab::ApplicationRateLimiter do ...@@ -42,7 +42,7 @@ RSpec.describe Gitlab::ApplicationRateLimiter do
it 'raises an InvalidKeyError exception' do it 'raises an InvalidKeyError exception' do
key = rate_limits.keys[0].to_s key = rate_limits.keys[0].to_s
expect { subject.throttled?(key) }.to raise_error(Gitlab::ApplicationRateLimiter::InvalidKeyError) expect { subject.throttled?(key, scope: [user]) }.to raise_error(Gitlab::ApplicationRateLimiter::InvalidKeyError)
end end
end end
...@@ -50,7 +50,7 @@ RSpec.describe Gitlab::ApplicationRateLimiter do ...@@ -50,7 +50,7 @@ RSpec.describe Gitlab::ApplicationRateLimiter do
it 'raises an InvalidKeyError exception' do it 'raises an InvalidKeyError exception' do
key = 'key_not_in_rate_limits_hash' key = 'key_not_in_rate_limits_hash'
expect { subject.throttled?(key) }.to raise_error(Gitlab::ApplicationRateLimiter::InvalidKeyError) expect { subject.throttled?(key, scope: [user]) }.to raise_error(Gitlab::ApplicationRateLimiter::InvalidKeyError)
end end
end end
end end
...@@ -89,6 +89,17 @@ RSpec.describe Gitlab::ApplicationRateLimiter do ...@@ -89,6 +89,17 @@ RSpec.describe Gitlab::ApplicationRateLimiter do
expect(subject.throttled?(:another_action, scope: scope)).to eq(true) expect(subject.throttled?(:another_action, scope: scope)).to eq(true)
end end
end end
it 'allows peeking at the current state without changing its value' do
travel_to(start_time) do
expect(subject.throttled?(:test_action, scope: scope)).to eq(false)
2.times do
expect(subject.throttled?(:test_action, scope: scope, peek: true)).to eq(false)
end
expect(subject.throttled?(:test_action, scope: scope)).to eq(true)
expect(subject.throttled?(:test_action, scope: scope, peek: true)).to eq(true)
end
end
end end
context 'when using ActiveRecord models as scope' do context 'when using ActiveRecord models as scope' do
...@@ -104,6 +115,20 @@ RSpec.describe Gitlab::ApplicationRateLimiter do ...@@ -104,6 +115,20 @@ RSpec.describe Gitlab::ApplicationRateLimiter do
end end
end end
describe '.peek' do
it 'peeks at the current state without changing its value' do
freeze_time do
expect(subject.peek(:test_action, scope: [user])).to eq(false)
expect(subject.throttled?(:test_action, scope: [user])).to eq(false)
2.times do
expect(subject.peek(:test_action, scope: [user])).to eq(false)
end
expect(subject.throttled?(:test_action, scope: [user])).to eq(true)
expect(subject.peek(:test_action, scope: [user])).to eq(true)
end
end
end
describe '.log_request' do describe '.log_request' do
let(:file_path) { 'master/README.md' } let(:file_path) { 'master/README.md' }
let(:type) { :raw_blob_request_limit } let(:type) { :raw_blob_request_limit }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment