2020-02-18 20:30:26 +00:00
|
|
|
# Copyright (C) 2018-2020 by BDE ENS Paris-Saclay
|
2019-07-16 10:43:23 +00:00
|
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
2020-05-29 19:43:24 +00:00
|
|
|
from django.core.exceptions import ValidationError
|
2020-08-05 17:42:44 +00:00
|
|
|
from django.db import models, transaction
|
2020-03-07 21:28:59 +00:00
|
|
|
from django.urls import reverse
|
2019-07-16 10:43:23 +00:00
|
|
|
from django.utils import timezone
|
|
|
|
from django.utils.translation import gettext_lazy as _
|
2020-02-24 09:36:04 +00:00
|
|
|
from polymorphic.models import PolymorphicModel
|
2019-07-16 10:43:23 +00:00
|
|
|
|
2020-08-06 10:50:24 +00:00
|
|
|
from .notes import Note, NoteClub, NoteSpecial
|
2020-04-22 01:26:45 +00:00
|
|
|
from ..templatetags.pretty_money import pretty_money
|
2019-07-16 10:43:23 +00:00
|
|
|
|
|
|
|
"""
|
|
|
|
Defines transactions
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
2020-02-23 16:27:55 +00:00
|
|
|
class TemplateCategory(models.Model):
|
2020-02-04 00:18:03 +00:00
|
|
|
"""
|
|
|
|
Defined a recurrent transaction category
|
|
|
|
|
|
|
|
Example: food, softs, ...
|
|
|
|
"""
|
|
|
|
name = models.CharField(
|
|
|
|
verbose_name=_("name"),
|
|
|
|
max_length=31,
|
|
|
|
unique=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
verbose_name = _("transaction category")
|
|
|
|
verbose_name_plural = _("transaction categories")
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return str(self.name)
|
2019-07-16 10:43:23 +00:00
|
|
|
|
2020-02-18 20:14:29 +00:00
|
|
|
|
2019-07-16 10:43:23 +00:00
|
|
|
class TransactionTemplate(models.Model):
|
2020-01-21 21:06:06 +00:00
|
|
|
"""
|
2020-02-04 00:18:03 +00:00
|
|
|
Defined a recurrent transaction
|
2020-01-21 21:06:06 +00:00
|
|
|
|
|
|
|
associated to selling something (a burger, a beer, ...)
|
|
|
|
"""
|
2019-07-16 10:43:23 +00:00
|
|
|
name = models.CharField(
|
|
|
|
verbose_name=_('name'),
|
|
|
|
max_length=255,
|
2019-08-13 16:23:15 +00:00
|
|
|
unique=True,
|
2020-03-07 21:28:59 +00:00
|
|
|
error_messages={'unique': _("A template with this name already exist")},
|
2019-07-16 10:43:23 +00:00
|
|
|
)
|
2020-04-01 18:56:24 +00:00
|
|
|
|
2019-07-16 10:43:23 +00:00
|
|
|
destination = models.ForeignKey(
|
2019-08-11 17:54:18 +00:00
|
|
|
NoteClub,
|
2019-07-16 10:43:23 +00:00
|
|
|
on_delete=models.PROTECT,
|
|
|
|
related_name='+', # no reverse
|
|
|
|
verbose_name=_('destination'),
|
|
|
|
)
|
2020-04-01 18:56:24 +00:00
|
|
|
|
2019-07-16 10:43:23 +00:00
|
|
|
amount = models.PositiveIntegerField(
|
|
|
|
verbose_name=_('amount'),
|
|
|
|
)
|
2020-02-23 15:48:35 +00:00
|
|
|
category = models.ForeignKey(
|
2020-02-23 16:27:55 +00:00
|
|
|
TemplateCategory,
|
2020-02-04 00:18:03 +00:00
|
|
|
on_delete=models.PROTECT,
|
2020-07-25 15:25:57 +00:00
|
|
|
related_name='templates',
|
2019-07-16 10:43:23 +00:00
|
|
|
verbose_name=_('type'),
|
2020-02-18 20:14:29 +00:00
|
|
|
max_length=31,
|
2019-07-16 10:43:23 +00:00
|
|
|
)
|
2020-04-01 18:56:24 +00:00
|
|
|
|
2020-02-23 15:48:35 +00:00
|
|
|
display = models.BooleanField(
|
2020-03-07 21:28:59 +00:00
|
|
|
default=True,
|
2020-04-01 18:56:24 +00:00
|
|
|
verbose_name=_("display"),
|
2020-02-23 15:48:35 +00:00
|
|
|
)
|
2020-04-01 18:56:24 +00:00
|
|
|
|
2020-07-25 15:25:57 +00:00
|
|
|
highlighted = models.BooleanField(
|
|
|
|
default=False,
|
|
|
|
verbose_name=_("highlighted"),
|
|
|
|
)
|
|
|
|
|
2020-02-22 15:31:01 +00:00
|
|
|
description = models.CharField(
|
|
|
|
verbose_name=_('description'),
|
|
|
|
max_length=255,
|
2020-03-14 18:00:20 +00:00
|
|
|
blank=True,
|
2020-02-22 15:31:01 +00:00
|
|
|
)
|
|
|
|
|
2019-07-16 11:50:05 +00:00
|
|
|
class Meta:
|
|
|
|
verbose_name = _("transaction template")
|
|
|
|
verbose_name_plural = _("transaction templates")
|
|
|
|
|
2019-08-11 21:24:54 +00:00
|
|
|
def get_absolute_url(self):
|
2020-03-07 21:28:59 +00:00
|
|
|
return reverse('note:template_update', args=(self.pk,))
|
2019-08-11 21:24:54 +00:00
|
|
|
|
2019-07-16 10:43:23 +00:00
|
|
|
|
2020-02-24 09:36:04 +00:00
|
|
|
class Transaction(PolymorphicModel):
|
2020-01-21 21:06:06 +00:00
|
|
|
"""
|
|
|
|
General transaction between two :model:`note.Note`
|
|
|
|
|
|
|
|
amount is store in centimes of currency, making it a positive integer
|
|
|
|
value. (from someone to someone else)
|
|
|
|
"""
|
|
|
|
|
2019-07-16 10:43:23 +00:00
|
|
|
source = models.ForeignKey(
|
|
|
|
Note,
|
|
|
|
on_delete=models.PROTECT,
|
|
|
|
related_name='+',
|
|
|
|
verbose_name=_('source'),
|
|
|
|
)
|
2020-03-26 13:45:48 +00:00
|
|
|
|
|
|
|
source_alias = models.CharField(
|
|
|
|
max_length=255,
|
|
|
|
default="", # Will be remplaced by the name of the note on save
|
|
|
|
verbose_name=_('used alias'),
|
|
|
|
)
|
|
|
|
|
2019-07-16 10:43:23 +00:00
|
|
|
destination = models.ForeignKey(
|
|
|
|
Note,
|
|
|
|
on_delete=models.PROTECT,
|
|
|
|
related_name='+',
|
|
|
|
verbose_name=_('destination'),
|
|
|
|
)
|
2020-03-26 13:45:48 +00:00
|
|
|
|
|
|
|
destination_alias = models.CharField(
|
|
|
|
max_length=255,
|
|
|
|
default="", # Will be remplaced by the name of the note on save
|
|
|
|
verbose_name=_('used alias'),
|
|
|
|
)
|
|
|
|
|
2019-07-17 10:48:03 +00:00
|
|
|
created_at = models.DateTimeField(
|
|
|
|
verbose_name=_('created at'),
|
2019-07-16 10:43:23 +00:00
|
|
|
default=timezone.now,
|
|
|
|
)
|
2019-07-19 08:12:46 +00:00
|
|
|
quantity = models.PositiveIntegerField(
|
2019-07-16 10:43:23 +00:00
|
|
|
verbose_name=_('quantity'),
|
2019-07-19 08:12:46 +00:00
|
|
|
default=1,
|
2019-07-16 10:43:23 +00:00
|
|
|
)
|
2020-03-16 11:11:16 +00:00
|
|
|
amount = models.PositiveIntegerField(
|
|
|
|
verbose_name=_('amount'),
|
|
|
|
)
|
|
|
|
|
2019-07-17 11:53:58 +00:00
|
|
|
reason = models.CharField(
|
|
|
|
verbose_name=_('reason'),
|
|
|
|
max_length=255,
|
2019-07-16 10:43:23 +00:00
|
|
|
)
|
2020-03-25 12:13:01 +00:00
|
|
|
|
2019-07-19 08:12:46 +00:00
|
|
|
valid = models.BooleanField(
|
2019-07-16 10:43:23 +00:00
|
|
|
verbose_name=_('valid'),
|
2019-07-19 08:12:46 +00:00
|
|
|
default=True,
|
2019-07-16 10:43:23 +00:00
|
|
|
)
|
|
|
|
|
2020-03-25 12:13:01 +00:00
|
|
|
invalidity_reason = models.CharField(
|
|
|
|
verbose_name=_('invalidity reason'),
|
|
|
|
max_length=255,
|
|
|
|
default=None,
|
|
|
|
null=True,
|
2020-04-01 01:42:19 +00:00
|
|
|
blank=True,
|
2020-03-25 12:13:01 +00:00
|
|
|
)
|
|
|
|
|
2019-07-16 11:50:05 +00:00
|
|
|
class Meta:
|
|
|
|
verbose_name = _("transaction")
|
|
|
|
verbose_name_plural = _("transactions")
|
2020-03-11 12:46:47 +00:00
|
|
|
indexes = [
|
|
|
|
models.Index(fields=['created_at']),
|
|
|
|
models.Index(fields=['source']),
|
|
|
|
models.Index(fields=['destination']),
|
|
|
|
]
|
2019-07-16 11:50:05 +00:00
|
|
|
|
2020-08-09 10:31:06 +00:00
|
|
|
def validate(self):
|
2020-08-05 14:23:32 +00:00
|
|
|
previous_source_balance = self.source.balance
|
|
|
|
previous_dest_balance = self.destination.balance
|
|
|
|
|
|
|
|
created = self.pk is None
|
|
|
|
to_transfer = self.amount * self.quantity
|
|
|
|
if not created:
|
|
|
|
# Revert old transaction
|
|
|
|
old_transaction = Transaction.objects.get(pk=self.pk)
|
|
|
|
if old_transaction.valid:
|
|
|
|
self.source.balance += to_transfer
|
|
|
|
self.destination.balance -= to_transfer
|
|
|
|
|
|
|
|
if self.valid:
|
|
|
|
self.source.balance -= to_transfer
|
|
|
|
self.destination.balance += to_transfer
|
|
|
|
|
|
|
|
# When a transaction is declared valid, we ensure that the invalidity reason is null, if it was
|
|
|
|
# previously invalid
|
|
|
|
self.invalidity_reason = None
|
|
|
|
|
|
|
|
source_balance = self.source.balance
|
|
|
|
dest_balance = self.destination.balance
|
|
|
|
|
|
|
|
if source_balance > 2147483647 or source_balance < -2147483648\
|
|
|
|
or dest_balance > 2147483647 or dest_balance < -2147483648:
|
|
|
|
raise ValidationError(_("The note balances must be between - 21 474 836.47 € and 21 474 836.47 €."))
|
|
|
|
|
2020-08-09 10:31:06 +00:00
|
|
|
return source_balance - previous_source_balance, dest_balance - previous_dest_balance
|
|
|
|
|
2020-08-05 17:42:44 +00:00
|
|
|
@transaction.atomic
|
2020-03-19 01:26:06 +00:00
|
|
|
def save(self, *args, **kwargs):
|
2019-07-19 13:00:44 +00:00
|
|
|
"""
|
|
|
|
When saving, also transfer money between two notes
|
|
|
|
"""
|
2020-08-05 17:42:44 +00:00
|
|
|
with transaction.atomic():
|
2020-08-09 10:31:06 +00:00
|
|
|
diff_source, diff_dest = self.validate()
|
2020-08-05 17:42:44 +00:00
|
|
|
|
|
|
|
if not self.source.is_active or not self.destination.is_active:
|
|
|
|
if 'force_insert' not in kwargs or not kwargs['force_insert']:
|
|
|
|
if 'force_update' not in kwargs or not kwargs['force_update']:
|
|
|
|
raise ValidationError(_("The transaction can't be saved since the source note "
|
|
|
|
"or the destination note is not active."))
|
|
|
|
|
|
|
|
# If the aliases are not entered, we assume that the used alias is the name of the note
|
|
|
|
if not self.source_alias:
|
|
|
|
self.source_alias = str(self.source)
|
|
|
|
|
|
|
|
if not self.destination_alias:
|
|
|
|
self.destination_alias = str(self.destination)
|
|
|
|
|
|
|
|
if self.source.pk == self.destination.pk:
|
2020-08-06 10:46:44 +00:00
|
|
|
# When source == destination, no money is transferred and no transaction is created
|
2020-08-05 17:42:44 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
self.log("Saving")
|
|
|
|
# We save first the transaction, in case of the user has no right to transfer money
|
2020-03-19 01:26:06 +00:00
|
|
|
super().save(*args, **kwargs)
|
2020-08-05 17:42:44 +00:00
|
|
|
self.log("Saved")
|
|
|
|
|
|
|
|
# Save notes
|
2020-08-09 10:31:06 +00:00
|
|
|
self.source.refresh_from_db()
|
|
|
|
self.source.balance += diff_source
|
2020-08-05 17:42:44 +00:00
|
|
|
self.source._force_save = True
|
|
|
|
self.source.save()
|
|
|
|
self.log("Source saved")
|
2020-08-09 10:31:06 +00:00
|
|
|
self.destination.refresh_from_db()
|
|
|
|
self.destination.balance += diff_dest
|
2020-08-05 17:42:44 +00:00
|
|
|
self.destination._force_save = True
|
|
|
|
self.destination.save()
|
|
|
|
self.log("Destination saved")
|
|
|
|
|
|
|
|
def log(self, msg):
|
|
|
|
with open("/tmp/log", "a") as f:
|
|
|
|
f.write(msg + "\n")
|
2019-07-19 13:00:44 +00:00
|
|
|
|
2020-04-22 01:26:45 +00:00
|
|
|
def delete(self, **kwargs):
|
|
|
|
"""
|
|
|
|
Whenever we want to delete a transaction (caution with this), we ensure the transaction is invalid first.
|
|
|
|
"""
|
|
|
|
self.valid = False
|
|
|
|
self.save(**kwargs)
|
|
|
|
super().delete(**kwargs)
|
|
|
|
|
2019-08-15 21:08:15 +00:00
|
|
|
@property
|
|
|
|
def total(self):
|
2020-02-18 20:14:29 +00:00
|
|
|
return self.amount * self.quantity
|
2019-08-15 21:08:15 +00:00
|
|
|
|
2020-03-16 12:14:06 +00:00
|
|
|
@property
|
|
|
|
def type(self):
|
|
|
|
return _('Transfer')
|
|
|
|
|
2020-04-22 01:26:45 +00:00
|
|
|
def __str__(self):
|
|
|
|
return self.__class__.__name__ + " from " + str(self.source) + " to " + str(self.destination) + " of "\
|
|
|
|
+ pretty_money(self.quantity * self.amount) + ("" if self.valid else " invalid")
|
|
|
|
|
2019-07-16 10:43:23 +00:00
|
|
|
|
2020-03-19 19:37:48 +00:00
|
|
|
class RecurrentTransaction(Transaction):
|
2020-02-23 16:27:55 +00:00
|
|
|
"""
|
|
|
|
Special type of :model:`note.Transaction` associated to a :model:`note.TransactionTemplate`.
|
|
|
|
"""
|
|
|
|
|
|
|
|
template = models.ForeignKey(
|
|
|
|
TransactionTemplate,
|
2020-04-27 01:56:22 +00:00
|
|
|
on_delete=models.PROTECT,
|
2020-02-23 16:27:55 +00:00
|
|
|
)
|
2020-08-03 16:49:15 +00:00
|
|
|
|
2020-02-23 16:27:55 +00:00
|
|
|
category = models.ForeignKey(
|
|
|
|
TemplateCategory,
|
2020-02-23 20:05:51 +00:00
|
|
|
on_delete=models.PROTECT,
|
2020-02-23 16:27:55 +00:00
|
|
|
)
|
|
|
|
|
2020-03-16 12:14:06 +00:00
|
|
|
@property
|
|
|
|
def type(self):
|
|
|
|
return _('Template')
|
|
|
|
|
2020-08-03 16:49:15 +00:00
|
|
|
class Meta:
|
|
|
|
verbose_name = _("recurrent transaction")
|
|
|
|
verbose_name_plural = _("recurrent transactions")
|
|
|
|
|
2020-03-07 21:28:59 +00:00
|
|
|
|
2020-03-14 14:13:58 +00:00
|
|
|
class SpecialTransaction(Transaction):
|
|
|
|
"""
|
|
|
|
Special type of :model:`note.Transaction` associated to transactions with special notes
|
|
|
|
"""
|
|
|
|
|
|
|
|
last_name = models.CharField(
|
|
|
|
max_length=255,
|
|
|
|
verbose_name=_("name"),
|
|
|
|
)
|
|
|
|
|
|
|
|
first_name = models.CharField(
|
|
|
|
max_length=255,
|
|
|
|
verbose_name=_("first_name"),
|
|
|
|
)
|
|
|
|
|
|
|
|
bank = models.CharField(
|
|
|
|
max_length=255,
|
2020-03-16 11:11:16 +00:00
|
|
|
verbose_name=_("bank"),
|
|
|
|
blank=True,
|
2020-03-14 14:13:58 +00:00
|
|
|
)
|
|
|
|
|
2020-03-16 12:14:06 +00:00
|
|
|
@property
|
|
|
|
def type(self):
|
|
|
|
return _('Credit') if isinstance(self.source, NoteSpecial) else _("Debit")
|
|
|
|
|
2020-04-18 13:59:06 +00:00
|
|
|
def is_credit(self):
|
|
|
|
return isinstance(self.source, NoteSpecial)
|
|
|
|
|
|
|
|
def is_debit(self):
|
|
|
|
return isinstance(self.destination, NoteSpecial)
|
|
|
|
|
|
|
|
def clean(self):
|
|
|
|
# SpecialTransaction are only possible with NoteSpecial object
|
|
|
|
if self.is_credit() == self.is_debit():
|
2020-05-29 19:43:24 +00:00
|
|
|
raise(ValidationError(_("A special transaction is only possible between a"
|
|
|
|
" Note associated to a payment method and a User or a Club")))
|
2020-04-18 13:59:06 +00:00
|
|
|
|
2020-08-03 16:49:15 +00:00
|
|
|
class Meta:
|
|
|
|
verbose_name = _("Special transaction")
|
|
|
|
verbose_name_plural = _("Special transactions")
|
|
|
|
|
2020-03-14 14:13:58 +00:00
|
|
|
|
2019-07-16 10:43:23 +00:00
|
|
|
class MembershipTransaction(Transaction):
|
2020-01-21 21:06:06 +00:00
|
|
|
"""
|
|
|
|
Special type of :model:`note.Transaction` associated to a :model:`member.Membership`.
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
2019-07-16 10:43:23 +00:00
|
|
|
membership = models.OneToOneField(
|
|
|
|
'member.Membership',
|
|
|
|
on_delete=models.PROTECT,
|
|
|
|
related_name='transaction',
|
|
|
|
)
|
2019-07-16 11:50:05 +00:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
verbose_name = _("membership transaction")
|
|
|
|
verbose_name_plural = _("membership transactions")
|
2020-03-16 12:14:06 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def type(self):
|
|
|
|
return _('membership transaction')
|