threads.py 5.1 KB

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