Fix poll update handler calling method was that was not available (#10246)
* Fix poll update handler calling method was that was not available Fix regression from #10209 * Refactor VoteService * Refactor ActivityPub::DistributePollUpdateWorker and optimize it * Fix typo * Fix typololsob-rspec
parent
45d257a20f
commit
d5377d0d45
|
@ -47,6 +47,15 @@ module JsonLdHelper
|
||||||
!uri.start_with?('http://', 'https://')
|
!uri.start_with?('http://', 'https://')
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def invalid_origin?(url)
|
||||||
|
return true if unsupported_uri_scheme?(url)
|
||||||
|
|
||||||
|
needle = Addressable::URI.parse(url).host
|
||||||
|
haystack = Addressable::URI.parse(@account.uri).host
|
||||||
|
|
||||||
|
!haystack.casecmp(needle).zero?
|
||||||
|
end
|
||||||
|
|
||||||
def canonicalize(json)
|
def canonicalize(json)
|
||||||
graph = RDF::Graph.new << JSON::LD::API.toRdf(json, documentLoader: method(:load_jsonld_context))
|
graph = RDF::Graph.new << JSON::LD::API.toRdf(json, documentLoader: method(:load_jsonld_context))
|
||||||
graph.dump(:normalize)
|
graph.dump(:normalize)
|
||||||
|
|
|
@ -241,9 +241,12 @@ class ActivityPub::Activity::Create < ActivityPub::Activity
|
||||||
|
|
||||||
def poll_vote?
|
def poll_vote?
|
||||||
return false if replied_to_status.nil? || replied_to_status.poll.nil? || !replied_to_status.local? || !replied_to_status.poll.options.include?(@object['name'])
|
return false if replied_to_status.nil? || replied_to_status.poll.nil? || !replied_to_status.local? || !replied_to_status.poll.options.include?(@object['name'])
|
||||||
return true if replied_to_status.poll.expired?
|
|
||||||
|
unless replied_to_status.poll.expired?
|
||||||
replied_to_status.poll.votes.create!(account: @account, choice: replied_to_status.poll.options.index(@object['name']), uri: @object['id'])
|
replied_to_status.poll.votes.create!(account: @account, choice: replied_to_status.poll.options.index(@object['name']), uri: @object['id'])
|
||||||
ActivityPub::DistributePollUpdateWorker.perform_in(3.minutes, replied_to_status.id) unless replied_to_status.poll.hide_totals
|
ActivityPub::DistributePollUpdateWorker.perform_in(3.minutes, replied_to_status.id) unless replied_to_status.poll.hide_totals?
|
||||||
|
end
|
||||||
|
|
||||||
true
|
true
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -371,15 +374,6 @@ class ActivityPub::Activity::Create < ActivityPub::Activity
|
||||||
@skip_download ||= DomainBlock.find_by(domain: @account.domain)&.reject_media?
|
@skip_download ||= DomainBlock.find_by(domain: @account.domain)&.reject_media?
|
||||||
end
|
end
|
||||||
|
|
||||||
def invalid_origin?(url)
|
|
||||||
return true if unsupported_uri_scheme?(url)
|
|
||||||
|
|
||||||
needle = Addressable::URI.parse(url).host
|
|
||||||
haystack = Addressable::URI.parse(@account.uri).host
|
|
||||||
|
|
||||||
!haystack.casecmp(needle).zero?
|
|
||||||
end
|
|
||||||
|
|
||||||
def reply_to_local?
|
def reply_to_local?
|
||||||
!replied_to_status.nil? && replied_to_status.account.local?
|
!replied_to_status.nil? && replied_to_status.account.local?
|
||||||
end
|
end
|
||||||
|
|
|
@ -75,13 +75,4 @@ class ActivityPub::Activity::Delete < ActivityPub::Activity
|
||||||
def lock_options
|
def lock_options
|
||||||
{ redis: Redis.current, key: "create:#{object_uri}" }
|
{ redis: Redis.current, key: "create:#{object_uri}" }
|
||||||
end
|
end
|
||||||
|
|
||||||
def invalid_origin?(url)
|
|
||||||
return true if unsupported_uri_scheme?(url)
|
|
||||||
|
|
||||||
needle = Addressable::URI.parse(url).host
|
|
||||||
haystack = Addressable::URI.parse(@account.uri).host
|
|
||||||
|
|
||||||
!haystack.casecmp(needle).zero?
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
|
@ -4,8 +4,11 @@ class ActivityPub::Activity::Update < ActivityPub::Activity
|
||||||
SUPPORTED_TYPES = %w(Application Group Organization Person Service).freeze
|
SUPPORTED_TYPES = %w(Application Group Organization Person Service).freeze
|
||||||
|
|
||||||
def perform
|
def perform
|
||||||
update_account if equals_or_includes_any?(@object['type'], SUPPORTED_TYPES)
|
if equals_or_includes_any?(@object['type'], SUPPORTED_TYPES)
|
||||||
update_poll if equals_or_includes_any?(@object['type'], %w(Question))
|
update_account
|
||||||
|
elsif equals_or_includes_any?(@object['type'], %w(Question))
|
||||||
|
update_poll
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
@ -18,11 +21,10 @@ class ActivityPub::Activity::Update < ActivityPub::Activity
|
||||||
|
|
||||||
def update_poll
|
def update_poll
|
||||||
return reject_payload! if invalid_origin?(@object['id'])
|
return reject_payload! if invalid_origin?(@object['id'])
|
||||||
status = Status.find_by(uri: object_uri, account_id: @account.id)
|
|
||||||
return if status.nil? || status.poll_id.nil?
|
|
||||||
poll = Poll.find(status.poll_id)
|
|
||||||
return if poll.nil?
|
|
||||||
|
|
||||||
ActivityPub::ProcessPollService.new.call(poll, @object)
|
status = Status.find_by(uri: object_uri, account_id: @account.id)
|
||||||
|
return if status.nil? || status.poll.nil?
|
||||||
|
|
||||||
|
ActivityPub::ProcessPollService.new.call(status.poll, @object)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -18,7 +18,11 @@ module Expireable
|
||||||
end
|
end
|
||||||
|
|
||||||
def expired?
|
def expired?
|
||||||
!expires_at.nil? && expires_at < Time.now.utc
|
expires? && expires_at < Time.now.utc
|
||||||
|
end
|
||||||
|
|
||||||
|
def expires?
|
||||||
|
!expires_at.nil?
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -46,13 +46,4 @@ class ActivityPub::FetchRepliesService < BaseService
|
||||||
# Also limit to 5 fetched replies to limit potential for DoS.
|
# Also limit to 5 fetched replies to limit potential for DoS.
|
||||||
@items.map { |item| value_or_id(item) }.reject { |uri| invalid_origin?(uri) }.take(5)
|
@items.map { |item| value_or_id(item) }.reject { |uri| invalid_origin?(uri) }.take(5)
|
||||||
end
|
end
|
||||||
|
|
||||||
def invalid_origin?(url)
|
|
||||||
return true if unsupported_uri_scheme?(url)
|
|
||||||
|
|
||||||
needle = Addressable::URI.parse(url).host
|
|
||||||
haystack = Addressable::URI.parse(@account.uri).host
|
|
||||||
|
|
||||||
!haystack.casecmp(needle).zero?
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
|
@ -92,7 +92,7 @@ class NotifyService < BaseService
|
||||||
|
|
||||||
def blocked?
|
def blocked?
|
||||||
blocked = @recipient.suspended? # Skip if the recipient account is suspended anyway
|
blocked = @recipient.suspended? # Skip if the recipient account is suspended anyway
|
||||||
blocked ||= from_self? unless @notification.type == :poll # Skip for interactions with self
|
blocked ||= from_self? && @notification.type != :poll # Skip for interactions with self
|
||||||
|
|
||||||
return blocked if message? && from_staff?
|
return blocked if message? && from_staff?
|
||||||
|
|
||||||
|
|
|
@ -20,8 +20,26 @@ class VoteService < BaseService
|
||||||
end
|
end
|
||||||
|
|
||||||
if @poll.account.local?
|
if @poll.account.local?
|
||||||
ActivityPub::DistributePollUpdateWorker.perform_in(3.minutes, @poll.status.id) unless @poll.hide_totals
|
distribute_poll!
|
||||||
else
|
else
|
||||||
|
deliver_votes!
|
||||||
|
queue_final_poll_check!
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def distribute_poll!
|
||||||
|
return if @poll.hide_totals?
|
||||||
|
ActivityPub::DistributePollUpdateWorker.perform_in(3.minutes, @poll.status.id)
|
||||||
|
end
|
||||||
|
|
||||||
|
def queue_final_poll_check!
|
||||||
|
return unless @poll.expires?
|
||||||
|
PollExpirationNotifyWorker.perform_at(@poll.expires_at + 5.minutes, @poll.id)
|
||||||
|
end
|
||||||
|
|
||||||
|
def deliver_votes!
|
||||||
@votes.each do |vote|
|
@votes.each do |vote|
|
||||||
ActivityPub::DeliveryWorker.perform_async(
|
ActivityPub::DeliveryWorker.perform_async(
|
||||||
build_json(vote),
|
build_json(vote),
|
||||||
|
@ -29,11 +47,7 @@ class VoteService < BaseService
|
||||||
@poll.account.inbox_url
|
@poll.account.inbox_url
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
PollExpirationNotifyWorker.perform_at(@poll.expires_at + 5.minutes, @poll.id) unless @poll.expires_at.nil?
|
|
||||||
end
|
end
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
|
|
||||||
def build_json(vote)
|
def build_json(vote)
|
||||||
ActiveModelSerializers::SerializableResource.new(
|
ActiveModelSerializers::SerializableResource.new(
|
||||||
|
|
|
@ -28,13 +28,16 @@ class ActivityPub::DistributePollUpdateWorker
|
||||||
|
|
||||||
def inboxes
|
def inboxes
|
||||||
return @inboxes if defined?(@inboxes)
|
return @inboxes if defined?(@inboxes)
|
||||||
target_accounts = @status.mentions.map(&:account).reject(&:local?)
|
|
||||||
target_accounts += @status.reblogs.map(&:account).reject(&:local?)
|
@inboxes = [@status.mentions, @status.reblogs, @status.poll.votes].flat_map do |relation|
|
||||||
target_accounts += @status.poll.votes.map(&:account).reject(&:local?)
|
relation.includes(:account).map do |record|
|
||||||
target_accounts.uniq!(&:id)
|
record.account.preferred_inbox_url if !record.account.local? && record.account.activitypub?
|
||||||
@inboxes = target_accounts.select(&:activitypub?).pluck(&:inbox_url)
|
end
|
||||||
@inboxes += @account.followers.inboxes unless @status.direct_visibility?
|
end
|
||||||
|
|
||||||
|
@inboxes.concat(@account.followers.inboxes) unless @status.direct_visibility?
|
||||||
@inboxes.uniq!
|
@inboxes.uniq!
|
||||||
|
@inboxes.compact!
|
||||||
@inboxes
|
@inboxes
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue