summaryrefslogtreecommitdiffstats
path: root/app/lib
diff options
context:
space:
mode:
authorTakeshi Umeda <noel.yoshiba@gmail.com>2020-07-20 18:25:26 +0900
committerGitHub <noreply@github.com>2020-07-20 11:25:26 +0200
commitfcb3f259e5a36dc4ac5300aa715d583f3a577c2b (patch)
treea2cb632fe0d36d18bb6d471ef6a90f5328c1b963 /app/lib
parent0ab97107c76732d3af2ff415d8d9d9d89da25d93 (diff)
Fix to add RedisLock to handle Announce activity (#14365)
Diffstat (limited to 'app/lib')
-rw-r--r--app/lib/activitypub/activity/announce.rb39
1 files changed, 25 insertions, 14 deletions
diff --git a/app/lib/activitypub/activity/announce.rb b/app/lib/activitypub/activity/announce.rb
index 34c646668b1..9e108985a3e 100644
--- a/app/lib/activitypub/activity/announce.rb
+++ b/app/lib/activitypub/activity/announce.rb
@@ -4,25 +4,32 @@ class ActivityPub::Activity::Announce < ActivityPub::Activity
def perform
return reject_payload! if delete_arrived_first?(@json['id']) || !related_to_local_activity?
- original_status = status_from_object
+ RedisLock.acquire(lock_options) do |lock|
+ if lock.acquired?
+ original_status = status_from_object
- return reject_payload! if original_status.nil? || !announceable?(original_status)
+ return reject_payload! if original_status.nil? || !announceable?(original_status)
- status = Status.find_by(account: @account, reblog: original_status)
+ @status = Status.find_by(account: @account, reblog: original_status)
- return status unless status.nil?
+ return @status unless @status.nil?
- status = Status.create!(
- account: @account,
- reblog: original_status,
- uri: @json['id'],
- created_at: @json['published'],
- override_timestamps: @options[:override_timestamps],
- visibility: visibility_from_audience
- )
+ @status = Status.create!(
+ account: @account,
+ reblog: original_status,
+ uri: @json['id'],
+ created_at: @json['published'],
+ override_timestamps: @options[:override_timestamps],
+ visibility: visibility_from_audience
+ )
- distribute(status)
- status
+ distribute(@status)
+ else
+ raise Mastodon::RaceConditionError
+ end
+ end
+
+ @status
end
private
@@ -54,4 +61,8 @@ class ActivityPub::Activity::Announce < ActivityPub::Activity
def reblog_of_local_status?
status_from_uri(object_uri)&.account&.local?
end
+
+ def lock_options
+ { redis: Redis.current, key: "announce:#{@object['id']}" }
+ end
end