mirror of
				https://github.com/ae-utbm/sith.git
				synced 2025-11-04 11:03:04 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			492 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			492 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
# -*- coding:utf-8 -*
 | 
						|
#
 | 
						|
# Copyright 2016,2017
 | 
						|
# - Skia <skia@libskia.so>
 | 
						|
# - Sli <antoine@bartuccio.fr>
 | 
						|
#
 | 
						|
# Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM,
 | 
						|
# http://ae.utbm.fr.
 | 
						|
#
 | 
						|
# This program is free software; you can redistribute it and/or modify it under
 | 
						|
# the terms of the GNU General Public License a published by the Free Software
 | 
						|
# Foundation; either version 3 of the License, or (at your option) any later
 | 
						|
# version.
 | 
						|
#
 | 
						|
# This program 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 General Public License for more
 | 
						|
# details.
 | 
						|
#
 | 
						|
# You should have received a copy of the GNU General Public License along with
 | 
						|
# this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple
 | 
						|
# Place - Suite 330, Boston, MA 02111-1307, USA.
 | 
						|
#
 | 
						|
#
 | 
						|
 | 
						|
from django.db import models
 | 
						|
from django.core import validators
 | 
						|
from django.conf import settings
 | 
						|
from django.utils.translation import gettext_lazy as _
 | 
						|
from django.core.exceptions import ValidationError, ObjectDoesNotExist
 | 
						|
from django.db import transaction
 | 
						|
from django.urls import reverse
 | 
						|
from django.utils import timezone
 | 
						|
from django.core.validators import RegexValidator, validate_email
 | 
						|
from django.utils.functional import cached_property
 | 
						|
 | 
						|
from core.models import User, MetaGroup, Group, SithFile, RealGroup, Notification, Page
 | 
						|
 | 
						|
# 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, new_name):
 | 
						|
        c = Club.objects.filter(unix_name=new_name).first()
 | 
						|
        if c is None:
 | 
						|
            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)
 | 
						|
 | 
						|
    def save(self, *args, **kwargs):
 | 
						|
        with transaction.atomic():
 | 
						|
            creation = False
 | 
						|
            old = Club.objects.filter(id=self.id).first()
 | 
						|
            if not old:
 | 
						|
                creation = True
 | 
						|
            else:
 | 
						|
                if 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()
 | 
						|
 | 
						|
    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
 | 
						|
        """
 | 
						|
        return user.is_in_group(settings.SITH_MAIN_BOARD_GROUP)
 | 
						|
 | 
						|
    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
 | 
						|
 | 
						|
    _memberships = {}
 | 
						|
 | 
						|
    def get_membership_for(self, user):
 | 
						|
        """
 | 
						|
        Returns the current membership the given user
 | 
						|
        """
 | 
						|
        try:
 | 
						|
            return Club._memberships[self.id][user.id]
 | 
						|
        except:
 | 
						|
            m = self.members.filter(user=user.id).filter(end_date=None).first()
 | 
						|
            try:
 | 
						|
                Club._memberships[self.id][user.id] = m
 | 
						|
            except:
 | 
						|
                Club._memberships[self.id] = {}
 | 
						|
                Club._memberships[self.id][user.id] = m
 | 
						|
            return m
 | 
						|
 | 
						|
    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 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
 | 
						|
    )
 | 
						|
 | 
						|
    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
 | 
						|
        """
 | 
						|
        return user.is_in_group(settings.SITH_MAIN_BOARD_GROUP)
 | 
						|
 | 
						|
    def can_be_edited_by(self, user, membership=None):
 | 
						|
        """
 | 
						|
        Method to see if that object can be edited by the given user
 | 
						|
        """
 | 
						|
        if user.memberships:
 | 
						|
            if membership:  # This is for optimisation purpose
 | 
						|
                ms = membership
 | 
						|
            else:
 | 
						|
                ms = user.memberships.filter(club=self.club, end_date=None).first()
 | 
						|
            return (ms and ms.role >= self.role) or user.is_in_group(
 | 
						|
                settings.SITH_MAIN_BOARD_GROUP
 | 
						|
            )
 | 
						|
        return user.is_in_group(settings.SITH_MAIN_BOARD_GROUP)
 | 
						|
 | 
						|
    def get_absolute_url(self):
 | 
						|
        return reverse("club:club_members", kwargs={"club_id": self.club.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_in_group(settings.SITH_GROUP_COM_ADMIN_ID)
 | 
						|
 | 
						|
    def is_owned_by(self, user):
 | 
						|
        return (
 | 
						|
            user.is_in_group(self)
 | 
						|
            or user.is_root
 | 
						|
            or user.is_in_group(settings.SITH_GROUP_COM_ADMIN_ID)
 | 
						|
        )
 | 
						|
 | 
						|
    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):
 | 
						|
        for sub in self.subscriptions.all():
 | 
						|
            sub.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):
 | 
						|
        return (
 | 
						|
            self.mailing.club.has_rights_in_club(user)
 | 
						|
            or user.is_root
 | 
						|
            or self.user.is_in_group(settings.SITH_GROUP_COM_ADMIN_ID)
 | 
						|
        )
 | 
						|
 | 
						|
    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)
 |