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_creator.py 16KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477
  1. import datetime
  2. import threading
  3. import time
  4. from decimal import Decimal as D
  5. import pytest
  6. from django.contrib.auth.models import AnonymousUser
  7. from django.http import HttpRequest
  8. from django.test import TestCase, TransactionTestCase
  9. from django.test.utils import override_settings
  10. from django.utils import timezone
  11. from oscar.apps.catalogue.models import Product, ProductClass
  12. from oscar.apps.checkout import calculators
  13. from oscar.apps.offer.utils import Applicator
  14. from oscar.apps.order.models import Order
  15. from oscar.apps.order.utils import OrderCreator
  16. from oscar.apps.shipping.methods import FixedPrice, Free
  17. from oscar.apps.shipping.repository import Repository
  18. from oscar.apps.voucher.models import Voucher
  19. from oscar.core.loading import get_class
  20. from oscar.test import factories
  21. from oscar.test.basket import add_product
  22. from oscar.test.utils import run_concurrently
  23. Range = get_class("offer.models", "Range")
  24. Benefit = get_class("offer.models", "Benefit")
  25. SurchargeApplicator = get_class("checkout.applicator", "SurchargeApplicator")
  26. def place_order(creator, **kwargs):
  27. """
  28. Helper function to place an order without the boilerplate
  29. """
  30. if "shipping_method" not in kwargs:
  31. kwargs["shipping_method"] = Free()
  32. shipping_charge = kwargs["shipping_method"].calculate(kwargs["basket"])
  33. kwargs["total"] = calculators.OrderTotalCalculator().calculate(
  34. basket=kwargs["basket"],
  35. shipping_charge=shipping_charge,
  36. surcharges=kwargs["surcharges"],
  37. )
  38. kwargs["shipping_charge"] = shipping_charge
  39. return creator.place_order(**kwargs)
  40. class TestOrderCreatorErrorCases(TestCase):
  41. def setUp(self):
  42. self.creator = OrderCreator()
  43. self.basket = factories.create_basket(empty=True)
  44. self.surcharges = SurchargeApplicator().get_applicable_surcharges(self.basket)
  45. def test_raises_exception_when_empty_basket_passed(self):
  46. with self.assertRaises(ValueError):
  47. place_order(self.creator, surcharges=self.surcharges, basket=self.basket)
  48. def test_raises_exception_if_duplicate_order_number_passed(self):
  49. add_product(self.basket, D("12.00"))
  50. place_order(
  51. self.creator,
  52. surcharges=self.surcharges,
  53. basket=self.basket,
  54. order_number="1234",
  55. )
  56. with self.assertRaises(ValueError):
  57. place_order(
  58. self.creator,
  59. surcharges=self.surcharges,
  60. basket=self.basket,
  61. order_number="1234",
  62. )
  63. class TestSuccessfulOrderCreation(TestCase):
  64. def setUp(self):
  65. self.creator = OrderCreator()
  66. self.basket = factories.create_basket(empty=True)
  67. self.surcharges = SurchargeApplicator().get_applicable_surcharges(self.basket)
  68. def test_saves_shipping_code(self):
  69. add_product(self.basket, D("12.00"))
  70. free_method = Free()
  71. order = place_order(
  72. self.creator,
  73. surcharges=self.surcharges,
  74. basket=self.basket,
  75. order_number="1234",
  76. shipping_method=free_method,
  77. )
  78. self.assertEqual(order.shipping_code, free_method.code)
  79. def test_creates_order_and_line_models(self):
  80. add_product(self.basket, D("12.00"))
  81. place_order(
  82. self.creator,
  83. surcharges=self.surcharges,
  84. basket=self.basket,
  85. order_number="1234",
  86. )
  87. order = Order.objects.get(number="1234")
  88. lines = order.lines.all()
  89. self.assertEqual(1, len(lines))
  90. def test_sets_correct_order_status(self):
  91. add_product(self.basket, D("12.00"))
  92. place_order(
  93. self.creator,
  94. surcharges=self.surcharges,
  95. basket=self.basket,
  96. order_number="1234",
  97. status="Active",
  98. )
  99. order = Order.objects.get(number="1234")
  100. self.assertEqual("Active", order.status)
  101. def test_defaults_to_using_free_shipping(self):
  102. add_product(self.basket, D("12.00"))
  103. place_order(
  104. self.creator,
  105. surcharges=self.surcharges,
  106. basket=self.basket,
  107. order_number="1234",
  108. )
  109. order = Order.objects.get(number="1234")
  110. self.assertEqual(
  111. order.total_incl_tax,
  112. self.basket.total_incl_tax + self.surcharges.total.incl_tax,
  113. )
  114. self.assertEqual(
  115. order.total_excl_tax,
  116. self.basket.total_excl_tax + self.surcharges.total.excl_tax,
  117. )
  118. def test_uses_default_order_status_from_settings(self):
  119. add_product(self.basket, D("12.00"))
  120. with override_settings(OSCAR_INITIAL_ORDER_STATUS="A"):
  121. place_order(
  122. self.creator,
  123. surcharges=self.surcharges,
  124. basket=self.basket,
  125. order_number="1234",
  126. )
  127. order = Order.objects.get(number="1234")
  128. self.assertEqual("A", order.status)
  129. def test_uses_default_line_status_from_settings(self):
  130. add_product(self.basket, D("12.00"))
  131. with override_settings(OSCAR_INITIAL_LINE_STATUS="A"):
  132. place_order(
  133. self.creator,
  134. surcharges=self.surcharges,
  135. basket=self.basket,
  136. order_number="1234",
  137. )
  138. order = Order.objects.get(number="1234")
  139. line = order.lines.all()[0]
  140. self.assertEqual("A", line.status)
  141. def test_partner_name_is_optional(self):
  142. for partner_name, order_number in [("", "A"), ("p1", "B")]:
  143. self.basket = factories.create_basket(empty=True)
  144. product = factories.create_product(partner_name=partner_name)
  145. add_product(self.basket, D("12.00"), product=product)
  146. place_order(
  147. self.creator,
  148. surcharges=self.surcharges,
  149. basket=self.basket,
  150. order_number=order_number,
  151. )
  152. line = Order.objects.get(number=order_number).lines.all()[0]
  153. partner = product.stockrecords.all()[0].partner
  154. self.assertTrue(partner_name == line.partner_name == partner.name)
  155. class TestPlacingOrderForDigitalGoods(TestCase):
  156. def setUp(self):
  157. self.creator = OrderCreator()
  158. self.basket = factories.create_basket(empty=True)
  159. self.surcharges = SurchargeApplicator().get_applicable_surcharges(self.basket)
  160. def test_does_not_allocate_stock(self):
  161. ProductClass.objects.create(name="Digital", track_stock=False)
  162. product = factories.create_product(product_class="Digital")
  163. record = factories.create_stockrecord(product, num_in_stock=None)
  164. self.assertTrue(record.num_allocated is None)
  165. add_product(self.basket, D("12.00"), product=product)
  166. place_order(
  167. self.creator,
  168. surcharges=self.surcharges,
  169. basket=self.basket,
  170. order_number="1234",
  171. )
  172. product = Product.objects.get(id=product.id)
  173. stockrecord = product.stockrecords.all()[0]
  174. self.assertTrue(stockrecord.num_in_stock is None)
  175. self.assertTrue(stockrecord.num_allocated is None)
  176. class TestShippingOfferForOrder(TestCase):
  177. def setUp(self):
  178. self.creator = OrderCreator()
  179. self.basket = factories.create_basket(empty=True)
  180. # add the product now so we can calculate the correct surcharges
  181. add_product(self.basket, D("12.00"))
  182. self.surcharges = SurchargeApplicator().get_applicable_surcharges(self.basket)
  183. def apply_20percent_shipping_offer(self):
  184. """Shipping offer 20% off"""
  185. product_range = Range.objects.create(
  186. name="All products range", includes_all_products=True
  187. )
  188. benefit = Benefit.objects.create(
  189. range=product_range, type=Benefit.SHIPPING_PERCENTAGE, value=20
  190. )
  191. offer = factories.create_offer(product_range=product_range, benefit=benefit)
  192. Applicator().apply_offers(self.basket, [offer])
  193. return offer
  194. def test_shipping_offer_is_applied(self):
  195. offer = self.apply_20percent_shipping_offer()
  196. shipping = FixedPrice(D("5.00"), D("5.00"))
  197. shipping = Repository().apply_shipping_offer(self.basket, shipping, offer)
  198. place_order(
  199. self.creator,
  200. surcharges=self.surcharges,
  201. basket=self.basket,
  202. order_number="1234",
  203. shipping_method=shipping,
  204. )
  205. order = Order.objects.get(number="1234")
  206. self.assertEqual(1, len(order.shipping_discounts))
  207. self.assertEqual(D("4.00"), order.shipping_incl_tax)
  208. self.assertEqual(D("38.00"), order.total_incl_tax)
  209. def test_zero_shipping_discount_is_not_created(self):
  210. offer = self.apply_20percent_shipping_offer()
  211. shipping = Free()
  212. shipping = Repository().apply_shipping_offer(self.basket, shipping, offer)
  213. place_order(
  214. self.creator,
  215. surcharges=self.surcharges,
  216. basket=self.basket,
  217. order_number="1234",
  218. shipping_method=shipping,
  219. )
  220. order = Order.objects.get(number="1234")
  221. # No shipping discount
  222. self.assertEqual(0, len(order.shipping_discounts))
  223. self.assertEqual(D("0.00"), order.shipping_incl_tax)
  224. self.assertEqual(D("34.00"), order.total_incl_tax)
  225. class TestMultiSiteOrderCreation(TestCase):
  226. def setUp(self):
  227. self.creator = OrderCreator()
  228. self.basket = factories.create_basket(empty=True)
  229. self.site1 = factories.SiteFactory()
  230. self.site2 = factories.SiteFactory()
  231. self.surcharges = SurchargeApplicator().get_applicable_surcharges(self.basket)
  232. def test_default_site(self):
  233. add_product(self.basket, D("12.00"))
  234. place_order(
  235. self.creator,
  236. surcharges=self.surcharges,
  237. basket=self.basket,
  238. order_number="1234",
  239. )
  240. order = Order.objects.get(number="1234")
  241. self.assertEqual(order.site_id, 1)
  242. def test_multi_sites(self):
  243. add_product(self.basket, D("12.00"))
  244. place_order(
  245. self.creator,
  246. surcharges=self.surcharges,
  247. basket=self.basket,
  248. order_number="12345",
  249. site=self.site1,
  250. )
  251. order1 = Order.objects.get(number="12345")
  252. self.assertEqual(order1.site, self.site1)
  253. add_product(self.basket, D("12.00"))
  254. place_order(
  255. self.creator,
  256. surcharges=self.surcharges,
  257. basket=self.basket,
  258. order_number="12346",
  259. site=self.site2,
  260. )
  261. order2 = Order.objects.get(number="12346")
  262. self.assertEqual(order2.site, self.site2)
  263. @override_settings(SITE_ID="")
  264. def test_request(self):
  265. request = HttpRequest()
  266. request.META["SERVER_PORT"] = 80
  267. request.META["SERVER_NAME"] = self.site1.domain
  268. add_product(self.basket, D("12.00"))
  269. place_order(
  270. self.creator,
  271. surcharges=self.surcharges,
  272. basket=self.basket,
  273. order_number="12345",
  274. request=request,
  275. )
  276. order1 = Order.objects.get(number="12345")
  277. self.assertEqual(order1.site, self.site1)
  278. add_product(self.basket, D("12.00"))
  279. request.META["SERVER_NAME"] = self.site2.domain
  280. place_order(
  281. self.creator,
  282. surcharges=self.surcharges,
  283. basket=self.basket,
  284. order_number="12346",
  285. request=request,
  286. )
  287. order2 = Order.objects.get(number="12346")
  288. self.assertEqual(order2.site, self.site2)
  289. class TestPlaceOrderWithVoucher(TestCase):
  290. def test_single_usage(self):
  291. user = AnonymousUser()
  292. basket = factories.create_basket()
  293. creator = OrderCreator()
  294. voucher = factories.VoucherFactory(usage=Voucher.SINGLE_USE)
  295. voucher.offers.add(factories.create_offer(offer_type="Voucher"))
  296. basket.vouchers.add(voucher)
  297. surcharges = SurchargeApplicator().get_applicable_surcharges(basket)
  298. place_order(
  299. creator,
  300. surcharges=surcharges,
  301. basket=basket,
  302. order_number="12346",
  303. user=user,
  304. )
  305. assert voucher.applications.count() == 1
  306. # Make sure the voucher usage is rechecked
  307. with pytest.raises(ValueError):
  308. place_order(
  309. creator,
  310. surcharges=surcharges,
  311. basket=basket,
  312. order_number="12347",
  313. user=user,
  314. )
  315. def test_expired_voucher(self):
  316. user = AnonymousUser()
  317. basket = factories.create_basket()
  318. creator = OrderCreator()
  319. voucher = factories.VoucherFactory(usage=Voucher.SINGLE_USE)
  320. voucher.offers.add(factories.create_offer(offer_type="Voucher"))
  321. basket.vouchers.add(voucher)
  322. voucher.end_datetime = timezone.now() - datetime.timedelta(days=100)
  323. voucher.save()
  324. surcharges = SurchargeApplicator().get_applicable_surcharges(basket)
  325. place_order(
  326. creator,
  327. surcharges=surcharges,
  328. basket=basket,
  329. order_number="12346",
  330. user=user,
  331. )
  332. assert voucher.applications.count() == 0
  333. class TestConcurrentOrderPlacement(TransactionTestCase):
  334. def test_single_usage(self):
  335. user = AnonymousUser()
  336. creator = OrderCreator()
  337. product = factories.ProductFactory(stockrecords__num_in_stock=1000)
  338. # Make the order creator a bit more slow too reliable trigger
  339. # concurrency issues
  340. org_create_order_model = OrderCreator.create_order_model
  341. def new_create_order_model(*args, **kwargs):
  342. time.sleep(0.5)
  343. return org_create_order_model(creator, *args, **kwargs)
  344. creator.create_order_model = new_create_order_model
  345. # Start 5 threads to place an order concurrently
  346. def worker():
  347. order_number = threading.current_thread().name
  348. basket = factories.BasketFactory()
  349. basket.add_product(product)
  350. surcharges = SurchargeApplicator().get_applicable_surcharges(basket)
  351. place_order(
  352. creator,
  353. surcharges=surcharges,
  354. basket=basket,
  355. order_number=order_number,
  356. user=user,
  357. )
  358. exceptions = run_concurrently(worker, num_threads=5)
  359. assert all(isinstance(x, ValueError) for x in exceptions), exceptions
  360. assert len(exceptions) == 0
  361. assert Order.objects.count() == 5
  362. stockrecord = product.stockrecords.first()
  363. assert stockrecord.num_allocated == 5
  364. def test_voucher_single_usage(self):
  365. user = AnonymousUser()
  366. creator = OrderCreator()
  367. product = factories.ProductFactory()
  368. voucher = factories.VoucherFactory(usage=Voucher.SINGLE_USE)
  369. voucher.offers.add(factories.create_offer(offer_type="Voucher"))
  370. # Make the order creator a bit more slow too reliable trigger
  371. # concurrency issues
  372. org_create_order_model = OrderCreator.create_order_model
  373. def new_create_order_model(*args, **kwargs):
  374. time.sleep(0.5)
  375. return org_create_order_model(creator, *args, **kwargs)
  376. creator.create_order_model = new_create_order_model
  377. org_record_voucher_usage = OrderCreator.record_voucher_usage
  378. def record_voucher_usage(*args, **kwargs):
  379. time.sleep(0.5)
  380. return org_record_voucher_usage(creator, *args, **kwargs)
  381. creator.record_voucher_usage = record_voucher_usage
  382. # Start 5 threads to place an order concurrently
  383. def worker():
  384. order_number = threading.current_thread().name
  385. basket = factories.BasketFactory()
  386. basket.add_product(product)
  387. basket.vouchers.add(voucher)
  388. surcharges = SurchargeApplicator().get_applicable_surcharges(basket)
  389. place_order(
  390. creator,
  391. surcharges=surcharges,
  392. basket=basket,
  393. order_number=order_number,
  394. user=user,
  395. )
  396. exceptions = run_concurrently(worker, num_threads=5)
  397. voucher.refresh_from_db()
  398. assert all(isinstance(x, ValueError) for x in exceptions), exceptions
  399. assert len(exceptions) == 4
  400. assert voucher.applications.count() == 1
  401. assert Order.objects.count() == 1