2018-03-04 09:19:11 +01:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
class ActivityPub::FetchFeaturedCollectionService < BaseService
|
|
|
|
include JsonLdHelper
|
|
|
|
|
2022-10-20 16:15:52 +09:00
|
|
|
def call(account, **options)
|
2019-07-09 03:27:35 +02:00
|
|
|
return if account.featured_collection_url.blank? || account.suspended? || account.local?
|
2018-05-02 15:44:22 +02:00
|
|
|
|
2018-03-04 09:19:11 +01:00
|
|
|
@account = account
|
2022-10-20 16:15:52 +09:00
|
|
|
@options = options
|
2022-05-02 17:41:01 +02:00
|
|
|
@json = fetch_resource(@account.featured_collection_url, true, local_follower)
|
2018-03-04 09:19:11 +01:00
|
|
|
|
2022-05-02 17:41:01 +02:00
|
|
|
return unless supported_context?(@json)
|
2018-03-04 09:19:11 +01:00
|
|
|
|
2022-05-02 17:41:01 +02:00
|
|
|
process_items(collection_items(@json))
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def collection_items(collection)
|
|
|
|
collection = fetch_collection(collection['first']) if collection['first'].present?
|
|
|
|
return unless collection.is_a?(Hash)
|
|
|
|
|
|
|
|
case collection['type']
|
2018-03-04 09:19:11 +01:00
|
|
|
when 'Collection', 'CollectionPage'
|
2024-01-19 13:43:10 +01:00
|
|
|
as_array(collection['items'])
|
2018-03-04 09:19:11 +01:00
|
|
|
when 'OrderedCollection', 'OrderedCollectionPage'
|
2024-01-19 13:43:10 +01:00
|
|
|
as_array(collection['orderedItems'])
|
2018-03-04 09:19:11 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-05-02 17:41:01 +02:00
|
|
|
def fetch_collection(collection_or_uri)
|
|
|
|
return collection_or_uri if collection_or_uri.is_a?(Hash)
|
2023-04-23 16:35:54 -04:00
|
|
|
return if non_matching_uri_hosts?(@account.uri, collection_or_uri)
|
2022-05-02 17:41:01 +02:00
|
|
|
|
2022-05-02 19:34:39 +02:00
|
|
|
fetch_resource_without_id_validation(collection_or_uri, local_follower, true)
|
2022-05-02 17:41:01 +02:00
|
|
|
end
|
2018-03-04 09:19:11 +01:00
|
|
|
|
|
|
|
def process_items(items)
|
2023-10-27 11:36:22 +09:00
|
|
|
return if items.nil?
|
|
|
|
|
2022-10-20 16:15:52 +09:00
|
|
|
process_note_items(items) if @options[:note]
|
|
|
|
process_hashtag_items(items) if @options[:hashtag]
|
|
|
|
end
|
|
|
|
|
|
|
|
def process_note_items(items)
|
2022-04-06 20:58:23 +02:00
|
|
|
status_ids = items.filter_map do |item|
|
2022-10-21 18:48:22 +09:00
|
|
|
next unless item.is_a?(String) || item['type'] == 'Note'
|
2022-10-20 16:15:52 +09:00
|
|
|
|
2022-04-06 20:58:23 +02:00
|
|
|
uri = value_or_id(item)
|
2023-04-23 16:35:54 -04:00
|
|
|
next if ActivityPub::TagManager.instance.local_uri?(uri) || non_matching_uri_hosts?(@account.uri, uri)
|
2022-04-06 20:58:23 +02:00
|
|
|
|
2022-12-07 00:15:24 +01:00
|
|
|
status = ActivityPub::FetchRemoteStatusService.new.call(uri, on_behalf_of: local_follower, expected_actor_uri: @account.uri, request_id: @options[:request_id])
|
2022-04-13 20:25:42 +09:00
|
|
|
next unless status&.account_id == @account.id
|
2022-04-06 20:58:23 +02:00
|
|
|
|
|
|
|
status.id
|
|
|
|
rescue ActiveRecord::RecordInvalid => e
|
2023-02-06 21:44:36 -05:00
|
|
|
Rails.logger.debug { "Invalid pinned status #{uri}: #{e.message}" }
|
2022-04-06 20:58:23 +02:00
|
|
|
nil
|
|
|
|
end
|
|
|
|
|
2018-03-04 09:19:11 +01:00
|
|
|
to_remove = []
|
|
|
|
to_add = status_ids
|
|
|
|
|
|
|
|
StatusPin.where(account: @account).pluck(:status_id).each do |status_id|
|
|
|
|
if status_ids.include?(status_id)
|
|
|
|
to_add.delete(status_id)
|
|
|
|
else
|
|
|
|
to_remove << status_id
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
StatusPin.where(account: @account, status_id: to_remove).delete_all unless to_remove.empty?
|
|
|
|
|
|
|
|
to_add.each do |status_id|
|
|
|
|
StatusPin.create!(account: @account, status_id: status_id)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-10-20 16:15:52 +09:00
|
|
|
def process_hashtag_items(items)
|
|
|
|
names = items.filter_map { |item| item['type'] == 'Hashtag' && item['name']&.delete_prefix('#') }.map { |name| HashtagNormalizer.new.normalize(name) }
|
|
|
|
to_remove = []
|
|
|
|
to_add = names
|
|
|
|
|
|
|
|
FeaturedTag.where(account: @account).map(&:name).each do |name|
|
|
|
|
if names.include?(name)
|
|
|
|
to_add.delete(name)
|
|
|
|
else
|
|
|
|
to_remove << name
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
FeaturedTag.includes(:tag).where(account: @account, tags: { name: to_remove }).delete_all unless to_remove.empty?
|
|
|
|
|
|
|
|
to_add.each do |name|
|
|
|
|
FeaturedTag.create!(account: @account, name: name)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-01-17 00:49:55 +01:00
|
|
|
def local_follower
|
2022-05-02 17:41:01 +02:00
|
|
|
return @local_follower if defined?(@local_follower)
|
|
|
|
|
|
|
|
@local_follower = @account.followers.local.without_suspended.first
|
2022-01-17 00:49:55 +01:00
|
|
|
end
|
2018-03-04 09:19:11 +01:00
|
|
|
end
|