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

test_basket.py 23KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582
  1. import datetime
  2. from decimal import Decimal as D
  3. from http import client as http_client
  4. from http.cookies import _unquote
  5. import django
  6. from django.contrib.messages.storage import cookie
  7. from django.core import signing
  8. from django.test import TestCase, override_settings
  9. from django.urls import reverse
  10. from django.utils.translation import gettext
  11. from oscar.apps.basket import reports
  12. from oscar.apps.basket.models import Basket
  13. from oscar.apps.partner import strategy
  14. from oscar.core.compat import get_user_model
  15. from oscar.test import factories
  16. from oscar.test.basket import add_product
  17. from oscar.test.factories import OptionFactory, create_product
  18. from oscar.test.testcases import WebTestCase
  19. User = get_user_model()
  20. class TestBasketMerging(TestCase):
  21. def setUp(self):
  22. self.product = create_product(num_in_stock=10)
  23. self.user_basket = Basket()
  24. self.user_basket.strategy = strategy.Default()
  25. add_product(self.user_basket, product=self.product)
  26. self.cookie_basket = Basket()
  27. self.cookie_basket.strategy = strategy.Default()
  28. add_product(self.cookie_basket, quantity=2, product=self.product)
  29. self.user_basket.merge(self.cookie_basket, add_quantities=False)
  30. def test_cookie_basket_has_status_set(self):
  31. self.assertEqual(Basket.MERGED, self.cookie_basket.status)
  32. def test_lines_are_moved_across(self):
  33. self.assertEqual(1, self.user_basket.lines.all().count())
  34. def test_merge_line_takes_max_quantity(self):
  35. line = self.user_basket.lines.get(product=self.product)
  36. self.assertEqual(2, line.quantity)
  37. class AnonAddToBasketViewTests(WebTestCase):
  38. csrf_checks = False
  39. def setUp(self):
  40. self.product = create_product(price=D("10.00"), num_in_stock=10)
  41. url = reverse("basket:add", kwargs={"pk": self.product.pk})
  42. post_params = {"product_id": self.product.id, "action": "add", "quantity": 1}
  43. self.response = self.app.post(url, params=post_params)
  44. def test_cookie_is_created(self):
  45. self.assertTrue("oscar_open_basket" in self.response.test_app.cookies)
  46. def test_price_is_recorded(self):
  47. oscar_open_basket_cookie = _unquote(
  48. self.response.test_app.cookies["oscar_open_basket"]
  49. )
  50. basket_id = oscar_open_basket_cookie.split(":")[0]
  51. basket = Basket.objects.get(id=basket_id)
  52. line = basket.lines.get(product=self.product)
  53. stockrecord = self.product.stockrecords.all()[0]
  54. self.assertEqual(stockrecord.price, line.price_excl_tax)
  55. class BasketSummaryViewTests(WebTestCase):
  56. def setUp(self):
  57. url = reverse("basket:summary")
  58. self.response = self.app.get(url)
  59. def test_shipping_method_in_context(self):
  60. self.assertTrue("shipping_method" in self.response.context)
  61. def test_order_total_in_context(self):
  62. self.assertTrue("order_total" in self.response.context)
  63. def test_view_does_not_error(self):
  64. self.assertEqual(http_client.OK, self.response.status_code)
  65. def test_basket_in_context(self):
  66. self.assertTrue("basket" in self.response.context)
  67. def test_basket_is_empty(self):
  68. basket = self.response.context["basket"]
  69. self.assertEqual(0, basket.num_lines)
  70. class BasketThresholdTest(WebTestCase):
  71. csrf_checks = False
  72. @override_settings(OSCAR_MAX_BASKET_QUANTITY_THRESHOLD=3)
  73. def test_adding_more_than_threshold_raises(self):
  74. dummy_product = create_product(price=D("10.00"), num_in_stock=10)
  75. url = reverse("basket:add", kwargs={"pk": dummy_product.pk})
  76. post_params = {"product_id": dummy_product.id, "action": "add", "quantity": 2}
  77. response = self.app.post(url, params=post_params)
  78. # pylint: disable=no-member
  79. self.assertIn("oscar_open_basket", response.test_app.cookies)
  80. post_params = {"product_id": dummy_product.id, "action": "add", "quantity": 2}
  81. response = self.app.post(url, params=post_params)
  82. expected = gettext(
  83. "Due to technical limitations we are not able to ship more "
  84. "than %(threshold)d items in one order. Your basket currently "
  85. "has %(basket)d items."
  86. ) % ({"threshold": 3, "basket": 2})
  87. if django.VERSION < (3, 2):
  88. self.assertIn(expected, response.test_app.cookies["messages"])
  89. else:
  90. signer = signing.get_cookie_signer(salt="django.contrib.messages")
  91. message_strings = [
  92. m.message
  93. # pylint: disable=no-member
  94. for m in signer.unsign_object(
  95. response.test_app.cookies["messages"],
  96. serializer=cookie.MessageSerializer,
  97. )
  98. ]
  99. self.assertIn(expected, message_strings)
  100. class BasketReportTests(TestCase):
  101. def test_open_report_doesnt_error(self):
  102. data = {
  103. "start_date": datetime.date(2012, 5, 1),
  104. "end_date": datetime.date(2012, 5, 17),
  105. "formatter": "CSV",
  106. }
  107. generator = reports.OpenBasketReportGenerator(**data)
  108. generator.generate()
  109. def test_submitted_report_doesnt_error(self):
  110. data = {
  111. "start_date": datetime.date(2012, 5, 1),
  112. "end_date": datetime.date(2012, 5, 17),
  113. "formatter": "CSV",
  114. }
  115. generator = reports.SubmittedBasketReportGenerator(**data)
  116. generator.generate()
  117. class SavedBasketTests(WebTestCase):
  118. csrf_checks = False
  119. def test_moving_to_saved_basket_creates_new(self):
  120. self.user = factories.UserFactory()
  121. product = factories.ProductFactory()
  122. basket = factories.BasketFactory(owner=self.user)
  123. basket.add_product(product)
  124. response = self.get(reverse("basket:summary"))
  125. formset = response.context["formset"]
  126. form = formset.forms[0]
  127. data = {
  128. formset.add_prefix("INITIAL_FORMS"): 1,
  129. formset.add_prefix("TOTAL_FORMS"): 1,
  130. formset.add_prefix("MIN_FORMS"): 0,
  131. formset.add_prefix("MAX_NUM_FORMS"): 1,
  132. form.add_prefix("id"): form.instance.pk,
  133. form.add_prefix("quantity"): form.initial["quantity"],
  134. form.add_prefix("save_for_later"): True,
  135. }
  136. response = self.post(reverse("basket:summary"), params=data)
  137. self.assertRedirects(response, reverse("basket:summary"))
  138. self.assertFalse(Basket.open.get(pk=basket.pk).lines.exists())
  139. self.assertEqual(
  140. Basket.saved.get(owner=self.user).lines.get(product=product).quantity, 1
  141. )
  142. def test_moving_to_saved_basket_updates_existing(self):
  143. self.user = factories.UserFactory()
  144. product = factories.ProductFactory()
  145. basket = factories.BasketFactory(owner=self.user)
  146. basket.add_product(product)
  147. saved_basket = factories.BasketFactory(owner=self.user, status=Basket.SAVED)
  148. saved_basket.add_product(product)
  149. response = self.get(reverse("basket:summary"))
  150. formset = response.context["formset"]
  151. form = formset.forms[0]
  152. data = {
  153. formset.add_prefix("INITIAL_FORMS"): 1,
  154. formset.add_prefix("TOTAL_FORMS"): 1,
  155. formset.add_prefix("MIN_FORMS"): 0,
  156. formset.add_prefix("MAX_NUM_FORMS"): 1,
  157. form.add_prefix("id"): form.instance.pk,
  158. form.add_prefix("quantity"): form.initial["quantity"],
  159. form.add_prefix("save_for_later"): True,
  160. }
  161. response = self.post(reverse("basket:summary"), params=data)
  162. self.assertRedirects(response, reverse("basket:summary"))
  163. self.assertFalse(Basket.open.get(pk=basket.pk).lines.exists())
  164. self.assertEqual(
  165. Basket.saved.get(pk=saved_basket.pk).lines.get(product=product).quantity, 2
  166. )
  167. def test_moving_from_saved_basket(self):
  168. self.user = User.objects.create_user(
  169. username="test", password="pass", email="test@example.com"
  170. )
  171. product = create_product(price=D("10.00"), num_in_stock=2)
  172. basket = factories.create_basket(empty=True)
  173. basket.owner = self.user
  174. basket.save()
  175. add_product(basket, product=product)
  176. saved_basket, _ = Basket.saved.get_or_create(owner=self.user)
  177. saved_basket.strategy = basket.strategy
  178. add_product(saved_basket, product=product)
  179. response = self.get(reverse("basket:summary"))
  180. saved_formset = response.context["saved_formset"]
  181. saved_form = saved_formset.forms[0]
  182. data = {
  183. saved_formset.add_prefix("INITIAL_FORMS"): 1,
  184. saved_formset.add_prefix("MAX_NUM_FORMS"): 1,
  185. saved_formset.add_prefix("TOTAL_FORMS"): 1,
  186. saved_form.add_prefix("id"): saved_form.initial["id"],
  187. saved_form.add_prefix("move_to_basket"): True,
  188. }
  189. response = self.post(reverse("basket:saved"), params=data)
  190. self.assertEqual(
  191. Basket.open.get(id=basket.id).lines.get(product=product).quantity, 2
  192. )
  193. self.assertRedirects(response, reverse("basket:summary"))
  194. def test_moving_from_saved_basket_more_than_stocklevel_raises(self):
  195. self.user = User.objects.create_user(
  196. username="test", password="pass", email="test@example.com"
  197. )
  198. product = create_product(price=D("10.00"), num_in_stock=1)
  199. basket, _ = Basket.open.get_or_create(owner=self.user)
  200. add_product(basket, product=product)
  201. saved_basket, _ = Basket.saved.get_or_create(owner=self.user)
  202. add_product(saved_basket, product=product)
  203. response = self.get(reverse("basket:summary"))
  204. saved_formset = response.context["saved_formset"]
  205. saved_form = saved_formset.forms[0]
  206. data = {
  207. saved_formset.add_prefix("INITIAL_FORMS"): 1,
  208. saved_formset.add_prefix("MAX_NUM_FORMS"): 1,
  209. saved_formset.add_prefix("TOTAL_FORMS"): 1,
  210. saved_form.add_prefix("id"): saved_form.initial["id"],
  211. saved_form.add_prefix("move_to_basket"): True,
  212. }
  213. response = self.post(reverse("basket:saved"), params=data)
  214. # we can't add more than stock level into basket
  215. self.assertEqual(
  216. Basket.open.get(id=basket.id).lines.get(product=product).quantity, 1
  217. )
  218. self.assertRedirects(response, reverse("basket:summary"))
  219. class BasketFormSetTests(WebTestCase):
  220. csrf_checks = False
  221. def test_formset_with_removed_line(self):
  222. products = [create_product() for i in range(3)]
  223. basket = factories.create_basket(empty=True)
  224. basket.owner = self.user
  225. basket.save()
  226. add_product(basket, product=products[0])
  227. add_product(basket, product=products[1])
  228. add_product(basket, product=products[2])
  229. response = self.get(reverse("basket:summary"))
  230. formset = response.context["formset"]
  231. self.assertEqual(len(formset.forms), 3)
  232. basket.lines.filter(product=products[0]).delete()
  233. management_form = formset.management_form
  234. data = {
  235. formset.add_prefix("INITIAL_FORMS"): management_form.initial[
  236. "INITIAL_FORMS"
  237. ],
  238. formset.add_prefix("MAX_NUM_FORMS"): management_form.initial[
  239. "MAX_NUM_FORMS"
  240. ],
  241. formset.add_prefix("TOTAL_FORMS"): management_form.initial["TOTAL_FORMS"],
  242. "form-0-quantity": 1,
  243. "form-0-id": formset.forms[0].instance.id,
  244. "form-1-quantity": 2,
  245. "form-1-id": formset.forms[1].instance.id,
  246. "form-2-quantity": 2,
  247. "form-2-id": formset.forms[2].instance.id,
  248. }
  249. response = self.post(reverse("basket:summary"), params=data)
  250. self.assertEqual(response.status_code, 302)
  251. formset = response.follow().context["formset"]
  252. self.assertEqual(len(formset.forms), 2)
  253. self.assertEqual(len(formset.forms_with_instances), 2)
  254. self.assertEqual(basket.lines.all()[0].quantity, 2)
  255. self.assertEqual(basket.lines.all()[1].quantity, 2)
  256. def test_invalid_formset_with_removed_line(self):
  257. products = [create_product() for i in range(3)]
  258. basket = factories.create_basket(empty=True)
  259. basket.owner = self.user
  260. basket.save()
  261. add_product(basket, product=products[0])
  262. add_product(basket, product=products[1])
  263. add_product(basket, product=products[2])
  264. response = self.get(reverse("basket:summary"))
  265. formset = response.context["formset"]
  266. self.assertEqual(len(formset.forms), 3)
  267. basket.lines.filter(product=products[0]).delete()
  268. stockrecord = products[1].stockrecords.first()
  269. stockrecord.num_in_stock = 0
  270. stockrecord.save()
  271. management_form = formset.management_form
  272. data = {
  273. formset.add_prefix("INITIAL_FORMS"): management_form.initial[
  274. "INITIAL_FORMS"
  275. ],
  276. formset.add_prefix("MIN_NUM_FORMS"): management_form.initial[
  277. "MIN_NUM_FORMS"
  278. ],
  279. formset.add_prefix("MAX_NUM_FORMS"): management_form.initial[
  280. "MAX_NUM_FORMS"
  281. ],
  282. formset.add_prefix("TOTAL_FORMS"): management_form.initial["TOTAL_FORMS"],
  283. "form-0-quantity": 1,
  284. "form-0-id": formset.forms[0].instance.id,
  285. "form-1-quantity": 2,
  286. "form-1-id": formset.forms[1].instance.id,
  287. "form-2-quantity": 2,
  288. "form-2-id": formset.forms[2].instance.id,
  289. }
  290. response = self.post(reverse("basket:summary"), params=data)
  291. self.assertEqual(response.status_code, 200)
  292. formset = response.context["formset"]
  293. self.assertEqual(len(formset.forms), 2)
  294. self.assertEqual(len(formset.forms_with_instances), 2)
  295. self.assertEqual(basket.lines.all()[0].quantity, 1)
  296. self.assertEqual(basket.lines.all()[1].quantity, 1)
  297. def test_deleting_valid_line_with_other_valid_line(self):
  298. product_1 = create_product()
  299. product_2 = create_product()
  300. basket = factories.create_basket(empty=True)
  301. basket.owner = self.user
  302. basket.save()
  303. add_product(basket, product=product_1)
  304. add_product(basket, product=product_2)
  305. response = self.get(reverse("basket:summary"))
  306. formset = response.context["formset"]
  307. self.assertEqual(len(formset.forms), 2)
  308. data = {
  309. formset.add_prefix("TOTAL_FORMS"): formset.management_form.initial[
  310. "TOTAL_FORMS"
  311. ],
  312. formset.add_prefix("INITIAL_FORMS"): formset.management_form.initial[
  313. "INITIAL_FORMS"
  314. ],
  315. formset.add_prefix("MIN_NUM_FORMS"): formset.management_form.initial[
  316. "MIN_NUM_FORMS"
  317. ],
  318. formset.add_prefix("MAX_NUM_FORMS"): formset.management_form.initial[
  319. "MAX_NUM_FORMS"
  320. ],
  321. formset.forms[0].add_prefix("id"): formset.forms[0].instance.pk,
  322. formset.forms[0].add_prefix("quantity"): formset.forms[0].instance.quantity,
  323. formset.forms[0].add_prefix("DELETE"): "on",
  324. formset.forms[1].add_prefix("id"): formset.forms[1].instance.pk,
  325. formset.forms[1].add_prefix("quantity"): formset.forms[1].instance.quantity,
  326. }
  327. response = self.post(reverse("basket:summary"), params=data, xhr=True)
  328. self.assertEqual(response.status_code, 200)
  329. self.assertEqual(len(response.context["formset"].forms), 1)
  330. self.assertFalse(
  331. response.context["formset"].is_bound
  332. ) # new formset is rendered
  333. self.assertEqual(basket.lines.count(), 1)
  334. self.assertEqual(basket.lines.all()[0].quantity, 1)
  335. def test_deleting_valid_line_with_other_invalid_line(self):
  336. product_1 = create_product()
  337. product_2 = create_product()
  338. basket = factories.create_basket(empty=True)
  339. basket.owner = self.user
  340. basket.save()
  341. add_product(basket, product=product_1)
  342. add_product(basket, product=product_2)
  343. response = self.get(reverse("basket:summary"))
  344. formset = response.context["formset"]
  345. self.assertEqual(len(formset.forms), 2)
  346. # Render product for other line out of stock
  347. product_2.stockrecords.update(num_in_stock=0)
  348. data = {
  349. formset.add_prefix("TOTAL_FORMS"): formset.management_form.initial[
  350. "TOTAL_FORMS"
  351. ],
  352. formset.add_prefix("INITIAL_FORMS"): formset.management_form.initial[
  353. "INITIAL_FORMS"
  354. ],
  355. formset.add_prefix("MIN_NUM_FORMS"): formset.management_form.initial[
  356. "MIN_NUM_FORMS"
  357. ],
  358. formset.add_prefix("MAX_NUM_FORMS"): formset.management_form.initial[
  359. "MAX_NUM_FORMS"
  360. ],
  361. formset.forms[0].add_prefix("id"): formset.forms[0].instance.pk,
  362. formset.forms[0].add_prefix("quantity"): formset.forms[0].instance.quantity,
  363. formset.forms[0].add_prefix("DELETE"): "on",
  364. formset.forms[1].add_prefix("id"): formset.forms[1].instance.pk,
  365. formset.forms[1].add_prefix("quantity"): formset.forms[1].instance.quantity,
  366. }
  367. response = self.post(reverse("basket:summary"), params=data, xhr=True)
  368. self.assertEqual(response.status_code, 200)
  369. self.assertEqual(len(response.context["formset"].forms), 1)
  370. self.assertTrue(
  371. response.context["formset"].is_bound
  372. ) # formset with errors is rendered
  373. self.assertFalse(response.context["formset"].forms[0].is_valid())
  374. self.assertEqual(basket.lines.count(), 1)
  375. self.assertEqual(basket.lines.all()[0].quantity, 1)
  376. def test_deleting_invalid_line_with_other_valid_line(self):
  377. product_1 = create_product()
  378. product_2 = create_product()
  379. basket = factories.create_basket(empty=True)
  380. basket.owner = self.user
  381. basket.save()
  382. add_product(basket, product=product_1)
  383. add_product(basket, product=product_2)
  384. response = self.get(reverse("basket:summary"))
  385. formset = response.context["formset"]
  386. self.assertEqual(len(formset.forms), 2)
  387. # Render product for to-be-deleted line out of stock
  388. product_1.stockrecords.update(num_in_stock=0)
  389. data = {
  390. formset.add_prefix("TOTAL_FORMS"): formset.management_form.initial[
  391. "TOTAL_FORMS"
  392. ],
  393. formset.add_prefix("INITIAL_FORMS"): formset.management_form.initial[
  394. "INITIAL_FORMS"
  395. ],
  396. formset.add_prefix("MIN_NUM_FORMS"): formset.management_form.initial[
  397. "MIN_NUM_FORMS"
  398. ],
  399. formset.add_prefix("MAX_NUM_FORMS"): formset.management_form.initial[
  400. "MAX_NUM_FORMS"
  401. ],
  402. formset.forms[0].add_prefix("id"): formset.forms[0].instance.pk,
  403. formset.forms[0].add_prefix("quantity"): formset.forms[0].instance.quantity,
  404. formset.forms[0].add_prefix("DELETE"): "on",
  405. formset.forms[1].add_prefix("id"): formset.forms[1].instance.pk,
  406. formset.forms[1].add_prefix("quantity"): formset.forms[1].instance.quantity,
  407. }
  408. response = self.post(reverse("basket:summary"), params=data, xhr=True)
  409. self.assertEqual(response.status_code, 200)
  410. self.assertEqual(len(response.context["formset"].forms), 1)
  411. self.assertFalse(
  412. response.context["formset"].is_bound
  413. ) # new formset is rendered
  414. self.assertEqual(basket.lines.count(), 1)
  415. self.assertEqual(basket.lines.all()[0].quantity, 1)
  416. def test_deleting_invalid_line_with_other_invalid_line(self):
  417. product_1 = create_product()
  418. product_2 = create_product()
  419. basket = factories.create_basket(empty=True)
  420. basket.owner = self.user
  421. basket.save()
  422. add_product(basket, product=product_1)
  423. add_product(basket, product=product_2)
  424. response = self.get(reverse("basket:summary"))
  425. formset = response.context["formset"]
  426. self.assertEqual(len(formset.forms), 2)
  427. # Render products for both lines out of stock
  428. product_1.stockrecords.update(num_in_stock=0)
  429. product_2.stockrecords.update(num_in_stock=0)
  430. data = {
  431. formset.add_prefix("TOTAL_FORMS"): formset.management_form.initial[
  432. "TOTAL_FORMS"
  433. ],
  434. formset.add_prefix("INITIAL_FORMS"): formset.management_form.initial[
  435. "INITIAL_FORMS"
  436. ],
  437. formset.add_prefix("MIN_NUM_FORMS"): formset.management_form.initial[
  438. "MIN_NUM_FORMS"
  439. ],
  440. formset.add_prefix("MAX_NUM_FORMS"): formset.management_form.initial[
  441. "MAX_NUM_FORMS"
  442. ],
  443. formset.forms[0].add_prefix("id"): formset.forms[0].instance.pk,
  444. formset.forms[0].add_prefix("quantity"): formset.forms[0].instance.quantity,
  445. formset.forms[0].add_prefix("DELETE"): "on",
  446. formset.forms[1].add_prefix("id"): formset.forms[1].instance.pk,
  447. formset.forms[1].add_prefix("quantity"): formset.forms[1].instance.quantity,
  448. }
  449. response = self.post(reverse("basket:summary"), params=data, xhr=True)
  450. self.assertEqual(response.status_code, 200)
  451. self.assertEqual(len(response.context["formset"].forms), 1)
  452. self.assertTrue(
  453. response.context["formset"].is_bound
  454. ) # formset with errors is rendered
  455. self.assertFalse(response.context["formset"].forms[0].is_valid())
  456. self.assertEqual(basket.lines.count(), 1)
  457. self.assertEqual(basket.lines.all()[0].quantity, 1)
  458. def test_formset_quantity_update_with_options(self):
  459. product = create_product(num_in_stock=2)
  460. option = OptionFactory()
  461. # Add the option to the product class
  462. product.get_product_class().options.add(option)
  463. basket = factories.create_basket(empty=True)
  464. basket.owner = self.user
  465. basket.save()
  466. basket.add_product(product, options=[{"option": option, "value": "Test 1"}])
  467. basket.add_product(product, options=[{"option": option, "value": "Test 2"}])
  468. response = self.get(reverse("basket:summary"))
  469. formset = response.context["formset"]
  470. self.assertEqual(len(formset.forms), 2)
  471. # Now update one of the quantities to 2
  472. data = {
  473. formset.add_prefix("TOTAL_FORMS"): formset.management_form.initial[
  474. "TOTAL_FORMS"
  475. ],
  476. formset.add_prefix("INITIAL_FORMS"): formset.management_form.initial[
  477. "INITIAL_FORMS"
  478. ],
  479. formset.add_prefix("MIN_NUM_FORMS"): formset.management_form.initial[
  480. "MIN_NUM_FORMS"
  481. ],
  482. formset.add_prefix("MAX_NUM_FORMS"): formset.management_form.initial[
  483. "MAX_NUM_FORMS"
  484. ],
  485. formset.forms[0].add_prefix("id"): formset.forms[0].instance.pk,
  486. formset.forms[0].add_prefix("quantity"): 2,
  487. formset.forms[1].add_prefix("id"): formset.forms[1].instance.pk,
  488. formset.forms[1].add_prefix("quantity"): formset.forms[1].instance.quantity,
  489. }
  490. response = self.post(reverse("basket:summary"), params=data, xhr=True)
  491. self.assertEqual(response.status_code, 200)
  492. self.assertFalse(response.context["formset"].forms[0].is_valid())
  493. self.assertIn(
  494. "Available stock is only %s, which has been exceeded because multiple lines contain the same product."
  495. % 2,
  496. str(response.context["formset"].forms[0].errors),
  497. )