您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

test_forms.py 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343
  1. from decimal import Decimal as D
  2. from unittest import mock
  3. from django.conf import settings
  4. from django.test import TestCase, override_settings
  5. from oscar.apps.basket import forms, formsets
  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. AttributeOptionFactory, AttributeOptionGroupFactory, BenefitFactory,
  12. ConditionalOfferFactory, ConditionFactory, OptionFactory, RangeFactory)
  13. Line = get_model('basket', 'Line')
  14. Option = get_model('catalogue', 'Option')
  15. class TestBasketLineForm(TestCase):
  16. def setUp(self):
  17. self.applicator = Applicator()
  18. rng = RangeFactory(includes_all_products=True)
  19. self.condition = ConditionFactory(
  20. range=rng, type=ConditionFactory._meta.model.VALUE,
  21. value=D('100'), proxy_class=None)
  22. self.benefit = BenefitFactory(
  23. range=rng, type=BenefitFactory._meta.model.FIXED,
  24. value=D('10'), max_affected_items=1)
  25. self.basket = factories.create_basket()
  26. self.line = self.basket.all_lines()[0]
  27. def mock_availability_return_value(self, is_available, reason=''):
  28. policy = self.line.purchase_info.availability
  29. policy.is_purchase_permitted = mock.MagicMock(
  30. return_value=(is_available, reason))
  31. def build_form(self, quantity=None):
  32. if quantity is None:
  33. quantity = self.line.quantity
  34. return forms.BasketLineForm(
  35. strategy=self.basket.strategy,
  36. data={'quantity': quantity},
  37. instance=self.line)
  38. def test_interpret_empty_quantity_field_as_zero(self):
  39. form = self.build_form(quantity="")
  40. self.assertTrue(form.is_valid())
  41. self.assertEqual(form.cleaned_data['quantity'], 0)
  42. def test_enforces_availability_policy_for_valid_quantities(self):
  43. self.mock_availability_return_value(True)
  44. form = self.build_form()
  45. self.assertTrue(form.is_valid())
  46. def test_enforces_availability_policy_for_invalid_quantities(self):
  47. self.mock_availability_return_value(False, "Some reason")
  48. form = self.build_form()
  49. self.assertFalse(form.is_valid())
  50. self.assertEqual(
  51. form.errors['quantity'], ['Some reason'])
  52. def test_skips_availability_policy_for_zero_quantities(self):
  53. self.mock_availability_return_value(True)
  54. form = self.build_form(quantity=0)
  55. self.assertTrue(form.is_valid())
  56. def test_enforces_max_line_quantity_for_new_product(self):
  57. invalid_qty = settings.OSCAR_MAX_BASKET_QUANTITY_THRESHOLD + 1
  58. form = self.build_form(quantity=invalid_qty)
  59. self.assertFalse(form.is_valid())
  60. @override_settings(OSCAR_MAX_BASKET_QUANTITY_THRESHOLD=10)
  61. def test_enforce_max_line_quantity_for_existing_product(self):
  62. self.basket.flush()
  63. product = factories.create_product(num_in_stock=20)
  64. add_product(self.basket, D('100'), 4, product)
  65. self.line = self.basket.all_lines()[0]
  66. form = self.build_form(quantity=6)
  67. self.assertTrue(form.is_valid())
  68. form.save()
  69. # We set the _lines to None because the basket caches the lines here.
  70. # We want the basket to do the query again.
  71. # basket.num_items() will otherwise not return the correct values
  72. self.basket._lines = None
  73. form = self.build_form(quantity=11)
  74. self.assertFalse(form.is_valid())
  75. def test_line_quantity_max_attribute_per_num_available(self):
  76. self.basket.flush()
  77. product = factories.create_product(num_in_stock=20)
  78. add_product(self.basket, D('100'), 4, product)
  79. self.line = self.basket.all_lines()[0]
  80. form = self.build_form()
  81. self.assertIn('max="20"', str(form['quantity']))
  82. @override_settings(OSCAR_MAX_BASKET_QUANTITY_THRESHOLD=10)
  83. def test_line_quantity_max_attribute_per_basket_threshold(self):
  84. self.basket.flush()
  85. product = factories.create_product(num_in_stock=20)
  86. add_product(self.basket, D('100'), 4, product)
  87. self.line = self.basket.all_lines()[0]
  88. form = self.build_form()
  89. self.assertIn('max="6"', str(form['quantity']))
  90. def test_basketline_formset_ordering(self):
  91. # when we use a unordered queryset in the Basketlineformset, the
  92. # discounts will be lost because django will query the database
  93. # again to enforce ordered results
  94. add_product(self.basket, D('100'), 5)
  95. offer = ConditionalOfferFactory(
  96. pk=1, condition=self.condition, benefit=self.benefit)
  97. # now we force an unordered queryset so we can see that our discounts
  98. # will disappear due to a new ordering query (see django/forms/model.py)
  99. default_line_ordering = Line._meta.ordering
  100. Line._meta.ordering = []
  101. self.basket._lines = self.basket.lines.all()
  102. self.applicator.apply_offers(self.basket, [offer])
  103. formset = formsets.BasketLineFormSet(
  104. strategy=self.basket.strategy,
  105. queryset=self.basket.all_lines())
  106. # the discount is in all_lines():
  107. self.assertTrue(self.basket.all_lines()[0].has_discount)
  108. # but not in the formset
  109. self.assertFalse(formset.forms[0].instance.has_discount)
  110. # Restore the ordering on the line
  111. Line._meta.ordering = default_line_ordering
  112. # clear the cached lines and apply the offer again
  113. self.basket._lines = None
  114. self.applicator.apply_offers(self.basket, [offer])
  115. formset = formsets.BasketLineFormSet(
  116. strategy=self.basket.strategy,
  117. queryset=self.basket.all_lines())
  118. self.assertTrue(formset.forms[0].instance.has_discount)
  119. def test_max_allowed_quantity_with_options(self):
  120. self.basket.flush()
  121. option = OptionFactory(required=False)
  122. product = factories.create_product(num_in_stock=2)
  123. product.get_product_class().options.add(option)
  124. self.basket.add_product(product, options=[{"option": option, "value": "Test 1"}])
  125. self.basket.add_product(product, options=[{"option": option, "value": "Test 2"}])
  126. form = forms.BasketLineForm(
  127. strategy=self.basket.strategy,
  128. data={'quantity': 2},
  129. instance=self.basket.all_lines()[0]
  130. )
  131. self.assertFalse(form.is_valid())
  132. self.assertIn(
  133. "Available stock is only %s, which has been exceeded because multiple lines contain the same product." % 2,
  134. str(form.errors)
  135. )
  136. class TestAddToBasketForm(TestCase):
  137. def test_allows_a_product_quantity_to_be_increased(self):
  138. basket = factories.create_basket()
  139. product = basket.all_lines()[0].product
  140. # Add more of the same product
  141. data = {'quantity': 1}
  142. form = forms.AddToBasketForm(
  143. basket=basket, product=product, data=data)
  144. self.assertTrue(form.is_valid())
  145. def test_checks_whether_passed_product_id_matches_a_real_product(self):
  146. basket = factories.create_basket()
  147. product = basket.all_lines()[0].product
  148. # Add more of the same product
  149. data = {'quantity': -1}
  150. form = forms.AddToBasketForm(
  151. basket=basket, product=product, data=data)
  152. self.assertFalse(form.is_valid())
  153. def test_checks_if_purchase_is_permitted(self):
  154. basket = factories.BasketFactory()
  155. product = factories.ProductFactory()
  156. # Build a 4-level mock monster so we can force the return value of
  157. # whether the product is available to buy. This is a serious code smell
  158. # and needs to be remedied.
  159. info = mock.Mock()
  160. info.availability = mock.Mock()
  161. info.availability.is_purchase_permitted = mock.Mock(
  162. return_value=(False, "Not on your nelly!"))
  163. basket.strategy.fetch_for_product = mock.Mock(
  164. return_value=info)
  165. data = {'quantity': 1}
  166. form = forms.AddToBasketForm(
  167. basket=basket, product=product, data=data)
  168. self.assertFalse(form.is_valid())
  169. self.assertEqual('Not on your nelly!', form.errors['__all__'][0])
  170. def test_mixed_currency_baskets_are_not_permitted(self):
  171. # Ensure basket is one currency
  172. basket = mock.Mock()
  173. basket.currency = 'GBP'
  174. basket.num_items = 1
  175. # Ensure new product has different currency
  176. info = mock.Mock()
  177. info.price.currency = 'EUR'
  178. basket.strategy.fetch_for_product = mock.Mock(
  179. return_value=info)
  180. product = factories.ProductFactory()
  181. data = {'quantity': 1}
  182. form = forms.AddToBasketForm(
  183. basket=basket, product=product, data=data)
  184. self.assertFalse(form.is_valid())
  185. def test_cannot_add_a_product_without_price(self):
  186. basket = factories.BasketFactory()
  187. product = factories.create_product(price=None)
  188. data = {'quantity': 1}
  189. form = forms.AddToBasketForm(
  190. basket=basket, product=product, data=data)
  191. self.assertFalse(form.is_valid())
  192. self.assertEqual(
  193. form.errors['__all__'][0],
  194. 'This product cannot be added to the basket because a price '
  195. 'could not be determined for it.',
  196. )
  197. class TestAddToBasketWithOptionForm(TestCase):
  198. def setUp(self):
  199. self.basket = factories.create_basket(empty=True)
  200. self.product = factories.create_product(num_in_stock=1)
  201. def _get_basket_form(self, basket, product, data=None):
  202. return forms.AddToBasketForm(basket=basket, product=product, data=data)
  203. def test_basket_option_field_exists(self):
  204. option = OptionFactory()
  205. self.product.product_class.options.add(option)
  206. form = self._get_basket_form(basket=self.basket, product=self.product)
  207. self.assertIn(option.code, form.fields)
  208. def test_add_to_basket_with_not_required_option(self):
  209. option = OptionFactory(required=False)
  210. self.product.product_class.options.add(option)
  211. data = {'quantity': 1}
  212. form = self._get_basket_form(
  213. basket=self.basket, product=self.product, data=data,
  214. )
  215. self.assertTrue(form.is_valid())
  216. self.assertFalse(form.fields[option.code].required)
  217. def test_add_to_basket_with_required_option(self):
  218. option = OptionFactory(required=True)
  219. self.product.product_class.options.add(option)
  220. data = {'quantity': 1}
  221. invalid_form = self._get_basket_form(
  222. basket=self.basket, product=self.product, data=data,
  223. )
  224. self.assertFalse(invalid_form.is_valid())
  225. self.assertTrue(invalid_form.fields[option.code].required)
  226. data[option.code] = 'Test value'
  227. valid_form = self._get_basket_form(
  228. basket=self.basket, product=self.product, data=data,
  229. )
  230. self.assertTrue(valid_form.is_valid())
  231. def _test_add_to_basket_with_specific_option_type(
  232. self, option_type, invalid_value, valid_value
  233. ):
  234. if option_type in [Option.SELECT, Option.RADIO, Option.MULTI_SELECT, Option.CHECKBOX]:
  235. group = AttributeOptionGroupFactory(name="minte")
  236. AttributeOptionFactory(option="henk", group=group)
  237. AttributeOptionFactory(option="klaas", group=group)
  238. option = OptionFactory(required=True, type=option_type, option_group=group)
  239. else:
  240. option = OptionFactory(required=True, type=option_type)
  241. self.product.product_class.options.add(option)
  242. data = {'quantity': 1, option.code: invalid_value}
  243. invalid_form = self._get_basket_form(
  244. basket=self.basket, product=self.product, data=data,
  245. )
  246. self.assertFalse(invalid_form.is_valid())
  247. data[option.code] = valid_value
  248. valid_form = self._get_basket_form(
  249. basket=self.basket, product=self.product, data=data,
  250. )
  251. self.assertTrue(valid_form.is_valid())
  252. def test_add_to_basket_with_integer_option(self):
  253. self._test_add_to_basket_with_specific_option_type(
  254. Option.INTEGER, 1.55, 1,
  255. )
  256. def test_add_to_basket_with_float_option(self):
  257. self._test_add_to_basket_with_specific_option_type(
  258. Option.FLOAT, 'invalid_float', 1,
  259. )
  260. def test_add_to_basket_with_bool_option(self):
  261. self._test_add_to_basket_with_specific_option_type(
  262. Option.BOOLEAN, None, True,
  263. )
  264. def test_add_to_basket_with_date_option(self):
  265. self._test_add_to_basket_with_specific_option_type(
  266. Option.DATE, 'invalid_date', '2019-03-03',
  267. )
  268. def test_add_to_basket_with_select_option(self):
  269. self._test_add_to_basket_with_specific_option_type(
  270. Option.SELECT, 'invalid_select', 'henk',
  271. )
  272. def test_add_to_basket_with_radio_option(self):
  273. self._test_add_to_basket_with_specific_option_type(
  274. Option.RADIO, 'invalid_radio', 'henk',
  275. )
  276. def test_add_to_basket_with_multi_select_option(self):
  277. self._test_add_to_basket_with_specific_option_type(
  278. Option.MULTI_SELECT, ['invalid_multi_select'], ['henk', 'klaas'],
  279. )
  280. def test_add_to_basket_with_checkbox_option(self):
  281. self._test_add_to_basket_with_specific_option_type(
  282. Option.CHECKBOX, ['invalid_checkbox'], ['henk', 'klaas'],
  283. )