-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathmodels.py
138 lines (97 loc) · 4.23 KB
/
models.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
from __future__ import unicode_literals
import datetime
from decimal import Decimal
from uuid import uuid4
from django.contrib.auth.models import User
from django.core.validators import MinValueValidator
from django.db import models
from django.utils import timezone
from recurrence.fields import RecurrenceField
from fields import UniqueBooleanField
class RateQuerySet(models.QuerySet):
def intransient(self):
return self.filter(transient=False)
def active(self):
return self.filter(active=True)
def total(self):
return self.active().aggregate(
models.Sum('amount_per_day')
)['amount_per_day__sum']
class Rate(models.Model):
objects = RateQuerySet.as_manager()
id = models.UUIDField(
primary_key=True, editable=False, default=uuid4, unique=True)
description = models.CharField(max_length=120)
amount = models.DecimalField(max_digits=8, decimal_places=2)
days = models.PositiveIntegerField(validators=[MinValueValidator(1)])
amount_per_day = models.DecimalField(
max_digits=8, decimal_places=3, editable=False, blank=True)
active = models.BooleanField(default=True)
def rount_amount_per_day(self, place='0.01'):
return Decimal(self.amount_per_day).quantize(Decimal(place))
def save(self, *args, **kwargs):
self.amount_per_day = self.amount / Decimal(self.days)
return super(Rate, self).save(*args, **kwargs)
def __unicode__(self):
return '{0} ({1})'.format(self.description,
self.rount_amount_per_day())
class Allowance(models.Model):
id = models.UUIDField(
primary_key=True, editable=False, default=uuid4, unique=True)
user = models.OneToOneField(User)
amount = models.DecimalField(max_digits=8, decimal_places=2)
def round_amount(self, place='0.01'):
return Decimal(self.amount).quantize(Decimal(place))
def __unicode__(self):
return '{} - {}'.format(self.user.username, self.round_amount())
class TransactionQuerySet(models.QuerySet):
def date(self, date):
return self.filter(
timestamp__month=date.month,
timestamp__day=date.day,
timestamp__year=date.year)
def date_range(self, start_of_day, end_of_day):
query_set = self
if start_of_day:
start_of_day = start_of_day.replace(hour=0, minute=0, second=59)
query_set = query_set.filter(timestamp__gte=start_of_day)
if end_of_day:
end_of_day = end_of_day.replace(hour=23, minute=59, second=59)
query_set = query_set.filter(timestamp__lte=end_of_day)
return query_set
def days_from_today(self, days):
end = timezone.now()
start = end - timezone.timedelta(days=days)
return self.date_range(start, end)
def today(self):
now = timezone.localtime(timezone.now())
now = datetime.datetime(now.year, now.month, now.day)
tomorrow = now + datetime.timedelta(days=1)
now, tomorrow = [timezone.make_aware(d) for d in [now, tomorrow]]
return self.filter(timestamp__gte=now, timestamp__lt=tomorrow)
def last_week(self):
return self.days_from_today(7)
def last_month(self):
return self.days_from_today(30)
def last_year(self):
return self.days_from_today(365)
def with_allowance(self):
return self.filter(allowance__isnull=False)
def without_allowance(self):
return self.filter(allowance__isnull=True)
def total(self):
return self.aggregate(models.Sum('amount'))['amount__sum'] or 0
class Transaction(models.Model):
objects = TransactionQuerySet.as_manager()
id = models.UUIDField(
primary_key=True, editable=False, default=uuid4, unique=True)
description = models.CharField(max_length=120)
timestamp = models.DateTimeField(default=timezone.now)
amount = models.DecimalField(max_digits=8, decimal_places=2)
allowance = models.ForeignKey(Allowance, blank=True, null=True)
def round_amount(self, place='0.01'):
return Decimal(self.amount).quantize(Decimal(place))
def __unicode__(self):
return '{} ({})'.format(self.description, self.round_amount())
class Meta:
ordering = ['-timestamp']