chewy 0.9.0 → 5.2.0

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