patch_post.py 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. from rest_framework import serializers
  2. from rest_framework.response import Response
  3. from django.core.exceptions import PermissionDenied
  4. from django.utils.translation import gettext as _
  5. from misago.acl import add_acl
  6. from misago.conf import settings
  7. from misago.core.apipatch import ApiPatch
  8. from misago.threads.models import PostLike
  9. from misago.threads.moderation import posts as moderation
  10. from misago.threads.permissions import (
  11. allow_approve_post, allow_hide_best_answer, allow_hide_post, allow_protect_post,
  12. allow_unhide_post)
  13. from misago.threads.permissions import exclude_invisible_posts
  14. PATCH_LIMIT = settings.MISAGO_POSTS_PER_PAGE + settings.MISAGO_POSTS_TAIL
  15. post_patch_dispatcher = ApiPatch()
  16. def patch_acl(request, post, value):
  17. """useful little op that updates post acl to current state"""
  18. if value:
  19. add_acl(request.user_acl, post)
  20. return {'acl': post.acl}
  21. else:
  22. return {'acl': None}
  23. post_patch_dispatcher.add('acl', patch_acl)
  24. def patch_is_liked(request, post, value):
  25. if not post.acl['can_like']:
  26. raise PermissionDenied(_("You can't like posts in this category."))
  27. # lock user to protect us from likes flood
  28. request.user.lock()
  29. # grab like state for this post and user
  30. try:
  31. user_like = post.postlike_set.get(liker=request.user)
  32. except PostLike.DoesNotExist:
  33. user_like = None
  34. # no change
  35. if (value and user_like) or (not value and not user_like):
  36. return {
  37. 'likes': post.likes,
  38. 'last_likes': post.last_likes or [],
  39. 'is_liked': value,
  40. }
  41. # like
  42. if value:
  43. post.postlike_set.create(
  44. category=post.category,
  45. thread=post.thread,
  46. liker=request.user,
  47. liker_name=request.user.username,
  48. liker_slug=request.user.slug,
  49. )
  50. post.likes += 1
  51. # unlike
  52. if not value:
  53. user_like.delete()
  54. post.likes -= 1
  55. post.last_likes = []
  56. for like in post.postlike_set.all()[:4]:
  57. post.last_likes.append({
  58. 'id': like.liker_id,
  59. 'username': like.liker_name,
  60. })
  61. post.save(update_fields=['likes', 'last_likes'])
  62. return {
  63. 'likes': post.likes,
  64. 'last_likes': post.last_likes or [],
  65. 'is_liked': value,
  66. }
  67. post_patch_dispatcher.replace('is-liked', patch_is_liked)
  68. def patch_is_protected(request, post, value):
  69. allow_protect_post(request.user_acl, post)
  70. if value:
  71. moderation.protect_post(request.user, post)
  72. else:
  73. moderation.unprotect_post(request.user, post)
  74. return {'is_protected': post.is_protected}
  75. post_patch_dispatcher.replace('is-protected', patch_is_protected)
  76. def patch_is_unapproved(request, post, value):
  77. allow_approve_post(request.user_acl, post)
  78. if value:
  79. raise PermissionDenied(_("Content approval can't be reversed."))
  80. moderation.approve_post(request.user, post)
  81. return {'is_unapproved': post.is_unapproved}
  82. post_patch_dispatcher.replace('is-unapproved', patch_is_unapproved)
  83. def patch_is_hidden(request, post, value):
  84. if value is True:
  85. allow_hide_post(request.user_acl, post)
  86. allow_hide_best_answer(request.user_acl, post)
  87. moderation.hide_post(request.user, post)
  88. elif value is False:
  89. allow_unhide_post(request.user_acl, post)
  90. moderation.unhide_post(request.user, post)
  91. return {'is_hidden': post.is_hidden}
  92. post_patch_dispatcher.replace('is-hidden', patch_is_hidden)
  93. def post_patch_endpoint(request, post):
  94. old_is_unapproved = post.is_unapproved
  95. response = post_patch_dispatcher.dispatch(request, post)
  96. # diff posts's state against pre-patch and resync thread/category if necessarys
  97. if old_is_unapproved != post.is_unapproved:
  98. post.thread.synchronize()
  99. post.thread.save()
  100. post.category.synchronize()
  101. post.category.save()
  102. return response
  103. def bulk_patch_endpoint(request, thread):
  104. serializer = BulkPatchSerializer(data=request.data)
  105. if not serializer.is_valid():
  106. return Response(serializer.errors, status=400)
  107. posts = clean_posts_for_patch(request, thread, serializer.data['ids'])
  108. old_unapproved_posts = [p.is_unapproved for p in posts].count(True)
  109. response = post_patch_dispatcher.dispatch_bulk(request, posts)
  110. new_unapproved_posts = [p.is_unapproved for p in posts].count(True)
  111. if old_unapproved_posts != new_unapproved_posts:
  112. thread.synchronize()
  113. thread.save()
  114. thread.category.synchronize()
  115. thread.category.save()
  116. return response
  117. def clean_posts_for_patch(request, thread, posts_ids):
  118. posts_queryset = exclude_invisible_posts(request.user, thread.category, thread.post_set)
  119. posts_queryset = posts_queryset.filter(
  120. id__in=posts_ids,
  121. is_event=False,
  122. ).order_by('id')
  123. posts = []
  124. for post in posts_queryset:
  125. post.category = thread.category
  126. post.thread = thread
  127. posts.append(post)
  128. if len(posts) != len(posts_ids):
  129. raise PermissionDenied(_("One or more posts to update could not be found."))
  130. return posts
  131. class BulkPatchSerializer(serializers.Serializer):
  132. ids = serializers.ListField(
  133. child=serializers.IntegerField(min_value=1),
  134. max_length=PATCH_LIMIT,
  135. min_length=1,
  136. )
  137. ops = serializers.ListField(
  138. child=serializers.DictField(),
  139. min_length=1,
  140. max_length=10,
  141. )