1
0
mirror of https://gitlab.crans.org/bde/nk20 synced 2025-01-19 06:31:20 +00:00
nk20/apps/permission/backends.py

170 lines
6.2 KiB
Python
Raw Normal View History

2020-03-07 13:12:17 +01:00
# Copyright (C) 2018-2020 by BDE ENS Paris-Saclay
# SPDX-License-Identifier: GPL-3.0-or-later
2020-04-01 03:42:19 +02:00
import datetime
2020-03-20 02:14:43 +01:00
from django.contrib.auth.backends import ModelBackend
2020-03-20 18:22:20 +01:00
from django.contrib.auth.models import User, AnonymousUser
from django.contrib.contenttypes.models import ContentType
from django.db.models import Q, F
2020-03-31 14:57:44 +02:00
from note.models import Note, NoteUser, NoteClub, NoteSpecial
2020-03-19 16:12:52 +01:00
from note_kfet.middlewares import get_current_session
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
def get_raw_permissions(user, t):
"""
Query permissions of a certain type for a user, then memoize it.
:param user: The owner of the permissions
: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
"""
if isinstance(user, AnonymousUser):
# Unauthenticated users have no permissions
return Permission.objects.none()
2020-04-23 18:28:16 +02:00
return Permission.objects.annotate(
club=F("rolepermissions__role__membership__club"),
membership=F("rolepermissions__role__membership"),
).filter(
rolepermissions__role__membership__user=user,
rolepermissions__role__membership__date_start__lte=datetime.date.today(),
rolepermissions__role__membership__date_end__gte=datetime.date.today(),
type=t,
mask__rank__lte=get_current_session().get("permission_mask", 0),
).distinct()
2020-04-02 00:30:22 +02:00
@staticmethod
def permissions(user, 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
:param user: The owner of the permissions
: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
clubs = {}
2020-04-23 18:28:16 +02:00
memberships = {}
2020-04-02 00:30:22 +02:00
for permission in PermissionBackend.get_raw_permissions(user, type):
if not isinstance(model.model_class()(), permission.model.model_class()) or not permission.club:
2020-03-20 15:58:14 +01:00
continue
2020-04-02 00:30:22 +02:00
if permission.club not in clubs:
clubs[permission.club] = club = Club.objects.get(pk=permission.club)
else:
club = clubs[permission.club]
2020-04-23 18:28:16 +02:00
if permission.membership not in memberships:
memberships[permission.membership] = membership = Membership.objects.get(pk=permission.membership)
else:
membership = memberships[permission.membership]
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,
now=datetime.datetime.now(),
today=datetime.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
@staticmethod
2020-04-02 00:30:22 +02:00
@memoize
def filter_queryset(user, model, t, field=None):
"""
Filter a queryset by considering the permissions of a given user.
:param user: The owner of the permissions that are fetched
:param model: The concerned model of the queryset
:param t: The type of modification (view, add, change, delete)
:param field: The field of the model to test, if concerned
:return: A query that corresponds to the filter to give to a queryset
"""
2020-03-20 18:22:20 +01:00
if user is None or isinstance(user, AnonymousUser):
# Anonymous users can't do anything
return Q(pk=-1)
2020-03-19 16:12:52 +01:00
if user.is_superuser and get_current_session().get("permission_mask", 0) >= 42:
# Superusers have all rights
return Q()
if not isinstance(model, ContentType):
model = ContentType.objects.get_for_model(model)
# Never satisfied
query = Q(pk=-1)
2020-03-20 01:46:59 +01:00
perms = PermissionBackend.permissions(user, model, t)
for perm in perms:
if perm.field and field != perm.field:
continue
if perm.type != t or perm.model != model:
continue
2020-03-20 01:46:59 +01:00
perm.update_query()
query = query | perm.query
return query
@staticmethod
2020-04-02 00:30:22 +02:00
@memoize
def check_perm(user_obj, perm, obj=None):
"""
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)
"""
2020-03-20 18:22:20 +01:00
if user_obj is None or isinstance(user_obj, AnonymousUser):
return False
2020-04-02 00:30:22 +02:00
sess = get_current_session()
if sess is not None and sess.session_key is None:
return Permission.objects.none()
2020-03-19 16:12:52 +01:00
if user_obj.is_superuser and get_current_session().get("permission_mask", 0) >= 42:
2020-03-07 13:12:17 +01:00
return True
2019-09-18 14:26:42 +02:00
if obj is None:
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
ct = ContentType.objects.get_for_model(obj)
if any(permission.applies(obj, perm_type, perm_field)
for permission in PermissionBackend.permissions(user_obj, ct, perm_type)):
return True
return False
2020-03-07 13:12:17 +01:00
def has_perm(self, user_obj, perm, obj=None):
return PermissionBackend.check_perm(user_obj, perm, obj)
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
def get_all_permissions(self, user_obj, obj=None):
ct = ContentType.objects.get_for_model(obj)
return list(self.permissions(user_obj, ct, "view"))