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,7 @@
1
+ require 'spec_helper'
2
+
3
+ describe Spree::ReturnAuthorizationReason do
4
+ describe 'Associations' do
5
+ it { is_expected.to have_many(:return_authorizations).dependent(:restrict_with_error) }
6
+ end
7
+ end
@@ -84,28 +84,8 @@ describe Spree::ReturnAuthorization, :type => :model do
84
84
  end
85
85
  end
86
86
 
87
- describe ".before_create" do
88
- describe "#generate_number" do
89
- context "number is assigned" do
90
- let(:return_authorization) { Spree::ReturnAuthorization.new(number: '123') }
91
-
92
- it "should return the assigned number" do
93
- return_authorization.save
94
- expect(return_authorization.number).to eq('123')
95
- end
96
- end
97
-
98
- context "number is not assigned" do
99
- let(:return_authorization) { Spree::ReturnAuthorization.new(number: nil) }
100
-
101
- before { allow(return_authorization).to receive_messages valid?: true }
102
-
103
- it "should assign number with random RA number" do
104
- return_authorization.save
105
- expect(return_authorization.number).to match(/RA\d{9}/)
106
- end
107
- end
108
- end
87
+ describe 'whitelisted_ransackable_attributes' do
88
+ it { expect(Spree::ReturnAuthorization.whitelisted_ransackable_attributes).to eq(%w(memo number state)) }
109
89
  end
110
90
 
111
91
  context "#currency" do
@@ -18,7 +18,7 @@ describe Spree::ReturnItem, :type => :model do
18
18
  end
19
19
 
20
20
  describe '#receive!' do
21
- let(:now) { Time.now }
21
+ let(:now) { Time.current }
22
22
  let(:inventory_unit) { create(:inventory_unit, state: 'shipped') }
23
23
  let(:return_item) { create(:return_item, inventory_unit: inventory_unit) }
24
24
 
@@ -679,4 +679,53 @@ describe Spree::ReturnItem, :type => :model do
679
679
  expect(return_item.total).to eq 20
680
680
  end
681
681
  end
682
+
683
+ describe '#process_inventory_unit!' do
684
+ let(:inventory_unit) { create(:inventory_unit, state: 'shipped') }
685
+ let(:return_item) { create(:return_item, inventory_unit: inventory_unit, reception_status: 'awaiting') }
686
+ let!(:stock_item) { inventory_unit.find_stock_item }
687
+ before { return_item.update_attributes!(reception_status: 'awaiting') }
688
+
689
+ subject { return_item.send(:process_inventory_unit!) }
690
+
691
+ it { expect { subject }.to change { inventory_unit.state }.to('returned').from('shipped') }
692
+
693
+ context 'stock should restock' do
694
+ let(:stock_movement_attributes) do
695
+ {
696
+ stock_item_id: stock_item.id,
697
+ quantity: 1,
698
+ originator: return_item.return_authorization
699
+ }
700
+ end
701
+
702
+ it { expect(subject).to eq(Spree::StockMovement.find_by(stock_movement_attributes)) }
703
+ end
704
+
705
+ context 'stock should not restock' do
706
+ context 'return_item is not resellable' do
707
+ before { return_item.resellable = false }
708
+ it { expect(subject).to be_nil }
709
+ it { expect { subject }.to_not change { stock_item.reload.count_on_hand } }
710
+ end
711
+
712
+ context 'variant should not track inventory' do
713
+ before { return_item.variant.track_inventory = false }
714
+ it { expect(subject).to be_nil }
715
+ it { expect { subject }.to_not change { stock_item.reload.count_on_hand } }
716
+ end
717
+
718
+ context 'stock_item not present' do
719
+ before { stock_item.destroy }
720
+ it { expect(subject).to be_nil }
721
+ it { expect { subject }.to_not change { stock_item.reload.count_on_hand } }
722
+ end
723
+
724
+ context 'when restock inventory preference false' do
725
+ before { Spree::Config[:restock_inventory] = false }
726
+ it { expect(subject).to be_nil }
727
+ it { expect { subject }.to_not change { stock_item.reload.count_on_hand } }
728
+ end
729
+ end
730
+ end
682
731
  end
@@ -8,7 +8,7 @@ module Spree
8
8
  it 'compute_shipment must be overridden' do
9
9
  expect {
10
10
  subject.compute(return_item)
11
- }.to raise_error
11
+ }.to raise_error(NotImplementedError)
12
12
  end
13
13
  end
14
14
  end
@@ -0,0 +1,7 @@
1
+ require 'spec_helper'
2
+
3
+ describe Spree::Role, type: :model do
4
+ describe 'Associations' do
5
+ it { is_expected.to have_many(:role_users).class_name('Spree::RoleUser').dependent(:destroy) }
6
+ end
7
+ end
@@ -41,7 +41,7 @@ describe Spree::Shipment, :type => :model do
41
41
  # Regression test for #4063
42
42
  context "number generation" do
43
43
  before do
44
- allow(order).to receive :update!
44
+ allow(order).to receive :update_with_updater!
45
45
  end
46
46
 
47
47
  it "generates a number containing a letter + 11 numbers" do
@@ -394,7 +394,7 @@ describe Spree::Shipment, :type => :model do
394
394
 
395
395
  context "#cancel" do
396
396
  it 'cancels the shipment' do
397
- allow(shipment.order).to receive(:update!)
397
+ allow(shipment.order).to receive(:update_with_updater!)
398
398
 
399
399
  shipment.state = 'pending'
400
400
  expect(shipment).to receive(:after_cancel)
@@ -439,33 +439,33 @@ describe Spree::Shipment, :type => :model do
439
439
  end
440
440
 
441
441
  context "#resume" do
442
- it 'will determine new state based on order' do
443
- allow(shipment.order).to receive(:update!)
442
+ it 'transitions state to ready if the order is ready' do
443
+ allow(shipment.order).to receive(:update_with_updater!)
444
444
 
445
445
  shipment.state = 'canceled'
446
- expect(shipment).to receive(:determine_state).and_return(:ready)
446
+ expect(shipment).to receive(:determine_state).and_return('ready')
447
447
  expect(shipment).to receive(:after_resume)
448
448
  shipment.resume!
449
449
  expect(shipment.state).to eq 'ready'
450
450
  end
451
451
 
452
- it 'unstocks them items' do
453
- allow(shipment).to receive_message_chain(inventory_units: [mock_model(Spree::InventoryUnit, line_item: line_item, variant: variant)])
454
- shipment.stock_location = mock_model(Spree::StockLocation)
455
- expect(shipment.stock_location).to receive(:unstock).with(variant, 1, shipment)
456
- shipment.after_resume
457
- end
458
-
459
- it 'will determine new state based on order' do
460
- allow(shipment.order).to receive(:update!)
452
+ it 'transitions state to pending if the order is not ready' do
453
+ allow(shipment.order).to receive(:update_with_updater!)
461
454
 
462
455
  shipment.state = 'canceled'
463
- expect(shipment).to receive(:determine_state).twice.and_return('ready')
456
+ expect(shipment).to receive(:determine_state).and_return('pending')
464
457
  expect(shipment).to receive(:after_resume)
465
458
  shipment.resume!
466
459
  # Shipment is pending because order is already paid
467
460
  expect(shipment.state).to eq 'pending'
468
461
  end
462
+
463
+ it 'unstocks them items' do
464
+ allow(shipment).to receive_message_chain(inventory_units: [mock_model(Spree::InventoryUnit, line_item: line_item, variant: variant)])
465
+ shipment.stock_location = mock_model(Spree::StockLocation)
466
+ expect(shipment.stock_location).to receive(:unstock).with(variant, 1, shipment)
467
+ shipment.after_resume
468
+ end
469
469
  end
470
470
 
471
471
  context "#ship" do
@@ -473,7 +473,7 @@ describe Spree::Shipment, :type => :model do
473
473
  let(:shipment_with_inventory_units) { create(:shipment, order: create(:order_with_line_items), state: 'canceled') }
474
474
  let(:subject) { shipment_with_inventory_units.ship! }
475
475
  before do
476
- allow(order).to receive(:update!)
476
+ allow(order).to receive(:update_with_updater!)
477
477
  allow(shipment_with_inventory_units).to receive_messages(require_inventory: false, update_order: true)
478
478
  end
479
479
 
@@ -489,7 +489,7 @@ describe Spree::Shipment, :type => :model do
489
489
  ['ready', 'canceled'].each do |state|
490
490
  context "from #{state}" do
491
491
  before do
492
- allow(order).to receive(:update!)
492
+ allow(order).to receive(:update_with_updater!)
493
493
  allow(shipment).to receive_messages(require_inventory: false, update_order: true, state: state)
494
494
  end
495
495
 
@@ -25,13 +25,13 @@ module Spree
25
25
  it 'compute_shipment must be overridden' do
26
26
  expect {
27
27
  subject.compute_shipment(shipment)
28
- }.to raise_error
28
+ }.to raise_error(NameError)
29
29
  end
30
30
 
31
31
  it 'compute_package must be overridden' do
32
32
  expect {
33
33
  subject.compute_package(package)
34
- }.to raise_error
34
+ }.to raise_error(NotImplementedError)
35
35
  end
36
36
 
37
37
  it 'checks availability for a package' do
@@ -2,4 +2,18 @@ require 'spec_helper'
2
2
 
3
3
  describe Spree::ShippingCategory, :type => :model do
4
4
 
5
+ describe '#validations' do
6
+ it 'should have a valid factory' do
7
+ expect(FactoryGirl.build(:shipping_category)).to be_valid
8
+ end
9
+
10
+ it 'should require name' do
11
+ expect(FactoryGirl.build(:shipping_category, name: '')).not_to be_valid
12
+ end
13
+
14
+ it 'should validate uniqueness' do
15
+ FactoryGirl.create(:shipping_category, name: 'Test')
16
+ expect(FactoryGirl.build(:shipping_category, name: 'Test')).not_to be_valid
17
+ end
18
+ end
5
19
  end
@@ -42,8 +42,15 @@ describe Spree::ShippingMethod, :type => :model do
42
42
  end
43
43
 
44
44
  context "shipping category" do
45
- it "validates presence of at least one" do
46
- expect(subject.error_on(:base).size).to eq(1)
45
+ context "is required" do
46
+ it { expect(subject.error_on(:base).size).to eq(1) }
47
+ it 'adds error to base' do
48
+ expect(subject.error_on(:base)).to include(I18n.t(:required_shipping_category,
49
+ scope: [
50
+ :activerecord, :errors, :models,
51
+ 'spree/shipping_method', :attributes, :base
52
+ ]))
53
+ end
47
54
  end
48
55
 
49
56
  context "one associated" do
@@ -2,29 +2,32 @@
2
2
 
3
3
  require 'spec_helper'
4
4
 
5
- describe Spree::ShippingRate, :type => :model do
5
+ describe Spree::ShippingRate, type: :model do
6
6
  let(:shipment) { create(:shipment) }
7
7
  let(:shipping_method) { create(:shipping_method) }
8
- let(:shipping_rate) { Spree::ShippingRate.new(:shipment => shipment,
9
- :shipping_method => shipping_method,
10
- :cost => 10) }
8
+ let(:shipping_rate) do
9
+ Spree::ShippingRate.new shipment: shipment,
10
+ shipping_method: shipping_method,
11
+ cost: 10
12
+ end
11
13
 
12
14
  context "#display_price" do
13
15
  context "when tax included in price" do
16
+ let!(:default_zone) { create(:zone, default_tax: true) }
17
+ let(:default_tax_rate) do
18
+ create :tax_rate,
19
+ name: "VAT",
20
+ amount: 0.1,
21
+ included_in_price: true,
22
+ zone: default_zone
23
+ end
14
24
  context "when the tax rate is from the default zone" do
15
- let!(:zone) { create(:zone, :default_tax => true) }
16
- let(:tax_rate) do
17
- create(:tax_rate,
18
- :name => "VAT",
19
- :amount => 0.1,
20
- :included_in_price => true,
21
- :zone => zone)
22
- end
23
25
 
24
- before { shipping_rate.tax_rate = tax_rate }
26
+ before { shipping_rate.tax_rate = default_tax_rate }
25
27
 
26
28
  it "shows correct tax amount" do
27
- expect(shipping_rate.display_price.to_s).to eq("$10.00 (incl. $0.91 #{tax_rate.name})")
29
+ expect(shipping_rate.display_price.to_s).
30
+ to eq("$10.00 (incl. $0.91 #{default_tax_rate.name})")
28
31
  end
29
32
 
30
33
  context "when cost is zero" do
@@ -38,20 +41,21 @@ describe Spree::ShippingRate, :type => :model do
38
41
  end
39
42
  end
40
43
 
41
- context "when the tax rate is from a non-default zone" do
42
- let!(:default_zone) { create(:zone, :default_tax => true) }
43
- let!(:non_default_zone) { create(:zone, :default_tax => false) }
44
- let(:tax_rate) do
45
- create(:tax_rate,
46
- :name => "VAT",
47
- :amount => 0.1,
48
- :included_in_price => true,
49
- :zone => non_default_zone)
44
+ context "when the tax rate is from another zone" do
45
+ let!(:non_default_zone) { create(:zone, default_tax: false) }
46
+
47
+ let(:non_default_tax_rate) do
48
+ create :tax_rate,
49
+ name: "VAT",
50
+ amount: 0.2,
51
+ included_in_price: true,
52
+ zone: non_default_zone
50
53
  end
51
- before { shipping_rate.tax_rate = tax_rate }
54
+ before { shipping_rate.tax_rate = non_default_tax_rate }
52
55
 
53
- it "shows correct tax amount" do
54
- expect(shipping_rate.display_price.to_s).to eq("$10.00 (excl. $0.91 #{tax_rate.name})")
56
+ it "deducts the other zone's VAT from the calculated shipping rate" do
57
+ expect(shipping_rate.display_price.to_s).
58
+ to eq("$10.00 (incl. $1.67 #{non_default_tax_rate.name})")
55
59
  end
56
60
 
57
61
  context "when cost is zero" do
@@ -67,11 +71,12 @@ describe Spree::ShippingRate, :type => :model do
67
71
  end
68
72
 
69
73
  context "when tax is additional to price" do
70
- let(:tax_rate) { create(:tax_rate, :name => "Sales Tax", :amount => 0.1) }
74
+ let(:tax_rate) { create(:tax_rate, name: "Sales Tax", amount: 0.1) }
71
75
  before { shipping_rate.tax_rate = tax_rate }
72
76
 
73
77
  it "shows correct tax amount" do
74
- expect(shipping_rate.display_price.to_s).to eq("$10.00 (+ $1.00 #{tax_rate.name})")
78
+ expect(shipping_rate.display_price.to_s).
79
+ to eq("$10.00 (+ $1.00 #{tax_rate.name})")
75
80
  end
76
81
 
77
82
  context "when cost is zero" do
@@ -86,9 +91,9 @@ describe Spree::ShippingRate, :type => :model do
86
91
  end
87
92
 
88
93
  context "when the currency is JPY" do
89
- let(:shipping_rate) { shipping_rate = Spree::ShippingRate.new(:cost => 205)
90
- allow(shipping_rate).to receive_messages(:currency => "JPY")
91
- shipping_rate }
94
+ let(:shipping_rate) { Spree::ShippingRate.new(cost: 205) }
95
+
96
+ before { allow(shipping_rate).to receive_messages(currency: "JPY") }
92
97
 
93
98
  it "displays the price in yen" do
94
99
  expect(shipping_rate.display_price.to_s).to eq("¥205")
@@ -103,7 +108,7 @@ describe Spree::ShippingRate, :type => :model do
103
108
  end
104
109
 
105
110
  it "can be retrieved even when deleted" do
106
- shipping_method.update_column(:deleted_at, Time.now)
111
+ shipping_method.update_column(:deleted_at, Time.current)
107
112
  shipping_rate.save
108
113
  shipping_rate.reload
109
114
  expect(shipping_rate.shipping_method).to eq(shipping_method)
@@ -122,20 +127,14 @@ describe Spree::ShippingRate, :type => :model do
122
127
  end
123
128
 
124
129
  it "can be retrieved even when deleted" do
125
- tax_rate.update_column(:deleted_at, Time.now)
130
+ tax_rate.update_column(:deleted_at, Time.current)
126
131
  shipping_rate.save
127
132
  shipping_rate.reload
128
133
  expect(shipping_rate.tax_rate).to eq(tax_rate)
129
134
  end
130
135
  end
131
136
 
132
- context "#shipping_method_code" do
133
- before do
134
- shipping_method.code = "THE_CODE"
135
- end
136
-
137
- it 'should be shipping_method.code' do
138
- expect(shipping_rate.shipping_method_code).to eq("THE_CODE")
139
- end
137
+ describe 'delegate' do
138
+ it { is_expected.to delegate_method(:code).to(:shipping_method).with_prefix(true) }
140
139
  end
141
140
  end
@@ -1,6 +1,17 @@
1
1
  require 'spec_helper'
2
2
 
3
- describe Spree::State, :type => :model do
3
+ describe Spree::State, type: :model do
4
+ describe 'Associations' do
5
+ it { is_expected.to have_many(:addresses).dependent(:restrict_with_error) }
6
+ end
7
+
8
+ describe 'validations' do
9
+ it { is_expected.to validate_presence_of(:name) }
10
+ it { is_expected.to validate_presence_of(:country) }
11
+ it { is_expected.to validate_uniqueness_of(:name).scoped_to(:country_id).case_insensitive }
12
+ it { is_expected.to validate_uniqueness_of(:abbr).scoped_to(:country_id).case_insensitive }
13
+ end
14
+
4
15
  it "can find a state by name or abbr" do
5
16
  state = create(:state, :name => "California", :abbr => "CA")
6
17
  expect(Spree::State.find_all_by_name_or_abbr("California")).to include(state)
@@ -6,6 +6,15 @@ module Spree
6
6
  let(:variant) { build(:variant, weight: 25.0) }
7
7
  subject { ContentItem.new(build(:inventory_unit, variant: variant)) }
8
8
 
9
+ describe 'Delegations' do
10
+ it { is_expected.to delegate_method(:line_item).to(:inventory_unit) }
11
+ it { is_expected.to delegate_method(:variant).to(:inventory_unit) }
12
+ it { is_expected.to delegate_method(:dimension).to(:variant).with_prefix }
13
+ it { is_expected.to delegate_method(:price).to(:variant) }
14
+ it { is_expected.to delegate_method(:volume).to(:variant).with_prefix }
15
+ it { is_expected.to delegate_method(:weight).to(:variant).with_prefix }
16
+ end
17
+
9
18
  context "#volume" do
10
19
  it "calculate the total volume of the variant" do
11
20
  expect(subject.volume).to eq variant.volume * subject.quantity
@@ -70,14 +70,6 @@ module Spree
70
70
  it_should_behave_like "shipping rate doesn't match"
71
71
  end
72
72
 
73
- context "when the shipping method's calculator raises an exception" do
74
- before do
75
- allow_any_instance_of(ShippingMethod).to receive_message_chain(:calculator, :available?).and_raise(Exception, "Something went wrong!")
76
- expect(subject).to receive(:log_calculator_exception)
77
- end
78
- it_should_behave_like "shipping rate doesn't match"
79
- end
80
-
81
73
  it "sorts shipping rates by cost" do
82
74
  shipping_methods = 3.times.map { create(:shipping_method) }
83
75
  allow(shipping_methods[0]).to receive_message_chain(:calculator, :compute).and_return(5.00)
@@ -148,6 +140,62 @@ module Spree
148
140
  expect(shipping_rates.first.tax_rate).to eq(tax_rate)
149
141
  end
150
142
  end
143
+
144
+ context "VAT price calculation" do
145
+ let(:tax_category) { create :tax_category }
146
+ let!(:shipping_method) { create(:shipping_method, tax_category: tax_category) }
147
+
148
+ let(:default_zone) { create(:zone_with_country, default_tax: true) }
149
+ let!(:default_vat) do
150
+ create :tax_rate,
151
+ included_in_price: true,
152
+ zone: default_zone,
153
+ amount: 0.2,
154
+ tax_category: shipping_method.tax_category
155
+ end
156
+
157
+ context "when the order does not have a tax zone" do
158
+ before { allow(order).to receive(:tax_zone).and_return nil }
159
+ it_should_behave_like "shipping rate matches"
160
+ end
161
+
162
+ context "when the order's tax zone is the default zone" do
163
+ before { allow(order).to receive(:tax_zone).and_return(default_zone) }
164
+ it_should_behave_like "shipping rate matches"
165
+ end
166
+
167
+ context "when the order's tax zone is a non-VAT zone" do
168
+ let!(:zone_without_vat) { create(:zone_with_country) }
169
+ before { allow(order).to receive(:tax_zone).and_return(zone_without_vat) }
170
+
171
+ it 'deducts the default VAT from the cost' do
172
+ shipping_rates = subject.shipping_rates(package)
173
+ # deduct default vat: 4.00 / 1.2 = 3.33 (rounded)
174
+ expect(shipping_rates.first.cost).to eq(3.33)
175
+ end
176
+ end
177
+
178
+ context "when the order's tax zone is a zone with VAT outside the default zone" do
179
+ let(:other_vat_zone) { create(:zone_with_country) }
180
+ let!(:other_vat) do
181
+ create :tax_rate,
182
+ included_in_price: true,
183
+ zone: other_vat_zone,
184
+ amount: 0.3,
185
+ tax_category: shipping_method.tax_category
186
+ end
187
+
188
+ before { allow(order).to receive(:tax_zone).and_return(other_vat_zone) }
189
+
190
+ it 'deducts the default vat and applies the foreign vat to calculate the price' do
191
+ shipping_rates = subject.shipping_rates(package)
192
+ #
193
+ # deduct default vat: 4.00 / 1.2 = 3.33 (rounded)
194
+ # apply foreign vat: 3.33 * 1.3 = 4.33 (rounded)
195
+ expect(shipping_rates.first.cost).to eq(4.33)
196
+ end
197
+ end
198
+ end
151
199
  end
152
200
  end
153
201
  end