threadpoll.py 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. from django.core.exceptions import PermissionDenied
  2. from django.db import transaction
  3. from django.http import Http404
  4. from django.utils.translation import gettext as _
  5. from rest_framework import viewsets
  6. from rest_framework.decorators import detail_route
  7. from rest_framework.response import Response
  8. from misago.acl.objectacl import add_acl_to_obj
  9. from misago.core.shortcuts import get_int_or_404
  10. from misago.threads.models import Poll
  11. from misago.threads.permissions import (
  12. allow_delete_poll,
  13. allow_edit_poll,
  14. allow_see_poll_votes,
  15. allow_start_poll,
  16. can_start_poll,
  17. )
  18. from misago.threads.serializers import (
  19. EditPollSerializer,
  20. NewPollSerializer,
  21. PollSerializer,
  22. PollVoteSerializer,
  23. )
  24. from misago.threads.viewmodels import ForumThread
  25. from misago.users.audittrail import create_audit_trail
  26. from .pollvotecreateendpoint import poll_vote_create
  27. class ViewSet(viewsets.ViewSet):
  28. thread = None
  29. def get_thread(self, request, thread_pk):
  30. return self.thread(request, get_int_or_404(thread_pk)).unwrap()
  31. def get_poll(self, thread, pk):
  32. try:
  33. poll_id = get_int_or_404(pk)
  34. if thread.poll.pk != poll_id:
  35. raise Http404()
  36. poll = Poll.objects.get(pk=thread.poll.pk)
  37. poll.thread = thread
  38. poll.category = thread.category
  39. return poll
  40. except Poll.DoesNotExist:
  41. raise Http404()
  42. @transaction.atomic
  43. def create(self, request, thread_pk):
  44. thread = self.get_thread(request, thread_pk)
  45. allow_start_poll(request.user_acl, thread)
  46. try:
  47. if thread.poll and thread.poll.pk:
  48. raise PermissionDenied(_("There's already a poll in this thread."))
  49. except Poll.DoesNotExist:
  50. pass
  51. instance = Poll(
  52. thread=thread,
  53. category=thread.category,
  54. poster=request.user,
  55. poster_name=request.user.username,
  56. poster_slug=request.user.slug,
  57. )
  58. serializer = NewPollSerializer(instance, data=request.data)
  59. serializer.is_valid(raise_exception=True)
  60. serializer.save()
  61. add_acl_to_obj(request.user_acl, instance)
  62. for choice in instance.choices:
  63. choice["selected"] = False
  64. thread.has_poll = True
  65. thread.save()
  66. create_audit_trail(request, instance)
  67. return Response(PollSerializer(instance).data)
  68. @transaction.atomic
  69. def update(self, request, thread_pk, pk=None):
  70. thread = self.get_thread(request, thread_pk)
  71. instance = self.get_poll(thread, pk)
  72. allow_edit_poll(request.user_acl, instance)
  73. serializer = EditPollSerializer(instance, data=request.data)
  74. serializer.is_valid(raise_exception=True)
  75. serializer.save()
  76. add_acl_to_obj(request.user_acl, instance)
  77. instance.make_choices_votes_aware(request.user)
  78. create_audit_trail(request, instance)
  79. return Response(PollSerializer(instance).data)
  80. @transaction.atomic
  81. def delete(self, request, thread_pk, pk=None):
  82. thread = self.get_thread(request, thread_pk)
  83. instance = self.get_poll(thread, pk)
  84. allow_delete_poll(request.user_acl, instance)
  85. thread.poll.delete()
  86. thread.has_poll = False
  87. thread.save()
  88. return Response({"can_start_poll": can_start_poll(request.user_acl, thread)})
  89. @detail_route(methods=["get", "post"])
  90. def votes(self, request, thread_pk, pk=None):
  91. if request.method == "POST":
  92. return self.post_votes(request, thread_pk, pk)
  93. else:
  94. return self.get_votes(request, thread_pk, pk)
  95. @transaction.atomic
  96. def post_votes(self, request, thread_pk, pk=None):
  97. thread = self.get_thread(request, thread_pk)
  98. instance = self.get_poll(thread, pk)
  99. return poll_vote_create(request, thread, instance)
  100. def get_votes(self, request, thread_pk, pk=None):
  101. poll_pk = get_int_or_404(pk)
  102. try:
  103. thread = self.get_thread(request, thread_pk)
  104. if thread.poll.pk != poll_pk:
  105. raise Http404()
  106. except Poll.DoesNotExist:
  107. raise Http404()
  108. allow_see_poll_votes(request.user_acl, thread.poll)
  109. choices = []
  110. voters = {}
  111. for choice in thread.poll.choices:
  112. choice["voters"] = []
  113. voters[choice["hash"]] = choice["voters"]
  114. choices.append(choice)
  115. queryset = thread.poll.pollvote_set.values(
  116. "voter_id", "voter_name", "voter_slug", "voted_on", "choice_hash"
  117. )
  118. for voter in queryset.order_by("voter_name").iterator():
  119. voters[voter["choice_hash"]].append(PollVoteSerializer(voter).data)
  120. return Response(choices)
  121. class ThreadPollViewSet(ViewSet):
  122. thread = ForumThread