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
@@ -0,0 +1,211 @@
1
+ require 'spec_helper'
2
+
3
+ describe Spree::Adjustable::Adjuster::Promotion, type: :model do
4
+ let(:order) { create :order_with_line_items, line_items_count: 1 }
5
+ let(:line_item) { order.line_items.first }
6
+ let(:subject) { Spree::Adjustable::AdjustmentsUpdater.new(line_item) }
7
+ let(:order_subject) { Spree::Adjustable::AdjustmentsUpdater.new(order) }
8
+
9
+ context "best promotion is always applied" do
10
+ let(:calculator) { Spree::Calculator::FlatRate.new(preferred_amount: 10) }
11
+ let(:source) { Spree::Promotion::Actions::CreateItemAdjustments.create calculator: calculator }
12
+
13
+ def create_adjustment(label, amount)
14
+ create(:adjustment,
15
+ order: order,
16
+ adjustable: line_item,
17
+ source: source,
18
+ amount: amount,
19
+ state: "closed",
20
+ label: label,
21
+ mandatory: false)
22
+ end
23
+
24
+ describe "competing promos" do
25
+ before { Spree::Adjustment.competing_promos_source_types = ['Spree::PromotionAction', 'Custom'] }
26
+
27
+ it "do not update promo_total" do
28
+ create(:adjustment,
29
+ order: order,
30
+ adjustable: line_item,
31
+ source_type: 'Custom',
32
+ source_id: nil,
33
+ amount: -3.50,
34
+ label: "Other",
35
+ mandatory: false)
36
+ create_adjustment("Promotion A", -2.50)
37
+
38
+ subject.update
39
+ expect(line_item.promo_total).to eql 0.0
40
+ end
41
+ end
42
+
43
+ it "should use only the most valuable promotion" do
44
+ create_adjustment("Promotion A", -100)
45
+ create_adjustment("Promotion B", -200)
46
+ create_adjustment("Promotion C", -300)
47
+ create(:adjustment,
48
+ order: order,
49
+ adjustable: line_item,
50
+ source: nil,
51
+ amount: -500,
52
+ state: "closed",
53
+ label: "Some other credit")
54
+ line_item.adjustments.each { |a| a.update_column(:eligible, true) }
55
+
56
+ subject.update
57
+
58
+ expect(line_item.adjustments.promotion.eligible.count).to eq(1)
59
+ expect(line_item.adjustments.promotion.eligible.first.label).to eq('Promotion C')
60
+ end
61
+
62
+ it "should choose the most recent promotion adjustment when amounts are equal" do
63
+ # Using Timecop is a regression test
64
+ Timecop.freeze do
65
+ create_adjustment("Promotion A", -200)
66
+ create_adjustment("Promotion B", -200)
67
+ end
68
+ line_item.adjustments.each { |a| a.update_column(:eligible, true) }
69
+
70
+ subject.update
71
+
72
+ expect(line_item.adjustments.promotion.eligible.count).to eq(1)
73
+ expect(line_item.adjustments.promotion.eligible.first.label).to eq('Promotion B')
74
+ end
75
+
76
+ context "when previously ineligible promotions become available" do
77
+ let(:order_promo1) do
78
+ create(:promotion,
79
+ :with_order_adjustment,
80
+ :with_item_total_rule,
81
+ weighted_order_adjustment_amount: 5,
82
+ item_total_threshold_amount: 10)
83
+ end
84
+
85
+ let(:order_promo2) do
86
+ create(:promotion,
87
+ :with_order_adjustment,
88
+ :with_item_total_rule,
89
+ weighted_order_adjustment_amount: 10,
90
+ item_total_threshold_amount: 20)
91
+ end
92
+
93
+ let(:order_promos) { [order_promo1, order_promo2] }
94
+
95
+ let(:line_item_promo1) do
96
+ create(:promotion,
97
+ :with_line_item_adjustment,
98
+ :with_item_total_rule,
99
+ adjustment_rate: 2.5,
100
+ item_total_threshold_amount: 10)
101
+ end
102
+
103
+ let(:line_item_promo2) do
104
+ create(:promotion,
105
+ :with_line_item_adjustment,
106
+ :with_item_total_rule,
107
+ adjustment_rate: 5,
108
+ item_total_threshold_amount: 20)
109
+ end
110
+
111
+ let(:line_item_promos) { [line_item_promo1, line_item_promo2] }
112
+ let(:order) { create(:order_with_line_items, line_items_count: 1) }
113
+
114
+ # Apply promotions in different sequences. Results should be the same.
115
+ promo_sequences = [[0, 1], [1, 0]]
116
+
117
+ promo_sequences.each do |promo_sequence|
118
+ it "should pick the best order-level promo according to current eligibility" do
119
+ # apply both promos to the order, even though only promo1 is eligible
120
+ order_promos[promo_sequence[0]].activate order: order
121
+ order_promos[promo_sequence[1]].activate order: order
122
+
123
+ order.reload
124
+ msg = "Expected two adjustments (using sequence #{promo_sequence})"
125
+ expect(order.all_adjustments.count).to eq(2), msg
126
+
127
+ msg = "Expected one elegible adjustment (using sequence #{promo_sequence})"
128
+ expect(order.all_adjustments.eligible.count).to eq(1), msg
129
+
130
+ msg = "Expected promo1 to be used (using sequence #{promo_sequence})"
131
+ expect(order.all_adjustments.eligible.first.source.promotion).to eq(order_promo1), msg
132
+
133
+ order.contents.add create(:variant, price: 10), 1
134
+ order.save
135
+
136
+ order.reload
137
+ msg = "Expected two adjustments (using sequence #{promo_sequence})"
138
+ expect(order.all_adjustments.count).to eq(2), msg
139
+
140
+ msg = "Expected one elegible adjustment (using sequence #{promo_sequence})"
141
+ expect(order.all_adjustments.eligible.count).to eq(1), msg
142
+
143
+ msg = "Expected promo2 to be used (using sequence #{promo_sequence})"
144
+ expect(order.all_adjustments.eligible.first.source.promotion).to eq(order_promo2), msg
145
+ end
146
+ end
147
+
148
+ promo_sequences.each do |promo_sequence|
149
+ it "should pick the best line-item-level promo according to current eligibility" do
150
+ # apply both promos to the order, even though only promo1 is eligible
151
+ line_item_promos[promo_sequence[0]].activate order: order
152
+ line_item_promos[promo_sequence[1]].activate order: order
153
+
154
+ order.reload
155
+ msg = "Expected one adjustment (using sequence #{promo_sequence})"
156
+ expect(order.all_adjustments.count).to eq(1), msg
157
+
158
+ msg = "Expected one elegible adjustment (using sequence #{promo_sequence})"
159
+ expect(order.all_adjustments.eligible.count).to eq(1), msg
160
+
161
+ # line_item_promo1 is the only one that has thus far met the order total threshold,
162
+ # it is the only promo which should be applied.
163
+ msg = "Expected line_item_promo1 to be used (using sequence #{promo_sequence})"
164
+ expect(order.all_adjustments.first.source.promotion).to eq(line_item_promo1), msg
165
+
166
+ order.contents.add create(:variant, price: 10), 1
167
+ order.save
168
+
169
+ order.reload
170
+ msg = "Expected four adjustments (using sequence #{promo_sequence})"
171
+ expect(order.all_adjustments.count).to eq(4), msg
172
+
173
+ msg = "Expected two elegible adjustments (using sequence #{promo_sequence})"
174
+ expect(order.all_adjustments.eligible.count).to eq(2), msg
175
+
176
+ order.all_adjustments.eligible.each do |adjustment|
177
+ msg = "Expected line_item_promo2 to be used (using sequence #{promo_sequence})"
178
+ expect(adjustment.source.promotion).to eq(line_item_promo2), msg
179
+ end
180
+ end
181
+ end
182
+ end
183
+
184
+ context "multiple adjustments and the best one is not eligible" do
185
+ let!(:promo_a) { create_adjustment("Promotion A", -100) }
186
+ let!(:promo_c) { create_adjustment("Promotion C", -300) }
187
+
188
+ before do
189
+ promo_a.update_column(:eligible, true)
190
+ promo_c.update_column(:eligible, false)
191
+ end
192
+
193
+ # regression for #3274
194
+ it "still makes the previous best eligible adjustment valid" do
195
+ subject.update
196
+ expect(line_item.adjustments.promotion.eligible.first.label).to eq('Promotion A')
197
+ end
198
+ end
199
+
200
+ it "should only leave one adjustment even if 2 have the same amount" do
201
+ create_adjustment("Promotion A", -100)
202
+ create_adjustment("Promotion B", -200)
203
+ create_adjustment("Promotion C", -200)
204
+
205
+ subject.update
206
+
207
+ expect(line_item.adjustments.promotion.eligible.count).to eq(1)
208
+ expect(line_item.adjustments.promotion.eligible.first.amount.to_i).to eq(-200)
209
+ end
210
+ end
211
+ end
@@ -0,0 +1,86 @@
1
+ require 'spec_helper'
2
+
3
+ describe Spree::Adjustable::Adjuster::Tax, type: :model do
4
+ let(:order) { create :order_with_line_items, line_items_count: 1 }
5
+ let(:line_item) { order.line_items.first }
6
+
7
+ let(:subject) { Spree::Adjustable::AdjustmentsUpdater.new(line_item) }
8
+ let(:order_subject) { Spree::Adjustable::AdjustmentsUpdater.new(order) }
9
+
10
+ context "taxes with promotions" do
11
+ let!(:tax_rate) do
12
+ create(:tax_rate, amount: 0.05)
13
+ end
14
+
15
+ let!(:promotion) do
16
+ Spree::Promotion.create(name: "$10 off")
17
+ end
18
+
19
+ let!(:promotion_action) do
20
+ calculator = Spree::Calculator::FlatRate.new(preferred_amount: 10)
21
+ Spree::Promotion::Actions::CreateItemAdjustments.create(calculator: calculator,
22
+ promotion: promotion)
23
+ end
24
+
25
+ before do
26
+ line_item.price = 20
27
+ line_item.tax_category = tax_rate.tax_category
28
+ line_item.save
29
+ create(:adjustment, order: order, source: promotion_action, adjustable: line_item)
30
+ end
31
+
32
+ context "tax included in price" do
33
+ before do
34
+ create(:adjustment,
35
+ source: tax_rate,
36
+ adjustable: line_item,
37
+ order: order,
38
+ included: true)
39
+ end
40
+
41
+ it "tax has no bearing on final price" do
42
+ subject.update
43
+ line_item.reload
44
+ expect(line_item.included_tax_total).to eq(0.5)
45
+ expect(line_item.additional_tax_total).to eq(0)
46
+ expect(line_item.promo_total).to eq(-10)
47
+ expect(line_item.adjustment_total).to eq(-10)
48
+ end
49
+
50
+ it "tax linked to order" do
51
+ order_subject.update
52
+ order.reload
53
+ expect(order.included_tax_total).to eq(0.5)
54
+ expect(order.additional_tax_total).to eq(00)
55
+ end
56
+ end
57
+
58
+ context "tax excluded from price" do
59
+ before do
60
+ create(:adjustment,
61
+ source: tax_rate,
62
+ adjustable: line_item,
63
+ order: order,
64
+ included: false)
65
+ end
66
+
67
+ it "tax applies to line item" do
68
+ subject.update
69
+ line_item.reload
70
+ # Taxable amount is: $20 (base) - $10 (promotion) = $10
71
+ # Tax rate is 5% (of $10).
72
+ expect(line_item.included_tax_total).to eq(0)
73
+ expect(line_item.additional_tax_total).to eq(0.5)
74
+ expect(line_item.promo_total).to eq(-10)
75
+ expect(line_item.adjustment_total).to eq(-9.5)
76
+ end
77
+
78
+ it "tax linked to order" do
79
+ order_subject.update
80
+ order.reload
81
+ expect(order.included_tax_total).to eq(0)
82
+ expect(order.additional_tax_total).to eq(0.5)
83
+ end
84
+ end
85
+ end
86
+ end
@@ -9,8 +9,8 @@ module Spree
9
9
  let(:subject) { AdjustmentsUpdater.new(line_item) }
10
10
  let(:order_subject) { AdjustmentsUpdater.new(order) }
11
11
 
12
- context '#update' do
13
- it "updates a linked adjustment" do
12
+ describe '#update' do
13
+ it "updates all linked adjusters" do
14
14
  tax_rate = create(:tax_rate, amount: 0.05)
15
15
  create(:adjustment, order: order, source: tax_rate, adjustable: line_item)
16
16
  line_item.price = 10
@@ -21,266 +21,6 @@ module Spree
21
21
  expect(line_item.additional_tax_total).to eq(0.5)
22
22
  end
23
23
  end
24
-
25
- context "taxes and promotions" do
26
- let!(:tax_rate) do
27
- create(:tax_rate, amount: 0.05)
28
- end
29
-
30
- let!(:promotion) do
31
- Spree::Promotion.create(name: "$10 off")
32
- end
33
-
34
- let!(:promotion_action) do
35
- calculator = Calculator::FlatRate.new(preferred_amount: 10)
36
- Promotion::Actions::CreateItemAdjustments.create(calculator: calculator,
37
- promotion: promotion)
38
- end
39
-
40
- before do
41
- line_item.price = 20
42
- line_item.tax_category = tax_rate.tax_category
43
- line_item.save
44
- create(:adjustment, order: order, source: promotion_action, adjustable: line_item)
45
- end
46
-
47
- context "tax included in price" do
48
- before do
49
- create(:adjustment, source: tax_rate,
50
- adjustable: line_item,
51
- order: order,
52
- included: true)
53
- end
54
-
55
- it "tax has no bearing on final price" do
56
- subject.update
57
- line_item.reload
58
- expect(line_item.included_tax_total).to eq(0.5)
59
- expect(line_item.additional_tax_total).to eq(0)
60
- expect(line_item.promo_total).to eq(-10)
61
- expect(line_item.adjustment_total).to eq(-10)
62
- end
63
-
64
- it "tax linked to order" do
65
- order_subject.update
66
- order.reload
67
- expect(order.included_tax_total).to eq(0.5)
68
- expect(order.additional_tax_total).to eq(00)
69
- end
70
- end
71
-
72
- context "tax excluded from price" do
73
- before do
74
- create(:adjustment, source: tax_rate,
75
- adjustable: line_item,
76
- order: order,
77
- included: false)
78
- end
79
-
80
- it "tax applies to line item" do
81
- subject.update
82
- line_item.reload
83
- # Taxable amount is: $20 (base) - $10 (promotion) = $10
84
- # Tax rate is 5% (of $10).
85
- expect(line_item.included_tax_total).to eq(0)
86
- expect(line_item.additional_tax_total).to eq(0.5)
87
- expect(line_item.promo_total).to eq(-10)
88
- expect(line_item.adjustment_total).to eq(-9.5)
89
- end
90
-
91
- it "tax linked to order" do
92
- order_subject.update
93
- order.reload
94
- expect(order.included_tax_total).to eq(0)
95
- expect(order.additional_tax_total).to eq(0.5)
96
- end
97
- end
98
- end
99
-
100
- context "best promotion is always applied" do
101
- let(:calculator) { Calculator::FlatRate.new(preferred_amount: 10) }
102
-
103
- let(:source) { Promotion::Actions::CreateItemAdjustments.create calculator: calculator }
104
-
105
- def create_adjustment(label, amount)
106
- create(:adjustment, order: order,
107
- adjustable: line_item,
108
- source: source,
109
- amount: amount,
110
- state: "closed",
111
- label: label,
112
- mandatory: false)
113
- end
114
-
115
- it "should make all but the most valuable promotion adjustment ineligible, " +
116
- "leaving non promotion adjustments alone" do
117
- create_adjustment("Promotion A", -100)
118
- create_adjustment("Promotion B", -200)
119
- create_adjustment("Promotion C", -300)
120
- create(:adjustment, order: order,
121
- adjustable: line_item,
122
- source: nil,
123
- amount: -500,
124
- state: "closed",
125
- label: "Some other credit")
126
- line_item.adjustments.each { |a| a.update_column(:eligible, true) }
127
-
128
- subject.update
129
-
130
- expect(line_item.adjustments.promotion.eligible.count).to eq(1)
131
- expect(line_item.adjustments.promotion.eligible.first.label).to eq('Promotion C')
132
- end
133
-
134
- it "should choose the most recent promotion adjustment when amounts are equal" do
135
- # Using Timecop is a regression test
136
- Timecop.freeze do
137
- create_adjustment("Promotion A", -200)
138
- create_adjustment("Promotion B", -200)
139
- end
140
- line_item.adjustments.each { |a| a.update_column(:eligible, true) }
141
-
142
- subject.update
143
-
144
- expect(line_item.adjustments.promotion.eligible.count).to eq(1)
145
- expect(line_item.adjustments.promotion.eligible.first.label).to eq('Promotion B')
146
- end
147
-
148
- context "when previously ineligible promotions become available" do
149
- let(:order_promo1) do
150
- create(:promotion,
151
- :with_order_adjustment,
152
- :with_item_total_rule,
153
- weighted_order_adjustment_amount: 5,
154
- item_total_threshold_amount: 10)
155
- end
156
-
157
- let(:order_promo2) do
158
- create(:promotion,
159
- :with_order_adjustment,
160
- :with_item_total_rule,
161
- weighted_order_adjustment_amount: 10,
162
- item_total_threshold_amount: 20)
163
- end
164
-
165
- let(:order_promos) { [order_promo1, order_promo2] }
166
-
167
- let(:line_item_promo1) do
168
- create(:promotion,
169
- :with_line_item_adjustment,
170
- :with_item_total_rule,
171
- adjustment_rate: 2.5,
172
- item_total_threshold_amount: 10)
173
- end
174
-
175
- let(:line_item_promo2) do
176
- create(:promotion,
177
- :with_line_item_adjustment,
178
- :with_item_total_rule,
179
- adjustment_rate: 5,
180
- item_total_threshold_amount: 20)
181
- end
182
-
183
- let(:line_item_promos) { [line_item_promo1, line_item_promo2] }
184
- let(:order) { create(:order_with_line_items, line_items_count: 1) }
185
-
186
- # Apply promotions in different sequences. Results should be the same.
187
- promo_sequences = [[0, 1], [1, 0]]
188
-
189
- promo_sequences.each do |promo_sequence|
190
- it "should pick the best order-level promo according to current eligibility" do
191
- # apply both promos to the order, even though only promo1 is eligible
192
- order_promos[promo_sequence[0]].activate order: order
193
- order_promos[promo_sequence[1]].activate order: order
194
-
195
- order.reload
196
- msg = "Expected two adjustments (using sequence #{promo_sequence})"
197
- expect(order.all_adjustments.count).to eq(2), msg
198
-
199
- msg = "Expected one elegible adjustment (using sequence #{promo_sequence})"
200
- expect(order.all_adjustments.eligible.count).to eq(1), msg
201
-
202
- msg = "Expected promo1 to be used (using sequence #{promo_sequence})"
203
- expect(order.all_adjustments.eligible.first.source.promotion).to eq(order_promo1), msg
204
-
205
- order.contents.add create(:variant, price: 10), 1
206
- order.save
207
-
208
- order.reload
209
- msg = "Expected two adjustments (using sequence #{promo_sequence})"
210
- expect(order.all_adjustments.count).to eq(2), msg
211
-
212
- msg = "Expected one elegible adjustment (using sequence #{promo_sequence})"
213
- expect(order.all_adjustments.eligible.count).to eq(1), msg
214
-
215
- msg = "Expected promo2 to be used (using sequence #{promo_sequence})"
216
- expect(order.all_adjustments.eligible.first.source.promotion).to eq(order_promo2), msg
217
- end
218
- end
219
-
220
- promo_sequences.each do |promo_sequence|
221
- it "should pick the best line-item-level promo according to current eligibility" do
222
- # apply both promos to the order, even though only promo1 is eligible
223
- line_item_promos[promo_sequence[0]].activate order: order
224
- line_item_promos[promo_sequence[1]].activate order: order
225
-
226
- order.reload
227
- msg = "Expected one adjustment (using sequence #{promo_sequence})"
228
- expect(order.all_adjustments.count).to eq(1), msg
229
-
230
- msg = "Expected one elegible adjustment (using sequence #{promo_sequence})"
231
- expect(order.all_adjustments.eligible.count).to eq(1), msg
232
-
233
- # line_item_promo1 is the only one that has thus far met the order total threshold,
234
- # it is the only promo which should be applied.
235
- msg = "Expected line_item_promo1 to be used (using sequence #{promo_sequence})"
236
- expect(order.all_adjustments.first.source.promotion).to eq(line_item_promo1), msg
237
-
238
- order.contents.add create(:variant, price: 10), 1
239
- order.save
240
-
241
- order.reload
242
- msg = "Expected four adjustments (using sequence #{promo_sequence})"
243
- expect(order.all_adjustments.count).to eq(4), msg
244
-
245
- msg = "Expected two elegible adjustments (using sequence #{promo_sequence})"
246
- expect(order.all_adjustments.eligible.count).to eq(2), msg
247
-
248
- order.all_adjustments.eligible.each do |adjustment|
249
- msg = "Expected line_item_promo2 to be used (using sequence #{promo_sequence})"
250
- expect(adjustment.source.promotion).to eq(line_item_promo2), msg
251
- end
252
- end
253
- end
254
- end
255
-
256
- context "multiple adjustments and the best one is not eligible" do
257
- let!(:promo_a) { create_adjustment("Promotion A", -100) }
258
- let!(:promo_c) { create_adjustment("Promotion C", -300) }
259
-
260
- before do
261
- promo_a.update_column(:eligible, true)
262
- promo_c.update_column(:eligible, false)
263
- end
264
-
265
- # regression for #3274
266
- it "still makes the previous best eligible adjustment valid" do
267
- subject.update
268
- expect(line_item.adjustments.promotion.eligible.first.label).to eq('Promotion A')
269
- end
270
- end
271
-
272
- it "should only leave one adjustment even if 2 have the same amount" do
273
- create_adjustment("Promotion A", -100)
274
- create_adjustment("Promotion B", -200)
275
- create_adjustment("Promotion C", -200)
276
-
277
- subject.update
278
-
279
- expect(line_item.adjustments.promotion.eligible.count).to eq(1)
280
- expect(line_item.adjustments.promotion.eligible.first.amount.to_i).to eq(-200)
281
- end
282
- end
283
-
284
24
  end
285
25
  end
286
26
  end