chewy 5.2.0 → 6.0.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (103) hide show
  1. checksums.yaml +4 -4
  2. data/.circleci/config.yml +36 -10
  3. data/CHANGELOG.md +219 -326
  4. data/Gemfile +1 -1
  5. data/README.md +9 -21
  6. data/chewy.gemspec +1 -1
  7. data/gemfiles/rails.6.1.activerecord.gemfile +1 -1
  8. data/lib/chewy/config.rb +2 -20
  9. data/lib/chewy/fields/base.rb +0 -6
  10. data/lib/chewy/index.rb +2 -0
  11. data/lib/chewy/index/actions.rb +5 -1
  12. data/lib/chewy/multi_search.rb +62 -0
  13. data/lib/chewy/search.rb +1 -4
  14. data/lib/chewy/search/pagination/will_paginate.rb +1 -1
  15. data/lib/chewy/search/parameters/none.rb +1 -3
  16. data/lib/chewy/search/request.rb +25 -24
  17. data/lib/chewy/search/scrolling.rb +4 -1
  18. data/lib/chewy/type.rb +4 -1
  19. data/lib/chewy/type/adapter/orm.rb +6 -3
  20. data/lib/chewy/type/import.rb +1 -0
  21. data/lib/chewy/type/syncer.rb +4 -5
  22. data/lib/chewy/type/witchcraft.rb +3 -1
  23. data/lib/chewy/type/wrapper.rb +11 -1
  24. data/lib/chewy/version.rb +1 -1
  25. data/migration_guide.md +18 -0
  26. data/spec/chewy/config_spec.rb +0 -21
  27. data/spec/chewy/index/actions_spec.rb +24 -0
  28. data/spec/chewy/index_spec.rb +16 -39
  29. data/spec/chewy/journal_spec.rb +21 -17
  30. data/spec/chewy/minitest/search_index_receiver_spec.rb +11 -9
  31. data/spec/chewy/multi_search_spec.rb +85 -0
  32. data/spec/chewy/rake_helper_spec.rb +102 -87
  33. data/spec/chewy/rspec/update_index_spec.rb +47 -46
  34. data/spec/chewy/runtime_spec.rb +2 -2
  35. data/spec/chewy/search/parameters/indices_spec.rb +6 -7
  36. data/spec/chewy/search/parameters/none_spec.rb +1 -1
  37. data/spec/chewy/search/parameters_spec.rb +1 -1
  38. data/spec/chewy/search/request_spec.rb +82 -67
  39. data/spec/chewy/search/response_spec.rb +19 -15
  40. data/spec/chewy/search/scrolling_spec.rb +25 -16
  41. data/spec/chewy/search_spec.rb +45 -41
  42. data/spec/chewy/stash_spec.rb +14 -12
  43. data/spec/chewy/type/adapter/active_record_spec.rb +13 -1
  44. data/spec/chewy/type/import/bulk_builder_spec.rb +9 -94
  45. data/spec/chewy/type/import/journal_builder_spec.rb +9 -7
  46. data/spec/chewy/type/mapping_spec.rb +3 -1
  47. data/spec/chewy/type/witchcraft_spec.rb +15 -0
  48. data/spec/chewy/type/wrapper_spec.rb +3 -1
  49. data/spec/chewy_spec.rb +0 -7
  50. data/spec/spec_helper.rb +0 -7
  51. data/spec/support/active_record.rb +20 -0
  52. metadata +8 -78
  53. data/LEGACY_DSL.md +0 -497
  54. data/gemfiles/ruby3.gemfile +0 -10
  55. data/lib/chewy/query.rb +0 -1142
  56. data/lib/chewy/query/compose.rb +0 -68
  57. data/lib/chewy/query/criteria.rb +0 -191
  58. data/lib/chewy/query/filters.rb +0 -244
  59. data/lib/chewy/query/loading.rb +0 -110
  60. data/lib/chewy/query/nodes/and.rb +0 -25
  61. data/lib/chewy/query/nodes/base.rb +0 -17
  62. data/lib/chewy/query/nodes/bool.rb +0 -34
  63. data/lib/chewy/query/nodes/equal.rb +0 -34
  64. data/lib/chewy/query/nodes/exists.rb +0 -20
  65. data/lib/chewy/query/nodes/expr.rb +0 -28
  66. data/lib/chewy/query/nodes/field.rb +0 -110
  67. data/lib/chewy/query/nodes/has_child.rb +0 -15
  68. data/lib/chewy/query/nodes/has_parent.rb +0 -15
  69. data/lib/chewy/query/nodes/has_relation.rb +0 -59
  70. data/lib/chewy/query/nodes/match_all.rb +0 -11
  71. data/lib/chewy/query/nodes/missing.rb +0 -20
  72. data/lib/chewy/query/nodes/not.rb +0 -25
  73. data/lib/chewy/query/nodes/or.rb +0 -25
  74. data/lib/chewy/query/nodes/prefix.rb +0 -19
  75. data/lib/chewy/query/nodes/query.rb +0 -20
  76. data/lib/chewy/query/nodes/range.rb +0 -63
  77. data/lib/chewy/query/nodes/raw.rb +0 -15
  78. data/lib/chewy/query/nodes/regexp.rb +0 -35
  79. data/lib/chewy/query/nodes/script.rb +0 -20
  80. data/lib/chewy/query/pagination.rb +0 -25
  81. data/spec/chewy/query/criteria_spec.rb +0 -700
  82. data/spec/chewy/query/filters_spec.rb +0 -201
  83. data/spec/chewy/query/loading_spec.rb +0 -124
  84. data/spec/chewy/query/nodes/and_spec.rb +0 -12
  85. data/spec/chewy/query/nodes/bool_spec.rb +0 -14
  86. data/spec/chewy/query/nodes/equal_spec.rb +0 -32
  87. data/spec/chewy/query/nodes/exists_spec.rb +0 -18
  88. data/spec/chewy/query/nodes/has_child_spec.rb +0 -59
  89. data/spec/chewy/query/nodes/has_parent_spec.rb +0 -59
  90. data/spec/chewy/query/nodes/match_all_spec.rb +0 -11
  91. data/spec/chewy/query/nodes/missing_spec.rb +0 -16
  92. data/spec/chewy/query/nodes/not_spec.rb +0 -14
  93. data/spec/chewy/query/nodes/or_spec.rb +0 -12
  94. data/spec/chewy/query/nodes/prefix_spec.rb +0 -16
  95. data/spec/chewy/query/nodes/query_spec.rb +0 -12
  96. data/spec/chewy/query/nodes/range_spec.rb +0 -32
  97. data/spec/chewy/query/nodes/raw_spec.rb +0 -11
  98. data/spec/chewy/query/nodes/regexp_spec.rb +0 -43
  99. data/spec/chewy/query/nodes/script_spec.rb +0 -15
  100. data/spec/chewy/query/pagination/kaminari_spec.rb +0 -5
  101. data/spec/chewy/query/pagination/will_paginate_spec.rb +0 -5
  102. data/spec/chewy/query/pagination_spec.rb +0 -39
  103. data/spec/chewy/query_spec.rb +0 -637
@@ -1,11 +0,0 @@
1
- module Chewy
2
- class Query
3
- module Nodes
4
- class MatchAll < Expr
5
- def __render__
6
- {match_all: {}}
7
- end
8
- end
9
- end
10
- end
11
- end
@@ -1,20 +0,0 @@
1
- module Chewy
2
- class Query
3
- module Nodes
4
- class Missing < Expr
5
- def initialize(name, options = {})
6
- @name = name.to_s
7
- @options = options.reverse_merge(existence: true, null_value: false)
8
- end
9
-
10
- def !
11
- Nodes::Exists.new @name
12
- end
13
-
14
- def __render__
15
- {missing: {field: @name}.merge(@options.slice(:existence, :null_value))}
16
- end
17
- end
18
- end
19
- end
20
- end
@@ -1,25 +0,0 @@
1
- module Chewy
2
- class Query
3
- module Nodes
4
- class Not < Expr
5
- def initialize(expr, options = {})
6
- @expr = expr
7
- @options = options
8
- end
9
-
10
- def !
11
- @expr
12
- end
13
-
14
- def __render__
15
- expr = @expr.__render__
16
- if @options.key?(:cache)
17
- {not: {filter: expr, _cache: !!@options[:cache]}}
18
- else
19
- {not: expr}
20
- end
21
- end
22
- end
23
- end
24
- end
25
- end
@@ -1,25 +0,0 @@
1
- module Chewy
2
- class Query
3
- module Nodes
4
- class Or < Expr
5
- def initialize(*nodes)
6
- @options = nodes.extract_options!
7
- @nodes = nodes.flatten.map { |node| node.is_a?(self.class) ? node.__nodes__ : node }.flatten
8
- end
9
-
10
- def __nodes__
11
- @nodes
12
- end
13
-
14
- def __render__
15
- nodes = @nodes.map(&:__render__)
16
- if @options.key?(:cache)
17
- {or: {filters: nodes, _cache: !!@options[:cache]}}
18
- else
19
- {or: nodes}
20
- end
21
- end
22
- end
23
- end
24
- end
25
- end
@@ -1,19 +0,0 @@
1
- module Chewy
2
- class Query
3
- module Nodes
4
- class Prefix < Expr
5
- def initialize(name, value, options = {})
6
- @name = name.to_s
7
- @value = value
8
- @options = options
9
- end
10
-
11
- def __render__
12
- filter = {prefix: {@name => @value}}
13
- filter[:prefix][:_cache] = !!@options[:cache] if @options.key?(:cache)
14
- filter
15
- end
16
- end
17
- end
18
- end
19
- end
@@ -1,20 +0,0 @@
1
- module Chewy
2
- class Query
3
- module Nodes
4
- class Query < Expr
5
- def initialize(query, options = {})
6
- @query = query
7
- @options = options
8
- end
9
-
10
- def __render__
11
- if @options.key?(:cache)
12
- {fquery: {query: @query, _cache: !!@options[:cache]}}
13
- else
14
- {query: @query}
15
- end
16
- end
17
- end
18
- end
19
- end
20
- end
@@ -1,63 +0,0 @@
1
- module Chewy
2
- class Query
3
- module Nodes
4
- class Range < Expr
5
- EXECUTION = {
6
- i: :index,
7
- index: :index,
8
- f: :fielddata,
9
- fielddata: :fielddata
10
- }.freeze
11
-
12
- def initialize(name, *args)
13
- @name = name.to_s
14
- @options = args.extract_options!
15
- @range = @options.select { |k, _v| %i[gt lt].include?(k) }
16
- @bounds = @options.select { |k, _v| %i[left_closed right_closed].include?(k) }
17
- execution = EXECUTION[args.first.to_sym] if args.first
18
- @options[:execution] = execution if execution
19
- end
20
-
21
- def &(other)
22
- if other.is_a?(self.class) && other.__name__ == @name
23
- state = __state__.merge(other.__state__)
24
-
25
- cache = other.__options__[:cache] || @options[:cache]
26
- state[:cache] = cache unless cache.nil?
27
-
28
- execution = other.__options__[:execution] || @options[:execution]
29
- state[:execution] = execution unless execution.nil?
30
-
31
- self.class.new(@name, state)
32
- else
33
- super
34
- end
35
- end
36
-
37
- def __name__
38
- @name
39
- end
40
-
41
- def __state__
42
- @range.merge(@bounds)
43
- end
44
-
45
- def __options__
46
- @options
47
- end
48
-
49
- def __render__
50
- body = {}
51
- body[@bounds[:left_closed] ? :gte : :gt] = @range[:gt] if @range.key?(:gt)
52
- body[@bounds[:right_closed] ? :lte : :lt] = @range[:lt] if @range.key?(:lt)
53
-
54
- filter = {@name => body}
55
- filter[:_cache] = !!@options[:cache] if @options.key?(:cache)
56
- filter.merge!(@options.slice(:execution))
57
-
58
- {range: filter}
59
- end
60
- end
61
- end
62
- end
63
- end
@@ -1,15 +0,0 @@
1
- module Chewy
2
- class Query
3
- module Nodes
4
- class Raw < Expr
5
- def initialize(raw)
6
- @raw = raw
7
- end
8
-
9
- def __render__
10
- @raw
11
- end
12
- end
13
- end
14
- end
15
- end
@@ -1,35 +0,0 @@
1
- module Chewy
2
- class Query
3
- module Nodes
4
- class Regexp < Expr
5
- FLAGS = %w[all anystring automaton complement empty intersection interval none].freeze
6
-
7
- def initialize(name, regexp, *args)
8
- @name = name.to_s
9
- @regexp = regexp.respond_to?(:source) ? regexp.source : regexp.to_s
10
- @options = args.extract_options!
11
- return if args.blank? && @options[:flags].blank?
12
- @options[:flags] = FLAGS & (args.present? ? args.flatten : @options[:flags]).map(&:to_s).map(&:downcase)
13
- end
14
-
15
- def __render__
16
- body = if @options[:flags]
17
- {value: @regexp, flags: @options[:flags].map(&:to_s).map(&:upcase).uniq.join('|')}
18
- else
19
- @regexp
20
- end
21
- filter = {@name => body}
22
- if @options.key?(:cache)
23
- filter[:_cache] = !!@options[:cache]
24
- filter[:_cache_key] = if @options[:cache].is_a?(TrueClass) || @options[:cache].is_a?(FalseClass)
25
- @regexp.underscore
26
- else
27
- @options[:cache]
28
- end
29
- end
30
- {regexp: filter}
31
- end
32
- end
33
- end
34
- end
35
- end
@@ -1,20 +0,0 @@
1
- module Chewy
2
- class Query
3
- module Nodes
4
- class Script < Expr
5
- def initialize(script, params = {})
6
- @script = script
7
- @params = params
8
- @options = params.select { |k, _v| [:cache].include?(k) }
9
- end
10
-
11
- def __render__
12
- script = {script: @script}
13
- script[:params] = @params if @params.present?
14
- script[:_cache] = !!@options[:cache] if @options.key?(:cache)
15
- {script: script}
16
- end
17
- end
18
- end
19
- end
20
- end
@@ -1,25 +0,0 @@
1
- module Chewy
2
- class Query
3
- module Pagination
4
- # Returns request total found documents count
5
- #
6
- # PlacesIndex.query(...).filter(...).total
7
- #
8
- def total
9
- _response['hits'].try(:[], 'total') || 0
10
- end
11
- alias_method :total_count, :total
12
- alias_method :total_entries, :total
13
-
14
- private
15
-
16
- def raw_limit_value
17
- criteria.request_options[:size]
18
- end
19
-
20
- def raw_offset_value
21
- criteria.request_options[:from]
22
- end
23
- end
24
- end
25
- end
@@ -1,700 +0,0 @@
1
- require 'spec_helper'
2
-
3
- describe Chewy::Query::Criteria do
4
- subject { described_class.new }
5
-
6
- its(:options) { should be_a Hash }
7
- its(:request_options) { should be_a Hash }
8
- its(:facets) { should == {} }
9
- its(:scores) { should == [] }
10
- its(:aggregations) { should == {} }
11
- its(:script_fields) { should == {} }
12
- its(:queries) { should == [] }
13
- its(:filters) { should == [] }
14
- its(:post_filters) { should == [] }
15
- its(:sort) { should == [] }
16
- its(:fields) { should == [] }
17
- its(:types) { should == [] }
18
-
19
- its(:request_options?) { should eq(false) }
20
- its(:facets?) { should eq(false) }
21
- its(:scores?) { should eq(false) }
22
- its(:aggregations?) { should eq(false) }
23
- its(:script_fields?) { should eq(false) }
24
- its(:queries?) { should eq(false) }
25
- its(:filters?) { should eq(false) }
26
- its(:post_filters?) { should eq(false) }
27
- its(:sort?) { should eq(false) }
28
- its(:fields?) { should eq(false) }
29
- its(:types?) { should eq(false) }
30
-
31
- its(:none?) { should eq(false) }
32
-
33
- describe '#update_options' do
34
- specify { expect { subject.update_options(field: 'hello') }.to change { subject.options }.to(hash_including(field: 'hello')) }
35
- end
36
-
37
- describe '#update_request_options' do
38
- specify { expect { subject.update_request_options(field: 'hello') }.to change { subject.request_options }.to(hash_including(field: 'hello')) }
39
- end
40
-
41
- describe '#update_facets' do
42
- specify { expect { subject.update_facets(field: 'hello') }.to change { subject.facets? }.to(true) }
43
- specify { expect { subject.update_facets(field: 'hello') }.to change { subject.facets }.to(field: 'hello') }
44
- end
45
-
46
- describe '#update_scores' do
47
- specify { expect { subject.update_scores(:score) }.to change { subject.scores? }.to(true) }
48
- specify { expect { subject.update_scores(:score) }.to change { subject.scores }.to([:score]) }
49
- specify { expect { subject.update_scores(%i[score score2]) }.to change { subject.scores }.to(%i[score score2]) }
50
- specify do
51
- expect { subject.tap { |s| s.update_scores(:score1) }.update_scores(%i[score2 score3]) }
52
- .to change { subject.scores }.to(%i[score1 score2 score3])
53
- end
54
- end
55
-
56
- describe '#update_aggregations' do
57
- specify { expect { subject.update_aggregations(field: 'hello') }.to change { subject.aggregations? }.to(true) }
58
- specify { expect { subject.update_aggregations(field: 'hello') }.to change { subject.aggregations }.to(field: 'hello') }
59
- end
60
-
61
- describe '#update_script_fields' do
62
- specify do
63
- expect { subject.update_script_fields(distance: {script: "doc['coordinates'].distanceInMiles(lat, lon)"}) }
64
- .to change { subject.script_fields? }.to(true)
65
- end
66
- specify do
67
- expect { subject.update_script_fields(distance_km: {script: "doc['coordinates'].distanceInKm(lat, lon)"}) }
68
- .to change { subject.script_fields }.to(distance_km: {script: "doc['coordinates'].distanceInKm(lat, lon)"})
69
- end
70
- end
71
-
72
- describe '#update_queries' do
73
- specify do
74
- expect { subject.update_queries(field: 'hello') }
75
- .to change { subject.queries? }.to(true)
76
- end
77
- specify do
78
- expect { subject.update_queries(field: 'hello') }
79
- .to change { subject.queries }.to([field: 'hello'])
80
- end
81
- specify do
82
- expect do
83
- subject.update_queries(field: 'hello')
84
- subject.update_queries(field: 'world')
85
- end
86
- .to change { subject.queries }.to([{field: 'hello'}, {field: 'world'}])
87
- end
88
- specify do
89
- expect { subject.update_queries([{field: 'hello'}, {field: 'world'}, nil]) }
90
- .to change { subject.queries }.to([{field: 'hello'}, {field: 'world'}])
91
- end
92
- end
93
-
94
- describe '#update_filters' do
95
- specify { expect { subject.update_filters(field: 'hello') }.to change { subject.filters? }.to(true) }
96
- specify { expect { subject.update_filters(field: 'hello') }.to change { subject.filters }.to([{field: 'hello'}]) }
97
- specify do
98
- expect do
99
- subject.update_filters(field: 'hello')
100
- subject.update_filters(field: 'world')
101
- end
102
- .to change { subject.filters }.to([{field: 'hello'}, {field: 'world'}])
103
- end
104
- specify do
105
- expect { subject.update_filters([{field: 'hello'}, {field: 'world'}, nil]) }
106
- .to change { subject.filters }.to([{field: 'hello'}, {field: 'world'}])
107
- end
108
- end
109
-
110
- describe '#update_post_filters' do
111
- specify { expect { subject.update_post_filters(field: 'hello') }.to change { subject.post_filters? }.to(true) }
112
- specify { expect { subject.update_post_filters(field: 'hello') }.to change { subject.post_filters }.to([{field: 'hello'}]) }
113
- specify do
114
- expect do
115
- subject.update_post_filters(field: 'hello')
116
- subject.update_post_filters(field: 'world')
117
- end
118
- .to change { subject.post_filters }.to([{field: 'hello'}, {field: 'world'}])
119
- end
120
- specify do
121
- expect { subject.update_post_filters([{field: 'hello'}, {field: 'world'}, nil]) }
122
- .to change { subject.post_filters }.to([{field: 'hello'}, {field: 'world'}])
123
- end
124
- end
125
-
126
- describe '#update_sort' do
127
- specify do
128
- expect { subject.update_sort(:field) }
129
- .to change { subject.sort? }.to(true)
130
- end
131
-
132
- specify do
133
- expect { subject.update_sort([:field]) }
134
- .to change { subject.sort }.to([:field])
135
- end
136
- specify do
137
- expect { subject.update_sort(%i[field1 field2]) }
138
- .to change { subject.sort }.to(%i[field1 field2])
139
- end
140
- specify do
141
- expect { subject.update_sort([{field: :asc}]) }
142
- .to change { subject.sort }.to([{field: :asc}])
143
- end
144
- specify do
145
- expect { subject.update_sort([:field1, field2: {order: :asc}]) }
146
- .to change { subject.sort }.to([:field1, {field2: {order: :asc}}])
147
- end
148
- specify do
149
- expect { subject.update_sort([{field1: {order: :asc}}, :field2]) }
150
- .to change { subject.sort }.to([{field1: {order: :asc}}, :field2])
151
- end
152
- specify do
153
- expect { subject.update_sort([field1: :asc, field2: {order: :asc}]) }
154
- .to change { subject.sort }.to([{field1: :asc}, {field2: {order: :asc}}])
155
- end
156
- specify do
157
- expect { subject.update_sort([{field1: {order: :asc}}, :field2, :field3]) }
158
- .to change { subject.sort }.to([{field1: {order: :asc}}, :field2, :field3])
159
- end
160
- specify do
161
- expect { subject.update_sort([{field1: {order: :asc}}, %i[field2 field3]]) }
162
- .to change { subject.sort }.to([{field1: {order: :asc}}, :field2, :field3])
163
- end
164
- specify do
165
- expect { subject.update_sort([{field1: {order: :asc}}, [:field2], :field3]) }
166
- .to change { subject.sort }.to([{field1: {order: :asc}}, :field2, :field3])
167
- end
168
- specify do
169
- expect { subject.update_sort([{field1: {order: :asc}, field2: :desc}, [:field3], :field4]) }
170
- .to change { subject.sort }.to([{field1: {order: :asc}}, {field2: :desc}, :field3, :field4])
171
- end
172
- specify do
173
- expect { subject.tap { |s| s.update_sort([field1: {order: :asc}, field2: :desc]) }.update_sort([[:field3], :field4]) }
174
- .to change { subject.sort }.to([{field1: {order: :asc}}, {field2: :desc}, :field3, :field4])
175
- end
176
- specify do
177
- expect { subject.tap { |s| s.update_sort([field1: {order: :asc}, field2: :desc]) }.update_sort([[:field3], :field4], purge: true) }
178
- .to change { subject.sort }.to(%i[field3 field4])
179
- end
180
- end
181
-
182
- describe '#update_fields' do
183
- specify { expect { subject.update_fields(:field) }.to change { subject.fields? }.to(true) }
184
- specify { expect { subject.update_fields(:field) }.to change { subject.fields }.to(['field']) }
185
- specify { expect { subject.update_fields(%i[field field]) }.to change { subject.fields }.to(['field']) }
186
- specify { expect { subject.update_fields(%i[field1 field2]) }.to change { subject.fields }.to(%w[field1 field2]) }
187
- specify do
188
- expect { subject.tap { |s| s.update_fields(:field1) }.update_fields(%i[field2 field3]) }
189
- .to change { subject.fields }.to(%w[field1 field2 field3])
190
- end
191
- specify do
192
- expect { subject.tap { |s| s.update_fields(:field1) }.update_fields(%i[field2 field3], purge: true) }
193
- .to change { subject.fields }.to(%w[field2 field3])
194
- end
195
- end
196
-
197
- describe '#update_types' do
198
- specify { expect { subject.update_types(:type) }.to change { subject.types? }.to(true) }
199
- specify { expect { subject.update_types(:type) }.to change { subject.types }.to(['type']) }
200
- specify { expect { subject.update_types(%i[type type]) }.to change { subject.types }.to(['type']) }
201
- specify { expect { subject.update_types(%i[type1 type2]) }.to change { subject.types }.to(%w[type1 type2]) }
202
- specify do
203
- expect { subject.tap { |s| s.update_types(:type1) }.update_types(%i[type2 type3]) }
204
- .to change { subject.types }.to(%w[type1 type2 type3])
205
- end
206
- specify do
207
- expect { subject.tap { |s| s.update_types(:type1) }.update_types(%i[type2 type3], purge: true) }
208
- .to change { subject.types }.to(%w[type2 type3])
209
- end
210
- end
211
-
212
- describe '#merge' do
213
- let(:criteria) { described_class.new }
214
-
215
- specify { expect(subject.merge(criteria)).not_to be_equal subject }
216
- specify { expect(subject.merge(criteria)).not_to be_equal criteria }
217
-
218
- specify do
219
- expect(subject.tap { |c| c.update_options(opt1: 'hello') }
220
- .merge(criteria.tap { |c| c.update_options(opt2: 'hello') }).options)
221
- .to include(opt1: 'hello', opt2: 'hello')
222
- end
223
- specify do
224
- expect(subject.tap { |c| c.update_request_options(opt1: 'hello') }
225
- .merge(criteria.tap { |c| c.update_request_options(opt2: 'hello') }).request_options)
226
- .to include(opt1: 'hello', opt2: 'hello')
227
- end
228
- specify do
229
- expect(subject.tap { |c| c.update_facets(field1: 'hello') }
230
- .merge(criteria.tap { |c| c.update_facets(field1: 'hello') }).facets)
231
- .to eq(field1: 'hello')
232
- end
233
- specify do
234
- expect(subject.tap { |c| c.update_script_fields(distance_m: {script: "doc['coordinates'].distanceInMiles(lat, lon)"}) }
235
- .merge(criteria.tap { |c| c.update_script_fields(distance_km: {script: "doc['coordinates'].distanceInKm(lat, lon)"}) }).script_fields)
236
- .to eq(distance_m: {script: "doc['coordinates'].distanceInMiles(lat, lon)"}, distance_km: {script: "doc['coordinates'].distanceInKm(lat, lon)"})
237
- end
238
- specify do
239
- expect(subject.tap { |c| c.update_scores(script: 'hello') }
240
- .merge(criteria.tap { |c| c.update_scores(script: 'foobar') }).scores)
241
- .to eq([{script: 'hello'}, {script: 'foobar'}])
242
- end
243
- specify do
244
- expect(subject.tap { |c| c.update_aggregations(field1: 'hello') }
245
- .merge(criteria.tap { |c| c.update_aggregations(field1: 'hello') }).aggregations)
246
- .to eq(field1: 'hello')
247
- end
248
- specify do
249
- expect(subject.tap { |c| c.update_queries(field1: 'hello') }
250
- .merge(criteria.tap { |c| c.update_queries(field2: 'hello') }).queries)
251
- .to eq([{field1: 'hello'}, {field2: 'hello'}])
252
- end
253
- specify do
254
- expect(subject.tap { |c| c.update_filters(field1: 'hello') }
255
- .merge(criteria.tap { |c| c.update_filters(field2: 'hello') }).filters)
256
- .to eq([{field1: 'hello'}, {field2: 'hello'}])
257
- end
258
- specify do
259
- expect(subject.tap { |c| c.update_post_filters(field1: 'hello') }
260
- .merge(criteria.tap { |c| c.update_post_filters(field2: 'hello') }).post_filters)
261
- .to eq([{field1: 'hello'}, {field2: 'hello'}])
262
- end
263
- specify do
264
- expect(subject.tap { |c| c.update_sort(:field1) }
265
- .merge(criteria.tap { |c| c.update_sort(:field2) }).sort).to eq(%i[field1 field2])
266
- end
267
- specify do
268
- expect(subject.tap { |c| c.update_fields(:field1) }
269
- .merge(criteria.tap { |c| c.update_fields(:field2) }).fields).to eq(%w[field1 field2])
270
- end
271
- specify do
272
- expect(subject.tap { |c| c.update_types(:type1) }
273
- .merge(criteria.tap { |c| c.update_types(:type2) }).types).to eq(%w[type1 type2])
274
- end
275
- end
276
-
277
- describe '#merge!' do
278
- let(:criteria) { described_class.new }
279
-
280
- specify { expect(subject.merge!(criteria)).to be_equal subject }
281
- specify { expect(subject.merge!(criteria)).not_to be_equal criteria }
282
-
283
- specify do
284
- expect(subject.tap { |c| c.update_options(opt1: 'hello') }
285
- .merge!(criteria.tap { |c| c.update_options(opt2: 'hello') }).options)
286
- .to include(opt1: 'hello', opt2: 'hello')
287
- end
288
- specify do
289
- expect(subject.tap { |c| c.update_request_options(opt1: 'hello') }
290
- .merge!(criteria.tap { |c| c.update_request_options(opt2: 'hello') }).request_options)
291
- .to include(opt1: 'hello', opt2: 'hello')
292
- end
293
- specify do
294
- expect(subject.tap { |c| c.update_facets(field1: 'hello') }
295
- .merge!(criteria.tap { |c| c.update_facets(field1: 'hello') }).facets)
296
- .to eq(field1: 'hello')
297
- end
298
- specify do
299
- expect(subject.tap { |c| c.update_script_fields(distance_m: {script: "doc['coordinates'].distanceInMiles(lat, lon)"}) }
300
- .merge(criteria.tap { |c| c.update_script_fields(distance_km: {script: "doc['coordinates'].distanceInKm(lat, lon)"}) }).script_fields)
301
- .to eq(distance_m: {script: "doc['coordinates'].distanceInMiles(lat, lon)"}, distance_km: {script: "doc['coordinates'].distanceInKm(lat, lon)"})
302
- end
303
- specify do
304
- expect(subject.tap { |c| c.update_aggregations(field1: 'hello') }
305
- .merge!(criteria.tap { |c| c.update_aggregations(field1: 'hello') }).aggregations)
306
- .to eq(field1: 'hello')
307
- end
308
- specify do
309
- expect(subject.tap { |c| c.update_queries(field1: 'hello') }
310
- .merge!(criteria.tap { |c| c.update_queries(field2: 'hello') }).queries)
311
- .to eq([{field1: 'hello'}, {field2: 'hello'}])
312
- end
313
- specify do
314
- expect(subject.tap { |c| c.update_filters(field1: 'hello') }
315
- .merge!(criteria.tap { |c| c.update_filters(field2: 'hello') }).filters)
316
- .to eq([{field1: 'hello'}, {field2: 'hello'}])
317
- end
318
- specify do
319
- expect(subject.tap { |c| c.update_post_filters(field1: 'hello') }
320
- .merge!(criteria.tap { |c| c.update_post_filters(field2: 'hello') }).post_filters)
321
- .to eq([{field1: 'hello'}, {field2: 'hello'}])
322
- end
323
- specify do
324
- expect(subject.tap { |c| c.update_sort(:field1) }
325
- .merge!(criteria.tap { |c| c.update_sort(:field2) }).sort)
326
- .to eq(%i[field1 field2])
327
- end
328
- specify do
329
- expect(subject.tap { |c| c.update_fields(:field1) }
330
- .merge!(criteria.tap { |c| c.update_fields(:field2) }).fields)
331
- .to eq(%w[field1 field2])
332
- end
333
- specify do
334
- expect(subject.tap { |c| c.update_types(:type1) }
335
- .merge!(criteria.tap { |c| c.update_types(:type2) }).types)
336
- .to eq(%w[type1 type2])
337
- end
338
- end
339
-
340
- describe '#request_body' do
341
- def request_body(&block)
342
- subject.instance_exec(&block) if block
343
- subject.request_body
344
- end
345
-
346
- specify { expect(request_body).to eq(body: {}) }
347
- specify { expect(request_body { update_request_options(size: 10) }).to eq(body: {size: 10}) }
348
- specify { expect(request_body { update_request_options(from: 10) }).to eq(body: {from: 10}) }
349
- specify { expect(request_body { update_request_options(explain: true) }).to eq(body: {explain: true}) }
350
- specify { expect(request_body { update_queries(:query) }).to eq(body: {query: :query}) }
351
- specify do
352
- expect(request_body do
353
- update_scores(script_score: {script: '_score'})
354
- end).to eq(body: {query: {function_score: {functions: [{script_score: {script: '_score'}}]}}}) end
355
- specify do
356
- expect(request_body do
357
- update_scores(script_score: {script: 'boost_me'})
358
- update_queries(:query)
359
- update_options(boost_mode: :add)
360
- update_options(score_mode: :avg)
361
- end).to eq(body: {query: {
362
- function_score: {
363
- functions: [{
364
- script_score: {script: 'boost_me'}
365
- }],
366
- query: :query,
367
- boost_mode: :add,
368
- score_mode: :avg
369
- }
370
- }})
371
- end
372
- specify do
373
- expect(request_body do
374
- update_request_options(from: 10)
375
- update_sort(:field)
376
- update_fields(:field)
377
- update_queries(:query)
378
- end).to eq(body: {query: :query, from: 10, sort: [:field], _source: ['field']}) end
379
-
380
- specify do
381
- expect(request_body do
382
- update_queries(:query)
383
- update_filters(:filters)
384
- end).to eq(body: {query: {filtered: {query: :query, filter: :filters}}}) end
385
- specify do
386
- expect(request_body do
387
- update_queries(:query)
388
- update_post_filters(:post_filter)
389
- end).to eq(body: {query: :query, post_filter: :post_filter}) end
390
- specify do
391
- expect(request_body do
392
- update_script_fields(distance_m: {script: "doc['coordinates'].distanceInMiles(lat, lon)"})
393
- end).to eq(body: {script_fields: {distance_m: {script: "doc['coordinates'].distanceInMiles(lat, lon)"}}}) end
394
- end
395
-
396
- describe '#_filtered_query' do
397
- def _filtered_query(options = {}, &block)
398
- subject.instance_exec(&block) if block
399
- subject.send(:_filtered_query, subject.send(:_request_query), subject.send(:_request_filter), options)
400
- end
401
-
402
- specify { expect(_filtered_query).to eq({}) }
403
- specify { expect(_filtered_query { update_queries(:query) }).to eq(query: :query) }
404
- specify { expect(_filtered_query(strategy: 'query_first') { update_queries(:query) }).to eq(query: :query) }
405
- specify do
406
- expect(_filtered_query { update_queries(%i[query1 query2]) })
407
- .to eq(query: {bool: {must: %i[query1 query2]}})
408
- end
409
- specify do
410
- expect(_filtered_query do
411
- update_options(query_mode: :should)
412
- update_queries(%i[query1 query2])
413
- end)
414
- .to eq(query: {bool: {should: %i[query1 query2]}})
415
- end
416
- specify do
417
- expect(_filtered_query do
418
- update_options(query_mode: :dis_max)
419
- update_queries(%i[query1 query2])
420
- end)
421
- .to eq(query: {dis_max: {queries: %i[query1 query2]}})
422
- end
423
-
424
- specify do
425
- expect(_filtered_query(strategy: 'query_first') { update_filters(%i[filter1 filter2]) })
426
- .to eq(query: {filtered: {query: {match_all: {}}, filter: {and: %i[filter1 filter2]}, strategy: 'query_first'}})
427
- end
428
- specify do
429
- expect(_filtered_query { update_filters(%i[filter1 filter2]) })
430
- .to eq(query: {filtered: {query: {match_all: {}}, filter: {and: %i[filter1 filter2]}}})
431
- end
432
-
433
- specify do
434
- expect(_filtered_query do
435
- update_filters(%i[filter1 filter2])
436
- update_queries(%i[query1 query2])
437
- end)
438
- .to eq(query: {filtered: {
439
- query: {bool: {must: %i[query1 query2]}},
440
- filter: {and: %i[filter1 filter2]}
441
- }})
442
- end
443
- specify do
444
- expect(_filtered_query(strategy: 'query_first') do
445
- update_filters(%i[filter1 filter2])
446
- update_queries(%i[query1 query2])
447
- end)
448
- .to eq(query: {filtered: {
449
- query: {bool: {must: %i[query1 query2]}},
450
- filter: {and: %i[filter1 filter2]},
451
- strategy: 'query_first'
452
- }})
453
- end
454
- specify do
455
- expect(_filtered_query do
456
- update_options(query_mode: :should)
457
- update_options(filter_mode: :or)
458
- update_filters(%i[filter1 filter2])
459
- update_queries(%i[query1 query2])
460
- end).to eq(query: {filtered: {
461
- query: {bool: {should: %i[query1 query2]}},
462
- filter: {or: %i[filter1 filter2]}
463
- }})
464
- end
465
- end
466
-
467
- describe '#_boost_query' do
468
- specify do
469
- expect(subject.send(:_boost_query, query: :query))
470
- .to eq(query: :query)
471
- end
472
- specify do
473
- subject.update_scores(boost_factor: 5)
474
- expect(subject.send(:_boost_query, query: :query))
475
- .to eq(query: {function_score: {functions: [{boost_factor: 5}], query: :query}})
476
- end
477
- specify do
478
- subject.update_scores(boost_factor: 5)
479
- subject.update_options(boost_mode: :multiply)
480
- subject.update_options(score_mode: :add)
481
- expect(subject.send(:_boost_query, query: :query))
482
- .to eq(query: {function_score: {functions: [{boost_factor: 5}], query: :query, boost_mode: :multiply, score_mode: :add}})
483
- end
484
- specify do
485
- subject.update_scores(boost_factor: 5)
486
- expect(subject.send(:_boost_query, query: :query, filter: :filter))
487
- .to eq(query: {function_score: {functions: [{boost_factor: 5}], query: {filtered: {query: :query, filter: :filter}}}})
488
- end
489
- end
490
-
491
- describe '#_request_filter' do
492
- def _request_filter(&block)
493
- subject.instance_exec(&block) if block
494
- subject.send(:_request_filter)
495
- end
496
-
497
- specify { expect(_request_filter).to be_nil }
498
-
499
- specify { expect(_request_filter { update_types(:type) }).to eq(type: {value: 'type'}) }
500
- specify do
501
- expect(_request_filter { update_types(%i[type1 type2]) })
502
- .to eq(or: [{type: {value: 'type1'}}, {type: {value: 'type2'}}])
503
- end
504
-
505
- specify do
506
- expect(_request_filter { update_filters(%i[filter1 filter2]) })
507
- .to eq(and: %i[filter1 filter2])
508
- end
509
- specify do
510
- expect(_request_filter do
511
- update_options(filter_mode: :or)
512
- update_filters(%i[filter1 filter2])
513
- end)
514
- .to eq(or: %i[filter1 filter2])
515
- end
516
- specify do
517
- expect(_request_filter do
518
- update_options(filter_mode: :must)
519
- update_filters(%i[filter1 filter2])
520
- end)
521
- .to eq(bool: {must: %i[filter1 filter2]})
522
- end
523
- specify do
524
- expect(_request_filter do
525
- update_options(filter_mode: :should)
526
- update_filters(%i[filter1 filter2])
527
- end)
528
- .to eq(bool: {should: %i[filter1 filter2]})
529
- end
530
- specify do
531
- expect(_request_filter do
532
- update_options(filter_mode: :must_not)
533
- update_filters(%i[filter1 filter2])
534
- end)
535
- .to eq(bool: {must_not: %i[filter1 filter2]})
536
- end
537
-
538
- specify do
539
- expect(_request_filter do
540
- update_types(%i[type1 type2])
541
- update_filters(%i[filter1 filter2])
542
- end)
543
- .to eq(and: [{or: [{type: {value: 'type1'}}, {type: {value: 'type2'}}]}, :filter1, :filter2])
544
- end
545
- specify do
546
- expect(_request_filter do
547
- update_options(filter_mode: :or)
548
- update_types(%i[type1 type2])
549
- update_filters(%i[filter1 filter2])
550
- end)
551
- .to eq(and: [{or: [{type: {value: 'type1'}}, {type: {value: 'type2'}}]}, {or: %i[filter1 filter2]}])
552
- end
553
- specify do
554
- expect(_request_filter do
555
- update_options(filter_mode: :must)
556
- update_types(%i[type1 type2])
557
- update_filters(%i[filter1 filter2])
558
- end)
559
- .to eq(and: [{or: [{type: {value: 'type1'}}, {type: {value: 'type2'}}]}, {bool: {must: %i[filter1 filter2]}}])
560
- end
561
- specify do
562
- expect(_request_filter do
563
- update_options(filter_mode: :should)
564
- update_types(%i[type1 type2])
565
- update_filters(%i[filter1 filter2])
566
- end)
567
- .to eq(and: [{or: [{type: {value: 'type1'}}, {type: {value: 'type2'}}]}, {bool: {should: %i[filter1 filter2]}}])
568
- end
569
- specify do
570
- expect(_request_filter do
571
- update_options(filter_mode: :must_not)
572
- update_types(%i[type1 type2])
573
- update_filters(%i[filter1 filter2])
574
- end)
575
- .to eq(and: [{or: [{type: {value: 'type1'}}, {type: {value: 'type2'}}]}, {bool: {must_not: %i[filter1 filter2]}}])
576
- end
577
- end
578
-
579
- describe '#_request_post_filter' do
580
- def _request_post_filter(&block)
581
- subject.instance_exec(&block) if block
582
- subject.send(:_request_post_filter)
583
- end
584
-
585
- specify { expect(_request_post_filter).to be_nil }
586
-
587
- specify do
588
- expect(_request_post_filter { update_post_filters(%i[post_filter1 post_filter2]) })
589
- .to eq(and: %i[post_filter1 post_filter2])
590
- end
591
- specify do
592
- expect(_request_post_filter do
593
- update_options(post_filter_mode: :or)
594
- update_post_filters(%i[post_filter1 post_filter2])
595
- end)
596
- .to eq(or: %i[post_filter1 post_filter2])
597
- end
598
- specify do
599
- expect(_request_post_filter do
600
- update_options(post_filter_mode: :must)
601
- update_post_filters(%i[post_filter1 post_filter2])
602
- end)
603
- .to eq(bool: {must: %i[post_filter1 post_filter2]})
604
- end
605
- specify do
606
- expect(_request_post_filter do
607
- update_options(post_filter_mode: :should)
608
- update_post_filters(%i[post_filter1 post_filter2])
609
- end)
610
- .to eq(bool: {should: %i[post_filter1 post_filter2]})
611
- end
612
-
613
- context do
614
- before { allow(Chewy).to receive_messages(filter_mode: :or) }
615
-
616
- specify do
617
- expect(_request_post_filter { update_post_filters(%i[post_filter1 post_filter2]) })
618
- .to eq(or: %i[post_filter1 post_filter2])
619
- end
620
- end
621
- end
622
-
623
- describe '#_request_types' do
624
- def _request_types(&block)
625
- subject.instance_exec(&block) if block
626
- subject.send(:_request_types)
627
- end
628
-
629
- specify { expect(_request_types).to be_nil }
630
- specify { expect(_request_types { update_types(:type1) }).to eq(type: {value: 'type1'}) }
631
- specify do
632
- expect(_request_types { update_types(%i[type1 type2]) })
633
- .to eq(or: [{type: {value: 'type1'}}, {type: {value: 'type2'}}])
634
- end
635
- end
636
-
637
- describe '#_queries_join' do
638
- def _queries_join(*args)
639
- subject.send(:_queries_join, *args)
640
- end
641
-
642
- let(:query) { {term: {field: 'value'}} }
643
-
644
- specify { expect(_queries_join([], :dis_max)).to be_nil }
645
- specify { expect(_queries_join([query], :dis_max)).to eq(query) }
646
- specify { expect(_queries_join([query, query], :dis_max)).to eq(dis_max: {queries: [query, query]}) }
647
-
648
- specify { expect(_queries_join([], 0.7)).to be_nil }
649
- specify { expect(_queries_join([query], 0.7)).to eq(query) }
650
- specify { expect(_queries_join([query, query], 0.7)).to eq(dis_max: {queries: [query, query], tie_breaker: 0.7}) }
651
-
652
- specify { expect(_queries_join([], :must)).to be_nil }
653
- specify { expect(_queries_join([query], :must)).to eq(query) }
654
- specify { expect(_queries_join([query, query], :must)).to eq(bool: {must: [query, query]}) }
655
-
656
- specify { expect(_queries_join([], :should)).to be_nil }
657
- specify { expect(_queries_join([query], :should)).to eq(query) }
658
- specify { expect(_queries_join([query, query], :should)).to eq(bool: {should: [query, query]}) }
659
-
660
- specify { expect(_queries_join([], :must_not)).to be_nil }
661
- specify { expect(_queries_join([query], :must_not)).to eq(bool: {must_not: [query]}) }
662
- specify { expect(_queries_join([query, query], :must_not)).to eq(bool: {must_not: [query, query]}) }
663
-
664
- specify { expect(_queries_join([], '25%')).to be_nil }
665
- specify { expect(_queries_join([query], '25%')).to eq(query) }
666
- specify { expect(_queries_join([query, query], '25%')).to eq(bool: {should: [query, query], minimum_should_match: '25%'}) }
667
- end
668
-
669
- describe '#_filters_join' do
670
- def _filters_join(*args)
671
- subject.send(:_filters_join, *args)
672
- end
673
-
674
- let(:filter) { {term: {field: 'value'}} }
675
-
676
- specify { expect(_filters_join([], :and)).to be_nil }
677
- specify { expect(_filters_join([filter], :and)).to eq(filter) }
678
- specify { expect(_filters_join([filter, filter], :and)).to eq(and: [filter, filter]) }
679
-
680
- specify { expect(_filters_join([], :or)).to be_nil }
681
- specify { expect(_filters_join([filter], :or)).to eq(filter) }
682
- specify { expect(_filters_join([filter, filter], :or)).to eq(or: [filter, filter]) }
683
-
684
- specify { expect(_filters_join([], :must)).to be_nil }
685
- specify { expect(_filters_join([filter], :must)).to eq(filter) }
686
- specify { expect(_filters_join([filter, filter], :must)).to eq(bool: {must: [filter, filter]}) }
687
-
688
- specify { expect(_filters_join([], :should)).to be_nil }
689
- specify { expect(_filters_join([filter], :should)).to eq(filter) }
690
- specify { expect(_filters_join([filter, filter], :should)).to eq(bool: {should: [filter, filter]}) }
691
-
692
- specify { expect(_filters_join([], :must_not)).to be_nil }
693
- specify { expect(_filters_join([filter], :must_not)).to eq(bool: {must_not: [filter]}) }
694
- specify { expect(_filters_join([filter, filter], :must_not)).to eq(bool: {must_not: [filter, filter]}) }
695
-
696
- specify { expect(_filters_join([], '25%')).to be_nil }
697
- specify { expect(_filters_join([filter], '25%')).to eq(filter) }
698
- specify { expect(_filters_join([filter, filter], '25%')).to eq(bool: {should: [filter, filter], minimum_should_match: '25%'}) }
699
- end
700
- end