2020-09-04 15:53:00 +02:00
|
|
|
# Copyright (C) 2018-2020 by BDE ENS Paris-Saclay
|
|
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
|
|
|
|
from django.contrib.auth.models import User
|
|
|
|
from django.core.exceptions import ValidationError
|
|
|
|
from django.db.models import Q
|
|
|
|
from django.test import TestCase
|
|
|
|
from django.urls import reverse
|
|
|
|
from member.models import Membership, Club
|
|
|
|
from note.models import SpecialTransaction, NoteSpecial, Transaction
|
|
|
|
from treasury.models import Invoice, Product, Remittance, RemittanceType, SogeCredit
|
|
|
|
|
|
|
|
|
|
|
|
class TestInvoices(TestCase):
|
|
|
|
"""
|
|
|
|
Check that invoices can be created and rendered properly.
|
|
|
|
"""
|
|
|
|
def setUp(self) -> None:
|
|
|
|
self.user = User.objects.create_superuser(
|
|
|
|
username="admintoto",
|
|
|
|
password="totototo",
|
|
|
|
email="admin@example.com",
|
|
|
|
)
|
|
|
|
self.client.force_login(self.user)
|
|
|
|
sess = self.client.session
|
|
|
|
sess["permission_mask"] = 42
|
|
|
|
sess.save()
|
|
|
|
|
|
|
|
self.invoice = Invoice.objects.create(
|
|
|
|
id=1,
|
|
|
|
object="Object",
|
|
|
|
description="Description",
|
|
|
|
name="Me",
|
|
|
|
address="Earth",
|
|
|
|
acquitted=False,
|
|
|
|
)
|
|
|
|
self.product = Product.objects.create(
|
|
|
|
invoice=self.invoice,
|
|
|
|
designation="Product",
|
|
|
|
quantity=3,
|
|
|
|
amount=3.14,
|
|
|
|
)
|
|
|
|
|
|
|
|
def test_admin_page(self):
|
|
|
|
"""
|
|
|
|
Display the invoice admin page.
|
|
|
|
"""
|
|
|
|
response = self.client.get(reverse("admin:index") + "treasury/invoice/")
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
def test_invoices_list(self):
|
|
|
|
"""
|
|
|
|
Display the list of invoices.
|
|
|
|
"""
|
|
|
|
response = self.client.get(reverse("treasury:invoice_list"))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
def test_invoice_create(self):
|
|
|
|
"""
|
|
|
|
Try to create a new invoice.
|
|
|
|
"""
|
|
|
|
response = self.client.get(reverse("treasury:invoice_create"))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
response = self.client.post(reverse("treasury:invoice_create"), data={
|
|
|
|
"id": 42,
|
|
|
|
"object": "Same object",
|
|
|
|
"description": "Longer description",
|
|
|
|
"name": "Me and others",
|
|
|
|
"address": "Alwways earth",
|
|
|
|
"acquitted": True,
|
|
|
|
"products-0-designation": "Designation",
|
|
|
|
"products-0-quantity": 1,
|
|
|
|
"products-0-amount": 42,
|
|
|
|
"products-TOTAL_FORMS": 1,
|
|
|
|
"products-INITIAL_FORMS": 0,
|
|
|
|
"products-MIN_NUM_FORMS": 0,
|
|
|
|
"products-MAX_NUM_FORMS": 1000,
|
|
|
|
})
|
|
|
|
self.assertRedirects(response, reverse("treasury:invoice_list"), 302, 200)
|
|
|
|
self.assertTrue(Invoice.objects.filter(object="Same object", id=42).exists())
|
|
|
|
self.assertTrue(Product.objects.filter(designation="Designation", invoice_id=42).exists())
|
|
|
|
self.assertTrue(Invoice.objects.get(id=42).tex)
|
|
|
|
|
|
|
|
def test_invoice_update(self):
|
|
|
|
"""
|
|
|
|
Try to update an invoice.
|
|
|
|
"""
|
|
|
|
response = self.client.get(reverse("treasury:invoice_update", args=(self.invoice.id,)))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
data = {
|
|
|
|
"object": "Same object",
|
|
|
|
"description": "Longer description",
|
|
|
|
"name": "Me and others",
|
|
|
|
"address": "Always earth",
|
|
|
|
"acquitted": True,
|
|
|
|
"locked": True,
|
|
|
|
"products-0-designation": "Designation",
|
|
|
|
"products-0-quantity": 1,
|
|
|
|
"products-0-amount": 4200,
|
|
|
|
"products-1-designation": "Second designation",
|
|
|
|
"products-1-quantity": 5,
|
|
|
|
"products-1-amount": -1800,
|
|
|
|
"products-TOTAL_FORMS": 2,
|
|
|
|
"products-INITIAL_FORMS": 0,
|
|
|
|
"products-MIN_NUM_FORMS": 0,
|
|
|
|
"products-MAX_NUM_FORMS": 1000,
|
|
|
|
}
|
|
|
|
|
|
|
|
response = self.client.post(reverse("treasury:invoice_update", args=(self.invoice.id,)), data=data)
|
|
|
|
self.assertRedirects(response, reverse("treasury:invoice_list"), 302, 200)
|
|
|
|
self.invoice.refresh_from_db()
|
|
|
|
self.assertTrue(Invoice.objects.filter(pk=1, object="Same object", locked=True).exists())
|
|
|
|
self.assertTrue(Product.objects.filter(designation="Second designation", invoice_id=1).exists())
|
|
|
|
|
|
|
|
# Resend the same data, but the invoice is locked.
|
|
|
|
response = self.client.get(reverse("treasury:invoice_update", args=(self.invoice.id,)))
|
|
|
|
self.assertTrue(response.status_code, 200)
|
|
|
|
response = self.client.post(reverse("treasury:invoice_update", args=(self.invoice.id,)), data=data)
|
|
|
|
self.assertTrue(response.status_code, 200)
|
|
|
|
|
|
|
|
def test_delete_invoice(self):
|
|
|
|
"""
|
|
|
|
Try to delete an invoice.
|
|
|
|
"""
|
|
|
|
response = self.client.get(reverse("treasury:invoice_delete", args=(self.invoice.id,)))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
# Can't delete a locked invoice
|
|
|
|
self.invoice.locked = True
|
|
|
|
self.invoice.save()
|
|
|
|
response = self.client.delete(reverse("treasury:invoice_delete", args=(self.invoice.id,)))
|
|
|
|
self.assertEqual(response.status_code, 403)
|
|
|
|
self.assertTrue(Invoice.objects.filter(pk=self.invoice.id).exists())
|
|
|
|
|
|
|
|
# Unlock invoice and truly delete it.
|
|
|
|
self.invoice.locked = False
|
|
|
|
self.invoice._force_save = True
|
|
|
|
self.invoice.save()
|
|
|
|
response = self.client.delete(reverse("treasury:invoice_delete", args=(self.invoice.id,)))
|
|
|
|
self.assertRedirects(response, reverse("treasury:invoice_list"), 302, 200)
|
|
|
|
self.assertFalse(Invoice.objects.filter(pk=self.invoice.id).exists())
|
|
|
|
|
|
|
|
def test_invoice_render_pdf(self):
|
|
|
|
"""
|
|
|
|
Generate the PDF file of an invoice.
|
|
|
|
"""
|
|
|
|
response = self.client.get(reverse("treasury:invoice_render", args=(self.invoice.id,)))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
def test_invoice_api(self):
|
|
|
|
"""
|
|
|
|
Load some API pages
|
|
|
|
"""
|
|
|
|
response = self.client.get("/api/treasury/invoice/")
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
response = self.client.get("/api/treasury/product/")
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
|
|
|
|
class TestRemittances(TestCase):
|
|
|
|
"""
|
|
|
|
Create some credits and close remittances.
|
|
|
|
"""
|
|
|
|
|
|
|
|
fixtures = ('initial',)
|
|
|
|
|
|
|
|
def setUp(self) -> None:
|
|
|
|
self.user = User.objects.create_superuser(
|
|
|
|
username="admintoto",
|
|
|
|
password="totototo",
|
|
|
|
email="admin@example.com",
|
|
|
|
)
|
|
|
|
self.client.force_login(self.user)
|
|
|
|
sess = self.client.session
|
|
|
|
sess["permission_mask"] = 42
|
|
|
|
sess.save()
|
|
|
|
|
|
|
|
self.credit = SpecialTransaction.objects.create(
|
|
|
|
source=NoteSpecial.objects.get(special_type="Chèque"),
|
|
|
|
destination=self.user.note,
|
|
|
|
amount=4200,
|
|
|
|
reason="Credit",
|
|
|
|
last_name="TOTO",
|
|
|
|
first_name="Toto",
|
|
|
|
bank="Société générale",
|
|
|
|
)
|
|
|
|
|
|
|
|
self.second_credit = SpecialTransaction.objects.create(
|
|
|
|
source=self.user.note,
|
|
|
|
destination=NoteSpecial.objects.get(special_type="Chèque"),
|
|
|
|
amount=424200,
|
|
|
|
reason="Second credit",
|
|
|
|
last_name="TOTO",
|
|
|
|
first_name="Toto",
|
|
|
|
bank="Société générale",
|
|
|
|
)
|
|
|
|
|
|
|
|
self.remittance = Remittance.objects.create(
|
|
|
|
remittance_type=RemittanceType.objects.get(),
|
|
|
|
comment="Test remittance",
|
|
|
|
closed=False,
|
|
|
|
)
|
|
|
|
self.credit.specialtransactionproxy.remittance = self.remittance
|
|
|
|
self.credit.specialtransactionproxy.save()
|
|
|
|
|
|
|
|
def test_admin_page(self):
|
|
|
|
"""
|
|
|
|
Load the admin page.
|
|
|
|
"""
|
|
|
|
response = self.client.get(reverse("admin:index") + "treasury/remittance/")
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
def test_remittances_list(self):
|
|
|
|
"""
|
|
|
|
Display the remittance list.
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
response = self.client.get(reverse("treasury:remittance_list"))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
def test_remittance_create(self):
|
|
|
|
"""
|
|
|
|
Create a new Remittance.
|
|
|
|
"""
|
|
|
|
response = self.client.get(reverse("treasury:remittance_create"))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
response = self.client.post(reverse("treasury:remittance_create"), data=dict(
|
|
|
|
remittance_type=RemittanceType.objects.get().pk,
|
|
|
|
comment="Created remittance",
|
|
|
|
))
|
|
|
|
self.assertRedirects(response, reverse("treasury:remittance_list"), 302, 200)
|
|
|
|
self.assertTrue(Remittance.objects.filter(comment="Created remittance").exists())
|
|
|
|
|
|
|
|
def test_remittance_update(self):
|
|
|
|
"""
|
|
|
|
Update an existing remittance.
|
|
|
|
"""
|
|
|
|
response = self.client.get(reverse("treasury:remittance_update", args=(self.remittance.pk,)))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
response = self.client.post(reverse("treasury:remittance_update", args=(self.remittance.pk,)), data=dict(
|
|
|
|
comment="Updated remittance",
|
|
|
|
))
|
|
|
|
self.assertRedirects(response, reverse("treasury:remittance_list"), 302, 200)
|
|
|
|
self.assertTrue(Remittance.objects.filter(comment="Updated remittance").exists())
|
|
|
|
|
|
|
|
def test_remittance_close(self):
|
|
|
|
"""
|
|
|
|
Try to close an open remittance.
|
|
|
|
"""
|
|
|
|
response = self.client.get(reverse("treasury:remittance_update", args=(self.remittance.pk,)))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
response = self.client.post(reverse("treasury:remittance_update", args=(self.remittance.pk,)), data=dict(
|
|
|
|
comment="Closed remittance",
|
|
|
|
close=True,
|
|
|
|
))
|
|
|
|
self.assertRedirects(response, reverse("treasury:remittance_list"), 302, 200)
|
|
|
|
self.assertTrue(Remittance.objects.filter(comment="Closed remittance", closed=True).exists())
|
|
|
|
|
|
|
|
def test_remittance_link_transaction(self):
|
|
|
|
"""
|
|
|
|
Link a transaction to an open remittance.
|
|
|
|
"""
|
|
|
|
response = self.client.get(reverse("treasury:link_transaction", args=(self.credit.pk,)))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
response = self.client.post(reverse("treasury:link_transaction", args=(self.credit.pk,)), data=dict(
|
|
|
|
remittance=self.remittance.pk,
|
|
|
|
last_name="Last Name",
|
|
|
|
first_name="First Name",
|
|
|
|
bank="Bank",
|
|
|
|
))
|
|
|
|
self.assertRedirects(response, reverse("treasury:remittance_list"), 302, 200)
|
|
|
|
self.credit.refresh_from_db()
|
|
|
|
self.assertEqual(self.credit.last_name, "Last Name")
|
|
|
|
self.assertEqual(self.remittance.transactions.count(), 1)
|
|
|
|
|
|
|
|
response = self.client.get(reverse("treasury:unlink_transaction", args=(self.credit.pk,)))
|
|
|
|
self.assertRedirects(response, reverse("treasury:remittance_list"), 302, 200)
|
|
|
|
|
|
|
|
def test_invoice_api(self):
|
|
|
|
"""
|
|
|
|
Load some API pages
|
|
|
|
"""
|
|
|
|
response = self.client.get("/api/treasury/remittance_type/")
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
response = self.client.get("/api/treasury/remittance/")
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
|
|
|
|
class TestSogeCredits(TestCase):
|
|
|
|
"""
|
|
|
|
Check that credits from the Société générale are working correctly.
|
|
|
|
"""
|
|
|
|
|
|
|
|
fixtures = ('initial',)
|
|
|
|
|
|
|
|
def setUp(self) -> None:
|
|
|
|
self.user = User.objects.create_superuser(
|
|
|
|
username="admintoto",
|
|
|
|
password="totototo",
|
|
|
|
email="admin@example.com",
|
|
|
|
)
|
|
|
|
self.client.force_login(self.user)
|
|
|
|
sess = self.client.session
|
|
|
|
sess["permission_mask"] = 42
|
|
|
|
sess.save()
|
|
|
|
|
|
|
|
self.kfet = Club.objects.get(name="Kfet")
|
|
|
|
self.bde = self.kfet.parent_club
|
|
|
|
|
|
|
|
self.kfet_membership = Membership(
|
|
|
|
user=self.user,
|
|
|
|
club=self.kfet,
|
|
|
|
)
|
|
|
|
self.kfet_membership._force_renew_parent = True
|
|
|
|
self.kfet_membership._soge = True
|
|
|
|
self.kfet_membership.save()
|
|
|
|
|
|
|
|
def test_admin_page(self):
|
|
|
|
"""
|
|
|
|
Render the admin page.
|
|
|
|
"""
|
|
|
|
response = self.client.get(reverse("admin:index") + "treasury/sogecredit/")
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
def test_sogecredit_list(self):
|
|
|
|
"""
|
|
|
|
Display the list of all credits.
|
|
|
|
"""
|
|
|
|
response = self.client.get(reverse("treasury:soge_credits"))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
response = self.client.get(reverse("treasury:soge_credits") + "?search=toto&valid=")
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
def test_validate_soge_credit(self):
|
|
|
|
"""
|
|
|
|
Try to validate a credit.
|
|
|
|
"""
|
|
|
|
soge_credit = SogeCredit.objects.get(user=self.user)
|
|
|
|
|
|
|
|
response = self.client.get(reverse("treasury:manage_soge_credit", args=(soge_credit.pk,)))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
response = self.client.post(reverse("treasury:manage_soge_credit", args=(soge_credit.pk,)), data=dict(
|
|
|
|
validate=True,
|
|
|
|
))
|
|
|
|
self.assertRedirects(response, reverse("treasury:manage_soge_credit", args=(soge_credit.pk,)), 302, 200)
|
|
|
|
soge_credit.refresh_from_db()
|
|
|
|
self.assertTrue(soge_credit.valid)
|
|
|
|
self.user.note.refresh_from_db()
|
|
|
|
self.assertEqual(
|
|
|
|
Transaction.objects.filter(Q(source=self.user.note) | Q(destination=self.user.note)).count(), 3)
|
|
|
|
self.assertTrue(self.user.profile.soge)
|
|
|
|
|
|
|
|
def test_delete_soge_credit(self):
|
|
|
|
"""
|
|
|
|
Try to invalidate a credit.
|
|
|
|
"""
|
|
|
|
soge_credit = SogeCredit.objects.get(user=self.user)
|
|
|
|
|
|
|
|
response = self.client.get(reverse("treasury:manage_soge_credit", args=(soge_credit.pk,)))
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
try:
|
|
|
|
self.client.post(reverse("treasury:manage_soge_credit", args=(soge_credit.pk,)), data=dict(delete=True))
|
|
|
|
raise AssertionError("It is not possible to delete the soge credit until the note is not credited.")
|
|
|
|
except ValidationError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
SpecialTransaction.objects.create(
|
|
|
|
source=NoteSpecial.objects.get(special_type="Carte bancaire"),
|
|
|
|
destination=self.user.note,
|
|
|
|
amount=self.bde.membership_fee_paid + self.kfet.membership_fee_paid,
|
|
|
|
quantity=1,
|
|
|
|
reason="Registration is not complete, pliz pay",
|
|
|
|
last_name="TOTO",
|
|
|
|
first_name="Toto",
|
|
|
|
)
|
|
|
|
|
|
|
|
response = self.client.post(reverse("treasury:manage_soge_credit", args=(soge_credit.pk,)),
|
|
|
|
data=dict(delete=True))
|
|
|
|
# 403 because no SogeCredit exists anymore, then a PermissionDenied is raised
|
|
|
|
self.assertRedirects(response, reverse("treasury:soge_credits"), 302, 403)
|
|
|
|
self.assertFalse(SogeCredit.objects.filter(pk=soge_credit.pk))
|
|
|
|
self.user.note.refresh_from_db()
|
|
|
|
self.assertEqual(self.user.note.balance, 0)
|
|
|
|
self.assertEqual(
|
2020-09-07 21:33:23 +02:00
|
|
|
Transaction.objects.filter(Q(source=self.user.note) | Q(destination=self.user.note)).count(), 4)
|
2020-09-04 15:53:00 +02:00
|
|
|
self.assertFalse(self.user.profile.soge)
|
|
|
|
|
|
|
|
def test_invoice_api(self):
|
|
|
|
"""
|
|
|
|
Load some API pages
|
|
|
|
"""
|
|
|
|
response = self.client.get("/api/treasury/soge_credit/")
|
|
|
|
self.assertEqual(response.status_code, 200)
|