2016-01-22 05:01:30 +00:00
|
|
|
import datetime
|
|
|
|
import pytz
|
|
|
|
|
|
|
|
from decimal import Decimal
|
|
|
|
from django.core.exceptions import ValidationError
|
|
|
|
|
2016-04-22 05:06:24 +00:00
|
|
|
from registrasion.models import commerce
|
|
|
|
from registrasion.models import conditions
|
|
|
|
from registrasion.models import inventory
|
2016-04-07 00:19:18 +00:00
|
|
|
from controller_helpers import TestingCartController
|
2016-04-10 04:41:43 +00:00
|
|
|
from controller_helpers import TestingCreditNoteController
|
2016-04-07 00:23:38 +00:00
|
|
|
from controller_helpers import TestingInvoiceController
|
2016-01-22 05:01:30 +00:00
|
|
|
|
|
|
|
from test_cart import RegistrationCartTestCase
|
|
|
|
|
|
|
|
UTC = pytz.timezone('UTC')
|
|
|
|
|
|
|
|
|
|
|
|
class InvoiceTestCase(RegistrationCartTestCase):
|
|
|
|
|
2016-04-24 21:42:05 +00:00
|
|
|
def _invoice_containing_prod_1(self, qty=1):
|
|
|
|
cart = TestingCartController.for_user(self.USER_1)
|
|
|
|
cart.add_to_cart(self.PROD_1, qty)
|
|
|
|
|
|
|
|
return TestingInvoiceController.for_cart(self.reget(cart.cart))
|
|
|
|
|
2016-04-24 22:13:44 +00:00
|
|
|
def _credit_note_for_invoice(self, invoice):
|
|
|
|
note = commerce.CreditNote.objects.get(invoice=invoice)
|
|
|
|
return TestingCreditNoteController(note)
|
|
|
|
|
2016-01-22 05:01:30 +00:00
|
|
|
def test_create_invoice(self):
|
2016-04-03 00:06:35 +00:00
|
|
|
current_cart = TestingCartController.for_user(self.USER_1)
|
2016-01-22 05:01:30 +00:00
|
|
|
|
|
|
|
# Should be able to create an invoice after the product is added
|
|
|
|
current_cart.add_to_cart(self.PROD_1, 1)
|
2016-04-07 00:23:38 +00:00
|
|
|
invoice_1 = TestingInvoiceController.for_cart(current_cart.cart)
|
2016-01-22 05:01:30 +00:00
|
|
|
# That invoice should have a single line item
|
2016-04-22 05:06:24 +00:00
|
|
|
line_items = commerce.LineItem.objects.filter(
|
|
|
|
invoice=invoice_1.invoice,
|
|
|
|
)
|
2016-01-22 05:01:30 +00:00
|
|
|
self.assertEqual(1, len(line_items))
|
|
|
|
# That invoice should have a value equal to cost of PROD_1
|
|
|
|
self.assertEqual(self.PROD_1.price, invoice_1.invoice.value)
|
|
|
|
|
2016-03-27 03:04:47 +00:00
|
|
|
# Adding item to cart should produce a new invoice
|
2016-01-22 05:01:30 +00:00
|
|
|
current_cart.add_to_cart(self.PROD_2, 1)
|
2016-04-07 00:23:38 +00:00
|
|
|
invoice_2 = TestingInvoiceController.for_cart(current_cart.cart)
|
2016-01-22 05:01:30 +00:00
|
|
|
self.assertNotEqual(invoice_1.invoice, invoice_2.invoice)
|
2016-03-27 03:04:47 +00:00
|
|
|
|
|
|
|
# The old invoice should automatically be voided
|
2016-04-22 05:06:24 +00:00
|
|
|
invoice_1_new = commerce.Invoice.objects.get(pk=invoice_1.invoice.id)
|
|
|
|
invoice_2_new = commerce.Invoice.objects.get(pk=invoice_2.invoice.id)
|
2016-04-06 22:28:43 +00:00
|
|
|
self.assertTrue(invoice_1_new.is_void)
|
|
|
|
self.assertFalse(invoice_2_new.is_void)
|
2016-03-27 03:04:47 +00:00
|
|
|
|
2016-01-22 05:01:30 +00:00
|
|
|
# Invoice should have two line items
|
2016-04-22 05:06:24 +00:00
|
|
|
line_items = commerce.LineItem.objects.filter(
|
|
|
|
invoice=invoice_2.invoice,
|
|
|
|
)
|
2016-01-22 05:01:30 +00:00
|
|
|
self.assertEqual(2, len(line_items))
|
|
|
|
# Invoice should have a value equal to cost of PROD_1 and PROD_2
|
|
|
|
self.assertEqual(
|
|
|
|
self.PROD_1.price + self.PROD_2.price,
|
|
|
|
invoice_2.invoice.value)
|
|
|
|
|
2016-04-24 21:13:37 +00:00
|
|
|
def test_invoice_controller_for_id_works(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
2016-04-24 21:13:37 +00:00
|
|
|
|
|
|
|
id_ = invoice.invoice.id
|
|
|
|
|
|
|
|
invoice1 = TestingInvoiceController.for_id(id_)
|
|
|
|
invoice2 = TestingInvoiceController.for_id(str(id_))
|
|
|
|
|
|
|
|
self.assertEqual(invoice.invoice, invoice1.invoice)
|
|
|
|
self.assertEqual(invoice.invoice, invoice2.invoice)
|
|
|
|
|
2016-01-22 05:01:30 +00:00
|
|
|
def test_create_invoice_fails_if_cart_invalid(self):
|
|
|
|
self.make_ceiling("Limit ceiling", limit=1)
|
|
|
|
self.set_time(datetime.datetime(2015, 01, 01, tzinfo=UTC))
|
2016-04-03 00:06:35 +00:00
|
|
|
current_cart = TestingCartController.for_user(self.USER_1)
|
2016-01-22 05:01:30 +00:00
|
|
|
current_cart.add_to_cart(self.PROD_1, 1)
|
|
|
|
|
|
|
|
self.add_timedelta(self.RESERVATION * 2)
|
2016-04-03 00:06:35 +00:00
|
|
|
cart_2 = TestingCartController.for_user(self.USER_2)
|
2016-01-22 05:01:30 +00:00
|
|
|
cart_2.add_to_cart(self.PROD_1, 1)
|
|
|
|
|
|
|
|
# Now try to invoice the first user
|
|
|
|
with self.assertRaises(ValidationError):
|
2016-04-07 00:23:38 +00:00
|
|
|
TestingInvoiceController.for_cart(current_cart.cart)
|
2016-01-22 05:01:30 +00:00
|
|
|
|
|
|
|
def test_paying_invoice_makes_new_cart(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
2016-01-22 05:01:30 +00:00
|
|
|
|
|
|
|
invoice.pay("A payment!", invoice.invoice.value)
|
|
|
|
|
|
|
|
# This payment is for the correct amount invoice should be paid.
|
2016-04-06 22:28:43 +00:00
|
|
|
self.assertTrue(invoice.invoice.is_paid)
|
2016-01-22 05:01:30 +00:00
|
|
|
|
|
|
|
# Cart should not be active
|
2016-04-25 04:31:25 +00:00
|
|
|
self.assertNotEqual(
|
|
|
|
commerce.Cart.STATUS_ACTIVE,
|
|
|
|
invoice.invoice.cart.status,
|
|
|
|
)
|
2016-01-22 05:01:30 +00:00
|
|
|
|
|
|
|
# Asking for a cart should generate a new one
|
2016-04-03 00:06:35 +00:00
|
|
|
new_cart = TestingCartController.for_user(self.USER_1)
|
2016-04-24 21:42:05 +00:00
|
|
|
self.assertNotEqual(invoice.invoice.cart, new_cart.cart)
|
2016-01-22 05:01:30 +00:00
|
|
|
|
|
|
|
def test_invoice_includes_discounts(self):
|
2016-04-22 05:06:24 +00:00
|
|
|
voucher = inventory.Voucher.objects.create(
|
2016-01-22 05:01:30 +00:00
|
|
|
recipient="Voucher recipient",
|
|
|
|
code="VOUCHER",
|
|
|
|
limit=1
|
|
|
|
)
|
2016-04-22 05:06:24 +00:00
|
|
|
discount = conditions.VoucherDiscount.objects.create(
|
2016-01-22 05:01:30 +00:00
|
|
|
description="VOUCHER RECIPIENT",
|
|
|
|
voucher=voucher,
|
|
|
|
)
|
2016-04-22 05:06:24 +00:00
|
|
|
conditions.DiscountForProduct.objects.create(
|
2016-01-22 05:01:30 +00:00
|
|
|
discount=discount,
|
|
|
|
product=self.PROD_1,
|
|
|
|
percentage=Decimal(50),
|
|
|
|
quantity=1
|
2016-04-06 07:19:09 +00:00
|
|
|
)
|
2016-01-22 05:01:30 +00:00
|
|
|
|
2016-04-03 00:06:35 +00:00
|
|
|
current_cart = TestingCartController.for_user(self.USER_1)
|
2016-03-23 02:29:18 +00:00
|
|
|
current_cart.apply_voucher(voucher.code)
|
2016-01-22 05:01:30 +00:00
|
|
|
|
|
|
|
# Should be able to create an invoice after the product is added
|
|
|
|
current_cart.add_to_cart(self.PROD_1, 1)
|
2016-04-07 00:23:38 +00:00
|
|
|
invoice_1 = TestingInvoiceController.for_cart(current_cart.cart)
|
2016-01-22 05:01:30 +00:00
|
|
|
|
|
|
|
# That invoice should have two line items
|
2016-04-22 05:06:24 +00:00
|
|
|
line_items = commerce.LineItem.objects.filter(
|
|
|
|
invoice=invoice_1.invoice,
|
|
|
|
)
|
2016-01-22 05:01:30 +00:00
|
|
|
self.assertEqual(2, len(line_items))
|
|
|
|
# That invoice should have a value equal to 50% of the cost of PROD_1
|
2016-01-22 06:02:07 +00:00
|
|
|
self.assertEqual(
|
|
|
|
self.PROD_1.price * Decimal("0.5"),
|
|
|
|
invoice_1.invoice.value)
|
2016-03-27 03:04:47 +00:00
|
|
|
|
2016-04-06 07:19:09 +00:00
|
|
|
def test_zero_value_invoice_is_automatically_paid(self):
|
2016-04-22 05:06:24 +00:00
|
|
|
voucher = inventory.Voucher.objects.create(
|
2016-04-06 07:19:09 +00:00
|
|
|
recipient="Voucher recipient",
|
|
|
|
code="VOUCHER",
|
|
|
|
limit=1
|
|
|
|
)
|
2016-04-22 05:06:24 +00:00
|
|
|
discount = conditions.VoucherDiscount.objects.create(
|
2016-04-06 07:19:09 +00:00
|
|
|
description="VOUCHER RECIPIENT",
|
|
|
|
voucher=voucher,
|
|
|
|
)
|
2016-04-22 05:06:24 +00:00
|
|
|
conditions.DiscountForProduct.objects.create(
|
2016-04-06 07:19:09 +00:00
|
|
|
discount=discount,
|
|
|
|
product=self.PROD_1,
|
|
|
|
percentage=Decimal(100),
|
|
|
|
quantity=1
|
|
|
|
)
|
|
|
|
|
|
|
|
current_cart = TestingCartController.for_user(self.USER_1)
|
|
|
|
current_cart.apply_voucher(voucher.code)
|
|
|
|
|
|
|
|
# Should be able to create an invoice after the product is added
|
|
|
|
current_cart.add_to_cart(self.PROD_1, 1)
|
2016-04-07 00:23:38 +00:00
|
|
|
invoice_1 = TestingInvoiceController.for_cart(current_cart.cart)
|
2016-04-06 07:19:09 +00:00
|
|
|
|
2016-04-06 22:28:43 +00:00
|
|
|
self.assertTrue(invoice_1.invoice.is_paid)
|
2016-04-06 07:19:09 +00:00
|
|
|
|
2016-03-27 03:04:47 +00:00
|
|
|
def test_invoice_voids_self_if_cart_is_invalid(self):
|
2016-04-03 00:06:35 +00:00
|
|
|
current_cart = TestingCartController.for_user(self.USER_1)
|
2016-03-27 03:04:47 +00:00
|
|
|
|
|
|
|
# Should be able to create an invoice after the product is added
|
|
|
|
current_cart.add_to_cart(self.PROD_1, 1)
|
2016-04-07 00:23:38 +00:00
|
|
|
invoice_1 = TestingInvoiceController.for_cart(current_cart.cart)
|
2016-03-27 03:04:47 +00:00
|
|
|
|
2016-04-06 22:28:43 +00:00
|
|
|
self.assertFalse(invoice_1.invoice.is_void)
|
2016-03-27 03:04:47 +00:00
|
|
|
|
|
|
|
# Adding item to cart should produce a new invoice
|
|
|
|
current_cart.add_to_cart(self.PROD_2, 1)
|
2016-04-07 00:23:38 +00:00
|
|
|
invoice_2 = TestingInvoiceController.for_cart(current_cart.cart)
|
2016-03-27 03:04:47 +00:00
|
|
|
self.assertNotEqual(invoice_1.invoice, invoice_2.invoice)
|
|
|
|
|
|
|
|
# Viewing invoice_1's invoice should show it as void
|
2016-04-07 00:23:38 +00:00
|
|
|
invoice_1_new = TestingInvoiceController(invoice_1.invoice)
|
2016-04-06 22:28:43 +00:00
|
|
|
self.assertTrue(invoice_1_new.invoice.is_void)
|
2016-03-27 03:04:47 +00:00
|
|
|
|
|
|
|
# Viewing invoice_2's invoice should *not* show it as void
|
2016-04-07 00:23:38 +00:00
|
|
|
invoice_2_new = TestingInvoiceController(invoice_2.invoice)
|
2016-04-06 22:28:43 +00:00
|
|
|
self.assertFalse(invoice_2_new.invoice.is_void)
|
2016-03-27 03:41:43 +00:00
|
|
|
|
|
|
|
def test_voiding_invoice_creates_new_invoice(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice_1 = self._invoice_containing_prod_1(1)
|
2016-03-27 03:41:43 +00:00
|
|
|
|
2016-04-06 22:28:43 +00:00
|
|
|
self.assertFalse(invoice_1.invoice.is_void)
|
2016-03-27 03:41:43 +00:00
|
|
|
invoice_1.void()
|
|
|
|
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice_2 = TestingInvoiceController.for_cart(invoice_1.invoice.cart)
|
2016-03-27 03:41:43 +00:00
|
|
|
self.assertNotEqual(invoice_1.invoice, invoice_2.invoice)
|
|
|
|
|
|
|
|
def test_cannot_pay_void_invoice(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice_1 = self._invoice_containing_prod_1(1)
|
2016-03-27 03:41:43 +00:00
|
|
|
|
|
|
|
invoice_1.void()
|
|
|
|
|
|
|
|
with self.assertRaises(ValidationError):
|
2016-04-07 08:26:31 +00:00
|
|
|
invoice_1.validate_allowed_to_pay()
|
2016-03-27 03:41:43 +00:00
|
|
|
|
|
|
|
def test_cannot_void_paid_invoice(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
2016-03-27 03:41:43 +00:00
|
|
|
|
2016-04-10 04:41:43 +00:00
|
|
|
invoice.pay("Reference", invoice.invoice.value)
|
2016-03-27 03:41:43 +00:00
|
|
|
|
|
|
|
with self.assertRaises(ValidationError):
|
2016-04-10 04:41:43 +00:00
|
|
|
invoice.void()
|
|
|
|
|
|
|
|
def test_cannot_void_partially_paid_invoice(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
invoice.pay("Reference", invoice.invoice.value - 1)
|
|
|
|
self.assertTrue(invoice.invoice.is_unpaid)
|
|
|
|
|
|
|
|
with self.assertRaises(ValidationError):
|
|
|
|
invoice.void()
|
2016-04-06 07:19:09 +00:00
|
|
|
|
|
|
|
def test_cannot_generate_blank_invoice(self):
|
|
|
|
current_cart = TestingCartController.for_user(self.USER_1)
|
|
|
|
with self.assertRaises(ValidationError):
|
2016-04-08 09:49:18 +00:00
|
|
|
TestingInvoiceController.for_cart(current_cart.cart)
|
2016-04-07 00:19:18 +00:00
|
|
|
|
2016-04-07 08:26:31 +00:00
|
|
|
def test_cannot_pay_implicitly_void_invoice(self):
|
|
|
|
cart = TestingCartController.for_user(self.USER_1)
|
|
|
|
cart.add_to_cart(self.PROD_1, 1)
|
|
|
|
invoice = TestingInvoiceController.for_cart(self.reget(cart.cart))
|
|
|
|
|
|
|
|
# Implicitly void the invoice
|
|
|
|
cart.add_to_cart(self.PROD_1, 1)
|
|
|
|
|
|
|
|
with self.assertRaises(ValidationError):
|
|
|
|
invoice.validate_allowed_to_pay()
|
|
|
|
|
2016-04-10 04:41:43 +00:00
|
|
|
def test_overpaid_invoice_results_in_credit_note(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
# Invoice is overpaid by 1 unit
|
|
|
|
to_pay = invoice.invoice.value + 1
|
|
|
|
invoice.pay("Reference", to_pay)
|
|
|
|
|
|
|
|
# The total paid should be equal to the value of the invoice only
|
|
|
|
self.assertEqual(invoice.invoice.value, invoice.total_payments())
|
|
|
|
self.assertTrue(invoice.invoice.is_paid)
|
|
|
|
|
|
|
|
# There should be a credit note generated out of the invoice.
|
2016-04-22 05:06:24 +00:00
|
|
|
credit_notes = commerce.CreditNote.objects.filter(
|
|
|
|
invoice=invoice.invoice,
|
|
|
|
)
|
2016-04-10 04:41:43 +00:00
|
|
|
self.assertEqual(1, credit_notes.count())
|
|
|
|
self.assertEqual(to_pay - invoice.invoice.value, credit_notes[0].value)
|
|
|
|
|
|
|
|
def test_full_paid_invoice_does_not_generate_credit_note(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
# Invoice is paid evenly
|
|
|
|
invoice.pay("Reference", invoice.invoice.value)
|
|
|
|
|
|
|
|
# The total paid should be equal to the value of the invoice only
|
|
|
|
self.assertEqual(invoice.invoice.value, invoice.total_payments())
|
|
|
|
self.assertTrue(invoice.invoice.is_paid)
|
|
|
|
|
|
|
|
# There should be no credit notes
|
2016-04-22 05:06:24 +00:00
|
|
|
credit_notes = commerce.CreditNote.objects.filter(
|
|
|
|
invoice=invoice.invoice,
|
|
|
|
)
|
2016-04-10 04:41:43 +00:00
|
|
|
self.assertEqual(0, credit_notes.count())
|
|
|
|
|
|
|
|
def test_refund_partially_paid_invoice_generates_correct_credit_note(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
# Invoice is underpaid by 1 unit
|
|
|
|
to_pay = invoice.invoice.value - 1
|
|
|
|
invoice.pay("Reference", to_pay)
|
|
|
|
invoice.refund()
|
|
|
|
|
|
|
|
# The total paid should be zero
|
|
|
|
self.assertEqual(0, invoice.total_payments())
|
|
|
|
self.assertTrue(invoice.invoice.is_void)
|
|
|
|
|
|
|
|
# There should be a credit note generated out of the invoice.
|
2016-04-22 05:06:24 +00:00
|
|
|
credit_notes = commerce.CreditNote.objects.filter(
|
|
|
|
invoice=invoice.invoice,
|
|
|
|
)
|
2016-04-10 04:41:43 +00:00
|
|
|
self.assertEqual(1, credit_notes.count())
|
|
|
|
self.assertEqual(to_pay, credit_notes[0].value)
|
|
|
|
|
|
|
|
def test_refund_fully_paid_invoice_generates_correct_credit_note(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
to_pay = invoice.invoice.value
|
|
|
|
invoice.pay("Reference", to_pay)
|
|
|
|
self.assertTrue(invoice.invoice.is_paid)
|
|
|
|
|
|
|
|
invoice.refund()
|
|
|
|
|
|
|
|
# The total paid should be zero
|
|
|
|
self.assertEqual(0, invoice.total_payments())
|
|
|
|
self.assertTrue(invoice.invoice.is_refunded)
|
|
|
|
|
|
|
|
# There should be a credit note generated out of the invoice.
|
2016-04-22 05:06:24 +00:00
|
|
|
credit_notes = commerce.CreditNote.objects.filter(
|
|
|
|
invoice=invoice.invoice,
|
|
|
|
)
|
2016-04-10 04:41:43 +00:00
|
|
|
self.assertEqual(1, credit_notes.count())
|
|
|
|
self.assertEqual(to_pay, credit_notes[0].value)
|
|
|
|
|
|
|
|
def test_apply_credit_note_pays_invoice(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
to_pay = invoice.invoice.value
|
|
|
|
invoice.pay("Reference", to_pay)
|
|
|
|
self.assertTrue(invoice.invoice.is_paid)
|
|
|
|
|
|
|
|
invoice.refund()
|
|
|
|
|
|
|
|
# There should be one credit note generated out of the invoice.
|
2016-04-24 22:13:44 +00:00
|
|
|
cn = self._credit_note_for_invoice(invoice.invoice)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
# That credit note should be in the unclaimed pile
|
2016-04-22 05:06:24 +00:00
|
|
|
self.assertEquals(1, commerce.CreditNote.unclaimed().count())
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
# Create a new (identical) cart with invoice
|
|
|
|
cart = TestingCartController.for_user(self.USER_1)
|
|
|
|
cart.add_to_cart(self.PROD_1, 1)
|
|
|
|
|
|
|
|
invoice2 = TestingInvoiceController.for_cart(self.reget(cart.cart))
|
|
|
|
|
|
|
|
cn.apply_to_invoice(invoice2.invoice)
|
|
|
|
self.assertTrue(invoice2.invoice.is_paid)
|
|
|
|
|
|
|
|
# That invoice should not show up as unclaimed any more
|
2016-04-22 05:06:24 +00:00
|
|
|
self.assertEquals(0, commerce.CreditNote.unclaimed().count())
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
def test_apply_credit_note_generates_new_credit_note_if_overpaying(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice = self._invoice_containing_prod_1(2)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
to_pay = invoice.invoice.value
|
|
|
|
invoice.pay("Reference", to_pay)
|
|
|
|
self.assertTrue(invoice.invoice.is_paid)
|
|
|
|
|
|
|
|
invoice.refund()
|
|
|
|
|
|
|
|
# There should be one credit note generated out of the invoice.
|
2016-04-24 22:13:44 +00:00
|
|
|
cn = self._credit_note_for_invoice(invoice.invoice)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
2016-04-22 05:06:24 +00:00
|
|
|
self.assertEquals(1, commerce.CreditNote.unclaimed().count())
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
# Create a new cart (of half value of inv 1) and get invoice
|
|
|
|
cart = TestingCartController.for_user(self.USER_1)
|
|
|
|
cart.add_to_cart(self.PROD_1, 1)
|
|
|
|
|
|
|
|
invoice2 = TestingInvoiceController.for_cart(self.reget(cart.cart))
|
|
|
|
|
|
|
|
cn.apply_to_invoice(invoice2.invoice)
|
|
|
|
self.assertTrue(invoice2.invoice.is_paid)
|
|
|
|
|
|
|
|
# We generated a new credit note, and spent the old one,
|
|
|
|
# unclaimed should still be 1.
|
2016-04-22 05:06:24 +00:00
|
|
|
self.assertEquals(1, commerce.CreditNote.unclaimed().count())
|
2016-04-10 04:41:43 +00:00
|
|
|
|
2016-04-22 05:06:24 +00:00
|
|
|
credit_note2 = commerce.CreditNote.objects.get(
|
|
|
|
invoice=invoice2.invoice,
|
|
|
|
)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
# The new credit note should be the residual of the cost of cart 1
|
|
|
|
# minus the cost of cart 2.
|
|
|
|
self.assertEquals(
|
|
|
|
invoice.invoice.value - invoice2.invoice.value,
|
|
|
|
credit_note2.value,
|
|
|
|
)
|
|
|
|
|
|
|
|
def test_cannot_apply_credit_note_on_invalid_invoices(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
to_pay = invoice.invoice.value
|
|
|
|
invoice.pay("Reference", to_pay)
|
|
|
|
self.assertTrue(invoice.invoice.is_paid)
|
|
|
|
|
|
|
|
invoice.refund()
|
|
|
|
|
|
|
|
# There should be one credit note generated out of the invoice.
|
2016-04-24 22:13:44 +00:00
|
|
|
cn = self._credit_note_for_invoice(invoice.invoice)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
# Create a new cart with invoice, pay it
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice_2 = self._invoice_containing_prod_1(1)
|
2016-04-10 04:41:43 +00:00
|
|
|
invoice_2.pay("LOL", invoice_2.invoice.value)
|
|
|
|
|
|
|
|
# Cannot pay paid invoice
|
|
|
|
with self.assertRaises(ValidationError):
|
|
|
|
cn.apply_to_invoice(invoice_2.invoice)
|
|
|
|
|
|
|
|
invoice_2.refund()
|
|
|
|
# Cannot pay refunded invoice
|
|
|
|
with self.assertRaises(ValidationError):
|
|
|
|
cn.apply_to_invoice(invoice_2.invoice)
|
|
|
|
|
|
|
|
# Create a new cart with invoice
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice_2 = self._invoice_containing_prod_1(1)
|
2016-04-10 04:41:43 +00:00
|
|
|
invoice_2.void()
|
|
|
|
# Cannot pay void invoice
|
|
|
|
with self.assertRaises(ValidationError):
|
|
|
|
cn.apply_to_invoice(invoice_2.invoice)
|
|
|
|
|
|
|
|
def test_cannot_apply_a_refunded_credit_note(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
to_pay = invoice.invoice.value
|
|
|
|
invoice.pay("Reference", to_pay)
|
|
|
|
self.assertTrue(invoice.invoice.is_paid)
|
|
|
|
|
|
|
|
invoice.refund()
|
|
|
|
|
2016-04-22 05:06:24 +00:00
|
|
|
self.assertEquals(1, commerce.CreditNote.unclaimed().count())
|
2016-04-10 04:41:43 +00:00
|
|
|
|
2016-04-24 22:13:44 +00:00
|
|
|
cn = self._credit_note_for_invoice(invoice.invoice)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
cn.refund()
|
|
|
|
|
|
|
|
# Refunding a credit note should mark it as claimed
|
2016-04-22 05:06:24 +00:00
|
|
|
self.assertEquals(0, commerce.CreditNote.unclaimed().count())
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
# Create a new cart with invoice
|
|
|
|
cart = TestingCartController.for_user(self.USER_1)
|
|
|
|
cart.add_to_cart(self.PROD_1, 1)
|
|
|
|
|
|
|
|
invoice_2 = TestingInvoiceController.for_cart(self.reget(cart.cart))
|
|
|
|
|
|
|
|
# Cannot pay with this credit note.
|
|
|
|
with self.assertRaises(ValidationError):
|
|
|
|
cn.apply_to_invoice(invoice_2.invoice)
|
|
|
|
|
|
|
|
def test_cannot_refund_an_applied_credit_note(self):
|
2016-04-24 21:42:05 +00:00
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
to_pay = invoice.invoice.value
|
|
|
|
invoice.pay("Reference", to_pay)
|
|
|
|
self.assertTrue(invoice.invoice.is_paid)
|
|
|
|
|
|
|
|
invoice.refund()
|
2016-04-07 00:19:18 +00:00
|
|
|
|
2016-04-22 05:06:24 +00:00
|
|
|
self.assertEquals(1, commerce.CreditNote.unclaimed().count())
|
2016-04-07 00:19:18 +00:00
|
|
|
|
2016-04-24 22:13:44 +00:00
|
|
|
cn = self._credit_note_for_invoice(invoice.invoice)
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
# Create a new cart with invoice
|
|
|
|
cart = TestingCartController.for_user(self.USER_1)
|
|
|
|
cart.add_to_cart(self.PROD_1, 1)
|
|
|
|
|
|
|
|
invoice_2 = TestingInvoiceController.for_cart(self.reget(cart.cart))
|
|
|
|
cn.apply_to_invoice(invoice_2.invoice)
|
|
|
|
|
2016-04-22 05:06:24 +00:00
|
|
|
self.assertEquals(0, commerce.CreditNote.unclaimed().count())
|
2016-04-10 04:41:43 +00:00
|
|
|
|
|
|
|
# Cannot refund this credit note as it is already applied.
|
|
|
|
with self.assertRaises(ValidationError):
|
|
|
|
cn.refund()
|
2016-04-24 22:13:44 +00:00
|
|
|
|
|
|
|
def test_money_into_void_invoice_generates_credit_note(self):
|
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
|
|
|
invoice.void()
|
|
|
|
|
|
|
|
val = invoice.invoice.value
|
|
|
|
|
|
|
|
invoice.pay("Paying into the void.", val, pre_validate=False)
|
|
|
|
cn = self._credit_note_for_invoice(invoice.invoice)
|
|
|
|
self.assertEqual(val, cn.credit_note.value)
|
|
|
|
|
|
|
|
def test_money_into_refunded_invoice_generates_credit_note(self):
|
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
|
|
|
|
|
|
|
val = invoice.invoice.value
|
|
|
|
|
|
|
|
invoice.pay("Paying the first time.", val)
|
|
|
|
invoice.refund()
|
|
|
|
|
|
|
|
cnval = val - 1
|
|
|
|
invoice.pay("Paying into the void.", cnval, pre_validate=False)
|
|
|
|
|
|
|
|
notes = commerce.CreditNote.objects.filter(invoice=invoice.invoice)
|
2016-04-25 07:37:33 +00:00
|
|
|
notes = sorted(notes, key=lambda note: note.value)
|
2016-04-25 04:31:25 +00:00
|
|
|
|
2016-04-24 22:13:44 +00:00
|
|
|
self.assertEqual(cnval, notes[0].value)
|
|
|
|
self.assertEqual(val, notes[1].value)
|
|
|
|
|
|
|
|
def test_money_into_paid_invoice_generates_credit_note(self):
|
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
|
|
|
|
|
|
|
val = invoice.invoice.value
|
|
|
|
|
|
|
|
invoice.pay("Paying the first time.", val)
|
|
|
|
|
|
|
|
invoice.pay("Paying into the void.", val, pre_validate=False)
|
|
|
|
cn = self._credit_note_for_invoice(invoice.invoice)
|
|
|
|
self.assertEqual(val, cn.credit_note.value)
|
2016-04-25 07:13:47 +00:00
|
|
|
|
|
|
|
def test_required_category_constraints_prevent_invoicing(self):
|
|
|
|
self.CAT_1.required = True
|
|
|
|
self.CAT_1.save()
|
|
|
|
|
|
|
|
cart = TestingCartController.for_user(self.USER_1)
|
|
|
|
cart.add_to_cart(self.PROD_3, 1)
|
|
|
|
|
|
|
|
# CAT_1 is required, we don't have CAT_1 yet
|
|
|
|
with self.assertRaises(ValidationError):
|
|
|
|
invoice = TestingInvoiceController.for_cart(cart.cart)
|
|
|
|
|
|
|
|
# Now that we have CAT_1, we can check out the cart
|
|
|
|
cart.add_to_cart(self.PROD_1, 1)
|
|
|
|
invoice = TestingInvoiceController.for_cart(cart.cart)
|
|
|
|
|
|
|
|
# Paying for the invoice should work fine
|
|
|
|
invoice.pay("Boop", invoice.invoice.value)
|
|
|
|
|
|
|
|
# We have an item in the first cart, so should be able to invoice
|
|
|
|
# for the second cart, even without CAT_1 in it.
|
|
|
|
cart = TestingCartController.for_user(self.USER_1)
|
|
|
|
cart.add_to_cart(self.PROD_3, 1)
|
|
|
|
|
|
|
|
invoice2 = TestingInvoiceController.for_cart(cart.cart)
|
|
|
|
|
|
|
|
# Void invoice2, and release the first cart
|
|
|
|
# now we don't have any CAT_1
|
|
|
|
invoice2.void()
|
|
|
|
invoice.refund()
|
|
|
|
|
|
|
|
# Now that we don't have CAT_1, we can't checkout this cart
|
|
|
|
with self.assertRaises(ValidationError):
|
|
|
|
invoice = TestingInvoiceController.for_cart(cart.cart)
|
2016-08-21 06:56:15 +00:00
|
|
|
|
2016-09-03 01:07:46 +00:00
|
|
|
def test_invoice_with_credit_note_applied_is_refunded(self):
|
|
|
|
''' Invoices with partial payments should void when cart is updated.
|
|
|
|
|
|
|
|
Test for issue #64 -- applying a credit note to an invoice
|
|
|
|
means that invoice cannot be voided, and new invoices cannot be
|
|
|
|
created. '''
|
|
|
|
|
|
|
|
cart = TestingCartController.for_user(self.USER_1)
|
|
|
|
|
|
|
|
cart.add_to_cart(self.PROD_1, 1)
|
|
|
|
invoice = TestingInvoiceController.for_cart(cart.cart)
|
|
|
|
|
|
|
|
# Now get a credit note
|
|
|
|
invoice.pay("Lol", invoice.invoice.value)
|
|
|
|
invoice.refund()
|
|
|
|
cn = self._credit_note_for_invoice(invoice.invoice)
|
|
|
|
|
|
|
|
# Create a cart of higher value than the credit note
|
|
|
|
cart = TestingCartController.for_user(self.USER_1)
|
|
|
|
cart.add_to_cart(self.PROD_1, 2)
|
|
|
|
|
|
|
|
# Create a current invoice, and apply partial payments
|
|
|
|
invoice = TestingInvoiceController.for_cart(cart.cart)
|
|
|
|
cn.apply_to_invoice(invoice.invoice)
|
|
|
|
|
|
|
|
# Adding to cart will mean that the old invoice for this cart
|
|
|
|
# will be invalidated. A new invoice should be generated.
|
|
|
|
cart.add_to_cart(self.PROD_1, 1)
|
2016-09-03 01:11:44 +00:00
|
|
|
invoice = TestingInvoiceController.for_id(invoice.invoice.id)
|
2016-09-03 01:07:46 +00:00
|
|
|
invoice2 = TestingInvoiceController.for_cart(cart.cart)
|
2016-09-03 01:46:24 +00:00
|
|
|
cn2 = self._credit_note_for_invoice(invoice.invoice)
|
2016-09-03 01:07:46 +00:00
|
|
|
|
2016-09-03 01:31:39 +00:00
|
|
|
invoice._refresh()
|
|
|
|
|
|
|
|
# The first invoice should be refunded
|
2016-09-03 01:07:46 +00:00
|
|
|
self.assertEquals(
|
2016-09-03 01:46:24 +00:00
|
|
|
commerce.Invoice.STATUS_VOID,
|
2016-09-03 01:07:46 +00:00
|
|
|
invoice.invoice.status,
|
|
|
|
)
|
|
|
|
|
2016-09-03 01:46:24 +00:00
|
|
|
# Both credit notes should be for the same amount
|
2016-09-03 01:31:39 +00:00
|
|
|
self.assertEquals(
|
|
|
|
cn.credit_note.value,
|
2016-09-03 01:46:24 +00:00
|
|
|
cn2.credit_note.value,
|
2016-09-03 01:31:39 +00:00
|
|
|
)
|
2016-09-03 01:07:46 +00:00
|
|
|
|
2016-09-14 23:08:29 +00:00
|
|
|
def test_can_generate_manual_invoice(self):
|
|
|
|
|
|
|
|
description_price_pairs = [
|
|
|
|
("Item 1", 15),
|
|
|
|
("Item 2", 30),
|
|
|
|
]
|
|
|
|
|
|
|
|
due_delta = datetime.timedelta(hours=24)
|
|
|
|
|
|
|
|
_invoice = TestingInvoiceController.manual_invoice(
|
|
|
|
self.USER_1, due_delta, description_price_pairs
|
|
|
|
)
|
|
|
|
inv = TestingInvoiceController(_invoice)
|
|
|
|
|
|
|
|
self.assertEquals(
|
|
|
|
inv.invoice.value,
|
|
|
|
sum(i[1] for i in description_price_pairs)
|
|
|
|
)
|
|
|
|
|
|
|
|
self.assertEquals(
|
|
|
|
len(inv.invoice.lineitem_set.all()),
|
|
|
|
len(description_price_pairs)
|
|
|
|
)
|
|
|
|
|
|
|
|
inv.pay("Demo payment", inv.invoice.value)
|
|
|
|
|
2016-08-21 06:56:15 +00:00
|
|
|
def test_sends_email_on_invoice_creation(self):
|
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
2016-08-21 07:14:19 +00:00
|
|
|
self.assertEquals(1, len(self.emails))
|
2016-08-21 06:56:15 +00:00
|
|
|
email = self.emails[0]
|
2016-08-21 08:28:16 +00:00
|
|
|
self.assertEquals([self.USER_1.email], email["to"])
|
2016-08-21 06:56:15 +00:00
|
|
|
self.assertEquals("invoice_created", email["kind"])
|
|
|
|
self.assertEquals(invoice.invoice, email["context"]["invoice"])
|
2016-08-21 07:14:19 +00:00
|
|
|
|
|
|
|
def test_sends_first_change_email_on_invoice_fully_paid(self):
|
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
|
|
|
|
|
|
|
self.assertEquals(1, len(self.emails))
|
|
|
|
invoice.pay("Partial", invoice.invoice.value - 1)
|
|
|
|
# Should have an "invoice_created" email and nothing else.
|
|
|
|
self.assertEquals(1, len(self.emails))
|
|
|
|
invoice.pay("Remainder", 1)
|
|
|
|
self.assertEquals(2, len(self.emails))
|
|
|
|
|
|
|
|
email = self.emails[1]
|
2016-08-21 08:28:16 +00:00
|
|
|
self.assertEquals([self.USER_1.email], email["to"])
|
2016-08-21 07:14:19 +00:00
|
|
|
self.assertEquals("invoice_updated", email["kind"])
|
|
|
|
self.assertEquals(invoice.invoice, email["context"]["invoice"])
|
|
|
|
|
|
|
|
def test_sends_email_when_invoice_refunded(self):
|
|
|
|
invoice = self._invoice_containing_prod_1(1)
|
|
|
|
|
|
|
|
self.assertEquals(1, len(self.emails))
|
|
|
|
invoice.pay("Payment", invoice.invoice.value)
|
|
|
|
self.assertEquals(2, len(self.emails))
|
|
|
|
invoice.refund()
|
|
|
|
self.assertEquals(3, len(self.emails))
|
|
|
|
|
|
|
|
email = self.emails[2]
|
2016-08-21 08:28:16 +00:00
|
|
|
self.assertEquals([self.USER_1.email], email["to"])
|
2016-08-21 07:14:19 +00:00
|
|
|
self.assertEquals("invoice_updated", email["kind"])
|
|
|
|
self.assertEquals(invoice.invoice, email["context"]["invoice"])
|