jay_api 27.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (74) hide show
  1. checksums.yaml +7 -0
  2. data/CHANGELOG.md +786 -0
  3. data/README.md +61 -0
  4. data/jay_api.gemspec +38 -0
  5. data/lib/jay_api/abstract/connection.rb +50 -0
  6. data/lib/jay_api/abstract/constant_wait.rb +17 -0
  7. data/lib/jay_api/abstract/geometric_wait.rb +35 -0
  8. data/lib/jay_api/abstract/wait_strategy.rb +43 -0
  9. data/lib/jay_api/configuration.rb +115 -0
  10. data/lib/jay_api/elasticsearch/async.rb +72 -0
  11. data/lib/jay_api/elasticsearch/batch_counter.rb +76 -0
  12. data/lib/jay_api/elasticsearch/client.rb +96 -0
  13. data/lib/jay_api/elasticsearch/client_factory.rb +100 -0
  14. data/lib/jay_api/elasticsearch/errors/elasticsearch_error.rb +13 -0
  15. data/lib/jay_api/elasticsearch/errors/end_of_query_results_error.rb +22 -0
  16. data/lib/jay_api/elasticsearch/errors/query_execution_error.rb +15 -0
  17. data/lib/jay_api/elasticsearch/errors/query_execution_failure.rb +17 -0
  18. data/lib/jay_api/elasticsearch/errors/query_execution_timeout.rb +13 -0
  19. data/lib/jay_api/elasticsearch/errors/search_after_error.rb +13 -0
  20. data/lib/jay_api/elasticsearch/index.rb +223 -0
  21. data/lib/jay_api/elasticsearch/query_builder/aggregations/aggregation.rb +66 -0
  22. data/lib/jay_api/elasticsearch/query_builder/aggregations/avg.rb +56 -0
  23. data/lib/jay_api/elasticsearch/query_builder/aggregations/errors/aggregations_error.rb +17 -0
  24. data/lib/jay_api/elasticsearch/query_builder/aggregations/errors.rb +14 -0
  25. data/lib/jay_api/elasticsearch/query_builder/aggregations/filter.rb +67 -0
  26. data/lib/jay_api/elasticsearch/query_builder/aggregations/max.rb +51 -0
  27. data/lib/jay_api/elasticsearch/query_builder/aggregations/scripted_metric.rb +72 -0
  28. data/lib/jay_api/elasticsearch/query_builder/aggregations/sum.rb +57 -0
  29. data/lib/jay_api/elasticsearch/query_builder/aggregations/terms.rb +73 -0
  30. data/lib/jay_api/elasticsearch/query_builder/aggregations/top_hits.rb +49 -0
  31. data/lib/jay_api/elasticsearch/query_builder/aggregations/value_count.rb +50 -0
  32. data/lib/jay_api/elasticsearch/query_builder/aggregations.rb +168 -0
  33. data/lib/jay_api/elasticsearch/query_builder/errors/query_builder_error.rb +16 -0
  34. data/lib/jay_api/elasticsearch/query_builder/query_clauses/bool.rb +179 -0
  35. data/lib/jay_api/elasticsearch/query_builder/query_clauses/exists.rb +33 -0
  36. data/lib/jay_api/elasticsearch/query_builder/query_clauses/match_all.rb +22 -0
  37. data/lib/jay_api/elasticsearch/query_builder/query_clauses/match_clauses.rb +140 -0
  38. data/lib/jay_api/elasticsearch/query_builder/query_clauses/match_none.rb +22 -0
  39. data/lib/jay_api/elasticsearch/query_builder/query_clauses/match_phrase.rb +35 -0
  40. data/lib/jay_api/elasticsearch/query_builder/query_clauses/negator.rb +42 -0
  41. data/lib/jay_api/elasticsearch/query_builder/query_clauses/query_clause.rb +17 -0
  42. data/lib/jay_api/elasticsearch/query_builder/query_clauses/query_string.rb +50 -0
  43. data/lib/jay_api/elasticsearch/query_builder/query_clauses/range.rb +49 -0
  44. data/lib/jay_api/elasticsearch/query_builder/query_clauses/regexp.rb +39 -0
  45. data/lib/jay_api/elasticsearch/query_builder/query_clauses/term.rb +37 -0
  46. data/lib/jay_api/elasticsearch/query_builder/query_clauses/terms.rb +37 -0
  47. data/lib/jay_api/elasticsearch/query_builder/query_clauses/wildcard.rb +37 -0
  48. data/lib/jay_api/elasticsearch/query_builder/query_clauses.rb +163 -0
  49. data/lib/jay_api/elasticsearch/query_builder/script.rb +36 -0
  50. data/lib/jay_api/elasticsearch/query_builder.rb +196 -0
  51. data/lib/jay_api/elasticsearch/query_results.rb +111 -0
  52. data/lib/jay_api/elasticsearch/response.rb +43 -0
  53. data/lib/jay_api/elasticsearch/search_after_results.rb +58 -0
  54. data/lib/jay_api/elasticsearch/tasks.rb +36 -0
  55. data/lib/jay_api/elasticsearch/time.rb +18 -0
  56. data/lib/jay_api/errors/configuration_error.rb +22 -0
  57. data/lib/jay_api/errors/error.rb +8 -0
  58. data/lib/jay_api/git/errors/invalid_repository_error.rb +11 -0
  59. data/lib/jay_api/git/errors/missing_url_error.rb +13 -0
  60. data/lib/jay_api/git/gerrit/gitiles_helper.rb +58 -0
  61. data/lib/jay_api/git/repository.rb +356 -0
  62. data/lib/jay_api/id_builder.rb +52 -0
  63. data/lib/jay_api/mergeables/merge_selector/configuration.rb +29 -0
  64. data/lib/jay_api/mergeables/merge_selector/merger.rb +58 -0
  65. data/lib/jay_api/mergeables/merge_selector.rb +15 -0
  66. data/lib/jay_api/prior_version_fetcher_base.rb +66 -0
  67. data/lib/jay_api/properties_fetcher.rb +196 -0
  68. data/lib/jay_api/rspec/configuration.rb +46 -0
  69. data/lib/jay_api/rspec/git.rb +60 -0
  70. data/lib/jay_api/rspec/test_data_collector.rb +189 -0
  71. data/lib/jay_api/rspec.rb +9 -0
  72. data/lib/jay_api/version.rb +6 -0
  73. data/lib/jay_api.rb +9 -0
  74. metadata +215 -0
@@ -0,0 +1,37 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'query_clause'
4
+
5
+ module JayAPI
6
+ module Elasticsearch
7
+ class QueryBuilder
8
+ class QueryClauses
9
+ # Represents a Terms query in Elasticsearch.
10
+ # Information about this type of query can be found here:
11
+ # https://www.elastic.co/guide/en/elasticsearch/reference/current/query-dsl-terms-query.html
12
+ class Terms < ::JayAPI::Elasticsearch::QueryBuilder::QueryClauses::QueryClause
13
+ attr_reader :field, :terms
14
+
15
+ # @param [String, Symbol] field The name of the field to search.
16
+ # @param [Array<String>] terms The array of terms to search for.
17
+ def initialize(field:, terms:)
18
+ super()
19
+
20
+ @field = field
21
+ @terms = terms
22
+ end
23
+
24
+ # @return [Hash] The Hash that represents this query (in
25
+ # Elasticsearch's DSL)
26
+ def to_h
27
+ {
28
+ terms: {
29
+ field => terms
30
+ }
31
+ }
32
+ end
33
+ end
34
+ end
35
+ end
36
+ end
37
+ end
@@ -0,0 +1,37 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'query_clause'
4
+
5
+ module JayAPI
6
+ module Elasticsearch
7
+ class QueryBuilder
8
+ class QueryClauses
9
+ # Represents a Wildcard query in Elasticsearch
10
+ # Documentation for this type of query can be found here:
11
+ # https://www.elastic.co/guide/en/elasticsearch/reference/current/query-dsl-wildcard-query.html
12
+ class Wildcard < ::JayAPI::Elasticsearch::QueryBuilder::QueryClauses::QueryClause
13
+ attr_accessor :field, :value
14
+
15
+ # @param [String, Symbol] field The name of the field to search
16
+ # @param [String] value The wildcard pattern.
17
+ def initialize(field:, value:)
18
+ @field = field
19
+ @value = value
20
+ end
21
+
22
+ # @return [Hash] The Hash that represents this query (in
23
+ # Elasticsearch's format)
24
+ def to_h
25
+ {
26
+ wildcard: {
27
+ "#{field}": {
28
+ value: value
29
+ }
30
+ }
31
+ }
32
+ end
33
+ end
34
+ end
35
+ end
36
+ end
37
+ end
@@ -0,0 +1,163 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'forwardable'
4
+
5
+ require_relative 'errors/query_builder_error'
6
+ require_relative 'query_clauses/bool'
7
+ require_relative 'query_clauses/match_clauses'
8
+ require_relative 'query_clauses/negator'
9
+
10
+ module JayAPI
11
+ module Elasticsearch
12
+ class QueryBuilder
13
+ # Represents the set of query clauses in an Elasticsearch query.
14
+ # An empty set of clauses produces a "match all" query clause.
15
+ class QueryClauses
16
+ extend Forwardable
17
+
18
+ include ::JayAPI::Elasticsearch::QueryBuilder::QueryClauses::MatchClauses
19
+
20
+ def_delegator :top_level_clause, :nil?, :empty?
21
+
22
+ # Turns the query into a Compound Boolean query by adding a +bool+
23
+ # clause and yields the latter so that sub-clauses can be added to it.
24
+ # If the query is already a boolean query the current boolean clause is
25
+ # yielded.
26
+ # @raise [JayAPI::Elasticsearch::QueryBuilder::Errors::QueryBuilderError]
27
+ # If the query already has a top-level query.
28
+ # @yield [JayAPI::Elasticsearch::QueryBuilder::QueryClauses::Bool]
29
+ # Yields the +bool+ query clause to the given block (if there is any).
30
+ # @return [self, JayAPI::Elasticsearch::QueryBuilder::QueryClauses::Bool]
31
+ # If a block is given then +self+ is returned, if no block is given
32
+ # then the +bool+ query clause is returned.
33
+ def bool
34
+ clause ||= boolean_clause || ::JayAPI::Elasticsearch::QueryBuilder::QueryClauses::Bool.new
35
+ replace_top_level_clause(clause, force: boolean_query?)
36
+
37
+ if block_given?
38
+ yield clause
39
+ self
40
+ else
41
+ clause
42
+ end
43
+ end
44
+
45
+ # Adds the given query clause as top-level clause if none exists yet.
46
+ # @param [JayAPI::Elasticsearch::QueryBuilder::QueryClauses::QueryClause]
47
+ # query_clause The query clause to add.
48
+ # @return [JayAPI::Elasticsearch::QueryBuilder::QueryClauses] Returns
49
+ # itself so that other methods can be chained.
50
+ # @raise [JayAPI::Elasticsearch::QueryBuilder::Errors::QueryBuilderError]
51
+ # If there is already a top level clause.
52
+ def <<(query_clause)
53
+ replace_top_level_clause(query_clause)
54
+ end
55
+
56
+ # @return [Hash] The Hash representation of the Query Clauses set.
57
+ # @raise [JayAPI::Elasticsearch::QueryBuilder::Errors::QueryBuilderError]
58
+ # If a boolean query was created but no inner clauses were added.
59
+ def to_h
60
+ return self.class.new.match_all.to_h if empty?
61
+
62
+ top_level_clause.to_h
63
+ end
64
+
65
+ # @return [Boolean] True if the current Query Clauses set includes a
66
+ # +bool+ clause, false otherwise.
67
+ def boolean_query?
68
+ top_level_clause.is_a?(::JayAPI::Elasticsearch::QueryBuilder::QueryClauses::Bool)
69
+ end
70
+
71
+ # Clones the receiver and the enclosed top-level clause (if any).
72
+ # @return [self] A copy of the receiver.
73
+ def clone
74
+ self.class.new.tap do |copy|
75
+ copy << top_level_clause.clone
76
+ end
77
+ end
78
+
79
+ # Creates a new +QueryClauses+ object by merging the receiver with the
80
+ # given object. The individual top-query clauses are merged together
81
+ # using a boolean clause.
82
+ # @param [self] other The +QueryClauses+ object the receiver should be
83
+ # merged with.
84
+ # @return [self] A new +QueryClauses+ object which is a combination of
85
+ # the receiver and the given object.
86
+ def merge(other)
87
+ klass = self.class
88
+ raise TypeError, "Cannot merge #{klass} with #{other.class}" unless other.is_a?(klass)
89
+
90
+ if other.empty?
91
+ clone
92
+ elsif empty?
93
+ other.clone
94
+ else
95
+ klass.new.tap do |merged|
96
+ merged.bool.merge!(top_level_clause).merge!(other.top_level_clause)
97
+ end
98
+ end
99
+ end
100
+
101
+ # Negates the receiver by wrapping its top-level query clause in a
102
+ # +must_not+ boolean clause or replacing it by its inverse clause.
103
+ # @return [self] Returns itself.
104
+ def negate!
105
+ if top_level_clause
106
+ @top_level_clause = ::JayAPI::Elasticsearch::QueryBuilder::QueryClauses::Negator.new(top_level_clause)
107
+ .negate
108
+ else
109
+ match_none
110
+ end
111
+
112
+ self
113
+ end
114
+
115
+ # @return [self] A negated version of the receiver (with its top-level
116
+ # query clause wrapped in a +must_not+ boolean query or replaced by
117
+ # its inverse clause)
118
+ def negate
119
+ clone.negate!
120
+ end
121
+
122
+ protected
123
+
124
+ attr_reader :top_level_clause
125
+
126
+ private
127
+
128
+ # Replaces the current top-level clause with the given clause.
129
+ # @param [JayAPI::Elasticsearch::QueryBuilder::QueryClauses::QueryClause]
130
+ # query_clause The query clause to add.
131
+ # @param [Boolean] force Forces the replacement of the top-level clause
132
+ # even if there is already a top-level clause in place.
133
+ # @return [JayAPI::Elasticsearch::QueryBuilder::QueryClauses] Returns
134
+ # itself so that other methods can be chained.
135
+ # @raise [JayAPI::Elasticsearch::QueryBuilder::Errors::QueryBuilderError]
136
+ # If there is already a top level clause and +force+ is +false+.
137
+ def replace_top_level_clause(query_clause, force: false)
138
+ single_top_level_clause! unless force
139
+ @top_level_clause = query_clause
140
+ self
141
+ end
142
+
143
+ # @return [JayAPI::Elasticsearch::QueryBuilder::QueryClauses::Bool, nil]
144
+ # The current top-level clause if it is a Boolean clause, +nil+
145
+ # otherwise.
146
+ def boolean_clause
147
+ boolean_query? ? top_level_clause : nil
148
+ end
149
+
150
+ # @raise [JayAPI::Elasticsearch::QueryBuilder::Errors::QueryBuilderError]
151
+ # If there is already a top level clause.
152
+ def single_top_level_clause!
153
+ return unless top_level_clause
154
+
155
+ raise ::JayAPI::Elasticsearch::QueryBuilder::Errors::QueryBuilderError,
156
+ 'Queries can only have one top-level query clause, ' \
157
+ 'to use multiple clauses add a compound query, ' \
158
+ 'for example: `bool`'
159
+ end
160
+ end
161
+ end
162
+ end
163
+ end
@@ -0,0 +1,36 @@
1
+ # frozen_string_literal: true
2
+
3
+ module JayAPI
4
+ module Elasticsearch
5
+ class QueryBuilder
6
+ # Represents a scripted element in a query. This scripted element can be
7
+ # used in different places. It can be used in a query clause, but can
8
+ # also be used to create custom aggregations.
9
+ class Script
10
+ attr_reader :source, :lang, :params
11
+
12
+ # @param [String] source The source for the script element.
13
+ # @param [String] lang The language the script is written in.
14
+ # @param [Hash] params A +Hash+ with key-value pairs for the script's
15
+ # parameters.
16
+ def initialize(source:, lang: 'painless', params: nil)
17
+ @source = source
18
+ @lang = lang
19
+
20
+ # Keeps the parameters from being modified from the outside after the
21
+ # class has been initialized.
22
+ @params = params.dup.freeze
23
+ end
24
+
25
+ # @return [Hash] The hash representation of the scripted element.
26
+ def to_h
27
+ {
28
+ source: source,
29
+ lang: lang,
30
+ params: params
31
+ }.compact
32
+ end
33
+ end
34
+ end
35
+ end
36
+ end
@@ -0,0 +1,196 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'query_builder/aggregations'
4
+ require_relative 'query_builder/query_clauses'
5
+ require_relative 'query_builder/script'
6
+
7
+ module JayAPI
8
+ module Elasticsearch
9
+ # A helper class to build simple and common queries for Elasticsearch.
10
+ # Queries are created with the Elasticsearch Query DSL:
11
+ # https://www.elastic.co/guide/en/elasticsearch/reference/current/query-dsl.html
12
+ class QueryBuilder
13
+ # @return [JayAPI::Elasticsearch::QueryBuilder::Aggregations]
14
+ attr_reader :aggregations
15
+
16
+ # @return [JayAPI::Elasticsearch::QueryBuilder::QueryClauses] The current
17
+ # set of query clauses
18
+ attr_reader :query
19
+
20
+ # Creates a new instance of the class.
21
+ # A new instance of the class will produce an empty query.
22
+ def initialize
23
+ @from = nil
24
+ @size = nil
25
+ @source = nil
26
+ @sort = {}
27
+ @collapse = nil
28
+ @query = ::JayAPI::Elasticsearch::QueryBuilder::QueryClauses.new
29
+ @aggregations = JayAPI::Elasticsearch::QueryBuilder::Aggregations.new
30
+ end
31
+
32
+ # Adds a +from+ clause to the query.
33
+ # @param [Integer] from The value for the from clause.
34
+ # @return [QueryBuilder] itself so that other methods can be chained.
35
+ def from(from)
36
+ check_argument(from, 'from', Integer)
37
+ check_positive_argument(from, 'from')
38
+ @from = from
39
+ self
40
+ end
41
+
42
+ # Adds a +size+ clause to the query.
43
+ # @param [Integer] size The value for the size clause.
44
+ # @return [QueryBuilder] itself so that other methods can be chained.
45
+ def size(size)
46
+ check_argument(size, 'size', Integer)
47
+ check_positive_argument(size, 'size')
48
+ @size = size
49
+ self
50
+ end
51
+
52
+ # Adds a +sort+ clause to the query.
53
+ # This method can be called with multiple fields at once or called
54
+ # multiple times.
55
+ #
56
+ # Example:
57
+ #
58
+ # query_builder.sort(name: 'asc', age: 'desc')
59
+ #
60
+ # or
61
+ #
62
+ # query_builder.sort(name: 'asc')
63
+ # query_builder.sort(age: 'desc')
64
+ #
65
+ # Both will produce the same +sort+ clause.
66
+ # @param [Hash] sort A Hash whose keys are the name of the fields
67
+ # and the keys are the direction of the sorting, either +asc+ or
68
+ # +desc+.
69
+ # @return [QueryBuilder] itself so that other methods can be chained.
70
+ def sort(sort)
71
+ check_argument(sort, 'sort', Hash)
72
+ @sort.merge!(sort)
73
+ self
74
+ end
75
+
76
+ # Adds a +collapse+ clause to the query.
77
+ # @param [String] field The field to use for collapsing the results.
78
+ # @return [QueryBuilder] itself so that other methods can be chained.
79
+ def collapse(field)
80
+ check_argument(field, 'field', String)
81
+ @collapse = field
82
+ self
83
+ end
84
+
85
+ # Adds a +_source+ clause to the query.
86
+ # @param [String] filter_expr Expression used for filtering source.
87
+ # @return [QueryBuilder] itself so that other methods can be chained.
88
+ def source(filter_expr)
89
+ check_argument(filter_expr, 'source', String)
90
+ @source = filter_expr
91
+ self
92
+ end
93
+
94
+ # @return [Hash] The generated query.
95
+ def to_h
96
+ build_query
97
+ end
98
+
99
+ alias to_query to_h
100
+
101
+ # Returns a new +QueryBuilder+ object which is the result of merging the
102
+ # receiver with +other+.
103
+ # @param [self] other Another instance of +QueryBuilder+.
104
+ # @return [self] A new +QueryBuilder+, the result of the merge of both
105
+ # objects.
106
+ # @raise [TypeError] If the given object is not a +QueryBuilder+.
107
+ def merge(other)
108
+ klass = self.class
109
+ raise TypeError, "Cannot merge #{klass} and #{other.class}" unless other.is_a?(klass)
110
+
111
+ other.combine(
112
+ from: @from, size: @size, source: @source, collapse: @collapse,
113
+ sort: @sort, query: @query, aggregations: @aggregations
114
+ )
115
+ end
116
+
117
+ protected
118
+
119
+ attr_writer :from, :size, :source, :collapse, :sort, :query, :aggregations
120
+
121
+ # Creates a new +QueryBuilder+ object whose attributes are a combination
122
+ # of the receiver's attributes and the provided values. The receiver's
123
+ # attributes take precedence over the given ones.
124
+ # @param [Integer, nil] from See {#from}
125
+ # @param [Integer, nil] size See {#size}
126
+ # @param [String, nil] source See {#source}
127
+ # @param [String, nil] collapse See {#collapse}
128
+ # @param [Hash] sort See {#sort}
129
+ # @param [JayAPI::Elasticsearch::QueryBuilder::QueryClauses] query See {#query}
130
+ # @param [JayAPI::Elasticsearch::QueryBuilder::Aggregations] aggregations See {#aggregations}
131
+ # @return [self] A new +QueryBuilder+ object.
132
+ def combine(from:, size:, source:, collapse:, sort:, query:, aggregations:)
133
+ self.class.new.tap do |combined|
134
+ combined.from = @from || from
135
+ combined.size = @size || size
136
+ # TODO: Improve the merging of this kind of clause (https://esrlabs.atlassian.net/browse/JAY-495)
137
+ combined.source = @source || source
138
+ combined.collapse = @collapse || collapse
139
+ combined.sort = sort.merge(@sort)
140
+ combined.query = query.merge(self.query)
141
+ combined.aggregations = aggregations.merge(self.aggregations)
142
+ end
143
+ end
144
+
145
+ private
146
+
147
+ # :reek:FeatureEnvy (cannot be avoided, is checking the argument)
148
+ # Checks that the given argument is an instance of the specified class.
149
+ # @param [Object] value The value of the argument.
150
+ # @param [String] argument_name The name of the argument (for the error
151
+ # message).
152
+ # @param [Class] expected_type The expected type of +value+
153
+ # @raise [ArgumentError] If the value is not an instance of the class.
154
+ def check_argument(value, argument_name, expected_type)
155
+ return if value.is_a?(expected_type)
156
+
157
+ raise ArgumentError, "Expected `#{argument_name}` to be #{expected_type}, #{value.class} given"
158
+ end
159
+
160
+ # Checks that the given argument is positive (>= 0)
161
+ # @param [Numeric] value The value of the argument.
162
+ # @param [String] argument_name The name of the argument (for the error
163
+ # message).
164
+ # @raise [ArgumentError] If the value is not positive.
165
+ def check_positive_argument(value, argument_name)
166
+ return if value >= 0
167
+
168
+ raise ArgumentError, "`#{argument_name}` should be a positive integer"
169
+ end
170
+
171
+ # Builds the query.
172
+ # @return [Hash] The Elasticsearch DSL Query.
173
+ def build_query
174
+ query_hash = {}
175
+ query_hash[:from] = @from if @from
176
+ query_hash[:size] = @size if @size
177
+ query_hash[:_source] = @source if @source
178
+ query_hash[:query] = query.to_h
179
+
180
+ if @sort.any?
181
+ query_hash[:sort] = @sort.map do |field, direction|
182
+ { field => { order: direction } }
183
+ end
184
+ end
185
+
186
+ if @collapse
187
+ query_hash[:collapse] = {
188
+ field: @collapse
189
+ }
190
+ end
191
+
192
+ query_hash.merge(aggregations.to_h)
193
+ end
194
+ end
195
+ end
196
+ end
@@ -0,0 +1,111 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'active_support'
4
+ require 'active_support/core_ext/hash/indifferent_access'
5
+ require 'forwardable'
6
+
7
+ require_relative 'errors/end_of_query_results_error'
8
+ require_relative 'batch_counter'
9
+
10
+ module JayAPI
11
+ module Elasticsearch
12
+ # Represents the results of an Elasticsearch query.
13
+ # It provides a facade in front of the returned Hash and allows more
14
+ # results to be fetched dynamically.
15
+ class QueryResults
16
+ extend Forwardable
17
+
18
+ attr_reader :index, :query, :response, :batch_counter
19
+
20
+ def_delegators :batch_counter, :batch_size, :start_next
21
+ def_delegators :response, :hits, :total, :size, :count, :first, :last, :any?, :empty?, :aggregations
22
+
23
+ # Creates a new instance of the class.
24
+ # @param [JayAPI::Elasticsearch::Index] index The Elasticsearch
25
+ # index used to perform the query.
26
+ # @param [Hash] query The query that produced the results.
27
+ # @param [JayAPI::Elasticsearch::Results] response An object containing Docs retrieved from Elasticsearch.
28
+ # @param [JayAPI::Elasticsearch::BatchCounter] batch_counter An object keeping track of the current batch.
29
+ def initialize(index:, query:, response:, batch_counter: nil)
30
+ @index = index
31
+ @query = query.with_indifferent_access
32
+ @response = response
33
+ @batch_counter = batch_counter
34
+ end
35
+
36
+ # @return [Boolean] True if there are still more documents matched by the
37
+ # query and a call to next_batch can be performed.
38
+ def more?
39
+ start_next < total
40
+ end
41
+
42
+ # Calls the given block for every document in the QueryResults object or
43
+ # returns an Enumerator with all the documents if no block is given.
44
+ # @yield [Hash] Each document in the current QueryResults object.
45
+ # @return [Enumerator, Array] An enumerator with all the objects in the
46
+ # QueryResults object if no block is given, or an array of all the
47
+ # documents in the QueryResults object.
48
+ def each(&block)
49
+ hits.each(&block)
50
+ end
51
+
52
+ # Allows the entire set of documents to be iterated in batches.
53
+ #
54
+ # - If the method is invoked with a block, the given block will be called
55
+ # for every document in the +QueryResults+ object. Upon reaching the
56
+ # end of the collection the next batch will be requested and the block
57
+ # will be called again for each of the documents in the next batch, the
58
+ # process will continue until there are no more documents. At the end,
59
+ # the last batch of documents will be returned.
60
+ #
61
+ # - If the method is called without a block an +Enumerator+ object will
62
+ # be returned. Said +Enumerator+ can be used to iterate through the
63
+ # whole set of documents. The +#all+ method will take care of fetching
64
+ # them in batches and yielding them to the enumerator.
65
+ #
66
+ # @yield [Hash] Each document in the current QueryResults object.
67
+ # @return [JayAPI::Elasticsearch::QueryResults, Enumerator] If a block is
68
+ # given the object with the last batch of documents (can be the receiver
69
+ # if there is only one batch) will be returned. If no block is given
70
+ # an +Enumerator+ will be returned.
71
+ def all(&block)
72
+ return enum_for(:all) { total - start_current } unless block
73
+
74
+ data = self
75
+
76
+ loop do
77
+ data.each(&block)
78
+ break unless data.more? && data.any?
79
+
80
+ data = data.next_batch
81
+ end
82
+
83
+ data
84
+ end
85
+
86
+ # Fetches the next batch of documents.
87
+ # @return [JayAPI::Elasticsearch::QueryResults] A new instance of the
88
+ # QueryResults that contains the next batch of documents fetched from
89
+ # Elasticsearch.
90
+ # @raise [Elasticsearch::Transport::Transport::ServerError] If the
91
+ # query fails.
92
+ def next_batch
93
+ raise Errors::EndOfQueryResultsError unless more?
94
+
95
+ modified_query = adapt_query
96
+ index.search(modified_query, batch_counter: batch_counter)
97
+ end
98
+
99
+ private
100
+
101
+ def_delegators :batch_counter, :start_current
102
+
103
+ def adapt_query
104
+ query.dup.tap do |modified_query|
105
+ modified_query[:size] = batch_size
106
+ modified_query[:from] = start_next
107
+ end
108
+ end
109
+ end
110
+ end
111
+ end
@@ -0,0 +1,43 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'forwardable'
4
+
5
+ module JayAPI
6
+ module Elasticsearch
7
+ # The `Response` class encapsulates and processes the results received
8
+ # from an Elasticsearch query. It provides a uniform interface for accessing
9
+ # and working with the retrieved data.
10
+ class Response
11
+ extend Forwardable
12
+
13
+ # @!attribute [r] raw_response
14
+ # @return [Hash] The raw results data returned from Elasticsearch
15
+ attr_reader :raw_response
16
+
17
+ def_delegators :hits, :size, :count, :first, :last, :any?, :empty?
18
+
19
+ # @param [Hash] raw_response The raw results data from Elasticsearch
20
+ def initialize(raw_response)
21
+ @raw_response = raw_response
22
+ end
23
+
24
+ # @return [Hash, nil] The aggregations present in the current result set
25
+ # (if there are any).
26
+ def aggregations
27
+ @aggregations ||= raw_response['aggregations']
28
+ end
29
+
30
+ # The actual "hits" results from the Elasticsearch response
31
+ # @return [Array<Hash>]
32
+ def hits
33
+ @hits ||= raw_response.dig('hits', 'hits') || []
34
+ end
35
+
36
+ # The total count of results that match the query criteria
37
+ # @return [Integer]
38
+ def total
39
+ @total ||= raw_response.dig('hits', 'total', 'value') || hits.size
40
+ end
41
+ end
42
+ end
43
+ end
@@ -0,0 +1,58 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'errors/search_after_error'
4
+ require_relative 'query_results'
5
+
6
+ module JayAPI
7
+ module Elasticsearch
8
+ # A QueryResults class for the 'search_after' type of query.
9
+ # See more: https://www.elastic.co/guide/en/elasticsearch/reference/current/paginate-search-results.html
10
+ class SearchAfterResults < QueryResults
11
+ # The default 'from' attribute for the 'search_after' query. See the link for more details.
12
+ # https://www.elastic.co/guide/en/elasticsearch/reference/current/paginate-search-results.html#:~:text=(default)%20or-,%2D1,-.
13
+ DEFAULT_FROM = -1
14
+
15
+ # @return [true] It should always be assumed that there are more results when
16
+ # using 'search_after' parameter
17
+ def more?
18
+ true
19
+ end
20
+
21
+ # Fetches the next batch of documents.
22
+ # @return [JayAPI::Elasticsearch::QueryResults] A new instance of the
23
+ # QueryResults that contains the next batch of documents fetched from
24
+ # Elasticsearch.
25
+ def next_batch
26
+ index.search(adapt_query, batch_counter: batch_counter, type: :search_after)
27
+ end
28
+
29
+ private
30
+
31
+ # @raise [JayAPI::Elasticsearch::Errors::SearchAfterError]
32
+ def raise_sort
33
+ raise(
34
+ JayAPI::Elasticsearch::Errors::SearchAfterError,
35
+ "'sort' attribute must be specified in the query when using 'search_after' parameter"
36
+ )
37
+ end
38
+
39
+ # @return [String] the 'sort' attribute of the last Doc hash in the batch.
40
+ # @raise [JayAPI::Elasticsearch::Errors::SearchAfterError] If 'sort' is not found in the Doc.
41
+ def sort
42
+ @sort ||= last['sort'] || raise_sort
43
+ end
44
+
45
+ # Adapts the query for the next batch.
46
+ # * The 'from' attribute must be set to a special value.
47
+ # * The 'search_after' attribute must contain the 'sort' attribute of the
48
+ # last received Doc.
49
+ # @return [Hash]
50
+ def adapt_query
51
+ super.tap do |modified_query|
52
+ modified_query[:from] = DEFAULT_FROM
53
+ modified_query[:search_after] = sort
54
+ end
55
+ end
56
+ end
57
+ end
58
+ end