1
0
mirror of https://gitlab.crans.org/bde/nk20 synced 2025-02-06 07:13:00 +00:00
nk20/apps/member/views.py

440 lines
16 KiB
Python
Raw Normal View History

2020-02-18 21:30:26 +01:00
# Copyright (C) 2018-2020 by BDE ENS Paris-Saclay
2019-08-10 19:01:15 +02:00
# SPDX-License-Identifier: GPL-3.0-or-later
2020-02-18 21:30:26 +01:00
2020-03-07 22:28:59 +01:00
import io
2020-03-31 23:54:14 +02:00
from datetime import datetime
2020-03-07 22:28:59 +01:00
from PIL import Image
from django.conf import settings
2019-08-10 19:01:15 +02:00
from django.contrib.auth.mixins import LoginRequiredMixin
2020-03-07 22:28:59 +01:00
from django.contrib.auth.models import User
2020-03-19 16:12:52 +01:00
from django.contrib.auth.views import LoginView
2020-03-07 22:28:59 +01:00
from django.db.models import Q
2020-04-01 03:42:19 +02:00
from django.forms import HiddenInput
2020-02-17 23:30:55 +01:00
from django.shortcuts import redirect
2020-03-07 22:28:59 +01:00
from django.urls import reverse_lazy
2019-08-10 19:01:15 +02:00
from django.utils.translation import gettext_lazy as _
2020-03-27 14:19:55 +01:00
from django.views.generic import CreateView, DetailView, UpdateView, TemplateView
2020-02-28 13:37:31 +01:00
from django.views.generic.edit import FormMixin
from django_tables2.views import SingleTableView
2020-02-17 19:25:33 +01:00
from rest_framework.authtoken.models import Token
from note.forms import ImageForm
2020-02-18 21:14:29 +01:00
from note.models import Alias, NoteUser
from note.models.transactions import Transaction
2020-03-31 14:57:44 +02:00
from note.tables import HistoryTable, AliasTable
2020-03-20 14:43:35 +01:00
from permission.backends import PermissionBackend
from permission.views import ProtectQuerysetMixin
2020-03-20 18:13:34 +01:00
2020-03-07 22:28:59 +01:00
from .filters import UserFilter, UserFilterFormHelper
2020-03-31 23:54:14 +02:00
from .forms import SignUpForm, ProfileForm, ClubForm, MembershipForm, CustomAuthenticationForm
2020-03-07 22:28:59 +01:00
from .models import Club, Membership
2020-04-01 03:42:19 +02:00
from .tables import ClubTable, UserTable, MembershipTable
2019-09-23 12:50:14 +02:00
2020-02-18 12:31:15 +01:00
2020-03-19 16:12:52 +01:00
class CustomLoginView(LoginView):
form_class = CustomAuthenticationForm
def form_valid(self, form):
self.request.session['permission_mask'] = form.cleaned_data['permission_mask'].rank
return super().form_valid(form)
2019-08-12 00:30:29 +02:00
class UserCreateView(CreateView):
2019-08-11 16:22:52 +02:00
"""
Une vue pour inscrire un utilisateur et lui créer un profile
"""
form_class = SignUpForm
2019-08-11 16:22:52 +02:00
success_url = reverse_lazy('login')
2020-02-18 12:31:15 +01:00
template_name = 'member/signup.html'
second_form = ProfileForm
2019-08-11 16:22:52 +02:00
2020-02-18 12:31:15 +01:00
def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
2020-02-03 19:24:23 +01:00
context["profile_form"] = self.second_form()
2019-08-11 16:22:52 +02:00
return context
2019-08-11 17:39:05 +02:00
def form_valid(self, form):
profile_form = ProfileForm(self.request.POST)
if form.is_valid() and profile_form.is_valid():
2020-03-09 18:08:37 +01:00
user = form.save(commit=False)
user.profile = profile_form.save(commit=False)
user.save()
user.profile.save()
2019-08-11 17:39:05 +02:00
return super().form_valid(form)
2019-08-11 23:25:27 +02:00
2020-02-18 12:31:15 +01:00
class UserUpdateView(ProtectQuerysetMixin, LoginRequiredMixin, UpdateView):
2020-02-03 19:25:05 +01:00
model = User
2020-02-18 12:31:15 +01:00
fields = ['first_name', 'last_name', 'username', 'email']
2020-02-03 19:25:05 +01:00
template_name = 'member/profile_update.html'
2020-02-27 20:56:06 +01:00
context_object_name = 'user_object'
2020-03-03 14:25:16 +01:00
profile_form = ProfileForm
2020-02-18 12:31:15 +01:00
def get_context_data(self, **kwargs):
2020-02-03 19:25:05 +01:00
context = super().get_context_data(**kwargs)
2020-03-03 14:25:16 +01:00
context['profile_form'] = self.profile_form(instance=context['user_object'].profile)
2020-02-21 11:17:14 +01:00
context['title'] = _("Update Profile")
2020-02-03 19:25:05 +01:00
return context
def get_form(self, form_class=None):
2020-02-17 11:36:46 +01:00
form = super().get_form(form_class)
if 'username' not in form.data:
return form
new_username = form.data['username']
2020-02-17 11:36:46 +01:00
# Si l'utilisateur cherche à modifier son pseudo, le nouveau pseudo ne doit pas être proche d'un alias existant
2020-02-18 12:31:15 +01:00
note = NoteUser.objects.filter(
alias__normalized_name=Alias.normalize(new_username))
2020-02-27 20:56:06 +01:00
if note.exists() and note.get().user != self.object:
2020-02-18 12:31:15 +01:00
form.add_error('username',
_("An alias with a similar name already exists."))
return form
2020-02-03 19:25:05 +01:00
def form_valid(self, form):
2020-02-18 12:31:15 +01:00
profile_form = ProfileForm(
data=self.request.POST,
2020-02-27 20:56:06 +01:00
instance=self.object.profile,
2020-02-18 12:31:15 +01:00
)
2020-02-03 19:25:05 +01:00
if form.is_valid() and profile_form.is_valid():
new_username = form.data['username']
alias = Alias.objects.filter(name=new_username)
# Si le nouveau pseudo n'est pas un de nos alias,
# on supprime éventuellement un alias similaire pour le remplacer
if not alias.exists():
2020-02-18 12:31:15 +01:00
similar = Alias.objects.filter(
normalized_name=Alias.normalize(new_username))
if similar.exists():
similar.delete()
2020-02-17 11:36:46 +01:00
user = form.save(commit=False)
2020-02-18 12:31:15 +01:00
profile = profile_form.save(commit=False)
2020-02-03 19:25:05 +01:00
profile.user = user
profile.save()
2020-02-17 11:36:46 +01:00
user.save()
2020-02-03 19:25:05 +01:00
return super().form_valid(form)
def get_success_url(self, **kwargs):
2020-02-18 12:31:15 +01:00
if kwargs:
return reverse_lazy('member:user_detail',
kwargs={'pk': kwargs['id']})
2020-02-03 19:25:05 +01:00
else:
2020-03-07 22:28:59 +01:00
return reverse_lazy('member:user_detail', args=(self.object.id,))
2020-02-18 12:31:15 +01:00
2020-02-03 19:25:05 +01:00
class UserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView):
"""
2020-02-21 11:53:37 +01:00
Affiche les informations sur un utilisateur, sa note, ses clubs...
"""
2020-02-25 22:55:27 +01:00
model = User
context_object_name = "user_object"
template_name = "member/profile_detail.html"
2020-02-18 12:31:15 +01:00
def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
2020-02-25 22:55:27 +01:00
user = context['user_object']
history_list = \
2020-03-31 01:03:30 +02:00
Transaction.objects.all().filter(Q(source=user.note) | Q(destination=user.note)).order_by("-id")\
.filter(PermissionBackend.filter_queryset(self.request.user, Transaction, "view"))
context['history_list'] = HistoryTable(history_list)
club_list = Membership.objects.all().filter(user=user)\
.filter(PermissionBackend.filter_queryset(self.request.user, Membership, "view")).only("club")
2020-04-01 03:42:19 +02:00
context['club_list'] = MembershipTable(data=club_list)
return context
2019-08-12 00:30:29 +02:00
2020-02-18 12:31:15 +01:00
class UserListView(ProtectQuerysetMixin, LoginRequiredMixin, SingleTableView):
"""
Affiche la liste des utilisateurs, avec une fonction de recherche statique
"""
2019-09-23 12:50:14 +02:00
model = User
table_class = UserTable
template_name = 'member/user_list.html'
filter_class = UserFilter
formhelper_class = UserFilterFormHelper
2020-02-18 12:31:15 +01:00
def get_queryset(self, **kwargs):
qs = super().get_queryset()
2020-02-18 12:31:15 +01:00
self.filter = self.filter_class(self.request.GET, queryset=qs)
2019-09-23 12:50:14 +02:00
self.filter.form.helper = self.formhelper_class()
return self.filter.qs
2020-02-18 12:31:15 +01:00
def get_context_data(self, **kwargs):
2019-09-23 12:50:14 +02:00
context = super().get_context_data(**kwargs)
context["filter"] = self.filter
return context
2020-03-27 14:19:55 +01:00
class ProfileAliasView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView):
2020-02-28 13:37:31 +01:00
model = User
2020-03-03 11:05:02 +01:00
template_name = 'member/profile_alias.html'
2020-02-28 13:37:31 +01:00
context_object_name = 'user_object'
2020-03-27 14:19:55 +01:00
2020-03-07 22:28:59 +01:00
def get_context_data(self, **kwargs):
2020-02-28 13:37:31 +01:00
context = super().get_context_data(**kwargs)
2020-03-25 18:00:40 +01:00
note = context['object'].note
2020-02-28 16:12:35 +01:00
context["aliases"] = AliasTable(note.alias_set.all())
2020-02-28 13:37:31 +01:00
return context
2020-03-07 22:28:59 +01:00
class PictureUpdateView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, DetailView):
2020-03-04 16:34:12 +01:00
form_class = ImageForm
2020-03-07 22:28:59 +01:00
2020-03-31 01:03:30 +02:00
def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
2020-03-07 22:28:59 +01:00
context['form'] = self.form_class(self.request.POST, self.request.FILES)
2020-03-04 16:34:12 +01:00
return context
2020-03-07 22:28:59 +01:00
2020-03-04 16:34:12 +01:00
def get_success_url(self):
return reverse_lazy('member:user_detail', kwargs={'pk': self.object.id})
2020-03-07 22:28:59 +01:00
def post(self, request, *args, **kwargs):
form = self.get_form()
2020-03-04 16:34:12 +01:00
self.object = self.get_object()
if form.is_valid():
return self.form_valid(form)
else:
print('is_invalid')
print(form)
return self.form_invalid(form)
2020-03-07 22:28:59 +01:00
def form_valid(self, form):
image_field = form.cleaned_data['image']
2020-03-05 23:32:01 +01:00
x = form.cleaned_data['x']
y = form.cleaned_data['y']
w = form.cleaned_data['width']
h = form.cleaned_data['height']
# image crop and resize
image_file = io.BytesIO(image_field.read())
2020-03-07 22:28:59 +01:00
# ext = image_field.name.split('.')[-1].lower()
# TODO: support GIF format
image = Image.open(image_file)
2020-03-07 22:28:59 +01:00
image = image.crop((x, y, x + w, y + h))
image_clean = image.resize((settings.PIC_WIDTH,
2020-03-07 22:28:59 +01:00
settings.PIC_RATIO * settings.PIC_WIDTH),
Image.ANTIALIAS)
image_file = io.BytesIO()
2020-03-07 22:28:59 +01:00
image_clean.save(image_file, "PNG")
image_field.file = image_file
# renaming
2020-03-07 17:58:41 +01:00
filename = "{}_pic.png".format(self.object.note.pk)
image_field.name = filename
self.object.note.display_image = image_field
2020-03-04 16:34:12 +01:00
self.object.note.save()
return super().form_valid(form)
2020-03-25 16:58:15 +01:00
class ProfilePictureUpdateView(PictureUpdateView):
model = User
template_name = 'member/profile_picture_update.html'
context_object_name = 'user_object'
2020-03-07 22:28:59 +01:00
2020-03-25 16:58:15 +01:00
2020-02-17 21:32:08 +01:00
class ManageAuthTokens(LoginRequiredMixin, TemplateView):
2020-02-17 19:25:33 +01:00
"""
2020-02-17 21:32:08 +01:00
Affiche le jeton d'authentification, et permet de le regénérer
2020-02-17 19:25:33 +01:00
"""
2020-02-17 21:32:08 +01:00
model = Token
template_name = "member/manage_auth_tokens.html"
2020-02-17 19:25:33 +01:00
2020-02-17 23:30:55 +01:00
def get(self, request, *args, **kwargs):
if 'regenerate' in request.GET and Token.objects.filter(user=request.user).exists():
2020-02-17 19:25:33 +01:00
Token.objects.get(user=self.request.user).delete()
2020-02-18 12:31:15 +01:00
return redirect(reverse_lazy('member:auth_token') + "?show",
permanent=True)
2020-02-17 19:25:33 +01:00
2020-02-17 23:30:55 +01:00
return super().get(request, *args, **kwargs)
2020-02-17 21:32:08 +01:00
2020-02-17 23:30:55 +01:00
def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
context['token'] = Token.objects.get_or_create(user=self.request.user)[0]
2020-02-17 19:25:33 +01:00
return context
2020-02-18 12:31:15 +01:00
2020-02-18 21:14:29 +01:00
# ******************************* #
# CLUB #
# ******************************* #
2019-08-12 00:30:29 +02:00
2020-02-18 12:31:15 +01:00
class ClubCreateView(ProtectQuerysetMixin, LoginRequiredMixin, CreateView):
2019-08-11 23:25:27 +02:00
"""
Create Club
"""
model = Club
form_class = ClubForm
success_url = reverse_lazy('member:club_list')
2020-03-25 16:58:15 +01:00
2020-02-18 12:31:15 +01:00
def form_valid(self, form):
2019-08-11 23:25:27 +02:00
return super().form_valid(form)
2020-03-27 00:40:35 +01:00
2020-02-18 12:31:15 +01:00
class ClubListView(ProtectQuerysetMixin, LoginRequiredMixin, SingleTableView):
2019-08-11 23:25:27 +02:00
"""
List existing Clubs
2019-08-11 23:25:27 +02:00
"""
model = Club
2019-08-15 21:49:32 +02:00
table_class = ClubTable
2019-08-12 00:30:29 +02:00
2020-02-18 12:31:15 +01:00
class ClubDetailView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView):
2019-08-11 23:25:27 +02:00
model = Club
2020-02-18 12:31:15 +01:00
context_object_name = "club"
2020-02-18 12:31:15 +01:00
def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
2020-03-31 23:54:14 +02:00
club = context["club"]
2020-03-31 23:54:14 +02:00
if PermissionBackend().has_perm(self.request.user, "member.change_club_membership_start", club):
club.update_membership_dates()
2020-03-31 01:03:30 +02:00
club_transactions = Transaction.objects.all().filter(Q(source=club.note) | Q(destination=club.note))\
.filter(PermissionBackend.filter_queryset(self.request.user, Transaction, "view")).order_by('-id')
2019-08-15 23:11:52 +02:00
context['history_list'] = HistoryTable(club_transactions)
2020-03-31 23:54:14 +02:00
club_member = Membership.objects.filter(
club=club,
date_start__lte=datetime.now().date(),
date_end__gte=datetime.now().date(),
).filter(PermissionBackend.filter_queryset(self.request.user, Membership, "view")).all()
2020-04-01 03:42:19 +02:00
context['member_list'] = MembershipTable(data=club_member)
empty_membership = Membership(
club=club,
user=User.objects.first(),
date_start=datetime.now().date(),
date_end=datetime.now().date(),
fee=0,
)
context["can_add_members"] = PermissionBackend()\
.has_perm(self.request.user, "member.add_membership", empty_membership)
return context
2020-03-27 14:19:55 +01:00
class ClubAliasView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView):
2020-03-25 18:00:40 +01:00
model = Club
template_name = 'member/club_alias.html'
context_object_name = 'club'
2020-03-27 14:19:55 +01:00
def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
note = context['object'].note
context["aliases"] = AliasTable(note.alias_set.all())
return context
2020-03-25 18:00:40 +01:00
2020-03-25 16:58:15 +01:00
class ClubUpdateView(ProtectQuerysetMixin, LoginRequiredMixin, UpdateView):
model = Club
context_object_name = "club"
form_class = ClubForm
template_name = "member/club_form.html"
2020-03-31 01:03:30 +02:00
def get_success_url(self):
return reverse_lazy("member:club_detail", kwargs={"pk": self.object.pk})
class ClubPictureUpdateView(PictureUpdateView):
model = Club
template_name = 'member/club_picture_update.html'
context_object_name = 'club'
def get_success_url(self):
return reverse_lazy('member:club_detail', kwargs={'pk': self.object.id})
2020-02-18 12:31:15 +01:00
class ClubAddMemberView(ProtectQuerysetMixin, LoginRequiredMixin, CreateView):
model = Membership
form_class = MembershipForm
template_name = 'member/add_members.html'
2020-02-18 12:31:15 +01:00
def get_context_data(self, **kwargs):
club = Club.objects.filter(PermissionBackend.filter_queryset(self.request.user, Club, "view"))\
.get(pk=self.kwargs["pk"])
context = super().get_context_data(**kwargs)
2020-03-25 17:42:54 +01:00
context['club'] = club
2020-02-21 18:28:21 +01:00
return context
2020-03-31 23:54:14 +02:00
def form_valid(self, form):
club = Club.objects.filter(PermissionBackend.filter_queryset(self.request.user, Club, "view"))\
.get(pk=self.kwargs["pk"])
user = self.request.user
2020-03-31 23:54:14 +02:00
form.instance.club = club
2020-04-01 03:42:19 +02:00
if user.profile.paid:
fee = club.membership_fee_paid
else:
fee = club.membership_fee_unpaid
if user.note.balance < fee and not Membership.objects.filter(
club=2,
user=user,
date_start__lte=datetime.now().date(),
date_end__gte=datetime.now().date(),
).exists():
# Users without a valid Kfet membership can't have a negative balance.
# Club 2 = Kfet (hard-code :'( )
# TODO Send a notification to the user (with a mail?) to tell her/him to credit her/his note
form.add_error('user',
_("This user don't have enough money to join this club, and can't have a negative balance."))
2020-04-01 03:42:19 +02:00
if club.parent_club is not None:
if not Membership.objects.filter(user=form.instance.user, club=club.parent_club).exists():
form.add_error('user', _('User is not a member of the parent club') + ' ' + club.parent_club.name)
return super().form_invalid(form)
if Membership.objects.filter(
user=form.instance.user,
club=club,
date_start__lte=form.instance.date_start,
date_end__gte=form.instance.date_start,
2020-04-01 03:42:19 +02:00
).exists():
form.add_error('user', _('User is already a member of the club'))
return super().form_invalid(form)
if form.instance.club.membership_start and form.instance.date_start < form.instance.club.membership_start:
2020-04-01 03:42:19 +02:00
form.add_error('user', _("The membership must start after {:%m-%d-%Y}.")
.format(form.instance.club.membership_start))
return super().form_invalid(form)
if form.instance.club.membership_end and form.instance.date_start > form.instance.club.membership_end:
form.add_error('user', _("The membership must begin before {:%m-%d-%Y}.")
2020-04-01 03:42:19 +02:00
.format(form.instance.club.membership_start))
return super().form_invalid(form)
return super().form_valid(form)
def get_success_url(self):
return reverse_lazy('member:club_detail', kwargs={'pk': self.object.club.id})
class ClubManageRolesView(ProtectQuerysetMixin, LoginRequiredMixin, UpdateView):
model = Membership
form_class = MembershipForm
template_name = 'member/add_members.html'
def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
club = self.object.club
context['club'] = club
form = context['form']
form.fields['user'].disabled = True
form.fields['date_start'].widget = HiddenInput()
return context
def form_valid(self, form):
if form.instance.club.membership_start and form.instance.date_start < form.instance.club.membership_start:
2020-04-01 03:42:19 +02:00
form.add_error('user', _("The membership must start after {:%m-%d-%Y}.")
.format(form.instance.club.membership_start))
return super().form_invalid(form)
if form.instance.club.membership_end and form.instance.date_start > form.instance.club.membership_end:
form.add_error('user', _("The membership must begin before {:%m-%d-%Y}.")
2020-04-01 03:42:19 +02:00
.format(form.instance.club.membership_start))
return super().form_invalid(form)
2020-03-31 23:54:14 +02:00
return super().form_valid(form)
def get_success_url(self):
return reverse_lazy('member:club_detail', kwargs={'pk': self.object.club.id})