gab-social/config/initializers/rack_attack.rb

128 lines
4.2 KiB
Ruby
Raw Normal View History

2019-07-02 08:10:25 +01:00
# frozen_string_literal: true
require 'doorkeeper/grape/authorization_decorator'
class Rack::Attack
class Request
def authenticated_token
return @token if defined?(@token)
@token = Doorkeeper::OAuth::Token.authenticate(
Doorkeeper::Grape::AuthorizationDecorator.new(self),
*Doorkeeper.configuration.access_token_methods
)
end
def remote_ip
2019-07-04 09:05:34 +01:00
# @remote_ip ||= (@env["action_dispatch.remote_ip"] || ip).to_s
# @env['HTTP_CF_CONNECTING_IP'] ? @env['HTTP_CF_CONNECTING_IP'] : ip
@remote_ip ||= (@env["HTTP_CF_CONNECTING_IP"] || ip).to_s
2019-07-02 08:10:25 +01:00
end
def authenticated_user_id
authenticated_token&.resource_owner_id
end
def unauthenticated?
!authenticated_user_id
end
def api_request?
path.start_with?('/api')
end
def web_request?
!api_request?
end
def paging_request?
params['page'].present? || params['min_id'].present? || params['max_id'].present? || params['since_id'].present?
end
end
PROTECTED_PATHS = %w(
/auth/sign_in
/auth
/auth/password
2020-05-01 06:50:27 +01:00
/auth/confirmation
2019-07-02 08:10:25 +01:00
).freeze
PROTECTED_PATHS_REGEX = Regexp.union(PROTECTED_PATHS.map { |path| /\A#{Regexp.escape(path)}/ })
# Always allow requests from localhost
# (blocklist & throttles are skipped)
Rack::Attack.safelist('allow from localhost') do |req|
# Requests are allowed if the return value is truthy
req.remote_ip == '127.0.0.1' || req.remote_ip == '::1'
end
2019-07-04 09:05:34 +01:00
throttle('throttle_authenticated_api', limit: 300, period: 5.minutes) do |req|
2019-07-02 08:10:25 +01:00
req.authenticated_user_id if req.api_request?
end
2019-07-04 09:05:34 +01:00
throttle('throttle_unauthenticated_api', limit: 300, period: 5.minutes) do |req|
2019-07-02 08:10:25 +01:00
req.remote_ip if req.api_request? && req.unauthenticated?
end
2019-07-04 09:05:34 +01:00
throttle('throttle_api_media', limit: 30, period: 30.minutes) do |req|
2019-07-02 08:10:25 +01:00
req.authenticated_user_id if req.post? && req.path.start_with?('/api/v1/media')
end
2019-07-04 09:05:34 +01:00
throttle('throttle_media_proxy', limit: 30, period: 30.minutes) do |req|
2019-07-02 08:10:25 +01:00
req.remote_ip if req.path.start_with?('/media_proxy')
end
2019-07-04 09:05:34 +01:00
throttle('throttle_api_sign_up', limit: 5, period: 30.minutes) do |req|
2019-07-02 08:10:25 +01:00
req.remote_ip if req.post? && req.path == '/api/v1/accounts'
end
# Throttle paging, as it is mainly used for public pages and AP collections
2019-07-04 09:05:34 +01:00
throttle('throttle_authenticated_paging', limit: 300, period: 15.minutes) do |req|
2019-07-02 08:10:25 +01:00
req.authenticated_user_id if req.paging_request?
end
2019-07-04 09:05:34 +01:00
throttle('throttle_unauthenticated_paging', limit: 300, period: 15.minutes) do |req|
2019-07-02 08:10:25 +01:00
req.remote_ip if req.paging_request? && req.unauthenticated?
end
API_DELETE_REBLOG_REGEX = /\A\/api\/v1\/statuses\/[\d]+\/unreblog/.freeze
API_DELETE_STATUS_REGEX = /\A\/api\/v1\/statuses\/[\d]+/.freeze
2020-12-16 00:31:30 +00:00
API_POST_CHAT_MESSAGE_REGEX = /\A\/api\/v1\/chat_messages/.freeze
API_POST_FOLLOW_REGEX = /\A\/api\/v1\/accounts\/[\d]+\/follow/.freeze
API_POST_GROUP_PASSWORD_CHECK_REGEX = /\A\/api\/v1\/groups\/[\d]+\/password/.freeze
2019-07-02 08:10:25 +01:00
2019-07-04 09:05:34 +01:00
throttle('throttle_api_delete', limit: 30, period: 30.minutes) do |req|
2019-07-02 08:10:25 +01:00
req.authenticated_user_id if (req.post? && req.path =~ API_DELETE_REBLOG_REGEX) || (req.delete? && req.path =~ API_DELETE_STATUS_REGEX)
end
2020-12-16 00:31:30 +00:00
throttle('throttle_api_chat_message', limit: 1000, period: 1.day) do |req|
req.authenticated_user_id if req.post? && req.path =~ API_POST_CHAT_MESSAGE_REGEX
end
throttle('throttle_api_follow', limit: 200, period: 1.day) do |req|
req.authenticated_user_id if req.post? && req.path =~ API_POST_FOLLOW_REGEX
end
throttle('throttle_group_password_check', limit: 5, period: 1.minute) do |req|
req.authenticated_user_id if req.post? && req.path =~ API_POST_GROUP_PASSWORD_CHECK_REGEX
end
2020-12-16 00:31:30 +00:00
throttle('protected_paths', limit: 10, period: 5.minutes) do |req|
2019-07-02 08:10:25 +01:00
req.remote_ip if req.post? && req.path =~ PROTECTED_PATHS_REGEX
end
self.throttled_response = lambda do |env|
now = Time.now.utc
match_data = env['rack.attack.match_data']
headers = {
'Content-Type' => 'application/json',
'X-RateLimit-Limit' => match_data[:limit].to_s,
'X-RateLimit-Remaining' => '0',
'X-RateLimit-Reset' => (now + (match_data[:period] - now.to_i % match_data[:period])).iso8601(6),
}
[429, headers, [{ error: I18n.t('errors.429') }.to_json]]
end
end