chewy 0.8.4 → 5.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (303) hide show
  1. checksums.yaml +5 -5
  2. data/.gitignore +1 -0
  3. data/.rubocop.yml +56 -0
  4. data/.rubocop_todo.yml +44 -0
  5. data/.travis.yml +36 -67
  6. data/.yardopts +5 -0
  7. data/Appraisals +63 -58
  8. data/CHANGELOG.md +168 -11
  9. data/Gemfile +16 -9
  10. data/Guardfile +5 -5
  11. data/LEGACY_DSL.md +497 -0
  12. data/README.md +403 -470
  13. data/Rakefile +11 -1
  14. data/chewy.gemspec +12 -15
  15. data/gemfiles/rails.4.0.activerecord.gemfile +9 -9
  16. data/gemfiles/rails.4.1.activerecord.gemfile +9 -9
  17. data/gemfiles/rails.4.2.activerecord.gemfile +8 -8
  18. data/gemfiles/rails.4.2.mongoid.5.2.gemfile +16 -0
  19. data/gemfiles/rails.5.0.activerecord.gemfile +16 -0
  20. data/gemfiles/rails.5.0.mongoid.6.1.gemfile +16 -0
  21. data/gemfiles/rails.5.1.activerecord.gemfile +16 -0
  22. data/gemfiles/rails.5.1.mongoid.6.3.gemfile +16 -0
  23. data/gemfiles/rails.5.2.activerecord.gemfile +16 -0
  24. data/gemfiles/sequel.4.45.gemfile +11 -0
  25. data/lib/chewy/backports/deep_dup.rb +1 -1
  26. data/lib/chewy/backports/duplicable.rb +1 -0
  27. data/lib/chewy/config.rb +53 -21
  28. data/lib/chewy/errors.rb +6 -6
  29. data/lib/chewy/fields/base.rb +59 -29
  30. data/lib/chewy/fields/root.rb +49 -14
  31. data/lib/chewy/index/actions.rb +95 -36
  32. data/lib/chewy/index/aliases.rb +2 -1
  33. data/lib/chewy/index/settings.rb +10 -5
  34. data/lib/chewy/index/specification.rb +60 -0
  35. data/lib/chewy/index.rb +239 -138
  36. data/lib/chewy/journal.rb +55 -0
  37. data/lib/chewy/log_subscriber.rb +8 -8
  38. data/lib/chewy/minitest/helpers.rb +77 -0
  39. data/lib/chewy/minitest/search_index_receiver.rb +80 -0
  40. data/lib/chewy/minitest.rb +1 -0
  41. data/lib/chewy/query/compose.rb +18 -19
  42. data/lib/chewy/query/criteria.rb +34 -24
  43. data/lib/chewy/query/filters.rb +28 -11
  44. data/lib/chewy/query/loading.rb +3 -4
  45. data/lib/chewy/query/nodes/and.rb +1 -1
  46. data/lib/chewy/query/nodes/base.rb +1 -1
  47. data/lib/chewy/query/nodes/bool.rb +6 -4
  48. data/lib/chewy/query/nodes/equal.rb +4 -4
  49. data/lib/chewy/query/nodes/exists.rb +1 -1
  50. data/lib/chewy/query/nodes/expr.rb +2 -2
  51. data/lib/chewy/query/nodes/field.rb +35 -31
  52. data/lib/chewy/query/nodes/has_child.rb +1 -0
  53. data/lib/chewy/query/nodes/has_parent.rb +1 -0
  54. data/lib/chewy/query/nodes/has_relation.rb +10 -12
  55. data/lib/chewy/query/nodes/missing.rb +1 -1
  56. data/lib/chewy/query/nodes/not.rb +1 -1
  57. data/lib/chewy/query/nodes/or.rb +1 -1
  58. data/lib/chewy/query/nodes/prefix.rb +3 -2
  59. data/lib/chewy/query/nodes/query.rb +1 -1
  60. data/lib/chewy/query/nodes/range.rb +9 -9
  61. data/lib/chewy/query/nodes/raw.rb +1 -1
  62. data/lib/chewy/query/nodes/regexp.rb +13 -9
  63. data/lib/chewy/query/nodes/script.rb +4 -4
  64. data/lib/chewy/query/pagination.rb +10 -1
  65. data/lib/chewy/query.rb +286 -170
  66. data/lib/chewy/railtie.rb +7 -6
  67. data/lib/chewy/rake_helper.rb +275 -37
  68. data/lib/chewy/repository.rb +2 -2
  69. data/lib/chewy/rspec/update_index.rb +70 -65
  70. data/lib/chewy/rspec.rb +1 -1
  71. data/lib/chewy/runtime/version.rb +4 -4
  72. data/lib/chewy/search/loader.rb +83 -0
  73. data/lib/chewy/{query → search}/pagination/kaminari.rb +13 -5
  74. data/lib/chewy/search/pagination/will_paginate.rb +43 -0
  75. data/lib/chewy/search/parameters/aggs.rb +16 -0
  76. data/lib/chewy/search/parameters/allow_partial_search_results.rb +27 -0
  77. data/lib/chewy/search/parameters/concerns/bool_storage.rb +24 -0
  78. data/lib/chewy/search/parameters/concerns/hash_storage.rb +23 -0
  79. data/lib/chewy/search/parameters/concerns/integer_storage.rb +14 -0
  80. data/lib/chewy/search/parameters/concerns/query_storage.rb +238 -0
  81. data/lib/chewy/search/parameters/concerns/string_array_storage.rb +23 -0
  82. data/lib/chewy/search/parameters/concerns/string_storage.rb +14 -0
  83. data/lib/chewy/search/parameters/docvalue_fields.rb +12 -0
  84. data/lib/chewy/search/parameters/explain.rb +16 -0
  85. data/lib/chewy/search/parameters/filter.rb +47 -0
  86. data/lib/chewy/search/parameters/highlight.rb +16 -0
  87. data/lib/chewy/search/parameters/indices.rb +123 -0
  88. data/lib/chewy/search/parameters/indices_boost.rb +52 -0
  89. data/lib/chewy/search/parameters/limit.rb +17 -0
  90. data/lib/chewy/search/parameters/load.rb +32 -0
  91. data/lib/chewy/search/parameters/min_score.rb +16 -0
  92. data/lib/chewy/search/parameters/none.rb +27 -0
  93. data/lib/chewy/search/parameters/offset.rb +17 -0
  94. data/lib/chewy/search/parameters/order.rb +64 -0
  95. data/lib/chewy/search/parameters/post_filter.rb +19 -0
  96. data/lib/chewy/search/parameters/preference.rb +16 -0
  97. data/lib/chewy/search/parameters/profile.rb +16 -0
  98. data/lib/chewy/search/parameters/query.rb +19 -0
  99. data/lib/chewy/search/parameters/request_cache.rb +27 -0
  100. data/lib/chewy/search/parameters/rescore.rb +29 -0
  101. data/lib/chewy/search/parameters/script_fields.rb +16 -0
  102. data/lib/chewy/search/parameters/search_after.rb +20 -0
  103. data/lib/chewy/search/parameters/search_type.rb +16 -0
  104. data/lib/chewy/search/parameters/source.rb +73 -0
  105. data/lib/chewy/search/parameters/storage.rb +95 -0
  106. data/lib/chewy/search/parameters/stored_fields.rb +63 -0
  107. data/lib/chewy/search/parameters/suggest.rb +16 -0
  108. data/lib/chewy/search/parameters/terminate_after.rb +16 -0
  109. data/lib/chewy/search/parameters/timeout.rb +16 -0
  110. data/lib/chewy/search/parameters/track_scores.rb +16 -0
  111. data/lib/chewy/search/parameters/types.rb +20 -0
  112. data/lib/chewy/search/parameters/version.rb +16 -0
  113. data/lib/chewy/search/parameters.rb +167 -0
  114. data/lib/chewy/search/query_proxy.rb +257 -0
  115. data/lib/chewy/search/request.rb +1045 -0
  116. data/lib/chewy/search/response.rb +119 -0
  117. data/lib/chewy/search/scoping.rb +50 -0
  118. data/lib/chewy/search/scrolling.rb +134 -0
  119. data/lib/chewy/search.rb +81 -26
  120. data/lib/chewy/stash.rb +79 -0
  121. data/lib/chewy/strategy/active_job.rb +1 -0
  122. data/lib/chewy/strategy/atomic.rb +2 -4
  123. data/lib/chewy/strategy/base.rb +4 -4
  124. data/lib/chewy/strategy/bypass.rb +1 -2
  125. data/lib/chewy/strategy/resque.rb +1 -0
  126. data/lib/chewy/strategy/shoryuken.rb +40 -0
  127. data/lib/chewy/strategy/sidekiq.rb +13 -1
  128. data/lib/chewy/strategy/urgent.rb +1 -1
  129. data/lib/chewy/strategy.rb +19 -10
  130. data/lib/chewy/type/actions.rb +26 -2
  131. data/lib/chewy/type/adapter/active_record.rb +50 -24
  132. data/lib/chewy/type/adapter/base.rb +29 -9
  133. data/lib/chewy/type/adapter/mongoid.rb +19 -10
  134. data/lib/chewy/type/adapter/object.rb +195 -31
  135. data/lib/chewy/type/adapter/orm.rb +69 -33
  136. data/lib/chewy/type/adapter/sequel.rb +37 -19
  137. data/lib/chewy/type/crutch.rb +5 -4
  138. data/lib/chewy/type/import/bulk_builder.rb +122 -0
  139. data/lib/chewy/type/import/bulk_request.rb +78 -0
  140. data/lib/chewy/type/import/journal_builder.rb +45 -0
  141. data/lib/chewy/type/import/routine.rb +138 -0
  142. data/lib/chewy/type/import.rb +150 -176
  143. data/lib/chewy/type/mapping.rb +58 -42
  144. data/lib/chewy/type/observe.rb +21 -15
  145. data/lib/chewy/type/syncer.rb +222 -0
  146. data/lib/chewy/type/witchcraft.rb +89 -34
  147. data/lib/chewy/type/wrapper.rb +48 -16
  148. data/lib/chewy/type.rb +77 -49
  149. data/lib/chewy/version.rb +1 -1
  150. data/lib/chewy.rb +95 -52
  151. data/lib/generators/chewy/install_generator.rb +3 -3
  152. data/lib/sequel/plugins/chewy_observe.rb +4 -19
  153. data/lib/tasks/chewy.rake +91 -28
  154. data/spec/chewy/config_spec.rb +130 -12
  155. data/spec/chewy/fields/base_spec.rb +194 -172
  156. data/spec/chewy/fields/root_spec.rb +123 -17
  157. data/spec/chewy/fields/time_fields_spec.rb +10 -9
  158. data/spec/chewy/index/actions_spec.rb +228 -43
  159. data/spec/chewy/index/aliases_spec.rb +2 -2
  160. data/spec/chewy/index/settings_spec.rb +100 -49
  161. data/spec/chewy/index/specification_spec.rb +169 -0
  162. data/spec/chewy/index_spec.rb +159 -63
  163. data/spec/chewy/journal_spec.rb +268 -0
  164. data/spec/chewy/minitest/helpers_spec.rb +90 -0
  165. data/spec/chewy/minitest/search_index_receiver_spec.rb +120 -0
  166. data/spec/chewy/query/criteria_spec.rb +503 -236
  167. data/spec/chewy/query/filters_spec.rb +96 -68
  168. data/spec/chewy/query/loading_spec.rb +80 -42
  169. data/spec/chewy/query/nodes/and_spec.rb +3 -7
  170. data/spec/chewy/query/nodes/bool_spec.rb +5 -13
  171. data/spec/chewy/query/nodes/equal_spec.rb +20 -20
  172. data/spec/chewy/query/nodes/exists_spec.rb +7 -7
  173. data/spec/chewy/query/nodes/has_child_spec.rb +42 -23
  174. data/spec/chewy/query/nodes/has_parent_spec.rb +42 -23
  175. data/spec/chewy/query/nodes/match_all_spec.rb +2 -2
  176. data/spec/chewy/query/nodes/missing_spec.rb +6 -5
  177. data/spec/chewy/query/nodes/not_spec.rb +5 -7
  178. data/spec/chewy/query/nodes/or_spec.rb +3 -7
  179. data/spec/chewy/query/nodes/prefix_spec.rb +6 -6
  180. data/spec/chewy/query/nodes/query_spec.rb +3 -3
  181. data/spec/chewy/query/nodes/range_spec.rb +19 -19
  182. data/spec/chewy/query/nodes/raw_spec.rb +2 -2
  183. data/spec/chewy/query/nodes/regexp_spec.rb +31 -19
  184. data/spec/chewy/query/nodes/script_spec.rb +5 -5
  185. data/spec/chewy/query/pagination/kaminari_spec.rb +3 -55
  186. data/spec/chewy/query/pagination/will_paginate_spec.rb +5 -0
  187. data/spec/chewy/query/pagination_spec.rb +25 -22
  188. data/spec/chewy/query_spec.rb +510 -505
  189. data/spec/chewy/rake_helper_spec.rb +381 -0
  190. data/spec/chewy/repository_spec.rb +8 -8
  191. data/spec/chewy/rspec/update_index_spec.rb +215 -113
  192. data/spec/chewy/runtime_spec.rb +2 -2
  193. data/spec/chewy/search/loader_spec.rb +117 -0
  194. data/spec/chewy/search/pagination/kaminari_examples.rb +71 -0
  195. data/spec/chewy/search/pagination/kaminari_spec.rb +21 -0
  196. data/spec/chewy/search/pagination/will_paginate_examples.rb +63 -0
  197. data/spec/chewy/search/pagination/will_paginate_spec.rb +23 -0
  198. data/spec/chewy/search/parameters/aggs_spec.rb +5 -0
  199. data/spec/chewy/search/parameters/bool_storage_examples.rb +53 -0
  200. data/spec/chewy/search/parameters/docvalue_fields_spec.rb +5 -0
  201. data/spec/chewy/search/parameters/explain_spec.rb +5 -0
  202. data/spec/chewy/search/parameters/filter_spec.rb +5 -0
  203. data/spec/chewy/search/parameters/hash_storage_examples.rb +59 -0
  204. data/spec/chewy/search/parameters/highlight_spec.rb +5 -0
  205. data/spec/chewy/search/parameters/indices_spec.rb +191 -0
  206. data/spec/chewy/search/parameters/integer_storage_examples.rb +32 -0
  207. data/spec/chewy/search/parameters/limit_spec.rb +5 -0
  208. data/spec/chewy/search/parameters/load_spec.rb +60 -0
  209. data/spec/chewy/search/parameters/min_score_spec.rb +32 -0
  210. data/spec/chewy/search/parameters/none_spec.rb +5 -0
  211. data/spec/chewy/search/parameters/offset_spec.rb +5 -0
  212. data/spec/chewy/search/parameters/order_spec.rb +65 -0
  213. data/spec/chewy/search/parameters/post_filter_spec.rb +5 -0
  214. data/spec/chewy/search/parameters/preference_spec.rb +5 -0
  215. data/spec/chewy/search/parameters/profile_spec.rb +5 -0
  216. data/spec/chewy/search/parameters/query_spec.rb +5 -0
  217. data/spec/chewy/search/parameters/query_storage_examples.rb +388 -0
  218. data/spec/chewy/search/parameters/request_cache_spec.rb +67 -0
  219. data/spec/chewy/search/parameters/rescore_spec.rb +62 -0
  220. data/spec/chewy/search/parameters/script_fields_spec.rb +5 -0
  221. data/spec/chewy/search/parameters/search_after_spec.rb +32 -0
  222. data/spec/chewy/search/parameters/search_type_spec.rb +5 -0
  223. data/spec/chewy/search/parameters/source_spec.rb +156 -0
  224. data/spec/chewy/search/parameters/storage_spec.rb +60 -0
  225. data/spec/chewy/search/parameters/stored_fields_spec.rb +126 -0
  226. data/spec/chewy/search/parameters/string_array_storage_examples.rb +63 -0
  227. data/spec/chewy/search/parameters/string_storage_examples.rb +32 -0
  228. data/spec/chewy/search/parameters/suggest_spec.rb +5 -0
  229. data/spec/chewy/search/parameters/terminate_after_spec.rb +5 -0
  230. data/spec/chewy/search/parameters/timeout_spec.rb +5 -0
  231. data/spec/chewy/search/parameters/track_scores_spec.rb +5 -0
  232. data/spec/chewy/search/parameters/types_spec.rb +5 -0
  233. data/spec/chewy/search/parameters/version_spec.rb +5 -0
  234. data/spec/chewy/search/parameters_spec.rb +145 -0
  235. data/spec/chewy/search/query_proxy_spec.rb +68 -0
  236. data/spec/chewy/search/request_spec.rb +685 -0
  237. data/spec/chewy/search/response_spec.rb +192 -0
  238. data/spec/chewy/search/scrolling_spec.rb +169 -0
  239. data/spec/chewy/search_spec.rb +37 -20
  240. data/spec/chewy/stash_spec.rb +95 -0
  241. data/spec/chewy/strategy/active_job_spec.rb +8 -2
  242. data/spec/chewy/strategy/atomic_spec.rb +4 -1
  243. data/spec/chewy/strategy/resque_spec.rb +8 -2
  244. data/spec/chewy/strategy/shoryuken_spec.rb +66 -0
  245. data/spec/chewy/strategy/sidekiq_spec.rb +10 -2
  246. data/spec/chewy/strategy_spec.rb +6 -6
  247. data/spec/chewy/type/actions_spec.rb +29 -10
  248. data/spec/chewy/type/adapter/active_record_spec.rb +357 -139
  249. data/spec/chewy/type/adapter/mongoid_spec.rb +220 -101
  250. data/spec/chewy/type/adapter/object_spec.rb +129 -40
  251. data/spec/chewy/type/adapter/sequel_spec.rb +304 -152
  252. data/spec/chewy/type/import/bulk_builder_spec.rb +279 -0
  253. data/spec/chewy/type/import/bulk_request_spec.rb +102 -0
  254. data/spec/chewy/type/import/journal_builder_spec.rb +95 -0
  255. data/spec/chewy/type/import/routine_spec.rb +110 -0
  256. data/spec/chewy/type/import_spec.rb +360 -244
  257. data/spec/chewy/type/mapping_spec.rb +96 -29
  258. data/spec/chewy/type/observe_spec.rb +25 -15
  259. data/spec/chewy/type/syncer_spec.rb +123 -0
  260. data/spec/chewy/type/witchcraft_spec.rb +122 -44
  261. data/spec/chewy/type/wrapper_spec.rb +63 -23
  262. data/spec/chewy/type_spec.rb +32 -10
  263. data/spec/chewy_spec.rb +82 -12
  264. data/spec/spec_helper.rb +16 -2
  265. data/spec/support/active_record.rb +6 -2
  266. data/spec/support/class_helpers.rb +4 -19
  267. data/spec/support/mongoid.rb +17 -5
  268. data/spec/support/sequel.rb +6 -1
  269. metadata +250 -57
  270. data/gemfiles/rails.3.2.activerecord.gemfile +0 -15
  271. data/gemfiles/rails.3.2.activerecord.kaminari.gemfile +0 -14
  272. data/gemfiles/rails.3.2.activerecord.will_paginate.gemfile +0 -14
  273. data/gemfiles/rails.4.0.activerecord.kaminari.gemfile +0 -14
  274. data/gemfiles/rails.4.0.activerecord.will_paginate.gemfile +0 -14
  275. data/gemfiles/rails.4.0.mongoid.4.0.0.gemfile +0 -15
  276. data/gemfiles/rails.4.0.mongoid.4.0.0.kaminari.gemfile +0 -14
  277. data/gemfiles/rails.4.0.mongoid.4.0.0.will_paginate.gemfile +0 -14
  278. data/gemfiles/rails.4.0.mongoid.5.1.0.gemfile +0 -15
  279. data/gemfiles/rails.4.0.mongoid.5.1.0.kaminari.gemfile +0 -14
  280. data/gemfiles/rails.4.0.mongoid.5.1.0.will_paginate.gemfile +0 -14
  281. data/gemfiles/rails.4.1.activerecord.kaminari.gemfile +0 -14
  282. data/gemfiles/rails.4.1.activerecord.will_paginate.gemfile +0 -14
  283. data/gemfiles/rails.4.1.mongoid.4.0.0.gemfile +0 -15
  284. data/gemfiles/rails.4.1.mongoid.4.0.0.kaminari.gemfile +0 -14
  285. data/gemfiles/rails.4.1.mongoid.4.0.0.will_paginate.gemfile +0 -14
  286. data/gemfiles/rails.4.1.mongoid.5.1.0.gemfile +0 -15
  287. data/gemfiles/rails.4.1.mongoid.5.1.0.kaminari.gemfile +0 -14
  288. data/gemfiles/rails.4.1.mongoid.5.1.0.will_paginate.gemfile +0 -14
  289. data/gemfiles/rails.4.2.activerecord.kaminari.gemfile +0 -15
  290. data/gemfiles/rails.4.2.activerecord.will_paginate.gemfile +0 -15
  291. data/gemfiles/rails.4.2.mongoid.4.0.0.gemfile +0 -15
  292. data/gemfiles/rails.4.2.mongoid.4.0.0.kaminari.gemfile +0 -14
  293. data/gemfiles/rails.4.2.mongoid.4.0.0.will_paginate.gemfile +0 -14
  294. data/gemfiles/rails.4.2.mongoid.5.1.0.gemfile +0 -15
  295. data/gemfiles/rails.4.2.mongoid.5.1.0.kaminari.gemfile +0 -14
  296. data/gemfiles/rails.4.2.mongoid.5.1.0.will_paginate.gemfile +0 -14
  297. data/gemfiles/rails.5.0.0.beta3.activerecord.gemfile +0 -16
  298. data/gemfiles/rails.5.0.0.beta3.activerecord.kaminari.gemfile +0 -16
  299. data/gemfiles/rails.5.0.0.beta3.activerecord.will_paginate.gemfile +0 -15
  300. data/gemfiles/sequel.4.31.gemfile +0 -13
  301. data/lib/chewy/query/pagination/will_paginate.rb +0 -27
  302. data/lib/chewy/query/scoping.rb +0 -20
  303. data/spec/chewy/query/pagination/will_paginage_spec.rb +0 -60
@@ -1,7 +1,12 @@
1
1
  require 'spec_helper'
2
2
 
3
3
  describe Chewy::Type::Adapter::ActiveRecord, :active_record do
4
- before { stub_model(:city) }
4
+ before do
5
+ stub_model(:city)
6
+ stub_model(:country)
7
+ City.belongs_to :country
8
+ Country.has_many :cities
9
+ end
5
10
 
6
11
  describe '#name' do
7
12
  specify { expect(described_class.new(City).name).to eq('City') }
@@ -41,7 +46,7 @@ describe Chewy::Type::Adapter::ActiveRecord, :active_record do
41
46
  subject { described_class.new(City) }
42
47
 
43
48
  context do
44
- let!(:cities) { 3.times.map { City.create! } }
49
+ let!(:cities) { Array.new(3) { City.create! } }
45
50
 
46
51
  specify { expect(subject.identify(City.where(nil))).to match_array(cities.map(&:id)) }
47
52
  specify { expect(subject.identify(cities)).to eq(cities.map(&:id)) }
@@ -51,7 +56,7 @@ describe Chewy::Type::Adapter::ActiveRecord, :active_record do
51
56
 
52
57
  context 'custom primary_key' do
53
58
  before { stub_model(:city) { self.primary_key = 'rating' } }
54
- let!(:cities) { 3.times.map { |i| City.create! { |c| c.rating = i } } }
59
+ let!(:cities) { Array.new(3) { |i| City.create! { |c| c.rating = i } } }
55
60
 
56
61
  specify { expect(subject.identify(City.where(nil))).to match_array([0, 1, 2]) }
57
62
  specify { expect(subject.identify(cities)).to eq([0, 1, 2]) }
@@ -68,46 +73,112 @@ describe Chewy::Type::Adapter::ActiveRecord, :active_record do
68
73
  end
69
74
 
70
75
  context do
71
- let!(:cities) { 3.times.map { City.create! } }
72
- let!(:deleted) { 4.times.map { City.create!.tap(&:destroy) } }
76
+ let!(:cities) { Array.new(3) { City.create! } }
77
+ let!(:deleted) { Array.new(4) { City.create!.tap(&:destroy) } }
73
78
  subject { described_class.new(City) }
74
79
 
75
80
  specify { expect(import).to eq([{index: cities}]) }
76
- specify { expect(import nil).to eq([]) }
81
+ specify { expect(import(nil)).to eq([]) }
77
82
 
78
83
  specify { expect(import(City.order(:id))).to eq([{index: cities}]) }
79
- specify { expect(import(City.order(:id), batch_size: 2))
80
- .to eq([{index: cities.first(2)}, {index: cities.last(1)}]) }
84
+ specify do
85
+ expect(import(City.order(:id), batch_size: 2))
86
+ .to eq([{index: cities.first(2)}, {index: cities.last(1)}])
87
+ end
81
88
 
82
89
  specify { expect(import(cities)).to eq([{index: cities}]) }
83
- specify { expect(import(cities, batch_size: 2))
84
- .to eq([{index: cities.first(2)}, {index: cities.last(1)}]) }
85
- specify { expect(import(cities, deleted))
86
- .to eq([{index: cities}, {delete: deleted}]) }
87
- specify { expect(import(cities, deleted, batch_size: 2)).to eq([
88
- {index: cities.first(2)},
89
- {index: cities.last(1)},
90
- {delete: deleted.first(2)},
91
- {delete: deleted.last(2)}]) }
90
+ specify do
91
+ expect(import(cities, batch_size: 2))
92
+ .to eq([{index: cities.first(2)}, {index: cities.last(1)}])
93
+ end
94
+ specify do
95
+ expect(import(cities, deleted))
96
+ .to eq([{index: cities}, {delete: deleted}])
97
+ end
98
+ specify do
99
+ expect(import(cities, deleted, batch_size: 2)).to eq([
100
+ {index: cities.first(2)},
101
+ {index: cities.last(1)},
102
+ {delete: deleted.first(2)},
103
+ {delete: deleted.last(2)}
104
+ ])
105
+ end
92
106
 
93
107
  specify { expect(import(cities.map(&:id))).to eq([{index: cities}]) }
94
108
  specify { expect(import(deleted.map(&:id))).to eq([{delete: deleted.map(&:id)}]) }
95
- specify { expect(import(cities.map(&:id), batch_size: 2))
96
- .to eq([{index: cities.first(2)}, {index: cities.last(1)}]) }
97
- specify { expect(import(cities.map(&:id), deleted.map(&:id)))
98
- .to eq([{index: cities}, {delete: deleted.map(&:id)}]) }
99
- specify { expect(import(cities.map(&:id), deleted.map(&:id), batch_size: 2)).to eq([
100
- {index: cities.first(2)},
101
- {index: cities.last(1)},
102
- {delete: deleted.first(2).map(&:id)},
103
- {delete: deleted.last(2).map(&:id)}]) }
109
+ specify do
110
+ expect(import(cities.map(&:id), batch_size: 2))
111
+ .to eq([{index: cities.first(2)}, {index: cities.last(1)}])
112
+ end
113
+ specify do
114
+ expect(import(cities.map(&:id), deleted.map(&:id)))
115
+ .to eq([{index: cities}, {delete: deleted.map(&:id)}])
116
+ end
117
+ specify do
118
+ expect(import(cities.map(&:id), deleted.map(&:id), batch_size: 2)).to eq([
119
+ {index: cities.first(2)},
120
+ {index: cities.last(1)},
121
+ {delete: deleted.first(2).map(&:id)},
122
+ {delete: deleted.last(2).map(&:id)}
123
+ ])
124
+ end
104
125
 
105
126
  specify { expect(import(cities.first, nil)).to eq([{index: [cities.first]}]) }
106
127
  specify { expect(import(cities.first.id, nil)).to eq([{index: [cities.first]}]) }
128
+
129
+ context 'raw_import' do
130
+ before do
131
+ stub_class(:dummy_city) do
132
+ def initialize(attributes = {})
133
+ @attributes = attributes
134
+ end
135
+
136
+ def method_missing(name, *args, &block)
137
+ if @attributes.key?(name.to_s)
138
+ @attributes[name.to_s]
139
+ else
140
+ super
141
+ end
142
+ end
143
+
144
+ def respond_to_missing?(name, _)
145
+ @attributes.key?(name.to_s)
146
+ end
147
+ end
148
+ end
149
+ let!(:cities) { Array.new(3) { |i| City.create!(id: i + 1, name: "City#{i + 1}") } }
150
+ let(:converter) { ->(hash) { DummyCity.new(hash) } }
151
+
152
+ it 'uses the raw import converter to make objects out of raw hashes from the database' do
153
+ expect(City).not_to receive(:new)
154
+
155
+ expect(import(City.where(nil), raw_import: converter)).to match([{index: match_array([
156
+ an_instance_of(DummyCity).and(have_attributes(id: 1, name: 'City1')),
157
+ an_instance_of(DummyCity).and(have_attributes(id: 2, name: 'City2')),
158
+ an_instance_of(DummyCity).and(have_attributes(id: 3, name: 'City3'))
159
+ ])}])
160
+ end
161
+
162
+ specify do
163
+ expect(import([1, 2, 3], raw_import: converter)).to match([{index: match_array([
164
+ an_instance_of(DummyCity).and(have_attributes(id: 1, name: 'City1')),
165
+ an_instance_of(DummyCity).and(have_attributes(id: 2, name: 'City2')),
166
+ an_instance_of(DummyCity).and(have_attributes(id: 3, name: 'City3'))
167
+ ])}])
168
+ end
169
+
170
+ specify do
171
+ expect(import(cities, raw_import: converter)).to match([{index: match_array([
172
+ an_instance_of(DummyCity).and(have_attributes(id: 1, name: 'City1')),
173
+ an_instance_of(DummyCity).and(have_attributes(id: 2, name: 'City2')),
174
+ an_instance_of(DummyCity).and(have_attributes(id: 3, name: 'City3'))
175
+ ])}])
176
+ end
177
+ end
107
178
  end
108
179
 
109
- context 'additional delete conitions' do
110
- let!(:cities) { 4.times.map { |i| City.create! rating: i } }
180
+ context 'additional delete conditions' do
181
+ let!(:cities) { Array.new(4) { |i| City.create! rating: i } }
111
182
  before { cities.last(2).map(&:destroy) }
112
183
  subject { described_class.new(City) }
113
184
 
@@ -118,111 +189,164 @@ describe Chewy::Type::Adapter::ActiveRecord, :active_record do
118
189
  end
119
190
  end
120
191
  end
121
- subject { described_class.new(City, delete_if: ->{ delete_already? }) }
122
-
123
- specify { expect(import(City.where(nil))).to eq([
124
- { index: [cities[0]], delete: [cities[1]] }
125
- ]) }
126
- specify { expect(import(cities)).to eq([
127
- { index: [cities[0]], delete: [cities[1]] },
128
- { delete: cities.last(2) }
129
- ]) }
130
- specify { expect(import(cities.map(&:id))).to eq([
131
- { index: [cities[0]], delete: [cities[1]] },
132
- { delete: cities.last(2).map(&:id) }
133
- ]) }
192
+ subject { described_class.new(City, delete_if: -> { delete_already? }) }
193
+
194
+ specify do
195
+ expect(import(City.where(nil))).to eq([
196
+ {index: [cities[0]], delete: [cities[1]]}
197
+ ])
198
+ end
199
+ specify do
200
+ expect(import(cities)).to eq([
201
+ {index: [cities[0]], delete: [cities[1]]},
202
+ {delete: cities.last(2)}
203
+ ])
204
+ end
205
+ specify do
206
+ expect(import(cities.map(&:id))).to eq([
207
+ {index: [cities[0]], delete: [cities[1]]},
208
+ {delete: cities.last(2).map(&:id)}
209
+ ])
210
+ end
134
211
  end
135
212
 
136
213
  context 'custom primary_key' do
137
214
  before { stub_model(:city) { self.primary_key = 'rating' } }
138
- let!(:cities) { 3.times.map { |i| City.create! { |c| c.rating = i + 7 } } }
139
- let!(:deleted) { 3.times.map { |i| City.create! { |c| c.rating = i + 10 }.tap(&:destroy) } }
215
+ let!(:cities) { Array.new(3) { |i| City.create! { |c| c.rating = i + 7 } } }
216
+ let!(:deleted) { Array.new(3) { |i| City.create! { |c| c.rating = i + 10 }.tap(&:destroy) } }
140
217
  subject { described_class.new(City) }
141
218
 
142
219
  specify { expect(import).to eq([{index: cities}]) }
143
220
 
144
221
  specify { expect(import(City.order(:rating))).to eq([{index: cities}]) }
145
- specify { expect(import(City.order(:rating), batch_size: 2))
146
- .to eq([{index: cities.first(2)}, {index: cities.last(1)}]) }
222
+ specify do
223
+ expect(import(City.order(:rating), batch_size: 2))
224
+ .to eq([{index: cities.first(2)}, {index: cities.last(1)}])
225
+ end
147
226
 
148
227
  specify { expect(import(cities)).to eq([{index: cities}]) }
149
- specify { expect(import(cities, batch_size: 2))
150
- .to eq([{index: cities.first(2)}, {index: cities.last(1)}]) }
151
- specify { expect(import(cities, deleted))
152
- .to eq([{index: cities}, {delete: deleted}]) }
153
- specify { expect(import(cities, deleted, batch_size: 2)).to eq([
154
- {index: cities.first(2)},
155
- {index: cities.last(1)},
156
- {delete: deleted.first(2)},
157
- {delete: deleted.last(1)}]) }
228
+ specify do
229
+ expect(import(cities, batch_size: 2))
230
+ .to eq([{index: cities.first(2)}, {index: cities.last(1)}])
231
+ end
232
+ specify do
233
+ expect(import(cities, deleted))
234
+ .to eq([{index: cities}, {delete: deleted}])
235
+ end
236
+ specify do
237
+ expect(import(cities, deleted, batch_size: 2)).to eq([
238
+ {index: cities.first(2)},
239
+ {index: cities.last(1)},
240
+ {delete: deleted.first(2)},
241
+ {delete: deleted.last(1)}
242
+ ])
243
+ end
158
244
 
159
245
  specify { expect(import(cities.map(&:id))).to eq([{index: cities}]) }
160
- specify { expect(import(cities.map(&:id), batch_size: 2))
161
- .to eq([{index: cities.first(2)}, {index: cities.last(1)}]) }
162
- specify { expect(import(cities.map(&:id), deleted.map(&:id)))
163
- .to eq([{index: cities}, {delete: deleted.map(&:id)}]) }
164
- specify { expect(import(cities.map(&:id), deleted.map(&:id), batch_size: 2)).to eq([
165
- {index: cities.first(2)},
166
- {index: cities.last(1)},
167
- {delete: deleted.first(2).map(&:id)},
168
- {delete: deleted.last(1).map(&:id)}]) }
246
+ specify do
247
+ expect(import(cities.map(&:id), batch_size: 2))
248
+ .to eq([{index: cities.first(2)}, {index: cities.last(1)}])
249
+ end
250
+ specify do
251
+ expect(import(cities.map(&:id), deleted.map(&:id)))
252
+ .to eq([{index: cities}, {delete: deleted.map(&:id)}])
253
+ end
254
+ specify do
255
+ expect(import(cities.map(&:id), deleted.map(&:id), batch_size: 2)).to eq([
256
+ {index: cities.first(2)},
257
+ {index: cities.last(1)},
258
+ {delete: deleted.first(2).map(&:id)},
259
+ {delete: deleted.last(1).map(&:id)}
260
+ ])
261
+ end
169
262
  end
170
263
 
171
264
  context 'default scope' do
172
- let!(:cities) { 4.times.map { |i| City.create!(rating: i/3) } }
173
- let!(:deleted) { 3.times.map { |i| City.create!.tap(&:destroy) } }
265
+ let!(:cities) { Array.new(4) { |i| City.create!(rating: i / 3) } }
266
+ let!(:deleted) { Array.new(3) { City.create!.tap(&:destroy) } }
174
267
  subject { described_class.new(City.where(rating: 0)) }
175
268
 
176
269
  specify { expect(import).to eq([{index: cities.first(3)}]) }
177
270
 
178
- specify { expect(import(City.where('rating < 2')))
179
- .to eq([{index: cities.first(3)}]) }
180
- specify { expect(import(City.where('rating < 2'), batch_size: 2))
181
- .to eq([{index: cities.first(2)}, {index: [cities[2]]}]) }
182
- specify { expect(import(City.where('rating < 1')))
183
- .to eq([{index: cities.first(3)}]) }
271
+ specify do
272
+ expect(import(City.where('rating < 2')))
273
+ .to eq([{index: cities.first(3)}])
274
+ end
275
+ specify do
276
+ expect(import(City.where('rating < 2'), batch_size: 2))
277
+ .to eq([{index: cities.first(2)}, {index: [cities[2]]}])
278
+ end
279
+ specify do
280
+ expect(import(City.where('rating < 1')))
281
+ .to eq([{index: cities.first(3)}])
282
+ end
184
283
  specify { expect(import(City.where('rating > 1'))).to eq([]) }
185
284
 
186
- specify { expect(import(cities.first(2)))
187
- .to eq([{index: cities.first(2)}]) }
188
- specify { expect(import(cities))
189
- .to eq([{index: cities.first(3)}, {delete: cities.last(1)}]) }
190
- specify { expect(import(cities, batch_size: 2))
191
- .to eq([{index: cities.first(2)}, {index: [cities[2]]}, {delete: cities.last(1)}]) }
192
- specify { expect(import(cities, deleted))
193
- .to eq([{index: cities.first(3)}, {delete: cities.last(1) + deleted}]) }
194
- specify { expect(import(cities, deleted, batch_size: 3)).to eq([
195
- {index: cities.first(3)},
196
- {delete: cities.last(1) + deleted.first(2)},
197
- {delete: deleted.last(1)}]) }
198
-
199
- specify { expect(import(cities.first(2).map(&:id)))
200
- .to eq([{index: cities.first(2)}]) }
201
- specify { expect(import(cities.map(&:id)))
202
- .to eq([{index: cities.first(3)}, {delete: [cities.last.id]}]) }
203
- specify { expect(import(cities.map(&:id), batch_size: 2))
204
- .to eq([{index: cities.first(2)}, {index: [cities[2]]}, {delete: [cities.last.id]}]) }
205
- specify { expect(import(cities.map(&:id), deleted.map(&:id)))
206
- .to eq([{index: cities.first(3)}, {delete: [cities.last.id] + deleted.map(&:id)}]) }
207
- specify { expect(import(cities.map(&:id), deleted.map(&:id), batch_size: 3)).to eq([
208
- {index: cities.first(3)},
209
- {delete: [cities.last.id] + deleted.first(2).map(&:id)},
210
- {delete: deleted.last(1).map(&:id)}]) }
285
+ specify do
286
+ expect(import(cities.first(2)))
287
+ .to eq([{index: cities.first(2)}])
288
+ end
289
+ specify do
290
+ expect(import(cities))
291
+ .to eq([{index: cities.first(3)}, {delete: cities.last(1)}])
292
+ end
293
+ specify do
294
+ expect(import(cities, batch_size: 2))
295
+ .to eq([{index: cities.first(2)}, {index: [cities[2]]}, {delete: cities.last(1)}])
296
+ end
297
+ specify do
298
+ expect(import(cities, deleted))
299
+ .to eq([{index: cities.first(3)}, {delete: cities.last(1) + deleted}])
300
+ end
301
+ specify do
302
+ expect(import(cities, deleted, batch_size: 3)).to eq([
303
+ {index: cities.first(3)},
304
+ {delete: cities.last(1) + deleted.first(2)},
305
+ {delete: deleted.last(1)}
306
+ ])
307
+ end
308
+
309
+ specify do
310
+ expect(import(cities.first(2).map(&:id)))
311
+ .to eq([{index: cities.first(2)}])
312
+ end
313
+ specify do
314
+ expect(import(cities.map(&:id)))
315
+ .to eq([{index: cities.first(3)}, {delete: [cities.last.id]}])
316
+ end
317
+ specify do
318
+ expect(import(cities.map(&:id), batch_size: 2))
319
+ .to eq([{index: cities.first(2)}, {index: [cities[2]]}, {delete: [cities.last.id]}])
320
+ end
321
+ specify do
322
+ expect(import(cities.map(&:id), deleted.map(&:id)))
323
+ .to eq([{index: cities.first(3)}, {delete: [cities.last.id] + deleted.map(&:id)}])
324
+ end
325
+ specify do
326
+ expect(import(cities.map(&:id), deleted.map(&:id), batch_size: 3)).to eq([
327
+ {index: cities.first(3)},
328
+ {delete: [cities.last.id] + deleted.first(2).map(&:id)},
329
+ {delete: deleted.last(1).map(&:id)}
330
+ ])
331
+ end
211
332
  end
212
333
 
213
334
  context 'error handling' do
214
- let!(:cities) { 3.times.map { |i| City.create! } }
215
- let!(:deleted) { 2.times.map { |i| City.create!.tap(&:destroy) } }
335
+ let!(:cities) { Array.new(3) { City.create! } }
336
+ let!(:deleted) { Array.new(2) { City.create!.tap(&:destroy) } }
216
337
  let(:ids) { (cities + deleted).map(&:id) }
217
338
  subject { described_class.new(City) }
218
339
 
219
340
  let(:data_comparer) do
220
- ->(id, data) { objects = data[:index] || data[:delete]; !objects.map { |o| o.respond_to?(:id) ? o.id : o }.include?(id) }
341
+ lambda do |id, data|
342
+ objects = data[:index] || data[:delete]
343
+ !objects.map { |o| o.respond_to?(:id) ? o.id : o }.include?(id)
344
+ end
221
345
  end
222
346
 
223
347
  context 'implicit scope' do
224
- specify { expect(subject.import { |data| true }).to eq(true) }
225
- specify { expect(subject.import { |data| false }).to eq(false) }
348
+ specify { expect(subject.import { |_data| true }).to eq(true) }
349
+ specify { expect(subject.import { |_data| false }).to eq(false) }
226
350
  specify { expect(subject.import(batch_size: 1, &data_comparer.curry[cities[0].id])).to eq(false) }
227
351
  specify { expect(subject.import(batch_size: 1, &data_comparer.curry[cities[1].id])).to eq(false) }
228
352
  specify { expect(subject.import(batch_size: 1, &data_comparer.curry[cities[2].id])).to eq(false) }
@@ -233,8 +357,8 @@ describe Chewy::Type::Adapter::ActiveRecord, :active_record do
233
357
  context 'explicit scope' do
234
358
  let(:scope) { City.where(id: ids) }
235
359
 
236
- specify { expect(subject.import(scope) { |data| true }).to eq(true) }
237
- specify { expect(subject.import(scope) { |data| false }).to eq(false) }
360
+ specify { expect(subject.import(scope) { |_data| true }).to eq(true) }
361
+ specify { expect(subject.import(scope) { |_data| false }).to eq(false) }
238
362
  specify { expect(subject.import(scope, batch_size: 1, &data_comparer.curry[cities[0].id])).to eq(false) }
239
363
  specify { expect(subject.import(scope, batch_size: 1, &data_comparer.curry[cities[1].id])).to eq(false) }
240
364
  specify { expect(subject.import(scope, batch_size: 1, &data_comparer.curry[cities[2].id])).to eq(false) }
@@ -243,8 +367,8 @@ describe Chewy::Type::Adapter::ActiveRecord, :active_record do
243
367
  end
244
368
 
245
369
  context 'objects' do
246
- specify { expect(subject.import(cities + deleted) { |data| true }).to eq(true) }
247
- specify { expect(subject.import(cities + deleted) { |data| false }).to eq(false) }
370
+ specify { expect(subject.import(cities + deleted) { |_data| true }).to eq(true) }
371
+ specify { expect(subject.import(cities + deleted) { |_data| false }).to eq(false) }
248
372
  specify { expect(subject.import(cities + deleted, batch_size: 1, &data_comparer.curry[cities[0].id])).to eq(false) }
249
373
  specify { expect(subject.import(cities + deleted, batch_size: 1, &data_comparer.curry[cities[1].id])).to eq(false) }
250
374
  specify { expect(subject.import(cities + deleted, batch_size: 1, &data_comparer.curry[cities[2].id])).to eq(false) }
@@ -253,8 +377,8 @@ describe Chewy::Type::Adapter::ActiveRecord, :active_record do
253
377
  end
254
378
 
255
379
  context 'ids' do
256
- specify { expect(subject.import(ids) { |data| true }).to eq(true) }
257
- specify { expect(subject.import(ids) { |data| false }).to eq(false) }
380
+ specify { expect(subject.import(ids) { |_data| true }).to eq(true) }
381
+ specify { expect(subject.import(ids) { |_data| false }).to eq(false) }
258
382
  specify { expect(subject.import(ids, batch_size: 1, &data_comparer.curry[cities[0].id])).to eq(false) }
259
383
  specify { expect(subject.import(ids, batch_size: 1, &data_comparer.curry[cities[1].id])).to eq(false) }
260
384
  specify { expect(subject.import(ids, batch_size: 1, &data_comparer.curry[cities[2].id])).to eq(false) }
@@ -264,54 +388,148 @@ describe Chewy::Type::Adapter::ActiveRecord, :active_record do
264
388
  end
265
389
  end
266
390
 
391
+ describe '#import_fields' do
392
+ subject { described_class.new(Country) }
393
+ let!(:countries) { Array.new(3) { |i| Country.create!(rating: i) { |c| c.id = i + 1 } } }
394
+ let!(:cities) { Array.new(6) { |i| City.create!(rating: i + 3, country_id: (i + 4) / 2) { |c| c.id = i + 3 } } }
395
+
396
+ specify { expect(subject.import_fields).to match([contain_exactly(1, 2, 3)]) }
397
+ specify { expect(subject.import_fields(fields: [:rating])).to match([contain_exactly([1, 0], [2, 1], [3, 2])]) }
398
+
399
+ context 'scopes' do
400
+ context do
401
+ subject { described_class.new(Country.includes(:cities)) }
402
+
403
+ specify { expect(subject.import_fields).to match([contain_exactly(1, 2, 3)]) }
404
+ specify { expect(subject.import_fields(fields: [:rating])).to match([contain_exactly([1, 0], [2, 1], [3, 2])]) }
405
+ end
406
+
407
+ context do
408
+ subject { described_class.new(Country.joins(:cities)) }
409
+
410
+ specify { expect(subject.import_fields).to match([contain_exactly(2, 3)]) }
411
+ specify { expect(subject.import_fields(fields: [:rating])).to match([contain_exactly([2, 1], [3, 2])]) }
412
+ end
413
+
414
+ context 'ignores default scope if another scope is passed' do
415
+ subject { described_class.new(Country.joins(:cities)) }
416
+
417
+ specify { expect(subject.import_fields(Country.where('rating < 2'))).to match([contain_exactly(1, 2)]) }
418
+ specify { expect(subject.import_fields(Country.where('rating < 2'), fields: [:rating])).to match([contain_exactly([1, 0], [2, 1])]) }
419
+ end
420
+ end
421
+
422
+ context 'objects/ids' do
423
+ specify { expect(subject.import_fields(1, 2)).to match([contain_exactly(1, 2)]) }
424
+ specify { expect(subject.import_fields(1, 2, fields: [:rating])).to match([contain_exactly([1, 0], [2, 1])]) }
425
+
426
+ specify { expect(subject.import_fields(countries.first(2))).to match([contain_exactly(1, 2)]) }
427
+ specify { expect(subject.import_fields(countries.first(2), fields: [:rating])).to match([contain_exactly([1, 0], [2, 1])]) }
428
+ end
429
+
430
+ context 'batch_size' do
431
+ specify { expect(subject.import_fields(batch_size: 2)).to match([contain_exactly(1, 2), [3]]) }
432
+ specify { expect(subject.import_fields(batch_size: 2, fields: [:rating])).to match([contain_exactly([1, 0], [2, 1]), [[3, 2]]]) }
433
+
434
+ specify { expect(subject.import_fields(Country.where('rating < 2'), batch_size: 2)).to match([contain_exactly(1, 2)]) }
435
+ specify { expect(subject.import_fields(Country.where('rating < 2'), batch_size: 2, fields: [:rating])).to match([contain_exactly([1, 0], [2, 1])]) }
436
+
437
+ specify { expect(subject.import_fields(1, 2, batch_size: 1)).to match([[1], [2]]) }
438
+ specify { expect(subject.import_fields(1, 2, batch_size: 1, fields: [:rating])).to match([[[1, 0]], [[2, 1]]]) }
439
+
440
+ specify { expect(subject.import_fields(countries.first(2), batch_size: 1)).to match([[1], [2]]) }
441
+ specify { expect(subject.import_fields(countries.first(2), batch_size: 1, fields: [:rating])).to match([[[1, 0]], [[2, 1]]]) }
442
+ end
443
+
444
+ context 'typecast' do
445
+ specify { expect(subject.import_fields(typecast: false)).to match([contain_exactly(1, 2, 3)]) }
446
+ specify do
447
+ expect(subject.import_fields(fields: [:updated_at]).to_a)
448
+ .to match([contain_exactly(
449
+ [1, an_instance_of(Time)],
450
+ [2, an_instance_of(Time)],
451
+ [3, an_instance_of(Time)]
452
+ )])
453
+ end
454
+ specify do
455
+ expect(subject.import_fields(fields: [:updated_at], typecast: false))
456
+ .to match([contain_exactly(
457
+ [1, match(/#{Time.now.strftime('%Y-%m-%d')}/)],
458
+ [2, match(/#{Time.now.strftime('%Y-%m-%d')}/)],
459
+ [3, match(/#{Time.now.strftime('%Y-%m-%d')}/)]
460
+ )])
461
+ end
462
+ end
463
+ end
464
+
267
465
  describe '#load' do
268
466
  context do
269
- let!(:cities) { 3.times.map { |i| City.create!(rating: i/2) } }
270
- let!(:deleted) { 2.times.map { |i| City.create!.tap(&:destroy) } }
467
+ let!(:cities) { Array.new(3) { |i| City.create!(rating: i / 2) } }
468
+ let!(:deleted) { Array.new(2) { City.create!.tap(&:destroy) } }
469
+ let(:city_ids) { cities.map(&:id) }
470
+ let(:deleted_ids) { deleted.map(&:id) }
271
471
 
272
472
  let(:type) { double(type_name: 'user') }
273
473
 
274
474
  subject { described_class.new(City) }
275
475
 
276
- specify { expect(subject.load(cities.map { |c| double(id: c.id) }, _type: type)).to eq(cities) }
277
- specify { expect(subject.load(cities.map { |c| double(id: c.id) }.reverse, _type: type)).to eq(cities.reverse) }
278
- specify { expect(subject.load(deleted.map { |c| double(id: c.id) }, _type: type)).to eq([nil, nil]) }
279
- specify { expect(subject.load((cities + deleted).map { |c| double(id: c.id) }, _type: type)).to eq([*cities, nil, nil]) }
280
- specify { expect(subject.load(cities.map { |c| double(id: c.id) }, _type: type, scope: ->{ where(rating: 0) }))
281
- .to eq(cities.first(2) + [nil]) }
282
- specify { expect(subject.load(cities.map { |c| double(id: c.id) },
283
- _type: type, scope: ->{ where(rating: 0) }, user: {scope: ->{ where(rating: 1)}}))
284
- .to eq([nil, nil] + cities.last(1)) }
285
- specify { expect(subject.load(cities.map { |c| double(id: c.id) }, _type: type, scope: City.where(rating: 1)))
286
- .to eq([nil, nil] + cities.last(1)) }
287
- specify { expect(subject.load(cities.map { |c| double(id: c.id) },
288
- _type: type, scope: City.where(rating: 1), user: {scope: ->{ where(rating: 0)}}))
289
- .to eq(cities.first(2) + [nil]) }
476
+ specify { expect(subject.load(city_ids, _type: type)).to eq(cities) }
477
+ specify { expect(subject.load(city_ids.reverse, _type: type)).to eq(cities.reverse) }
478
+ specify { expect(subject.load(deleted_ids, _type: type)).to eq([nil, nil]) }
479
+ specify { expect(subject.load(city_ids + deleted_ids, _type: type)).to eq([*cities, nil, nil]) }
480
+ specify do
481
+ expect(subject.load(city_ids, _type: type, scope: -> { where(rating: 0) }))
482
+ .to eq(cities.first(2) + [nil])
483
+ end
484
+ specify do
485
+ expect(subject.load(city_ids,
486
+ _type: type, scope: -> { where(rating: 0) }, user: {scope: -> { where(rating: 1) }}))
487
+ .to eq([nil, nil] + cities.last(1))
488
+ end
489
+ specify do
490
+ expect(subject.load(city_ids, _type: type, scope: City.where(rating: 1)))
491
+ .to eq([nil, nil] + cities.last(1))
492
+ end
493
+ specify do
494
+ expect(subject.load(city_ids,
495
+ _type: type, scope: City.where(rating: 1), user: {scope: -> { where(rating: 0) }}))
496
+ .to eq(cities.first(2) + [nil])
497
+ end
290
498
  end
291
499
 
292
500
  context 'custom primary_key' do
293
501
  before { stub_model(:city) { self.primary_key = 'rating' } }
294
- let!(:cities) { 3.times.map { |i| City.create!(country_id: i/2) { |c| c.rating = i + 7 } } }
295
- let!(:deleted) { 2.times.map { |i| City.create! { |c| c.rating = i + 10 }.tap(&:destroy) } }
502
+ let!(:cities) { Array.new(3) { |i| City.create!(country_id: i / 2) { |c| c.rating = i + 7 } } }
503
+ let!(:deleted) { Array.new(2) { |i| City.create! { |c| c.rating = i + 10 }.tap(&:destroy) } }
504
+ let(:city_ids) { cities.map(&:rating) }
505
+ let(:deleted_ids) { deleted.map(&:rating) }
296
506
 
297
507
  let(:type) { double(type_name: 'user') }
298
508
 
299
509
  subject { described_class.new(City) }
300
510
 
301
- specify { expect(subject.load(cities.map { |c| double(id: c.id) }, _type: type)).to eq(cities) }
302
- specify { expect(subject.load(cities.map { |c| double(id: c.id) }.reverse, _type: type)).to eq(cities.reverse) }
303
- specify { expect(subject.load(deleted.map { |c| double(id: c.id) }, _type: type)).to eq([nil, nil]) }
304
- specify { expect(subject.load((cities + deleted).map { |c| double(id: c.id) }, _type: type)).to eq([*cities, nil, nil]) }
305
- specify { expect(subject.load(cities.map { |c| double(id: c.id) }, _type: type, scope: ->{ where(country_id: 0) }))
306
- .to eq(cities.first(2) + [nil]) }
307
- specify { expect(subject.load(cities.map { |c| double(id: c.id) },
308
- _type: type, scope: ->{ where(country_id: 0) }, user: {scope: ->{ where(country_id: 1)}}))
309
- .to eq([nil, nil] + cities.last(1)) }
310
- specify { expect(subject.load(cities.map { |c| double(id: c.id) }, _type: type, scope: City.where(country_id: 1)))
311
- .to eq([nil, nil] + cities.last(1)) }
312
- specify { expect(subject.load(cities.map { |c| double(id: c.id) },
313
- _type: type, scope: City.where(country_id: 1), user: {scope: ->{ where(country_id: 0)}}))
314
- .to eq(cities.first(2) + [nil]) }
511
+ specify { expect(subject.load(city_ids, _type: type)).to eq(cities) }
512
+ specify { expect(subject.load(city_ids.reverse, _type: type)).to eq(cities.reverse) }
513
+ specify { expect(subject.load(deleted_ids, _type: type)).to eq([nil, nil]) }
514
+ specify { expect(subject.load(city_ids + deleted_ids, _type: type)).to eq([*cities, nil, nil]) }
515
+ specify do
516
+ expect(subject.load(city_ids, _type: type, scope: -> { where(country_id: 0) }))
517
+ .to eq(cities.first(2) + [nil])
518
+ end
519
+ specify do
520
+ expect(subject.load(city_ids,
521
+ _type: type, scope: -> { where(country_id: 0) }, user: {scope: -> { where(country_id: 1) }}))
522
+ .to eq([nil, nil] + cities.last(1))
523
+ end
524
+ specify do
525
+ expect(subject.load(city_ids, _type: type, scope: City.where(country_id: 1)))
526
+ .to eq([nil, nil] + cities.last(1))
527
+ end
528
+ specify do
529
+ expect(subject.load(city_ids,
530
+ _type: type, scope: City.where(country_id: 1), user: {scope: -> { where(country_id: 0) }}))
531
+ .to eq(cities.first(2) + [nil])
532
+ end
315
533
  end
316
534
  end
317
535
  end