# frozen_string_literal: true module Esse module Search class Query attr_reader :client_proxy, :definition # @param client_proxy [Esse::ClientProxy] The client proxy to use for the query # @param indices [] The class of the index to search or the index name # @param definition [Hash] The options to pass to the search. def initialize(client_proxy, *indices, suffix: nil, **definition, &_block) @client_proxy = client_proxy @definition = definition @definition[:index] = indices.map do |index| if index.is_a?(Class) && index < Esse::Index index.index_name(suffix: suffix) elsif index.is_a?(String) || index.is_a?(Symbol) [index, suffix].compact.join('_') else raise ArgumentError, format('Invalid index type: %p. It should be a Esse::Index class or a String index name', index: index) end end.join(',') end def response @response ||= execute_search_query! end def results return paginated_results if respond_to?(:paginated_results, true) response.hits end def scroll_hits(batch_size: 1_000, scroll: '1m') response = execute_search_query!(size: batch_size, scroll: scroll) scroll_id = nil fetched = 0 total = response.total loop do fetched += response.hits.size yield(response.hits) if response.hits.any? break if fetched >= total scroll_id = response.raw_response['scroll_id'] || response.raw_response['_scroll_id'] break unless scroll_id response = execute_scroll_query(scroll: scroll, scroll_id: scroll_id) end ensure begin client_proxy.clear_scroll(body: {scroll_id: scroll_id}) if scroll_id rescue Esse::Backend::NotFoundError end end private def execute_search_query!(**execution_options) resp, err = nil Esse::Events.instrument('elasticsearch.execute_search_query') do |payload| payload[:query] = self begin resp = Response.new(self, client_proxy.search(**definition, **execution_options)) rescue => e err = e end payload[:error] = err if err payload[:response] = resp end raise err if err resp end def execute_scroll_query(scroll:, scroll_id:) resp, err = nil Esse::Events.instrument('elasticsearch.execute_search_query') do |payload| payload[:query] = self begin resp = Response.new(self, client_proxy.scroll(scroll: scroll, body: { scroll_id: scroll_id })) rescue => e err = e end payload[:error] = err if err payload[:response] = resp end raise err if err resp end def reset! @response = nil end def raw_limit_value definition.dig(:body, :size) || definition.dig(:body, 'size') || definition.dig(:size) || definition.dig('size') end def raw_offset_value definition.dig(:body, :from) || definition.dig(:body, 'from') || definition.dig(:from) || definition.dig('from') end end end end