mirror of
https://github.com/mastodon/mastodon.git
synced 2025-07-15 08:48:15 +00:00
Add FASP account search support (#34033)
This commit is contained in:
parent
6d017dbf10
commit
ac4b735c67
|
@ -1,6 +1,7 @@
|
||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
class Api::V2::SearchController < Api::BaseController
|
class Api::V2::SearchController < Api::BaseController
|
||||||
|
include AsyncRefreshesConcern
|
||||||
include Authorization
|
include Authorization
|
||||||
|
|
||||||
RESULTS_LIMIT = 20
|
RESULTS_LIMIT = 20
|
||||||
|
@ -13,6 +14,7 @@ class Api::V2::SearchController < Api::BaseController
|
||||||
before_action :remote_resolve_error, if: :remote_resolve_requested?
|
before_action :remote_resolve_error, if: :remote_resolve_requested?
|
||||||
end
|
end
|
||||||
before_action :require_valid_pagination_options!
|
before_action :require_valid_pagination_options!
|
||||||
|
before_action :handle_fasp_requests
|
||||||
|
|
||||||
def index
|
def index
|
||||||
@search = Search.new(search_results)
|
@search = Search.new(search_results)
|
||||||
|
@ -37,6 +39,21 @@ class Api::V2::SearchController < Api::BaseController
|
||||||
render json: { error: 'Search queries that resolve remote resources are not supported without authentication' }, status: 401
|
render json: { error: 'Search queries that resolve remote resources are not supported without authentication' }, status: 401
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def handle_fasp_requests
|
||||||
|
return unless Mastodon::Feature.fasp_enabled?
|
||||||
|
return if params[:q].blank?
|
||||||
|
|
||||||
|
# Do not schedule a new retrieval if the request is a follow-up
|
||||||
|
# to an earlier retrieval
|
||||||
|
return if request.headers['Mastodon-Async-Refresh-Id'].present?
|
||||||
|
|
||||||
|
refresh_key = "fasp:account_search:#{Digest::MD5.base64digest(params[:q])}"
|
||||||
|
return if AsyncRefresh.new(refresh_key).running?
|
||||||
|
|
||||||
|
add_async_refresh_header(AsyncRefresh.create(refresh_key))
|
||||||
|
@query_fasp = true
|
||||||
|
end
|
||||||
|
|
||||||
def remote_resolve_requested?
|
def remote_resolve_requested?
|
||||||
truthy_param?(:resolve)
|
truthy_param?(:resolve)
|
||||||
end
|
end
|
||||||
|
@ -58,7 +75,8 @@ class Api::V2::SearchController < Api::BaseController
|
||||||
search_params.merge(
|
search_params.merge(
|
||||||
resolve: truthy_param?(:resolve),
|
resolve: truthy_param?(:resolve),
|
||||||
exclude_unreviewed: truthy_param?(:exclude_unreviewed),
|
exclude_unreviewed: truthy_param?(:exclude_unreviewed),
|
||||||
following: truthy_param?(:following)
|
following: truthy_param?(:following),
|
||||||
|
query_fasp: @query_fasp
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -178,6 +178,12 @@ class AccountSearchService < BaseService
|
||||||
'search.backend' => Chewy.enabled? ? 'elasticsearch' : 'database'
|
'search.backend' => Chewy.enabled? ? 'elasticsearch' : 'database'
|
||||||
)
|
)
|
||||||
|
|
||||||
|
# Trigger searching accounts using providers.
|
||||||
|
# This will not return any immediate results but has the
|
||||||
|
# potential to fill the local database with relevant
|
||||||
|
# accounts for the next time the search is performed.
|
||||||
|
Fasp::AccountSearchWorker.perform_async(@query) if options[:query_fasp]
|
||||||
|
|
||||||
search_service_results.compact.uniq.tap do |results|
|
search_service_results.compact.uniq.tap do |results|
|
||||||
span.set_attribute('search.results.count', results.size)
|
span.set_attribute('search.results.count', results.size)
|
||||||
end
|
end
|
||||||
|
|
|
@ -11,6 +11,7 @@ class SearchService < BaseService
|
||||||
@offset = options[:type].blank? ? 0 : options[:offset].to_i
|
@offset = options[:type].blank? ? 0 : options[:offset].to_i
|
||||||
@resolve = options[:resolve] || false
|
@resolve = options[:resolve] || false
|
||||||
@following = options[:following] || false
|
@following = options[:following] || false
|
||||||
|
@query_fasp = options[:query_fasp] || false
|
||||||
|
|
||||||
default_results.tap do |results|
|
default_results.tap do |results|
|
||||||
next if @query.blank? || @limit.zero?
|
next if @query.blank? || @limit.zero?
|
||||||
|
@ -36,7 +37,8 @@ class SearchService < BaseService
|
||||||
offset: @offset,
|
offset: @offset,
|
||||||
use_searchable_text: true,
|
use_searchable_text: true,
|
||||||
following: @following,
|
following: @following,
|
||||||
start_with_hashtag: @query.start_with?('#')
|
start_with_hashtag: @query.start_with?('#'),
|
||||||
|
query_fasp: @options[:query_fasp]
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
30
app/workers/fasp/account_search_worker.rb
Normal file
30
app/workers/fasp/account_search_worker.rb
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class Fasp::AccountSearchWorker
|
||||||
|
include Sidekiq::Worker
|
||||||
|
|
||||||
|
sidekiq_options queue: 'fasp', retry: 0
|
||||||
|
|
||||||
|
def perform(query)
|
||||||
|
return unless Mastodon::Feature.fasp_enabled?
|
||||||
|
|
||||||
|
async_refresh = AsyncRefresh.new("fasp:account_search:#{Digest::MD5.base64digest(query)}")
|
||||||
|
|
||||||
|
account_search_providers = Fasp::Provider.with_capability('account_search')
|
||||||
|
return if account_search_providers.none?
|
||||||
|
|
||||||
|
params = { term: query, limit: 10 }.to_query
|
||||||
|
fetch_service = ActivityPub::FetchRemoteActorService.new
|
||||||
|
|
||||||
|
account_search_providers.each do |provider|
|
||||||
|
Fasp::Request.new(provider).get("/account_search/v0/search?#{params}").each do |uri|
|
||||||
|
next if Account.where(uri:).any?
|
||||||
|
|
||||||
|
account = fetch_service.call(uri)
|
||||||
|
async_refresh.increment_result_count(by: 1) if account.present?
|
||||||
|
end
|
||||||
|
end
|
||||||
|
ensure
|
||||||
|
async_refresh&.finish!
|
||||||
|
end
|
||||||
|
end
|
|
@ -41,3 +41,15 @@ Fabricator(:follow_recommendation_fasp, from: :fasp_provider) do
|
||||||
def fasp.update_remote_capabilities = true
|
def fasp.update_remote_capabilities = true
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
Fabricator(:account_search_fasp, from: :fasp_provider) do
|
||||||
|
confirmed true
|
||||||
|
capabilities [
|
||||||
|
{ id: 'account_search', version: '0.1', enabled: true },
|
||||||
|
]
|
||||||
|
|
||||||
|
after_build do |fasp|
|
||||||
|
# Prevent fabrication from attempting an HTTP call to the provider
|
||||||
|
def fasp.update_remote_capabilities = true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
|
@ -118,6 +118,15 @@ RSpec.describe 'Search API' do
|
||||||
.to start_with('application/json')
|
.to start_with('application/json')
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
context 'when `account_search` FASP is enabled', feature: :fasp do
|
||||||
|
it 'enqueues a retrieval job and adds a header to inform the client' do
|
||||||
|
get '/api/v2/search', headers: headers, params: params
|
||||||
|
|
||||||
|
expect(Fasp::AccountSearchWorker).to have_enqueued_sidekiq_job
|
||||||
|
expect(response.headers['Mastodon-Async-Refresh']).to be_present
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -66,7 +66,7 @@ RSpec.describe SearchService do
|
||||||
allow(AccountSearchService).to receive(:new).and_return(service)
|
allow(AccountSearchService).to receive(:new).and_return(service)
|
||||||
|
|
||||||
results = subject.call(query, nil, 10)
|
results = subject.call(query, nil, 10)
|
||||||
expect(service).to have_received(:call).with(query, nil, limit: 10, offset: 0, resolve: false, start_with_hashtag: false, use_searchable_text: true, following: false)
|
expect(service).to have_received(:call).with(query, nil, limit: 10, offset: 0, resolve: false, start_with_hashtag: false, use_searchable_text: true, following: false, query_fasp: nil)
|
||||||
expect(results).to eq empty_results.merge(accounts: [account])
|
expect(results).to eq empty_results.merge(accounts: [account])
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
50
spec/workers/fasp/account_search_worker_spec.rb
Normal file
50
spec/workers/fasp/account_search_worker_spec.rb
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'rails_helper'
|
||||||
|
|
||||||
|
RSpec.describe Fasp::AccountSearchWorker, feature: :fasp do
|
||||||
|
include ProviderRequestHelper
|
||||||
|
|
||||||
|
let(:provider) { Fabricate(:account_search_fasp) }
|
||||||
|
let(:account) { Fabricate(:account) }
|
||||||
|
let(:fetch_service) { instance_double(ActivityPub::FetchRemoteActorService, call: true) }
|
||||||
|
|
||||||
|
let!(:stubbed_request) do
|
||||||
|
path = '/account_search/v0/search?term=cats&limit=10'
|
||||||
|
stub_provider_request(provider,
|
||||||
|
method: :get,
|
||||||
|
path:,
|
||||||
|
response_body: [
|
||||||
|
'https://fedi.example.com/accounts/2',
|
||||||
|
'https://fedi.example.com/accounts/9',
|
||||||
|
])
|
||||||
|
end
|
||||||
|
|
||||||
|
before do
|
||||||
|
allow(ActivityPub::FetchRemoteActorService).to receive(:new).and_return(fetch_service)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'requests search results and fetches received account uris' do
|
||||||
|
described_class.new.perform('cats')
|
||||||
|
|
||||||
|
expect(stubbed_request).to have_been_made
|
||||||
|
expect(fetch_service).to have_received(:call).with('https://fedi.example.com/accounts/2')
|
||||||
|
expect(fetch_service).to have_received(:call).with('https://fedi.example.com/accounts/9')
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'marks a running async refresh as finished' do
|
||||||
|
async_refresh = AsyncRefresh.create("fasp:account_search:#{Digest::MD5.base64digest('cats')}", count_results: true)
|
||||||
|
|
||||||
|
described_class.new.perform('cats')
|
||||||
|
|
||||||
|
expect(async_refresh.reload).to be_finished
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'tracks the number of fetched accounts in the async refresh' do
|
||||||
|
async_refresh = AsyncRefresh.create("fasp:account_search:#{Digest::MD5.base64digest('cats')}", count_results: true)
|
||||||
|
|
||||||
|
described_class.new.perform('cats')
|
||||||
|
|
||||||
|
expect(async_refresh.reload.result_count).to eq 2
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in New Issue
Block a user