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
@@ -19,45 +19,60 @@ module Spree
19
19
  subject { described_class.new(stock_item.variant) }
20
20
 
21
21
  specify { expect(subject.stock_items).to eq([stock_item]) }
22
+ specify { expect(subject.variant).to eq(stock_item.variant) }
22
23
 
23
24
  context 'with a single stock location/item' do
24
25
  it 'total_on_hand should match stock_item' do
25
26
  expect(subject.total_on_hand).to eq(stock_item.count_on_hand)
26
27
  end
27
28
 
28
- context 'when track_inventory_levels is false' do
29
- before { configure_spree_preferences { |config| config.track_inventory_levels = false } }
29
+ context 'when variant is available' do
30
+ before(:each) do
31
+ allow(subject.variant).to receive(:available?) { true }
32
+ end
30
33
 
31
- specify { expect(subject.total_on_hand).to eq(Float::INFINITY) }
34
+ context 'when track_inventory_levels is false' do
35
+ before { configure_spree_preferences { |config| config.track_inventory_levels = false } }
32
36
 
33
- it_should_behave_like 'unlimited supply'
34
- end
37
+ specify { expect(subject.total_on_hand).to eq(Float::INFINITY) }
35
38
 
36
- context 'when variant inventory tracking is off' do
37
- before { stock_item.variant.track_inventory = false }
39
+ it_should_behave_like 'unlimited supply'
40
+ end
38
41
 
39
- specify { expect(subject.total_on_hand).to eq(Float::INFINITY) }
42
+ context 'when variant inventory tracking is off' do
43
+ before { stock_item.variant.track_inventory = false }
40
44
 
41
- it_should_behave_like 'unlimited supply'
42
- end
45
+ specify { expect(subject.total_on_hand).to eq(Float::INFINITY) }
46
+
47
+ it_should_behave_like 'unlimited supply'
48
+ end
43
49
 
44
- context 'when stock item allows backordering' do
50
+ context 'when stock item allows backordering' do
45
51
 
46
- specify { expect(subject.backorderable?).to be true }
52
+ specify { expect(subject.backorderable?).to be true }
47
53
 
48
- it_should_behave_like 'unlimited supply'
49
- end
54
+ it_should_behave_like 'unlimited supply'
55
+ end
50
56
 
51
- context 'when stock item prevents backordering' do
52
- before { stock_item.update_attributes(backorderable: false) }
57
+ context 'when stock item prevents backordering' do
58
+ before { stock_item.update_attributes(backorderable: false) }
53
59
 
54
- specify { expect(subject.backorderable?).to be false }
60
+ specify { expect(subject.backorderable?).to be false }
61
+
62
+ it 'can_supply? only upto total_on_hand' do
63
+ expect(subject.can_supply?(1)).to be true
64
+ expect(subject.can_supply?(10)).to be true
65
+ expect(subject.can_supply?(11)).to be false
66
+ end
67
+ end
68
+ end
55
69
 
56
- it 'can_supply? only upto total_on_hand' do
57
- expect(subject.can_supply?(1)).to be true
58
- expect(subject.can_supply?(10)).to be true
59
- expect(subject.can_supply?(11)).to be false
70
+ context 'when variant is not available' do
71
+ before(:each) do
72
+ allow(subject.variant).to receive(:available?) { false }
60
73
  end
74
+
75
+ it { expect(subject.can_supply?).to be false }
61
76
  end
62
77
  end
63
78
 
@@ -74,23 +89,37 @@ module Spree
74
89
  expect(subject.total_on_hand).to eq(15)
75
90
  end
76
91
 
77
- context 'when any stock item allows backordering' do
78
- specify { expect(subject.backorderable?).to be true }
92
+ context 'when variant is available' do
93
+ before(:each) do
94
+ allow(subject.variant).to receive(:available?) { true }
95
+ end
79
96
 
80
- it_should_behave_like 'unlimited supply'
81
- end
97
+ context 'when any stock item allows backordering' do
98
+ specify { expect(subject.backorderable?).to be true }
99
+
100
+ it_should_behave_like 'unlimited supply'
101
+ end
82
102
 
83
- context 'when all stock items prevent backordering' do
84
- before { stock_item.update_attributes(backorderable: false) }
103
+ context 'when all stock items prevent backordering' do
104
+ before { stock_item.update_attributes(backorderable: false) }
85
105
 
86
- specify { expect(subject.backorderable?).to be false }
106
+ specify { expect(subject.backorderable?).to be false }
87
107
 
88
- it 'can_supply? upto total_on_hand' do
89
- expect(subject.can_supply?(1)).to be true
90
- expect(subject.can_supply?(15)).to be true
91
- expect(subject.can_supply?(16)).to be false
108
+ it 'can_supply? upto total_on_hand' do
109
+ expect(subject.can_supply?(1)).to be true
110
+ expect(subject.can_supply?(15)).to be true
111
+ expect(subject.can_supply?(16)).to be false
112
+ end
92
113
  end
93
114
  end
115
+
116
+ context 'when variant is not available' do
117
+ before(:each) do
118
+ allow(subject.variant).to receive(:available?) { false }
119
+ end
120
+
121
+ it { expect(subject.can_supply?).to be false }
122
+ end
94
123
  end
95
124
  end
96
125
  end
@@ -167,7 +167,7 @@ describe Spree::StockItem, :type => :model do
167
167
 
168
168
  expect {
169
169
  stock_location.stock_items.create!(variant: subject.variant)
170
- }.to raise_error
170
+ }.to raise_error(StandardError)
171
171
  end
172
172
  end
173
173
 
@@ -407,4 +407,22 @@ describe Spree::StockItem, :type => :model do
407
407
  end
408
408
  end
409
409
  end
410
+
411
+ describe 'scopes' do
412
+ context '.with_active_stock_location' do
413
+ let(:stock_items_with_active_location) { Spree::StockItem.with_active_stock_location }
414
+
415
+ context 'when stock location is active' do
416
+ before { stock_location.update_column(:active, true) }
417
+
418
+ it { expect(stock_items_with_active_location).to include(subject) }
419
+ end
420
+
421
+ context 'when stock location is inactive' do
422
+ before { stock_location.update_column(:active, false) }
423
+
424
+ it { expect(stock_items_with_active_location).to_not include(subject) }
425
+ end
426
+ end
427
+ end
410
428
  end
@@ -0,0 +1,101 @@
1
+ require 'spec_helper'
2
+
3
+ describe 'StoreCreditEvent' do
4
+ describe '#display_amount' do
5
+ let(:event_amount) { 120.0 }
6
+
7
+ subject { create(:store_credit_auth_event, amount: event_amount) }
8
+
9
+ it 'returns a Spree::Money instance' do
10
+ expect(subject.display_amount).to be_instance_of(Spree::Money)
11
+ end
12
+
13
+ it 'uses the events amount attribute' do
14
+ expect(subject.display_amount).to eq Spree::Money.new(event_amount, currency: subject.currency)
15
+ end
16
+ end
17
+
18
+ describe '#display_user_total_amount' do
19
+ let(:user_total_amount) { 300.0 }
20
+
21
+ subject { create(:store_credit_auth_event, user_total_amount: user_total_amount) }
22
+
23
+ it 'returns a Spree::Money instance' do
24
+ expect(subject.display_user_total_amount).to be_instance_of(Spree::Money)
25
+ end
26
+
27
+ it 'uses the events user_total_amount attribute' do
28
+ amount = Spree::Money.new(user_total_amount, currency: subject.currency)
29
+ expect(subject.display_user_total_amount).to eq amount
30
+ end
31
+ end
32
+
33
+ describe '#display_action' do
34
+ subject { create(:store_credit_auth_event, action: action) }
35
+
36
+ context 'capture event' do
37
+ let(:action) { Spree::StoreCredit::CAPTURE_ACTION }
38
+
39
+ it 'returns used' do
40
+ expect(subject.display_action).to eq Spree.t('store_credit.captured')
41
+ end
42
+ end
43
+
44
+ context 'authorize event' do
45
+ let(:action) { Spree::StoreCredit::AUTHORIZE_ACTION }
46
+
47
+ it 'returns authorized' do
48
+ expect(subject.display_action).to eq Spree.t('store_credit.authorized')
49
+ end
50
+ end
51
+
52
+ context 'allocation event' do
53
+ let(:action) { Spree::StoreCredit::ALLOCATION_ACTION }
54
+
55
+ it 'returns added' do
56
+ expect(subject.display_action).to eq Spree.t('store_credit.allocated')
57
+ end
58
+ end
59
+
60
+ context 'void event' do
61
+ let(:action) { Spree::StoreCredit::VOID_ACTION }
62
+
63
+ it 'returns credit' do
64
+ expect(subject.display_action).to eq Spree.t('store_credit.credit')
65
+ end
66
+ end
67
+
68
+ context 'credit event' do
69
+ let(:action) { Spree::StoreCredit::CREDIT_ACTION }
70
+
71
+ it 'returns credit' do
72
+ expect(subject.display_action).to eq Spree.t('store_credit.credit')
73
+ end
74
+ end
75
+ end
76
+
77
+ describe '#order' do
78
+ context 'there is no associated payment with the event' do
79
+ subject { create(:store_credit_auth_event) }
80
+
81
+ it 'returns nil' do
82
+ expect(subject.order).to be_nil
83
+ end
84
+ end
85
+
86
+ context 'there is an associated payment with the event' do
87
+ let(:authorization_code) { '1-SC-TEST' }
88
+ let(:order) { create(:order) }
89
+ let!(:payment) { create(:store_credit_payment, order: order, response_code: authorization_code) }
90
+
91
+ subject do
92
+ create(:store_credit_auth_event, action: Spree::StoreCredit::CAPTURE_ACTION,
93
+ authorization_code: authorization_code)
94
+ end
95
+
96
+ it 'returns the order associated with the payment' do
97
+ expect(subject.order).to eq order
98
+ end
99
+ end
100
+ end
101
+ end
@@ -0,0 +1,786 @@
1
+ require 'spec_helper'
2
+
3
+ describe 'StoreCredit' do
4
+ let(:currency) { 'TEST' }
5
+ let(:store_credit) { build(:store_credit, store_credit_attrs) }
6
+ let(:store_credit_attrs) { {} }
7
+
8
+ describe 'callbacks' do
9
+ subject { store_credit.save }
10
+
11
+ context 'amount used is greater than zero' do
12
+ let(:store_credit) { create(:store_credit, amount: 100, amount_used: 1) }
13
+ let(:validation_message) { I18n.t('activerecord.errors.models.spree/store_credit.attributes.amount_used.greater_than_zero_restrict_delete') }
14
+ subject { store_credit.destroy }
15
+
16
+ it 'can not delete the store credit' do
17
+ subject
18
+ expect(store_credit.reload).to eq store_credit
19
+ expect(store_credit.errors[:amount_used]).to include(validation_message)
20
+ end
21
+ end
22
+
23
+ context 'category is a non-expiring type' do
24
+ let!(:secondary_credit_type) { create(:secondary_credit_type) }
25
+ let(:store_credit) { build(:store_credit, credit_type: nil) }
26
+ before { allow(store_credit.category).to receive_message_chain(:non_expiring?).and_return(true) }
27
+
28
+ it 'sets the credit type to non-expiring' do
29
+ subject
30
+ expect(store_credit.credit_type.name).to eq secondary_credit_type.name
31
+ end
32
+ end
33
+
34
+ context 'category is an expiring type' do
35
+ before { allow(store_credit.category).to receive_message_chain(:non_expiring?).and_return(false) }
36
+
37
+ it 'sets the credit type to non-expiring' do
38
+ subject
39
+ expect(store_credit.credit_type.name).to eq 'Expiring'
40
+ end
41
+ end
42
+
43
+ context 'the type is set' do
44
+ let!(:secondary_credit_type) { create(:secondary_credit_type) }
45
+ let(:store_credit) { build(:store_credit, credit_type: secondary_credit_type) }
46
+ before { allow(store_credit.category).to receive_message_chain(:non_expiring?).and_return(false) }
47
+
48
+ it "doesn't overwrite the type" do
49
+ expect { subject }.to_not change { store_credit.credit_type }
50
+ end
51
+ end
52
+ end
53
+
54
+ describe 'validations' do
55
+ describe 'used amount should not be greater than the credited amount' do
56
+ context 'the used amount is defined' do
57
+ let(:invalid_store_credit) { build(:store_credit, amount: 100, amount_used: 150) }
58
+
59
+ it 'should not be valid' do
60
+ expect(invalid_store_credit).to_not be_valid
61
+ end
62
+
63
+ it 'should set the correct error message' do
64
+ invalid_store_credit.valid?
65
+ attribute_name = I18n.t('activerecord.attributes.spree/store_credit.amount_used')
66
+ validation_message = I18n.t('activerecord.errors.models.spree/store_credit.attributes.amount_used.cannot_be_greater_than_amount')
67
+ expected_error_message = "#{attribute_name} #{validation_message}"
68
+ expect(invalid_store_credit.errors.full_messages).to include(expected_error_message)
69
+ end
70
+ end
71
+
72
+ context 'the used amount is not defined yet' do
73
+ let(:store_credit) { build(:store_credit, amount: 100) }
74
+
75
+ it 'should be valid' do
76
+ expect(store_credit).to be_valid
77
+ end
78
+ end
79
+ end
80
+
81
+ describe 'amount used less than or equal to amount' do
82
+ subject { build(:store_credit, amount_used: 101.0, amount: 100.0) }
83
+
84
+ it 'is not valid' do
85
+ expect(subject).to_not be_valid
86
+ end
87
+
88
+ it 'adds an error message about the invalid amount used' do
89
+ subject.valid?
90
+ text = I18n.t('activerecord.errors.models.spree/store_credit.attributes.amount_used.cannot_be_greater_than_amount')
91
+ expect(subject.errors[:amount_used]).to include(text)
92
+ end
93
+ end
94
+
95
+ describe 'amount authorized less than or equal to amount' do
96
+ subject { build(:store_credit, amount_authorized: 101.0, amount: 100.0) }
97
+
98
+ it 'is not valid' do
99
+ expect(subject).to_not be_valid
100
+ end
101
+
102
+ it 'adds an error message about the invalid authorized amount' do
103
+ subject.valid?
104
+ text = I18n.t('activerecord.errors.models.spree/store_credit.attributes.amount_authorized.exceeds_total_credits')
105
+ expect(subject.errors[:amount_authorized]).to include(text)
106
+ end
107
+ end
108
+ end
109
+
110
+ describe '#display_amount' do
111
+ it 'returns a Spree::Money instance' do
112
+ expect(store_credit.display_amount).to be_instance_of(Spree::Money)
113
+ end
114
+ end
115
+
116
+ describe '#display_amount_used' do
117
+ it 'returns a Spree::Money instance' do
118
+ expect(store_credit.display_amount_used).to be_instance_of(Spree::Money)
119
+ end
120
+ end
121
+
122
+ describe '#amount_remaining' do
123
+ context 'the amount_used is not defined' do
124
+ context 'the authorized amount is not defined' do
125
+ it 'returns the credited amount' do
126
+ expect(store_credit.amount_remaining).to eq store_credit.amount
127
+ end
128
+ end
129
+
130
+ context 'the authorized amount is defined' do
131
+ let(:authorized_amount) { 15.00 }
132
+
133
+ before { store_credit.update_attributes(amount_authorized: authorized_amount) }
134
+
135
+ it 'subtracts the authorized amount from the credited amount' do
136
+ expect(store_credit.amount_remaining).to eq (store_credit.amount - authorized_amount)
137
+ end
138
+ end
139
+ end
140
+
141
+ context 'the amount_used is defined' do
142
+ let(:amount_used) { 10.0 }
143
+
144
+ before { store_credit.update_attributes(amount_used: amount_used) }
145
+
146
+ context 'the authorized amount is not defined' do
147
+ it 'subtracts the amount used from the credited amount' do
148
+ expect(store_credit.amount_remaining).to eq (store_credit.amount - amount_used)
149
+ end
150
+ end
151
+
152
+ context 'the authorized amount is defined' do
153
+ let(:authorized_amount) { 15.00 }
154
+
155
+ before { store_credit.update_attributes(amount_authorized: authorized_amount) }
156
+
157
+ it 'subtracts the amount used and the authorized amount from the credited amount' do
158
+ expect(store_credit.amount_remaining).to eq (store_credit.amount - amount_used - authorized_amount)
159
+ end
160
+ end
161
+ end
162
+ end
163
+
164
+ describe '#authorize' do
165
+ context 'amount is valid' do
166
+ let(:authorization_amount) { 1.0 }
167
+ let(:added_authorization_amount) { 3.0 }
168
+ let(:originator) { nil }
169
+
170
+ context 'amount has not been authorized yet' do
171
+ before { store_credit.update_attributes(amount_authorized: authorization_amount) }
172
+
173
+ it 'returns true' do
174
+ expect(store_credit.authorize(store_credit.amount - authorization_amount, store_credit.currency)).to be_truthy
175
+ end
176
+
177
+ it 'adds the new amount to authorized amount' do
178
+ store_credit.authorize(added_authorization_amount, store_credit.currency)
179
+ expect(store_credit.reload.amount_authorized).to eq (authorization_amount + added_authorization_amount)
180
+ end
181
+
182
+ context 'originator is present' do
183
+ with_model 'OriginatorThing'
184
+
185
+ let(:originator) { OriginatorThing.create! } # won't actually be a user. just giving it a valid model here
186
+
187
+ subject do
188
+ store_credit.authorize(added_authorization_amount, store_credit.currency,
189
+ action_originator: originator)
190
+ end
191
+
192
+ it 'records the originator' do
193
+ expect { subject }.to change { Spree::StoreCreditEvent.count }.by(1)
194
+ expect(Spree::StoreCreditEvent.last.originator).to eq originator
195
+ end
196
+ end
197
+ end
198
+
199
+ context 'authorization has already happened' do
200
+ let!(:auth_event) { create(:store_credit_auth_event, store_credit: store_credit) }
201
+
202
+ before { store_credit.update_attributes(amount_authorized: store_credit.amount) }
203
+
204
+ it 'returns true' do
205
+ expect(store_credit.authorize(store_credit.amount, store_credit.currency,
206
+ action_authorization_code: auth_event.authorization_code)
207
+ ).to be true
208
+ end
209
+ end
210
+ end
211
+
212
+ context 'amount is invalid' do
213
+ it 'returns false' do
214
+ expect(store_credit.authorize(store_credit.amount * 2, store_credit.currency)).to be false
215
+ end
216
+ end
217
+ end
218
+
219
+ describe '#validate_authorization' do
220
+ context 'insufficient funds' do
221
+ subject { store_credit.validate_authorization(store_credit.amount * 2, store_credit.currency) }
222
+
223
+ it 'returns false' do
224
+ expect(subject).to be false
225
+ end
226
+
227
+ it 'adds an error to the model' do
228
+ subject
229
+ text = Spree.t('store_credit_payment_method.insufficient_funds')
230
+ expect(store_credit.errors.full_messages).to include(text)
231
+ end
232
+ end
233
+
234
+ context 'currency mismatch' do
235
+ subject { store_credit.validate_authorization(store_credit.amount, "EUR") }
236
+
237
+ it 'returns false' do
238
+ expect(subject).to be false
239
+ end
240
+
241
+ it 'adds an error to the model' do
242
+ subject
243
+ text = Spree.t('store_credit_payment_method.currency_mismatch')
244
+ expect(store_credit.errors.full_messages).to include(text)
245
+ end
246
+ end
247
+
248
+ context 'valid authorization' do
249
+ subject { store_credit.validate_authorization(store_credit.amount, store_credit.currency) }
250
+
251
+ it 'returns true' do
252
+ expect(subject).to be true
253
+ end
254
+ end
255
+
256
+ context 'troublesome floats' do
257
+ # 8.21.to_d < 8.21 => true
258
+ let(:store_credit_attrs) { { amount: 8.21 } }
259
+
260
+ subject { store_credit.validate_authorization(store_credit_attrs[:amount], store_credit.currency) }
261
+
262
+ it 'returns true' do
263
+ expect(subject).to be_truthy
264
+ end
265
+ end
266
+ end
267
+
268
+ describe '#capture' do
269
+ let(:authorized_amount) { 10.00 }
270
+ let(:auth_code) { '23-SC-20140602164814476128' }
271
+
272
+ before do
273
+ store_credit.update_attributes(amount_authorized: authorized_amount, amount_used: 0.0)
274
+ allow(store_credit).to receive_messages(authorize: true)
275
+ end
276
+
277
+ context 'insufficient funds' do
278
+ subject { store_credit.capture(authorized_amount * 2, auth_code, store_credit.currency) }
279
+
280
+ it 'returns false' do
281
+ expect(subject).to be false
282
+ end
283
+
284
+ it 'adds an error to the model' do
285
+ subject
286
+ text = Spree.t('store_credit_payment_method.insufficient_authorized_amount')
287
+ expect(store_credit.errors.full_messages).to include(text)
288
+ end
289
+
290
+ it 'does not update the store credit model' do
291
+ expect { subject }.to_not change { store_credit }
292
+ end
293
+ end
294
+
295
+ context 'currency mismatch' do
296
+ subject { store_credit.capture(authorized_amount, auth_code, "EUR") }
297
+
298
+ it 'returns false' do
299
+ expect(subject).to be false
300
+ end
301
+
302
+ it 'adds an error to the model' do
303
+ subject
304
+ text = Spree.t('store_credit_payment_method.currency_mismatch')
305
+ expect(store_credit.errors.full_messages).to include(text)
306
+ end
307
+
308
+ it 'does not update the store credit model' do
309
+ expect { subject }.to_not change { store_credit }
310
+ end
311
+ end
312
+
313
+ context 'valid capture' do
314
+ let(:remaining_authorized_amount) { 1 }
315
+ let(:originator) { nil }
316
+
317
+ subject do
318
+ amount = authorized_amount - remaining_authorized_amount
319
+ store_credit.capture(amount, auth_code, store_credit.currency,
320
+ action_originator: originator)
321
+ end
322
+
323
+ it 'returns true' do
324
+ expect(subject).to be_truthy
325
+ end
326
+
327
+ it 'updates the authorized amount to the difference between the captured amount and the authorized amount' do
328
+ subject
329
+ expect(store_credit.reload.amount_authorized).to eq remaining_authorized_amount
330
+ end
331
+
332
+ it 'updates the used amount to the current used amount plus the captured amount' do
333
+ subject
334
+ expect(store_credit.reload.amount_used).to eq authorized_amount - remaining_authorized_amount
335
+ end
336
+
337
+ context 'originator is present' do
338
+ with_model 'OriginatorThing'
339
+
340
+ let(:originator) { OriginatorThing.create! } # won't actually be a user. just giving it a valid model here
341
+
342
+ it 'records the originator' do
343
+ expect { subject }.to change { Spree::StoreCreditEvent.count }.by(1)
344
+ expect(Spree::StoreCreditEvent.last.originator).to eq originator
345
+ end
346
+ end
347
+ end
348
+ end
349
+
350
+ describe '#void' do
351
+ let(:auth_code) { '1-SC-20141111111111' }
352
+ let(:store_credit) { create(:store_credit, amount_used: 150.0) }
353
+ let(:originator) { nil }
354
+
355
+ subject do
356
+ store_credit.void(auth_code, action_originator: originator)
357
+ end
358
+
359
+ context 'no event found for auth_code' do
360
+ it 'returns false' do
361
+ expect(subject).to be false
362
+ end
363
+
364
+ it 'adds an error to the model' do
365
+ subject
366
+ text = Spree.t('store_credit_payment_method.unable_to_void', auth_code: auth_code)
367
+ expect(store_credit.errors.full_messages).to include(text)
368
+ end
369
+ end
370
+
371
+ context 'capture event found for auth_code' do
372
+ let(:captured_amount) { 10.0 }
373
+ let!(:capture_event) do
374
+ create(:store_credit_auth_event,
375
+ action: Spree::StoreCredit::CAPTURE_ACTION,
376
+ authorization_code: auth_code,
377
+ amount: captured_amount,
378
+ store_credit: store_credit)
379
+ end
380
+
381
+ it 'returns false' do
382
+ expect(subject).to be false
383
+ end
384
+
385
+ it 'does not change the amount used on the store credit' do
386
+ expect { subject }.to_not change { store_credit.amount_used.to_f }
387
+ end
388
+ end
389
+
390
+ context 'auth event found for auth_code' do
391
+ let(:auth_event) { create(:store_credit_auth_event) }
392
+
393
+ let(:authorized_amount) { 10.0 }
394
+ let!(:auth_event) do
395
+ create(:store_credit_auth_event,
396
+ authorization_code: auth_code,
397
+ amount: authorized_amount,
398
+ store_credit: store_credit)
399
+ end
400
+
401
+ it 'returns true' do
402
+ expect(subject).to be true
403
+ end
404
+
405
+ it 'returns the capture amount to the store credit' do
406
+ expect { subject }.to change { store_credit.amount_authorized.to_f }.by(-authorized_amount)
407
+ end
408
+
409
+ context 'originator is present' do
410
+ with_model 'OriginatorThing'
411
+
412
+ let(:originator) { OriginatorThing.create! } # won't actually be a user. just giving it a valid model here
413
+
414
+ it 'records the originator' do
415
+ expect { subject }.to change { Spree::StoreCreditEvent.count }.by(1)
416
+ expect(Spree::StoreCreditEvent.last.originator).to eq originator
417
+ end
418
+ end
419
+ end
420
+ end
421
+
422
+ describe '#credit' do
423
+ let(:event_auth_code) { '1-SC-20141111111111' }
424
+ let(:amount_used) { 10.0 }
425
+ let(:store_credit) { create(:store_credit, amount_used: amount_used) }
426
+ let!(:capture_event) do
427
+ create(:store_credit_auth_event,
428
+ action: Spree::StoreCredit::CAPTURE_ACTION,
429
+ authorization_code: event_auth_code,
430
+ amount: captured_amount,
431
+ store_credit: store_credit)
432
+ end
433
+ let(:originator) { nil }
434
+
435
+ subject do
436
+ store_credit.credit(credit_amount, auth_code, currency, action_originator: originator)
437
+ end
438
+
439
+ context 'currency does not match' do
440
+ let(:currency) { 'AUD' }
441
+ let(:credit_amount) { 5.0 }
442
+ let(:captured_amount) { 100.0 }
443
+ let(:auth_code) { event_auth_code }
444
+
445
+ it 'returns false' do
446
+ expect(subject).to be false
447
+ end
448
+
449
+ it 'adds an error message about the currency mismatch' do
450
+ subject
451
+ text = Spree.t('store_credit_payment_method.currency_mismatch')
452
+ expect(store_credit.errors.full_messages).to include(text)
453
+ end
454
+ end
455
+
456
+ context 'unable to find capture event' do
457
+ let(:currency) { 'USD' }
458
+ let(:credit_amount) { 5.0 }
459
+ let(:captured_amount) { 100.0 }
460
+ let(:auth_code) { 'UNKNOWN_CODE' }
461
+
462
+ it 'returns false' do
463
+ expect(subject).to be false
464
+ end
465
+
466
+ it 'adds an error message about the currency mismatch' do
467
+ subject
468
+ text = Spree.t('store_credit_payment_method.unable_to_credit', auth_code: auth_code)
469
+ expect(store_credit.errors.full_messages).to include(text)
470
+ end
471
+ end
472
+
473
+ context 'amount is more than what is captured' do
474
+ let(:currency) { 'USD' }
475
+ let(:credit_amount) { 100.0 }
476
+ let(:captured_amount) { 5.0 }
477
+ let(:auth_code) { event_auth_code }
478
+
479
+ it 'returns false' do
480
+ expect(subject).to be false
481
+ end
482
+
483
+ it 'adds an error message about the currency mismatch' do
484
+ subject
485
+ text = Spree.t('store_credit_payment_method.unable_to_credit', auth_code: auth_code)
486
+ expect(store_credit.errors.full_messages).to include(text)
487
+ end
488
+ end
489
+
490
+ context 'amount is successfully credited' do
491
+ let(:currency) { 'USD' }
492
+ let(:credit_amount) { 5.0 }
493
+ let(:captured_amount) { 100.0 }
494
+ let(:auth_code) { event_auth_code }
495
+
496
+ context 'credit_to_new_allocation is set' do
497
+ before { Spree::Config[:credit_to_new_allocation] = true }
498
+
499
+ it 'returns true' do
500
+ expect(subject).to be true
501
+ end
502
+
503
+ it 'creates a new store credit record' do
504
+ expect { subject }.to change { Spree::StoreCredit.count }.by(1)
505
+ end
506
+
507
+ it 'does not create a new store credit event on the parent store credit' do
508
+ expect { subject }.to_not change { store_credit.store_credit_events.count }
509
+ end
510
+
511
+ context 'credits the passed amount to a new store credit record' do
512
+ before do
513
+ subject
514
+ @new_store_credit = Spree::StoreCredit.last
515
+ end
516
+
517
+ it 'does not set the amount used on hte originating store credit' do
518
+ expect(store_credit.reload.amount_used).to eq amount_used
519
+ end
520
+
521
+ it 'sets the correct amount on the new store credit' do
522
+ expect(@new_store_credit.amount).to eq credit_amount
523
+ end
524
+
525
+ [:user_id, :category_id, :created_by_id, :currency, :type_id].each do |attr|
526
+ it "sets attribute #{attr} inherited from the originating store credit" do
527
+ expect(@new_store_credit.send(attr)).to eq store_credit.send(attr)
528
+ end
529
+ end
530
+
531
+ it 'sets a memo' do
532
+ expect(@new_store_credit.memo).to eq "This is a credit from store credit ID #{store_credit.id}"
533
+ end
534
+ end
535
+
536
+ context 'originator is present' do
537
+ with_model 'OriginatorThing'
538
+
539
+ let(:originator) { OriginatorThing.create! } # won't actually be a user. just giving it a valid model here
540
+
541
+ it 'records the originator' do
542
+ expect { subject }.to change { Spree::StoreCreditEvent.count }.by(1)
543
+ expect(Spree::StoreCreditEvent.last.originator).to eq originator
544
+ end
545
+ end
546
+ end
547
+
548
+ context 'credit_to_new_allocation is not set' do
549
+ it 'returns true' do
550
+ expect(subject).to be true
551
+ end
552
+
553
+ it 'credits the passed amount to the store credit amount used' do
554
+ subject
555
+ expect(store_credit.reload.amount_used).to eq (amount_used - credit_amount)
556
+ end
557
+
558
+ it 'creates a new store credit event' do
559
+ expect { subject }.to change { store_credit.store_credit_events.count }.by(1)
560
+ end
561
+ end
562
+ end
563
+ end
564
+
565
+ describe '#amount_used' do
566
+ context 'amount used is not defined' do
567
+ subject { Spree::StoreCredit.new }
568
+
569
+ it 'returns zero' do
570
+ expect(subject.amount_used).to be_zero
571
+ end
572
+ end
573
+
574
+ context 'amount used is defined' do
575
+ let(:amount_used) { 100.0 }
576
+
577
+ subject { create(:store_credit, amount_used: amount_used) }
578
+
579
+ it 'returns the attribute value' do
580
+ expect(subject.amount_used).to eq amount_used
581
+ end
582
+ end
583
+ end
584
+
585
+ describe '#amount_authorized' do
586
+ context 'amount authorized is not defined' do
587
+ subject { Spree::StoreCredit.new }
588
+
589
+ it 'returns zero' do
590
+ expect(subject.amount_authorized).to be_zero
591
+ end
592
+ end
593
+
594
+ context 'amount authorized is defined' do
595
+ let(:amount_authorized) { 100.0 }
596
+
597
+ subject { create(:store_credit, amount_authorized: amount_authorized) }
598
+
599
+ it 'returns the attribute value' do
600
+ expect(subject.amount_authorized).to eq amount_authorized
601
+ end
602
+ end
603
+ end
604
+
605
+ describe '#can_capture?' do
606
+ let(:store_credit) { create(:store_credit) }
607
+ let(:payment) { create(:payment, state: payment_state) }
608
+
609
+ subject { store_credit.can_capture?(payment) }
610
+
611
+ context 'pending payment' do
612
+ let(:payment_state) { 'pending' }
613
+
614
+ it 'returns true' do
615
+ expect(subject).to be true
616
+ end
617
+ end
618
+
619
+ context 'checkout payment' do
620
+ let(:payment_state) { 'checkout' }
621
+
622
+ it 'returns true' do
623
+ expect(subject).to be true
624
+ end
625
+ end
626
+
627
+ context 'void payment' do
628
+ let(:payment_state) { Spree::StoreCredit::VOID_ACTION }
629
+
630
+ it 'returns false' do
631
+ expect(subject).to be false
632
+ end
633
+ end
634
+
635
+ context 'invalid payment' do
636
+ let(:payment_state) { 'invalid' }
637
+
638
+ it 'returns false' do
639
+ expect(subject).to be false
640
+ end
641
+ end
642
+
643
+ context 'complete payment' do
644
+ let(:payment_state) { 'completed' }
645
+
646
+ it 'returns false' do
647
+ expect(subject).to be false
648
+ end
649
+ end
650
+ end
651
+
652
+ describe '#can_void?' do
653
+ let(:store_credit) { create(:store_credit) }
654
+ let(:payment) { create(:payment, state: payment_state) }
655
+
656
+ subject { store_credit.can_void?(payment) }
657
+
658
+ context 'pending payment' do
659
+ let(:payment_state) { 'pending' }
660
+
661
+ it 'returns true' do
662
+ expect(subject).to be true
663
+ end
664
+ end
665
+
666
+ context 'checkout payment' do
667
+ let(:payment_state) { 'checkout' }
668
+
669
+ it 'returns false' do
670
+ expect(subject).to be false
671
+ end
672
+ end
673
+
674
+ context 'void payment' do
675
+ let(:payment_state) { Spree::StoreCredit::VOID_ACTION }
676
+
677
+ it 'returns false' do
678
+ expect(subject).to be false
679
+ end
680
+ end
681
+
682
+ context 'invalid payment' do
683
+ let(:payment_state) { 'invalid' }
684
+
685
+ it 'returns false' do
686
+ expect(subject).to be false
687
+ end
688
+ end
689
+
690
+ context 'complete payment' do
691
+ let(:payment_state) { 'completed' }
692
+
693
+ it 'returns false' do
694
+ expect(subject).to be false
695
+ end
696
+ end
697
+ end
698
+
699
+ describe '#can_credit?' do
700
+ let(:store_credit) { create(:store_credit) }
701
+ let(:payment) { create(:payment, state: payment_state) }
702
+
703
+ subject { store_credit.can_credit?(payment) }
704
+
705
+ context 'payment is not completed' do
706
+ let(:payment_state) { 'pending' }
707
+
708
+ it 'returns false' do
709
+ expect(subject).to be false
710
+ end
711
+ end
712
+
713
+ context 'payment is completed' do
714
+ let(:payment_state) { 'completed' }
715
+
716
+ context 'credit is owed on the order' do
717
+ before { allow(payment.order).to receive_messages(payment_state: 'credit_owed') }
718
+
719
+ context "payment doesn't have allowed credit" do
720
+ before { allow(payment).to receive_messages(credit_allowed: 0.0) }
721
+
722
+ it 'returns false' do
723
+ expect(subject).to be false
724
+ end
725
+ end
726
+
727
+ context 'payment has allowed credit' do
728
+ before { allow(payment).to receive_messages(credit_allowed: 5.0) }
729
+
730
+ it 'returns true' do
731
+ expect(subject).to be true
732
+ end
733
+ end
734
+ end
735
+ end
736
+
737
+ describe '#store_events' do
738
+ context 'create' do
739
+ context 'user has one store credit' do
740
+ let(:store_credit_amount) { 100.0 }
741
+
742
+ subject { create(:store_credit, amount: store_credit_amount) }
743
+
744
+ it 'creates a store credit event' do
745
+ expect { subject }.to change { Spree::StoreCreditEvent.count }.by(1)
746
+ end
747
+
748
+ it 'makes the store credit event an allocation event' do
749
+ expect(subject.store_credit_events.first.action).to eq Spree::StoreCredit::ALLOCATION_ACTION
750
+ end
751
+
752
+ it "saves the user's total store credit in the event" do
753
+ expect(subject.store_credit_events.first.user_total_amount).to eq store_credit_amount
754
+ end
755
+ end
756
+
757
+ context 'user has multiple store credits' do
758
+ let(:store_credit_amount) { 100.0 }
759
+ let(:additional_store_credit_amount) { 200.0 }
760
+
761
+ let(:user) { create(:user) }
762
+ let!(:store_credit) { create(:store_credit, user: user, amount: store_credit_amount) }
763
+
764
+ subject { create(:store_credit, user: user, amount: additional_store_credit_amount) }
765
+
766
+ it "saves the user's total store credit in the event" do
767
+ amount = store_credit_amount + additional_store_credit_amount
768
+ expect(subject.store_credit_events.first.user_total_amount).to eq amount
769
+ end
770
+ end
771
+
772
+ context 'an action is specified' do
773
+ it 'creates an event with the set action' do
774
+ store_credit = build(:store_credit)
775
+ store_credit.action = Spree::StoreCredit::VOID_ACTION
776
+ store_credit.action_authorization_code = '1-SC-TEST'
777
+
778
+ expect { store_credit.save! }.to change {
779
+ Spree::StoreCreditEvent.where(action: Spree::StoreCredit::VOID_ACTION).count
780
+ }.by(1)
781
+ end
782
+ end
783
+ end
784
+ end
785
+ end
786
+ end