summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--app/controllers/account_follow_controller.rb2
-rw-r--r--app/controllers/api/base_controller.rb4
-rw-r--r--app/controllers/api/v1/accounts_controller.rb4
-rw-r--r--app/controllers/api/v1/statuses/reblogs_controller.rb3
-rw-r--r--app/controllers/api/v1/statuses_controller.rb5
-rw-r--r--app/controllers/application_controller.rb5
-rw-r--r--app/controllers/authorize_interactions_controller.rb2
-rw-r--r--app/controllers/concerns/rate_limit_headers.rb16
-rw-r--r--app/lib/exceptions.rb1
-rw-r--r--app/lib/rate_limiter.rb64
-rw-r--r--app/models/concerns/account_interactions.rb16
-rw-r--r--app/models/concerns/rate_limitable.rb36
-rw-r--r--app/models/follow.rb3
-rw-r--r--app/models/follow_request.rb3
-rw-r--r--app/models/status.rb3
-rw-r--r--app/services/follow_service.rb78
-rw-r--r--app/services/post_status_service.rb11
-rw-r--r--app/services/reblog_service.rb16
-rw-r--r--app/views/errors/429.html.haml5
-rw-r--r--config/initializers/rack_attack.rb1
-rw-r--r--config/locales/en.yml2
-rw-r--r--spec/controllers/account_follow_controller_spec.rb2
-rw-r--r--spec/controllers/api/v1/statuses_controller_spec.rb46
23 files changed, 275 insertions, 53 deletions
diff --git a/app/controllers/account_follow_controller.rb b/app/controllers/account_follow_controller.rb
index 185a355f8f3..33394074db4 100644
--- a/app/controllers/account_follow_controller.rb
+++ b/app/controllers/account_follow_controller.rb
@@ -6,7 +6,7 @@ class AccountFollowController < ApplicationController
before_action :authenticate_user!
def create
- FollowService.new.call(current_user.account, @account.acct)
+ FollowService.new.call(current_user.account, @account, with_rate_limit: true)
redirect_to account_path(@account)
end
end
diff --git a/app/controllers/api/base_controller.rb b/app/controllers/api/base_controller.rb
index 68bf425f4d0..153ade253d6 100644
--- a/app/controllers/api/base_controller.rb
+++ b/app/controllers/api/base_controller.rb
@@ -44,6 +44,10 @@ class Api::BaseController < ApplicationController
render json: { error: 'There was a temporary problem serving your request, please try again' }, status: 503
end
+ rescue_from Mastodon::RateLimitExceededError do
+ render json: { error: I18n.t('errors.429') }, status: 429
+ end
+
rescue_from ActionController::ParameterMissing do |e|
render json: { error: e.to_s }, status: 400
end
diff --git a/app/controllers/api/v1/accounts_controller.rb b/app/controllers/api/v1/accounts_controller.rb
index f5b06862b9a..0080faf3307 100644
--- a/app/controllers/api/v1/accounts_controller.rb
+++ b/app/controllers/api/v1/accounts_controller.rb
@@ -14,6 +14,8 @@ class Api::V1::AccountsController < Api::BaseController
skip_before_action :require_authenticated_user!, only: :create
+ override_rate_limit_headers :follow, family: :follows
+
def show
render json: @account, serializer: REST::AccountSerializer
end
@@ -29,7 +31,7 @@ class Api::V1::AccountsController < Api::BaseController
end
def follow
- FollowService.new.call(current_user.account, @account, reblogs: truthy_param?(:reblogs))
+ FollowService.new.call(current_user.account, @account, reblogs: truthy_param?(:reblogs), with_rate_limit: true)
options = @account.locked? || current_user.account.silenced? ? {} : { following_map: { @account.id => { reblogs: truthy_param?(:reblogs) } }, requested_map: { @account.id => false } }
diff --git a/app/controllers/api/v1/statuses/reblogs_controller.rb b/app/controllers/api/v1/statuses/reblogs_controller.rb
index 9abeb0759e3..7fa774a4d72 100644
--- a/app/controllers/api/v1/statuses/reblogs_controller.rb
+++ b/app/controllers/api/v1/statuses/reblogs_controller.rb
@@ -7,8 +7,11 @@ class Api::V1::Statuses::ReblogsController < Api::BaseController
before_action :require_user!
before_action :set_reblog
+ override_rate_limit_headers :create, family: :statuses
+
def create
@status = ReblogService.new.call(current_account, @reblog, reblog_params)
+
render json: @status, serializer: REST::StatusSerializer
end
diff --git a/app/controllers/api/v1/statuses_controller.rb b/app/controllers/api/v1/statuses_controller.rb
index 85521fadf4a..2f55e95fd2a 100644
--- a/app/controllers/api/v1/statuses_controller.rb
+++ b/app/controllers/api/v1/statuses_controller.rb
@@ -8,6 +8,8 @@ class Api::V1::StatusesController < Api::BaseController
before_action :require_user!, except: [:show, :context]
before_action :set_status, only: [:show, :context]
+ override_rate_limit_headers :create, family: :statuses
+
# This API was originally unlimited, pagination cannot be introduced without
# breaking backwards-compatibility. Arbitrarily high number to cover most
# conversations as quasi-unlimited, it would be too much work to render more
@@ -42,7 +44,8 @@ class Api::V1::StatusesController < Api::BaseController
scheduled_at: status_params[:scheduled_at],
application: doorkeeper_token.application,
poll: status_params[:poll],
- idempotency: request.headers['Idempotency-Key'])
+ idempotency: request.headers['Idempotency-Key'],
+ with_rate_limit: true)
render json: @status, serializer: @status.is_a?(ScheduledStatus) ? REST::ScheduledStatusSerializer : REST::StatusSerializer
end
diff --git a/app/controllers/application_controller.rb b/app/controllers/application_controller.rb
index 0cfa2b38606..973db6aca97 100644
--- a/app/controllers/application_controller.rb
+++ b/app/controllers/application_controller.rb
@@ -29,6 +29,7 @@ class ApplicationController < ActionController::Base
rescue_from Mastodon::NotPermittedError, with: :forbidden
rescue_from HTTP::Error, OpenSSL::SSL::SSLError, with: :internal_server_error
rescue_from Mastodon::RaceConditionError, with: :service_unavailable
+ rescue_from Mastodon::RateLimitExceededError, with: :too_many_requests
before_action :store_current_location, except: :raise_not_found, unless: :devise_controller?
before_action :require_functional!, if: :user_signed_in?
@@ -111,6 +112,10 @@ class ApplicationController < ActionController::Base
respond_with_error(503)
end
+ def too_many_requests
+ respond_with_error(429)
+ end
+
def single_user_mode?
@single_user_mode ||= Rails.configuration.x.single_user_mode && Account.where('id > 0').exists?
end
diff --git a/app/controllers/authorize_interactions_controller.rb b/app/controllers/authorize_interactions_controller.rb
index e27366ea363..29c0288d09d 100644
--- a/app/controllers/authorize_interactions_controller.rb
+++ b/app/controllers/authorize_interactions_controller.rb
@@ -20,7 +20,7 @@ class AuthorizeInteractionsController < ApplicationController
end
def create
- if @resource.is_a?(Account) && FollowService.new.call(current_account, @resource)
+ if @resource.is_a?(Account) && FollowService.new.call(current_account, @resource, with_rate_limit: true)
render :success
else
render :error
diff --git a/app/controllers/concerns/rate_limit_headers.rb b/app/controllers/concerns/rate_limit_headers.rb
index b79c558d815..86fe58a71c9 100644
--- a/app/controllers/concerns/rate_limit_headers.rb
+++ b/app/controllers/concerns/rate_limit_headers.rb
@@ -3,6 +3,20 @@
module RateLimitHeaders
extend ActiveSupport::Concern
+ class_methods do
+ def override_rate_limit_headers(method_name, options = {})
+ around_action(only: method_name, if: :current_account) do |_controller, block|
+ begin
+ block.call
+ ensure
+ rate_limiter = RateLimiter.new(current_account, options)
+ rate_limit_headers = rate_limiter.to_headers
+ response.headers.merge!(rate_limit_headers) unless response.headers['X-RateLimit-Remaining'].present? && rate_limit_headers['X-RateLimit-Remaining'].to_i > response.headers['X-RateLimit-Remaining'].to_i
+ end
+ end
+ end
+ end
+
included do
before_action :set_rate_limit_headers, if: :rate_limited_request?
end
@@ -44,7 +58,7 @@ module RateLimitHeaders
end
def api_throttle_data
- most_limited_type, = request.env['rack.attack.throttle_data'].min_by { |_, v| v[:limit] }
+ most_limited_type, = request.env['rack.attack.throttle_data'].min_by { |_, v| v[:limit] - v[:count] }
request.env['rack.attack.throttle_data'][most_limited_type]
end
diff --git a/app/lib/exceptions.rb b/app/lib/exceptions.rb
index 01346bfe5ae..3362576b0bc 100644
--- a/app/lib/exceptions.rb
+++ b/app/lib/exceptions.rb
@@ -8,6 +8,7 @@ module Mastodon
class LengthValidationError < ValidationError; end
class DimensionsValidationError < ValidationError; end
class RaceConditionError < Error; end
+ class RateLimitExceededError < Error; end
class UnexpectedResponseError < Error
def initialize(response = nil)
diff --git a/app/lib/rate_limiter.rb b/app/lib/rate_limiter.rb
new file mode 100644
index 00000000000..68dae9add43
--- /dev/null
+++ b/app/lib/rate_limiter.rb
@@ -0,0 +1,64 @@
+# 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,
+
+ media: {
+ limit: 30,
+ period: 30.minutes.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
diff --git a/app/models/concerns/account_interactions.rb b/app/models/concerns/account_interactions.rb
index 14bcf7bb10f..32fcb539752 100644
--- a/app/models/concerns/account_interactions.rb
+++ b/app/models/concerns/account_interactions.rb
@@ -87,10 +87,10 @@ module AccountInteractions
has_many :announcement_mutes, dependent: :destroy
end
- def follow!(other_account, reblogs: nil, uri: nil)
+ def follow!(other_account, reblogs: nil, uri: nil, rate_limit: false)
reblogs = true if reblogs.nil?
- rel = active_relationships.create_with(show_reblogs: reblogs, uri: uri)
+ rel = active_relationships.create_with(show_reblogs: reblogs, uri: uri, rate_limit: rate_limit)
.find_or_create_by!(target_account: other_account)
rel.update!(show_reblogs: reblogs)
@@ -99,6 +99,18 @@ module AccountInteractions
rel
end
+ def request_follow!(other_account, reblogs: nil, uri: nil, rate_limit: false)
+ reblogs = true if reblogs.nil?
+
+ rel = follow_requests.create_with(show_reblogs: reblogs, uri: uri, rate_limit: rate_limit)
+ .find_or_create_by!(target_account: other_account)
+
+ rel.update!(show_reblogs: reblogs)
+ remove_potential_friendship(other_account)
+
+ rel
+ end
+
def block!(other_account, uri: nil)
remove_potential_friendship(other_account)
block_relationships.create_with(uri: uri)
diff --git a/app/models/concerns/rate_limitable.rb b/app/models/concerns/rate_limitable.rb
new file mode 100644
index 00000000000..ad1b5e44e36
--- /dev/null
+++ b/app/models/concerns/rate_limitable.rb
@@ -0,0 +1,36 @@
+# frozen_string_literal: true
+
+module RateLimitable
+ extend ActiveSupport::Concern
+
+ def rate_limit=(value)
+ @rate_limit = value
+ end
+
+ def rate_limit?
+ @rate_limit
+ end
+
+ def rate_limiter(by, options = {})
+ return @rate_limiter if defined?(@rate_limiter)
+
+ @rate_limiter = RateLimiter.new(by, options)
+ end
+
+ class_methods do
+ def rate_limit(options = {})
+ after_create do
+ by = public_send(options[:by])
+
+ if rate_limit? && by&.local?
+ rate_limiter(by, options).record!
+ @rate_limit_recorded = true
+ end
+ end
+
+ after_rollback do
+ rate_limiter(public_send(options[:by]), options).rollback! if @rate_limit_recorded
+ end
+ end
+ end
+end
diff --git a/app/models/follow.rb b/app/models/follow.rb
index 87fa114253b..f3e48a2ed7e 100644
--- a/app/models/follow.rb
+++ b/app/models/follow.rb
@@ -15,6 +15,9 @@
class Follow < ApplicationRecord
include Paginable
include RelationshipCacheable
+ include RateLimitable
+
+ rate_limit by: :account, family: :follows
belongs_to :account
belongs_to :target_account, class_name: 'Account'
diff --git a/app/models/follow_request.rb b/app/models/follow_request.rb
index 96ac7eaa593..3325e264cce 100644
--- a/app/models/follow_request.rb
+++ b/app/models/follow_request.rb
@@ -15,6 +15,9 @@
class FollowRequest < ApplicationRecord
include Paginable
include RelationshipCacheable
+ include RateLimitable
+
+ rate_limit by: :account, family: :follows
belongs_to :account
belongs_to :target_account, class_name: 'Account'
diff --git a/app/models/status.rb b/app/models/status.rb
index 1e630196bde..1610da2450a 100644
--- a/app/models/status.rb
+++ b/app/models/status.rb
@@ -32,6 +32,9 @@ class Status < ApplicationRecord
include Paginable
include Cacheable
include StatusThreadingConcern
+ include RateLimitable
+
+ rate_limit by: :account, family: :statuses
self.discard_column = :deleted_at
diff --git a/app/services/follow_service.rb b/app/services/follow_service.rb
index 4d19002c40e..311ae7fa688 100644
--- a/app/services/follow_service.rb
+++ b/app/services/follow_service.rb
@@ -7,54 +7,68 @@ class FollowService < BaseService
# Follow a remote user, notify remote user about the follow
# @param [Account] source_account From which to follow
# @param [String, Account] uri User URI to follow in the form of username@domain (or account record)
- # @param [true, false, nil] reblogs Whether or not to show reblogs, defaults to true
- def call(source_account, target_account, reblogs: nil, bypass_locked: false)
- reblogs = true if reblogs.nil?
- target_account = ResolveAccountService.new.call(target_account, skip_webfinger: true)
-
- raise ActiveRecord::RecordNotFound if target_account.nil? || target_account.id == source_account.id || target_account.suspended?
- raise Mastodon::NotPermittedError if target_account.blocking?(source_account) || source_account.blocking?(target_account) || target_account.moved? || (!target_account.local? && target_account.ostatus?) || source_account.domain_blocking?(target_account.domain)
-
- if source_account.following?(target_account)
- # We're already following this account, but we'll call follow! again to
- # make sure the reblogs status is set correctly.
- return source_account.follow!(target_account, reblogs: reblogs)
- elsif source_account.requested?(target_account)
- # This isn't managed by a method in AccountInteractions, so we modify it
- # ourselves if necessary.
- req = source_account.follow_requests.find_by(target_account: target_account)
- req.update!(show_reblogs: reblogs)
- return req
+ # @param [Hash] options
+ # @option [Boolean] :reblogs Whether or not to show reblogs, defaults to true
+ # @option [Boolean] :bypass_locked
+ # @option [Boolean] :with_rate_limit
+ def call(source_account, target_account, options = {})
+ @source_account = source_account
+ @target_account = ResolveAccountService.new.call(target_account, skip_webfinger: true)
+ @options = { reblogs: true, bypass_locked: false, with_rate_limit: false }.merge(options)
+
+ raise ActiveRecord::RecordNotFound if following_not_possible?
+ raise Mastodon::NotPermittedError if following_not_allowed?
+
+ if @source_account.following?(@target_account)
+ return change_follow_options!
+ elsif @source_account.requested?(@target_account)
+ return change_follow_request_options!
end
ActivityTracker.increment('activity:interactions')
- if (target_account.locked? && !bypass_locked) || source_account.silenced? || target_account.activitypub?
- request_follow(source_account, target_account, reblogs: reblogs)
- elsif target_account.local?
- direct_follow(source_account, target_account, reblogs: reblogs)
+ if (@target_account.locked? && !@options[:bypass_locked]) || @source_account.silenced? || @target_account.activitypub?
+ request_follow!
+ elsif @target_account.local?
+ direct_follow!
end
end
private
- def request_follow(source_account, target_account, reblogs: true)
- follow_request = FollowRequest.create!(account: source_account, target_account: target_account, show_reblogs: reblogs)
+ def following_not_possible?
+ @target_account.nil? || @target_account.id == @source_account.id || @target_account.suspended?
+ end
+
+ def following_not_allowed?
+ @target_account.blocking?(@source_account) || @source_account.blocking?(@target_account) || @target_account.moved? || (!@target_account.local? && @target_account.ostatus?) || @source_account.domain_blocking?(@target_account.domain)
+ end
+
+ def change_follow_options!
+ @source_account.follow!(@target_account, reblogs: @options[:reblogs])
+ end
+
+ def change_follow_request_options!
+ @source_account.request_follow!(@target_account, reblogs: @options[:reblogs])
+ end
+
+ def request_follow!
+ follow_request = @source_account.request_follow!(@target_account, reblogs: @options[:reblogs], rate_limit: @options[:with_rate_limit])
- if target_account.local?
- LocalNotificationWorker.perform_async(target_account.id, follow_request.id, follow_request.class.name)
- elsif target_account.activitypub?
- ActivityPub::DeliveryWorker.perform_async(build_json(follow_request), source_account.id, target_account.inbox_url)
+ if @target_account.local?
+ LocalNotificationWorker.perform_async(@target_account.id, follow_request.id, follow_request.class.name)
+ elsif @target_account.activitypub?
+ ActivityPub::DeliveryWorker.perform_async(build_json(follow_request), @source_account.id, @target_account.inbox_url)
end
follow_request
end
- def direct_follow(source_account, target_account, reblogs: true)
- follow = source_account.follow!(target_account, reblogs: reblogs)
+ def direct_follow!
+ follow = @source_account.follow!(@target_account, reblogs: @options[:reblogs], rate_limit: @options[:with_rate_limit])
- LocalNotificationWorker.perform_async(target_account.id, follow.id, follow.class.name)
- MergeWorker.perform_async(target_account.id, source_account.id)
+ LocalNotificationWorker.perform_async(@target_account.id, follow.id, follow.class.name)
+ MergeWorker.perform_async(@target_account.id, @source_account.id)
follow
end
diff --git a/app/services/post_status_service.rb b/app/services/post_status_service.rb
index a0a650d6241..1bbd625b4d6 100644
--- a/app/services/post_status_service.rb
+++ b/app/services/post_status_service.rb
@@ -19,6 +19,7 @@ class PostStatusService < BaseService
# @option [Enumerable] :media_ids Optional array of media IDs to attach
# @option [Doorkeeper::Application] :application
# @option [String] :idempotency Optional idempotency key
+ # @option [Boolean] :with_rate_limit
# @return [Status]
def call(account, options = {})
@account = account
@@ -160,6 +161,7 @@ class PostStatusService < BaseService
visibility: @visibility,
language: language_from_option(@options[:language]) || @account.user&.setting_default_language&.presence || LanguageDetector.instance.detect(@text, @account),
application: @options[:application],
+ rate_limit: @options[:with_rate_limit],
}.compact
end
@@ -179,10 +181,11 @@ class PostStatusService < BaseService
def scheduled_options
@options.tap do |options_hash|
- options_hash[:in_reply_to_id] = options_hash.delete(:thread)&.id
- options_hash[:application_id] = options_hash.delete(:application)&.id
- options_hash[:scheduled_at] = nil
- options_hash[:idempotency] = nil
+ options_hash[:in_reply_to_id] = options_hash.delete(:thread)&.id
+ options_hash[:application_id] = options_hash.delete(:application)&.id
+ options_hash[:scheduled_at] = nil
+ options_hash[:idempotency] = nil
+ options_hash[:with_rate_limit] = false
end
end
end
diff --git a/app/services/reblog_service.rb b/app/services/reblog_service.rb
index 3bb460fcaf9..4b5ae949268 100644
--- a/app/services/reblog_service.rb
+++ b/app/services/reblog_service.rb
@@ -8,6 +8,8 @@ class ReblogService < BaseService
# @param [Account] account Account to reblog from
# @param [Status] reblogged_status Status to be reblogged
# @param [Hash] options
+ # @option [String] :visibility
+ # @op