spree_core 3.0.10 → 3.1.0.rc1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (289) hide show
  1. checksums.yaml +4 -4
  2. data/app/assets/javascripts/spree.js.coffee.erb +12 -3
  3. data/app/helpers/spree/base_helper.rb +4 -1
  4. data/app/helpers/spree/products_helper.rb +37 -6
  5. data/app/mailers/spree/base_mailer.rb +11 -2
  6. data/app/models/concerns/spree/adjustment_source.rb +3 -10
  7. data/app/models/concerns/spree/default_price.rb +7 -1
  8. data/app/models/concerns/spree/named_type.rb +1 -1
  9. data/app/models/concerns/spree/user_api_authentication.rb +7 -1
  10. data/app/models/concerns/spree/user_methods.rb +47 -0
  11. data/app/models/concerns/spree/user_reporting.rb +2 -2
  12. data/app/models/concerns/spree/vat_price_calculation.rb +47 -0
  13. data/app/models/spree/address.rb +8 -7
  14. data/app/models/spree/adjustable/adjuster/base.rb +25 -0
  15. data/app/models/spree/adjustable/adjuster/promotion.rb +41 -0
  16. data/app/models/spree/adjustable/adjuster/tax.rb +26 -0
  17. data/app/models/spree/adjustable/adjustments_updater.rb +22 -45
  18. data/app/models/spree/adjustment.rb +8 -10
  19. data/app/models/spree/app_configuration.rb +5 -2
  20. data/app/models/spree/base.rb +4 -0
  21. data/app/models/spree/calculator.rb +0 -5
  22. data/app/models/spree/calculator/default_tax.rb +2 -10
  23. data/app/models/spree/classification.rb +7 -3
  24. data/app/models/spree/country.rb +14 -3
  25. data/app/models/spree/credit_card.rb +21 -31
  26. data/app/models/spree/customer_return.rb +7 -15
  27. data/app/models/spree/gateway.rb +7 -6
  28. data/app/models/spree/image.rb +1 -1
  29. data/app/models/spree/inventory_unit.rb +9 -6
  30. data/app/models/spree/legacy_user.rb +1 -6
  31. data/app/models/spree/line_item.rb +69 -68
  32. data/app/models/spree/log_entry.rb +1 -4
  33. data/app/models/spree/option_type.rb +15 -6
  34. data/app/models/spree/option_type_prototype.rb +9 -0
  35. data/app/models/spree/option_value.rb +11 -3
  36. data/app/models/spree/option_value_variant.rb +6 -0
  37. data/app/models/spree/order.rb +113 -64
  38. data/app/models/spree/order/checkout.rb +8 -11
  39. data/app/models/spree/order/currency_updater.rb +1 -1
  40. data/app/models/spree/order/store_credit.rb +96 -0
  41. data/app/models/spree/order_contents.rb +6 -1
  42. data/app/models/spree/order_inventory.rb +4 -8
  43. data/app/models/spree/order_promotion.rb +6 -0
  44. data/app/models/spree/order_updater.rb +2 -7
  45. data/app/models/spree/payment.rb +46 -19
  46. data/app/models/spree/payment/gateway_options.rb +8 -4
  47. data/app/models/spree/payment_method.rb +12 -13
  48. data/app/models/spree/payment_method/store_credit.rb +130 -0
  49. data/app/models/spree/preference.rb +1 -1
  50. data/app/models/spree/price.rb +16 -6
  51. data/app/models/spree/product.rb +52 -49
  52. data/app/models/spree/product/scopes.rb +7 -2
  53. data/app/models/spree/product_option_type.rb +7 -2
  54. data/app/models/spree/product_promotion_rule.rb +9 -0
  55. data/app/models/spree/product_property.rb +8 -10
  56. data/app/models/spree/promotion.rb +19 -19
  57. data/app/models/spree/promotion/rules/product.rb +3 -1
  58. data/app/models/spree/promotion/rules/taxon.rb +2 -1
  59. data/app/models/spree/promotion/rules/user.rb +4 -4
  60. data/app/models/spree/promotion_action.rb +3 -3
  61. data/app/models/spree/promotion_category.rb +1 -1
  62. data/app/models/spree/promotion_rule_taxon.rb +9 -0
  63. data/app/models/spree/promotion_rule_user.rb +9 -0
  64. data/app/models/spree/property.rb +2 -1
  65. data/app/models/spree/property_prototype.rb +9 -0
  66. data/app/models/spree/prototype.rb +8 -3
  67. data/app/models/spree/prototype_taxon.rb +9 -0
  68. data/app/models/spree/refund.rb +10 -7
  69. data/app/models/spree/refund_reason.rb +1 -1
  70. data/app/models/spree/reimbursement.rb +12 -16
  71. data/app/models/spree/reimbursement_type/reimbursement_helpers.rb +23 -6
  72. data/app/models/spree/reimbursement_type/store_credit.rb +28 -0
  73. data/app/models/spree/return_authorization.rb +8 -13
  74. data/app/models/spree/return_authorization_reason.rb +1 -1
  75. data/app/models/spree/return_item.rb +13 -12
  76. data/app/models/spree/return_item/eligibility_validator/time_since_purchase.rb +1 -1
  77. data/app/models/spree/role.rb +3 -2
  78. data/app/models/spree/role_user.rb +6 -0
  79. data/app/models/spree/shipment.rb +18 -23
  80. data/app/models/spree/shipment_handler.rb +2 -2
  81. data/app/models/spree/shipping_category.rb +6 -3
  82. data/app/models/spree/shipping_method.rb +11 -10
  83. data/app/models/spree/shipping_method_zone.rb +6 -0
  84. data/app/models/spree/shipping_rate.rb +16 -29
  85. data/app/models/spree/state.rb +3 -2
  86. data/app/models/spree/state_change.rb +1 -1
  87. data/app/models/spree/stock/content_item.rb +10 -12
  88. data/app/models/spree/stock/coordinator.rb +13 -14
  89. data/app/models/spree/stock/estimator.rb +28 -30
  90. data/app/models/spree/stock/inventory_unit_builder.rb +1 -9
  91. data/app/models/spree/stock/packer.rb +1 -1
  92. data/app/models/spree/stock/quantifier.rb +5 -5
  93. data/app/models/spree/stock/splitter/backordered.rb +2 -2
  94. data/app/models/spree/stock_item.rb +12 -18
  95. data/app/models/spree/stock_location.rb +4 -7
  96. data/app/models/spree/stock_movement.rb +11 -9
  97. data/app/models/spree/stock_transfer.rb +11 -12
  98. data/app/models/spree/store.rb +14 -6
  99. data/app/models/spree/store_credit.rb +252 -0
  100. data/app/models/spree/store_credit_category.rb +22 -0
  101. data/app/models/spree/store_credit_event.rb +38 -0
  102. data/app/models/spree/store_credit_type.rb +6 -0
  103. data/app/models/spree/tax_category.rb +3 -8
  104. data/app/models/spree/tax_rate.rb +56 -122
  105. data/app/models/spree/taxon.rb +11 -5
  106. data/app/models/spree/tracker.rb +12 -1
  107. data/app/models/spree/validations/db_maximum_length_validator.rb +2 -1
  108. data/app/models/spree/variant.rb +82 -50
  109. data/app/models/spree/zone.rb +21 -17
  110. data/app/models/spree/zone_member.rb +6 -0
  111. data/app/validators/db_maximum_length_validator.rb +11 -0
  112. data/app/views/layouts/spree/base_mailer.html.erb +38 -781
  113. data/app/views/spree/order_mailer/_adjustment.html.erb +8 -0
  114. data/app/views/spree/order_mailer/_subtotal.html.erb +8 -0
  115. data/app/views/spree/order_mailer/_total.html.erb +8 -0
  116. data/app/views/spree/order_mailer/cancel_email.html.erb +13 -28
  117. data/app/views/spree/order_mailer/cancel_email.text.erb +1 -1
  118. data/app/views/spree/order_mailer/confirm_email.html.erb +49 -63
  119. data/app/views/spree/order_mailer/confirm_email.text.erb +1 -1
  120. data/app/views/spree/shared/_base_mailer_header.html.erb +5 -7
  121. data/app/views/spree/shared/_base_mailer_stylesheets.html.erb +777 -0
  122. data/app/views/spree/shared/_error_messages.html.erb +1 -1
  123. data/app/views/spree/shared/_mailer_line_item.html.erb +12 -0
  124. data/app/views/spree/shipment_mailer/shipped_email.html.erb +21 -14
  125. data/app/views/spree/shipment_mailer/shipped_email.text.erb +3 -3
  126. data/config/initializers/user_class_extensions.rb +7 -38
  127. data/config/locales/en.yml +113 -13
  128. data/config/routes.rb +7 -0
  129. data/db/default/spree/default_reimbursement_type.rb +1 -1
  130. data/db/default/spree/zones.rb +4 -5
  131. data/db/migrate/20150118210639_create_spree_store_credits.rb +24 -0
  132. data/db/migrate/20150118211500_create_spree_store_credit_categories.rb +8 -0
  133. data/db/migrate/20150118212051_create_spree_store_credit_events.rb +17 -0
  134. data/db/migrate/20150118212101_create_spree_store_credit_types.rb +10 -0
  135. data/db/migrate/20150314013438_add_missing_indexes.rb +25 -0
  136. data/db/migrate/20150317174308_remove_duplicated_indexes_from_multi_columns.rb +18 -0
  137. data/db/migrate/20150324104002_remove_user_index_from_spree_state_changes.rb +14 -0
  138. data/db/migrate/20150522071831_add_position_to_spree_payment_methods.rb +5 -0
  139. data/db/migrate/20150626181949_add_taxable_adjustment_total_to_line_item.rb +19 -0
  140. data/db/migrate/20150627090949_migrate_payment_methods_display.rb +12 -0
  141. data/db/migrate/20150714154102_spree_payment_method_store_credits.rb +12 -0
  142. data/db/migrate/20150726141425_rename_has_and_belongs_to_associations_to_model_names.rb +18 -0
  143. data/db/migrate/20150727191614_spree_store_credit_types.rb +11 -0
  144. data/db/migrate/20150819154308_add_discontinued_to_products_and_variants.rb +68 -0
  145. data/db/migrate/20151220072838_remove_shipping_method_id_from_spree_orders.rb +13 -0
  146. data/db/migrate/20160207191757_add_id_column_to_earlier_habtm_tables.rb +16 -0
  147. data/db/migrate/20160219165458_add_indexes.rb +14 -0
  148. data/lib/generators/spree/dummy/templates/rails/database.yml +31 -24
  149. data/lib/generators/spree/dummy/templates/rails/test.rb +2 -1
  150. data/lib/spree/core.rb +16 -0
  151. data/lib/spree/core/controller_helpers/auth.rb +1 -1
  152. data/lib/spree/core/controller_helpers/common.rb +3 -3
  153. data/lib/spree/core/controller_helpers/order.rb +6 -5
  154. data/lib/spree/core/controller_helpers/search.rb +1 -1
  155. data/lib/spree/core/controller_helpers/store.rb +29 -0
  156. data/lib/spree/core/delegate_belongs_to.rb +2 -2
  157. data/lib/spree/core/engine.rb +30 -25
  158. data/lib/spree/core/environment.rb +1 -1
  159. data/lib/spree/core/importer/order.rb +37 -40
  160. data/lib/spree/core/number_generator.rb +52 -0
  161. data/lib/spree/core/product_filters.rb +1 -1
  162. data/lib/spree/core/search/base.rb +4 -3
  163. data/lib/spree/core/version.rb +1 -1
  164. data/lib/spree/localized_number.rb +3 -1
  165. data/lib/spree/permitted_attributes.rb +5 -2
  166. data/lib/spree/testing_support/common_rake.rb +3 -3
  167. data/lib/spree/testing_support/factories.rb +3 -3
  168. data/lib/spree/testing_support/factories/address_factory.rb +1 -1
  169. data/lib/spree/testing_support/factories/country_factory.rb +2 -2
  170. data/lib/spree/testing_support/factories/order_factory.rb +2 -2
  171. data/lib/spree/testing_support/factories/payment_factory.rb +5 -0
  172. data/lib/spree/testing_support/factories/payment_method_factory.rb +8 -0
  173. data/lib/spree/testing_support/factories/promotion_rule_factory.rb +5 -0
  174. data/lib/spree/testing_support/factories/refund_factory.rb +9 -1
  175. data/lib/spree/testing_support/factories/return_authorization_factory.rb +2 -0
  176. data/lib/spree/testing_support/factories/state_factory.rb +2 -2
  177. data/lib/spree/testing_support/factories/store_credit_category_factory.rb +9 -0
  178. data/lib/spree/testing_support/factories/store_credit_event_factory.rb +8 -0
  179. data/lib/spree/testing_support/factories/store_credit_factory.rb +17 -0
  180. data/lib/spree/testing_support/factories/store_credit_type_factory.rb +11 -0
  181. data/lib/spree/testing_support/factories/user_factory.rb +1 -1
  182. data/lib/spree/testing_support/factories/zone_member_factory.rb +6 -0
  183. data/lib/spree/testing_support/microdata.rb +189 -0
  184. data/lib/tasks/core.rake +68 -0
  185. data/lib/tasks/exchanges.rake +2 -2
  186. data/spec/fixtures/microdata.html +22 -0
  187. data/spec/fixtures/microdata_itemref.html +15 -0
  188. data/spec/fixtures/microdata_no_itemscope.html +20 -0
  189. data/spec/helpers/base_helper_spec.rb +64 -1
  190. data/spec/helpers/products_helper_spec.rb +75 -3
  191. data/spec/lib/i18n_spec.rb +2 -2
  192. data/spec/lib/search/base_spec.rb +2 -2
  193. data/spec/lib/spree/core/controller_helpers/auth_spec.rb +4 -2
  194. data/spec/lib/spree/core/controller_helpers/store_spec.rb +56 -0
  195. data/spec/lib/spree/core/importer/order_spec.rb +226 -123
  196. data/spec/lib/spree/core/number_generator_spec.rb +175 -0
  197. data/spec/lib/spree/core_spec.rb +23 -0
  198. data/spec/lib/spree/localized_number_spec.rb +10 -0
  199. data/spec/mailers/order_mailer_spec.rb +11 -13
  200. data/spec/mailers/shipment_mailer_spec.rb +26 -8
  201. data/spec/mailers/test_mailer_spec.rb +15 -1
  202. data/spec/models/option_type_prototype_spec.rb +9 -0
  203. data/spec/models/spree/ability_spec.rb +6 -13
  204. data/spec/models/spree/address_spec.rb +1 -1
  205. data/spec/models/spree/adjustable/adjuster/base_spec.rb +10 -0
  206. data/spec/models/spree/adjustable/adjuster/promotion_spec.rb +211 -0
  207. data/spec/models/spree/adjustable/adjuster/tax_spec.rb +86 -0
  208. data/spec/models/spree/adjustable/adjustments_updater_spec.rb +2 -262
  209. data/spec/models/spree/adjustment_spec.rb +27 -1
  210. data/spec/models/spree/app_configuration_spec.rb +5 -2
  211. data/spec/models/spree/calculator/default_tax_spec.rb +39 -14
  212. data/spec/models/spree/concerns/user_methods_spec.rb +55 -0
  213. data/spec/models/spree/concerns/vat_price_calculation_spec.rb +66 -0
  214. data/spec/models/spree/country_spec.rb +45 -8
  215. data/spec/models/spree/credit_card_spec.rb +8 -8
  216. data/spec/models/spree/customer_return_spec.rb +4 -26
  217. data/spec/models/spree/gateway_spec.rb +7 -0
  218. data/spec/models/spree/image_spec.rb +3 -0
  219. data/spec/models/spree/inventory_unit_spec.rb +1 -18
  220. data/spec/models/spree/line_item_spec.rb +78 -18
  221. data/spec/models/spree/option_type_spec.rb +2 -2
  222. data/spec/models/spree/option_value_spec.rb +8 -3
  223. data/spec/models/spree/order/checkout_spec.rb +49 -39
  224. data/spec/models/spree/order/currency_updater_spec.rb +3 -3
  225. data/spec/models/spree/order/finalizing_spec.rb +0 -3
  226. data/spec/models/spree/order/payment_spec.rb +1 -1
  227. data/spec/models/spree/order/state_machine_spec.rb +1 -6
  228. data/spec/models/spree/order/store_credit_spec.rb +423 -0
  229. data/spec/models/spree/order/updating_spec.rb +2 -2
  230. data/spec/models/spree/order_contents_spec.rb +42 -1
  231. data/spec/models/spree/order_inventory_spec.rb +27 -17
  232. data/spec/models/spree/order_spec.rb +65 -52
  233. data/spec/models/spree/payment/gateway_options_spec.rb +10 -2
  234. data/spec/models/spree/payment/store_credit_spec.rb +60 -0
  235. data/spec/models/spree/payment_method/store_credit_spec.rb +291 -0
  236. data/spec/models/spree/payment_method_spec.rb +22 -14
  237. data/spec/models/spree/payment_spec.rb +37 -44
  238. data/spec/models/spree/price_spec.rb +86 -0
  239. data/spec/models/spree/product/scopes_spec.rb +35 -0
  240. data/spec/models/spree/product_option_type_spec.rb +6 -2
  241. data/spec/models/spree/product_promotion_rule_spec.rb +9 -0
  242. data/spec/models/spree/product_property_spec.rb +11 -0
  243. data/spec/models/spree/product_spec.rb +82 -15
  244. data/spec/models/spree/promotion/actions/create_item_adjustments_spec.rb +1 -1
  245. data/spec/models/spree/promotion/rules/user_spec.rb +8 -0
  246. data/spec/models/spree/promotion_action_spec.rb +1 -1
  247. data/spec/models/spree/promotion_rule_spec.rb +1 -1
  248. data/spec/models/spree/promotion_rule_taxon_spec.rb +9 -0
  249. data/spec/models/spree/promotion_rule_user_spec.rb +9 -0
  250. data/spec/models/spree/promotion_spec.rb +57 -36
  251. data/spec/models/spree/property_prototype_spec.rb +9 -0
  252. data/spec/models/spree/prototype_taxon_spec.rb +9 -0
  253. data/spec/models/spree/refund_reason_spec.rb +7 -0
  254. data/spec/models/spree/reimbursement_spec.rb +3 -30
  255. data/spec/models/spree/reimbursement_tax_calculator_spec.rb +17 -5
  256. data/spec/models/spree/reimbursement_type/store_credit_spec.rb +101 -0
  257. data/spec/models/spree/return_authorization_reason_spec.rb +7 -0
  258. data/spec/models/spree/return_authorization_spec.rb +2 -22
  259. data/spec/models/spree/return_item_spec.rb +50 -1
  260. data/spec/models/spree/returns_calculator_spec.rb +1 -1
  261. data/spec/models/spree/role_spec.rb +7 -0
  262. data/spec/models/spree/shipment_spec.rb +17 -17
  263. data/spec/models/spree/shipping_calculator_spec.rb +2 -2
  264. data/spec/models/spree/shipping_category_spec.rb +14 -0
  265. data/spec/models/spree/shipping_method_spec.rb +9 -2
  266. data/spec/models/spree/shipping_rate_spec.rb +40 -41
  267. data/spec/models/spree/state_spec.rb +12 -1
  268. data/spec/models/spree/stock/content_item_spec.rb +9 -0
  269. data/spec/models/spree/stock/estimator_spec.rb +56 -8
  270. data/spec/models/spree/stock/quantifier_spec.rb +61 -32
  271. data/spec/models/spree/stock_item_spec.rb +19 -1
  272. data/spec/models/spree/store_credit_event_spec.rb +101 -0
  273. data/spec/models/spree/store_credit_spec.rb +786 -0
  274. data/spec/models/spree/store_spec.rb +39 -11
  275. data/spec/models/spree/tax_category_spec.rb +6 -1
  276. data/spec/models/spree/tax_rate_spec.rb +204 -44
  277. data/spec/models/spree/user_spec.rb +105 -38
  278. data/spec/models/spree/variant_spec.rb +281 -9
  279. data/spec/models/spree/zone_member_spec.rb +38 -0
  280. data/spec/models/spree/zone_spec.rb +32 -8
  281. data/spec/spec_helper.rb +3 -0
  282. data/spec/support/concerns/{adjustment_source_spec.rb → adjustment_source.rb} +0 -0
  283. data/spec/support/concerns/{default_price_spec.rb → default_price.rb} +9 -0
  284. data/spec/validators/db_maximum_length_validator_spec.rb +22 -0
  285. data/spree_core.gemspec +5 -6
  286. metadata +99 -36
  287. data/CHANGELOG.md +0 -4
  288. data/app/models/concerns/spree/number_generator.rb +0 -39
  289. data/spec/models/spree/validations/db_maximum_length_validator_spec.rb +0 -24
@@ -4,9 +4,15 @@ require 'spec_helper'
4
4
 
5
5
  describe Spree::Variant, :type => :model do
6
6
  let!(:variant) { create(:variant) }
7
+ let(:master_variant) { create(:master_variant) }
7
8
 
8
9
  it_behaves_like 'default_price'
9
10
 
11
+ describe 'validations' do
12
+ it { expect(master_variant).to_not validate_presence_of(:option_values) }
13
+ it { expect(variant).to validate_presence_of(:option_values) }
14
+ end
15
+
10
16
  context 'sorting' do
11
17
  it 'responds to set_list_position' do
12
18
  expect(variant.respond_to?(:set_list_position)).to eq(true)
@@ -30,7 +36,7 @@ describe Spree::Variant, :type => :model do
30
36
 
31
37
  it "propagate to stock items" do
32
38
  expect_any_instance_of(Spree::StockLocation).to receive(:propagate_variant)
33
- product.variants.create(:name => "Foobar")
39
+ create(:variant, product: product)
34
40
  end
35
41
 
36
42
  context "stock location has disable propagate all variants" do
@@ -51,15 +57,119 @@ describe Spree::Variant, :type => :model do
51
57
  end
52
58
 
53
59
  context 'when a variant is created' do
54
- before(:each) do
55
- product.variants.create!(:name => 'any-name')
56
- end
60
+ let!(:new_variant) { create(:variant, product: product) }
57
61
 
58
62
  it { expect(product.master).to_not be_in_stock }
59
63
  end
60
64
  end
61
65
  end
62
66
 
67
+ describe 'scope' do
68
+ describe '.not_discontinued' do
69
+ context 'when discontinued' do
70
+ let!(:discontinued_variant) { create(:variant, discontinue_on: Time.current - 1.day) }
71
+
72
+ it { expect(Spree::Variant.not_discontinued).not_to include(discontinued_variant) }
73
+ end
74
+
75
+ context 'when not discontinued' do
76
+ let!(:variant_2) { create(:variant, discontinue_on: Time.current + 1.day) }
77
+
78
+ it { expect(Spree::Variant.not_discontinued).to include(variant_2) }
79
+ end
80
+
81
+ context 'when discontinue_on not present' do
82
+ let!(:variant_2) { create(:variant, discontinue_on: nil) }
83
+
84
+ it { expect(Spree::Variant.not_discontinued).to include(variant_2) }
85
+ end
86
+ end
87
+
88
+ describe '.not_deleted' do
89
+ context 'when deleted' do
90
+ let!(:deleted_variant) { create(:variant, deleted_at: Time.current) }
91
+
92
+ it { expect(Spree::Variant.not_deleted).not_to include(deleted_variant) }
93
+ end
94
+
95
+ context 'when not deleted' do
96
+ let!(:variant_2) { create(:variant, deleted_at: nil) }
97
+
98
+ it { expect(Spree::Variant.not_deleted).to include(variant_2) }
99
+ end
100
+ end
101
+
102
+ describe '.for_currency_and_available_price_amount' do
103
+ let(:currency) { 'EUR' }
104
+
105
+ context 'when price with currency present' do
106
+ context 'when price has amount' do
107
+ let!(:price_1) { create(:price, currency: currency, variant: variant, amount: 10) }
108
+
109
+ it { expect(Spree::Variant.for_currency_and_available_price_amount(currency)).to include(variant) }
110
+ end
111
+
112
+ context 'when price do not have amount' do
113
+ let!(:price_1) { create(:price, currency: currency, variant: variant, amount: nil) }
114
+
115
+ it { expect(Spree::Variant.for_currency_and_available_price_amount(currency)).not_to include(variant) }
116
+ end
117
+ end
118
+
119
+ context 'when price with currency not present' do
120
+ let!(:unavailable_currency) { 'INR' }
121
+ context 'when price has amount' do
122
+ let!(:price_1) { create(:price, currency: unavailable_currency, variant: variant, amount: 10) }
123
+
124
+ it { expect(Spree::Variant.for_currency_and_available_price_amount(currency)).not_to include(variant) }
125
+ end
126
+
127
+ context 'when price do not have amount' do
128
+ let!(:price_1) { create(:price, currency: unavailable_currency, variant: variant, amount: nil) }
129
+
130
+ it { expect(Spree::Variant.for_currency_and_available_price_amount(currency)).not_to include(variant) }
131
+ end
132
+ end
133
+
134
+ context 'when multiple prices for same currency present' do
135
+ let!(:price_1) { create(:price, currency: currency, variant: variant) }
136
+ let!(:price_2) { create(:price, currency: currency, variant: variant) }
137
+
138
+ it 'should not duplicate variant' do
139
+ expect(Spree::Variant.for_currency_and_available_price_amount(currency)).to eq([variant])
140
+ end
141
+ end
142
+ end
143
+
144
+ describe '.active' do
145
+ let!(:variants) { [variant] }
146
+ let!(:currency) { 'EUR' }
147
+
148
+ before(:each) do
149
+ allow(Spree::Variant).to receive(:not_discontinued).and_return(variants)
150
+ allow(variants).to receive(:not_deleted).and_return(variants)
151
+ allow(variants).to receive(:for_currency_and_available_price_amount).with(currency).and_return(variants)
152
+ end
153
+
154
+ it 'should find not_discontinued variants' do
155
+ expect(Spree::Variant).to receive(:not_discontinued).and_return(variants)
156
+ Spree::Variant.active(currency)
157
+ end
158
+
159
+ it 'should find not_deleted variants' do
160
+ expect(variants).to receive(:not_deleted).and_return(variants)
161
+ Spree::Variant.active(currency)
162
+ end
163
+
164
+ it 'should find variants for_currency_and_available_price_amount' do
165
+ expect(variants).to receive(:for_currency_and_available_price_amount).with(currency).and_return(variants)
166
+ Spree::Variant.active(currency)
167
+ end
168
+
169
+ it { expect(Spree::Variant.active(currency)).to eq(variants) }
170
+ end
171
+ end
172
+
63
173
  context "product has other variants" do
64
174
  describe "option value accessors" do
65
175
  before {
@@ -235,13 +345,14 @@ describe Spree::Variant, :type => :model do
235
345
 
236
346
  # Regression test for #2432
237
347
  describe 'options_text' do
238
- let!(:variant) { create(:variant, option_values: []) }
348
+ let!(:variant) { build(:variant, option_values: []) }
239
349
  let!(:master) { create(:master_variant) }
240
350
 
241
351
  before do
242
352
  # Order bar than foo
243
353
  variant.option_values << create(:option_value, {name: 'Foo', presentation: 'Foo', option_type: create(:option_type, position: 2, name: 'Foo Type', presentation: 'Foo Type')})
244
354
  variant.option_values << create(:option_value, {name: 'Bar', presentation: 'Bar', option_type: create(:option_type, position: 1, name: 'Bar Type', presentation: 'Bar Type')})
355
+ variant.save
245
356
  end
246
357
 
247
358
  it 'should order by bar than foo' do
@@ -251,7 +362,7 @@ describe Spree::Variant, :type => :model do
251
362
  end
252
363
 
253
364
  describe 'exchange_name' do
254
- let!(:variant) { create(:variant, option_values: []) }
365
+ let!(:variant) { build(:variant, option_values: []) }
255
366
  let!(:master) { create(:master_variant) }
256
367
 
257
368
  before do
@@ -260,6 +371,7 @@ describe Spree::Variant, :type => :model do
260
371
  presentation: 'Foo',
261
372
  option_type: create(:option_type, position: 2, name: 'Foo Type', presentation: 'Foo Type')
262
373
  })
374
+ variant.save
263
375
  end
264
376
 
265
377
  context 'master variant' do
@@ -277,7 +389,7 @@ describe Spree::Variant, :type => :model do
277
389
  end
278
390
 
279
391
  describe 'exchange_name' do
280
- let!(:variant) { create(:variant, option_values: []) }
392
+ let!(:variant) { build(:variant, option_values: []) }
281
393
  let!(:master) { create(:master_variant) }
282
394
 
283
395
  before do
@@ -286,6 +398,7 @@ describe Spree::Variant, :type => :model do
286
398
  presentation: 'Foo',
287
399
  option_type: create(:option_type, position: 2, name: 'Foo Type', presentation: 'Foo Type')
288
400
  })
401
+ variant.save
289
402
  end
290
403
 
291
404
  context 'master variant' do
@@ -303,7 +416,7 @@ describe Spree::Variant, :type => :model do
303
416
  end
304
417
 
305
418
  describe 'descriptive_name' do
306
- let!(:variant) { create(:variant, option_values: []) }
419
+ let!(:variant) { build(:variant, option_values: []) }
307
420
  let!(:master) { create(:master_variant) }
308
421
 
309
422
  before do
@@ -312,6 +425,7 @@ describe Spree::Variant, :type => :model do
312
425
  presentation: 'Foo',
313
426
  option_type: create(:option_type, position: 2, name: 'Foo Type', presentation: 'Foo Type')
314
427
  })
428
+ variant.save
315
429
  end
316
430
 
317
431
  context 'master variant' do
@@ -441,7 +555,7 @@ describe Spree::Variant, :type => :model do
441
555
  it "updates a product" do
442
556
  variant.product.update_column(:updated_at, 1.day.ago)
443
557
  variant.touch
444
- expect(variant.product.reload.updated_at).to be_within(3.seconds).of(Time.now)
558
+ expect(variant.product.reload.updated_at).to be_within(3.seconds).of(Time.current)
445
559
  end
446
560
 
447
561
  it "clears the in_stock cache key" do
@@ -520,4 +634,162 @@ describe Spree::Variant, :type => :model do
520
634
  expect(variant.dimension).to eq (dimension_expected)
521
635
  end
522
636
  end
637
+
638
+ context "#discontinue!" do
639
+ let(:variant) { create(:variant) }
640
+
641
+ it "sets the discontinued" do
642
+ variant.discontinue!
643
+ variant.reload
644
+ expect(variant.discontinued?).to be(true)
645
+ end
646
+
647
+ it "changes updated_at" do
648
+ expect { variant.discontinue! }.to change { variant.updated_at }
649
+ end
650
+ end
651
+
652
+ context "#discontinued?" do
653
+ let(:variant_live) { build(:variant) }
654
+ it "should be false" do
655
+ expect(variant_live.discontinued?).to be(false)
656
+ end
657
+
658
+ let(:variant_discontinued) { build(:variant, discontinue_on: Time.now - 1.day) }
659
+ it "should be true" do
660
+ expect(variant_discontinued.discontinued?).to be(true)
661
+ end
662
+ end
663
+
664
+ describe "#available?" do
665
+ let(:variant) { create(:variant) }
666
+ context 'when discontinued' do
667
+ before(:each) do
668
+ variant.discontinue_on = Time.current - 1.day
669
+ end
670
+
671
+ context 'when product is available' do
672
+ before(:each) do
673
+ allow(variant.product).to receive(:available?) { true }
674
+ end
675
+
676
+ it { expect(variant.available?).to be(false) }
677
+ end
678
+
679
+ context 'when product is not available' do
680
+ before(:each) do
681
+ allow(variant.product).to receive(:available?) { false }
682
+ end
683
+
684
+ it { expect(variant.available?).to be(false) }
685
+ end
686
+ end
687
+
688
+ context 'when not discontinued' do
689
+ before(:each) do
690
+ variant.discontinue_on = Time.current + 1.day
691
+ end
692
+
693
+ context 'when product is available' do
694
+ before(:each) do
695
+ allow(variant.product).to receive(:available?) { true }
696
+ end
697
+
698
+ it { expect(variant.available?).to be(true) }
699
+ end
700
+
701
+ context 'when product is not available' do
702
+ before(:each) do
703
+ allow(variant.product).to receive(:available?) { false }
704
+ end
705
+
706
+ it { expect(variant.available?).to be(false) }
707
+ end
708
+ end
709
+ end
710
+
711
+ describe "#check_price" do
712
+ let(:variant) { create(:variant) }
713
+ let(:variant2) { create(:variant) }
714
+
715
+ context 'require_master_price set false' do
716
+ before { Spree::Config.set(require_master_price: false) }
717
+
718
+ context 'price present and currency present' do
719
+ it { expect(variant.send(:check_price)).to be(nil) }
720
+ end
721
+
722
+ context 'price present and currency nil' do
723
+ before { variant.currency = nil }
724
+
725
+ it { expect(variant.send(:check_price)).to be(Spree::Config[:currency]) }
726
+ end
727
+
728
+ context 'price nil and currency present' do
729
+ before { variant.price = nil }
730
+
731
+ it { expect(variant.send(:check_price)).to be(nil) }
732
+ end
733
+
734
+ context 'price nil and currency nil' do
735
+ before { variant.price = nil }
736
+
737
+ it { expect(variant.send(:check_price)).to be(nil) }
738
+ end
739
+ end
740
+
741
+ context 'require_master_price set true' do
742
+ before { Spree::Config.set(require_master_price: true) }
743
+
744
+ context 'price present and currency present' do
745
+ it { expect(variant.send(:check_price)).to be(nil) }
746
+ end
747
+
748
+ context 'price present and currency nil' do
749
+ before { variant.currency = nil }
750
+
751
+ it { expect(variant.send(:check_price)).to be(Spree::Config[:currency]) }
752
+ end
753
+
754
+ context 'product and master_variant present and equal' do
755
+ context 'price nil and currency present' do
756
+ before { variant.price = nil }
757
+ it { expect(variant.send(:check_price)).to be(nil) }
758
+
759
+ context 'check variant price' do
760
+ before { variant.send(:check_price) }
761
+ it { expect(variant.price).to eq(variant.product.master.price) }
762
+ end
763
+ end
764
+
765
+ context 'price nil and currency nil' do
766
+ before do
767
+ variant.price = nil
768
+ variant.send(:check_price)
769
+ end
770
+
771
+ it { expect(variant.price).to eq(variant.product.master.price) }
772
+ it { expect(variant.currency).to eq(Spree::Config[:currency]) }
773
+ end
774
+ end
775
+
776
+ context 'product not present' do
777
+ context 'product not present' do
778
+ before { variant.product = nil }
779
+
780
+ context 'price nil and currency present' do
781
+ before { variant.price = nil }
782
+
783
+ it { expect { variant.send(:check_price) }.to raise_error(RuntimeError, 'No master variant found to infer price') }
784
+ end
785
+
786
+ context 'price nil and currency nil' do
787
+ before { variant.price = nil }
788
+
789
+ it { expect { variant.send(:check_price) }.to raise_error(RuntimeError, 'No master variant found to infer price') }
790
+ end
791
+ end
792
+ end
793
+ end
794
+ end
523
795
  end
@@ -0,0 +1,38 @@
1
+ require 'spec_helper'
2
+
3
+ describe Spree::ZoneMember, type: :model do
4
+ let(:country) { create(:country) }
5
+ let(:state) { create(:state) }
6
+ let(:zone) { create(:zone, kind: 'country') }
7
+ let(:zone_member) { create(:zone_member, zone: zone, zoneable: country) }
8
+
9
+ describe 'associations' do
10
+ it { is_expected.to belong_to(:zoneable) }
11
+ it { is_expected.to belong_to(:zone).counter_cache(true).inverse_of(:zone_members) }
12
+ end
13
+
14
+ describe 'validations' do
15
+ it { is_expected.to validate_presence_of(:zone) }
16
+ it { is_expected.to validate_presence_of(:zoneable) }
17
+ end
18
+
19
+ describe 'scopes' do
20
+ describe '.defunct_without_kind' do
21
+ let(:defunct_without_kind) { Spree::ZoneMember.defunct_without_kind('country') }
22
+
23
+ context 'zoneable is present and is of defunct kind' do
24
+ it { expect(defunct_without_kind).to_not include(zone_member) }
25
+ end
26
+
27
+ context 'zoneable is not of defunct kind' do
28
+ before { zone_member.update(zoneable: state) }
29
+ it { expect(defunct_without_kind).to include(zone_member) }
30
+ end
31
+
32
+ context 'zoneable is absent' do
33
+ before { zone_member.update_column(:zoneable_id, nil) }
34
+ it { expect(defunct_without_kind).to include(zone_member) }
35
+ end
36
+ end
37
+ end
38
+ end
@@ -14,6 +14,38 @@ describe Spree::Zone, :type => :model do
14
14
 
15
15
  before { country_zone.members.create(zoneable: country) }
16
16
 
17
+ describe 'scopes' do
18
+ describe '.remove_previous_default' do
19
+ let(:zone_with_default_tax) { create(:zone, kind: 'country', default_tax: true) }
20
+ let(:zone_not_with_default_tax) { create(:zone, kind: 'country', default_tax: false) }
21
+
22
+ subject { Spree::Zone.with_default_tax }
23
+
24
+ it 'is expected to include zone with default tax' do
25
+ is_expected.to include(zone_with_default_tax)
26
+ end
27
+
28
+ it 'is expected to not include zone with default tax' do
29
+ is_expected.to_not include(zone_not_with_default_tax)
30
+ end
31
+ end
32
+ end
33
+
34
+ describe 'callbacks' do
35
+ it { is_expected.to callback(:remove_previous_default).after(:save).if(:default_tax?).if(:default_tax_changed?) }
36
+
37
+ describe '#remove_previous_default' do
38
+ let!(:zone_with_default_tax) { create(:zone, kind: 'country', default_tax: true) }
39
+ let!(:zone_not_with_default_tax) { create(:zone, kind: 'country', default_tax: false) }
40
+
41
+ it 'is expected to make previous default tax zones to non default tax zones' do
42
+ expect(zone_with_default_tax).to be_default_tax
43
+ zone_not_with_default_tax.update(default_tax: true)
44
+ expect(zone_with_default_tax.reload).to_not be_default_tax
45
+ end
46
+ end
47
+ end
48
+
17
49
  context "when there is only one qualifying zone" do
18
50
  let(:address) { create(:address, country: country, state: state) }
19
51
 
@@ -373,10 +405,6 @@ describe Spree::Zone, :type => :model do
373
405
  it "only returns each zone once" do
374
406
  expect(@result.select { |z| z == zone }.size).to be 1
375
407
  end
376
-
377
- it "will include the default_tax zone" do
378
- expect(@result).to include(default_tax_zone)
379
- end
380
408
  end
381
409
 
382
410
  context "finding potential matches for a state zone" do
@@ -410,10 +438,6 @@ describe Spree::Zone, :type => :model do
410
438
  it "only returns each zone once" do
411
439
  expect(@result.select { |z| z == zone }.size).to be 1
412
440
  end
413
-
414
- it "will include the default tax zone" do
415
- expect(@result).to include(default_tax_zone)
416
- end
417
441
  end
418
442
  end
419
443