2024-07-18 18:23:30 +00:00
|
|
|
"""Permission classes to be used within ninja-extra controllers.
|
|
|
|
|
|
|
|
Some permissions are global (like `IsInGroup` or `IsRoot`),
|
|
|
|
and some others are per-object (like `CanView` or `CanEdit`).
|
|
|
|
|
2025-01-14 16:14:59 +00:00
|
|
|
Example:
|
|
|
|
```python
|
2024-07-18 18:23:30 +00:00
|
|
|
# restrict all the routes of this controller
|
|
|
|
# to subscribed users
|
|
|
|
@api_controller("/foo", permissions=[IsSubscriber])
|
|
|
|
class FooController(ControllerBase):
|
|
|
|
@route.get("/bar")
|
|
|
|
def bar_get(self):
|
|
|
|
# This route inherits the permissions of the controller
|
|
|
|
# ...
|
|
|
|
|
|
|
|
@route.bar("/bar/{bar_id}", permissions=[CanView])
|
|
|
|
def bar_get_one(self, bar_id: int):
|
|
|
|
# per-object permission resolution happens
|
|
|
|
# when calling either the `get_object_or_exception`
|
|
|
|
# or `get_object_or_none` method.
|
|
|
|
bar = self.get_object_or_exception(Counter, pk=bar_id)
|
|
|
|
|
|
|
|
# you can also call the `check_object_permission` manually
|
|
|
|
other_bar = Counter.objects.first()
|
|
|
|
self.check_object_permissions(other_bar)
|
|
|
|
|
|
|
|
# ...
|
|
|
|
|
|
|
|
# This route is restricted to counter admins and root users
|
|
|
|
@route.delete(
|
|
|
|
"/bar/{bar_id}",
|
|
|
|
permissions=[IsRoot | IsInGroup(settings.SITH_GROUP_COUNTER_ADMIN_ID)
|
|
|
|
]
|
|
|
|
def bar_delete(self, bar_id: int):
|
|
|
|
# ...
|
2025-01-14 16:14:59 +00:00
|
|
|
```
|
2024-07-18 18:23:30 +00:00
|
|
|
"""
|
|
|
|
|
2025-01-17 08:48:34 +00:00
|
|
|
import operator
|
|
|
|
from functools import reduce
|
2024-07-18 18:23:30 +00:00
|
|
|
from typing import Any
|
|
|
|
|
2025-01-17 08:48:34 +00:00
|
|
|
from django.contrib.auth.models import Permission
|
2024-07-18 18:23:30 +00:00
|
|
|
from django.http import HttpRequest
|
|
|
|
from ninja_extra import ControllerBase
|
|
|
|
from ninja_extra.permissions import BasePermission
|
|
|
|
|
2024-08-01 17:02:29 +00:00
|
|
|
from counter.models import Counter
|
|
|
|
|
2024-07-18 18:23:30 +00:00
|
|
|
|
|
|
|
class IsInGroup(BasePermission):
|
2024-07-23 21:25:17 +00:00
|
|
|
"""Check that the user is in the group whose primary key is given."""
|
|
|
|
|
2024-07-18 18:23:30 +00:00
|
|
|
def __init__(self, group_pk: int):
|
|
|
|
self._group_pk = group_pk
|
|
|
|
|
|
|
|
def has_permission(self, request: HttpRequest, controller: ControllerBase) -> bool:
|
|
|
|
return request.user.is_in_group(pk=self._group_pk)
|
|
|
|
|
|
|
|
|
2025-01-17 08:48:34 +00:00
|
|
|
class HasPerm(BasePermission):
|
|
|
|
"""Check that the user has the required perm.
|
|
|
|
|
|
|
|
If multiple perms are given, a comparer function can also be passed,
|
|
|
|
in order to change the way perms are checked.
|
|
|
|
|
|
|
|
Example:
|
|
|
|
```python
|
|
|
|
# this route will require both permissions
|
|
|
|
@route.put("/foo", permissions=[HasPerm(["foo.change_foo", "foo.add_foo"])]
|
|
|
|
def foo(self): ...
|
|
|
|
|
|
|
|
# This route will require at least one of the perm,
|
|
|
|
# but it's not mandatory to have all of them
|
|
|
|
@route.put(
|
|
|
|
"/bar",
|
|
|
|
permissions=[HasPerm(["foo.change_bar", "foo.add_bar"], op=operator.or_)],
|
|
|
|
)
|
|
|
|
def bar(self): ...
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(
|
|
|
|
self, perms: str | Permission | list[str | Permission], op=operator.and_
|
|
|
|
):
|
|
|
|
"""
|
|
|
|
Args:
|
|
|
|
perms: a permission or a list of permissions the user must have
|
|
|
|
op: An operator to combine multiple permissions (in most cases,
|
|
|
|
it will be either `operator.and_` or `operator.or_`)
|
|
|
|
"""
|
|
|
|
super().__init__()
|
|
|
|
if not isinstance(perms, (list, tuple, set)):
|
|
|
|
perms = [perms]
|
|
|
|
self._operator = op
|
|
|
|
self._perms = perms
|
|
|
|
|
|
|
|
def has_permission(self, request: HttpRequest, controller: ControllerBase) -> bool:
|
|
|
|
return reduce(self._operator, (request.user.has_perm(p) for p in self._perms))
|
|
|
|
|
|
|
|
|
2024-07-18 18:23:30 +00:00
|
|
|
class IsRoot(BasePermission):
|
2024-07-23 21:25:17 +00:00
|
|
|
"""Check that the user is root."""
|
|
|
|
|
2024-07-18 18:23:30 +00:00
|
|
|
def has_permission(self, request: HttpRequest, controller: ControllerBase) -> bool:
|
|
|
|
return request.user.is_root
|
|
|
|
|
|
|
|
|
|
|
|
class IsSubscriber(BasePermission):
|
2024-07-23 21:25:17 +00:00
|
|
|
"""Check that the user is currently subscribed."""
|
|
|
|
|
2024-07-18 18:23:30 +00:00
|
|
|
def has_permission(self, request: HttpRequest, controller: ControllerBase) -> bool:
|
|
|
|
return request.user.is_subscribed
|
|
|
|
|
|
|
|
|
|
|
|
class IsOldSubscriber(BasePermission):
|
2024-07-23 21:25:17 +00:00
|
|
|
"""Check that the user has at least one subscription in its history."""
|
|
|
|
|
2024-07-18 18:23:30 +00:00
|
|
|
def has_permission(self, request: HttpRequest, controller: ControllerBase) -> bool:
|
|
|
|
return request.user.was_subscribed
|
|
|
|
|
|
|
|
|
|
|
|
class CanView(BasePermission):
|
2024-07-23 21:25:17 +00:00
|
|
|
"""Check that this user has the permission to view the object of this route.
|
|
|
|
|
|
|
|
Wrap the `user.can_view(obj)` method.
|
2024-09-17 10:10:06 +00:00
|
|
|
To see an example, look at the example in the module docstring.
|
2024-07-23 21:25:17 +00:00
|
|
|
"""
|
|
|
|
|
2024-07-18 18:23:30 +00:00
|
|
|
def has_permission(self, request: HttpRequest, controller: ControllerBase) -> bool:
|
|
|
|
return True
|
|
|
|
|
|
|
|
def has_object_permission(
|
|
|
|
self, request: HttpRequest, controller: ControllerBase, obj: Any
|
|
|
|
) -> bool:
|
|
|
|
return request.user.can_view(obj)
|
|
|
|
|
|
|
|
|
|
|
|
class CanEdit(BasePermission):
|
2024-07-23 21:25:17 +00:00
|
|
|
"""Check that this user has the permission to edit the object of this route.
|
|
|
|
|
|
|
|
Wrap the `user.can_edit(obj)` method.
|
2024-09-17 10:10:06 +00:00
|
|
|
To see an example, look at the example in the module docstring.
|
2024-07-23 21:25:17 +00:00
|
|
|
"""
|
|
|
|
|
2024-07-18 18:23:30 +00:00
|
|
|
def has_permission(self, request: HttpRequest, controller: ControllerBase) -> bool:
|
|
|
|
return True
|
|
|
|
|
|
|
|
def has_object_permission(
|
|
|
|
self, request: HttpRequest, controller: ControllerBase, obj: Any
|
|
|
|
) -> bool:
|
|
|
|
return request.user.can_edit(obj)
|
|
|
|
|
|
|
|
|
|
|
|
class IsOwner(BasePermission):
|
2024-07-23 21:25:17 +00:00
|
|
|
"""Check that this user owns the object of this route.
|
|
|
|
|
|
|
|
Wrap the `user.is_owner(obj)` method.
|
2024-09-17 10:10:06 +00:00
|
|
|
To see an example, look at the example in the module docstring.
|
2024-07-23 21:25:17 +00:00
|
|
|
"""
|
|
|
|
|
2024-07-18 18:23:30 +00:00
|
|
|
def has_permission(self, request: HttpRequest, controller: ControllerBase) -> bool:
|
|
|
|
return True
|
|
|
|
|
|
|
|
def has_object_permission(
|
|
|
|
self, request: HttpRequest, controller: ControllerBase, obj: Any
|
|
|
|
) -> bool:
|
|
|
|
return request.user.is_owner(obj)
|
2024-08-01 17:02:29 +00:00
|
|
|
|
|
|
|
|
|
|
|
class IsLoggedInCounter(BasePermission):
|
|
|
|
"""Check that a user is logged in a counter."""
|
|
|
|
|
|
|
|
def has_permission(self, request: HttpRequest, controller: ControllerBase) -> bool:
|
2024-10-19 22:18:53 +00:00
|
|
|
if "/counter/" not in request.META.get("HTTP_REFERER", ""):
|
2024-08-01 17:02:29 +00:00
|
|
|
return False
|
|
|
|
token = request.session.get("counter_token")
|
|
|
|
if not token:
|
|
|
|
return False
|
|
|
|
return Counter.objects.filter(token=token).exists()
|
2024-10-19 22:18:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
CanAccessLookup = IsOldSubscriber | IsRoot | IsLoggedInCounter
|