threads.py 3.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091
  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 detail_route, list_route
  6. from rest_framework.parsers import JSONParser
  7. from rest_framework.response import Response
  8. from misago.acl import add_acl
  9. from misago.categories.models import CATEGORIES_TREE_ID, Category
  10. from misago.categories.permissions import (
  11. allow_see_category, allow_browse_category)
  12. from misago.core.shortcuts import get_int_or_404, get_object_or_404
  13. from misago.readtracker.categoriestracker import read_category
  14. from misago.users.rest_permissions import IsAuthenticatedOrReadOnly
  15. from misago.threads.api.threadendpoints.lists import threads_list_endpoint
  16. from misago.threads.api.threadendpoints.merge import threads_merge_endpoint
  17. from misago.threads.api.threadendpoints.patch import thread_patch_endpoint
  18. from misago.threads.models import Thread, Subscription
  19. from misago.threads.moderation import threads as moderation
  20. from misago.threads.permissions.threads import allow_see_thread
  21. from misago.threads.serializers import ThreadSerializer
  22. from misago.threads.subscriptions import make_subscription_aware
  23. class ThreadViewSet(viewsets.ViewSet):
  24. permission_classes = (IsAuthenticatedOrReadOnly, )
  25. parser_classes=(JSONParser, )
  26. TREE_ID = CATEGORIES_TREE_ID
  27. def validate_thread_visible(self, user, thread):
  28. allow_see_thread(user, thread)
  29. def get_thread(self, user, thread_id):
  30. thread = get_object_or_404(Thread.objects.select_related('category'),
  31. id=get_int_or_404(thread_id),
  32. category__tree_id=self.TREE_ID,
  33. )
  34. add_acl(user, thread.category)
  35. add_acl(user, thread)
  36. self.validate_thread_visible(user, thread)
  37. return thread
  38. def list(self, request):
  39. return threads_list_endpoint(request)
  40. def retrieve(self, request, pk=None):
  41. thread = self.get_thread(request.user, pk)
  42. make_subscription_aware(request.user, thread)
  43. return Response(ThreadSerializer(thread).data)
  44. def partial_update(self, request, pk=None):
  45. thread = self.get_thread(request.user, pk)
  46. return thread_patch_endpoint.dispatch(request, thread)
  47. def destroy(self, request, pk=None):
  48. thread = self.get_thread(request.user, pk)
  49. if thread.acl.get('can_hide') == 2:
  50. moderation.delete_thread(request.user, thread)
  51. return Response({'detail': 'ok'})
  52. else:
  53. raise PermissionDenied(
  54. _("You don't have permission to delete this thread."))
  55. @list_route(methods=['post'])
  56. def read(self, request):
  57. if request.query_params.get('category'):
  58. category_id = get_int_or_404(request.query_params.get('category'))
  59. category = get_object_or_404(Category.objects,
  60. id=category_id,
  61. tree_id=self.TREE_ID,
  62. )
  63. allow_see_category(request.user, category)
  64. allow_browse_category(request.user, category)
  65. else:
  66. category = Category.objects.root_category()
  67. read_category(request.user, category)
  68. return Response({'detail': 'ok'})
  69. @list_route(methods=['post'])
  70. def merge(self, request):
  71. return threads_merge_endpoint(request)