chewy 0.8.4 → 5.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (303) hide show
  1. checksums.yaml +5 -5
  2. data/.gitignore +1 -0
  3. data/.rubocop.yml +56 -0
  4. data/.rubocop_todo.yml +44 -0
  5. data/.travis.yml +36 -67
  6. data/.yardopts +5 -0
  7. data/Appraisals +63 -58
  8. data/CHANGELOG.md +168 -11
  9. data/Gemfile +16 -9
  10. data/Guardfile +5 -5
  11. data/LEGACY_DSL.md +497 -0
  12. data/README.md +403 -470
  13. data/Rakefile +11 -1
  14. data/chewy.gemspec +12 -15
  15. data/gemfiles/rails.4.0.activerecord.gemfile +9 -9
  16. data/gemfiles/rails.4.1.activerecord.gemfile +9 -9
  17. data/gemfiles/rails.4.2.activerecord.gemfile +8 -8
  18. data/gemfiles/rails.4.2.mongoid.5.2.gemfile +16 -0
  19. data/gemfiles/rails.5.0.activerecord.gemfile +16 -0
  20. data/gemfiles/rails.5.0.mongoid.6.1.gemfile +16 -0
  21. data/gemfiles/rails.5.1.activerecord.gemfile +16 -0
  22. data/gemfiles/rails.5.1.mongoid.6.3.gemfile +16 -0
  23. data/gemfiles/rails.5.2.activerecord.gemfile +16 -0
  24. data/gemfiles/sequel.4.45.gemfile +11 -0
  25. data/lib/chewy/backports/deep_dup.rb +1 -1
  26. data/lib/chewy/backports/duplicable.rb +1 -0
  27. data/lib/chewy/config.rb +53 -21
  28. data/lib/chewy/errors.rb +6 -6
  29. data/lib/chewy/fields/base.rb +59 -29
  30. data/lib/chewy/fields/root.rb +49 -14
  31. data/lib/chewy/index/actions.rb +95 -36
  32. data/lib/chewy/index/aliases.rb +2 -1
  33. data/lib/chewy/index/settings.rb +10 -5
  34. data/lib/chewy/index/specification.rb +60 -0
  35. data/lib/chewy/index.rb +239 -138
  36. data/lib/chewy/journal.rb +55 -0
  37. data/lib/chewy/log_subscriber.rb +8 -8
  38. data/lib/chewy/minitest/helpers.rb +77 -0
  39. data/lib/chewy/minitest/search_index_receiver.rb +80 -0
  40. data/lib/chewy/minitest.rb +1 -0
  41. data/lib/chewy/query/compose.rb +18 -19
  42. data/lib/chewy/query/criteria.rb +34 -24
  43. data/lib/chewy/query/filters.rb +28 -11
  44. data/lib/chewy/query/loading.rb +3 -4
  45. data/lib/chewy/query/nodes/and.rb +1 -1
  46. data/lib/chewy/query/nodes/base.rb +1 -1
  47. data/lib/chewy/query/nodes/bool.rb +6 -4
  48. data/lib/chewy/query/nodes/equal.rb +4 -4
  49. data/lib/chewy/query/nodes/exists.rb +1 -1
  50. data/lib/chewy/query/nodes/expr.rb +2 -2
  51. data/lib/chewy/query/nodes/field.rb +35 -31
  52. data/lib/chewy/query/nodes/has_child.rb +1 -0
  53. data/lib/chewy/query/nodes/has_parent.rb +1 -0
  54. data/lib/chewy/query/nodes/has_relation.rb +10 -12
  55. data/lib/chewy/query/nodes/missing.rb +1 -1
  56. data/lib/chewy/query/nodes/not.rb +1 -1
  57. data/lib/chewy/query/nodes/or.rb +1 -1
  58. data/lib/chewy/query/nodes/prefix.rb +3 -2
  59. data/lib/chewy/query/nodes/query.rb +1 -1
  60. data/lib/chewy/query/nodes/range.rb +9 -9
  61. data/lib/chewy/query/nodes/raw.rb +1 -1
  62. data/lib/chewy/query/nodes/regexp.rb +13 -9
  63. data/lib/chewy/query/nodes/script.rb +4 -4
  64. data/lib/chewy/query/pagination.rb +10 -1
  65. data/lib/chewy/query.rb +286 -170
  66. data/lib/chewy/railtie.rb +7 -6
  67. data/lib/chewy/rake_helper.rb +275 -37
  68. data/lib/chewy/repository.rb +2 -2
  69. data/lib/chewy/rspec/update_index.rb +70 -65
  70. data/lib/chewy/rspec.rb +1 -1
  71. data/lib/chewy/runtime/version.rb +4 -4
  72. data/lib/chewy/search/loader.rb +83 -0
  73. data/lib/chewy/{query → search}/pagination/kaminari.rb +13 -5
  74. data/lib/chewy/search/pagination/will_paginate.rb +43 -0
  75. data/lib/chewy/search/parameters/aggs.rb +16 -0
  76. data/lib/chewy/search/parameters/allow_partial_search_results.rb +27 -0
  77. data/lib/chewy/search/parameters/concerns/bool_storage.rb +24 -0
  78. data/lib/chewy/search/parameters/concerns/hash_storage.rb +23 -0
  79. data/lib/chewy/search/parameters/concerns/integer_storage.rb +14 -0
  80. data/lib/chewy/search/parameters/concerns/query_storage.rb +238 -0
  81. data/lib/chewy/search/parameters/concerns/string_array_storage.rb +23 -0
  82. data/lib/chewy/search/parameters/concerns/string_storage.rb +14 -0
  83. data/lib/chewy/search/parameters/docvalue_fields.rb +12 -0
  84. data/lib/chewy/search/parameters/explain.rb +16 -0
  85. data/lib/chewy/search/parameters/filter.rb +47 -0
  86. data/lib/chewy/search/parameters/highlight.rb +16 -0
  87. data/lib/chewy/search/parameters/indices.rb +123 -0
  88. data/lib/chewy/search/parameters/indices_boost.rb +52 -0
  89. data/lib/chewy/search/parameters/limit.rb +17 -0
  90. data/lib/chewy/search/parameters/load.rb +32 -0
  91. data/lib/chewy/search/parameters/min_score.rb +16 -0
  92. data/lib/chewy/search/parameters/none.rb +27 -0
  93. data/lib/chewy/search/parameters/offset.rb +17 -0
  94. data/lib/chewy/search/parameters/order.rb +64 -0
  95. data/lib/chewy/search/parameters/post_filter.rb +19 -0
  96. data/lib/chewy/search/parameters/preference.rb +16 -0
  97. data/lib/chewy/search/parameters/profile.rb +16 -0
  98. data/lib/chewy/search/parameters/query.rb +19 -0
  99. data/lib/chewy/search/parameters/request_cache.rb +27 -0
  100. data/lib/chewy/search/parameters/rescore.rb +29 -0
  101. data/lib/chewy/search/parameters/script_fields.rb +16 -0
  102. data/lib/chewy/search/parameters/search_after.rb +20 -0
  103. data/lib/chewy/search/parameters/search_type.rb +16 -0
  104. data/lib/chewy/search/parameters/source.rb +73 -0
  105. data/lib/chewy/search/parameters/storage.rb +95 -0
  106. data/lib/chewy/search/parameters/stored_fields.rb +63 -0
  107. data/lib/chewy/search/parameters/suggest.rb +16 -0
  108. data/lib/chewy/search/parameters/terminate_after.rb +16 -0
  109. data/lib/chewy/search/parameters/timeout.rb +16 -0
  110. data/lib/chewy/search/parameters/track_scores.rb +16 -0
  111. data/lib/chewy/search/parameters/types.rb +20 -0
  112. data/lib/chewy/search/parameters/version.rb +16 -0
  113. data/lib/chewy/search/parameters.rb +167 -0
  114. data/lib/chewy/search/query_proxy.rb +257 -0
  115. data/lib/chewy/search/request.rb +1045 -0
  116. data/lib/chewy/search/response.rb +119 -0
  117. data/lib/chewy/search/scoping.rb +50 -0
  118. data/lib/chewy/search/scrolling.rb +134 -0
  119. data/lib/chewy/search.rb +81 -26
  120. data/lib/chewy/stash.rb +79 -0
  121. data/lib/chewy/strategy/active_job.rb +1 -0
  122. data/lib/chewy/strategy/atomic.rb +2 -4
  123. data/lib/chewy/strategy/base.rb +4 -4
  124. data/lib/chewy/strategy/bypass.rb +1 -2
  125. data/lib/chewy/strategy/resque.rb +1 -0
  126. data/lib/chewy/strategy/shoryuken.rb +40 -0
  127. data/lib/chewy/strategy/sidekiq.rb +13 -1
  128. data/lib/chewy/strategy/urgent.rb +1 -1
  129. data/lib/chewy/strategy.rb +19 -10
  130. data/lib/chewy/type/actions.rb +26 -2
  131. data/lib/chewy/type/adapter/active_record.rb +50 -24
  132. data/lib/chewy/type/adapter/base.rb +29 -9
  133. data/lib/chewy/type/adapter/mongoid.rb +19 -10
  134. data/lib/chewy/type/adapter/object.rb +195 -31
  135. data/lib/chewy/type/adapter/orm.rb +69 -33
  136. data/lib/chewy/type/adapter/sequel.rb +37 -19
  137. data/lib/chewy/type/crutch.rb +5 -4
  138. data/lib/chewy/type/import/bulk_builder.rb +122 -0
  139. data/lib/chewy/type/import/bulk_request.rb +78 -0
  140. data/lib/chewy/type/import/journal_builder.rb +45 -0
  141. data/lib/chewy/type/import/routine.rb +138 -0
  142. data/lib/chewy/type/import.rb +150 -176
  143. data/lib/chewy/type/mapping.rb +58 -42
  144. data/lib/chewy/type/observe.rb +21 -15
  145. data/lib/chewy/type/syncer.rb +222 -0
  146. data/lib/chewy/type/witchcraft.rb +89 -34
  147. data/lib/chewy/type/wrapper.rb +48 -16
  148. data/lib/chewy/type.rb +77 -49
  149. data/lib/chewy/version.rb +1 -1
  150. data/lib/chewy.rb +95 -52
  151. data/lib/generators/chewy/install_generator.rb +3 -3
  152. data/lib/sequel/plugins/chewy_observe.rb +4 -19
  153. data/lib/tasks/chewy.rake +91 -28
  154. data/spec/chewy/config_spec.rb +130 -12
  155. data/spec/chewy/fields/base_spec.rb +194 -172
  156. data/spec/chewy/fields/root_spec.rb +123 -17
  157. data/spec/chewy/fields/time_fields_spec.rb +10 -9
  158. data/spec/chewy/index/actions_spec.rb +228 -43
  159. data/spec/chewy/index/aliases_spec.rb +2 -2
  160. data/spec/chewy/index/settings_spec.rb +100 -49
  161. data/spec/chewy/index/specification_spec.rb +169 -0
  162. data/spec/chewy/index_spec.rb +159 -63
  163. data/spec/chewy/journal_spec.rb +268 -0
  164. data/spec/chewy/minitest/helpers_spec.rb +90 -0
  165. data/spec/chewy/minitest/search_index_receiver_spec.rb +120 -0
  166. data/spec/chewy/query/criteria_spec.rb +503 -236
  167. data/spec/chewy/query/filters_spec.rb +96 -68
  168. data/spec/chewy/query/loading_spec.rb +80 -42
  169. data/spec/chewy/query/nodes/and_spec.rb +3 -7
  170. data/spec/chewy/query/nodes/bool_spec.rb +5 -13
  171. data/spec/chewy/query/nodes/equal_spec.rb +20 -20
  172. data/spec/chewy/query/nodes/exists_spec.rb +7 -7
  173. data/spec/chewy/query/nodes/has_child_spec.rb +42 -23
  174. data/spec/chewy/query/nodes/has_parent_spec.rb +42 -23
  175. data/spec/chewy/query/nodes/match_all_spec.rb +2 -2
  176. data/spec/chewy/query/nodes/missing_spec.rb +6 -5
  177. data/spec/chewy/query/nodes/not_spec.rb +5 -7
  178. data/spec/chewy/query/nodes/or_spec.rb +3 -7
  179. data/spec/chewy/query/nodes/prefix_spec.rb +6 -6
  180. data/spec/chewy/query/nodes/query_spec.rb +3 -3
  181. data/spec/chewy/query/nodes/range_spec.rb +19 -19
  182. data/spec/chewy/query/nodes/raw_spec.rb +2 -2
  183. data/spec/chewy/query/nodes/regexp_spec.rb +31 -19
  184. data/spec/chewy/query/nodes/script_spec.rb +5 -5
  185. data/spec/chewy/query/pagination/kaminari_spec.rb +3 -55
  186. data/spec/chewy/query/pagination/will_paginate_spec.rb +5 -0
  187. data/spec/chewy/query/pagination_spec.rb +25 -22
  188. data/spec/chewy/query_spec.rb +510 -505
  189. data/spec/chewy/rake_helper_spec.rb +381 -0
  190. data/spec/chewy/repository_spec.rb +8 -8
  191. data/spec/chewy/rspec/update_index_spec.rb +215 -113
  192. data/spec/chewy/runtime_spec.rb +2 -2
  193. data/spec/chewy/search/loader_spec.rb +117 -0
  194. data/spec/chewy/search/pagination/kaminari_examples.rb +71 -0
  195. data/spec/chewy/search/pagination/kaminari_spec.rb +21 -0
  196. data/spec/chewy/search/pagination/will_paginate_examples.rb +63 -0
  197. data/spec/chewy/search/pagination/will_paginate_spec.rb +23 -0
  198. data/spec/chewy/search/parameters/aggs_spec.rb +5 -0
  199. data/spec/chewy/search/parameters/bool_storage_examples.rb +53 -0
  200. data/spec/chewy/search/parameters/docvalue_fields_spec.rb +5 -0
  201. data/spec/chewy/search/parameters/explain_spec.rb +5 -0
  202. data/spec/chewy/search/parameters/filter_spec.rb +5 -0
  203. data/spec/chewy/search/parameters/hash_storage_examples.rb +59 -0
  204. data/spec/chewy/search/parameters/highlight_spec.rb +5 -0
  205. data/spec/chewy/search/parameters/indices_spec.rb +191 -0
  206. data/spec/chewy/search/parameters/integer_storage_examples.rb +32 -0
  207. data/spec/chewy/search/parameters/limit_spec.rb +5 -0
  208. data/spec/chewy/search/parameters/load_spec.rb +60 -0
  209. data/spec/chewy/search/parameters/min_score_spec.rb +32 -0
  210. data/spec/chewy/search/parameters/none_spec.rb +5 -0
  211. data/spec/chewy/search/parameters/offset_spec.rb +5 -0
  212. data/spec/chewy/search/parameters/order_spec.rb +65 -0
  213. data/spec/chewy/search/parameters/post_filter_spec.rb +5 -0
  214. data/spec/chewy/search/parameters/preference_spec.rb +5 -0
  215. data/spec/chewy/search/parameters/profile_spec.rb +5 -0
  216. data/spec/chewy/search/parameters/query_spec.rb +5 -0
  217. data/spec/chewy/search/parameters/query_storage_examples.rb +388 -0
  218. data/spec/chewy/search/parameters/request_cache_spec.rb +67 -0
  219. data/spec/chewy/search/parameters/rescore_spec.rb +62 -0
  220. data/spec/chewy/search/parameters/script_fields_spec.rb +5 -0
  221. data/spec/chewy/search/parameters/search_after_spec.rb +32 -0
  222. data/spec/chewy/search/parameters/search_type_spec.rb +5 -0
  223. data/spec/chewy/search/parameters/source_spec.rb +156 -0
  224. data/spec/chewy/search/parameters/storage_spec.rb +60 -0
  225. data/spec/chewy/search/parameters/stored_fields_spec.rb +126 -0
  226. data/spec/chewy/search/parameters/string_array_storage_examples.rb +63 -0
  227. data/spec/chewy/search/parameters/string_storage_examples.rb +32 -0
  228. data/spec/chewy/search/parameters/suggest_spec.rb +5 -0
  229. data/spec/chewy/search/parameters/terminate_after_spec.rb +5 -0
  230. data/spec/chewy/search/parameters/timeout_spec.rb +5 -0
  231. data/spec/chewy/search/parameters/track_scores_spec.rb +5 -0
  232. data/spec/chewy/search/parameters/types_spec.rb +5 -0
  233. data/spec/chewy/search/parameters/version_spec.rb +5 -0
  234. data/spec/chewy/search/parameters_spec.rb +145 -0
  235. data/spec/chewy/search/query_proxy_spec.rb +68 -0
  236. data/spec/chewy/search/request_spec.rb +685 -0
  237. data/spec/chewy/search/response_spec.rb +192 -0
  238. data/spec/chewy/search/scrolling_spec.rb +169 -0
  239. data/spec/chewy/search_spec.rb +37 -20
  240. data/spec/chewy/stash_spec.rb +95 -0
  241. data/spec/chewy/strategy/active_job_spec.rb +8 -2
  242. data/spec/chewy/strategy/atomic_spec.rb +4 -1
  243. data/spec/chewy/strategy/resque_spec.rb +8 -2
  244. data/spec/chewy/strategy/shoryuken_spec.rb +66 -0
  245. data/spec/chewy/strategy/sidekiq_spec.rb +10 -2
  246. data/spec/chewy/strategy_spec.rb +6 -6
  247. data/spec/chewy/type/actions_spec.rb +29 -10
  248. data/spec/chewy/type/adapter/active_record_spec.rb +357 -139
  249. data/spec/chewy/type/adapter/mongoid_spec.rb +220 -101
  250. data/spec/chewy/type/adapter/object_spec.rb +129 -40
  251. data/spec/chewy/type/adapter/sequel_spec.rb +304 -152
  252. data/spec/chewy/type/import/bulk_builder_spec.rb +279 -0
  253. data/spec/chewy/type/import/bulk_request_spec.rb +102 -0
  254. data/spec/chewy/type/import/journal_builder_spec.rb +95 -0
  255. data/spec/chewy/type/import/routine_spec.rb +110 -0
  256. data/spec/chewy/type/import_spec.rb +360 -244
  257. data/spec/chewy/type/mapping_spec.rb +96 -29
  258. data/spec/chewy/type/observe_spec.rb +25 -15
  259. data/spec/chewy/type/syncer_spec.rb +123 -0
  260. data/spec/chewy/type/witchcraft_spec.rb +122 -44
  261. data/spec/chewy/type/wrapper_spec.rb +63 -23
  262. data/spec/chewy/type_spec.rb +32 -10
  263. data/spec/chewy_spec.rb +82 -12
  264. data/spec/spec_helper.rb +16 -2
  265. data/spec/support/active_record.rb +6 -2
  266. data/spec/support/class_helpers.rb +4 -19
  267. data/spec/support/mongoid.rb +17 -5
  268. data/spec/support/sequel.rb +6 -1
  269. metadata +250 -57
  270. data/gemfiles/rails.3.2.activerecord.gemfile +0 -15
  271. data/gemfiles/rails.3.2.activerecord.kaminari.gemfile +0 -14
  272. data/gemfiles/rails.3.2.activerecord.will_paginate.gemfile +0 -14
  273. data/gemfiles/rails.4.0.activerecord.kaminari.gemfile +0 -14
  274. data/gemfiles/rails.4.0.activerecord.will_paginate.gemfile +0 -14
  275. data/gemfiles/rails.4.0.mongoid.4.0.0.gemfile +0 -15
  276. data/gemfiles/rails.4.0.mongoid.4.0.0.kaminari.gemfile +0 -14
  277. data/gemfiles/rails.4.0.mongoid.4.0.0.will_paginate.gemfile +0 -14
  278. data/gemfiles/rails.4.0.mongoid.5.1.0.gemfile +0 -15
  279. data/gemfiles/rails.4.0.mongoid.5.1.0.kaminari.gemfile +0 -14
  280. data/gemfiles/rails.4.0.mongoid.5.1.0.will_paginate.gemfile +0 -14
  281. data/gemfiles/rails.4.1.activerecord.kaminari.gemfile +0 -14
  282. data/gemfiles/rails.4.1.activerecord.will_paginate.gemfile +0 -14
  283. data/gemfiles/rails.4.1.mongoid.4.0.0.gemfile +0 -15
  284. data/gemfiles/rails.4.1.mongoid.4.0.0.kaminari.gemfile +0 -14
  285. data/gemfiles/rails.4.1.mongoid.4.0.0.will_paginate.gemfile +0 -14
  286. data/gemfiles/rails.4.1.mongoid.5.1.0.gemfile +0 -15
  287. data/gemfiles/rails.4.1.mongoid.5.1.0.kaminari.gemfile +0 -14
  288. data/gemfiles/rails.4.1.mongoid.5.1.0.will_paginate.gemfile +0 -14
  289. data/gemfiles/rails.4.2.activerecord.kaminari.gemfile +0 -15
  290. data/gemfiles/rails.4.2.activerecord.will_paginate.gemfile +0 -15
  291. data/gemfiles/rails.4.2.mongoid.4.0.0.gemfile +0 -15
  292. data/gemfiles/rails.4.2.mongoid.4.0.0.kaminari.gemfile +0 -14
  293. data/gemfiles/rails.4.2.mongoid.4.0.0.will_paginate.gemfile +0 -14
  294. data/gemfiles/rails.4.2.mongoid.5.1.0.gemfile +0 -15
  295. data/gemfiles/rails.4.2.mongoid.5.1.0.kaminari.gemfile +0 -14
  296. data/gemfiles/rails.4.2.mongoid.5.1.0.will_paginate.gemfile +0 -14
  297. data/gemfiles/rails.5.0.0.beta3.activerecord.gemfile +0 -16
  298. data/gemfiles/rails.5.0.0.beta3.activerecord.kaminari.gemfile +0 -16
  299. data/gemfiles/rails.5.0.0.beta3.activerecord.will_paginate.gemfile +0 -15
  300. data/gemfiles/sequel.4.31.gemfile +0 -13
  301. data/lib/chewy/query/pagination/will_paginate.rb +0 -27
  302. data/lib/chewy/query/scoping.rb +0 -20
  303. data/spec/chewy/query/pagination/will_paginage_spec.rb +0 -60
@@ -1,632 +1,637 @@
1
1
  require 'spec_helper'
2
2
 
3
3
  describe Chewy::Query do
4
- before { Chewy.massacre }
4
+ if Chewy::Runtime.version < '5.0'
5
+ before { Chewy.massacre }
5
6
 
6
- before do
7
- stub_index(:products) do
8
- define_type :product do
9
- field :name, :age
7
+ before do
8
+ stub_index(:products) do
9
+ define_type :product do
10
+ field :name, :age
11
+ end
12
+ define_type :city
13
+ define_type :country
10
14
  end
11
- define_type :city
12
- define_type :country
13
15
  end
14
- end
15
-
16
- subject { described_class.new(ProductsIndex) }
17
-
18
- context 'unexistent index' do
19
- specify { expect(subject.to_a).to eq([]) }
20
- end
21
-
22
- context 'integration' do
23
- let(:products) { 3.times.map { |i| {id: i.next.to_s, name: "Name#{i.next}", age: 10 * i.next}.stringify_keys! } }
24
- let(:cities) { 3.times.map { |i| {id: i.next.to_s}.stringify_keys! } }
25
- let(:countries) { 3.times.map { |i| {id: i.next.to_s}.stringify_keys! } }
26
- before do
27
- ProductsIndex::Product.import!(products.map { |h| double(h) })
28
- ProductsIndex::City.import!(cities.map { |h| double(h) })
29
- ProductsIndex::Country.import!(countries.map { |h| double(h) })
30
- end
31
-
32
- specify { expect(subject.count).to eq(9) }
33
- specify { expect(subject.first._data).to be_a Hash }
34
- specify { expect(subject.limit(6).count).to eq(6) }
35
- specify { expect(subject.offset(6).count).to eq(3) }
36
- specify { expect(subject.query(match: {name: 'name3'}).highlight(fields: {name: {}}).first.name).to eq('Name3') }
37
- specify { expect(subject.query(match: {name: 'name3'}).highlight(fields: {name: {}}).first.name_highlight).to eq('<em>Name3</em>') }
38
- specify { expect(subject.query(match: {name: 'name3'}).highlight(fields: {name: {}}).first._data['_source']['name']).to eq('Name3') }
39
- specify { expect(subject.types(:product).count).to eq(3) }
40
- specify { expect(subject.types(:product, :country).count).to eq(6) }
41
- specify { expect(subject.filter(term: {age: 10}).count).to eq(1) }
42
- specify { expect(subject.query(term: {age: 10}).count).to eq(1) }
43
- specify { expect(subject.search_type(:count).count).to eq(0) }
44
- specify { expect(subject.search_type(:count).total).to eq(9) }
45
- end
46
-
47
- describe '#==' do
48
- let(:data) { 3.times.map { |i| {id: i.next.to_s, name: "Name#{i.next}", age: 10 * i.next}.stringify_keys! } }
49
- before { ProductsIndex::Product.import!(data.map { |h| double(h) }) }
50
-
51
- specify { expect(subject.query(match: 'hello')).to eq(subject.query(match: 'hello')) }
52
- specify { expect(subject.query(match: 'hello')).not_to eq(subject.query(match: 'world')) }
53
- specify { expect(subject.limit(10)).to eq(subject.limit(10)) }
54
- specify { expect(subject.limit(10)).not_to eq(subject.limit(11)) }
55
- specify { expect(subject.limit(2)).to eq(subject.limit(2).to_a) }
56
- end
57
16
 
58
- describe '#query_mode' do
59
- specify { expect(subject.query_mode(:should)).to be_a described_class }
60
- specify { expect(subject.query_mode(:should)).not_to eq(subject) }
61
- specify { expect(subject.query_mode(:should).criteria.options).to include(query_mode: :should) }
62
- specify { expect { subject.query_mode(:should) }.not_to change { subject.criteria.options } }
63
- end
64
-
65
- describe '#filter_mode' do
66
- specify { expect(subject.filter_mode(:or)).to be_a described_class }
67
- specify { expect(subject.filter_mode(:or)).not_to eq(subject) }
68
- specify { expect(subject.filter_mode(:or).criteria.options).to include(filter_mode: :or) }
69
- specify { expect { subject.filter_mode(:or) }.not_to change { subject.criteria.options } }
70
- end
17
+ subject { described_class.new(ProductsIndex) }
71
18
 
72
- describe '#post_filter_mode' do
73
- specify { expect(subject.post_filter_mode(:or)).to be_a described_class }
74
- specify { expect(subject.post_filter_mode(:or)).not_to eq(subject) }
75
- specify { expect(subject.post_filter_mode(:or).criteria.options).to include(post_filter_mode: :or) }
76
- specify { expect { subject.post_filter_mode(:or) }.not_to change { subject.criteria.options } }
77
- end
19
+ context 'unexistent index' do
20
+ specify { expect(subject.to_a).to eq([]) }
21
+ end
78
22
 
79
- describe '#boost_mode' do
80
- specify { expect(subject.boost_mode(:replace)).to be_a described_class }
81
- specify { expect(subject.boost_mode(:replace)).not_to eq(subject) }
82
- specify { expect(subject.boost_mode(:replace).criteria.options).to include(boost_mode: :replace) }
83
- specify { expect { subject.boost_mode(:replace) }.not_to change { subject.criteria.options } }
84
- end
23
+ context 'integration' do
24
+ let(:products) { Array.new(3) { |i| {id: i.next.to_s, name: "Name#{i.next}", age: 10 * i.next}.stringify_keys! } }
25
+ let(:cities) { Array.new(3) { |i| {id: i.next.to_s}.stringify_keys! } }
26
+ let(:countries) { Array.new(3) { |i| {id: i.next.to_s}.stringify_keys! } }
27
+ before do
28
+ ProductsIndex::Product.import!(products.map { |h| double(h) })
29
+ ProductsIndex::City.import!(cities.map { |h| double(h) })
30
+ ProductsIndex::Country.import!(countries.map { |h| double(h) })
31
+ end
85
32
 
86
- describe '#score_mode' do
87
- specify { expect(subject.score_mode(:first)).to be_a described_class }
88
- specify { expect(subject.score_mode(:first)).not_to eq(subject) }
89
- specify { expect(subject.score_mode(:first).criteria.options).to include(score_mode: :first) }
90
- specify { expect { subject.score_mode(:first) }.not_to change { subject.criteria.options } }
91
- end
33
+ specify { expect(subject.count).to eq(9) }
34
+ specify { expect(subject.first._data).to be_a Hash }
35
+ specify { expect(subject.limit(6).count).to eq(6) }
36
+ specify { expect(subject.offset(6).count).to eq(3) }
37
+ specify { expect(subject.query(match: {name: 'name3'}).highlight(fields: {name: {}}).first.name).to eq('Name3') }
38
+ specify { expect(subject.query(match: {name: 'name3'}).highlight(fields: {name: {}}).first.name_highlight).to eq('<em>Name3</em>') }
39
+ specify { expect(subject.query({}).highlight(fields: {name: {}}).first.name_highlight).to eq(nil) }
40
+ specify { expect(subject.query(match: {name: 'name3'}).highlight(fields: {name: {}}).first._data['_source']['name']).to eq('Name3') }
41
+ specify { expect(subject.types(:product).count).to eq(3) }
42
+ specify { expect(subject.types(:product, :country).count).to eq(6) }
43
+ specify { expect(subject.filter(term: {age: 10}).count).to eq(1) }
44
+ specify { expect(subject.query(term: {age: 10}).count).to eq(1) }
45
+ specify { expect(subject.order(nil).count).to eq(9) }
46
+ specify { expect(subject.search_type(:count).count).to eq(0) }
47
+ specify { expect(subject.search_type(:count).total).to eq(9) }
48
+ end
92
49
 
93
- describe '#limit' do
94
- specify { expect(subject.limit(10)).to be_a described_class }
95
- specify { expect(subject.limit(10)).not_to eq(subject) }
96
- specify { expect(subject.limit(10).criteria.request_options).to include(size: 10) }
97
- specify { expect { subject.limit(10) }.not_to change { subject.criteria.request_options } }
98
- end
50
+ describe '#==' do
51
+ let(:data) { Array.new(3) { |i| {id: i.next.to_s, name: "Name#{i.next}", age: 10 * i.next}.stringify_keys! } }
52
+ before { ProductsIndex::Product.import!(data.map { |h| double(h) }) }
99
53
 
100
- describe '#offset' do
101
- specify { expect(subject.offset(10)).to be_a described_class }
102
- specify { expect(subject.offset(10)).not_to eq(subject) }
103
- specify { expect(subject.offset(10).criteria.request_options).to include(from: 10) }
104
- specify { expect { subject.offset(10) }.not_to change { subject.criteria.request_options } }
105
- end
54
+ specify { expect(subject.query(match: 'hello')).to eq(subject.query(match: 'hello')) }
55
+ specify { expect(subject.query(match: 'hello')).not_to eq(subject.query(match: 'world')) }
56
+ specify { expect(subject.limit(10)).to eq(subject.limit(10)) }
57
+ specify { expect(subject.limit(10)).not_to eq(subject.limit(11)) }
58
+ specify { expect(subject.limit(2)).to eq(subject.limit(2).to_a) }
59
+ end
106
60
 
107
- describe '#script_fields' do
108
- specify { expect(subject.script_fields(distance: 'test()')).to be_a described_class }
109
- specify { expect(subject.script_fields(distance: 'test()')).not_to eq(subject) }
110
- specify { expect(subject.script_fields(distance: 'test()').criteria.script_fields).to include(distance: 'test()') }
111
- specify { expect { subject.script_fields(distance: 'test()') }.not_to change { subject.criteria.script_fields } }
112
- end
61
+ describe '#query_mode' do
62
+ specify { expect(subject.query_mode(:should)).to be_a described_class }
63
+ specify { expect(subject.query_mode(:should)).not_to eq(subject) }
64
+ specify { expect(subject.query_mode(:should).criteria.options).to include(query_mode: :should) }
65
+ specify { expect { subject.query_mode(:should) }.not_to change { subject.criteria.options } }
66
+ end
113
67
 
114
- describe '#script_score' do
115
- specify { expect(subject.script_score('23')).to be_a described_class }
116
- specify { expect(subject.script_score('23')).not_to eq(subject) }
117
- specify { expect(subject.script_score('23').criteria.scores).to eq([ { script_score: { script: '23' } } ]) }
118
- specify { expect { subject.script_score('23') }.not_to change { subject.criteria.scores } }
119
- specify { expect(subject.script_score('23 * factor', params: { factor: 0.5}).criteria.scores).to eq([{ script_score: { script: '23 * factor', params: { factor: 0.5} } }]) }
120
- end
68
+ describe '#filter_mode' do
69
+ specify { expect(subject.filter_mode(:or)).to be_a described_class }
70
+ specify { expect(subject.filter_mode(:or)).not_to eq(subject) }
71
+ specify { expect(subject.filter_mode(:or).criteria.options).to include(filter_mode: :or) }
72
+ specify { expect { subject.filter_mode(:or) }.not_to change { subject.criteria.options } }
73
+ end
121
74
 
122
- describe '#boost_factor' do
123
- specify { expect(subject.boost_factor('23')).to be_a described_class }
124
- specify { expect(subject.boost_factor('23')).not_to eq(subject) }
125
- specify { expect(subject.boost_factor('23').criteria.scores).to eq([ { boost_factor: 23 } ]) }
126
- specify { expect { subject.boost_factor('23') }.not_to change { subject.criteria.scores } }
127
- specify { expect(subject.boost_factor('23', filter: { foo: :bar}).criteria.scores).to eq([{ boost_factor: 23, filter: { foo: :bar } }]) }
128
- end
75
+ describe '#post_filter_mode' do
76
+ specify { expect(subject.post_filter_mode(:or)).to be_a described_class }
77
+ specify { expect(subject.post_filter_mode(:or)).not_to eq(subject) }
78
+ specify { expect(subject.post_filter_mode(:or).criteria.options).to include(post_filter_mode: :or) }
79
+ specify { expect { subject.post_filter_mode(:or) }.not_to change { subject.criteria.options } }
80
+ end
129
81
 
130
- describe '#random_score' do
131
- specify { expect(subject.random_score('23')).to be_a described_class }
132
- specify { expect(subject.random_score('23')).not_to eq(subject) }
133
- specify { expect(subject.random_score('23').criteria.scores).to eq([ { random_score: { seed: 23 } } ]) }
134
- specify { expect { subject.random_score('23') }.not_to change { subject.criteria.scores } }
135
- specify { expect(subject.random_score('23', filter: { foo: :bar}).criteria.scores).to eq([{ random_score: { seed: 23 }, filter: { foo: :bar } }]) }
136
- end
82
+ describe '#boost_mode' do
83
+ specify { expect(subject.boost_mode(:replace)).to be_a described_class }
84
+ specify { expect(subject.boost_mode(:replace)).not_to eq(subject) }
85
+ specify { expect(subject.boost_mode(:replace).criteria.options).to include(boost_mode: :replace) }
86
+ specify { expect { subject.boost_mode(:replace) }.not_to change { subject.criteria.options } }
87
+ end
137
88
 
138
- describe '#field_value_score' do
139
- specify { expect(subject.field_value_factor(field: :boost)).to be_a described_class }
140
- specify { expect(subject.field_value_factor(field: :boost)).not_to eq(subject) }
141
- specify { expect(subject.field_value_factor(field: :boost).criteria.scores).to eq([ { field_value_factor: { field: :boost } } ]) }
142
- specify { expect { subject.field_value_factor(field: :boost) }.not_to change { subject.criteria.scores } }
143
- specify { expect(subject.field_value_factor({ field: :boost }, filter: { foo: :bar}).criteria.scores).to eq([{ field_value_factor: { field: :boost }, filter: { foo: :bar } }]) }
144
- end
89
+ describe '#score_mode' do
90
+ specify { expect(subject.score_mode(:first)).to be_a described_class }
91
+ specify { expect(subject.score_mode(:first)).not_to eq(subject) }
92
+ specify { expect(subject.score_mode(:first).criteria.options).to include(score_mode: :first) }
93
+ specify { expect { subject.score_mode(:first) }.not_to change { subject.criteria.options } }
94
+ end
145
95
 
146
- describe '#decay' do
147
- specify { expect(subject.decay(:gauss, :field)).to be_a described_class }
148
- specify { expect(subject.decay(:gauss, :field)).not_to eq(subject) }
149
- specify { expect(subject.decay(:gauss, :field).criteria.scores).to eq([ {
150
- gauss: {
151
- field: {}
152
- }
153
- }]) }
154
- specify { expect { subject.decay(:gauss, :field) }.not_to change { subject.criteria.scores } }
155
- specify {
156
- expect(subject.decay(:gauss, :field,
157
- origin: '11, 12',
158
- scale: '2km',
159
- offset: '5km',
160
- decay: 0.4,
161
- filter: { foo: :bar }).criteria.scores).to eq([
162
- {
163
- gauss: {
164
- field: {
165
- origin: '11, 12',
166
- scale: '2km',
167
- offset: '5km',
168
- decay: 0.4
169
- }
170
- },
171
- filter: { foo: :bar }
172
- }
173
- ])
174
- }
175
- end
96
+ describe '#limit' do
97
+ specify { expect(subject.limit(10)).to be_a described_class }
98
+ specify { expect(subject.limit(10)).not_to eq(subject) }
99
+ specify { expect(subject.limit(10).criteria.request_options).to include(size: 10) }
100
+ specify { expect { subject.limit(10) }.not_to change { subject.criteria.request_options } }
101
+ specify { expect(subject.limit { 20 / 2 }.criteria.request_body[:body]).to include(size: 10) }
102
+ end
176
103
 
177
- describe '#facets' do
178
- specify do
179
- skip_on_version_lt('2.0')
180
- expect { subject.facets }.to raise_error(Chewy::RemovedFeature).with_message('removed in elasticsearch 2.0')
104
+ describe '#offset' do
105
+ specify { expect(subject.offset(10)).to be_a described_class }
106
+ specify { expect(subject.offset(10)).not_to eq(subject) }
107
+ specify { expect(subject.offset(10).criteria.request_options).to include(from: 10) }
108
+ specify { expect { subject.offset(10) }.not_to change { subject.criteria.request_options } }
109
+ specify { expect(subject.offset { 20 / 2 }.criteria.request_body[:body]).to include(from: 10) }
181
110
  end
182
111
 
183
- specify do
184
- skip_on_version_gte('2.0')
185
- expect(subject.facets(term: {field: 'hello'})).to be_a described_class
112
+ describe '#track_scores' do
113
+ specify { expect(subject.track_scores(true)).to be_a described_class }
114
+ specify { expect(subject.track_scores(true)).not_to eq(subject) }
115
+ specify { expect(subject.track_scores(true).criteria.request_options).to include(track_scores: true) }
116
+ specify { expect { subject.track_scores(true) }.not_to change { subject.criteria.request_options } }
117
+ specify { expect(subject.track_scores(false).criteria.request_body[:body]).to include(track_scores: false) }
186
118
  end
187
- specify do
188
- skip_on_version_gte('2.0')
189
- expect(subject.facets(term: {field: 'hello'})).not_to eq(subject)
119
+
120
+ describe '#script_fields' do
121
+ specify { expect(subject.script_fields(distance: 'test()')).to be_a described_class }
122
+ specify { expect(subject.script_fields(distance: 'test()')).not_to eq(subject) }
123
+ specify { expect(subject.script_fields(distance: 'test()').criteria.script_fields).to include(distance: 'test()') }
124
+ specify { expect { subject.script_fields(distance: 'test()') }.not_to change { subject.criteria.script_fields } }
190
125
  end
191
- specify do
192
- skip_on_version_gte('2.0')
193
- expect(subject.facets(term: {field: 'hello'}).criteria.facets).to include(term: {field: 'hello'})
126
+
127
+ describe '#script_score' do
128
+ specify { expect(subject.script_score('23')).to be_a described_class }
129
+ specify { expect(subject.script_score('23')).not_to eq(subject) }
130
+ specify { expect(subject.script_score('23').criteria.scores).to eq([{script_score: {script: '23'}}]) }
131
+ specify { expect { subject.script_score('23') }.not_to change { subject.criteria.scores } }
132
+ specify { expect(subject.script_score('23 * factor', params: {factor: 0.5}).criteria.scores).to eq([{script_score: {script: '23 * factor', params: {factor: 0.5}}}]) }
194
133
  end
195
- specify do
196
- skip_on_version_gte('2.0')
197
- expect { subject.facets(term: {field: 'hello'}) }.not_to change { subject.criteria.facets }
134
+
135
+ describe '#boost_factor' do
136
+ specify { expect(subject.boost_factor('23')).to be_a described_class }
137
+ specify { expect(subject.boost_factor('23')).not_to eq(subject) }
138
+ specify { expect(subject.boost_factor('23').criteria.scores).to eq([{boost_factor: 23}]) }
139
+ specify { expect { subject.boost_factor('23') }.not_to change { subject.criteria.scores } }
140
+ specify { expect(subject.boost_factor('23', filter: {foo: :bar}).criteria.scores).to eq([{boost_factor: 23, filter: {foo: :bar}}]) }
198
141
  end
199
142
 
200
- context 'results', :orm do
201
- before { stub_model(:city) }
202
- let(:cities) { 10.times.map { |i| City.create! id: i + 1, name: "name#{i}", rating: i % 3 } }
143
+ describe '#weight' do
144
+ specify { expect(subject.weight('23')).to be_a described_class }
145
+ specify { expect(subject.weight('23')).not_to eq(subject) }
146
+ specify { expect(subject.weight('23').criteria.scores).to eq([{weight: 23}]) }
147
+ specify { expect { subject.weight('23') }.not_to change { subject.criteria.scores } }
148
+ specify { expect(subject.weight('23', filter: {foo: :bar}).criteria.scores).to eq([{weight: 23, filter: {foo: :bar}}]) }
149
+ end
203
150
 
204
- before do
205
- stub_index(:cities) do
206
- define_type :city do
207
- field :rating, type: 'integer'
208
- end
209
- end
210
- end
151
+ describe '#random_score' do
152
+ specify { expect(subject.random_score('23')).to be_a described_class }
153
+ specify { expect(subject.random_score('23')).not_to eq(subject) }
154
+ specify { expect(subject.random_score('23').criteria.scores).to eq([{random_score: {seed: 23}}]) }
155
+ specify { expect { subject.random_score('23') }.not_to change { subject.criteria.scores } }
156
+ specify { expect(subject.random_score('23', filter: {foo: :bar}).criteria.scores).to eq([{random_score: {seed: 23}, filter: {foo: :bar}}]) }
157
+ end
211
158
 
212
- before { CitiesIndex::City.import! cities }
159
+ describe '#field_value_score' do
160
+ specify { expect(subject.field_value_factor(field: :boost)).to be_a described_class }
161
+ specify { expect(subject.field_value_factor(field: :boost)).not_to eq(subject) }
162
+ specify { expect(subject.field_value_factor(field: :boost).criteria.scores).to eq([{field_value_factor: {field: :boost}}]) }
163
+ specify { expect { subject.field_value_factor(field: :boost) }.not_to change { subject.criteria.scores } }
164
+ specify { expect(subject.field_value_factor({field: :boost}, filter: {foo: :bar}).criteria.scores).to eq([{field_value_factor: {field: :boost}, filter: {foo: :bar}}]) }
165
+ end
213
166
 
167
+ describe '#decay' do
168
+ specify { expect(subject.decay(:gauss, :field)).to be_a described_class }
169
+ specify { expect(subject.decay(:gauss, :field)).not_to eq(subject) }
214
170
  specify do
215
- skip_on_version_gte('2.0')
216
- expect(CitiesIndex.facets).to eq({})
171
+ expect(subject.decay(:gauss, :field).criteria.scores).to eq([{
172
+ gauss: {
173
+ field: {}
174
+ }
175
+ }])
217
176
  end
177
+ specify { expect { subject.decay(:gauss, :field) }.not_to change { subject.criteria.scores } }
218
178
  specify do
219
- skip_on_version_gte('2.0')
220
- expect(CitiesIndex.facets(ratings: {terms: {field: 'rating'}}).facets).to eq({
221
- 'ratings' => {
222
- '_type' => 'terms', 'missing' => 0, 'total' => 10, 'other' => 0,
223
- 'terms' => [
224
- {'term' => 0, 'count' => 4},
225
- {'term' => 2, 'count' => 3},
226
- {'term' => 1, 'count' => 3}
227
- ]
228
- }
229
- })
179
+ expect(subject.decay(:gauss, :field,
180
+ origin: '11, 12',
181
+ scale: '2km',
182
+ offset: '5km',
183
+ decay: 0.4,
184
+ filter: {foo: :bar}).criteria.scores).to eq([
185
+ {
186
+ gauss: {
187
+ field: {
188
+ origin: '11, 12',
189
+ scale: '2km',
190
+ offset: '5km',
191
+ decay: 0.4
192
+ }
193
+ },
194
+ filter: {foo: :bar}
195
+ }
196
+ ])
230
197
  end
231
198
  end
232
- end
233
-
234
- describe '#aggregations' do
235
- specify { expect(subject.aggregations(aggregation1: {field: 'hello'})).to be_a described_class }
236
- specify { expect(subject.aggregations(aggregation1: {field: 'hello'})).not_to eq(subject) }
237
- specify { expect(subject.aggregations(aggregation1: {field: 'hello'}).criteria.aggregations).to include(aggregation1: {field: 'hello'}) }
238
- specify { expect { subject.aggregations(aggregation1: {field: 'hello'}) }.not_to change { subject.criteria.aggregations } }
239
199
 
240
- context 'when requesting a named aggregation' do
200
+ describe '#preference' do
201
+ specify { expect(subject.preference(:_primary)).to be_a described_class }
202
+ specify { expect(subject.preference(:_primary)).not_to eq(subject) }
203
+ specify { expect(subject.preference(:_primary).criteria.search_options).to include(preference: :_primary) }
204
+ specify { expect { subject.preference(:_primary) }.not_to change { subject.criteria.search_options } }
205
+ specify { expect(subject.preference(:_primary).criteria.request_body).to include(preference: :_primary) }
206
+ end
241
207
 
242
- before do
243
- stub_index(:products) do
244
- define_type :product do
245
- root do
246
- field :name, 'surname'
247
- field :title, type: 'string' do
248
- field :subfield1
249
- end
250
- field 'price', type: 'float' do
251
- field :subfield2
252
- end
253
- agg :uniquely_named_agg do
254
- { min: { field: 'title.subfield1' } }
255
- end
256
- end
257
- end
258
- end
259
- end
260
- specify { expect(subject.aggregations(:uniquely_named_agg).criteria.aggregations).to include(uniquely_named_agg: { min: { field: 'title.subfield1' } }) }
208
+ describe '#aggregations' do
209
+ specify { expect(subject.aggregations(aggregation1: {field: 'hello'})).to be_a described_class }
210
+ specify { expect(subject.aggregations(aggregation1: {field: 'hello'})).not_to eq(subject) }
211
+ specify { expect(subject.aggregations(aggregation1: {field: 'hello'}).criteria.aggregations).to include(aggregation1: {field: 'hello'}) }
212
+ specify { expect { subject.aggregations(aggregation1: {field: 'hello'}) }.not_to change { subject.criteria.aggregations } }
261
213
 
262
- context 'when more than one aggregation of the same name exists' do
214
+ context 'when requesting a named aggregation' do
263
215
  before do
264
216
  stub_index(:products) do
265
217
  define_type :product do
266
218
  root do
267
219
  field :name, 'surname'
268
- field :title, type: 'string' do
220
+ field :title do
269
221
  field :subfield1
270
222
  end
271
223
  field 'price', type: 'float' do
272
224
  field :subfield2
273
225
  end
274
226
  agg :uniquely_named_agg do
275
- { min: { field: 'title.subfield1' } }
276
- end
277
- agg :named_agg do
278
- { avg: { field: 'title.subfield1' } }
227
+ {min: {field: 'title.subfield1'}}
279
228
  end
280
229
  end
281
230
  end
282
- define_type :review do
283
- field :title, :body
284
- field :comments do
285
- field :message
286
- field :rating, type: 'long'
231
+ end
232
+ end
233
+ specify { expect(subject.aggregations(:uniquely_named_agg).criteria.aggregations).to include(uniquely_named_agg: {min: {field: 'title.subfield1'}}) }
234
+
235
+ context 'when more than one aggregation of the same name exists' do
236
+ before do
237
+ stub_index(:products) do
238
+ define_type :product do
239
+ root do
240
+ field :name, 'surname'
241
+ field :title do
242
+ field :subfield1
243
+ end
244
+ field 'price', type: 'float' do
245
+ field :subfield2
246
+ end
247
+ agg :uniquely_named_agg do
248
+ {min: {field: 'title.subfield1'}}
249
+ end
250
+ agg :named_agg do
251
+ {avg: {field: 'title.subfield1'}}
252
+ end
253
+ end
287
254
  end
288
- agg :named_agg do
289
- { avg: { field: 'comments.rating' } }
255
+ define_type :review do
256
+ field :title, :body
257
+ field :comments do
258
+ field :message
259
+ field :rating, type: 'long'
260
+ end
261
+ agg :named_agg do
262
+ {avg: {field: 'comments.rating'}}
263
+ end
290
264
  end
291
265
  end
292
266
  end
293
- end
294
267
 
295
- it "is the aggregation definition that was last defined" do
296
- expect(subject.aggregations(:named_agg).criteria.aggregations).to include(named_agg: { avg: { field: 'comments.rating' } })
297
- end
268
+ it 'is the aggregation definition that was last defined' do
269
+ expect(subject.aggregations(:named_agg).criteria.aggregations).to include(named_agg: {avg: {field: 'comments.rating'}})
270
+ end
298
271
 
299
- context "when the fully qualified aggregation name is provided" do
300
- specify { expect(subject
301
- .aggregations("products#product.named_agg")
302
- .criteria
303
- .aggregations)
304
- .to include({ "products#product.named_agg" => { avg: { field: 'title.subfield1' } } }) }
272
+ context 'when the fully qualified aggregation name is provided' do
273
+ specify do
274
+ expect(subject
275
+ .aggregations('products#product.named_agg')
276
+ .criteria
277
+ .aggregations)
278
+ .to include('products#product.named_agg' => {avg: {field: 'title.subfield1'}})
279
+ end
280
+ end
305
281
  end
306
282
  end
307
- end
308
283
 
309
- context 'results', :orm do
310
- before { stub_model(:city) }
311
- let(:cities) { 10.times.map { |i| City.create! id: i + 1, name: "name#{i}", rating: i % 3 } }
284
+ context 'results', :orm do
285
+ before { stub_model(:city) }
286
+ let(:cities) { Array.new(10) { |i| City.create! id: i + 1, name: "name#{i}", rating: i % 3 } }
312
287
 
313
- context do
314
- before do
315
- stub_index(:cities) do
316
- define_type :city do
317
- field :rating, type: 'integer'
288
+ context do
289
+ before do
290
+ stub_index(:cities) do
291
+ define_type :city do
292
+ field :rating, type: 'integer'
293
+ end
318
294
  end
319
295
  end
320
- end
321
296
 
322
- before { CitiesIndex::City.import! cities }
323
-
324
- specify { expect(CitiesIndex.aggregations).to eq({}) }
325
- specify { expect(CitiesIndex.aggregations(ratings: {terms: {field: 'rating'}})
326
- .aggregations['ratings']['buckets'].map { |h| h.slice('key', 'doc_count') }).to eq([
327
- { 'key' => 0, 'doc_count' => 4 },
328
- { 'key' => 1, 'doc_count' => 3 },
329
- { 'key' => 2, 'doc_count' => 3 }
330
- ]) }
297
+ before { CitiesIndex::City.import! cities }
298
+ subject { described_class.new(CitiesIndex) }
299
+
300
+ specify { expect(subject.aggregations).to eq({}) }
301
+ specify do
302
+ expect(subject.aggregations(ratings: {terms: {field: 'rating'}})
303
+ .aggregations['ratings']['buckets'].map { |h| h.slice('key', 'doc_count') }).to eq([
304
+ {'key' => 0, 'doc_count' => 4},
305
+ {'key' => 1, 'doc_count' => 3},
306
+ {'key' => 2, 'doc_count' => 3}
307
+ ])
308
+ end
309
+ end
331
310
  end
332
311
  end
333
- end
334
-
335
- describe '#suggest' do
336
- specify { subject.suggest(name1: {text: 'hello', term: {field: 'name'}}) }
337
- specify { expect(subject.suggest(name1: {text: 'hello'})).not_to eq(subject) }
338
- specify { expect(subject.suggest(name1: {text: 'hello'}).criteria.suggest).to include(name1: {text: 'hello'}) }
339
- specify { expect { subject.suggest(name1: {text: 'hello'}) }.not_to change { subject.criteria.suggest } }
340
312
 
341
- context 'results', :orm do
342
- before { stub_model(:city) }
343
- let(:cities) { 10.times.map { |i| City.create! id: i + 1, name: "name#{i}" } }
344
-
345
- context do
346
- before do
347
- stub_index(:cities) do
348
- define_type :city do
349
- field :name
313
+ describe '#suggest' do
314
+ specify { subject.suggest(name1: {text: 'hello', term: {field: 'name'}}) }
315
+ specify { expect(subject.suggest(name1: {text: 'hello'})).not_to eq(subject) }
316
+ specify { expect(subject.suggest(name1: {text: 'hello'}).criteria.suggest).to include(name1: {text: 'hello'}) }
317
+ specify { expect { subject.suggest(name1: {text: 'hello'}) }.not_to change { subject.criteria.suggest } }
318
+
319
+ context 'results', :orm do
320
+ before { stub_model(:city) }
321
+ let(:cities) { Array.new(10) { |i| City.create! id: i + 1, name: "name#{i}" } }
322
+
323
+ context do
324
+ before do
325
+ stub_index(:cities) do
326
+ define_type :city do
327
+ field :name
328
+ end
350
329
  end
351
330
  end
352
- end
353
331
 
354
- before { CitiesIndex::City.import! cities }
332
+ before { CitiesIndex::City.import! cities }
333
+ subject { described_class.new(CitiesIndex) }
355
334
 
356
- specify { expect(CitiesIndex.suggest).to eq({}) }
357
- specify { expect(CitiesIndex.suggest(name: {text: 'name', term: {field: 'name'}}).suggest).to eq({
358
- 'name' => [
359
- {'text' => 'name', 'offset' => 0, 'length' => 4, 'options' => [
335
+ specify { expect(subject.suggest).to eq({}) }
336
+ specify do
337
+ expect(subject.suggest(name: {text: 'name', term: {field: 'name'}}).suggest).to eq('name' => [
338
+ {'text' => 'name', 'offset' => 0, 'length' => 4, 'options' => [
360
339
  {'text' => 'name0', 'score' => 0.75, 'freq' => 1},
361
340
  {'text' => 'name1', 'score' => 0.75, 'freq' => 1},
362
341
  {'text' => 'name2', 'score' => 0.75, 'freq' => 1},
363
342
  {'text' => 'name3', 'score' => 0.75, 'freq' => 1},
364
343
  {'text' => 'name4', 'score' => 0.75, 'freq' => 1}
365
- ]
366
- }
367
- ] })
368
- }
344
+ ]}
345
+ ])
346
+ end
347
+ end
369
348
  end
370
349
  end
371
- end
372
350
 
373
- describe '#delete_all' do
374
- let(:products) { 3.times.map { |i| {id: i.next.to_s, name: "Name#{i.next}", age: 10 * i.next}.stringify_keys! } }
375
- let(:cities) { 3.times.map { |i| {id: i.next.to_s}.stringify_keys! } }
376
- let(:countries) { 3.times.map { |i| {id: i.next.to_s}.stringify_keys! } }
351
+ describe '#delete_all' do
352
+ let(:products) { Array.new(3) { |i| {id: i.next.to_s, name: "Name#{i.next}", age: 10 * i.next}.stringify_keys! } }
353
+ let(:cities) { Array.new(3) { |i| {id: i.next.to_s}.stringify_keys! } }
354
+ let(:countries) { Array.new(3) { |i| {id: i.next.to_s}.stringify_keys! } }
377
355
 
378
- before do
379
- ProductsIndex::Product.import!(products.map { |h| double(h) })
380
- ProductsIndex::City.import!(cities.map { |h| double(h) })
381
- ProductsIndex::Country.import!(countries.map { |h| double(h) })
382
- end
383
-
384
- specify do
385
- skip_on_plugin_missing_from_version('delete-by-query', '2.0')
386
- expect {
387
- subject.query(match: {name: 'name3'}).delete_all
388
- Chewy.client.indices.refresh(index: 'products') }.to change { ProductsIndex.total }.from(9).to(8)
389
- end
390
- specify do
391
- skip_on_plugin_missing_from_version('delete-by-query', '2.0')
392
- expect {
393
- subject.filter { age == [10, 20] }.delete_all
394
- Chewy.client.indices.refresh(index: 'products') }.to change { ProductsIndex.total_count }.from(9).to(7)
395
- end
396
- specify do
397
- skip_on_plugin_missing_from_version('delete-by-query', '2.0')
398
- expect {
399
- subject.types(:product).delete_all
400
- Chewy.client.indices.refresh(index: 'products') }.to change { ProductsIndex::Product.total_entries }.from(3).to(0)
401
- end
402
- specify do
403
- skip_on_plugin_missing_from_version('delete-by-query', '2.0')
404
- expect {
405
- ProductsIndex.delete_all
406
- Chewy.client.indices.refresh(index: 'products') }.to change { ProductsIndex.total }.from(9).to(0)
407
- end
408
- specify do
409
- skip_on_plugin_missing_from_version('delete-by-query', '2.0')
410
- expect {
411
- ProductsIndex::City.delete_all
412
- Chewy.client.indices.refresh(index: 'products') }.to change { ProductsIndex.total }.from(9).to(6)
413
- end
414
-
415
- specify do
416
- skip_on_version_lt('2.0')
417
- expect(Chewy.client.nodes).to receive(:info).and_return({"nodes" => {"a" => {"plugins" => {"name" => "hello"}}}})
418
- expect { ProductsIndex.delete_all }.to raise_error(Chewy::PluginMissing).with_message("install delete-by-query plugin")
356
+ before do
357
+ ProductsIndex::Product.import!(products.map { |h| double(h) })
358
+ ProductsIndex::City.import!(cities.map { |h| double(h) })
359
+ ProductsIndex::Country.import!(countries.map { |h| double(h) })
360
+ end
361
+
362
+ specify do
363
+ expect do
364
+ subject.query(match: {name: 'name3'}).delete_all
365
+ Chewy.client.indices.refresh(index: 'products')
366
+ end.to change { described_class.new(ProductsIndex).total }.from(9).to(8)
367
+ end
368
+ specify do
369
+ expect do
370
+ subject.filter { age == [10, 20] }.delete_all
371
+ Chewy.client.indices.refresh(index: 'products')
372
+ end.to change { described_class.new(ProductsIndex).total_count }.from(9).to(7)
373
+ end
374
+ specify do
375
+ expect do
376
+ subject.types(:product).delete_all
377
+ Chewy.client.indices.refresh(index: 'products')
378
+ end.to change { described_class.new(ProductsIndex::Product).total_entries }.from(3).to(0)
379
+ end
380
+ specify do
381
+ expect do
382
+ subject.delete_all
383
+ Chewy.client.indices.refresh(index: 'products')
384
+ end.to change { described_class.new(ProductsIndex).total }.from(9).to(0)
385
+ end
386
+ specify do
387
+ expect do
388
+ described_class.new(ProductsIndex::City).delete_all
389
+ Chewy.client.indices.refresh(index: 'products')
390
+ end.to change { described_class.new(ProductsIndex).total }.from(9).to(6)
391
+ end
419
392
  end
420
- end
421
393
 
422
- describe '#find' do
423
- let(:products) { 3.times.map { |i| {id: i.next.to_s, name: "Name#{i.next}", age: 10 * i.next}.stringify_keys! } }
424
- let(:cities) { 1.times.map { |i| {id: '4'}.stringify_keys! } }
425
- let(:countries) { 1.times.map { |i| {id: '4'}.stringify_keys! } }
394
+ describe '#find' do
395
+ let(:products) { Array.new(3) { |i| {id: i.next.to_s, name: "Name#{i.next}", age: 10 * i.next}.stringify_keys! } }
396
+ let(:cities) { Array.new(1) { {id: '4'}.stringify_keys! } }
397
+ let(:countries) { Array.new(1) { {id: '4'}.stringify_keys! } }
426
398
 
427
- before do
428
- ProductsIndex::Product.import!(products.map { |h| double(h) })
429
- ProductsIndex::City.import!(cities.map { |h| double(h) })
430
- ProductsIndex::Country.import!(countries.map { |h| double(h) })
431
- end
432
-
433
- specify { expect(subject.find(1)).to be_a(ProductsIndex::Product) }
434
- specify { expect(subject.find(1).id).to eq('1') }
435
- specify { expect(subject.find(4).id).to eq('4') }
436
- specify { expect(subject.find([1]).map(&:id)).to match_array(%w(1)) }
437
- specify { expect(subject.find([4]).map(&:id)).to match_array(%w(4 4)) }
438
- specify { expect(subject.find([1, 3]).map(&:id)).to match_array(%w(1 3)) }
439
- specify { expect(subject.find(1, 3).map(&:id)).to match_array(%w(1 3)) }
440
- specify { expect(subject.find(1, 10).map(&:id)).to match_array(%w(1)) }
441
-
442
- specify { expect { subject.find(10) }.to raise_error Chewy::DocumentNotFound }
443
- specify { expect { subject.find([10]) }.to raise_error Chewy::DocumentNotFound }
444
- specify { expect { subject.find([10, 20]) }.to raise_error Chewy::DocumentNotFound }
445
- end
399
+ before do
400
+ ProductsIndex::Product.import!(products.map { |h| double(h) })
401
+ ProductsIndex::City.import!(cities.map { |h| double(h) })
402
+ ProductsIndex::Country.import!(countries.map { |h| double(h) })
403
+ end
446
404
 
447
- describe '#none' do
448
- specify { expect(subject.none).to be_a described_class }
449
- specify { expect(subject.none).not_to eq(subject) }
450
- specify { expect(subject.none.criteria).to be_none }
405
+ specify { expect(subject.find(1)).to be_a(ProductsIndex::Product) }
406
+ specify { expect(subject.find(1).id).to eq('1') }
407
+ specify { expect(subject.find(4).id).to eq('4') }
408
+ specify { expect(subject.find([1]).map(&:id)).to match_array(%w[1]) }
409
+ specify { expect(subject.find([4]).map(&:id)).to match_array(%w[4 4]) }
410
+ specify { expect(subject.find([1, 3]).map(&:id)).to match_array(%w[1 3]) }
411
+ specify { expect(subject.find(1, 3).map(&:id)).to match_array(%w[1 3]) }
412
+ specify { expect(subject.find(1, 10).map(&:id)).to match_array(%w[1]) }
413
+
414
+ specify { expect { subject.find(10) }.to raise_error Chewy::DocumentNotFound }
415
+ specify { expect { subject.find([10]) }.to raise_error Chewy::DocumentNotFound }
416
+ specify { expect { subject.find([10, 20]) }.to raise_error Chewy::DocumentNotFound }
417
+ end
451
418
 
452
- context do
453
- before { expect_any_instance_of(described_class).not_to receive(:_response) }
419
+ describe '#exists?' do
420
+ let(:data) { Array.new(10) { |i| {id: i.next.to_s, name: "Name#{i.next}", age: 10 * i.next} } }
454
421
 
455
- specify { expect(subject.none.to_a).to eq([]) }
456
- specify { expect(subject.query(match: 'hello').none.to_a).to eq([]) }
457
- specify { expect(subject.none.query(match: 'hello').to_a).to eq([]) }
458
- end
459
- end
422
+ before { ProductsIndex::Product.import!(data.map { |h| double(h) }) }
460
423
 
461
- describe '#strategy' do
462
- specify { expect(subject.strategy('query_first')).to be_a described_class }
463
- specify { expect(subject.strategy('query_first')).not_to eq(subject) }
464
- specify { expect(subject.strategy('query_first').criteria.options).to include(strategy: 'query_first') }
465
- specify { expect { subject.strategy('query_first') }.not_to change { subject.criteria.options } }
466
- end
424
+ specify { expect(subject.exists?).to eq true }
425
+ specify { expect(subject.limit(5).exists?).to eq true }
426
+ specify { expect(subject.filter(range: {age: {gt: 20}}).limit(3).exists?).to eq true }
427
+ specify { expect(subject.filter(range: {age: {lt: 0}}).exists?).to eq false }
428
+ end
467
429
 
468
- describe '#query' do
469
- specify { expect(subject.query(match: 'hello')).to be_a described_class }
470
- specify { expect(subject.query(match: 'hello')).not_to eq(subject) }
471
- specify { expect(subject.query(match: 'hello').criteria.queries).to include(match: 'hello') }
472
- specify { expect { subject.query(match: 'hello') }.not_to change { subject.criteria.queries } }
473
- end
430
+ describe '#unlimited' do
431
+ let(:data_length) { 10 }
432
+ let(:data) { Array.new(data_length) { |i| {id: i.next.to_s, name: "Name#{i.next}", age: 10 * i.next} } }
474
433
 
475
- describe '#filter' do
476
- specify { expect(subject.filter(term: {field: 'hello'})).to be_a described_class }
477
- specify { expect(subject.filter(term: {field: 'hello'})).not_to eq(subject) }
478
- specify { expect { subject.filter(term: {field: 'hello'}) }.not_to change { subject.criteria.filters } }
479
- specify { expect(subject.filter([{term: {field: 'hello'}}, {term: {field: 'world'}}]).criteria.filters)
480
- .to eq([{term: {field: 'hello'}}, {term: {field: 'world'}}]) }
434
+ before { ProductsIndex::Product.import!(data.map { |h| double(h) }) }
481
435
 
482
- specify { expect { subject.filter{ name == 'John' } }.not_to change { subject.criteria.filters } }
483
- specify { expect(subject.filter{ name == 'John' }.criteria.filters).to eq([{term: {'name' => 'John'}}]) }
484
- end
436
+ specify { expect(subject.unlimited.count).to eq data_length }
437
+ specify { expect(subject.offset(5).unlimited.count).to eq data_length }
438
+ specify { expect(subject.limit(1).unlimited.count).to eq data_length }
439
+ specify { expect(subject.unlimited.limit(1).count).to eq 1 }
440
+ end
485
441
 
486
- describe '#post_filter' do
487
- specify { expect(subject.post_filter(term: {field: 'hello'})).to be_a described_class }
488
- specify { expect(subject.post_filter(term: {field: 'hello'})).not_to eq(subject) }
489
- specify { expect { subject.post_filter(term: {field: 'hello'}) }.not_to change { subject.criteria.post_filters } }
490
- specify { expect(subject.post_filter([{term: {field: 'hello'}}, {term: {field: 'world'}}]).criteria.post_filters)
491
- .to eq([{term: {field: 'hello'}}, {term: {field: 'world'}}]) }
442
+ describe '#none' do
443
+ specify { expect(subject.none).to be_a described_class }
444
+ specify { expect(subject.none).not_to eq(subject) }
445
+ specify { expect(subject.none.criteria).to be_none }
492
446
 
493
- specify { expect { subject.post_filter{ name == 'John' } }.not_to change { subject.criteria.post_filters } }
494
- specify { expect(subject.post_filter{ name == 'John' }.criteria.post_filters).to eq([{term: {'name' => 'John'}}]) }
495
- end
447
+ context do
448
+ before { expect_any_instance_of(described_class).not_to receive(:_response) }
496
449
 
497
- describe '#order' do
498
- specify { expect(subject.order(field: 'hello')).to be_a described_class }
499
- specify { expect(subject.order(field: 'hello')).not_to eq(subject) }
500
- specify { expect { subject.order(field: 'hello') }.not_to change { subject.criteria.sort } }
450
+ specify { expect(subject.none.to_a).to eq([]) }
451
+ specify { expect(subject.query(match: 'hello').none.to_a).to eq([]) }
452
+ specify { expect(subject.none.query(match: 'hello').to_a).to eq([]) }
453
+ end
454
+ end
501
455
 
502
- specify { expect(subject.order(:field).criteria.sort).to eq([:field]) }
503
- specify { expect(subject.order([:field1, :field2]).criteria.sort).to eq([:field1, :field2]) }
504
- specify { expect(subject.order(field: :asc).criteria.sort).to eq([{field: :asc}]) }
505
- specify { expect(subject.order({field1: :asc, field2: :desc}).criteria.sort).to eq([{field1: :asc}, {field2: :desc}]) }
506
- specify { expect(subject.order({field1: {order: :asc}, field2: :desc}).order([:field3], :field4).criteria.sort).to eq([{field1: {order: :asc}}, {field2: :desc}, :field3, :field4]) }
507
- end
456
+ describe '#strategy' do
457
+ specify { expect(subject.strategy('query_first')).to be_a described_class }
458
+ specify { expect(subject.strategy('query_first')).not_to eq(subject) }
459
+ specify { expect(subject.strategy('query_first').criteria.options).to include(strategy: 'query_first') }
460
+ specify { expect { subject.strategy('query_first') }.not_to change { subject.criteria.options } }
461
+ end
508
462
 
509
- describe '#reorder' do
510
- specify { expect(subject.reorder(field: 'hello')).to be_a described_class }
511
- specify { expect(subject.reorder(field: 'hello')).not_to eq(subject) }
512
- specify { expect { subject.reorder(field: 'hello') }.not_to change { subject.criteria.sort } }
463
+ describe '#query' do
464
+ specify { expect(subject.query(match: 'hello')).to be_a described_class }
465
+ specify { expect(subject.query(match: 'hello')).not_to eq(subject) }
466
+ specify { expect(subject.query(match: 'hello').criteria.queries).to include(match: 'hello') }
467
+ specify { expect { subject.query(match: 'hello') }.not_to change { subject.criteria.queries } }
468
+ end
513
469
 
514
- specify { expect(subject.order(:field1).reorder(:field2).criteria.sort).to eq([:field2]) }
515
- specify { expect(subject.order(:field1).reorder(:field2).order(:field3).criteria.sort).to eq([:field2, :field3]) }
516
- specify { expect(subject.order(:field1).reorder(:field2).reorder(:field3).criteria.sort).to eq([:field3]) }
517
- end
470
+ describe '#filter' do
471
+ specify { expect(subject.filter(term: {field: 'hello'})).to be_a described_class }
472
+ specify { expect(subject.filter(term: {field: 'hello'})).not_to eq(subject) }
473
+ specify { expect { subject.filter(term: {field: 'hello'}) }.not_to change { subject.criteria.filters } }
474
+ specify do
475
+ expect(subject.filter([{term: {field: 'hello'}}, {term: {field: 'world'}}]).criteria.filters)
476
+ .to eq([{term: {field: 'hello'}}, {term: {field: 'world'}}])
477
+ end
518
478
 
519
- describe '#only' do
520
- specify { expect(subject.only(:field)).to be_a described_class }
521
- specify { expect(subject.only(:field)).not_to eq(subject) }
522
- specify { expect { subject.only(:field) }.not_to change { subject.criteria.fields } }
479
+ specify { expect { subject.filter { name == 'John' } }.not_to change { subject.criteria.filters } }
480
+ specify { expect(subject.filter { name == 'John' }.criteria.filters).to eq([{term: {'name' => 'John'}}]) }
481
+ end
523
482
 
524
- specify { expect(subject.only(:field1, :field2).criteria.fields).to match_array(['field1', 'field2']) }
525
- specify { expect(subject.only([:field1, :field2]).only(:field3).criteria.fields).to match_array(['field1', 'field2', 'field3']) }
526
- end
483
+ describe '#post_filter' do
484
+ specify { expect(subject.post_filter(term: {field: 'hello'})).to be_a described_class }
485
+ specify { expect(subject.post_filter(term: {field: 'hello'})).not_to eq(subject) }
486
+ specify { expect { subject.post_filter(term: {field: 'hello'}) }.not_to change { subject.criteria.post_filters } }
487
+ specify do
488
+ expect(subject.post_filter([{term: {field: 'hello'}}, {term: {field: 'world'}}]).criteria.post_filters)
489
+ .to eq([{term: {field: 'hello'}}, {term: {field: 'world'}}])
490
+ end
527
491
 
528
- describe '#only!' do
529
- specify { expect(subject.only!(:field)).to be_a described_class }
530
- specify { expect(subject.only!(:field)).not_to eq(subject) }
531
- specify { expect { subject.only!(:field) }.not_to change { subject.criteria.fields } }
492
+ specify { expect { subject.post_filter { name == 'John' } }.not_to change { subject.criteria.post_filters } }
493
+ specify { expect(subject.post_filter { name == 'John' }.criteria.post_filters).to eq([{term: {'name' => 'John'}}]) }
494
+ end
532
495
 
533
- specify { expect(subject.only!(:field1, :field2).criteria.fields).to match_array(['field1', 'field2']) }
534
- specify { expect(subject.only!([:field1, :field2]).only!(:field3).criteria.fields).to match_array(['field3']) }
535
- specify { expect(subject.only([:field1, :field2]).only!(:field3).criteria.fields).to match_array(['field3']) }
536
- end
496
+ describe '#order' do
497
+ specify { expect(subject.order(field: 'hello')).to be_a described_class }
498
+ specify { expect(subject.order(field: 'hello')).not_to eq(subject) }
499
+ specify { expect { subject.order(field: 'hello') }.not_to change { subject.criteria.sort } }
537
500
 
538
- describe '#types' do
539
- specify { expect(subject.types(:product)).to be_a described_class }
540
- specify { expect(subject.types(:product)).not_to eq(subject) }
541
- specify { expect { subject.types(:product) }.not_to change { subject.criteria.types } }
501
+ specify { expect(subject.order(:field).criteria.sort).to eq([:field]) }
502
+ specify { expect(subject.order(%i[field1 field2]).criteria.sort).to eq(%i[field1 field2]) }
503
+ specify { expect(subject.order(field: :asc).criteria.sort).to eq([{field: :asc}]) }
504
+ specify { expect(subject.order(field1: :asc, field2: :desc).criteria.sort).to eq([{field1: :asc}, {field2: :desc}]) }
505
+ specify { expect(subject.order(field1: {order: :asc}, field2: :desc).order([:field3], :field4).criteria.sort).to eq([{field1: {order: :asc}}, {field2: :desc}, :field3, :field4]) }
506
+ end
542
507
 
543
- specify { expect(subject.types(:user).criteria.types).to eq(['user']) }
544
- specify { expect(subject.types(:product, :city).criteria.types).to match_array(['product', 'city']) }
545
- specify { expect(subject.types([:product, :city]).types(:country).criteria.types).to match_array(['product', 'city', 'country']) }
546
- end
508
+ describe '#reorder' do
509
+ specify { expect(subject.reorder(field: 'hello')).to be_a described_class }
510
+ specify { expect(subject.reorder(field: 'hello')).not_to eq(subject) }
511
+ specify { expect { subject.reorder(field: 'hello') }.not_to change { subject.criteria.sort } }
547
512
 
548
- describe '#types!' do
549
- specify { expect(subject.types!(:product)).to be_a described_class }
550
- specify { expect(subject.types!(:product)).not_to eq(subject) }
551
- specify { expect { subject.types!(:product) }.not_to change { subject.criteria.types } }
513
+ specify { expect(subject.order(:field1).reorder(:field2).criteria.sort).to eq([:field2]) }
514
+ specify { expect(subject.order(:field1).reorder(:field2).order(:field3).criteria.sort).to eq(%i[field2 field3]) }
515
+ specify { expect(subject.order(:field1).reorder(:field2).reorder(:field3).criteria.sort).to eq([:field3]) }
516
+ end
552
517
 
553
- specify { expect(subject.types!(:user).criteria.types).to eq(['user']) }
554
- specify { expect(subject.types!(:product, :city).criteria.types).to match_array(['product', 'city']) }
555
- specify { expect(subject.types!([:product, :city]).types!(:country).criteria.types).to match_array(['country']) }
556
- specify { expect(subject.types([:product, :city]).types!(:country).criteria.types).to match_array(['country']) }
557
- end
518
+ describe '#only' do
519
+ specify { expect(subject.only(:field)).to be_a described_class }
520
+ specify { expect(subject.only(:field)).not_to eq(subject) }
521
+ specify { expect { subject.only(:field) }.not_to change { subject.criteria.fields } }
558
522
 
559
- describe '#search_type' do
560
- specify { expect(subject.search_type(:count).options).to include(search_type: :count) }
561
- end
523
+ specify { expect(subject.only(:field1, :field2).criteria.fields).to match_array(%w[field1 field2]) }
524
+ specify { expect(subject.only(%i[field1 field2]).only(:field3).criteria.fields).to match_array(%w[field1 field2 field3]) }
525
+ end
562
526
 
563
- describe '#aggregations' do
564
- specify { expect(subject.aggregations(attribute: {terms: {field: 'attribute'}})).to be_a described_class }
565
- specify { expect(subject.aggregations(attribute: {terms: {field: 'attribute'}})).not_to eq(subject) }
566
- specify { expect(subject.aggregations(attribute: {terms: {field: 'attribute'}}).criteria.request_body[:body]).to include(aggregations: {attribute: {terms: {field: 'attribute'}}}) }
567
- end
527
+ describe '#only!' do
528
+ specify { expect(subject.only!(:field)).to be_a described_class }
529
+ specify { expect(subject.only!(:field)).not_to eq(subject) }
530
+ specify { expect { subject.only!(:field) }.not_to change { subject.criteria.fields } }
568
531
 
569
- describe '#merge' do
570
- let(:query) { described_class.new(ProductsIndex) }
532
+ specify { expect(subject.only!(:field1, :field2).criteria.fields).to match_array(%w[field1 field2]) }
533
+ specify { expect(subject.only!(%i[field1 field2]).only!(:field3).criteria.fields).to match_array(['field3']) }
534
+ specify { expect(subject.only(%i[field1 field2]).only!(:field3).criteria.fields).to match_array(['field3']) }
535
+ end
571
536
 
572
- specify { expect(subject.filter { name == 'name' }.merge(query.filter { age == 42 }).criteria.filters)
573
- .to eq([{term: {'name' => 'name'}}, {term: {'age' => 42}}]) }
574
- end
537
+ describe '#types' do
538
+ specify { expect(subject.types(:product)).to be_a described_class }
539
+ specify { expect(subject.types(:product)).not_to eq(subject) }
540
+ specify { expect { subject.types(:product) }.not_to change { subject.criteria.types } }
575
541
 
576
- describe '#to_a', :orm do
577
- before { stub_model(:city) }
578
- let(:cities) { 3.times.map { |i| City.create! id: i + 1, name: "name#{i}", rating: i } }
542
+ specify { expect(subject.types(:user).criteria.types).to eq(['user']) }
543
+ specify { expect(subject.types(:product, :city).criteria.types).to match_array(%w[product city]) }
544
+ specify { expect(subject.types(%i[product city]).types(:country).criteria.types).to match_array(%w[product city country]) }
545
+ end
579
546
 
580
- context do
581
- before do
582
- stub_index(:cities) do
583
- define_type :city do
584
- field :name
585
- field :rating, type: 'integer'
586
- field :nested, type: 'object', value: ->{ {name: name} }
587
- end
588
- end
589
- end
547
+ describe '#types!' do
548
+ specify { expect(subject.types!(:product)).to be_a described_class }
549
+ specify { expect(subject.types!(:product)).not_to eq(subject) }
550
+ specify { expect { subject.types!(:product) }.not_to change { subject.criteria.types } }
590
551
 
591
- before { CitiesIndex::City.import! cities }
552
+ specify { expect(subject.types!(:user).criteria.types).to eq(['user']) }
553
+ specify { expect(subject.types!(:product, :city).criteria.types).to match_array(%w[product city]) }
554
+ specify { expect(subject.types!(%i[product city]).types!(:country).criteria.types).to match_array(['country']) }
555
+ specify { expect(subject.types(%i[product city]).types!(:country).criteria.types).to match_array(['country']) }
556
+ end
592
557
 
593
- specify { expect(CitiesIndex.order(:rating).first).to be_a CitiesIndex::City }
594
- specify { expect(CitiesIndex.order(:rating).first.name).to eq('name0') }
595
- specify { expect(CitiesIndex.order(:rating).first.rating).to eq(0) }
596
- specify { expect(CitiesIndex.order(:rating).first.nested).to eq({'name' => 'name0'}) }
597
- specify { expect(CitiesIndex.order(:rating).first.id).to eq(cities.first.id.to_s) }
558
+ describe '#search_type' do
559
+ specify { expect(subject.search_type(:count).criteria.search_options).to include(search_type: :count) }
560
+ end
598
561
 
599
- specify { expect(CitiesIndex.order(:rating).only(:name).first.name).to eq('name0') }
600
- specify { expect(CitiesIndex.order(:rating).only(:name).first.rating).to be_nil }
601
- specify { expect(CitiesIndex.order(:rating).only(:nested).first.nested).to eq({'name' => 'name0'}) }
562
+ describe '#aggregations' do
563
+ specify { expect(subject.aggregations(attribute: {terms: {field: 'attribute'}})).to be_a described_class }
564
+ specify { expect(subject.aggregations(attribute: {terms: {field: 'attribute'}})).not_to eq(subject) }
565
+ specify { expect(subject.aggregations(attribute: {terms: {field: 'attribute'}}).criteria.request_body[:body]).to include(aggregations: {attribute: {terms: {field: 'attribute'}}}) }
566
+ end
602
567
 
603
- specify { expect(CitiesIndex.order(:rating).first._score).to be_nil }
604
- specify { expect(CitiesIndex.all.first._score).to be > 0 }
605
- specify { expect(CitiesIndex.query(match: {name: 'name0'}).first._score).to be > 0 }
606
- specify { expect(CitiesIndex.query(match: {name: 'name0'}).took).to be >= 0 }
568
+ describe '#merge' do
569
+ let(:query) { described_class.new(ProductsIndex) }
607
570
 
608
- specify { expect(CitiesIndex.order(:rating).first._explanation).to be_nil }
609
- specify { expect(CitiesIndex.order(:rating).explain.first._explanation).to be_present }
571
+ specify do
572
+ expect(subject.filter { name == 'name' }.merge(query.filter { age == 42 }).criteria.filters)
573
+ .to eq([{term: {'name' => 'name'}}, {term: {'age' => 42}}])
574
+ end
610
575
  end
611
576
 
612
- context 'sourceless' do
613
- before do
614
- stub_index(:cities) do
615
- define_type :city do
616
- root _source: {enabled: false} do
577
+ describe '#to_a', :orm do
578
+ before { stub_model(:city) }
579
+ let(:cities) { Array.new(3) { |i| City.create! id: i + 1, name: "name#{i}", rating: i } }
580
+ subject { described_class.new(CitiesIndex) }
581
+
582
+ context do
583
+ before do
584
+ stub_index(:cities) do
585
+ define_type :city do
617
586
  field :name
618
587
  field :rating, type: 'integer'
619
- field :nested, type: 'object', value: ->{ {name: name} }
588
+ field :nested, type: 'object', value: -> { {name: name} }
620
589
  end
621
590
  end
622
591
  end
592
+
593
+ before { CitiesIndex::City.import! cities }
594
+
595
+ specify { expect(subject.order(:rating).first).to be_a CitiesIndex::City }
596
+ specify { expect(subject.order(:rating).first.name).to eq('name0') }
597
+ specify { expect(subject.order(:rating).first.rating).to eq(0) }
598
+ specify { expect(subject.order(:rating).first.nested).to eq('name' => 'name0') }
599
+ specify { expect(subject.order(:rating).first.id).to eq(cities.first.id.to_s) }
600
+
601
+ specify { expect(subject.order(:rating).only(:name).first.name).to eq('name0') }
602
+ specify { expect(subject.order(:rating).only(:name).first.rating).to be_nil }
603
+ specify { expect(subject.order(:rating).only(:nested).first.nested).to eq('name' => 'name0') }
604
+
605
+ specify { expect(subject.order(:rating).first._score).to be_nil }
606
+ specify { expect(subject.first._score).to be > 0 }
607
+ specify { expect(subject.query(match: {name: 'name0'}).first._score).to be > 0 }
608
+ specify { expect(subject.query(match: {name: 'name0'}).took).to be >= 0 }
609
+
610
+ specify { expect(subject.order(:rating).first._explanation).to be_nil }
611
+ specify { expect(subject.order(:rating).explain.first._explanation).to be_present }
623
612
  end
624
- before { CitiesIndex::City.import! cities }
625
613
 
626
- specify { expect(CitiesIndex.order(:rating).first).to be_a CitiesIndex::City }
627
- specify { expect(CitiesIndex.order(:rating).first.name).to be_nil }
628
- specify { expect(CitiesIndex.order(:rating).first.rating).to be_nil }
629
- specify { expect(CitiesIndex.order(:rating).first.nested).to be_nil }
614
+ context 'sourceless' do
615
+ before do
616
+ stub_index(:cities) do
617
+ define_type :city do
618
+ root _source: {enabled: false} do
619
+ field :name
620
+ field :rating, type: 'integer'
621
+ field :nested, type: 'object', value: -> { {name: name} }
622
+ end
623
+ end
624
+ end
625
+ end
626
+ before { CitiesIndex::City.import! cities }
627
+
628
+ specify { expect(subject.order(:rating).first).to be_a CitiesIndex::City }
629
+ specify { expect(subject.order(:rating).first.name).to be_nil }
630
+ specify { expect(subject.order(:rating).first.rating).to be_nil }
631
+ specify { expect(subject.order(:rating).first.nested).to be_nil }
632
+ end
630
633
  end
634
+ else
635
+ xspecify 'Skip Chewy::Query specs for 5.0'
631
636
  end
632
637
  end