chewy 5.1.0 → 7.2.7

Sign up to get free protection for your applications and to get access to all the features.
Files changed (234) hide show
  1. checksums.yaml +4 -4
  2. data/.github/CODEOWNERS +1 -0
  3. data/.github/ISSUE_TEMPLATE/bug_report.md +39 -0
  4. data/.github/ISSUE_TEMPLATE/feature_request.md +20 -0
  5. data/.github/PULL_REQUEST_TEMPLATE.md +16 -0
  6. data/.github/workflows/ruby.yml +73 -0
  7. data/.rubocop.yml +13 -8
  8. data/.rubocop_todo.yml +110 -22
  9. data/CHANGELOG.md +449 -347
  10. data/CODE_OF_CONDUCT.md +14 -0
  11. data/CONTRIBUTING.md +63 -0
  12. data/Gemfile +3 -7
  13. data/Guardfile +3 -1
  14. data/LICENSE.txt +1 -1
  15. data/README.md +423 -311
  16. data/chewy.gemspec +8 -10
  17. data/gemfiles/rails.5.2.activerecord.gemfile +9 -14
  18. data/gemfiles/rails.6.0.activerecord.gemfile +11 -0
  19. data/gemfiles/rails.6.1.activerecord.gemfile +13 -0
  20. data/gemfiles/rails.7.0.activerecord.gemfile +13 -0
  21. data/lib/chewy/config.rb +42 -60
  22. data/lib/chewy/errors.rb +4 -10
  23. data/lib/chewy/fields/base.rb +80 -20
  24. data/lib/chewy/fields/root.rb +7 -17
  25. data/lib/chewy/index/actions.rb +62 -35
  26. data/lib/chewy/{type → index}/adapter/active_record.rb +18 -4
  27. data/lib/chewy/{type → index}/adapter/base.rb +2 -3
  28. data/lib/chewy/{type → index}/adapter/object.rb +28 -32
  29. data/lib/chewy/{type → index}/adapter/orm.rb +26 -24
  30. data/lib/chewy/index/aliases.rb +14 -5
  31. data/lib/chewy/{type → index}/crutch.rb +5 -5
  32. data/lib/chewy/index/import/bulk_builder.rb +311 -0
  33. data/lib/chewy/{type → index}/import/bulk_request.rb +6 -7
  34. data/lib/chewy/{type → index}/import/journal_builder.rb +11 -12
  35. data/lib/chewy/{type → index}/import/routine.rb +17 -16
  36. data/lib/chewy/{type → index}/import.rb +51 -33
  37. data/lib/chewy/{type → index}/mapping.rb +32 -37
  38. data/lib/chewy/index/observe/active_record_methods.rb +87 -0
  39. data/lib/chewy/index/observe/callback.rb +34 -0
  40. data/lib/chewy/index/observe.rb +17 -0
  41. data/lib/chewy/index/specification.rb +1 -0
  42. data/lib/chewy/{type → index}/syncer.rb +61 -62
  43. data/lib/chewy/{type → index}/witchcraft.rb +15 -9
  44. data/lib/chewy/{type → index}/wrapper.rb +13 -3
  45. data/lib/chewy/index.rb +46 -96
  46. data/lib/chewy/journal.rb +25 -14
  47. data/lib/chewy/minitest/helpers.rb +86 -13
  48. data/lib/chewy/minitest/search_index_receiver.rb +22 -26
  49. data/lib/chewy/multi_search.rb +62 -0
  50. data/lib/chewy/railtie.rb +6 -20
  51. data/lib/chewy/rake_helper.rb +136 -108
  52. data/lib/chewy/rspec/build_query.rb +12 -0
  53. data/lib/chewy/rspec/helpers.rb +55 -0
  54. data/lib/chewy/rspec/update_index.rb +55 -44
  55. data/lib/chewy/rspec.rb +2 -0
  56. data/lib/chewy/runtime.rb +1 -1
  57. data/lib/chewy/search/loader.rb +19 -41
  58. data/lib/chewy/search/parameters/collapse.rb +16 -0
  59. data/lib/chewy/search/parameters/concerns/query_storage.rb +2 -2
  60. data/lib/chewy/search/parameters/ignore_unavailable.rb +27 -0
  61. data/lib/chewy/search/parameters/indices.rb +12 -57
  62. data/lib/chewy/search/parameters/none.rb +1 -3
  63. data/lib/chewy/search/parameters/order.rb +6 -19
  64. data/lib/chewy/search/parameters/source.rb +5 -1
  65. data/lib/chewy/search/parameters/track_total_hits.rb +16 -0
  66. data/lib/chewy/search/parameters.rb +7 -4
  67. data/lib/chewy/search/query_proxy.rb +9 -2
  68. data/lib/chewy/search/request.rb +180 -154
  69. data/lib/chewy/search/response.rb +5 -5
  70. data/lib/chewy/search/scoping.rb +7 -8
  71. data/lib/chewy/search/scrolling.rb +16 -13
  72. data/lib/chewy/search.rb +7 -22
  73. data/lib/chewy/stash.rb +19 -30
  74. data/lib/chewy/strategy/active_job.rb +2 -2
  75. data/lib/chewy/strategy/atomic_no_refresh.rb +18 -0
  76. data/lib/chewy/strategy/base.rb +10 -0
  77. data/lib/chewy/strategy/lazy_sidekiq.rb +64 -0
  78. data/lib/chewy/strategy/sidekiq.rb +3 -2
  79. data/lib/chewy/strategy.rb +5 -19
  80. data/lib/chewy/version.rb +1 -1
  81. data/lib/chewy.rb +36 -80
  82. data/lib/generators/chewy/install_generator.rb +1 -1
  83. data/lib/tasks/chewy.rake +26 -32
  84. data/migration_guide.md +56 -0
  85. data/spec/chewy/config_spec.rb +15 -61
  86. data/spec/chewy/fields/base_spec.rb +432 -145
  87. data/spec/chewy/fields/root_spec.rb +20 -28
  88. data/spec/chewy/fields/time_fields_spec.rb +5 -5
  89. data/spec/chewy/index/actions_spec.rb +388 -55
  90. data/spec/chewy/{type → index}/adapter/active_record_spec.rb +110 -44
  91. data/spec/chewy/{type → index}/adapter/object_spec.rb +21 -6
  92. data/spec/chewy/index/aliases_spec.rb +3 -3
  93. data/spec/chewy/index/import/bulk_builder_spec.rb +494 -0
  94. data/spec/chewy/{type → index}/import/bulk_request_spec.rb +5 -12
  95. data/spec/chewy/{type → index}/import/journal_builder_spec.rb +14 -22
  96. data/spec/chewy/{type → index}/import/routine_spec.rb +19 -19
  97. data/spec/chewy/{type → index}/import_spec.rb +149 -96
  98. data/spec/chewy/index/mapping_spec.rb +135 -0
  99. data/spec/chewy/index/observe/active_record_methods_spec.rb +68 -0
  100. data/spec/chewy/index/observe/callback_spec.rb +139 -0
  101. data/spec/chewy/index/observe_spec.rb +143 -0
  102. data/spec/chewy/index/settings_spec.rb +3 -1
  103. data/spec/chewy/index/specification_spec.rb +20 -30
  104. data/spec/chewy/{type → index}/syncer_spec.rb +14 -19
  105. data/spec/chewy/{type → index}/witchcraft_spec.rb +34 -21
  106. data/spec/chewy/index/wrapper_spec.rb +100 -0
  107. data/spec/chewy/index_spec.rb +69 -137
  108. data/spec/chewy/journal_spec.rb +46 -91
  109. data/spec/chewy/minitest/helpers_spec.rb +122 -14
  110. data/spec/chewy/minitest/search_index_receiver_spec.rb +24 -26
  111. data/spec/chewy/multi_search_spec.rb +84 -0
  112. data/spec/chewy/rake_helper_spec.rb +293 -101
  113. data/spec/chewy/rspec/build_query_spec.rb +34 -0
  114. data/spec/chewy/rspec/helpers_spec.rb +61 -0
  115. data/spec/chewy/rspec/update_index_spec.rb +106 -102
  116. data/spec/chewy/runtime_spec.rb +2 -2
  117. data/spec/chewy/search/loader_spec.rb +19 -53
  118. data/spec/chewy/search/pagination/kaminari_examples.rb +3 -5
  119. data/spec/chewy/search/pagination/kaminari_spec.rb +1 -1
  120. data/spec/chewy/search/parameters/collapse_spec.rb +5 -0
  121. data/spec/chewy/search/parameters/ignore_unavailable_spec.rb +67 -0
  122. data/spec/chewy/search/parameters/indices_spec.rb +26 -118
  123. data/spec/chewy/search/parameters/none_spec.rb +1 -1
  124. data/spec/chewy/search/parameters/order_spec.rb +18 -11
  125. data/spec/chewy/search/parameters/query_storage_examples.rb +67 -21
  126. data/spec/chewy/search/parameters/search_after_spec.rb +4 -1
  127. data/spec/chewy/search/parameters/source_spec.rb +8 -2
  128. data/spec/chewy/search/parameters/track_total_hits_spec.rb +5 -0
  129. data/spec/chewy/search/parameters_spec.rb +23 -7
  130. data/spec/chewy/search/query_proxy_spec.rb +68 -17
  131. data/spec/chewy/search/request_spec.rb +344 -149
  132. data/spec/chewy/search/response_spec.rb +35 -25
  133. data/spec/chewy/search/scrolling_spec.rb +28 -26
  134. data/spec/chewy/search_spec.rb +69 -59
  135. data/spec/chewy/stash_spec.rb +16 -26
  136. data/spec/chewy/strategy/active_job_spec.rb +23 -10
  137. data/spec/chewy/strategy/atomic_no_refresh_spec.rb +60 -0
  138. data/spec/chewy/strategy/atomic_spec.rb +9 -10
  139. data/spec/chewy/strategy/lazy_sidekiq_spec.rb +214 -0
  140. data/spec/chewy/strategy/sidekiq_spec.rb +14 -10
  141. data/spec/chewy/strategy_spec.rb +19 -15
  142. data/spec/chewy_spec.rb +17 -110
  143. data/spec/spec_helper.rb +6 -29
  144. data/spec/support/active_record.rb +43 -5
  145. metadata +102 -198
  146. data/.travis.yml +0 -45
  147. data/Appraisals +0 -81
  148. data/LEGACY_DSL.md +0 -497
  149. data/gemfiles/rails.4.0.activerecord.gemfile +0 -15
  150. data/gemfiles/rails.4.1.activerecord.gemfile +0 -15
  151. data/gemfiles/rails.4.2.activerecord.gemfile +0 -16
  152. data/gemfiles/rails.4.2.mongoid.5.2.gemfile +0 -16
  153. data/gemfiles/rails.5.0.activerecord.gemfile +0 -16
  154. data/gemfiles/rails.5.0.mongoid.6.1.gemfile +0 -16
  155. data/gemfiles/rails.5.1.activerecord.gemfile +0 -16
  156. data/gemfiles/rails.5.1.mongoid.6.3.gemfile +0 -16
  157. data/gemfiles/sequel.4.45.gemfile +0 -11
  158. data/lib/chewy/backports/deep_dup.rb +0 -46
  159. data/lib/chewy/backports/duplicable.rb +0 -91
  160. data/lib/chewy/query/compose.rb +0 -68
  161. data/lib/chewy/query/criteria.rb +0 -191
  162. data/lib/chewy/query/filters.rb +0 -244
  163. data/lib/chewy/query/loading.rb +0 -110
  164. data/lib/chewy/query/nodes/and.rb +0 -25
  165. data/lib/chewy/query/nodes/base.rb +0 -17
  166. data/lib/chewy/query/nodes/bool.rb +0 -34
  167. data/lib/chewy/query/nodes/equal.rb +0 -34
  168. data/lib/chewy/query/nodes/exists.rb +0 -20
  169. data/lib/chewy/query/nodes/expr.rb +0 -28
  170. data/lib/chewy/query/nodes/field.rb +0 -110
  171. data/lib/chewy/query/nodes/has_child.rb +0 -15
  172. data/lib/chewy/query/nodes/has_parent.rb +0 -15
  173. data/lib/chewy/query/nodes/has_relation.rb +0 -59
  174. data/lib/chewy/query/nodes/match_all.rb +0 -11
  175. data/lib/chewy/query/nodes/missing.rb +0 -20
  176. data/lib/chewy/query/nodes/not.rb +0 -25
  177. data/lib/chewy/query/nodes/or.rb +0 -25
  178. data/lib/chewy/query/nodes/prefix.rb +0 -19
  179. data/lib/chewy/query/nodes/query.rb +0 -20
  180. data/lib/chewy/query/nodes/range.rb +0 -63
  181. data/lib/chewy/query/nodes/raw.rb +0 -15
  182. data/lib/chewy/query/nodes/regexp.rb +0 -35
  183. data/lib/chewy/query/nodes/script.rb +0 -20
  184. data/lib/chewy/query/pagination.rb +0 -25
  185. data/lib/chewy/query.rb +0 -1142
  186. data/lib/chewy/search/pagination/will_paginate.rb +0 -43
  187. data/lib/chewy/search/parameters/types.rb +0 -20
  188. data/lib/chewy/strategy/resque.rb +0 -27
  189. data/lib/chewy/strategy/shoryuken.rb +0 -40
  190. data/lib/chewy/type/actions.rb +0 -43
  191. data/lib/chewy/type/adapter/mongoid.rb +0 -67
  192. data/lib/chewy/type/adapter/sequel.rb +0 -93
  193. data/lib/chewy/type/import/bulk_builder.rb +0 -122
  194. data/lib/chewy/type/observe.rb +0 -82
  195. data/lib/chewy/type.rb +0 -117
  196. data/lib/sequel/plugins/chewy_observe.rb +0 -63
  197. data/spec/chewy/query/criteria_spec.rb +0 -700
  198. data/spec/chewy/query/filters_spec.rb +0 -201
  199. data/spec/chewy/query/loading_spec.rb +0 -124
  200. data/spec/chewy/query/nodes/and_spec.rb +0 -12
  201. data/spec/chewy/query/nodes/bool_spec.rb +0 -14
  202. data/spec/chewy/query/nodes/equal_spec.rb +0 -32
  203. data/spec/chewy/query/nodes/exists_spec.rb +0 -18
  204. data/spec/chewy/query/nodes/has_child_spec.rb +0 -59
  205. data/spec/chewy/query/nodes/has_parent_spec.rb +0 -59
  206. data/spec/chewy/query/nodes/match_all_spec.rb +0 -11
  207. data/spec/chewy/query/nodes/missing_spec.rb +0 -16
  208. data/spec/chewy/query/nodes/not_spec.rb +0 -14
  209. data/spec/chewy/query/nodes/or_spec.rb +0 -12
  210. data/spec/chewy/query/nodes/prefix_spec.rb +0 -16
  211. data/spec/chewy/query/nodes/query_spec.rb +0 -12
  212. data/spec/chewy/query/nodes/range_spec.rb +0 -32
  213. data/spec/chewy/query/nodes/raw_spec.rb +0 -11
  214. data/spec/chewy/query/nodes/regexp_spec.rb +0 -43
  215. data/spec/chewy/query/nodes/script_spec.rb +0 -15
  216. data/spec/chewy/query/pagination/kaminari_spec.rb +0 -5
  217. data/spec/chewy/query/pagination/will_paginate_spec.rb +0 -5
  218. data/spec/chewy/query/pagination_spec.rb +0 -39
  219. data/spec/chewy/query_spec.rb +0 -637
  220. data/spec/chewy/search/pagination/will_paginate_examples.rb +0 -63
  221. data/spec/chewy/search/pagination/will_paginate_spec.rb +0 -23
  222. data/spec/chewy/search/parameters/types_spec.rb +0 -5
  223. data/spec/chewy/strategy/resque_spec.rb +0 -46
  224. data/spec/chewy/strategy/shoryuken_spec.rb +0 -66
  225. data/spec/chewy/type/actions_spec.rb +0 -50
  226. data/spec/chewy/type/adapter/mongoid_spec.rb +0 -372
  227. data/spec/chewy/type/adapter/sequel_spec.rb +0 -472
  228. data/spec/chewy/type/import/bulk_builder_spec.rb +0 -279
  229. data/spec/chewy/type/mapping_spec.rb +0 -173
  230. data/spec/chewy/type/observe_spec.rb +0 -137
  231. data/spec/chewy/type/wrapper_spec.rb +0 -98
  232. data/spec/chewy/type_spec.rb +0 -55
  233. data/spec/support/mongoid.rb +0 -93
  234. data/spec/support/sequel.rb +0 -80
@@ -10,8 +10,12 @@ describe Chewy::Fields::Base do
10
10
  specify { expect(field.compose(double(value: 'hello'))).to eq(name: 'hello') }
11
11
  specify { expect(field.compose(double(value: %w[hello world]))).to eq(name: %w[hello world]) }
12
12
 
13
- specify { expect(described_class.new(:name, value: :last_name).compose(double(last_name: 'hello'))).to eq(name: 'hello') }
14
- specify { expect(described_class.new(:name, value: :last_name).compose('last_name' => 'hello')).to eq(name: 'hello') }
13
+ specify do
14
+ expect(described_class.new(:name, value: :last_name).compose(double(last_name: 'hello'))).to eq(name: 'hello')
15
+ end
16
+ specify do
17
+ expect(described_class.new(:name, value: :last_name).compose('last_name' => 'hello')).to eq(name: 'hello')
18
+ end
15
19
  specify { expect(described_class.new(:name).compose(double(name: 'hello'))).to eq(name: 'hello') }
16
20
  specify { expect(described_class.new(:false_value).compose(false_value: false)).to eq(false_value: false) }
17
21
  specify { expect(described_class.new(:true_value).compose(true_value: true)).to eq(true_value: true) }
@@ -122,14 +126,12 @@ describe Chewy::Fields::Base do
122
126
  context 'default field type' do
123
127
  before do
124
128
  stub_index(:events) do
125
- define_type :event do
129
+ field :id
130
+ field :category do
126
131
  field :id
127
- field :category do
132
+ field :licenses do
128
133
  field :id
129
- field :licenses do
130
- field :id
131
- field :created_at, type: 'time'
132
- end
134
+ field :created_at, type: 'time'
133
135
  end
134
136
  end
135
137
  end
@@ -143,175 +145,217 @@ describe Chewy::Fields::Base do
143
145
  end
144
146
 
145
147
  specify do
146
- expect(EventsIndex::Event.mappings_hash).to eq(event: {
147
- properties: {
148
- id: {type: 'integer'},
149
- category: {
150
- type: 'object',
151
- properties: {
152
- id: {type: 'integer'},
153
- licenses: {
154
- type: 'object',
155
- properties: {
156
- id: {type: 'integer'},
157
- created_at: {type: 'time'}
148
+ expect(EventsIndex.mappings_hash).to eq(
149
+ mappings: {
150
+ properties: {
151
+ id: {type: 'integer'},
152
+ category: {
153
+ type: 'object',
154
+ properties: {
155
+ id: {type: 'integer'},
156
+ licenses: {
157
+ type: 'object',
158
+ properties: {
159
+ id: {type: 'integer'},
160
+ created_at: {type: 'time'}
161
+ }
158
162
  }
159
163
  }
160
164
  }
161
165
  }
162
166
  }
163
- })
167
+ )
164
168
  end
165
169
  end
166
170
 
167
171
  context 'objects, hashes and arrays' do
168
172
  before do
169
173
  stub_index(:events) do
170
- define_type :event do
174
+ field :id
175
+ field :category do
171
176
  field :id
172
- field :category do
177
+ field :licenses do
173
178
  field :id
174
- field :licenses do
175
- field :id
176
- field :name
177
- end
179
+ field :name
178
180
  end
179
181
  end
180
182
  end
181
183
  end
182
184
 
183
185
  specify do
184
- expect(EventsIndex::Event.root.compose(
185
- id: 1, category: {id: 2, licenses: {id: 3, name: 'Name'}}
186
- )).to eq('id' => 1, 'category' => {'id' => 2, 'licenses' => {'id' => 3, 'name' => 'Name'}})
186
+ expect(
187
+ EventsIndex.root.compose({id: 1, category: {id: 2, licenses: {id: 3, name: 'Name'}}})
188
+ ).to eq('id' => 1, 'category' => {'id' => 2, 'licenses' => {'id' => 3, 'name' => 'Name'}})
187
189
  end
188
190
 
189
191
  specify do
190
- expect(EventsIndex::Event.root.compose(id: 1, category: [
191
- {id: 2, 'licenses' => {id: 3, name: 'Name1'}},
192
- {id: 4, licenses: nil}
193
- ])).to eq('id' => 1, 'category' => [
192
+ expect(
193
+ EventsIndex.root.compose({id: 1, category: [
194
+ {id: 2, 'licenses' => {id: 3, name: 'Name1'}},
195
+ {id: 4, licenses: nil}
196
+ ]})
197
+ ).to eq('id' => 1, 'category' => [
194
198
  {'id' => 2, 'licenses' => {'id' => 3, 'name' => 'Name1'}},
195
199
  {'id' => 4, 'licenses' => nil.as_json}
196
200
  ])
197
201
  end
198
202
 
199
203
  specify do
200
- expect(EventsIndex::Event.root.compose('id' => 1, category: {id: 2, licenses: [
201
- {id: 3, name: 'Name1'}, {id: 4, name: 'Name2'}
202
- ]})).to eq('id' => 1, 'category' => {'id' => 2, 'licenses' => [
203
- {'id' => 3, 'name' => 'Name1'}, {'id' => 4, 'name' => 'Name2'}
204
- ]})
204
+ expect(
205
+ EventsIndex.root.compose({
206
+ 'id' => 1,
207
+ category: {
208
+ id: 2, licenses: [
209
+ {id: 3, name: 'Name1'},
210
+ {id: 4, name: 'Name2'}
211
+ ]
212
+ }
213
+ })
214
+ ).to eq(
215
+ 'id' => 1,
216
+ 'category' => {
217
+ 'id' => 2,
218
+ 'licenses' => [
219
+ {'id' => 3, 'name' => 'Name1'},
220
+ {'id' => 4, 'name' => 'Name2'}
221
+ ]
222
+ }
223
+ )
205
224
  end
206
225
 
207
226
  specify do
208
- expect(EventsIndex::Event.root.compose(id: 1, category: [
209
- {id: 2, licenses: [
210
- {id: 3, 'name' => 'Name1'}, {id: 4, name: 'Name2'}
211
- ]},
212
- {id: 5, licenses: []}
213
- ])).to eq('id' => 1, 'category' => [
214
- {'id' => 2, 'licenses' => [
215
- {'id' => 3, 'name' => 'Name1'}, {'id' => 4, 'name' => 'Name2'}
216
- ]},
217
- {'id' => 5, 'licenses' => []}
218
- ])
227
+ expect(
228
+ EventsIndex.root.compose({id: 1, category: [
229
+ {id: 2, licenses: [
230
+ {id: 3, 'name' => 'Name1'}, {id: 4, name: 'Name2'}
231
+ ]},
232
+ {id: 5, licenses: []}
233
+ ]})
234
+ ).to eq(
235
+ 'id' => 1,
236
+ 'category' => [
237
+ {'id' => 2, 'licenses' => [
238
+ {'id' => 3, 'name' => 'Name1'},
239
+ {'id' => 4, 'name' => 'Name2'}
240
+ ]},
241
+ {'id' => 5, 'licenses' => []}
242
+ ]
243
+ )
219
244
  end
220
-
221
245
  specify do
222
- expect(EventsIndex::Event.root.compose(
223
- double(id: 1, category: double(id: 2, licenses: double(id: 3, name: 'Name')))
224
- )).to eq('id' => 1, 'category' => {'id' => 2, 'licenses' => {'id' => 3, 'name' => 'Name'}})
246
+ expect(
247
+ EventsIndex.root.compose(double(id: 1, category: double(id: 2, licenses: double(id: 3, name: 'Name'))))
248
+ ).to eq('id' => 1, 'category' => {'id' => 2, 'licenses' => {'id' => 3, 'name' => 'Name'}})
225
249
  end
226
250
 
227
251
  specify do
228
- expect(EventsIndex::Event.root.compose(double(id: 1, category: [
229
- double(id: 2, licenses: double(id: 3, name: 'Name1')),
230
- double(id: 4, licenses: nil)
231
- ]))).to eq('id' => 1, 'category' => [
252
+ expect(
253
+ EventsIndex.root.compose(double(id: 1, category: [
254
+ double(id: 2, licenses: double(id: 3, name: 'Name1')),
255
+ double(id: 4, licenses: nil)
256
+ ]))
257
+ ).to eq('id' => 1, 'category' => [
232
258
  {'id' => 2, 'licenses' => {'id' => 3, 'name' => 'Name1'}},
233
259
  {'id' => 4, 'licenses' => nil.as_json}
234
260
  ])
235
261
  end
236
262
 
237
263
  specify do
238
- expect(EventsIndex::Event.root.compose(double(id: 1, category: double(id: 2, licenses: [
239
- double(id: 3, name: 'Name1'), double(id: 4, name: 'Name2')
240
- ])))).to eq('id' => 1, 'category' => {'id' => 2, 'licenses' => [
264
+ expect(
265
+ EventsIndex.root.compose(double(id: 1, category: double(id: 2, licenses: [
266
+ double(id: 3, name: 'Name1'), double(id: 4, name: 'Name2')
267
+ ])))
268
+ ).to eq('id' => 1, 'category' => {'id' => 2, 'licenses' => [
241
269
  {'id' => 3, 'name' => 'Name1'}, {'id' => 4, 'name' => 'Name2'}
242
270
  ]})
243
271
  end
244
272
 
245
273
  specify do
246
- expect(EventsIndex::Event.root.compose(double(id: 1, category: [
247
- double(id: 2, licenses: [
248
- double(id: 3, name: 'Name1'), double(id: 4, name: 'Name2')
249
- ]),
250
- double(id: 5, licenses: [])
251
- ]))).to eq('id' => 1, 'category' => [
252
- {'id' => 2, 'licenses' => [
253
- {'id' => 3, 'name' => 'Name1'}, {'id' => 4, 'name' => 'Name2'}
254
- ]},
255
- {'id' => 5, 'licenses' => []}
256
- ])
274
+ expect(
275
+ EventsIndex.root.compose(double(id: 1, category: [
276
+ double(id: 2, licenses: [
277
+ double(id: 3, name: 'Name1'), double(id: 4, name: 'Name2')
278
+ ]),
279
+ double(id: 5, licenses: [])
280
+ ]))
281
+ ).to eq(
282
+ 'id' => 1, 'category' => [
283
+ {'id' => 2, 'licenses' => [
284
+ {'id' => 3, 'name' => 'Name1'}, {'id' => 4, 'name' => 'Name2'}
285
+ ]},
286
+ {'id' => 5, 'licenses' => []}
287
+ ]
288
+ )
257
289
  end
258
290
  end
259
291
 
260
292
  context 'custom methods' do
261
293
  before do
262
294
  stub_index(:events) do
263
- define_type :event do
295
+ field :id, type: 'integer'
296
+ field :category, value: -> { categories } do
264
297
  field :id, type: 'integer'
265
- field :category, value: -> { categories } do
298
+ field :licenses, value: -> { license } do
266
299
  field :id, type: 'integer'
267
- field :licenses, value: -> { license } do
268
- field :id, type: 'integer'
269
- field :name
270
- end
300
+ field :name
271
301
  end
272
302
  end
273
303
  end
274
304
  end
275
305
 
276
306
  specify do
277
- expect(EventsIndex::Event.root.compose(
278
- double(id: 1, categories: double(id: 2, license: double(id: 3, name: 'Name')))
279
- )).to eq('id' => 1, 'category' => {'id' => 2, 'licenses' => {'id' => 3, 'name' => 'Name'}})
307
+ expect(
308
+ EventsIndex.root.compose(
309
+ double(
310
+ id: 1, categories: double(
311
+ id: 2, license: double(
312
+ id: 3, name: 'Name'
313
+ )
314
+ )
315
+ )
316
+ )
317
+ ).to eq('id' => 1, 'category' => {'id' => 2, 'licenses' => {'id' => 3, 'name' => 'Name'}})
280
318
  end
281
319
  end
282
320
 
283
321
  context 'objects and multi_fields' do
284
322
  before do
285
323
  stub_index(:events) do
286
- define_type :event do
287
- field :id, type: 'integer'
288
- field :name, type: 'integer' do
289
- field :raw, analyzer: 'my_own'
290
- end
291
- field :category, type: 'object'
324
+ field :id, type: 'integer'
325
+ field :name, type: 'integer' do
326
+ field :raw, analyzer: 'my_own'
292
327
  end
328
+ field :category, type: 'object'
293
329
  end
294
330
  end
295
331
 
296
332
  specify do
297
- expect(EventsIndex::Event.mappings_hash).to eq(event: {
298
- properties: {
299
- id: {type: 'integer'},
300
- name: {
301
- type: 'integer',
302
- fields: {
303
- raw: {analyzer: 'my_own', type: Chewy.default_field_type}
304
- }
305
- },
306
- category: {type: 'object'}
333
+ expect(EventsIndex.mappings_hash).to eq(
334
+ mappings: {
335
+ properties: {
336
+ id: {type: 'integer'},
337
+ name: {
338
+ type: 'integer',
339
+ fields: {
340
+ raw: {analyzer: 'my_own', type: Chewy.default_field_type}
341
+ }
342
+ },
343
+ category: {type: 'object'}
344
+ }
307
345
  }
308
- })
346
+ )
309
347
  end
310
348
 
311
349
  specify do
312
- expect(EventsIndex::Event.root.compose(
313
- double(id: 1, name: 'Jonny', category: double(id: 2, as_json: {'name' => 'Borogoves'}))
314
- )).to eq(
350
+ expect(
351
+ EventsIndex.root.compose(
352
+ double(
353
+ id: 1, name: 'Jonny', category: double(
354
+ id: 2, as_json: {'name' => 'Borogoves'}
355
+ )
356
+ )
357
+ )
358
+ ).to eq(
315
359
  'id' => 1,
316
360
  'name' => 'Jonny',
317
361
  'category' => {'name' => 'Borogoves'}
@@ -319,12 +363,14 @@ describe Chewy::Fields::Base do
319
363
  end
320
364
 
321
365
  specify do
322
- expect(EventsIndex::Event.root.compose(
323
- double(id: 1, name: 'Jonny', category: [
324
- double(id: 2, as_json: {'name' => 'Borogoves1'}),
325
- double(id: 3, as_json: {'name' => 'Borogoves2'})
326
- ])
327
- )).to eq(
366
+ expect(
367
+ EventsIndex.root.compose(
368
+ double(id: 1, name: 'Jonny', category: [
369
+ double(id: 2, as_json: {'name' => 'Borogoves1'}),
370
+ double(id: 3, as_json: {'name' => 'Borogoves2'})
371
+ ])
372
+ )
373
+ ).to eq(
328
374
  'id' => 1,
329
375
  'name' => 'Jonny',
330
376
  'category' => [
@@ -335,56 +381,298 @@ describe Chewy::Fields::Base do
335
381
  end
336
382
  end
337
383
 
338
- context 'objects and scopes', :orm do
384
+ context 'ignore_blank option for field method', :orm do
339
385
  before do
386
+ stub_model(:location)
340
387
  stub_model(:city)
341
388
  stub_model(:country)
342
389
 
343
- case adapter
344
- when :active_record
345
- City.belongs_to :country
346
- if ActiveRecord::VERSION::MAJOR >= 4
347
- Country.has_many :cities, -> { order :id }
348
- else
349
- Country.has_many :cities, order: :id
350
- end
351
- when :mongoid
352
- if Mongoid::VERSION.start_with?('6')
353
- City.belongs_to :country, optional: true
354
- else
355
- City.belongs_to :country
356
- end
357
- Country.has_many :cities, order: :id.asc
358
- when :sequel
359
- City.many_to_one :country
360
- Country.one_to_many :cities, order: :id
361
- end
390
+ City.belongs_to :country
391
+ Location.belongs_to :city
392
+ City.has_one :location, -> { order :id }
393
+ Country.has_many :cities, -> { order :id }
394
+ end
362
395
 
363
- stub_index(:countries) do
364
- define_type Country do
396
+ context 'text fields with and without ignore_blank option' do
397
+ before do
398
+ stub_index(:countries) do
399
+ index_scope Country
365
400
  field :id
366
401
  field :cities do
367
402
  field :id
368
403
  field :name
404
+ field :historical_name, ignore_blank: false
405
+ field :description, ignore_blank: true
369
406
  end
370
407
  end
371
408
  end
409
+
410
+ let(:country_with_cities) do
411
+ cities = [
412
+ City.create!(id: 1, name: '', historical_name: '', description: ''),
413
+ City.create!(id: 2, name: '', historical_name: '', description: '')
414
+ ]
415
+
416
+ Country.create!(id: 1, cities: cities)
417
+ end
418
+
419
+ specify do
420
+ expect(CountriesIndex.root.compose(country_with_cities)).to eq(
421
+ 'id' => 1, 'cities' => [
422
+ {'id' => 1, 'name' => '', 'historical_name' => ''},
423
+ {'id' => 2, 'name' => '', 'historical_name' => ''}
424
+ ]
425
+ )
426
+ end
427
+ specify { expect(CountriesIndex.reset).to eq(true) }
372
428
  end
373
429
 
374
- let(:country_with_cities) do
375
- cities = [City.create!(id: 1, name: 'City1'), City.create!(id: 2, name: 'City2')]
430
+ context 'nested fields' do
431
+ context 'with ignore_blank: true option' do
432
+ before do
433
+ stub_index(:countries) do
434
+ index_scope Country
435
+ field :id
436
+ field :cities, ignore_blank: true do
437
+ field :id
438
+ field :name
439
+ field :historical_name, ignore_blank: true
440
+ field :description
441
+ end
442
+ end
443
+ end
376
444
 
377
- if adapter == :sequel
378
- Country.create(id: 1).tap do |country|
379
- cities.each { |city| country.add_city(city) }
445
+ let(:country) { Country.create!(id: 1, cities: cities) }
446
+ context('without cities') do
447
+ let(:cities) { [] }
448
+ specify do
449
+ expect(CountriesIndex.root.compose(country))
450
+ .to eq('id' => 1)
451
+ end
452
+ end
453
+ context('with cities') do
454
+ let(:cities) { [City.create!(id: 1, name: '', historical_name: '')] }
455
+ specify do
456
+ expect(CountriesIndex.root.compose(country)).to eq(
457
+ 'id' => 1, 'cities' => [
458
+ {'id' => 1, 'name' => '', 'description' => nil}
459
+ ]
460
+ )
461
+ end
462
+ end
463
+ end
464
+
465
+ context 'with ignore_blank: false option' do
466
+ before do
467
+ stub_index(:countries) do
468
+ index_scope Country
469
+ field :id
470
+ field :cities, ignore_blank: false do
471
+ field :id
472
+ field :name
473
+ field :historical_name
474
+ field :description
475
+ end
476
+ end
477
+ end
478
+
479
+ let(:country_with_cities) { Country.create!(id: 1) }
480
+
481
+ specify do
482
+ expect(CountriesIndex.root.compose(country_with_cities))
483
+ .to eq('id' => 1, 'cities' => [])
484
+ end
485
+ end
486
+
487
+ context 'without ignore_blank: true option' do
488
+ before do
489
+ stub_index(:countries) do
490
+ index_scope Country
491
+ field :id
492
+ field :cities do
493
+ field :id
494
+ field :name
495
+ field :historical_name
496
+ field :description
497
+ end
498
+ end
499
+ end
500
+
501
+ let(:country_with_cities) { Country.create!(id: 1) }
502
+
503
+ specify do
504
+ expect(CountriesIndex.root.compose(country_with_cities))
505
+ .to eq('id' => 1, 'cities' => [])
506
+ end
507
+ end
508
+ end
509
+
510
+ context 'geo_point field type' do
511
+ context 'with ignore_blank: true option' do
512
+ before do
513
+ stub_index(:countries) do
514
+ index_scope Country
515
+ field :id
516
+ field :cities do
517
+ field :id
518
+ field :name
519
+ field :location, type: :geo_point, ignore_blank: true do
520
+ field :lat
521
+ field :lon
522
+ end
523
+ end
524
+ end
525
+ end
526
+
527
+ specify do
528
+ expect(
529
+ CountriesIndex.root.compose({
530
+ 'id' => 1,
531
+ 'cities' => [
532
+ {'id' => 1, 'name' => 'City1', 'location' => {}},
533
+ {'id' => 2, 'name' => 'City2', 'location' => {}}
534
+ ]
535
+ })
536
+ ).to eq(
537
+ 'id' => 1, 'cities' => [
538
+ {'id' => 1, 'name' => 'City1'},
539
+ {'id' => 2, 'name' => 'City2'}
540
+ ]
541
+ )
542
+ end
543
+ end
544
+
545
+ context 'join field type' do
546
+ before do
547
+ stub_model(:comment)
548
+ stub_index(:comments) do
549
+ index_scope Comment
550
+ field :id
551
+ field :hierarchy, type: :join, relations: {question: %i[answer comment], answer: :vote, vote: :subvote}, join: {type: :comment_type, id: :commented_id}
552
+ end
553
+ end
554
+
555
+ specify do
556
+ expect(
557
+ CommentsIndex.root.compose(
558
+ {'id' => 1, 'comment_type' => 'question'}
559
+ )
560
+ ).to eq(
561
+ {'id' => 1, 'hierarchy' => 'question'}
562
+ )
563
+
564
+ expect(
565
+ CommentsIndex.root.compose(
566
+ {'id' => 2, 'comment_type' => 'answer', 'commented_id' => 1}
567
+ )
568
+ ).to eq(
569
+ {'id' => 2, 'hierarchy' => {'name' => 'answer', 'parent' => 1}}
570
+ )
571
+
572
+ expect do
573
+ CommentsIndex.root.compose(
574
+ {'id' => 2, 'comment_type' => 'asd', 'commented_id' => 1}
575
+ )
576
+ end.to raise_error Chewy::InvalidJoinFieldType
577
+ end
578
+ end
579
+
580
+ context 'without ignore_blank option' do
581
+ before do
582
+ stub_index(:countries) do
583
+ index_scope Country
584
+ field :id
585
+ field :cities do
586
+ field :id
587
+ field :name
588
+ field :location, type: :geo_point do
589
+ field :lat
590
+ field :lon
591
+ end
592
+ end
593
+ end
594
+ end
595
+
596
+ specify do
597
+ expect(
598
+ CountriesIndex.root.compose({
599
+ 'id' => 1,
600
+ 'cities' => [
601
+ {'id' => 1, 'name' => 'City1', 'location' => {}},
602
+ {'id' => 2, 'name' => 'City2', 'location' => {}}
603
+ ]
604
+ })
605
+ ).to eq(
606
+ 'id' => 1, 'cities' => [
607
+ {'id' => 1, 'name' => 'City1'},
608
+ {'id' => 2, 'name' => 'City2'}
609
+ ]
610
+ )
611
+ end
612
+ end
613
+
614
+ context 'with ignore_blank: false flag' do
615
+ before do
616
+ stub_index(:countries) do
617
+ index_scope Country
618
+ field :id
619
+ field :cities do
620
+ field :id
621
+ field :name
622
+ field :location, type: :geo_point, ignore_blank: false do
623
+ field :lat
624
+ field :lon
625
+ end
626
+ end
627
+ end
628
+ end
629
+
630
+ specify do
631
+ expect(
632
+ CountriesIndex.root.compose({
633
+ 'id' => 1,
634
+ 'cities' => [
635
+ {'id' => 1, 'location' => {}, 'name' => 'City1'},
636
+ {'id' => 2, 'location' => '', 'name' => 'City2'}
637
+ ]
638
+ })
639
+ ).to eq(
640
+ 'id' => 1, 'cities' => [
641
+ {'id' => 1, 'location' => {}, 'name' => 'City1'},
642
+ {'id' => 2, 'location' => '', 'name' => 'City2'}
643
+ ]
644
+ )
380
645
  end
381
- else
382
- Country.create!(id: 1, cities: cities)
383
646
  end
384
647
  end
648
+ end
649
+
650
+ context 'objects and scopes', :orm do
651
+ before do
652
+ stub_model(:city)
653
+ stub_model(:country)
654
+
655
+ City.belongs_to :country
656
+ Country.has_many :cities, -> { order :id }
657
+
658
+ stub_index(:countries) do
659
+ index_scope Country
660
+ field :id
661
+ field :cities do
662
+ field :id
663
+ field :name
664
+ end
665
+ end
666
+ end
667
+
668
+ let(:country_with_cities) do
669
+ cities = [City.create!(id: 1, name: 'City1'), City.create!(id: 2, name: 'City2')]
670
+
671
+ Country.create!(id: 1, cities: cities)
672
+ end
385
673
 
386
674
  specify do
387
- expect(CountriesIndex::Country.root.compose(country_with_cities)).to eq('id' => 1, 'cities' => [
675
+ expect(CountriesIndex.root.compose(country_with_cities)).to eq('id' => 1, 'cities' => [
388
676
  {'id' => 1, 'name' => 'City1'}, {'id' => 2, 'name' => 'City2'}
389
677
  ])
390
678
  end
@@ -392,20 +680,19 @@ describe Chewy::Fields::Base do
392
680
  context 'nested object' do
393
681
  before do
394
682
  stub_index(:cities) do
395
- define_type City do
683
+ index_scope City
684
+ field :id
685
+ field :country do
396
686
  field :id
397
- field :country do
398
- field :id
399
- field :name
400
- end
687
+ field :name
401
688
  end
402
689
  end
403
690
  end
404
691
 
405
692
  specify do
406
- expect(CitiesIndex::City.root.compose(
407
- City.create!(id: 1, country: Country.create!(id: 1, name: 'Country'))
408
- )).to eq('id' => 1, 'country' => {'id' => 1, 'name' => 'Country'})
693
+ expect(
694
+ CitiesIndex.root.compose(City.create!(id: 1, country: Country.create!(id: 1, name: 'Country')))
695
+ ).to eq('id' => 1, 'country' => {'id' => 1, 'name' => 'Country'})
409
696
  end
410
697
  end
411
698
  end