178 lines
5 KiB
Ruby
178 lines
5 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
class SearchService < BaseService
|
|
def call(query, account, limit, options = {})
|
|
@query = query&.strip
|
|
@account = account
|
|
@options = options
|
|
@limit = limit.to_i
|
|
@offset = options[:type].blank? ? 0 : options[:offset].to_i
|
|
@resolve = options[:resolve] || false
|
|
|
|
default_results.tap do |results|
|
|
next if @query.blank? || @limit.zero?
|
|
|
|
if url_query?
|
|
results.merge!(url_resource_results) unless url_resource.nil? || @offset.positive? || (@options[:type].present? && url_resource_symbol != @options[:type].to_sym)
|
|
elsif @query.present?
|
|
# Account and status searches use different sets of prefix operators.
|
|
# Throw a syntax error only if the syntax is invalid in all search contexts.
|
|
search_succeeded = false
|
|
syntax_error = nil
|
|
|
|
if account_searchable?
|
|
begin
|
|
results[:accounts] = perform_accounts_search!
|
|
search_succeeded = true
|
|
rescue Mastodon::SyntaxError => e
|
|
syntax_error = e
|
|
end
|
|
end
|
|
|
|
if status_searchable?
|
|
begin
|
|
results[:statuses] = perform_statuses_search!
|
|
search_succeeded = true
|
|
rescue Mastodon::SyntaxError => e
|
|
syntax_error = e
|
|
end
|
|
end
|
|
|
|
if hashtag_searchable?
|
|
begin
|
|
results[:hashtags] = perform_hashtags_search!
|
|
search_succeeded = true
|
|
rescue Mastodon::SyntaxError => e
|
|
syntax_error = e
|
|
end
|
|
end
|
|
|
|
raise syntax_error unless syntax_error.nil? || search_succeeded
|
|
end
|
|
end
|
|
end
|
|
|
|
private
|
|
|
|
def perform_accounts_search!
|
|
AccountSearchService.new.call(
|
|
@query,
|
|
@account,
|
|
limit: @limit,
|
|
resolve: @resolve,
|
|
offset: @offset
|
|
)
|
|
end
|
|
|
|
def perform_statuses_search!
|
|
required_account_ids = parsed_query.statuses_required_account_ids
|
|
return [] if @account.blocked_by.exists?(id: required_account_ids)
|
|
|
|
statuses_index = StatusesIndex.filter(term: { searchable_by: @account.id })
|
|
|
|
status_search_scope = Rails.configuration.x.status_search_scope
|
|
case status_search_scope
|
|
when :discoverable
|
|
statuses_index = statuses_index.filter.or(
|
|
bool: {
|
|
filter: [
|
|
{ term: { visibility: 'public' } },
|
|
{ term: { discoverable: true } },
|
|
{ term: { silenced: false } },
|
|
],
|
|
}
|
|
)
|
|
when :public
|
|
statuses_index = statuses_index.filter.or(term: { visibility: 'public' })
|
|
when :public_or_unlisted
|
|
statuses_index = statuses_index.filter.or(terms: { visibility: %w(public unlisted) })
|
|
when :classic
|
|
# No alternate filter queries.
|
|
else
|
|
raise InvalidParameterError, "Unexpected status search scope: #{status_search_scope}"
|
|
end
|
|
|
|
definition = parsed_query.statuses_apply(statuses_index, @account.id, following_ids)
|
|
|
|
definition = definition.filter(term: { account_id: @options[:account_id] }) if @options[:account_id].present?
|
|
|
|
if @options[:min_id].present? || @options[:max_id].present?
|
|
range = {}
|
|
range[:gt] = @options[:min_id].to_i if @options[:min_id].present?
|
|
range[:lt] = @options[:max_id].to_i if @options[:max_id].present?
|
|
definition = definition.filter(range: { id: range })
|
|
end
|
|
|
|
results = definition.limit(@limit).offset(@offset).objects.compact
|
|
account_ids = results.map(&:account_id)
|
|
account_domains = results.map(&:account_domain)
|
|
preloaded_relations = @account.relations_map(account_ids, account_domains)
|
|
|
|
results.reject { |status| StatusFilter.new(status, @account, preloaded_relations).filtered? }
|
|
rescue Faraday::ConnectionFailed, Parslet::ParseFailed
|
|
[]
|
|
end
|
|
|
|
def perform_hashtags_search!
|
|
TagSearchService.new.call(
|
|
@query,
|
|
limit: @limit,
|
|
offset: @offset,
|
|
exclude_unreviewed: @options[:exclude_unreviewed]
|
|
)
|
|
end
|
|
|
|
def default_results
|
|
{ accounts: [], hashtags: [], statuses: [] }
|
|
end
|
|
|
|
def url_query?
|
|
@resolve && /\Ahttps?:\/\//.match?(@query)
|
|
end
|
|
|
|
def url_resource_results
|
|
{ url_resource_symbol => [url_resource] }
|
|
end
|
|
|
|
def url_resource
|
|
@_url_resource ||= ResolveURLService.new.call(@query, on_behalf_of: @account)
|
|
end
|
|
|
|
def url_resource_symbol
|
|
url_resource.class.name.downcase.pluralize.to_sym
|
|
end
|
|
|
|
def status_searchable?
|
|
return false unless Chewy.enabled?
|
|
|
|
statuses_search? && !@account.nil?
|
|
end
|
|
|
|
def account_searchable?
|
|
account_search?
|
|
end
|
|
|
|
def hashtag_searchable?
|
|
hashtag_search? && !@query.include?('@')
|
|
end
|
|
|
|
def account_search?
|
|
@options[:type].blank? || @options[:type] == 'accounts'
|
|
end
|
|
|
|
def hashtag_search?
|
|
@options[:type].blank? || @options[:type] == 'hashtags'
|
|
end
|
|
|
|
def statuses_search?
|
|
@options[:type].blank? || @options[:type] == 'statuses'
|
|
end
|
|
|
|
def parsed_query
|
|
SearchQueryTransformer.new.apply(SearchQueryParser.new.parse(@query))
|
|
end
|
|
|
|
def following_ids
|
|
@following_ids ||= @account.active_relationships.pluck(:target_account_id) + [@account.id]
|
|
end
|
|
end
|