2023-08-11 09:38:47 +08:00
|
|
|
from django.core.exceptions import ValidationError, ObjectDoesNotExist
|
2023-07-29 12:56:42 +08:00
|
|
|
|
2023-07-31 12:53:39 +08:00
|
|
|
from rest_framework import status
|
2023-08-11 09:38:47 +08:00
|
|
|
# from rest_framework.decorators import action
|
2023-08-31 22:02:13 +08:00
|
|
|
from rest_framework.parsers import (
|
|
|
|
FormParser,
|
|
|
|
MultiPartParser,
|
|
|
|
FileUploadParser
|
|
|
|
)
|
2023-08-11 09:38:47 +08:00
|
|
|
from rest_framework.permissions import IsAuthenticated
|
|
|
|
from rest_framework.response import Response
|
|
|
|
from rest_framework import viewsets
|
2023-08-11 21:12:51 +08:00
|
|
|
from rest_framework import filters
|
|
|
|
|
|
|
|
from django_filters.rest_framework import DjangoFilterBackend
|
2023-07-31 12:53:39 +08:00
|
|
|
|
2023-08-16 23:53:13 +08:00
|
|
|
from drf_spectacular.utils import extend_schema
|
|
|
|
from drf_spectacular.openapi import OpenApiParameter
|
|
|
|
|
2023-08-15 12:00:24 +08:00
|
|
|
from collector.models import Archive, Ticket, Platform
|
2023-08-14 15:56:17 +08:00
|
|
|
|
2023-08-11 21:12:51 +08:00
|
|
|
from .filters import ArchiveFilter, TicketFilter
|
2023-08-11 09:38:47 +08:00
|
|
|
from .permissions import IsGuestUpload
|
|
|
|
from .serializers import (
|
|
|
|
PublicArchiveUploadSerializer,
|
|
|
|
ArchiveSerializer,
|
|
|
|
PlatformSerializer,
|
|
|
|
TicketSerializer
|
|
|
|
)
|
2023-08-08 13:42:57 +08:00
|
|
|
|
2023-07-25 14:29:17 +08:00
|
|
|
|
2023-08-11 09:38:47 +08:00
|
|
|
class ArchiveViewSet(viewsets.ModelViewSet):
|
2023-08-08 13:42:57 +08:00
|
|
|
queryset = Archive.objects.order_by('-time_create')
|
2023-08-11 09:38:47 +08:00
|
|
|
serializer_class = ArchiveSerializer
|
2023-08-31 22:02:13 +08:00
|
|
|
parser_classes = (MultiPartParser, FormParser, FileUploadParser)
|
2023-08-11 09:38:47 +08:00
|
|
|
permission_classes = (IsGuestUpload, )
|
2023-08-11 21:12:51 +08:00
|
|
|
filter_backends = [DjangoFilterBackend]
|
|
|
|
filterset_class = ArchiveFilter
|
2023-08-08 13:42:57 +08:00
|
|
|
|
2023-08-16 23:53:13 +08:00
|
|
|
@extend_schema(
|
|
|
|
operation_id='upload_file',
|
|
|
|
request={
|
|
|
|
'multipart/form-data': {
|
|
|
|
'type': 'object',
|
|
|
|
'properties': {
|
|
|
|
'file': {
|
|
|
|
'type': 'string',
|
|
|
|
'format': 'binary'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
parameters=[
|
|
|
|
OpenApiParameter(
|
|
|
|
name='Upload-Token',
|
|
|
|
type=str,
|
|
|
|
location=OpenApiParameter.HEADER,
|
|
|
|
description="upload permission token",
|
|
|
|
),
|
|
|
|
]
|
|
|
|
)
|
2023-08-09 19:24:16 +08:00
|
|
|
def create(self, request, *args, **kwargs):
|
|
|
|
# ! upload-token protection:
|
|
|
|
upload_token = request.headers.get('upload-token', '')
|
2023-08-16 23:53:13 +08:00
|
|
|
if upload_token:
|
2023-08-09 19:24:16 +08:00
|
|
|
try:
|
|
|
|
bound_ticket = Ticket.objects.get(token=upload_token)
|
|
|
|
if bound_ticket.resolved:
|
|
|
|
return Response(
|
2023-08-16 23:53:13 +08:00
|
|
|
{'error': f'ticket {bound_ticket} already resolved'},
|
2023-08-09 19:24:16 +08:00
|
|
|
status=status.HTTP_423_LOCKED
|
|
|
|
)
|
|
|
|
if bound_ticket.attempts <= 0:
|
|
|
|
return Response(
|
|
|
|
{'error': f'token {upload_token} expired'},
|
|
|
|
status=status.HTTP_423_LOCKED
|
|
|
|
)
|
|
|
|
bound_ticket.attempts -= 1
|
|
|
|
bound_ticket.save()
|
2023-08-13 15:31:40 +08:00
|
|
|
# ? mixin bound ticket number to request.data from user
|
2023-08-11 09:38:47 +08:00
|
|
|
request.data['ticket'] = bound_ticket.number
|
|
|
|
# ? change serializer for guest user
|
2023-08-16 23:53:13 +08:00
|
|
|
if not request.user.is_authenticated:
|
|
|
|
self.serializer_class = PublicArchiveUploadSerializer
|
2023-08-11 09:38:47 +08:00
|
|
|
except (ValidationError, ObjectDoesNotExist,):
|
2023-08-09 19:24:16 +08:00
|
|
|
return Response(
|
|
|
|
{'error': f'token {upload_token} is not valid'},
|
|
|
|
status=status.HTTP_403_FORBIDDEN
|
|
|
|
)
|
2023-08-16 23:53:13 +08:00
|
|
|
else:
|
2023-08-09 19:24:16 +08:00
|
|
|
return Response(
|
|
|
|
{'error': 'Header Upload-Token is required'},
|
|
|
|
status=status.HTTP_401_UNAUTHORIZED
|
|
|
|
)
|
|
|
|
# ! default create method:
|
|
|
|
serializer = self.get_serializer(data=request.data)
|
|
|
|
serializer.is_valid(raise_exception=True)
|
|
|
|
self.perform_create(serializer)
|
|
|
|
headers = self.get_success_headers(serializer.data)
|
|
|
|
return Response(
|
|
|
|
serializer.data,
|
|
|
|
status=status.HTTP_201_CREATED,
|
|
|
|
headers=headers
|
|
|
|
)
|
2023-08-08 17:08:53 +08:00
|
|
|
|
2023-08-11 09:38:47 +08:00
|
|
|
|
|
|
|
class PlatformViewSet(viewsets.ModelViewSet):
|
|
|
|
queryset = Platform.objects.all()
|
|
|
|
lookup_field = 'name'
|
|
|
|
serializer_class = PlatformSerializer
|
|
|
|
permission_classes = (IsAuthenticated, )
|
|
|
|
|
|
|
|
|
|
|
|
class TicketViewSet(viewsets.ModelViewSet):
|
|
|
|
queryset = Ticket.objects.order_by('-time_create')
|
|
|
|
lookup_field = 'number'
|
|
|
|
serializer_class = TicketSerializer
|
|
|
|
permission_classes = (IsAuthenticated, )
|
2023-08-11 21:12:51 +08:00
|
|
|
filter_backends = [DjangoFilterBackend, filters.SearchFilter]
|
|
|
|
filterset_class = TicketFilter
|
|
|
|
search_fields = ['number']
|
|
|
|
|
|
|
|
def perform_create(self, serializer):
|
|
|
|
serializer.save(user=self.request.user)
|