# Copyright (C) 2019 Alban Gruin # # celcatsanitizer is free software: you can redistribute it and/or modify # it under the terms of the GNU Affero General Public License as published # by the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # celcatsanitizer is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU Affero General Public License for more details. # # You should have received a copy of the GNU Affero General Public License # along with celcatsanitizer. If not, see <http://www.gnu.org/licenses/>. import datetime from django.db.models import Count from django.db.models.functions import ExtractWeek, ExtractYear from django.utils import timezone from rest_framework import viewsets from rest_framework.decorators import action, detail_route from rest_framework.response import Response from ..forms import QSJPSForm from ..models import Course, Group, Room, Source, Timetable, Year from ..utils import get_current_or_next_week, get_week from .serializers import CourseSerializer, GroupSerializer, RoomSerializer, \ SourceSerializer, TimetableSerializer, YearSerializer class YearViewSet(viewsets.ReadOnlyModelViewSet): queryset = Year.objects.all().order_by("name") serializer_class = YearSerializer @detail_route(methods=["get"], url_path="timetables") def timetable_list(self, request, pk): year = self.get_object() timetables = Timetable.objects.filter(year=year).distinct() \ .order_by("name") timetables_json = TimetableSerializer( self.paginate_queryset(timetables), many=True) return self.get_paginated_response(timetables_json.data) class SourceViewSet(viewsets.ReadOnlyModelViewSet): queryset = Source.objects.all().order_by("pk") serializer_class = SourceSerializer @detail_route(methods=["get"], url_path="timetables") def timetable_list(self, request, pk): source = self.get_object() timetables = Timetable.objects.filter(source=source).distinct() \ .order_by("name") timetables_json = TimetableSerializer( self.paginate_queryset(timetables), many=True) return self.get_paginated_response(timetables_json.data) class TimetableViewSet(viewsets.ReadOnlyModelViewSet): queryset = Timetable.objects.all().select_related("source") \ .order_by("year", "name") serializer_class = TimetableSerializer @detail_route(methods=["get"], url_path="groups") def group_list(self, request, pk): timetable = self.get_object() groups = Group.objects.filter(source=timetable.source).distinct() \ .order_by("name") groups_json = GroupSerializer(self.paginate_queryset(groups), many=True) return self.get_paginated_response(groups_json.data) class CourseListGroupSet(viewsets.ReadOnlyModelViewSet): @detail_route(methods=["get"], url_path="courses") def course_list(self, request, pk): obj = self.get_object() courses = Course.objects.get_courses(obj).prefetch_related("groups") courses_json = CourseSerializer(self.paginate_queryset(courses), many=True) return self.get_paginated_response(courses_json.data) def __get_courses(self, obj, start, end): courses = Course.objects.get_courses(obj, begin__gte=start, end__lt=end) \ .prefetch_related("groups") courses_json = CourseSerializer(self.paginate_queryset(courses), many=True) return self.get_paginated_response(courses_json.data) @detail_route(methods=["get"], url_path="courses/days/current") def current_day(self, request, pk): obj = self.get_object() start = datetime.date.today() end = start + datetime.timedelta(days=1) return self.__get_courses(obj, start, end) @detail_route(methods=["get"], url_path="courses/days/(?P<year>\d+)/(?P<month>\d+)/(?P<day>\d+)") def other_day(self, request, pk, year, month, day): obj = self.get_object() try: start = datetime.date(int(year), int(month), int(day)) except ValueError as v: errors = {} message = v.args[0] if message.split(" ")[0] == "month": errors["month"] = "Rentrez un mois invalide" else: errors["day"] = "Numéro de jour invalide pour le mois" return Response(errors, status=400) end = start + datetime.timedelta(days=1) return self.__get_courses(obj, start, end) @detail_route(methods=["get"], url_path="courses/weeks/current") def current_week(self, request, pk): obj = self.get_object() start, end = get_week(*get_current_or_next_week()) return self.__get_courses(obj, start, end) @detail_route(methods=["get"], url_path="courses/weeks/(?P<year>\d+)/(?P<week>\d+)") def other_week(self, request, pk, year, week): obj = self.get_object() errors = {} if not year.isdigit(): errors["year"] = "Rentrez une année valide" if not week.isdigit() or not 0 < int(week) <= 53: errors["week"] = "Rentrez une semaine valide" if errors: return Response(errors, status=400) return self.__get_courses(obj, *get_week(int(year), int(week))) class GroupViewSet(CourseListGroupSet): queryset = Group.objects.all().order_by("name") serializer_class = GroupSerializer @detail_route(methods=["get"], url_path="courses/weeks") def weeks(self, request, pk): group = self.get_object() groups = Group.objects.get_parents(group) courses = Course.objects.filter(groups__in=groups) \ .order_by("year", "week") \ .annotate(year=ExtractYear("begin"), week=ExtractWeek("begin")) \ .values("year", "week") \ .annotate(c=Count("*")) weeks = [get_week(course["year"], course["week"])[0] for course in courses] return Response(weeks) class RoomViewSet(CourseListGroupSet): queryset = Room.objects.all().order_by("name") serializer_class = RoomSerializer @action( methods=["get"], detail=False, url_path="qsjps/(?P<day>[0-9\-]+)/(?P<begin>[0-9:]+)/(?P<end>[0-9:]+)") def qsjps(self, request, day, begin, end): form = QSJPSForm({"day": day, "begin": begin, "end": end}) if not form.is_valid(): return Response(form.errors, status=400) day = form.cleaned_data["day"] begin_hour = form.cleaned_data["begin"] end_hour = form.cleaned_data["end"] begin = timezone.make_aware(datetime.datetime.combine(day, begin_hour)) end = timezone.make_aware(datetime.datetime.combine(day, end_hour)) rooms = Room.objects.qsjps(begin, end) rooms_json = RoomSerializer(rooms, many=True) return Response(rooms_json.data) class CourseViewSet(viewsets.ReadOnlyModelViewSet): queryset = Course.objects.all().prefetch_related("groups", "rooms") serializer_class = CourseSerializer