patch_post.py 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. from django.core.exceptions import PermissionDenied
  2. from django.utils.translation import ugettext as _
  3. from misago.acl import add_acl
  4. from misago.conf import settings
  5. from misago.core.apipatch import ApiPatch
  6. from misago.threads.models import PostLike
  7. from misago.threads.moderation import posts as moderation
  8. from misago.threads.permissions import (
  9. allow_approve_post, allow_hide_post, allow_protect_post, allow_unhide_post)
  10. from misago.threads.permissions import exclude_invisible_posts
  11. PATCH_LIMIT = settings.MISAGO_POSTS_PER_PAGE + settings.MISAGO_POSTS_TAIL
  12. post_patch_dispatcher = ApiPatch()
  13. def patch_acl(request, post, value):
  14. """useful little op that updates post acl to current state"""
  15. if value:
  16. add_acl(request.user, post)
  17. return {'acl': post.acl}
  18. else:
  19. return {'acl': None}
  20. post_patch_dispatcher.add('acl', patch_acl)
  21. def patch_is_liked(request, post, value):
  22. if not post.acl['can_like']:
  23. raise PermissionDenied(_("You can't like posts in this category."))
  24. # lock user to protect us from likes flood
  25. request.user.lock()
  26. # grab like state for this post and user
  27. try:
  28. user_like = post.postlike_set.get(liker=request.user)
  29. except PostLike.DoesNotExist:
  30. user_like = None
  31. # no change
  32. if (value and user_like) or (not value and not user_like):
  33. return {
  34. 'likes': post.likes,
  35. 'last_likes': post.last_likes or [],
  36. 'is_liked': value,
  37. }
  38. # like
  39. if value:
  40. post.postlike_set.create(
  41. category=post.category,
  42. thread=post.thread,
  43. liker=request.user,
  44. liker_name=request.user.username,
  45. liker_slug=request.user.slug,
  46. liker_ip=request.user_ip,
  47. )
  48. post.likes += 1
  49. # unlike
  50. if not value:
  51. user_like.delete()
  52. post.likes -= 1
  53. post.last_likes = []
  54. for like in post.postlike_set.all()[:4]:
  55. post.last_likes.append({
  56. 'id': like.liker_id,
  57. 'username': like.liker_name,
  58. })
  59. post.save(update_fields=['likes', 'last_likes'])
  60. return {
  61. 'likes': post.likes,
  62. 'last_likes': post.last_likes or [],
  63. 'is_liked': value,
  64. }
  65. post_patch_dispatcher.replace('is-liked', patch_is_liked)
  66. def patch_is_protected(request, post, value):
  67. allow_protect_post(request.user, post)
  68. if value:
  69. moderation.protect_post(request.user, post)
  70. else:
  71. moderation.unprotect_post(request.user, post)
  72. return {'is_protected': post.is_protected}
  73. post_patch_dispatcher.replace('is-protected', patch_is_protected)
  74. def patch_is_unapproved(request, post, value):
  75. allow_approve_post(request.user, post)
  76. if value:
  77. raise PermissionDenied(_("Content approval can't be reversed."))
  78. moderation.approve_post(request.user, post)
  79. return {'is_unapproved': post.is_unapproved}
  80. post_patch_dispatcher.replace('is-unapproved', patch_is_unapproved)
  81. def patch_is_hidden(request, post, value):
  82. if value is True:
  83. allow_hide_post(request.user, post)
  84. moderation.hide_post(request.user, post)
  85. elif value is False:
  86. allow_unhide_post(request.user, post)
  87. moderation.unhide_post(request.user, post)
  88. return {'is_hidden': post.is_hidden}
  89. post_patch_dispatcher.replace('is-hidden', patch_is_hidden)
  90. def post_patch_endpoint(request, post):
  91. old_is_hidden = post.is_hidden
  92. old_is_unapproved = post.is_unapproved
  93. old_thread = post.thread
  94. old_category = post.category
  95. response = post_patch_dispatcher.dispatch(request, post)
  96. # diff posts's state against pre-patch and resync category if necessary
  97. hidden_changed = old_is_hidden != post.is_hidden
  98. unapproved_changed = old_is_unapproved != post.is_unapproved
  99. thread_changed = old_thread != post.thread
  100. category_changed = old_category != post.category
  101. if hidden_changed or unapproved_changed or thread_changed or category_changed:
  102. post.thread.synchronize()
  103. post.thread.save()
  104. post.category.synchronize()
  105. post.category.save()
  106. if thread_changed:
  107. old_thread.synchronize()
  108. old_thread.save()
  109. if category_changed:
  110. old_category.synchronize()
  111. old_category.save()
  112. return response
  113. def bulk_patch_endpoint(request, thread):
  114. posts = clean_posts_for_patch(request, thread)
  115. hidden_posts = 0
  116. revealed_posts = 0
  117. moved_posts = 0
  118. response = post_patch_dispatcher.dispatch_bulk(request, posts)
  119. def clean_posts_for_patch(request, thread):
  120. if not isinstance(request.data, dict):
  121. raise PermissionDenied(_("Bulk PATCH request should be a dict with ids and ops keys."))
  122. # todo: move this ids list cleanup step to utility
  123. try:
  124. posts_ids = list(map(int, request.data.get('ids', [])))
  125. except (ValueError, TypeError):
  126. raise PermissionDenied(_("One or more post ids received were invalid."))
  127. if not posts_ids:
  128. raise PermissionDenied(_("You have to specify at least one post to update."))
  129. elif len(posts_ids) > PATCH_LIMIT:
  130. message = ungettext(
  131. "No more than %(limit)s post can be updated at single time.",
  132. "No more than %(limit)s posts can be updated at single time.",
  133. PATCH_LIMIT,
  134. )
  135. raise PermissionDenied(message % {'limit': PATCH_LIMIT})
  136. posts_queryset = exclude_invisible_posts(request.user, thread.category, thread.post_set)
  137. posts_queryset = posts_queryset.filter(id__in=posts_ids).order_by('id')
  138. posts = []
  139. for post in posts_queryset:
  140. post.category = thread.category
  141. post.thread = thread
  142. posts.append(post)
  143. if len(posts) != len(posts_ids):
  144. raise PermissionDenied(_("One or more posts to update could not be found."))
  145. return posts