#    Copyright (C) 2017-2018  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/>.

from functools import reduce

from django.db import models
from django.db.models import Manager, Q
from django.db.models.functions import ExtractWeek, ExtractYear
from django.utils import timezone
from django.utils.text import slugify

from .utils import parse_group


class SlugModel(models.Model):
    def save(self, *args, **kwargs):
        if not self.slug:
            self.slug = slugify(self.name)

        super(SlugModel, self).save(*args, **kwargs)

    class Meta:
        abstract = True


class Year(SlugModel):
    name = models.CharField(max_length=16, verbose_name="année")
    slug = models.SlugField(max_length=16, unique=True, default="")

    def __str__(self):
        return self.name

    class Meta:
        verbose_name = "année"
        verbose_name_plural = "années"


class Source(models.Model):
    url = models.URLField(max_length=255, verbose_name="URL", unique=True)
    last_update_date = models.DateTimeField(null=True, blank=True,
                                            verbose_name="dernière mise à jour"
                                            " Celcat")

    def __str__(self):
        return self.url

    @property
    def formatted_timetables(self):
        return ", ".join([str(timetable) for timetable in
                          self.timetables.all()])

    class Meta:
        verbose_name = "source d’emploi du temps"
        verbose_name_plural = "sources d’emploi du temps"


class TimetableManager(Manager):
    def get_queryset(self):
        return super(Manager, self).get_queryset().select_related("year")


class Timetable(SlugModel):
    objects = TimetableManager()

    year = models.ForeignKey(Year, on_delete=models.CASCADE,
                             verbose_name="année")
    name = models.CharField(max_length=64, verbose_name="nom")
    slug = models.SlugField(max_length=64, default="")
    source = models.ForeignKey(Source, on_delete=models.CASCADE,
                               verbose_name="source",
                               related_name="timetables")

    def __str__(self):
        return self.year.name + " " + self.name

    class Meta:
        unique_together = (("year", "name"), ("year", "slug"),)
        verbose_name = "emploi du temps"
        verbose_name_plural = "emplois du temps"


class GroupManager(Manager):
    def get_parents(self, group):
        groups_criteria = reduce(lambda x, y: x | y,
                                 [Q(subgroup=group.subgroup[:i])
                                  for i in range(1, len(group.subgroup) + 1)],
                                 Q(subgroup=""))

        return self.get_queryset().filter(groups_criteria,
                                          Q(semester=None) |
                                          Q(semester=group.semester),
                                          mention=group.mention,
                                          source=group.source)


class Group(SlugModel):
    objects = GroupManager()

    name = models.CharField(max_length=255, verbose_name="nom")
    celcat_name = models.CharField(max_length=255,
                                   verbose_name="nom dans Celcat")
    source = models.ForeignKey(Source, on_delete=models.CASCADE,
                               verbose_name="source d’emploi du temps")

    mention = models.CharField(max_length=128)
    semester = models.IntegerField(verbose_name="semestre", null=True)
    subgroup = models.CharField(max_length=16, verbose_name="sous-groupe",
                                default="")

    slug = models.SlugField(max_length=64, default="")

    hidden = models.BooleanField(verbose_name="caché", default=False)

    def corresponds_to(self, mention, semester, subgroup):
        subgroup_corresponds = True
        if self.subgroup is not None and subgroup is not None:
            subgroup_corresponds = self.subgroup.startswith(subgroup)

        return (self.mention.startswith(mention) or
                mention.startswith(self.mention)) and \
               (self.semester == semester or semester is None) and \
            subgroup_corresponds

    @property
    def group_info(self):
        return self.mention, self.semester, self.subgroup

    def __str__(self):
        return self.name

    def save(self, *args, **kwargs):
        if self.name == "":
            self.name = self.celcat_name

        self.mention, self.semester, self.subgroup = parse_group(self.name)
        if self.subgroup is None:
            self.subgroup = ""

        super(Group, self).save(*args, **kwargs)

    class Meta:
        index_together = ("mention", "semester", "subgroup",)
        unique_together = (("name", "source",),
                           ("celcat_name", "source",),
                           ("slug", "source",),)

        verbose_name = "groupe"
        verbose_name_plural = "groupes"


class RoomManager(Manager):
    def qsjps(self, begin, end):
        # On récupère la liste des cours qui commencent avant la fin
        # de l’intervalle sélectionné et qui terminent après le début
        # de l’intervalle, c’est-à-dire qu’au moins une partie du
        # cours se déroule pendant l’intervalle. On récupère ensuite
        # la liste des salles dans lesquelles se déroulent ces
        # cours. On exclu les cours n’ayant aucune salle assignée.
        courses = Course.objects.filter(begin__lt=end, end__gt=begin,
                                        rooms__isnull=False) \
                                .values_list("rooms", flat=True)

        # On sélectionne ensuite les salles qui ne sont pas dans la
        # liste récupérée plus haut, et on les trie par leur nom.
        return self.get_queryset().exclude(pk__in=courses).order_by("name")


class Room(SlugModel):
    objects = RoomManager()

    name = models.CharField(max_length=255, unique=True, verbose_name="nom")
    slug = models.SlugField(max_length=64, default="", unique=True)

    def __str__(self):
        return self.name

    class Meta:
        verbose_name = "salle"
        verbose_name_plural = "salles"


class CourseManager(Manager):
    def get_courses(self, obj, **criteria):
        qs = self.get_queryset()
        if isinstance(obj, Group):
            qs = qs.filter(
                groups__in=Group.objects.get_parents(obj), **criteria) \
                   .prefetch_related("rooms")
        elif isinstance(obj, Room):
            qs = qs.filter(rooms=obj, **criteria) \
                   .prefetch_related("rooms", "groups")
        else:
            raise(TypeError, "obj must be a Group or a Room")

        return qs.order_by("begin")

    def get_weeks(self, **criteria):
        return self.get_queryset() \
                   .filter(**criteria) \
                   .order_by("groups__name", "year", "week") \
                   .annotate(year=ExtractYear("begin"),
                             week=ExtractWeek("begin")) \
                   .values("groups__mention", "groups__semester",
                           "groups__subgroup", "year", "week")


class Course(models.Model):
    objects = CourseManager()

    name = models.CharField(max_length=255, verbose_name="nom",
                            default="Sans nom")
    type_ = models.CharField(name="type", max_length=255,
                             verbose_name="type de cours", null=True)
    source = models.ForeignKey(Source, on_delete=models.CASCADE,
                               verbose_name="emploi du temps")
    notes = models.TextField(verbose_name="remarques", blank=True, null=True)

    groups = models.ManyToManyField(Group, verbose_name="groupes")
    rooms = models.ManyToManyField(Room, verbose_name="salles")

    begin = models.DateTimeField(verbose_name="début du cours", db_index=True)
    end = models.DateTimeField(verbose_name="fin du cours")

    last_update = models.DateTimeField(verbose_name="dernière mise à jour",
                                       default=timezone.now)

    def __str__(self):
        return self.name

    def save(self, *args, **kwargs):
        if self.type is not None:
            self.type = self.type.replace("COURS", "cours")
            self.type = self.type.replace("REUNION", "réunion")
        if self.name is not None:
            self.name = self.name.split("(")[0].strip()

        super(Course, self).save(*args, **kwargs)

    class Meta:
        verbose_name = "cours"
        verbose_name_plural = "cours"