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.

utils.py 7.7KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. from django.contrib.sites.models import Site
  2. from oscar.core.loading import import_module
  3. import_module('order.models', ['ShippingAddress', 'Order', 'Line',
  4. 'LinePrice', 'LineAttribute', 'OrderDiscount'], locals())
  5. import_module('order.signals', ['order_placed'], locals())
  6. class OrderNumberGenerator(object):
  7. """
  8. Simple object for generating order numbers.
  9. We need this as the order number is often required for payment
  10. which takes place before the order model has been created.
  11. """
  12. def order_number(self, basket):
  13. """
  14. Return an order number for a given basket
  15. """
  16. return 100000 + basket.id
  17. class OrderCreator(object):
  18. """
  19. Places the order by writing out the various models
  20. """
  21. def place_order(self, user, basket, shipping_address, shipping_method,
  22. billing_address, total_incl_tax, total_excl_tax,
  23. order_number=None, status=None, **kwargs):
  24. """
  25. Placing an order involves creating all the relevant models based on the
  26. basket and session data.
  27. """
  28. if not order_number:
  29. generator = OrderNumberGenerator()
  30. order_number = generator.order_number(basket)
  31. order = self.create_order_model(user, basket, shipping_address,
  32. shipping_method, billing_address, total_incl_tax,
  33. total_excl_tax, order_number, status, **kwargs)
  34. for line in basket.all_lines():
  35. self.create_line_models(order, line)
  36. self.update_stock_records(line)
  37. for discount in basket.get_discounts():
  38. self.create_discount_model(order, discount)
  39. for voucher in basket.vouchers.all():
  40. self.record_voucher_usage(order, voucher, user)
  41. # Send signal for analytics to pick up
  42. order_placed.send(sender=self, order=order, user=user)
  43. return order
  44. def create_order_model(self, user, basket, shipping_address, shipping_method,
  45. billing_address, total_incl_tax, total_excl_tax,
  46. order_number, status, **extra_order_fields):
  47. """
  48. Creates an order model.
  49. """
  50. order_data = {'basket': basket,
  51. 'number': order_number,
  52. 'site': Site._default_manager.get_current(),
  53. 'total_incl_tax': total_incl_tax,
  54. 'total_excl_tax': total_excl_tax,
  55. 'shipping_address': shipping_address,
  56. 'shipping_incl_tax': shipping_method.basket_charge_incl_tax(),
  57. 'shipping_excl_tax': shipping_method.basket_charge_excl_tax(),
  58. 'shipping_method': shipping_method.name}
  59. if billing_address:
  60. order_data['billing_address'] = billing_address
  61. if user.is_authenticated():
  62. order_data['user_id'] = user.id
  63. if status:
  64. order_data['status'] = status
  65. if extra_order_fields:
  66. order_data.update(extra_order_fields)
  67. order = Order(**order_data)
  68. order.save()
  69. return order
  70. def get_partner_for_product(self, product):
  71. """
  72. Return the partner for a product
  73. """
  74. if product.has_stockrecord:
  75. return product.stockrecord.partner
  76. raise AttributeError("No partner found for product '%s'" % product)
  77. def create_line_models(self, order, basket_line, extra_line_fields=None):
  78. """
  79. Create the batch line model.
  80. You can set extra fields by passing a dictionary as the extra_line_fields value
  81. """
  82. partner = self.get_partner_for_product(basket_line.product)
  83. stockrecord = basket_line.product.stockrecord
  84. line_data = {'order': order,
  85. # Partner details
  86. 'partner': partner,
  87. 'partner_name': partner.name,
  88. 'partner_sku': stockrecord.partner_sku,
  89. # Product details
  90. 'product': basket_line.product,
  91. 'title': basket_line.product.get_title(),
  92. 'quantity': basket_line.quantity,
  93. # Price details
  94. 'line_price_excl_tax': basket_line.line_price_excl_tax_and_discounts,
  95. 'line_price_incl_tax': basket_line.line_price_incl_tax_and_discounts,
  96. 'line_price_before_discounts_excl_tax': basket_line.line_price_excl_tax,
  97. 'line_price_before_discounts_incl_tax': basket_line.line_price_incl_tax,
  98. # Reporting details
  99. 'unit_cost_price': stockrecord.cost_price,
  100. 'unit_price_incl_tax': basket_line.unit_price_incl_tax,
  101. 'unit_price_excl_tax': basket_line.unit_price_excl_tax,
  102. 'unit_retail_price': stockrecord.price_retail,
  103. # Shipping details
  104. 'est_dispatch_date': basket_line.product.stockrecord.dispatch_date
  105. }
  106. if extra_line_fields:
  107. line_data.update(extra_line_fields)
  108. order_line = Line._default_manager.create(**line_data)
  109. self.create_line_price_models(order, order_line, basket_line)
  110. self.create_line_attributes(order, order_line, basket_line)
  111. self.create_additional_line_models(order, order_line, basket_line)
  112. return order_line
  113. def update_stock_records(self, line):
  114. line.product.stockrecord.allocate(line.quantity)
  115. def create_additional_line_models(self, order, order_line, basket_line):
  116. """
  117. Empty method designed to be overridden.
  118. Some applications require additional information about lines, this
  119. method provides a clean place to create additional models that
  120. relate to a given line.
  121. """
  122. pass
  123. def create_line_price_models(self, order, order_line, basket_line):
  124. """
  125. Creates the batch line price models
  126. """
  127. breakdown = basket_line.get_price_breakdown()
  128. for price_incl_tax, price_excl_tax, quantity in breakdown:
  129. LinePrice._default_manager.create(order=order,
  130. line=order_line,
  131. quantity=quantity,
  132. price_incl_tax=price_incl_tax,
  133. price_excl_tax=price_excl_tax)
  134. def create_line_attributes(self, order, order_line, basket_line):
  135. """
  136. Creates the batch line attributes.
  137. """
  138. for attr in basket_line.attributes.all():
  139. LineAttribute._default_manager.create(line=order_line,
  140. option=attr.option,
  141. type=attr.option.code,
  142. value=attr.value)
  143. def create_discount_model(self, order, discount):
  144. """
  145. Creates an order discount model for each discount attached to the basket.
  146. """
  147. order_discount = OrderDiscount(order=order, offer=discount['offer'], amount=discount['discount'])
  148. if discount['voucher']:
  149. order_discount.voucher = discount['voucher']
  150. order_discount.voucher_code = discount['voucher'].code
  151. order_discount.save()
  152. def record_voucher_usage(self, order, voucher, user):
  153. """
  154. Updates the models that care about this voucher.
  155. """
  156. voucher.record_usage(order, user)
  157. voucher.num_orders += 1
  158. voucher.save()