Skip to content
Snippets Groups Projects

refactor collection filters with django_filters

Merged Konrad Mohrfeldt requested to merge refactor-filters into master
2 files
+ 104
97
Compare changes
  • Side-by-side
  • Inline
Files
2
  • This change re-implements all existing collection filters for the
    APITimeSlotViewSet with a FilterSet. No breaking changes are expected,
    though there are some changes in semantics:
    
    * The start and end query parameters no longer need to be specified
      together. If users only want to modify the start or end date they
      can now do that.
      If start is specified and end is not, end will be start + 60 days.
    * If end was not set it would default to start + 60 days at 00:00.
      This is now fixed and end will be start + 60 days at 23:59:59.
    * end now uses time.max, which selects the latest possible time on
      the specified date.
    * The surrounding-filter now uses the value of the start filter
      by default. There might not be a use-case for this, but it seemed
      like a good default.
    * All filters are now applied in series. This wasn’t the case for
      every filter, e.g. the surrounding-filter would return early.
+ 88
0
import datetime
from django_filters import rest_framework as filters
from django import forms
from django.contrib.auth.models import User
from django.db.models import Q, QuerySet
from django.utils import timezone
@@ -90,3 +93,88 @@ class ShowFilterSet(StaticFilterHelpTextMixin, filters.FilterSet):
"topic",
"type",
]
class TimeSlotFilterSet(filters.FilterSet):
order = filters.OrderingFilter(
fields=[field.name for field in models.TimeSlot._meta.get_fields()]
)
surrounding = filters.BooleanFilter(
method="filter_surrounding",
label="Return surrounding timeslots",
help_text=(
"Returns the 10 nearest timeslots for the current date if set to true. "
"No filtering is performed if set to false. "
"If specified without a value true is assumed."
),
)
# The start/end filters will always be applied even if no query parameter has been set.
# This is because we enforce a value in the clean_start and clean_end methods
# of the filterset form.
start = filters.DateFilter(
method="filter_start",
help_text=(
"Only returns timeslots after that start on or after the specified date. "
"By default, this is set to the current date."
),
)
end = filters.DateFilter(
method="filter_end",
help_text=(
"Only returns timeslots that end on or before the specified date. "
"By default, this is set to value of the start filter + 60 days."
),
)
def filter_surrounding(self, queryset: QuerySet, name: str, value: bool):
if value is not True:
return queryset
start = self.form.cleaned_data.get("start", None) or timezone.now()
nearest_timeslots_in_future = (
models.TimeSlot.objects.filter(start__gte=start)
.order_by("start")
.values_list("id", flat=True)[:5]
)
nearest_timeslots_in_past = (
models.TimeSlot.objects.filter(start__lt=start)
.order_by("-start")
.values_list("id", flat=True)[:5]
)
relevant_timeslot_ids = list(nearest_timeslots_in_future) + list(
nearest_timeslots_in_past
)
return queryset.filter(id__in=relevant_timeslot_ids)
def filter_start(self, queryset: QuerySet, name: str, value: datetime.date):
start = timezone.make_aware(datetime.datetime.combine(value, datetime.time.min))
return queryset.filter(start__gte=start)
def filter_end(self, queryset: QuerySet, name: str, value: datetime.date):
end = timezone.make_aware(datetime.datetime.combine(value, datetime.time.max))
return queryset.filter(end__lte=end)
def filter_queryset(self, queryset):
queryset = super().filter_queryset(queryset)
# This is for backwards compatibility as the surrounding-filter was formerly implemented
# by just checking for the existence of the query parameter.
if self.request.GET.get("surrounding", None) == "":
queryset = self.filter_surrounding(queryset, "surrounding", True)
return queryset
class Meta:
model = models.TimeSlot
fields = [
"order",
"start",
"end",
"surrounding",
]
class form(forms.Form):
def clean_start(self):
start = self.cleaned_data.get("start", None)
return start or timezone.now().date()
def clean_end(self):
end = self.cleaned_data.get("end", None)
return end or self.cleaned_data["start"] + datetime.timedelta(days=60)
Loading