Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

form_tests.py 6.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. from decimal import Decimal as D
  2. from django.test import TestCase
  3. from django.conf import settings
  4. import mock
  5. from oscar.apps.basket import forms
  6. from oscar.apps.offer.utils import Applicator
  7. from oscar.core.loading import get_model
  8. from oscar.test import factories
  9. from oscar.test.basket import add_product
  10. from oscar.test.factories import (
  11. BenefitFactory, ConditionalOfferFactory, ConditionFactory, RangeFactory)
  12. Line = get_model('basket', 'Line')
  13. class TestBasketLineForm(TestCase):
  14. def setUp(self):
  15. self.applicator = Applicator()
  16. rng = RangeFactory(includes_all_products=True)
  17. self.condition = ConditionFactory(
  18. range=rng, type=ConditionFactory._meta.model.VALUE,
  19. value=D('100'), proxy_class=None)
  20. self.benefit = BenefitFactory(
  21. range=rng, type=BenefitFactory._meta.model.FIXED,
  22. value=D('10'), max_affected_items=1)
  23. self.basket = factories.create_basket()
  24. self.line = self.basket.all_lines()[0]
  25. def mock_availability_return_value(self, is_available, reason=''):
  26. policy = self.line.purchase_info.availability
  27. policy.is_purchase_permitted = mock.MagicMock(
  28. return_value=(is_available, reason))
  29. def build_form(self, quantity=None):
  30. if quantity is None:
  31. quantity = self.line.quantity
  32. return forms.BasketLineForm(
  33. strategy=self.basket.strategy,
  34. data={'quantity': quantity},
  35. instance=self.line)
  36. def test_enforces_availability_policy_for_valid_quantities(self):
  37. self.mock_availability_return_value(True)
  38. form = self.build_form()
  39. self.assertTrue(form.is_valid())
  40. def test_enforces_availability_policy_for_invalid_quantities(self):
  41. self.mock_availability_return_value(False, "Some reason")
  42. form = self.build_form()
  43. self.assertFalse(form.is_valid())
  44. self.assertEqual(
  45. form.errors['quantity'], ['Some reason'])
  46. def test_skips_availability_policy_for_zero_quantities(self):
  47. self.mock_availability_return_value(True)
  48. form = self.build_form(quantity=0)
  49. self.assertTrue(form.is_valid())
  50. def test_enforces_max_line_quantity(self):
  51. invalid_qty = settings.OSCAR_MAX_BASKET_QUANTITY_THRESHOLD + 1
  52. form = self.build_form(quantity=invalid_qty)
  53. self.assertFalse(form.is_valid())
  54. def test_basketline_formset_ordering(self):
  55. # when we use a unordered queryset in the Basketlineformset, the
  56. # discounts will be lost because django will query the database
  57. # again to enforce ordered results
  58. add_product(self.basket, D('100'), 5)
  59. offer = ConditionalOfferFactory(
  60. pk=1, condition=self.condition, benefit=self.benefit)
  61. # now we force an unordered queryset so we can see that our discounts
  62. # will disappear due to a new ordering query (see django/forms/model.py)
  63. default_line_ordering = Line._meta.ordering
  64. Line._meta.ordering = []
  65. self.basket._lines = self.basket.lines.all()
  66. self.applicator.apply_offers(self.basket, [offer])
  67. formset = forms.BasketLineFormSet(
  68. strategy=self.basket.strategy,
  69. queryset=self.basket.all_lines())
  70. # the discount is in all_lines():
  71. self.assertTrue(self.basket.all_lines()[0].has_discount)
  72. # but not in the formset
  73. self.assertFalse(formset.forms[0].instance.has_discount)
  74. # Restore the ordering on the line
  75. Line._meta.ordering = default_line_ordering
  76. # clear the cached lines and apply the offer again
  77. self.basket._lines = None
  78. self.applicator.apply_offers(self.basket, [offer])
  79. formset = forms.BasketLineFormSet(
  80. strategy=self.basket.strategy,
  81. queryset=self.basket.all_lines())
  82. self.assertTrue(formset.forms[0].instance.has_discount)
  83. class TestAddToBasketForm(TestCase):
  84. def test_allows_a_product_quantity_to_be_increased(self):
  85. basket = factories.create_basket()
  86. product = basket.all_lines()[0].product
  87. # Add more of the same product
  88. data = {'quantity': 1}
  89. form = forms.AddToBasketForm(
  90. basket=basket, product=product, data=data)
  91. self.assertTrue(form.is_valid())
  92. def test_checks_whether_passed_product_id_matches_a_real_product(self):
  93. basket = factories.create_basket()
  94. product = basket.all_lines()[0].product
  95. # Add more of the same product
  96. data = {'quantity': -1}
  97. form = forms.AddToBasketForm(
  98. basket=basket, product=product, data=data)
  99. self.assertFalse(form.is_valid())
  100. def test_checks_if_purchase_is_permitted(self):
  101. basket = factories.BasketFactory()
  102. product = factories.ProductFactory()
  103. # Build a 4-level mock monster so we can force the return value of
  104. # whether the product is available to buy. This is a serious code smell
  105. # and needs to be remedied.
  106. info = mock.Mock()
  107. info.availability = mock.Mock()
  108. info.availability.is_purchase_permitted = mock.Mock(
  109. return_value=(False, "Not on your nelly!"))
  110. basket.strategy.fetch_for_product = mock.Mock(
  111. return_value=info)
  112. data = {'quantity': 1}
  113. form = forms.AddToBasketForm(
  114. basket=basket, product=product, data=data)
  115. self.assertFalse(form.is_valid())
  116. self.assertEqual('Not on your nelly!', form.errors['__all__'][0])
  117. def test_mixed_currency_baskets_are_not_permitted(self):
  118. # Ensure basket is one currency
  119. basket = mock.Mock()
  120. basket.currency = 'GBP'
  121. basket.num_items = 1
  122. # Ensure new product has different currency
  123. info = mock.Mock()
  124. info.price.currency = 'EUR'
  125. basket.strategy.fetch_for_product = mock.Mock(
  126. return_value=info)
  127. product = factories.ProductFactory()
  128. data = {'quantity': 1}
  129. form = forms.AddToBasketForm(
  130. basket=basket, product=product, data=data)
  131. self.assertFalse(form.is_valid())