From dc9266fbeacd24446b52e4dad328c8286be40b31 Mon Sep 17 00:00:00 2001 From: Michael Kozono Date: Fri, 15 Sep 2017 10:31:32 -0700 Subject: Add request throttles --- config/application.rb | 2 +- config/initializers/rack_attack_global.rb | 73 +++++++++++++++++++++++++++++++ 2 files changed, 74 insertions(+), 1 deletion(-) create mode 100644 config/initializers/rack_attack_global.rb (limited to 'config') diff --git a/config/application.rb b/config/application.rb index 5100ec5d2b7..6436f887d14 100644 --- a/config/application.rb +++ b/config/application.rb @@ -113,7 +113,7 @@ module Gitlab config.action_view.sanitized_allowed_protocols = %w(smb) - config.middleware.insert_before Warden::Manager, Rack::Attack + config.middleware.insert_after Warden::Manager, Rack::Attack # Allow access to GitLab API from other domains config.middleware.insert_before Warden::Manager, Rack::Cors do diff --git a/config/initializers/rack_attack_global.rb b/config/initializers/rack_attack_global.rb new file mode 100644 index 00000000000..0b51fadbd02 --- /dev/null +++ b/config/initializers/rack_attack_global.rb @@ -0,0 +1,73 @@ +class Rack::Attack + def self.settings + Gitlab::CurrentSettings.current_application_settings + end + + def self.throttle_unauthenticated_options + limit_proc = proc { |req| settings.throttle_unauthenticated_requests_per_period } + period_proc = proc { |req| settings.throttle_unauthenticated_period_in_seconds.seconds } + { limit: limit_proc, period: period_proc } + end + + def self.throttle_authenticated_api_options + limit_proc = proc { |req| settings.throttle_authenticated_api_requests_per_period } + period_proc = proc { |req| settings.throttle_authenticated_api_period_in_seconds.seconds } + { limit: limit_proc, period: period_proc } + end + + def self.throttle_authenticated_web_options + limit_proc = proc { |req| settings.throttle_authenticated_web_requests_per_period } + period_proc = proc { |req| settings.throttle_authenticated_web_period_in_seconds.seconds } + { limit: limit_proc, period: period_proc } + end + + def self.define_throttles + throttle('throttle_unauthenticated', throttle_unauthenticated_options) do |req| + settings.throttle_unauthenticated_enabled && + req.unauthenticated? && + req.ip + end + + throttle('throttle_authenticated_api', throttle_authenticated_api_options) do |req| + settings.throttle_authenticated_api_enabled && + req.api_request? && + req.authenticated_user_id + end + + throttle('throttle_authenticated_web', throttle_authenticated_web_options) do |req| + settings.throttle_authenticated_web_enabled && + req.web_request? && + req.authenticated_user_id + end + end + + define_throttles unless Rails.env.test? + + class Request + def unauthenticated? + !authenticated_user_id + end + + def authenticated_user_id + session_user_id || sessionless_user_id + end + + def api_request? + path.start_with?('/api') + end + + def web_request? + !api_request? + end + + private + + def session_user_id + Gitlab::Auth.find_session_user(self)&.id + end + + def sessionless_user_id + Gitlab::Auth.find_sessionless_user(self)&.id + end + end +end -- cgit v1.2.3