2020-03-20 23:30:49 +00:00
|
|
|
# Copyright (C) 2018-2020 by BDE ENS Paris-Saclay
|
|
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
|
2020-03-21 08:22:38 +00:00
|
|
|
import os
|
|
|
|
import shutil
|
|
|
|
import subprocess
|
|
|
|
from tempfile import mkdtemp
|
|
|
|
|
2020-03-21 15:49:18 +00:00
|
|
|
from crispy_forms.helper import FormHelper
|
2020-03-20 23:30:49 +00:00
|
|
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
2020-03-23 22:42:37 +00:00
|
|
|
from django.core.exceptions import ValidationError
|
2020-03-21 15:49:18 +00:00
|
|
|
from django.db.models import Q
|
2020-03-21 08:22:38 +00:00
|
|
|
from django.http import HttpResponse
|
2020-03-23 22:42:37 +00:00
|
|
|
from django.shortcuts import redirect
|
2020-03-21 08:22:38 +00:00
|
|
|
from django.template.loader import render_to_string
|
2020-03-21 15:49:18 +00:00
|
|
|
from django.urls import reverse_lazy
|
2020-03-20 23:52:26 +00:00
|
|
|
from django.views.generic import CreateView, UpdateView
|
2020-03-23 21:43:16 +00:00
|
|
|
from django.views.generic.base import View, TemplateView
|
2020-03-20 23:30:49 +00:00
|
|
|
from django_tables2 import SingleTableView
|
2020-03-24 16:06:50 +00:00
|
|
|
from note.models import SpecialTransaction, NoteSpecial
|
2020-03-21 08:22:38 +00:00
|
|
|
from note_kfet.settings.base import BASE_DIR
|
2020-03-20 23:30:49 +00:00
|
|
|
|
2020-03-23 22:42:37 +00:00
|
|
|
from .forms import InvoiceForm, ProductFormSet, ProductFormSetHelper, RemittanceForm, LinkTransactionToRemittanceForm
|
2020-03-24 19:22:15 +00:00
|
|
|
from .models import Invoice, Product, Remittance, SpecialTransactionProxy
|
2020-03-23 21:43:16 +00:00
|
|
|
from .tables import InvoiceTable, RemittanceTable, SpecialTransactionTable
|
2020-03-20 23:30:49 +00:00
|
|
|
|
|
|
|
|
2020-03-22 00:22:27 +00:00
|
|
|
class InvoiceCreateView(LoginRequiredMixin, CreateView):
|
2020-03-20 23:52:26 +00:00
|
|
|
"""
|
2020-03-22 00:22:27 +00:00
|
|
|
Create Invoice
|
2020-03-20 23:52:26 +00:00
|
|
|
"""
|
2020-03-22 00:22:27 +00:00
|
|
|
model = Invoice
|
|
|
|
form_class = InvoiceForm
|
2020-03-21 15:49:18 +00:00
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
context = super().get_context_data(**kwargs)
|
2020-03-24 19:22:15 +00:00
|
|
|
|
2020-03-21 15:49:18 +00:00
|
|
|
form = context['form']
|
|
|
|
form.helper = FormHelper()
|
2020-03-24 19:22:15 +00:00
|
|
|
# Remove form tag on the generation of the form in the template (already present on the template)
|
2020-03-21 15:49:18 +00:00
|
|
|
form.helper.form_tag = False
|
2020-03-24 19:22:15 +00:00
|
|
|
# The formset handles the set of the products
|
2020-03-21 15:49:18 +00:00
|
|
|
form_set = ProductFormSet(instance=form.instance)
|
|
|
|
context['formset'] = form_set
|
|
|
|
context['helper'] = ProductFormSetHelper()
|
|
|
|
context['no_cache'] = True
|
|
|
|
|
|
|
|
return context
|
|
|
|
|
|
|
|
def form_valid(self, form):
|
|
|
|
ret = super().form_valid(form)
|
|
|
|
|
2020-03-21 17:59:13 +00:00
|
|
|
kwargs = {}
|
2020-03-24 19:22:15 +00:00
|
|
|
|
|
|
|
# The user type amounts in cents. We convert it in euros.
|
2020-03-21 17:59:13 +00:00
|
|
|
for key in self.request.POST:
|
|
|
|
value = self.request.POST[key]
|
2020-03-22 14:24:54 +00:00
|
|
|
if key.endswith("amount") and value:
|
2020-03-21 17:59:13 +00:00
|
|
|
kwargs[key] = str(int(100 * float(value)))
|
2020-03-22 14:24:54 +00:00
|
|
|
elif value:
|
2020-03-21 17:59:13 +00:00
|
|
|
kwargs[key] = value
|
|
|
|
|
2020-03-24 19:22:15 +00:00
|
|
|
# For each product, we save it
|
2020-03-21 17:59:13 +00:00
|
|
|
formset = ProductFormSet(kwargs, instance=form.instance)
|
2020-03-21 15:49:18 +00:00
|
|
|
if formset.is_valid():
|
|
|
|
for f in formset:
|
2020-03-24 19:22:15 +00:00
|
|
|
# We don't save the product if the designation is not entered, ie. if the line is empty
|
2020-03-21 15:49:18 +00:00
|
|
|
if f.is_valid() and f.instance.designation:
|
|
|
|
f.save()
|
|
|
|
f.instance.save()
|
|
|
|
else:
|
|
|
|
f.instance = None
|
|
|
|
|
|
|
|
return ret
|
|
|
|
|
|
|
|
def get_success_url(self):
|
2020-03-22 17:27:22 +00:00
|
|
|
return reverse_lazy('treasury:invoice_list')
|
2020-03-20 23:52:26 +00:00
|
|
|
|
|
|
|
|
2020-03-22 00:22:27 +00:00
|
|
|
class InvoiceListView(LoginRequiredMixin, SingleTableView):
|
2020-03-20 23:30:49 +00:00
|
|
|
"""
|
2020-03-22 00:22:27 +00:00
|
|
|
List existing Invoices
|
2020-03-20 23:30:49 +00:00
|
|
|
"""
|
2020-03-22 00:22:27 +00:00
|
|
|
model = Invoice
|
|
|
|
table_class = InvoiceTable
|
2020-03-20 23:52:26 +00:00
|
|
|
|
|
|
|
|
2020-03-22 00:22:27 +00:00
|
|
|
class InvoiceUpdateView(LoginRequiredMixin, UpdateView):
|
2020-03-20 23:52:26 +00:00
|
|
|
"""
|
2020-03-22 00:22:27 +00:00
|
|
|
Create Invoice
|
2020-03-20 23:52:26 +00:00
|
|
|
"""
|
2020-03-22 00:22:27 +00:00
|
|
|
model = Invoice
|
|
|
|
form_class = InvoiceForm
|
2020-03-21 15:49:18 +00:00
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
context = super().get_context_data(**kwargs)
|
2020-03-24 19:22:15 +00:00
|
|
|
|
2020-03-21 15:49:18 +00:00
|
|
|
form = context['form']
|
|
|
|
form.helper = FormHelper()
|
2020-03-24 19:22:15 +00:00
|
|
|
# Remove form tag on the generation of the form in the template (already present on the template)
|
2020-03-21 15:49:18 +00:00
|
|
|
form.helper.form_tag = False
|
2020-03-24 19:22:15 +00:00
|
|
|
# Fill the intial value for the date field, with the initial date of the model instance
|
2020-03-22 14:24:54 +00:00
|
|
|
form.fields['date'].initial = form.instance.date
|
2020-03-24 19:22:15 +00:00
|
|
|
# The formset handles the set of the products
|
2020-03-21 15:49:18 +00:00
|
|
|
form_set = ProductFormSet(instance=form.instance)
|
|
|
|
context['formset'] = form_set
|
|
|
|
context['helper'] = ProductFormSetHelper()
|
|
|
|
context['no_cache'] = True
|
|
|
|
|
|
|
|
return context
|
|
|
|
|
|
|
|
def form_valid(self, form):
|
|
|
|
ret = super().form_valid(form)
|
|
|
|
|
2020-03-21 17:59:13 +00:00
|
|
|
kwargs = {}
|
2020-03-24 19:22:15 +00:00
|
|
|
# The user type amounts in cents. We convert it in euros.
|
2020-03-21 17:59:13 +00:00
|
|
|
for key in self.request.POST:
|
|
|
|
value = self.request.POST[key]
|
2020-03-22 14:24:54 +00:00
|
|
|
if key.endswith("amount") and value:
|
2020-03-21 17:59:13 +00:00
|
|
|
kwargs[key] = str(int(100 * float(value)))
|
2020-03-22 14:24:54 +00:00
|
|
|
elif value:
|
2020-03-21 17:59:13 +00:00
|
|
|
kwargs[key] = value
|
|
|
|
|
|
|
|
formset = ProductFormSet(kwargs, instance=form.instance)
|
2020-03-21 15:49:18 +00:00
|
|
|
saved = []
|
2020-03-24 19:22:15 +00:00
|
|
|
# For each product, we save it
|
2020-03-21 15:49:18 +00:00
|
|
|
if formset.is_valid():
|
|
|
|
for f in formset:
|
2020-03-24 19:22:15 +00:00
|
|
|
# We don't save the product if the designation is not entered, ie. if the line is empty
|
2020-03-21 15:49:18 +00:00
|
|
|
if f.is_valid() and f.instance.designation:
|
|
|
|
f.save()
|
|
|
|
f.instance.save()
|
|
|
|
saved.append(f.instance.pk)
|
|
|
|
else:
|
|
|
|
f.instance = None
|
2020-03-24 19:22:15 +00:00
|
|
|
# Remove old products that weren't given in the form
|
2020-03-22 14:24:54 +00:00
|
|
|
Product.objects.filter(~Q(pk__in=saved), invoice=form.instance).delete()
|
2020-03-21 15:49:18 +00:00
|
|
|
|
|
|
|
return ret
|
|
|
|
|
|
|
|
def get_success_url(self):
|
2020-03-22 17:27:22 +00:00
|
|
|
return reverse_lazy('treasury:invoice_list')
|
2020-03-21 06:36:07 +00:00
|
|
|
|
|
|
|
|
2020-03-22 00:22:27 +00:00
|
|
|
class InvoiceRenderView(LoginRequiredMixin, View):
|
2020-03-21 06:36:07 +00:00
|
|
|
"""
|
2020-03-24 19:22:15 +00:00
|
|
|
Render Invoice as a generated PDF with the given information and a LaTeX template
|
2020-03-21 06:36:07 +00:00
|
|
|
"""
|
2020-03-21 08:22:38 +00:00
|
|
|
|
|
|
|
def get(self, request, **kwargs):
|
|
|
|
pk = kwargs["pk"]
|
2020-03-22 00:22:27 +00:00
|
|
|
invoice = Invoice.objects.get(pk=pk)
|
|
|
|
products = Product.objects.filter(invoice=invoice).all()
|
2020-03-21 08:22:38 +00:00
|
|
|
|
2020-03-24 19:22:15 +00:00
|
|
|
# Informations of the BDE. Should be updated when the school will move.
|
2020-03-22 14:24:54 +00:00
|
|
|
invoice.place = "Cachan"
|
|
|
|
invoice.my_name = "BDE ENS Cachan"
|
|
|
|
invoice.my_address_street = "61 avenue du Président Wilson"
|
|
|
|
invoice.my_city = "94230 Cachan"
|
|
|
|
invoice.bank_code = 30003
|
|
|
|
invoice.desk_code = 3894
|
|
|
|
invoice.account_number = 37280662
|
|
|
|
invoice.rib_key = 14
|
|
|
|
invoice.bic = "SOGEFRPP"
|
|
|
|
|
2020-03-24 19:22:15 +00:00
|
|
|
# Replace line breaks with the LaTeX equivalent
|
2020-03-22 14:24:54 +00:00
|
|
|
invoice.description = invoice.description.replace("\r", "").replace("\n", "\\\\ ")
|
|
|
|
invoice.address = invoice.address.replace("\r", "").replace("\n", "\\\\ ")
|
2020-03-24 19:22:15 +00:00
|
|
|
# Fill the template with the information
|
2020-03-22 00:22:27 +00:00
|
|
|
tex = render_to_string("treasury/invoice_sample.tex", dict(obj=invoice, products=products))
|
2020-03-24 19:22:15 +00:00
|
|
|
|
2020-03-21 08:22:38 +00:00
|
|
|
try:
|
|
|
|
os.mkdir(BASE_DIR + "/tmp")
|
|
|
|
except FileExistsError:
|
|
|
|
pass
|
2020-03-24 19:22:15 +00:00
|
|
|
# We render the file in a temporary directory
|
2020-03-21 08:22:38 +00:00
|
|
|
tmp_dir = mkdtemp(prefix=BASE_DIR + "/tmp/")
|
|
|
|
|
2020-03-21 16:29:39 +00:00
|
|
|
try:
|
2020-03-22 00:22:27 +00:00
|
|
|
with open("{}/invoice-{:d}.tex".format(tmp_dir, pk), "wb") as f:
|
2020-03-21 16:29:39 +00:00
|
|
|
f.write(tex.encode("UTF-8"))
|
|
|
|
del tex
|
|
|
|
|
2020-03-24 19:22:15 +00:00
|
|
|
# The file has to be rendered twice
|
2020-03-21 16:54:08 +00:00
|
|
|
for _ in range(2):
|
|
|
|
error = subprocess.Popen(
|
2020-03-22 00:22:27 +00:00
|
|
|
["pdflatex", "invoice-{}.tex".format(pk)],
|
2020-03-21 16:54:08 +00:00
|
|
|
cwd=tmp_dir,
|
2020-03-24 19:22:15 +00:00
|
|
|
stdin=open(os.devnull, "r"),
|
|
|
|
stderr=open(os.devnull, "wb"),
|
|
|
|
stdout=open(os.devnull, "wb"),
|
2020-03-21 16:54:08 +00:00
|
|
|
).wait()
|
|
|
|
|
|
|
|
if error:
|
2020-03-22 00:22:27 +00:00
|
|
|
raise IOError("An error attempted while generating a invoice (code=" + str(error) + ")")
|
2020-03-21 16:29:39 +00:00
|
|
|
|
2020-03-24 19:22:15 +00:00
|
|
|
# Display the generated pdf as a HTTP Response
|
2020-03-22 00:22:27 +00:00
|
|
|
pdf = open("{}/invoice-{}.pdf".format(tmp_dir, pk), 'rb').read()
|
2020-03-21 16:29:39 +00:00
|
|
|
response = HttpResponse(pdf, content_type="application/pdf")
|
2020-03-22 00:22:27 +00:00
|
|
|
response['Content-Disposition'] = "inline;filename=invoice-{:d}.pdf".format(pk)
|
2020-03-21 16:29:39 +00:00
|
|
|
except IOError as e:
|
|
|
|
raise e
|
|
|
|
finally:
|
2020-03-24 19:22:15 +00:00
|
|
|
# Delete all temporary files
|
2020-03-21 16:29:39 +00:00
|
|
|
shutil.rmtree(tmp_dir)
|
2020-03-21 08:22:38 +00:00
|
|
|
|
|
|
|
return response
|
2020-03-22 17:27:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
class RemittanceCreateView(LoginRequiredMixin, CreateView):
|
|
|
|
"""
|
|
|
|
Create Remittance
|
|
|
|
"""
|
|
|
|
model = Remittance
|
|
|
|
form_class = RemittanceForm
|
|
|
|
|
|
|
|
def get_success_url(self):
|
|
|
|
return reverse_lazy('treasury:remittance_list')
|
|
|
|
|
2020-03-23 21:43:16 +00:00
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
ctx = super().get_context_data(**kwargs)
|
|
|
|
|
|
|
|
ctx["table"] = RemittanceTable(data=Remittance.objects.all())
|
|
|
|
ctx["special_transactions"] = SpecialTransactionTable(data=SpecialTransaction.objects.none())
|
|
|
|
|
|
|
|
return ctx
|
|
|
|
|
2020-03-22 17:27:22 +00:00
|
|
|
|
2020-03-23 21:43:16 +00:00
|
|
|
class RemittanceListView(LoginRequiredMixin, TemplateView):
|
2020-03-22 17:27:22 +00:00
|
|
|
"""
|
|
|
|
List existing Remittances
|
|
|
|
"""
|
2020-03-23 21:43:16 +00:00
|
|
|
template_name = "treasury/remittance_list.html"
|
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
ctx = super().get_context_data(**kwargs)
|
|
|
|
|
|
|
|
ctx["opened_remittances"] = RemittanceTable(data=Remittance.objects.filter(closed=False).all())
|
|
|
|
ctx["closed_remittances"] = RemittanceTable(data=Remittance.objects.filter(closed=True).reverse().all())
|
2020-03-24 19:22:15 +00:00
|
|
|
|
2020-03-23 21:43:16 +00:00
|
|
|
ctx["special_transactions_no_remittance"] = SpecialTransactionTable(
|
2020-03-24 16:06:50 +00:00
|
|
|
data=SpecialTransaction.objects.filter(source__in=NoteSpecial.objects.filter(~Q(remittancetype=None)),
|
2020-03-23 21:43:16 +00:00
|
|
|
specialtransactionproxy__remittance=None).all(),
|
|
|
|
exclude=('remittance_remove', ))
|
|
|
|
ctx["special_transactions_with_remittance"] = SpecialTransactionTable(
|
2020-03-24 16:06:50 +00:00
|
|
|
data=SpecialTransaction.objects.filter(source__in=NoteSpecial.objects.filter(~Q(remittancetype=None)),
|
2020-03-23 21:43:16 +00:00
|
|
|
specialtransactionproxy__remittance__closed=False).all(),
|
|
|
|
exclude=('remittance_add', ))
|
|
|
|
|
|
|
|
return ctx
|
2020-03-22 17:27:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
class RemittanceUpdateView(LoginRequiredMixin, UpdateView):
|
|
|
|
"""
|
|
|
|
Update Remittance
|
|
|
|
"""
|
|
|
|
model = Remittance
|
|
|
|
form_class = RemittanceForm
|
|
|
|
|
|
|
|
def get_success_url(self):
|
|
|
|
return reverse_lazy('treasury:remittance_list')
|
2020-03-23 21:43:16 +00:00
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
ctx = super().get_context_data(**kwargs)
|
|
|
|
|
|
|
|
ctx["table"] = RemittanceTable(data=Remittance.objects.all())
|
2020-03-23 23:50:55 +00:00
|
|
|
data = SpecialTransaction.objects.filter(specialtransactionproxy__remittance=self.object).all()
|
2020-03-23 21:43:16 +00:00
|
|
|
ctx["special_transactions"] = SpecialTransactionTable(
|
2020-03-23 23:50:55 +00:00
|
|
|
data=data,
|
|
|
|
exclude=('remittance_add', 'remittance_remove', ) if self.object.closed else ('remittance_add', ))
|
2020-03-23 21:43:16 +00:00
|
|
|
|
|
|
|
return ctx
|
2020-03-23 22:42:37 +00:00
|
|
|
|
|
|
|
|
|
|
|
class LinkTransactionToRemittanceView(LoginRequiredMixin, UpdateView):
|
2020-03-24 19:22:15 +00:00
|
|
|
"""
|
|
|
|
Attach a special transaction to a remittance
|
|
|
|
"""
|
|
|
|
|
2020-03-23 22:42:37 +00:00
|
|
|
model = SpecialTransactionProxy
|
|
|
|
form_class = LinkTransactionToRemittanceForm
|
|
|
|
|
|
|
|
def get_success_url(self):
|
|
|
|
return reverse_lazy('treasury:remittance_list')
|
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
ctx = super().get_context_data(**kwargs)
|
|
|
|
|
|
|
|
form = ctx["form"]
|
|
|
|
form.fields["last_name"].initial = self.object.transaction.last_name
|
|
|
|
form.fields["first_name"].initial = self.object.transaction.first_name
|
|
|
|
form.fields["bank"].initial = self.object.transaction.bank
|
|
|
|
form.fields["amount"].initial = self.object.transaction.amount
|
|
|
|
form.fields["remittance"].queryset = form.fields["remittance"] \
|
2020-03-24 16:06:50 +00:00
|
|
|
.queryset.filter(remittance_type__note=self.object.transaction.source)
|
2020-03-23 22:42:37 +00:00
|
|
|
|
|
|
|
return ctx
|
|
|
|
|
|
|
|
|
|
|
|
class UnlinkTransactionToRemittanceView(LoginRequiredMixin, View):
|
2020-03-24 19:22:15 +00:00
|
|
|
"""
|
|
|
|
Unlink a special transaction and its remittance
|
|
|
|
"""
|
|
|
|
|
2020-03-23 22:42:37 +00:00
|
|
|
def get(self, *args, **kwargs):
|
|
|
|
pk = kwargs["pk"]
|
|
|
|
transaction = SpecialTransactionProxy.objects.get(pk=pk)
|
|
|
|
|
2020-03-24 19:22:15 +00:00
|
|
|
# The remittance must be open (or inexistant)
|
2020-03-23 22:42:37 +00:00
|
|
|
if transaction.remittance and transaction.remittance.closed:
|
|
|
|
raise ValidationError("Remittance is already closed.")
|
|
|
|
|
|
|
|
transaction.remittance = None
|
|
|
|
transaction.save()
|
|
|
|
|
|
|
|
return redirect('treasury:remittance_list')
|