# frozen_string_literal: true class Importer::StatusesIndexImporter < Importer::BaseImporter def import! # The idea is that instead of iterating over all statuses in the database # and calculating the searchable_by for each of them (majority of which # would be empty), we approach the index from the other end scopes.each do |scope| # We could be tempted to keep track of status IDs we have already processed # from a different scope to avoid indexing them multiple times, but that # could end up being a very large array scope.find_in_batches(batch_size: @batch_size) do |tmp| in_work_unit(tmp.map(&:status_id)) do |status_ids| bulk = ActiveRecord::Base.connection_pool.with_connection do Chewy::Index::Import::BulkBuilder.new(index, to_index: Status.includes(:media_attachments, :preloadable_poll).where(id: status_ids)).bulk_body end indexed = 0 deleted = 0 # We can't use the delete_if proc to do the filtering because delete_if # is called before rendering the data and we need to filter based # on the results of the filter, so this filtering happens here instead bulk.map! do |entry| new_entry = begin if entry[:index] && entry.dig(:index, :data, 'searchable_by').blank? { delete: entry[:index].except(:data) } else entry end end if new_entry[:index] indexed += 1 else deleted += 1 end new_entry end Chewy::Index::Import::BulkRequest.new(index).perform(bulk) [indexed, deleted] end end end wait! end private def index StatusesIndex end def scopes classic_scopes = [ local_statuses_scope, local_mentions_scope, local_favourites_scope, local_votes_scope, local_bookmarks_scope, ] case Rails.configuration.x.search_scope when :public classic_scopes + [public_scope] when :public_or_unlisted classic_scopes + [public_or_unlisted_scope] else classic_scopes end end def local_mentions_scope Mention.where(account: Account.local, silent: false).select(:id, :status_id) end def local_favourites_scope Favourite.where(account: Account.local).select(:id, :status_id) end def local_bookmarks_scope Bookmark.select(:id, :status_id) end def local_votes_scope Poll.joins(:votes).where(votes: { account: Account.local }).select('polls.id, polls.status_id') end def local_statuses_scope Status.local.select('"statuses"."id", COALESCE("statuses"."reblog_of_id", "statuses"."id") AS status_id') end # The `id` field in the above queries isn't used anywhere, so we leave it out of these. def public_scope Status.with_public_visibility.select('"statuses"."id" AS status_id') end def public_or_unlisted_scope Status.with_public_or_unlisted_visibility.select('"statuses"."id" AS status_id') end end