# frozen_string_literal: true class SearchQueryTransformer < Parslet::Transform class Query attr_reader :should_clauses, :must_not_clauses, :must_clauses, :filter_clauses, :order_clauses def initialize(clauses) grouped = clauses.group_by(&:operator).to_h @should_clauses = grouped.fetch(:should, []) @must_not_clauses = grouped.fetch(:must_not, []) @must_clauses = grouped.fetch(:must, []) @filter_clauses = grouped.fetch(:filter, []) @order_clauses = grouped.fetch(:order, []) end # Return account IDs that must not block the searching account for the search to be legal. def statuses_required_account_ids ids = Set.new positive_clauses = [should_clauses, must_clauses, filter_clauses] positive_clauses.each do |clauses| clauses.each do |clause| id = clause_to_account_id(clause) ids << id unless id.nil? end end ids end # Modifies a statuses search to include clauses from this query. def statuses_apply(search, account_id, following_ids) search_type = :statuses check_search_type(search_type) search_fields = %w(text text.stemmed) should_clauses.each { |clause| search = search.query.should(clause_to_query(clause, search_type, search_fields, account_id: account_id, following_ids: following_ids)) } must_clauses.each { |clause| search = search.query.must(clause_to_query(clause, search_type, search_fields, account_id: account_id, following_ids: following_ids)) } must_not_clauses.each { |clause| search = search.query.must_not(clause_to_query(clause, search_type, search_fields, account_id: account_id, following_ids: following_ids)) } filter_clauses.each { |clause| search = search.filter(clause_to_query(clause, search_type, search_fields, account_id: account_id, following_ids: following_ids)) } if order_clauses.empty? # Default to most recent results first. search = search.order(created_at: :desc) else order_clauses.each { |clause| search = search.order(clause_to_order(clause)) } end search.query.minimum_should_match(1) end # Generates the core query used for an accounts search. def accounts_query(likely_acct, search_scope, account_exists, following, following_ids) search_type = :accounts check_search_type(search_type) full_text_enabled = account_exists && search_scope != :classic search_fields = %w(acct.edge_ngram acct) search_fields += %w(display_name.edge_ngram display_name) unless likely_acct search_fields += %w(text.stemmed text) if full_text_enabled params = { must: must_clauses.map { |clause| clause_to_query(clause, search_type, search_fields, following_ids: following_ids) }, must_not: must_not_clauses.map { |clause| clause_to_query(clause, search_type, search_fields, following_ids: following_ids) }, should: should_clauses.map { |clause| clause_to_query(clause, search_type, search_fields, following_ids: following_ids) }, filter: filter_clauses.map { |clause| clause_to_query(clause, search_type, search_fields, following_ids: following_ids) }, } if account_exists if following params[:filter] << { terms: { id: following_ids } } elsif following_ids.any? params[:should] << { terms: { id: following_ids, boost: 0.5 } } end end if full_text_enabled && search_scope == :discoverable params[:filter] << { term: { discoverable: true } } params[:filter] << { term: { silenced: false } } end { bool: params } end private # Raise an exception if there are clauses that don't work with this search type. def check_search_type(search_type) [ @should_clauses, @must_not_clauses, @must_clauses, @filter_clauses, @order_clauses, ].each do |clauses| clauses.each do |clause| raise Mastodon::SyntaxError, "Unexpected clause for search type #{search_type}" if clause.respond_to?(:search_types) && clause.search_types.exclude?(search_type) end end end # Return any account ID related to a clause. def clause_to_account_id(clause) clause.term if clause.is_a?(PrefixClause) && %i(account_id mentions_ids).include?(clause.filter.to_sym) end def clause_to_query(clause, search_type, search_fields, account_id: nil, following_ids: nil) case clause when TermClause { multi_match: { type: 'most_fields', query: clause.term, fields: search_fields, operator: 'and' } } when PhraseClause { match_phrase: { text: { query: clause.phrase } } } when PrefixClause # Some prefix clauses yield queries that depend on the search type or account. filter = case clause.filter when :account_id_filter_placeholder case search_type when :accounts 'id' when :statuses 'account_id' else raise Mastodon::SyntaxError, "Unexpected search type for query: #{search_type}" end else clause.filter end term = case clause.term when :account_id_placeholder account_id when :following_ids_placeholder following_ids else clause.term end { clause.query => { filter => term } } when EmojiClause { term: { emojis: clause.shortcode } } when TagClause { term: { tags: clause.tag } } else raise Mastodon::SyntaxError, "Unexpected clause type for query: #{clause}" end end def clause_to_order(clause) case clause when PrefixClause { clause.term => clause.order } else raise Mastodon::SyntaxError, "Unexpected clause type for filter: #{clause}" end end end class Operator class << self def symbol(str) case str when '+' :must when '-' :must_not when nil :should else raise Mastodon::SyntaxError, "Unknown operator: #{str}" end end def filter_context_symbol(str) case str when '+', nil :filter when '-' :must_not else raise Mastodon::SyntaxError, "Unknown operator: #{str}" end end end end class TermClause attr_reader :prefix, :operator, :term def initialize(prefix, operator, term) @prefix = prefix @operator = Operator.symbol(operator) @term = term end end class PhraseClause attr_reader :prefix, :operator, :phrase def initialize(prefix, operator, phrase) @prefix = prefix @operator = Operator.symbol(operator) @phrase = phrase end end class EmojiClause attr_reader :prefix, :operator, :shortcode def initialize(prefix, operator, shortcode) @prefix = prefix @operator = Operator.filter_context_symbol(operator) @shortcode = shortcode end end class TagClause attr_reader :prefix, :operator, :tag def initialize(prefix, operator, tag) @prefix = prefix @operator = Operator.filter_context_symbol(operator) @tag = tag end end # If you add a new prefix here, make sure to add it to SearchQueryParser as well. class PrefixClause attr_reader :filter, :operator, :term, :order, :query, :search_types def initialize(prefix, operator, term) # These defaults may be modified by prefix-operator-specific initializers below. @query = :term @filter = prefix @term = term # Some prefixes don't apply to all search types. @search_types = %i(accounts statuses) @operator = Operator.filter_context_symbol(operator) case prefix when 'domain' initialize_is_local if TagManager.instance.local_domain?(term) when 'is' initialize_is(term) when 'has' initialize_has(term) when 'lang', 'visibility' @search_types = %i(statuses) when 'before', 'after' initialize_date_range(prefix, operator, term) when 'from', 'mentions', 'to' initialize_account(prefix, term) when 'scope' initialize_scope(operator, term) when 'sort' initialize_sort(operator, term) else raise Mastodon::SyntaxError end end private def initialize_is(term) case term when 'bot', 'group' # These apply to all search types. No action required. when 'local' initialize_is_local when 'local_only', 'reply', 'sensitive' @search_types = %i(statuses) else raise Mastodon::SyntaxError, "Unknown keyword for is: prefix: #{term}" end end # We can identify local objects by querying for objects that don't have a domain field. def initialize_is_local @operator = @operator == :filter ? :must_not : :filter @query = :exists @filter = :field @term = 'domain' end def initialize_has(term) @search_types = %i(statuses) case term when 'link', 'media', 'poll', 'warning', *(MediaAttachment.types.keys.reject { |t| t == 'unknown' }) # Pass all of these through. when 'cw', 'spoiler' @term = 'warning' else raise Mastodon::SyntaxError, "Unknown keyword for has: prefix: #{term}" end end def initialize_date_range(prefix, operator, term) raise Mastodon::SyntaxError, 'Operator not allowed for date range' unless operator.nil? @query = :range @filter = 'created_at' @term = { before: { lt: term }, after: { gt: term }, }[prefix.to_sym] or raise Mastodon::SyntaxError, "Unknown date range prefix: #{prefix}" end def initialize_account(prefix, term) @search_types = %i(statuses) @filter = { from: :account_id, mentions: :mentions_ids, to: :mentions_ids, }[prefix.to_sym] or raise Mastodon::SyntaxError, "Unknown account filter prefix: #{prefix}" username, domain = term.gsub(/\A@/, '').split('@') domain = nil if TagManager.instance.local_domain?(domain) account = Account.find_remote!(username, domain) @term = account.id end def initialize_scope(operator, term) raise Mastodon::SyntaxError, 'Operator not allowed for scope: prefix' unless operator.nil? case term when 'classic' @search_types = %i(statuses) @filter = 'searchable_by' @term = :account_id_placeholder when 'following' @query = :terms # This scope queries different fields depending on search context. @filter = :account_id_filter_placeholder @term = :following_ids_placeholder else raise Mastodon::SyntaxError, "Unknown scope: #{term}" end end def initialize_sort(operator, term) raise Mastodon::SyntaxError, 'Operator not allowed for sort: prefix' unless operator.nil? @operator = :order @term = :created_at @order = { oldest: :asc, newest: :desc, }[term.to_sym] or raise Mastodon::SyntaxError, "Unknown sort: #{term}" end end rule(clause: subtree(:clause)) do prefix = clause[:prefix]&.to_s operator = clause[:operator]&.to_s if clause[:prefix] PrefixClause.new(prefix, operator, clause[:term].to_s) elsif clause[:term] TermClause.new(prefix, operator, clause[:term].to_s) elsif clause[:shortcode] EmojiClause.new(prefix, operator, clause[:shortcode].to_s) elsif clause[:hashtag] TagClause.new(prefix, operator, clause[:hashtag].to_s) elsif clause[:phrase] PhraseClause.new(prefix, operator, clause[:phrase].to_s) else raise Mastodon::SyntaxError, "Unexpected clause type: #{clause}" end end rule(query: sequence(:clauses)) { Query.new(clauses) } end