chewy 0.9.0 → 0.10.0

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