-
Notifications
You must be signed in to change notification settings - Fork 51
/
Copy pathmanagers.py
125 lines (99 loc) · 4.5 KB
/
managers.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
from datetime import datetime, time, timedelta
from django.db import models
from django.utils import timezone
from django.utils.timezone import get_current_timezone
from places import models as place_models
from .settings import DEFAULT_SHIFT_CONFLICT_GRACE
class ShiftQuerySet(models.QuerySet):
"""Custom QuerySet for Shift. Defines several methods for filtering
Shift objects.
"""
def on_shiftdate(self, shiftdate):
"""Shifts that end on or after shiftdate and begin before or on
shiftdate. That means shifts that intersect with the day of
shiftdate.
"""
shiftdate = datetime.combine(shiftdate, time(tzinfo=get_current_timezone()))
return self.filter(
ending_time__gte=shiftdate, starting_time__lt=shiftdate + timedelta(days=1)
)
def at_place(self, place):
"""Shifts at a certain place (geographical location of a facility).
See places.models.
"""
return self.filter(facility__place=place)
def in_area(self, area):
"""Shifts at a certain area (subdivision of a region).
See places.models.
"""
return self.filter(facility__place__area=area)
def in_region(self, region):
"""Shifts at a certain region. See places.models."""
return self.filter(facility__place__area__region=region)
def in_country(self, country):
"""Shifts at a certain country. See places.models."""
return self.filter(facility__place__area__region__country=country)
def by_geography(self, geo_affiliation):
"""Shifts at a certain geo_affiliation which can be a place, area,
region or country. See places.models.
"""
if isinstance(geo_affiliation, place_models.Place):
return self.at_place(geo_affiliation)
elif isinstance(geo_affiliation, place_models.Area):
return self.in_area(geo_affiliation)
elif isinstance(geo_affiliation, place_models.Region):
return self.in_region(geo_affiliation)
elif isinstance(geo_affiliation, place_models.Country):
return self.in_country(geo_affiliation)
def open(self): # noqa: A003
return self.filter(ending_time__gte=timezone.now())
# Create manager from custom QuerySet ShiftQuerySet
ShiftManager = models.Manager.from_queryset(ShiftQuerySet)
class OpenShiftManager(ShiftManager):
"""Manager for Shift. Overwrites get_queryset with a filter on QuerySet
that holds all shifts that end now or in the future.
"""
def get_queryset(self):
return (
super(OpenShiftManager, self)
.get_queryset()
.filter(ending_time__gte=timezone.now())
)
class ShiftHelperManager(models.Manager):
"""Manager for ShiftHelper. Defines one method for filtering the
QuerySet on conflicting shifts.
"""
def conflicting(self, shift, user_account=None, grace=DEFAULT_SHIFT_CONFLICT_GRACE):
"""Filters QuerySet of ShiftHelper objects by selecting those that
intersect with respect to time.
:param shift
:param user_account - default is None (-Does a default value for
user make sense in that connection?)
:param grace - some "buffer" which reduces the time of the shift.
default is 1 hour
"""
grace = grace or timedelta(0)
# correct grace for short shifts, otherwise a user could join two
# concurrent 1-hour-shifts
if shift.duration <= grace:
grace = shift.duration / 2
graced_start = shift.starting_time + grace
graced_end = shift.ending_time - grace
query_set = self.get_queryset().select_related("shift", "user_account")
if user_account:
query_set = query_set.filter(user_account=user_account)
soft_conflict_query_set = query_set.exclude(
shift__starting_time__lt=shift.starting_time,
shift__ending_time__lte=shift.starting_time,
)
soft_conflict_query_set = soft_conflict_query_set.exclude(
shift__starting_time__gte=shift.ending_time,
shift__ending_time__gte=shift.ending_time,
)
hard_conflict_query_set = query_set.exclude(
shift__starting_time__lt=graced_start, shift__ending_time__lte=graced_start
)
hard_conflict_query_set = hard_conflict_query_set.exclude(
shift__starting_time__gte=graced_end, shift__ending_time__gte=graced_end
)
return hard_conflict_query_set, soft_conflict_query_set