* Add customizable user roles * Various fixes and improvements * Add migration for old settings and fix tootctl role management
		
			
				
	
	
		
			184 lines
		
	
	
	
		
			4.4 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			184 lines
		
	
	
	
		
			4.4 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
# frozen_string_literal: true
 | 
						|
 | 
						|
class AccountSearchService < BaseService
 | 
						|
  attr_reader :query, :limit, :offset, :options, :account
 | 
						|
 | 
						|
  def call(query, account = nil, options = {})
 | 
						|
    @acct_hint = query&.start_with?('@')
 | 
						|
    @query     = query&.strip&.gsub(/\A@/, '')
 | 
						|
    @limit     = options[:limit].to_i
 | 
						|
    @offset    = options[:offset].to_i
 | 
						|
    @options   = options
 | 
						|
    @account   = account
 | 
						|
 | 
						|
    search_service_results.compact.uniq
 | 
						|
  end
 | 
						|
 | 
						|
  private
 | 
						|
 | 
						|
  def search_service_results
 | 
						|
    return [] if query.blank? || limit < 1
 | 
						|
 | 
						|
    [exact_match] + search_results
 | 
						|
  end
 | 
						|
 | 
						|
  def exact_match
 | 
						|
    return unless offset.zero? && username_complete?
 | 
						|
 | 
						|
    return @exact_match if defined?(@exact_match)
 | 
						|
 | 
						|
    match = begin
 | 
						|
      if options[:resolve]
 | 
						|
        ResolveAccountService.new.call(query)
 | 
						|
      elsif domain_is_local?
 | 
						|
        Account.find_local(query_username)
 | 
						|
      else
 | 
						|
        Account.find_remote(query_username, query_domain)
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    match = nil if !match.nil? && !account.nil? && options[:following] && !account.following?(match)
 | 
						|
 | 
						|
    @exact_match = match
 | 
						|
  end
 | 
						|
 | 
						|
  def search_results
 | 
						|
    return [] if limit_for_non_exact_results.zero?
 | 
						|
 | 
						|
    @search_results ||= begin
 | 
						|
      results = from_elasticsearch if Chewy.enabled?
 | 
						|
      results ||= from_database
 | 
						|
      results
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def from_database
 | 
						|
    if account
 | 
						|
      advanced_search_results
 | 
						|
    else
 | 
						|
      simple_search_results
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def advanced_search_results
 | 
						|
    Account.advanced_search_for(terms_for_query, account, limit: limit_for_non_exact_results, following: options[:following], offset: offset)
 | 
						|
  end
 | 
						|
 | 
						|
  def simple_search_results
 | 
						|
    Account.search_for(terms_for_query, limit: limit_for_non_exact_results, offset: offset)
 | 
						|
  end
 | 
						|
 | 
						|
  def from_elasticsearch
 | 
						|
    must_clauses   = [{ multi_match: { query: terms_for_query, fields: likely_acct? ? %w(acct.edge_ngram acct) : %w(acct.edge_ngram acct display_name.edge_ngram display_name), type: 'most_fields', operator: 'and' } }]
 | 
						|
    should_clauses = []
 | 
						|
 | 
						|
    if account
 | 
						|
      return [] if options[:following] && following_ids.empty?
 | 
						|
 | 
						|
      if options[:following]
 | 
						|
        must_clauses << { terms: { id: following_ids } }
 | 
						|
      elsif following_ids.any?
 | 
						|
        should_clauses << { terms: { id: following_ids, boost: 100 } }
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    query     = { bool: { must: must_clauses, should: should_clauses } }
 | 
						|
    functions = [reputation_score_function, followers_score_function, time_distance_function]
 | 
						|
 | 
						|
    records = AccountsIndex.query(function_score: { query: query, functions: functions, boost_mode: 'multiply', score_mode: 'avg' })
 | 
						|
                           .limit(limit_for_non_exact_results)
 | 
						|
                           .offset(offset)
 | 
						|
                           .objects
 | 
						|
                           .compact
 | 
						|
 | 
						|
    ActiveRecord::Associations::Preloader.new.preload(records, :account_stat)
 | 
						|
 | 
						|
    records
 | 
						|
  rescue Faraday::ConnectionFailed, Parslet::ParseFailed
 | 
						|
    nil
 | 
						|
  end
 | 
						|
 | 
						|
  def reputation_score_function
 | 
						|
    {
 | 
						|
      script_score: {
 | 
						|
        script: {
 | 
						|
          source: "(doc['followers_count'].value + 0.0) / (doc['followers_count'].value + doc['following_count'].value + 1)",
 | 
						|
        },
 | 
						|
      },
 | 
						|
    }
 | 
						|
  end
 | 
						|
 | 
						|
  def followers_score_function
 | 
						|
    {
 | 
						|
      field_value_factor: {
 | 
						|
        field: 'followers_count',
 | 
						|
        modifier: 'log2p',
 | 
						|
        missing: 0,
 | 
						|
      },
 | 
						|
    }
 | 
						|
  end
 | 
						|
 | 
						|
  def time_distance_function
 | 
						|
    {
 | 
						|
      gauss: {
 | 
						|
        last_status_at: {
 | 
						|
          scale: '30d',
 | 
						|
          offset: '30d',
 | 
						|
          decay: 0.3,
 | 
						|
        },
 | 
						|
      },
 | 
						|
    }
 | 
						|
  end
 | 
						|
 | 
						|
  def following_ids
 | 
						|
    @following_ids ||= account.active_relationships.pluck(:target_account_id) + [account.id]
 | 
						|
  end
 | 
						|
 | 
						|
  def limit_for_non_exact_results
 | 
						|
    if exact_match?
 | 
						|
      limit - 1
 | 
						|
    else
 | 
						|
      limit
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def terms_for_query
 | 
						|
    if domain_is_local?
 | 
						|
      query_username
 | 
						|
    else
 | 
						|
      query
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def split_query_string
 | 
						|
    @split_query_string ||= query.split('@')
 | 
						|
  end
 | 
						|
 | 
						|
  def query_username
 | 
						|
    @query_username ||= split_query_string.first || ''
 | 
						|
  end
 | 
						|
 | 
						|
  def query_domain
 | 
						|
    @query_domain ||= query_without_split? ? nil : split_query_string.last
 | 
						|
  end
 | 
						|
 | 
						|
  def query_without_split?
 | 
						|
    split_query_string.size == 1
 | 
						|
  end
 | 
						|
 | 
						|
  def domain_is_local?
 | 
						|
    @domain_is_local ||= TagManager.instance.local_domain?(query_domain)
 | 
						|
  end
 | 
						|
 | 
						|
  def exact_match?
 | 
						|
    exact_match.present?
 | 
						|
  end
 | 
						|
 | 
						|
  def username_complete?
 | 
						|
    query.include?('@') && "@#{query}".match?(/\A#{Account::MENTION_RE}\Z/)
 | 
						|
  end
 | 
						|
 | 
						|
  def likely_acct?
 | 
						|
    @acct_hint || username_complete?
 | 
						|
  end
 | 
						|
end
 |