threads.py 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. from rest_framework import viewsets
  2. from rest_framework.decorators import detail_route, list_route
  3. from rest_framework.response import Response
  4. from django.core.exceptions import PermissionDenied
  5. from django.db import transaction
  6. from django.utils.translation import gettext as _
  7. from misago.categories import PRIVATE_THREADS_ROOT_NAME, THREADS_ROOT_NAME
  8. from misago.core.shortcuts import get_int_or_404
  9. from misago.threads.models import Post, Thread
  10. from misago.threads.moderation import threads as moderation
  11. from misago.threads.permissions import allow_use_private_threads
  12. from misago.threads.viewmodels import ForumThread, PrivateThread
  13. from .postingendpoint import PostingEndpoint
  14. from .threadendpoints.editor import thread_start_editor
  15. from .threadendpoints.list import private_threads_list_endpoint, threads_list_endpoint
  16. from .threadendpoints.merge import thread_merge_endpoint, threads_merge_endpoint
  17. from .threadendpoints.patch import thread_patch_endpoint
  18. from .threadendpoints.read import read_private_threads, read_threads
  19. class ViewSet(viewsets.ViewSet):
  20. thread = None
  21. def get_thread(self, request, pk, read_aware=True, subscription_aware=True, select_for_update=False):
  22. return self.thread(
  23. request,
  24. get_int_or_404(pk),
  25. None,
  26. read_aware,
  27. subscription_aware,
  28. select_for_update
  29. )
  30. def get_thread_for_update(self, request, pk):
  31. return self.get_thread(
  32. request, pk,
  33. read_aware=False,
  34. subscription_aware=False,
  35. select_for_update=True
  36. )
  37. def retrieve(self, request, pk):
  38. thread = self.get_thread(request, pk)
  39. return Response(thread.get_frontend_context())
  40. @transaction.atomic
  41. def partial_update(self, request, pk):
  42. thread = self.get_thread_for_update(request, pk).unwrap()
  43. return thread_patch_endpoint(request, thread)
  44. @transaction.atomic
  45. def destroy(self, request, pk):
  46. thread = self.get_thread_for_update(request, pk)
  47. if thread.acl.get('can_hide') == 2:
  48. moderation.delete_thread(request.user, thread)
  49. return Response({'detail': 'ok'})
  50. else:
  51. raise PermissionDenied(_("You don't have permission to delete this thread."))
  52. class ThreadViewSet(ViewSet):
  53. thread = ForumThread
  54. def list(self, request):
  55. return threads_list_endpoint(request)
  56. @transaction.atomic
  57. def create(self, request):
  58. # Initialize empty instances for new thread
  59. thread = Thread()
  60. post = Post(thread=thread)
  61. # Put them through posting pipeline
  62. posting = PostingEndpoint(
  63. request,
  64. PostingEndpoint.START,
  65. tree_name=THREADS_ROOT_NAME,
  66. thread=thread,
  67. post=post
  68. )
  69. if posting.is_valid():
  70. posting.save()
  71. return Response({
  72. 'id': thread.pk,
  73. 'title': thread.title,
  74. 'url': thread.get_absolute_url()
  75. })
  76. else:
  77. return Response(posting.errors, status=400)
  78. @detail_route(methods=['post'], url_path='merge')
  79. @transaction.atomic
  80. def thread_merge(self, request, pk):
  81. thread = self.get_thread_for_update(request, pk).unwrap()
  82. return thread_merge_endpoint(request, thread, self.thread)
  83. @list_route(methods=['post'], url_path='merge')
  84. @transaction.atomic
  85. def threads_merge(self, request):
  86. return threads_merge_endpoint(request)
  87. @list_route(methods=['get'])
  88. def editor(self, request):
  89. return thread_start_editor(request)
  90. @list_route(methods=['post'])
  91. @transaction.atomic
  92. def read(self, request):
  93. read_threads(request.user, request.GET.get('category'))
  94. return Response({'detail': 'ok'})
  95. class PrivateThreadViewSet(ViewSet):
  96. thread = PrivateThread
  97. def list(self, request):
  98. return private_threads_list_endpoint(request)
  99. @transaction.atomic
  100. def create(self, request):
  101. allow_use_private_threads(request.user)
  102. if not request.user.acl_cache['can_start_private_threads']:
  103. raise PermissionDenied(_("You can't start private threads."))
  104. # Initialize empty instances for new thread
  105. thread = Thread()
  106. post = Post(thread=thread)
  107. # Put them through posting pipeline
  108. posting = PostingEndpoint(
  109. request,
  110. PostingEndpoint.START,
  111. tree_name=PRIVATE_THREADS_ROOT_NAME,
  112. thread=thread,
  113. post=post
  114. )
  115. if posting.is_valid():
  116. posting.save()
  117. return Response({
  118. 'id': thread.pk,
  119. 'title': thread.title,
  120. 'url': thread.get_absolute_url()
  121. })
  122. else:
  123. return Response(posting.errors, status=400)
  124. @list_route(methods=['post'])
  125. @transaction.atomic
  126. def read(self, request):
  127. allow_use_private_threads(request.user)
  128. read_private_threads(request.user)
  129. return Response({'detail': 'ok'})