!uri.start_with?('http://', 'https://')
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)
graph = RDF::Graph.new << JSON::LD::API.toRdf(json, documentLoader: method(:load_jsonld_context))
graph.dump(:normalize)
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 true if 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'])
- ActivityPub::DistributePollUpdateWorker.perform_in(3.minutes, replied_to_status.id) unless replied_to_status.poll.hide_totals
+
+ 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'])
+ ActivityPub::DistributePollUpdateWorker.perform_in(3.minutes, replied_to_status.id) unless replied_to_status.poll.hide_totals?
+ end
+
true
end
@skip_download ||= DomainBlock.find_by(domain: @account.domain)&.reject_media?
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?
!replied_to_status.nil? && replied_to_status.account.local?
end
def lock_options
{ redis: Redis.current, key: "create:#{object_uri}" }
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
SUPPORTED_TYPES = %w(Application Group Organization Person Service).freeze
def perform
- update_account if equals_or_includes_any?(@object['type'], SUPPORTED_TYPES)
- update_poll if equals_or_includes_any?(@object['type'], %w(Question))
+ if equals_or_includes_any?(@object['type'], SUPPORTED_TYPES)
+ update_account
+ elsif equals_or_includes_any?(@object['type'], %w(Question))
+ update_poll
+ end
end
private
def update_poll
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?
+ return if status.nil? || status.poll.nil?
- ActivityPub::ProcessPollService.new.call(poll, @object)
+ ActivityPub::ProcessPollService.new.call(status.poll, @object)
end
end
end
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
# 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)
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
def blocked?
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?
end
if @poll.account.local?
- ActivityPub::DistributePollUpdateWorker.perform_in(3.minutes, @poll.status.id) unless @poll.hide_totals
+ distribute_poll!
else
- @votes.each do |vote|
- ActivityPub::DeliveryWorker.perform_async(
- build_json(vote),
- @account.id,
- @poll.account.inbox_url
- )
- end
- PollExpirationNotifyWorker.perform_at(@poll.expires_at + 5.minutes, @poll.id) unless @poll.expires_at.nil?
+ 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|
+ ActivityPub::DeliveryWorker.perform_async(
+ build_json(vote),
+ @account.id,
+ @poll.account.inbox_url
+ )
+ end
+ end
+
def build_json(vote)
ActiveModelSerializers::SerializableResource.new(
vote,
def inboxes
return @inboxes if defined?(@inboxes)
- target_accounts = @status.mentions.map(&:account).reject(&:local?)
- target_accounts += @status.reblogs.map(&:account).reject(&:local?)
- target_accounts += @status.poll.votes.map(&:account).reject(&:local?)
- target_accounts.uniq!(&:id)
- @inboxes = target_accounts.select(&:activitypub?).pluck(&:inbox_url)
- @inboxes += @account.followers.inboxes unless @status.direct_visibility?
+
+ @inboxes = [@status.mentions, @status.reblogs, @status.poll.votes].flat_map do |relation|
+ relation.includes(:account).map do |record|
+ record.account.preferred_inbox_url if !record.account.local? && record.account.activitypub?
+ end
+ end
+
+ @inboxes.concat(@account.followers.inboxes) unless @status.direct_visibility?
@inboxes.uniq!
+ @inboxes.compact!
@inboxes
end