2024-02-07 02:26:49 +01:00
|
|
|
# Copyright (C) 2018-2024 by BDE ENS Paris-Saclay
|
2020-03-07 13:12:17 +01:00
|
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
|
2020-08-16 00:35:13 +02:00
|
|
|
from datetime import date
|
|
|
|
|
2020-03-20 02:14:43 +01:00
|
|
|
from django.contrib.auth.backends import ModelBackend
|
2021-06-15 14:40:32 +02:00
|
|
|
from django.contrib.auth.models import User
|
2020-03-19 02:26:06 +01:00
|
|
|
from django.contrib.contenttypes.models import ContentType
|
2020-03-18 14:42:35 +01:00
|
|
|
from django.db.models import Q, F
|
2020-05-30 15:46:09 +02:00
|
|
|
from django.utils import timezone
|
2020-03-31 14:57:44 +02:00
|
|
|
from note.models import Note, NoteUser, NoteClub, NoteSpecial
|
2021-06-15 14:40:32 +02:00
|
|
|
from note_kfet.middlewares import get_current_request
|
2020-03-20 14:43:35 +01:00
|
|
|
from member.models import Membership, Club
|
2019-09-18 14:26:42 +02:00
|
|
|
|
2020-04-02 00:42:00 +02:00
|
|
|
from .decorators import memoize
|
2020-03-20 18:22:20 +01:00
|
|
|
from .models import Permission
|
|
|
|
|
2019-09-18 14:26:42 +02:00
|
|
|
|
2020-03-07 13:12:17 +01:00
|
|
|
class PermissionBackend(ModelBackend):
|
2020-03-20 15:58:14 +01:00
|
|
|
"""
|
|
|
|
Manage permissions of users
|
|
|
|
"""
|
2019-09-18 14:26:42 +02:00
|
|
|
supports_object_permissions = True
|
|
|
|
supports_anonymous_user = False
|
|
|
|
supports_inactive_user = False
|
|
|
|
|
2020-04-02 00:30:22 +02:00
|
|
|
@staticmethod
|
|
|
|
@memoize
|
2021-06-15 15:50:36 +02:00
|
|
|
def get_raw_permissions(request, t):
|
2020-04-02 00:30:22 +02:00
|
|
|
"""
|
|
|
|
Query permissions of a certain type for a user, then memoize it.
|
2021-06-15 15:50:36 +02:00
|
|
|
:param request: The current request
|
2020-04-02 00:30:22 +02:00
|
|
|
:param t: The type of the permissions: view, change, add or delete
|
|
|
|
:return: The queryset of the permissions of the user (memoized) grouped by clubs
|
|
|
|
"""
|
2021-06-15 15:50:36 +02:00
|
|
|
if hasattr(request, 'auth') and request.auth is not None and hasattr(request.auth, 'scope'):
|
|
|
|
# OAuth2 Authentication
|
|
|
|
user = request.auth.user
|
|
|
|
|
|
|
|
def permission_filter(membership_obj):
|
|
|
|
query = Q(pk=-1)
|
|
|
|
for scope in request.auth.scope.split(' '):
|
|
|
|
permission_id, club_id = scope.split('_')
|
|
|
|
if int(club_id) == membership_obj.club_id:
|
|
|
|
query |= Q(pk=permission_id)
|
|
|
|
return query
|
|
|
|
else:
|
|
|
|
user = request.user
|
|
|
|
|
|
|
|
def permission_filter(membership_obj):
|
2021-06-17 22:29:57 +02:00
|
|
|
return Q(mask__rank__lte=request.session.get("permission_mask", 42))
|
2021-06-15 15:50:36 +02:00
|
|
|
|
|
|
|
if user.is_anonymous:
|
2020-04-02 00:30:22 +02:00
|
|
|
# Unauthenticated users have no permissions
|
|
|
|
return Permission.objects.none()
|
|
|
|
|
2020-07-28 15:25:08 +02:00
|
|
|
memberships = Membership.objects.filter(user=user).all()
|
2020-07-30 15:53:23 +02:00
|
|
|
|
2020-07-28 15:25:08 +02:00
|
|
|
perms = []
|
|
|
|
|
|
|
|
for membership in memberships:
|
|
|
|
for role in membership.roles.all():
|
2021-06-15 15:50:36 +02:00
|
|
|
for perm in role.permissions.filter(permission_filter(membership), type=t).all():
|
2020-07-28 15:25:08 +02:00
|
|
|
if not perm.permanent:
|
2020-08-16 00:35:13 +02:00
|
|
|
if membership.date_start > date.today() or membership.date_end < date.today():
|
2020-07-28 15:25:08 +02:00
|
|
|
continue
|
|
|
|
perm.membership = membership
|
|
|
|
perms.append(perm)
|
|
|
|
return perms
|
2020-05-29 21:11:51 +02:00
|
|
|
|
2020-03-19 02:26:06 +01:00
|
|
|
@staticmethod
|
2021-06-15 15:50:36 +02:00
|
|
|
def permissions(request, model, type):
|
2020-03-20 15:58:14 +01:00
|
|
|
"""
|
|
|
|
List all permissions of the given user that applies to a given model and a give type
|
2021-06-15 15:50:36 +02:00
|
|
|
:param request: The current request
|
2020-03-20 15:58:14 +01:00
|
|
|
:param model: The model that the permissions shoud apply
|
|
|
|
:param type: The type of the permissions: view, change, add or delete
|
|
|
|
:return: A generator of the requested permissions
|
|
|
|
"""
|
2020-04-02 00:30:22 +02:00
|
|
|
|
2021-06-15 15:50:36 +02:00
|
|
|
if hasattr(request, 'auth') and request.auth is not None and hasattr(request.auth, 'scope'):
|
|
|
|
# OAuth2 Authentication
|
|
|
|
user = request.auth.user
|
|
|
|
else:
|
|
|
|
user = request.user
|
|
|
|
|
|
|
|
for permission in PermissionBackend.get_raw_permissions(request, type):
|
2020-07-28 15:25:08 +02:00
|
|
|
if not isinstance(model.model_class()(), permission.model.model_class()) or not permission.membership:
|
2020-03-20 15:58:14 +01:00
|
|
|
continue
|
|
|
|
|
2020-07-28 15:25:08 +02:00
|
|
|
membership = permission.membership
|
|
|
|
club = membership.club
|
2020-04-23 18:28:16 +02:00
|
|
|
|
2020-03-20 01:46:59 +01:00
|
|
|
permission = permission.about(
|
|
|
|
user=user,
|
|
|
|
club=club,
|
2020-04-23 18:28:16 +02:00
|
|
|
membership=membership,
|
2020-03-20 01:46:59 +01:00
|
|
|
User=User,
|
|
|
|
Club=Club,
|
|
|
|
Membership=Membership,
|
|
|
|
Note=Note,
|
|
|
|
NoteUser=NoteUser,
|
|
|
|
NoteClub=NoteClub,
|
|
|
|
NoteSpecial=NoteSpecial,
|
|
|
|
F=F,
|
2020-04-22 13:28:52 +02:00
|
|
|
Q=Q,
|
2020-05-30 15:46:09 +02:00
|
|
|
now=timezone.now(),
|
2020-08-16 00:35:13 +02:00
|
|
|
today=date.today(),
|
2020-03-20 01:46:59 +01:00
|
|
|
)
|
2020-04-02 00:30:22 +02:00
|
|
|
yield permission
|
2019-09-18 14:26:42 +02:00
|
|
|
|
2020-03-19 02:26:06 +01:00
|
|
|
@staticmethod
|
2020-04-02 00:30:22 +02:00
|
|
|
@memoize
|
2021-06-15 14:40:32 +02:00
|
|
|
def filter_queryset(request, model, t, field=None):
|
2020-03-18 14:42:35 +01:00
|
|
|
"""
|
|
|
|
Filter a queryset by considering the permissions of a given user.
|
2021-06-15 14:40:32 +02:00
|
|
|
:param request: The current request
|
2020-03-18 14:42:35 +01:00
|
|
|
:param model: The concerned model of the queryset
|
2020-03-19 02:26:06 +01:00
|
|
|
:param t: The type of modification (view, add, change, delete)
|
2020-03-18 14:42:35 +01:00
|
|
|
:param field: The field of the model to test, if concerned
|
|
|
|
:return: A query that corresponds to the filter to give to a queryset
|
|
|
|
"""
|
2021-06-15 15:50:36 +02:00
|
|
|
if hasattr(request, 'auth') and request.auth is not None and hasattr(request.auth, 'scope'):
|
|
|
|
# OAuth2 Authentication
|
|
|
|
user = request.auth.user
|
|
|
|
else:
|
|
|
|
user = request.user
|
|
|
|
|
|
|
|
if user is None or user.is_anonymous:
|
2021-06-17 22:13:43 +02:00
|
|
|
# Anonymous users can't do anything
|
2020-03-20 18:22:20 +01:00
|
|
|
return Q(pk=-1)
|
|
|
|
|
2021-06-15 15:50:36 +02:00
|
|
|
if user.is_superuser and request.session.get("permission_mask", -1) >= 42:
|
2020-03-18 14:42:35 +01:00
|
|
|
# Superusers have all rights
|
|
|
|
return Q()
|
|
|
|
|
2020-03-19 02:26:06 +01:00
|
|
|
if not isinstance(model, ContentType):
|
|
|
|
model = ContentType.objects.get_for_model(model)
|
|
|
|
|
2020-03-18 14:42:35 +01:00
|
|
|
# Never satisfied
|
|
|
|
query = Q(pk=-1)
|
2021-06-15 15:50:36 +02:00
|
|
|
perms = PermissionBackend.permissions(request, model, t)
|
2020-03-20 01:46:59 +01:00
|
|
|
for perm in perms:
|
2020-03-19 02:26:06 +01:00
|
|
|
if perm.field and field != perm.field:
|
2020-03-18 14:42:35 +01:00
|
|
|
continue
|
2020-03-20 00:06:28 +01:00
|
|
|
if perm.type != t or perm.model != model:
|
2020-03-18 14:42:35 +01:00
|
|
|
continue
|
2020-03-20 01:46:59 +01:00
|
|
|
perm.update_query()
|
2020-03-18 14:42:35 +01:00
|
|
|
query = query | perm.query
|
|
|
|
return query
|
|
|
|
|
2020-04-02 14:50:28 +02:00
|
|
|
@staticmethod
|
2020-04-02 00:30:22 +02:00
|
|
|
@memoize
|
2021-06-15 14:40:32 +02:00
|
|
|
def check_perm(request, perm, obj=None):
|
2020-04-02 14:50:28 +02:00
|
|
|
"""
|
|
|
|
Check is the given user has the permission over a given object.
|
|
|
|
The result is then memoized.
|
|
|
|
Exception: for add permissions, since the object is not hashable since it doesn't have any
|
|
|
|
primary key, the result is not memoized. Moreover, the right could change
|
|
|
|
(e.g. for a transaction, the balance of the user could change)
|
|
|
|
"""
|
2021-09-08 16:59:44 +02:00
|
|
|
# Requested by a shell
|
|
|
|
if request is None:
|
|
|
|
return False
|
|
|
|
|
2021-06-15 14:40:32 +02:00
|
|
|
user_obj = request.user
|
2021-06-15 15:50:36 +02:00
|
|
|
sess = request.session
|
2021-06-15 14:40:32 +02:00
|
|
|
|
2021-06-15 15:50:36 +02:00
|
|
|
if hasattr(request, 'auth') and request.auth is not None and hasattr(request.auth, 'scope'):
|
|
|
|
# OAuth2 Authentication
|
|
|
|
user_obj = request.auth.user
|
2020-03-20 18:22:20 +01:00
|
|
|
|
2021-06-15 15:50:36 +02:00
|
|
|
if user_obj is None or user_obj.is_anonymous:
|
|
|
|
return False
|
2020-04-02 00:30:22 +02:00
|
|
|
|
2020-07-30 12:50:48 +02:00
|
|
|
if user_obj.is_superuser and sess.get("permission_mask", -1) >= 42:
|
2020-03-07 13:12:17 +01:00
|
|
|
return True
|
|
|
|
|
2019-09-18 14:26:42 +02:00
|
|
|
if obj is None:
|
2020-03-18 14:42:35 +01:00
|
|
|
return True
|
|
|
|
|
|
|
|
perm = perm.split('.')[-1].split('_', 2)
|
|
|
|
perm_type = perm[0]
|
2019-09-18 14:26:42 +02:00
|
|
|
perm_field = perm[2] if len(perm) == 3 else None
|
2020-07-28 15:25:08 +02:00
|
|
|
|
2020-03-20 00:06:28 +01:00
|
|
|
ct = ContentType.objects.get_for_model(obj)
|
|
|
|
if any(permission.applies(obj, perm_type, perm_field)
|
2021-06-15 15:50:36 +02:00
|
|
|
for permission in PermissionBackend.permissions(request, ct, perm_type)):
|
2020-03-18 14:42:35 +01:00
|
|
|
return True
|
|
|
|
return False
|
2020-03-07 13:12:17 +01:00
|
|
|
|
2020-04-02 14:50:28 +02:00
|
|
|
def has_perm(self, user_obj, perm, obj=None):
|
2021-06-15 14:40:32 +02:00
|
|
|
# Warning: this does not check that user_obj has the permission,
|
|
|
|
# but if the current request has the permission.
|
|
|
|
# This function is implemented for backward compatibility, and should not be used.
|
|
|
|
return PermissionBackend.check_perm(get_current_request(), perm, obj)
|
2020-04-02 14:50:28 +02:00
|
|
|
|
2020-03-07 13:12:17 +01:00
|
|
|
def has_module_perms(self, user_obj, app_label):
|
|
|
|
return False
|
2019-09-18 14:26:42 +02:00
|
|
|
|
2023-09-28 18:48:57 +02:00
|
|
|
@staticmethod
|
|
|
|
@memoize
|
|
|
|
def has_model_perm(request, model, type):
|
|
|
|
"""
|
|
|
|
Check is the given user has the permission over a given model for a given action.
|
|
|
|
The result is then memoized.
|
|
|
|
:param request: The current request
|
|
|
|
:param model: The model that the permissions shoud apply
|
|
|
|
:param type: The type of the permissions: view, change, add or delete
|
|
|
|
For view action, it is consider possible if user can view or change the model
|
|
|
|
"""
|
|
|
|
# Requested by a shell
|
|
|
|
if request is None:
|
|
|
|
return False
|
|
|
|
|
|
|
|
user_obj = request.user
|
|
|
|
sess = request.session
|
|
|
|
|
|
|
|
if hasattr(request, 'auth') and request.auth is not None and hasattr(request.auth, 'scope'):
|
|
|
|
# OAuth2 Authentication
|
|
|
|
user_obj = request.auth.user
|
|
|
|
|
|
|
|
if user_obj is None or user_obj.is_anonymous:
|
|
|
|
return False
|
|
|
|
|
|
|
|
if user_obj.is_superuser and sess.get("permission_mask", -1) >= 42:
|
|
|
|
return True
|
|
|
|
|
|
|
|
ct = ContentType.objects.get_for_model(model)
|
|
|
|
if any(PermissionBackend.permissions(request, ct, type)):
|
|
|
|
return True
|
|
|
|
if type == "view" and any(PermissionBackend.permissions(request, ct, "change")):
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
2019-09-18 14:26:42 +02:00
|
|
|
def get_all_permissions(self, user_obj, obj=None):
|
2020-03-20 00:06:28 +01:00
|
|
|
ct = ContentType.objects.get_for_model(obj)
|
2021-06-15 15:50:36 +02:00
|
|
|
return list(self.permissions(get_current_request(), ct, "view"))
|