2020-02-07 17:02:07 +01:00
|
|
|
# Copyright (C) 2018-2020 by BDE ENS Paris-Saclay
|
|
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
2020-09-02 18:01:41 +02:00
|
|
|
from django.conf import settings
|
2020-02-08 17:17:00 +01:00
|
|
|
from django.db.models import Q
|
2020-03-26 23:05:37 +01:00
|
|
|
from django.core.exceptions import ValidationError
|
2020-03-11 11:15:03 +01:00
|
|
|
from django_filters.rest_framework import DjangoFilterBackend
|
2020-03-13 10:29:27 +01:00
|
|
|
from rest_framework.filters import OrderingFilter, SearchFilter
|
2020-03-24 22:12:44 +01:00
|
|
|
from rest_framework import viewsets
|
2020-03-26 23:05:37 +01:00
|
|
|
from rest_framework.response import Response
|
|
|
|
from rest_framework import status
|
2020-03-19 19:29:52 +01:00
|
|
|
from api.viewsets import ReadProtectedModelViewSet, ReadOnlyProtectedModelViewSet
|
2020-08-06 12:50:24 +02:00
|
|
|
from note_kfet.middlewares import get_current_session
|
2020-08-03 16:11:05 +02:00
|
|
|
from permission.backends import PermissionBackend
|
2020-03-20 02:14:43 +01:00
|
|
|
|
2020-03-28 17:42:29 +01:00
|
|
|
from .serializers import NotePolymorphicSerializer, AliasSerializer, ConsumerSerializer,\
|
|
|
|
TemplateCategorySerializer, TransactionTemplateSerializer, TransactionPolymorphicSerializer
|
2020-03-19 19:29:52 +01:00
|
|
|
from ..models.notes import Note, Alias
|
2020-03-11 11:15:03 +01:00
|
|
|
from ..models.transactions import TransactionTemplate, Transaction, TemplateCategory
|
2020-02-07 17:02:07 +01:00
|
|
|
|
|
|
|
|
2020-08-31 20:15:48 +02:00
|
|
|
class NotePolymorphicViewSet(ReadProtectedModelViewSet):
|
2020-02-07 20:47:49 +01:00
|
|
|
"""
|
|
|
|
REST API View set.
|
2020-02-08 17:17:00 +01:00
|
|
|
The djangorestframework plugin will get all `Note` objects (with polymorhism), serialize it to JSON with the given serializer,
|
|
|
|
then render it on /api/note/note/
|
2020-02-07 20:47:49 +01:00
|
|
|
"""
|
|
|
|
queryset = Note.objects.all()
|
|
|
|
serializer_class = NotePolymorphicSerializer
|
2020-03-27 16:19:33 +01:00
|
|
|
filter_backends = [DjangoFilterBackend, SearchFilter, OrderingFilter]
|
|
|
|
filterset_fields = ['polymorphic_ctype', 'is_active', ]
|
2020-03-13 10:29:27 +01:00
|
|
|
search_fields = ['$alias__normalized_name', '$alias__name', '$polymorphic_ctype__model', ]
|
|
|
|
ordering_fields = ['alias__name', 'alias__normalized_name']
|
2020-02-07 20:47:49 +01:00
|
|
|
|
2020-02-08 17:17:00 +01:00
|
|
|
def get_queryset(self):
|
|
|
|
"""
|
|
|
|
Parse query and apply filters.
|
|
|
|
:return: The filtered set of requested notes
|
|
|
|
"""
|
2020-08-31 20:15:48 +02:00
|
|
|
queryset = super().get_queryset().distinct()
|
2020-02-08 17:17:00 +01:00
|
|
|
|
|
|
|
alias = self.request.query_params.get("alias", ".*")
|
2020-02-18 12:31:15 +01:00
|
|
|
queryset = queryset.filter(
|
2020-08-31 20:15:48 +02:00
|
|
|
Q(alias__name__iregex="^" + alias)
|
|
|
|
| Q(alias__normalized_name__iregex="^" + Alias.normalize(alias))
|
|
|
|
| Q(alias__normalized_name__iregex="^" + alias.lower())
|
|
|
|
)
|
2020-02-08 17:17:00 +01:00
|
|
|
|
2020-09-01 15:54:56 +02:00
|
|
|
return queryset.order_by("id")
|
2020-02-08 17:17:00 +01:00
|
|
|
|
2020-02-07 20:47:49 +01:00
|
|
|
|
2020-03-18 14:42:35 +01:00
|
|
|
class AliasViewSet(ReadProtectedModelViewSet):
|
2020-02-08 15:08:55 +01:00
|
|
|
"""
|
|
|
|
REST API View set.
|
|
|
|
The djangorestframework plugin will get all `Alias` objects, serialize it to JSON with the given serializer,
|
|
|
|
then render it on /api/aliases/
|
|
|
|
"""
|
|
|
|
queryset = Alias.objects.all()
|
|
|
|
serializer_class = AliasSerializer
|
2020-03-13 10:29:27 +01:00
|
|
|
filter_backends = [SearchFilter, OrderingFilter]
|
2020-03-12 01:10:52 +01:00
|
|
|
search_fields = ['$normalized_name', '$name', '$note__polymorphic_ctype__model', ]
|
2020-03-13 10:29:27 +01:00
|
|
|
ordering_fields = ['name', 'normalized_name']
|
2020-02-08 15:08:55 +01:00
|
|
|
|
2020-03-26 17:45:24 +01:00
|
|
|
def get_serializer_class(self):
|
|
|
|
serializer_class = self.serializer_class
|
|
|
|
if self.request.method in ['PUT', 'PATCH']:
|
2020-03-27 14:19:55 +01:00
|
|
|
# alias owner cannot be change once establish
|
2020-03-26 17:45:24 +01:00
|
|
|
setattr(serializer_class.Meta, 'read_only_fields', ('note',))
|
|
|
|
return serializer_class
|
2020-03-27 14:19:55 +01:00
|
|
|
|
2020-03-26 23:05:37 +01:00
|
|
|
def destroy(self, request, *args, **kwargs):
|
|
|
|
instance = self.get_object()
|
|
|
|
try:
|
|
|
|
self.perform_destroy(instance)
|
|
|
|
except ValidationError as e:
|
2020-03-27 14:19:55 +01:00
|
|
|
return Response({e.code: e.message}, status.HTTP_400_BAD_REQUEST)
|
2020-03-26 23:05:37 +01:00
|
|
|
return Response(status=status.HTTP_204_NO_CONTENT)
|
2020-03-27 14:19:55 +01:00
|
|
|
|
2020-02-08 17:17:00 +01:00
|
|
|
def get_queryset(self):
|
|
|
|
"""
|
|
|
|
Parse query and apply filters.
|
|
|
|
:return: The filtered set of requested aliases
|
|
|
|
"""
|
|
|
|
|
2020-08-31 21:32:45 +02:00
|
|
|
queryset = super().get_queryset().distinct()
|
2020-02-08 17:17:00 +01:00
|
|
|
|
2020-08-31 21:32:45 +02:00
|
|
|
alias = self.request.query_params.get("alias", None)
|
|
|
|
if alias:
|
|
|
|
queryset = queryset.filter(
|
|
|
|
name__iregex="^" + alias
|
|
|
|
).union(
|
|
|
|
queryset.filter(
|
|
|
|
Q(normalized_name__iregex="^" + Alias.normalize(alias))
|
|
|
|
& ~Q(name__iregex="^" + alias)
|
|
|
|
),
|
|
|
|
all=True).union(
|
|
|
|
queryset.filter(
|
|
|
|
Q(normalized_name__iregex="^" + alias.lower())
|
|
|
|
& ~Q(normalized_name__iregex="^" + Alias.normalize(alias))
|
|
|
|
& ~Q(name__iregex="^" + alias)
|
|
|
|
),
|
|
|
|
all=True)
|
2020-02-08 17:17:00 +01:00
|
|
|
|
2020-09-01 15:54:56 +02:00
|
|
|
return queryset.order_by("name")
|
2020-02-08 17:17:00 +01:00
|
|
|
|
2020-04-10 00:02:22 +02:00
|
|
|
|
2020-03-28 17:42:29 +01:00
|
|
|
class ConsumerViewSet(ReadOnlyProtectedModelViewSet):
|
|
|
|
queryset = Alias.objects.all()
|
|
|
|
serializer_class = ConsumerSerializer
|
|
|
|
filter_backends = [SearchFilter, OrderingFilter]
|
|
|
|
search_fields = ['$normalized_name', '$name', '$note__polymorphic_ctype__model', ]
|
|
|
|
ordering_fields = ['name', 'normalized_name']
|
|
|
|
|
|
|
|
def get_queryset(self):
|
|
|
|
"""
|
|
|
|
Parse query and apply filters.
|
|
|
|
:return: The filtered set of requested aliases
|
|
|
|
"""
|
|
|
|
|
|
|
|
queryset = super().get_queryset()
|
2020-09-02 18:01:41 +02:00
|
|
|
# Sqlite doesn't support ORDER BY in subqueries
|
2020-09-02 22:54:01 +02:00
|
|
|
queryset = queryset.order_by("name") \
|
2020-09-02 18:01:41 +02:00
|
|
|
if settings.DATABASES[queryset.db]["ENGINE"] == 'django.db.backends.postgresql' else queryset
|
2020-03-28 17:42:29 +01:00
|
|
|
|
|
|
|
alias = self.request.query_params.get("alias", ".*")
|
2020-09-01 15:54:56 +02:00
|
|
|
queryset = queryset.prefetch_related('note')
|
2020-08-30 22:33:59 +02:00
|
|
|
# We match first an alias if it is matched without normalization,
|
|
|
|
# then if the normalized pattern matches a normalized alias.
|
2020-07-31 17:07:14 +02:00
|
|
|
queryset = queryset.filter(
|
2020-08-30 22:33:59 +02:00
|
|
|
name__iregex="^" + alias
|
|
|
|
).union(
|
|
|
|
queryset.filter(
|
|
|
|
Q(normalized_name__iregex="^" + Alias.normalize(alias))
|
|
|
|
& ~Q(name__iregex="^" + alias)
|
|
|
|
),
|
|
|
|
all=True).union(
|
|
|
|
queryset.filter(
|
|
|
|
Q(normalized_name__iregex="^" + alias.lower())
|
|
|
|
& ~Q(normalized_name__iregex="^" + Alias.normalize(alias))
|
|
|
|
& ~Q(name__iregex="^" + alias)
|
|
|
|
),
|
|
|
|
all=True)
|
2020-03-28 17:42:29 +01:00
|
|
|
|
2020-09-02 22:54:01 +02:00
|
|
|
queryset = queryset if settings.DATABASES[queryset.db]["ENGINE"] == 'django.db.backends.postgresql' \
|
|
|
|
else queryset.order_by("name")
|
|
|
|
|
2020-09-02 18:01:41 +02:00
|
|
|
return queryset.distinct()
|
2020-04-10 00:02:22 +02:00
|
|
|
|
2020-02-08 15:08:55 +01:00
|
|
|
|
2020-03-18 14:42:35 +01:00
|
|
|
class TemplateCategoryViewSet(ReadProtectedModelViewSet):
|
2020-03-11 01:03:15 +01:00
|
|
|
"""
|
|
|
|
REST API View set.
|
|
|
|
The djangorestframework plugin will get all `TemplateCategory` objects, serialize it to JSON with the given serializer,
|
|
|
|
then render it on /api/note/transaction/category/
|
|
|
|
"""
|
2020-09-01 15:54:56 +02:00
|
|
|
queryset = TemplateCategory.objects.order_by("name").all()
|
2020-03-11 01:03:15 +01:00
|
|
|
serializer_class = TemplateCategorySerializer
|
2020-03-11 11:15:03 +01:00
|
|
|
filter_backends = [SearchFilter]
|
|
|
|
search_fields = ['$name', ]
|
2020-03-11 01:03:15 +01:00
|
|
|
|
|
|
|
|
2020-03-24 22:12:44 +01:00
|
|
|
class TransactionTemplateViewSet(viewsets.ModelViewSet):
|
2020-02-07 17:02:07 +01:00
|
|
|
"""
|
|
|
|
REST API View set.
|
|
|
|
The djangorestframework plugin will get all `TransactionTemplate` objects, serialize it to JSON with the given serializer,
|
|
|
|
then render it on /api/note/transaction/template/
|
|
|
|
"""
|
2020-09-01 15:54:56 +02:00
|
|
|
queryset = TransactionTemplate.objects.order_by("name").all()
|
2020-02-07 17:02:07 +01:00
|
|
|
serializer_class = TransactionTemplateSerializer
|
2020-03-23 20:21:25 +01:00
|
|
|
filter_backends = [SearchFilter, DjangoFilterBackend]
|
2020-03-11 11:15:03 +01:00
|
|
|
filterset_fields = ['name', 'amount', 'display', 'category', ]
|
2020-03-23 20:21:25 +01:00
|
|
|
search_fields = ['$name', ]
|
2020-02-07 17:02:07 +01:00
|
|
|
|
|
|
|
|
2020-03-18 14:42:35 +01:00
|
|
|
class TransactionViewSet(ReadProtectedModelViewSet):
|
2020-02-07 17:02:07 +01:00
|
|
|
"""
|
|
|
|
REST API View set.
|
|
|
|
The djangorestframework plugin will get all `Transaction` objects, serialize it to JSON with the given serializer,
|
|
|
|
then render it on /api/note/transaction/transaction/
|
|
|
|
"""
|
2020-09-01 15:54:56 +02:00
|
|
|
queryset = Transaction.objects.order_by("-created_at").all()
|
2020-03-11 11:15:03 +01:00
|
|
|
serializer_class = TransactionPolymorphicSerializer
|
|
|
|
filter_backends = [SearchFilter]
|
|
|
|
search_fields = ['$reason', ]
|
2020-08-03 16:11:05 +02:00
|
|
|
|
|
|
|
def get_queryset(self):
|
2020-08-05 19:42:44 +02:00
|
|
|
user = self.request.user
|
|
|
|
get_current_session().setdefault("permission_mask", 42)
|
|
|
|
return self.model.objects.filter(PermissionBackend.filter_queryset(user, self.model, "view"))\
|
|
|
|
.order_by("created_at", "id")
|