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.
- checksums.yaml +7 -0
- data/CHANGELOG.md +786 -0
- data/README.md +61 -0
- data/jay_api.gemspec +38 -0
- data/lib/jay_api/abstract/connection.rb +50 -0
- data/lib/jay_api/abstract/constant_wait.rb +17 -0
- data/lib/jay_api/abstract/geometric_wait.rb +35 -0
- data/lib/jay_api/abstract/wait_strategy.rb +43 -0
- data/lib/jay_api/configuration.rb +115 -0
- data/lib/jay_api/elasticsearch/async.rb +72 -0
- data/lib/jay_api/elasticsearch/batch_counter.rb +76 -0
- data/lib/jay_api/elasticsearch/client.rb +96 -0
- data/lib/jay_api/elasticsearch/client_factory.rb +100 -0
- data/lib/jay_api/elasticsearch/errors/elasticsearch_error.rb +13 -0
- data/lib/jay_api/elasticsearch/errors/end_of_query_results_error.rb +22 -0
- data/lib/jay_api/elasticsearch/errors/query_execution_error.rb +15 -0
- data/lib/jay_api/elasticsearch/errors/query_execution_failure.rb +17 -0
- data/lib/jay_api/elasticsearch/errors/query_execution_timeout.rb +13 -0
- data/lib/jay_api/elasticsearch/errors/search_after_error.rb +13 -0
- data/lib/jay_api/elasticsearch/index.rb +223 -0
- data/lib/jay_api/elasticsearch/query_builder/aggregations/aggregation.rb +66 -0
- data/lib/jay_api/elasticsearch/query_builder/aggregations/avg.rb +56 -0
- data/lib/jay_api/elasticsearch/query_builder/aggregations/errors/aggregations_error.rb +17 -0
- data/lib/jay_api/elasticsearch/query_builder/aggregations/errors.rb +14 -0
- data/lib/jay_api/elasticsearch/query_builder/aggregations/filter.rb +67 -0
- data/lib/jay_api/elasticsearch/query_builder/aggregations/max.rb +51 -0
- data/lib/jay_api/elasticsearch/query_builder/aggregations/scripted_metric.rb +72 -0
- data/lib/jay_api/elasticsearch/query_builder/aggregations/sum.rb +57 -0
- data/lib/jay_api/elasticsearch/query_builder/aggregations/terms.rb +73 -0
- data/lib/jay_api/elasticsearch/query_builder/aggregations/top_hits.rb +49 -0
- data/lib/jay_api/elasticsearch/query_builder/aggregations/value_count.rb +50 -0
- data/lib/jay_api/elasticsearch/query_builder/aggregations.rb +168 -0
- data/lib/jay_api/elasticsearch/query_builder/errors/query_builder_error.rb +16 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses/bool.rb +179 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses/exists.rb +33 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses/match_all.rb +22 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses/match_clauses.rb +140 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses/match_none.rb +22 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses/match_phrase.rb +35 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses/negator.rb +42 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses/query_clause.rb +17 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses/query_string.rb +50 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses/range.rb +49 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses/regexp.rb +39 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses/term.rb +37 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses/terms.rb +37 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses/wildcard.rb +37 -0
- data/lib/jay_api/elasticsearch/query_builder/query_clauses.rb +163 -0
- data/lib/jay_api/elasticsearch/query_builder/script.rb +36 -0
- data/lib/jay_api/elasticsearch/query_builder.rb +196 -0
- data/lib/jay_api/elasticsearch/query_results.rb +111 -0
- data/lib/jay_api/elasticsearch/response.rb +43 -0
- data/lib/jay_api/elasticsearch/search_after_results.rb +58 -0
- data/lib/jay_api/elasticsearch/tasks.rb +36 -0
- data/lib/jay_api/elasticsearch/time.rb +18 -0
- data/lib/jay_api/errors/configuration_error.rb +22 -0
- data/lib/jay_api/errors/error.rb +8 -0
- data/lib/jay_api/git/errors/invalid_repository_error.rb +11 -0
- data/lib/jay_api/git/errors/missing_url_error.rb +13 -0
- data/lib/jay_api/git/gerrit/gitiles_helper.rb +58 -0
- data/lib/jay_api/git/repository.rb +356 -0
- data/lib/jay_api/id_builder.rb +52 -0
- data/lib/jay_api/mergeables/merge_selector/configuration.rb +29 -0
- data/lib/jay_api/mergeables/merge_selector/merger.rb +58 -0
- data/lib/jay_api/mergeables/merge_selector.rb +15 -0
- data/lib/jay_api/prior_version_fetcher_base.rb +66 -0
- data/lib/jay_api/properties_fetcher.rb +196 -0
- data/lib/jay_api/rspec/configuration.rb +46 -0
- data/lib/jay_api/rspec/git.rb +60 -0
- data/lib/jay_api/rspec/test_data_collector.rb +189 -0
- data/lib/jay_api/rspec.rb +9 -0
- data/lib/jay_api/version.rb +6 -0
- data/lib/jay_api.rb +9 -0
- 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
|