threads.py 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. from django.core.exceptions import PermissionDenied
  2. from django.db import transaction
  3. from django.utils.translation import gettext as _
  4. from rest_framework import viewsets
  5. from rest_framework.decorators import list_route
  6. from rest_framework.response import Response
  7. from misago.acl import add_acl
  8. from misago.categories.models import THREADS_ROOT_NAME, Category
  9. from misago.categories.permissions import allow_browse_category, allow_see_category
  10. from misago.core.shortcuts import get_int_or_404, get_object_or_404
  11. from misago.readtracker.categoriestracker import read_category
  12. from ..models import Post, Subscription, Thread
  13. from ..moderation import threads as moderation
  14. from ..permissions.threads import can_start_thread
  15. from ..subscriptions import make_subscription_aware
  16. from ..threadtypes import trees_map
  17. from ..viewmodels.thread import ForumThread
  18. from .postingendpoint import PostingEndpoint
  19. from .threadendpoints.list import threads_list_endpoint
  20. from .threadendpoints.merge import threads_merge_endpoint
  21. from .threadendpoints.patch import thread_patch_endpoint
  22. class ViewSet(viewsets.ViewSet):
  23. thread = None
  24. TREE_ID = None
  25. def get_thread(self, request, pk):
  26. return self.thread(request, get_int_or_404(pk), read_aware=True, subscription_aware=True)
  27. def list(self, request):
  28. return threads_list_endpoint(request)
  29. def retrieve(self, request, pk):
  30. thread = self.get_thread(request, pk)
  31. return Response(thread.get_frontend_context())
  32. def partial_update(self, request, pk):
  33. thread = self.get_thread(request, pk).thread
  34. old_is_hidden = thread.is_hidden
  35. old_is_unapproved = thread.is_unapproved
  36. old_category = thread.category
  37. response = thread_patch_endpoint.dispatch(request, thread)
  38. # diff thread's state against pre-patch and resync category if necessary
  39. hidden_changed = old_is_hidden != thread.is_hidden
  40. unapproved_changed = old_is_unapproved != thread.is_unapproved
  41. category_changed = old_category != thread.category
  42. if hidden_changed or unapproved_changed or category_changed:
  43. thread.category.synchronize()
  44. thread.category.save()
  45. if category_changed:
  46. old_category.synchronize()
  47. old_category.save()
  48. # return response
  49. return response
  50. def destroy(self, request, pk):
  51. thread = self.get_thread(request, pk).thread
  52. if thread.acl.get('can_hide') == 2:
  53. moderation.delete_thread(request.user, thread)
  54. return Response({'detail': 'ok'})
  55. else:
  56. raise PermissionDenied(_("You don't have permission to delete this thread."))
  57. class ThreadViewSet(ViewSet):
  58. thread = ForumThread
  59. def create(self, request):
  60. # Initialize empty instances for new thread
  61. thread = Thread()
  62. post = Post(thread=thread)
  63. # Put them through posting pipeline
  64. posting = PostingEndpoint(
  65. request,
  66. PostingEndpoint.START,
  67. tree_name=THREADS_ROOT_NAME,
  68. thread=thread,
  69. post=post
  70. )
  71. if posting.is_valid():
  72. posting.save()
  73. return Response({
  74. 'id': thread.pk,
  75. 'title': thread.title,
  76. 'url': thread.get_absolute_url()
  77. })
  78. else:
  79. return Response(posting.errors, status=400)
  80. @list_route(methods=['post'])
  81. def merge(self, request):
  82. return threads_merge_endpoint(request)
  83. @list_route(methods=['post'])
  84. def read(self, request):
  85. if request.query_params.get('category'):
  86. threads_tree_id = trees_map.get_tree_id_for_root(THREADS_ROOT_NAME)
  87. category_id = get_int_or_404(request.query_params.get('category'))
  88. category = get_object_or_404(Category,
  89. id=category_id,
  90. tree_id=threads_tree_id,
  91. )
  92. allow_see_category(request.user, category)
  93. allow_browse_category(request.user, category)
  94. else:
  95. category = Category.objects.root_category()
  96. read_category(request.user, category)
  97. return Response({'detail': 'ok'})
  98. @list_route(methods=['get'])
  99. def editor(self, request):
  100. if request.user.is_anonymous():
  101. raise PermissionDenied(_("You need to be signed in to start threads."))
  102. # list of categories that allow or contain subcategories that allow new threads
  103. available = []
  104. categories = []
  105. for category in Category.objects.filter(pk__in=request.user.acl['browseable_categories']).order_by('-lft'):
  106. add_acl(request.user, category)
  107. post = False
  108. if can_start_thread(request.user, category):
  109. post = {
  110. 'close': bool(category.acl['can_close_threads']),
  111. 'hide': bool(category.acl['can_hide_threads']),
  112. 'pin': category.acl['can_pin_threads']
  113. }
  114. available.append(category.pk)
  115. available.append(category.parent_id)
  116. elif category.pk in available:
  117. available.append(category.parent_id)
  118. categories.append({
  119. 'id': category.pk,
  120. 'name': category.name,
  121. 'level': category.level - 1,
  122. 'post': post
  123. })
  124. # list only categories that allow new threads, or contains subcategory that allows one
  125. cleaned_categories = []
  126. for category in reversed(categories):
  127. if category['id'] in available:
  128. cleaned_categories.append(category)
  129. if not cleaned_categories:
  130. raise PermissionDenied(_("No categories that allow new threads are available to you at the moment."))
  131. return Response(cleaned_categories)