# -*- coding:utf-8 -*- # # Copyright 2023 © AE UTBM # ae@utbm.fr / ae.info@utbm.fr # All contributors are listed in the CONTRIBUTORS file. # # This file is part of the website of the UTBM Student Association (AE UTBM), # https://ae.utbm.fr. # # You can find the whole source code at https://github.com/ae-utbm/sith3 # # LICENSED UNDER THE GNU GENERAL PUBLIC LICENSE VERSION 3 (GPLv3) # SEE : https://raw.githubusercontent.com/ae-utbm/sith3/master/LICENSE # OR WITHIN THE LOCAL FILE "LICENSE" # # PREVIOUSLY LICENSED UNDER THE MIT LICENSE, # SEE : https://raw.githubusercontent.com/ae-utbm/sith3/master/LICENSE.old # OR WITHIN THE LOCAL FILE "LICENSE.old" # from typing import Optional from django.conf import settings from django.core import validators from django.core.cache import cache from django.core.exceptions import ObjectDoesNotExist, ValidationError from django.core.validators import RegexValidator, validate_email from django.db import models, transaction from django.db.models import Q from django.urls import reverse from django.utils import timezone from django.utils.functional import cached_property from django.utils.translation import gettext_lazy as _ from core.models import Group, MetaGroup, Notification, Page, RealGroup, SithFile, User # Create your models here. class Club(models.Model): """ The Club class, made as a tree to allow nice tidy organization """ id = models.AutoField(primary_key=True, db_index=True) name = models.CharField(_("name"), max_length=64) parent = models.ForeignKey( "Club", related_name="children", null=True, blank=True, on_delete=models.CASCADE ) unix_name = models.CharField( _("unix name"), max_length=30, unique=True, validators=[ validators.RegexValidator( r"^[a-z0-9][a-z0-9._-]*[a-z0-9]$", _( "Enter a valid unix name. This value may contain only " "letters, numbers ./-/_ characters." ), ) ], error_messages={"unique": _("A club with that unix name already exists.")}, ) logo = models.ImageField( upload_to="club_logos", verbose_name=_("logo"), null=True, blank=True ) is_active = models.BooleanField(_("is active"), default=True) short_description = models.CharField( _("short description"), max_length=1000, default="", blank=True, null=True ) address = models.CharField(_("address"), max_length=254) # This function prevents generating migration upon settings change def get_default_owner_group(): return settings.SITH_GROUP_ROOT_ID owner_group = models.ForeignKey( Group, related_name="owned_club", default=get_default_owner_group, on_delete=models.CASCADE, ) edit_groups = models.ManyToManyField( Group, related_name="editable_club", blank=True ) view_groups = models.ManyToManyField( Group, related_name="viewable_club", blank=True ) home = models.OneToOneField( SithFile, related_name="home_of_club", verbose_name=_("home"), null=True, blank=True, on_delete=models.SET_NULL, ) page = models.OneToOneField( Page, related_name="club", blank=True, null=True, on_delete=models.CASCADE ) class Meta: ordering = ["name", "unix_name"] @cached_property def president(self): return self.members.filter( role=settings.SITH_CLUB_ROLES_ID["President"], end_date=None ).first() def check_loop(self): """Raise a validation error when a loop is found within the parent list""" objs = [] cur = self while cur.parent is not None: if cur in objs: raise ValidationError(_("You can not make loops in clubs")) objs.append(cur) cur = cur.parent def clean(self): self.check_loop() def _change_unixname(self, old_name, new_name): c = Club.objects.filter(unix_name=new_name).first() if c is None: # Update all the groups names Group.objects.filter(name=old_name).update(name=new_name) Group.objects.filter(name=old_name + settings.SITH_BOARD_SUFFIX).update( name=new_name + settings.SITH_BOARD_SUFFIX ) Group.objects.filter(name=old_name + settings.SITH_MEMBER_SUFFIX).update( name=new_name + settings.SITH_MEMBER_SUFFIX ) if self.home: self.home.name = new_name self.home.save() else: raise ValidationError(_("A club with that unix_name already exists")) def make_home(self): if not self.home: home_root = SithFile.objects.filter(parent=None, name="clubs").first() root = User.objects.filter(username="root").first() if home_root and root: home = SithFile(parent=home_root, name=self.unix_name, owner=root) home.save() self.home = home self.save() def make_page(self): root = User.objects.filter(username="root").first() if not self.page: club_root = Page.objects.filter(name=settings.SITH_CLUB_ROOT_PAGE).first() if root and club_root: public = Group.objects.filter(id=settings.SITH_GROUP_PUBLIC_ID).first() p = Page(name=self.unix_name) p.parent = club_root p.save(force_lock=True) if public: p.view_groups.add(public) p.save(force_lock=True) if self.parent and self.parent.page: p.parent = self.parent.page self.page = p self.save() elif self.page and self.page.name != self.unix_name: self.page.unset_lock() self.page.name = self.unix_name self.page.save(force_lock=True) elif ( self.page and self.parent and self.parent.page and self.page.parent != self.parent.page ): self.page.unset_lock() self.page.parent = self.parent.page self.page.save(force_lock=True) @transaction.atomic() def save(self, *args, **kwargs): old = Club.objects.filter(id=self.id).first() creation = old is None if not creation and old.unix_name != self.unix_name: self._change_unixname(self.unix_name) super(Club, self).save(*args, **kwargs) if creation: board = MetaGroup(name=self.unix_name + settings.SITH_BOARD_SUFFIX) board.save() member = MetaGroup(name=self.unix_name + settings.SITH_MEMBER_SUFFIX) member.save() subscribers = Group.objects.filter( name=settings.SITH_MAIN_MEMBERS_GROUP ).first() self.make_home() self.home.edit_groups.set([board]) self.home.view_groups.set([member, subscribers]) self.home.save() self.make_page() cache.set(f"sith_club_{self.unix_name}", self) def delete(self, *args, **kwargs): super().delete(*args, **kwargs) # Invalidate the cache of this club and of its memberships for membership in self.members.ongoing().select_related("user"): cache.delete(f"membership_{self.id}_{membership.user.id}") cache.delete(f"sith_club_{self.unix_name}") def __str__(self): return self.name def get_absolute_url(self): return reverse("club:club_view", kwargs={"club_id": self.id}) def get_display_name(self): return self.name def is_owned_by(self, user): """ Method to see if that object can be super edited by the given user """ if user.is_anonymous: return False return user.is_board_member def get_full_logo_url(self): return "https://%s%s" % (settings.SITH_URL, self.logo.url) def can_be_edited_by(self, user): """ Method to see if that object can be edited by the given user """ return self.has_rights_in_club(user) def can_be_viewed_by(self, user): """ Method to see if that object can be seen by the given user """ sub = User.objects.filter(pk=user.pk).first() if sub is None: return False return sub.was_subscribed def get_membership_for(self, user: User) -> Optional["Membership"]: """ Return the current membership the given user. The result is cached. """ if user.is_anonymous: return None membership = cache.get(f"membership_{self.id}_{user.id}") if membership == "not_member": return None if membership is None: membership = self.members.filter(user=user, end_date=None).first() if membership is None: cache.set(f"membership_{self.id}_{user.id}", "not_member") else: cache.set(f"membership_{self.id}_{user.id}", membership) return membership def has_rights_in_club(self, user): m = self.get_membership_for(user) return m is not None and m.role > settings.SITH_MAXIMUM_FREE_ROLE class MembershipQuerySet(models.QuerySet): def ongoing(self) -> "MembershipQuerySet": """ Filter all memberships which are not finished yet """ # noinspection PyTypeChecker return self.filter(Q(end_date=None) | Q(end_date__gte=timezone.now())) def board(self) -> "MembershipQuerySet": """ Filter all memberships where the user is/was in the board. Be aware that users who were in the board in the past are included, even if there are no more members. If you want to get the users who are currently in the board, mind combining this with the :meth:`ongoing` queryset method """ # noinspection PyTypeChecker return self.filter(role__gt=settings.SITH_MAXIMUM_FREE_ROLE) def update(self, **kwargs): """ Work just like the default Django's update() method, but add a cache refresh for the elements of the queryset. Be aware that this adds a db query to retrieve the updated objects """ nb_rows = super().update(**kwargs) if nb_rows > 0: # if at least a row was affected, refresh the cache for membership in self.all(): if membership.end_date is not None: cache.set( f"membership_{membership.club_id}_{membership.user_id}", "not_member", ) else: cache.set( f"membership_{membership.club_id}_{membership.user_id}", membership, ) def delete(self): """ Work just like the default Django's delete() method, but add a cache invalidation for the elements of the queryset before the deletion. Be aware that this adds a db query to retrieve the deleted element. As this first query take place before the deletion operation, it will be performed even if the deletion fails. """ ids = list(self.values_list("club_id", "user_id")) nb_rows, _ = super().delete() if nb_rows > 0: for club_id, user_id in ids: cache.set(f"membership_{club_id}_{user_id}", "not_member") class Membership(models.Model): """ The Membership class makes the connection between User and Clubs Both Users and Clubs can have many Membership objects: - a user can be a member of many clubs at a time - a club can have many members at a time too A User is currently member of all the Clubs where its Membership has an end_date set to null/None. Otherwise, it's a past membership kept because it can be very useful to see who was in which Club in the past. """ user = models.ForeignKey( User, verbose_name=_("user"), related_name="memberships", null=False, blank=False, on_delete=models.CASCADE, ) club = models.ForeignKey( Club, verbose_name=_("club"), related_name="members", null=False, blank=False, on_delete=models.CASCADE, ) start_date = models.DateField(_("start date"), default=timezone.now) end_date = models.DateField(_("end date"), null=True, blank=True) role = models.IntegerField( _("role"), choices=sorted(settings.SITH_CLUB_ROLES.items()), default=sorted(settings.SITH_CLUB_ROLES.items())[0][0], ) description = models.CharField( _("description"), max_length=128, null=False, blank=True ) objects = MembershipQuerySet.as_manager() def __str__(self): return ( self.club.name + " - " + self.user.username + " - " + str(settings.SITH_CLUB_ROLES[self.role]) + str(" - " + str(_("past member")) if self.end_date is not None else "") ) def is_owned_by(self, user): """ Method to see if that object can be super edited by the given user """ if user.is_anonymous: return False return user.is_board_member def can_be_edited_by(self, user: User) -> bool: """ Check if that object can be edited by the given user """ if user.is_root or user.is_board_member: return True membership = self.club.get_membership_for(user) if membership is not None and membership.role >= self.role: return True return False def get_absolute_url(self): return reverse("club:club_members", kwargs={"club_id": self.club_id}) def save(self, *args, **kwargs): super().save(*args, **kwargs) if self.end_date is None: cache.set(f"membership_{self.club_id}_{self.user_id}", self) else: cache.set(f"membership_{self.club_id}_{self.user_id}", "not_member") def delete(self, *args, **kwargs): super().delete(*args, **kwargs) cache.delete(f"membership_{self.club_id}_{self.user_id}") class Mailing(models.Model): """ This class correspond to a mailing list Remember that mailing lists should be validated by UTBM """ club = models.ForeignKey( Club, verbose_name=_("Club"), related_name="mailings", null=False, blank=False, on_delete=models.CASCADE, ) email = models.CharField( _("Email address"), unique=True, null=False, blank=False, max_length=256, validators=[ RegexValidator( validate_email.user_regex, _("Enter a valid address. Only the root of the address is needed."), ) ], ) is_moderated = models.BooleanField(_("is moderated"), default=False) moderator = models.ForeignKey( User, related_name="moderated_mailings", verbose_name=_("moderator"), null=True, on_delete=models.CASCADE, ) def clean(self): if Mailing.objects.filter(email=self.email).exists(): raise ValidationError(_("This mailing list already exists.")) if self.can_moderate(self.moderator): self.is_moderated = True else: self.moderator = None super(Mailing, self).clean() @property def email_full(self): return self.email + "@" + settings.SITH_MAILING_DOMAIN def can_moderate(self, user): return user.is_root or user.is_com_admin def is_owned_by(self, user): if user.is_anonymous: return False return user.is_root or user.is_com_admin def can_view(self, user): return self.club.has_rights_in_club(user) def can_be_edited_by(self, user): return self.club.has_rights_in_club(user) def delete(self, *args, **kwargs): self.subscriptions.all().delete() super(Mailing, self).delete() def fetch_format(self): resp = self.email + ": " for sub in self.subscriptions.all(): resp += sub.fetch_format() return resp def save(self): if not self.is_moderated: for user in ( RealGroup.objects.filter(id=settings.SITH_GROUP_COM_ADMIN_ID) .first() .users.all() ): if not user.notifications.filter( type="MAILING_MODERATION", viewed=False ).exists(): Notification( user=user, url=reverse("com:mailing_admin"), type="MAILING_MODERATION", ).save() super(Mailing, self).save() def __str__(self): return "%s - %s" % (self.club, self.email_full) class MailingSubscription(models.Model): """ This class makes the link between user and mailing list """ mailing = models.ForeignKey( Mailing, verbose_name=_("Mailing"), related_name="subscriptions", null=False, blank=False, on_delete=models.CASCADE, ) user = models.ForeignKey( User, verbose_name=_("User"), related_name="mailing_subscriptions", null=True, blank=True, on_delete=models.CASCADE, ) email = models.EmailField(_("Email address"), blank=False, null=False) class Meta: unique_together = (("user", "email", "mailing"),) def clean(self): if not self.user and not self.email: raise ValidationError(_("At least user or email is required")) try: if self.user and not self.email: self.email = self.user.email if MailingSubscription.objects.filter( mailing=self.mailing, email=self.email ).exists(): raise ValidationError( _("This email is already suscribed in this mailing") ) except ObjectDoesNotExist: pass super(MailingSubscription, self).clean() def is_owned_by(self, user): if user.is_anonymous: return False return ( self.mailing.club.has_rights_in_club(user) or user.is_root or self.user.is_com_admin ) def can_be_edited_by(self, user): return self.user is not None and user.id == self.user.id @cached_property def get_email(self): if self.user and not self.email: return self.user.email return self.email @cached_property def get_username(self): if self.user: return str(self.user) return _("Unregistered user") def fetch_format(self): return self.get_email + " " def __str__(self): return "(%s) - %s : %s" % (self.mailing, self.get_username, self.email)