2023-08-01 19:13:04 +08:00
|
|
|
import json
|
2023-08-11 09:38:47 +08:00
|
|
|
from django.core.exceptions import ValidationError, ObjectDoesNotExist
|
2023-08-01 19:13:04 +08:00
|
|
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
2023-08-04 09:15:25 +08:00
|
|
|
from django.http import FileResponse, JsonResponse
|
2023-07-29 12:56:42 +08:00
|
|
|
from django.views import generic
|
2023-08-04 09:15:25 +08:00
|
|
|
from django.views.generic.detail import SingleObjectMixin
|
2023-08-03 10:57:02 +08:00
|
|
|
from django.urls import reverse_lazy
|
2023-08-07 11:31:30 +08:00
|
|
|
from django.db.models import Q
|
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-08 13:42:57 +08:00
|
|
|
from rest_framework.parsers import FormParser, MultiPartParser
|
2023-08-11 09:38:47 +08:00
|
|
|
from rest_framework.permissions import IsAuthenticated
|
|
|
|
from rest_framework.response import Response
|
2023-08-08 13:42:57 +08:00
|
|
|
|
2023-08-11 09:38:47 +08:00
|
|
|
# from rest_framework import mixins
|
|
|
|
from rest_framework import viewsets
|
2023-07-31 12:53:39 +08:00
|
|
|
|
2023-08-11 09:38:47 +08:00
|
|
|
from .models import Archive, Ticket, Platform
|
2023-08-06 10:53:09 +08:00
|
|
|
from .forms import TicketForm
|
|
|
|
from .utils import PageTitleViewMixin, is_ajax
|
2023-08-11 09:38:47 +08:00
|
|
|
from .permissions import IsGuestUpload
|
2023-07-27 10:26:27 +08:00
|
|
|
|
2023-08-11 09:38:47 +08:00
|
|
|
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-04 09:15:25 +08:00
|
|
|
class ArchiveHandlerView(LoginRequiredMixin, SingleObjectMixin, generic.View):
|
|
|
|
model = Archive
|
|
|
|
slug_field = 'file'
|
|
|
|
slug_url_kwarg = 'path'
|
|
|
|
|
2023-07-31 12:53:39 +08:00
|
|
|
def get(self, request, path):
|
2023-08-04 09:15:25 +08:00
|
|
|
self.object = self.get_object()
|
|
|
|
return FileResponse(self.object.file)
|
2023-07-31 12:53:39 +08:00
|
|
|
|
|
|
|
def delete(self, request, path):
|
2023-08-04 09:15:25 +08:00
|
|
|
if is_ajax(request):
|
|
|
|
self.object = self.get_object()
|
|
|
|
self.object.delete()
|
|
|
|
return JsonResponse({'file': path}, status=status.HTTP_200_OK)
|
2023-07-31 12:53:39 +08:00
|
|
|
|
|
|
|
|
2023-08-06 10:53:09 +08:00
|
|
|
class CreateTicket(LoginRequiredMixin, PageTitleViewMixin, generic.CreateView):
|
2023-08-05 14:43:48 +08:00
|
|
|
model = Ticket
|
2023-08-06 10:53:09 +08:00
|
|
|
form_class = TicketForm
|
2023-08-05 14:43:48 +08:00
|
|
|
template_name = 'collector/ticket_create.html'
|
|
|
|
|
2023-08-06 10:53:09 +08:00
|
|
|
def get_title(self):
|
|
|
|
return f'{self.title} - create'
|
|
|
|
|
|
|
|
def form_valid(self, form):
|
|
|
|
form.instance.user = self.request.user
|
|
|
|
return super().form_valid(form)
|
|
|
|
|
|
|
|
|
|
|
|
class UpdateTicket(LoginRequiredMixin, PageTitleViewMixin, generic.UpdateView):
|
|
|
|
model = Ticket
|
|
|
|
form_class = TicketForm
|
|
|
|
template_name = 'collector/ticket_create.html'
|
|
|
|
slug_field = 'number'
|
|
|
|
slug_url_kwarg = 'ticket'
|
|
|
|
|
|
|
|
def get_title(self, **kwargs):
|
|
|
|
return f'{self.title} - {self.kwargs.get("ticket", "update")}'
|
|
|
|
|
2023-08-05 14:43:48 +08:00
|
|
|
def form_valid(self, form):
|
2023-08-08 13:42:57 +08:00
|
|
|
print(self.request.user)
|
2023-08-05 14:43:48 +08:00
|
|
|
form.instance.user = self.request.user
|
|
|
|
return super().form_valid(form)
|
|
|
|
|
|
|
|
|
2023-08-06 10:53:09 +08:00
|
|
|
class ListAllTickets(PageTitleViewMixin, generic.ListView):
|
2023-07-29 12:56:42 +08:00
|
|
|
model = Ticket
|
|
|
|
template_name = 'collector/tickets.html'
|
|
|
|
context_object_name = 'tickets'
|
2023-07-29 15:36:08 +08:00
|
|
|
paginate_by = 5
|
2023-08-06 10:53:09 +08:00
|
|
|
title = 'Collector - tickets'
|
2023-07-29 12:56:42 +08:00
|
|
|
|
2023-08-07 11:31:30 +08:00
|
|
|
def get_queryset(self):
|
|
|
|
search_query = self.request.GET.get('search', '')
|
|
|
|
if search_query:
|
|
|
|
query_list = []
|
|
|
|
try:
|
|
|
|
for item in search_query.split(','):
|
|
|
|
query_list.append(int(item))
|
|
|
|
except ValueError:
|
|
|
|
return super().get_queryset()
|
|
|
|
queryset = self.model.objects.filter(
|
|
|
|
Q(number__in=query_list) | Q(number__icontains=query_list[0])
|
|
|
|
)
|
|
|
|
self.paginate_by = 100 # fake disable pagination)
|
|
|
|
return queryset
|
|
|
|
|
|
|
|
return super().get_queryset()
|
|
|
|
|
2023-07-29 12:56:42 +08:00
|
|
|
|
2023-08-06 10:53:09 +08:00
|
|
|
class ListPlatformTickets(PageTitleViewMixin, generic.ListView):
|
2023-07-29 12:56:42 +08:00
|
|
|
model = Ticket
|
|
|
|
template_name = 'collector/tickets.html'
|
|
|
|
context_object_name = 'tickets'
|
2023-08-04 10:53:53 +08:00
|
|
|
# allow_empty = False
|
2023-07-29 15:36:08 +08:00
|
|
|
paginate_by = 5
|
2023-07-29 12:56:42 +08:00
|
|
|
|
2023-08-06 10:53:09 +08:00
|
|
|
def get_title(self, **kwargs):
|
|
|
|
return f'{self.title} - {self.kwargs.get("platform", "tickets")}'
|
|
|
|
|
2023-07-29 12:56:42 +08:00
|
|
|
def get_queryset(self):
|
|
|
|
return Ticket.objects.filter(
|
|
|
|
platform__name=self.kwargs.get('platform')
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2023-08-06 10:53:09 +08:00
|
|
|
class DetailTicket(PageTitleViewMixin, generic.DetailView):
|
2023-07-29 12:56:42 +08:00
|
|
|
model = Ticket
|
|
|
|
template_name = 'collector/ticket.html'
|
|
|
|
context_object_name = 'ticket'
|
2023-08-03 10:57:02 +08:00
|
|
|
slug_field = 'number'
|
|
|
|
slug_url_kwarg = 'ticket'
|
2023-07-27 10:26:27 +08:00
|
|
|
|
2023-08-06 10:53:09 +08:00
|
|
|
def get_title(self, **kwargs):
|
|
|
|
return f'{self.title} - {self.kwargs.get("ticket", "show")}'
|
|
|
|
|
2023-08-04 09:15:25 +08:00
|
|
|
|
2023-08-06 10:53:09 +08:00
|
|
|
class DeleteTicket(PageTitleViewMixin, generic.DeleteView):
|
2023-08-04 09:15:25 +08:00
|
|
|
model = Ticket
|
2023-08-05 14:43:48 +08:00
|
|
|
template_name = 'collector/ticket_delete.html'
|
2023-08-04 09:15:25 +08:00
|
|
|
context_object_name = 'ticket'
|
|
|
|
slug_field = 'number'
|
|
|
|
slug_url_kwarg = 'ticket'
|
|
|
|
success_url = reverse_lazy('tickets')
|
|
|
|
|
|
|
|
|
|
|
|
class UpdateTicketStateHandler(SingleObjectMixin, generic.View):
|
|
|
|
model = Ticket
|
|
|
|
slug_field = 'number'
|
|
|
|
slug_url_kwarg = 'ticket'
|
|
|
|
|
2023-08-03 10:57:02 +08:00
|
|
|
def post(self, request, **kwargs):
|
2023-08-01 19:13:04 +08:00
|
|
|
if is_ajax(request):
|
2023-08-04 09:15:25 +08:00
|
|
|
self.object = self.get_object()
|
2023-08-01 19:13:04 +08:00
|
|
|
if request.body:
|
|
|
|
data = json.loads(request.body)
|
2023-08-03 10:57:02 +08:00
|
|
|
resolved_field = data.get('resolved')
|
|
|
|
if isinstance(resolved_field, bool):
|
2023-08-04 09:15:25 +08:00
|
|
|
self.object.resolved = not resolved_field
|
|
|
|
self.object.save()
|
2023-08-03 10:57:02 +08:00
|
|
|
return JsonResponse(
|
|
|
|
{'resolved': not resolved_field},
|
|
|
|
status=status.HTTP_201_CREATED
|
|
|
|
)
|
|
|
|
return JsonResponse(
|
|
|
|
{'resolved': 'must be a boolean'},
|
|
|
|
status=status.HTTP_400_BAD_REQUEST
|
|
|
|
)
|
2023-08-04 09:15:25 +08:00
|
|
|
return JsonResponse(
|
|
|
|
{'error': 'header XMLHttpRequest is required'},
|
|
|
|
status=status.HTTP_406_NOT_ACCEPTABLE
|
|
|
|
)
|
2023-08-03 10:57:02 +08:00
|
|
|
|
|
|
|
|
2023-08-04 09:15:25 +08:00
|
|
|
class DeleteTicketHandler(SingleObjectMixin, generic.View):
|
2023-08-03 10:57:02 +08:00
|
|
|
model = Ticket
|
|
|
|
slug_field = 'number'
|
|
|
|
slug_url_kwarg = 'ticket'
|
|
|
|
|
2023-08-04 09:15:25 +08:00
|
|
|
def delete(self, request, ticket):
|
2023-08-03 10:57:02 +08:00
|
|
|
if is_ajax(request):
|
|
|
|
self.object = self.get_object()
|
|
|
|
self.object.delete()
|
|
|
|
return JsonResponse(
|
|
|
|
{'status': status.HTTP_200_OK},
|
|
|
|
status=status.HTTP_200_OK
|
|
|
|
)
|
2023-08-04 09:15:25 +08:00
|
|
|
return JsonResponse(
|
|
|
|
{'error': 'header XMLHttpRequest is required'},
|
|
|
|
status=status.HTTP_406_NOT_ACCEPTABLE
|
|
|
|
)
|
2023-08-08 13:42:57 +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-08 13:42:57 +08:00
|
|
|
parser_classes = (MultiPartParser, FormParser)
|
2023-08-11 09:38:47 +08:00
|
|
|
permission_classes = (IsGuestUpload, )
|
2023-08-08 13:42:57 +08:00
|
|
|
|
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-11 09:38:47 +08:00
|
|
|
if not request.user.is_authenticated and 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(
|
|
|
|
{'error': f'ticket {upload_token} already resolved'},
|
|
|
|
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()
|
|
|
|
# ? mixin bound ticket to request.data from user
|
2023-08-11 09:38:47 +08:00
|
|
|
request.data['ticket'] = bound_ticket.number
|
|
|
|
# ? change serializer for guest user
|
|
|
|
self.serializer_class = PublicArchiveUploadSerializer
|
|
|
|
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-11 09:38:47 +08:00
|
|
|
elif not request.user.is_authenticated:
|
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, )
|