You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

test_forms.py 14KB

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